コントローラにselectListを作成し、ビューに表示します。
私はその場でそれを作成しようとしています、このように...
myViewData.PageOptionsDropDown =
new SelectList(new [] {"10", "15", "25", "50", "100", "1000"}, "15");
コンパイルされますが、出力が悪いです...
<select id="PageOptionsDropDown" name="PageOptionsDropDown">
<option>10</option>
<option>15</option>
<option>25</option>
<option>50</option>
<option>100</option>
<option>1000</option>
</select>
アイテムが選択されていないことに注意してください
どうすれば修正できますか?
これは私がそれをする方法です
IList<Customer> customers = repository.GetAll<Customer>();
IEnumerable<SelectListItem> selectList =
from c in customers
select new SelectListItem
{
Selected = (c.CustomerID == invoice.CustomerID),
Text = c.Name,
Value = c.CustomerID.ToString()
};
一見すると、あなたが何を求めているのかわからない...
私は拡張メソッドを使用します:
使用法
var departmentItems = departments.ToSelectList(d => d.Code +
" - " + d.Description,
d => d.Id.ToString(),
" - ");
var functionItems = customerFunctions.ToSelectList(f => f.Description,
f => f.Id.ToString(),
" - ");
with
public static class MCVExtentions
{
public static List<SelectListItem> ToSelectList<T>(
this IEnumerable<T> enumerable,
Func<T, string> text,
Func<T, string> value,
string defaultOption)
{
var items = enumerable.Select(f => new SelectListItem()
{
Text = text(f),
Value = value(f)
}).ToList();
items.Insert(0, new SelectListItem()
{
Text = defaultOption,
Value = "-1"
});
return items;
}
}
items, dataValueField, dataTextField, selectedValue
をパラメーターとして受け入れるコンストラクターの使用:
ViewData["myList"] =
new SelectList(new[] { "10", "15", "25", "50", "100", "1000" }
.Select(x => new {value = x, text = x}),
"value", "text", "15");
次に、あなたの意見で:
<%=Html.DropDownList("myList") %>
Thomas Stockの答えを基に、これらのオーバーロードされたToSelectList
メソッドを作成しました。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web.Mvc;
public static partial class Helpers
{
public static IEnumerable<SelectListItem> ToSelectList<T>(this IEnumerable<T> enumerable, Func<T, object> value, bool selectAll = false)
{
return enumerable.ToSelectList(value, value, selectAll);
}
public static IEnumerable<SelectListItem> ToSelectList<T>(this IEnumerable<T> enumerable, Func<T, object> value, object selectedValue)
{
return enumerable.ToSelectList(value, value, new List<object>() { selectedValue });
}
public static IEnumerable<SelectListItem> ToSelectList<T>(this IEnumerable<T> enumerable, Func<T, object> value, IEnumerable<object> selectedValues)
{
return enumerable.ToSelectList(value, value, selectedValues);
}
public static IEnumerable<SelectListItem> ToSelectList<T>(this IEnumerable<T> enumerable, Func<T, object> value, Func<T, object> text, bool selectAll = false)
{
foreach (var f in enumerable.Where(x => x != null))
{
yield return new SelectListItem()
{
Value = value(f).ToString(),
Text = text(f).ToString(),
Selected = selectAll
};
}
}
public static IEnumerable<SelectListItem> ToSelectList<T>(this IEnumerable<T> enumerable, Func<T, object> value, Func<T, object> text, object selectedValue)
{
return enumerable.ToSelectList(value, text, new List<object>() { selectedValue });
}
public static IEnumerable<SelectListItem> ToSelectList<T>(this IEnumerable<T> enumerable, Func<T, object> value, Func<T, object> text, IEnumerable<object> selectedValues)
{
var sel = selectedValues != null
? selectedValues.Where(x => x != null).ToList().ConvertAll<string>(x => x.ToString())
: new List<string>();
foreach (var f in enumerable.Where(x => x != null))
{
yield return new SelectListItem()
{
Value = value(f).ToString(),
Text = text(f).ToString(),
Selected = sel.Contains(value(f).ToString())
};
}
}
}
コントローラーでは、次のことを実行できます。
var pageOptions = new[] { "10", "15", "25", "50", "100", "1000" };
ViewBag.PageOptions = pageOptions.ToSelectList(o => o, "15" /*selectedValue*/);
最後に、ビューに次のように入力します。
@Html.DropDownList("PageOptionsDropDown", ViewBag.PageOptions as IEnumerable<SelectListItem>, "(Select one)")
目的の出力が得られます。もちろん、最初の空の項目が必要ない場合は、上記の"(Select one)"
optionLabelを省略できます。
<select id="PageOptionsDropDown" name="PageOptionsDropDown">
<option value="">(Select one)</option>
<option value="10">10</option>
<option selected="selected" value="15">15</option>
<option value="25">25</option>
<option value="50">50</option>
<option value="100">100</option>
<option value="1000">1000</option>
</select>
更新:修正されたコードのリストは、XMLコメント付きの ここにあります です。
問題は、SelectListが設計どおりに機能することです。バグは設計にあります。 SelectedItemでSelectedプロパティを設定できますが、GetEnumerator()でリストを走査する場合(またはMvcがそれを行う場合)、これは完全に無視されます。 Mvcは代わりに新しいSelectListItemsを作成します。
SelectListItem []、Text-Name、Value-Name、SelectedValueでSelectList ctorを使用する必要があります。 SelectValueとして、SelectListItem自体ではなく、選択するSelectListItemのVALUEを渡すことに注意してください!例:
SelectList sl = new SelectList( new[]{
new SelectListItem{ Text="one", Value="1"},
new SelectListItem{ Text="two", Value="2"},
new SelectListItem{ Text="three", Value="3"}
}, "Text", "Value", "2" );
(これはテストしていませんが、同じ問題がありました)
2番目のオプションはselected = "selected"属性を取得します。それは古き良きデータセットのように見えます;-)
これはオプションです:
myViewData.PageOptionsDropDown = new[]
{
new SelectListItem { Text = "10", Value = "10" },
new SelectListItem { Text = "15", Value = "15", Selected = true }
new SelectListItem { Text = "25", Value = "25" },
new SelectListItem { Text = "50", Value = "50" },
new SelectListItem { Text = "100", Value = "100" },
new SelectListItem { Text = "1000", Value = "1000" },
}
それが文字通りあなたがやりたいことなら、文字列として配列を宣言するだけで選択されたアイテムの問題が修正されます:
myViewData.PageOptionsDropDown =
new SelectList(new string[] {"10", "15", "25", "50", "100", "1000"}, "15");
プロパティがInt、String、Double値などの単純なオブジェクトではない場合でも、SelectListとSelectedValueを連携させるのは非常に簡単です。
例:
Regionオブジェクトが次のようなものであると仮定します。
public class Region {
public Guid ID { get; set; }
public Guid Name { get; set; }
}
そして、あなたのビューモデルは次のようなものです:
public class ContactViewModel {
public DateTime Date { get; set; }
public Region Region { get; set; }
public List<Region> Regions { get; set; }
}
以下のコードを使用できます。
@Html.DropDownListFor(x => x.Region, new SelectList(Model.Regions, "ID", "Name"))
RegionオブジェクトのToStringメソッドを次のようにオーバーライドする場合のみ:
public class Region {
public Guid ID { get; set; }
public Guid Name { get; set; }
public override string ToString()
{
return ID.ToString();
}
}
これには100%の保証が必要です。
しかし、すべての状況でSelectListを100%動作させるための最良の方法は、Equalsメソッドを使用して、アイテムコレクションの各アイテムに対してDropDownListまたはListBoxプロパティの値をテストすることです。
強く型付けされたビューがある場合、継承されたクラスのプロパティの名前ではないようにドロップダウンのIDを変更する必要があるようです。次に、変更をコミットする前に、FORMCollectionから選択した値を取り出してインスタンスに適用するために、編集(POST)メソッドにロジックを配置する必要があります。
これは確かに少し奇妙ですが、私はそれを試してみましたが、動作します。
クラスにCountryId sayというフィールドがあり、国名のリストを表示している場合、ドロップダウンにCountryIdではなくCountryNameのIDを設定し、投稿でCollection ["CountryName"]で何かを行うことができます。
私はまったく同じ問題を抱えていました。解決策は簡単です。 DropDownListヘルパーに渡される「name」パラメーターを、ViewModelに存在するプロパティのいずれとも一致しないものに変更するだけです。詳細はこちら: http://www.dotnetguy.co.uk/post/2009/06/25/net-mvc-selectlists-selected-value-does-not-get-set-in-the-ビュー
ダンワトソンを引用します。
MVCでは、ビューが強く入力された場合、選択リストの選択オプションがオーバーライドされ、コンストラクターで設定された選択オプションプロパティがビューに到達せず、ドロップダウンの最初のオプションが代わりに選択されます(理由はまだ少し謎です) 。
乾杯!
シンプル:
string[] someName = new string[] {"10", "15", "25", "50", "100", "1000"};
myViewData.PageOptionsDropDown = new SelectList(someName, "15");
これらのすべての答えは素晴らしいようですが、コントローラがよく知られている構造化形式でビューのデータを準備しているのに対して、ビューがモデルを介して配信されるIEnumerable <>を単純に反復し、標準選択リストを作成してからDefaultModelBinderを許可しているようですアクションパラメータを使用して、選択したアイテムを送り返します。はい、いいえ、なぜですか?懸念の分離、はい?コントローラーを構築して、UIとビュー固有のものを作成するのは奇妙に思えます。
このように実行しただけで問題ありませんでしたが、
public class myViewDataObj
{
public SelectList PageOptionsDropDown { get; set; }
}
public ActionResult About()
{
myViewDataObj myViewData = new myViewDataObj();
myViewData.PageOptionsDropDown =
new SelectList(new[] { "10", "15", "25", "50", "100", "1000" }, "15");
ViewData["myList"] = myViewData.PageOptionsDropDown;
return View();
}
そして
<%=Html.DropDownList("myList") %>
これを行うとうまくいきました
public ActionResult About()
{
myViewDataObj myViewData = new myViewDataObj();
myViewData.PageOptionsDropDown =
new SelectList(new[] { "10", "15", "25", "50", "100", "1000" });
ViewData["myListValues"] = myViewData.PageOptionsDropDown;
ViewData["myList"] = "15";
return View();
}
そして
<%=Html.DropDownList("myList",(IEnumerable<SelectListItem>)ViewData["myListValues"]) %>
MonthRepository monthRepository = new MonthRepository();
IQueryable<MonthEntity> entities = monthRepository.GetAllMonth();
List<MonthEntity> monthEntities = new List<MonthEntity>();
foreach(var r in entities)
{
monthEntities.Add(r);
}
ViewData["Month"] = new SelectList(monthEntities, "MonthID", "Month", "Mars");
-Select-など、ランダムなテキストをDropDownListに渡す場合は、次のコードを使用して簡単に実行できます。
@Html.DropDownListFor(x => x.CategoryId, new SelectList(Model.Categories, "Id", "Name"), "--Select--", new { @class = "form-control" })
私はこのようにします:
List<SelectListItem> list = new List<SelectListItem>{
new SelectListItem {Selected = true, Text = "Select", Value = "0"},
new SelectListItem {Selected = true, Text = "1", Value = "1"},
new SelectListItem {Selected = true, Text = "2", Value = "2"}
};
return list.ToArray();
ToArray()は問題を処理します。
あなたの例を使用して、これは私のために働いた:
コントローラー:
ViewData["PageOptionsDropDown"] = new SelectList(new[] { "10", "15", "25", "50", "100", "1000" }, "15");
表示:
<%= Html.DropDownList("PageOptionsDropDown")%>
Html.DropDownList拡張メソッドでMVC 2のソースコードを見ると、SelectListクラスのSelectedValueプロパティはチェックされません。モデルとの一致のみを試みます。
上記はすべてテーマのバリエーションです。つまり、ドロップダウンリストのためにビューに大量のデータを送信するにはどうすればよいですか?.
問題はビューにあります。設定したselectvalueを尊重する独自のDropDownList拡張メソッドを作成するか、手動で繰り返します。どれがあなたに最適か。
モデルにコレクションがあり、ビューが厳密に型である場合、これのいくつかのバリエーションが機能します。
@Html.DropDownListFor(x => x.RegionID,
new SelectList(Model.Regions,"RegionID", "RegionName", Model.RegionID))
-または-
@Html.DropDownList("RegionID",
new SelectList(Model.Regions, "RegionID", "RegionName", Model.RegionID))
私はmvc 1がどのようにセットアップされたか覚えていませんが、それが属するフィールドと同じ名前の選択リストが必要だったようです...
上記の誰かが言ったように、私が見つけたのは、送信されたViewDataがフィールドと同じ名前であったときに、mvc2で私の選択リストが機能していなかったということです。
例えば:
<%= Html.DropDownListFor((model => model.ForID), (SelectList)ViewData["ForName"]) %>
動作するとき
<%= Html.DropDownListFor((model => model.ForID), (SelectList)ViewData["ForID"]) %>
viewDataの名前「ForID」は、作業対象のフィールドと同じ名前が付けられているため機能しません
考えられる説明は、バインドしているselectlist値が文字列ではないということです。
その例では、パラメータ「PageOptionsDropDown」はモデルの文字列ですか?そうでない場合は、リストで選択した値が表示されないためです。
モデルで選択された値は、デフォルトのアイテムの代わりに利用されます。 (すべての投稿を読んでいないことに同意します)
ViewDataにあいまいさがある場合があります。
ご覧ください ここ