我正在尝试从这样的模型创建表单:
class NewContractorModel
{
//...
public PhotoModel photos { get; set; }
//...
}
class PhotoModel
{
public List<Photo> f { get; set; }
}
从控制器我在模型对象上进行一些操作(实际上我从集合中删除了一些照片)并使用它将它们放入视图页面:
return new View("SomeView", model);
我试图为每张照片创建输入(比如说隐藏的输入)。
for (int i = 0; i < Model.photos.f.Count; ++i)
{
@Html.HiddenFor(m => m.photos.f[i].Uri)
@Html.HiddenFor(m => m.photos.f[i].ThumbnailUri)
@Html.HiddenFor(m => m.photos.f[i].SmallThumbnailUri)
@Html.TextBoxFor(m => m.photos.f[i].Description, new { placeholder = "Dodaj opis" })
}
但是我注意到这不起作用,因为它解雇了所有的模型修改(尽管我已经在Controler方法中删除了它们,但它仍然存储了列表中的所有照片)。
然后我尝试了这段代码:
for (int i = 0; i < Model.photos.f.Count; ++i)
{
Photo photo = Model.photos.f[i];
<input id="photos_f_@{@i}__Uri" name="photos.f[@{@i}].Uri" type="hidden" value="@photo.Uri"/>
<input id="photos_f_@{@i}__ThumbnailUri" name="photos.f[@{@i}].ThumbnailUri" type="hidden" value="@photo.ThumbnailUri"/>
<input id="photos_f_@{@i}__SmallThumbnailUri" name="photos.f[@{@i}].SmallThumbnailUri" type="hidden" value="@photo.SmallThumbnailUri"/>
<input id="photos_f_@{@i}__Description" name="photos.f[@{@i}].Description" placeholder="Dodaj opis" type="text" value="@photo.Description"/>
}
......这次它工作!
任何人都可以解释一下这两部分代码之间的区别是什么? 我试过这个代码超过十次,并且它总是工作相同所以这不是我的错。 ;) 我认为HtmlHelper方法中存在一个错误,但有没有任何转发?我想使用helpers方法而不是raw html。
编辑: 这是简化的控制器类。
public class MyController
{
private NewContractorModel _model = null;
protected override void OnActionExecuting(ActionExecutingContext filterContext)
{
_model = SerializationUtility.Deserialize(Request.Form["Data"]) as NewContractorModel;
if (_model == null)
_model = TempData["Data"] as NewContractorModel;
if (_model == null)
_model = new NewContractorModel() as NewContractorModel;
TryUpdateModel(_model);
}
protected override void OnResultExecuted(ResultExecutedContext filterContext)
{
if (filterContext.Result is RedirectToRouteResult)
TempData["Data"] = _model;
}
private bool CheckModel(object model)
{
Type type = model.GetType();
PropertyInfo[] properties = type.GetProperties();
foreach (PropertyInfo p in properties)
{
object[] attr = p.GetCustomAttributes(true);
foreach (object a in attr)
{
if (a is ValidationAttribute)
{
object value = p.GetValue(model, null);
if (!((ValidationAttribute)a).IsValid(value))
return false;
}
}
}
return true;
}
protected ActionResult SelectPage(string delPhoto)
{
if (!CheckModel(_model))
{
// Do some action
}
//.....
foreach (ZAY.Database.Photo p in _model.photos.f)
{
if (p.Uri == Request["delPhoto"])
{
_model.photos.f.Remove(p);
break;
}
}
//.....
return View("SomeView", _model);
}
}
我注意到在lambda中,模型看起来就像在TryUpdateModel调用之后(在修改之前)。如果我不使用lambdas,则修改模型......:/
还有我的Photo类(从EntityFramework生成 - 所以没什么有趣的)也简化了:
public class Photo : EntityObject
{
[Required]
public string Uri { get; set; }
[Required]
public string ThumbnailUri { get; set; }
[Required]
public string SmallThumbnailUri { get; set; }
public string Description { get; set; }
}
对不起,我只写了这么小的片段,但整个代码更复杂 - 只有最有趣的部分。
答案 0 :(得分:1)
这是我的问题的答案:
我想知道为什么在文档中没有提到......:/
答案 1 :(得分:0)
根据您的描述,我真的不明白您的第一个样本出了什么问题。但是你肯定对循环变量i
的范围有疑问。
由于表达式m => m.photos.f[i]
涉及闭包,因此将在稍后的时间,即for循环已经完成时对其进行求值。表达式捕获变量i
(而不是变量i
的值)。最终评估它时,它会在变量Model.photos.f.Count
中找到值i
。因此,所有隐藏字段和文本框都将使用相同的无效值 i 。
您的第二个代码示例通过在for循环中使用局部变量来避免此问题。