配列(string [])を返すメソッドがあり、この文字列の配列をアクションリンクに渡して、次のようなクエリ文字列を作成しようとしています。
/Controller/Action?str=val1&str=val2&str=val3...etc
しかし、new {str = GetStringArray()}を渡すと、次のURLが取得されます。
/Controller/Action?str=System.String%5B%5D
つまり、基本的には、私のstring []を受け取り、その上で.ToString()を実行して値を取得しています。
何か案は?ありがとう!
値を保持するRouteValueDictionaryを作成してみてください。各エントリに異なるキーを指定する必要があります。
<% var rv = new RouteValueDictionary();
var strings = GetStringArray();
for (int i = 0; i < strings.Length; ++i)
{
rv["str[" + i + "]"] = strings[i];
}
%>
<%= Html.ActionLink( "Link", "Action", "Controller", rv, null ) %>
次のようなリンクが表示されます
<a href='/Controller/Action?str=val0&str=val1&...'>Link</a>
[〜#〜] edit [〜#〜]:MVC2はValueProviderインターフェイスを変更して、元の回答を廃止しました。プロパティとして文字列の配列を持つモデルを使用する必要があります。
public class Model
{
public string Str[] { get; set; }
}
次に、モデルバインダーは、URLで渡す値をモデルに入力します。
public ActionResult Action( Model model )
{
var str0 = model.Str[0];
}
Scott Hanselmanからのインスピレーション で、これは本当に私を困らせました。私は次の(流暢)拡張メソッドを書きました。
public static RedirectToRouteResult WithRouteValue(
this RedirectToRouteResult result,
string key,
object value)
{
if (value == null)
throw new ArgumentException("value cannot be null");
result.RouteValues.Add(key, value);
return result;
}
public static RedirectToRouteResult WithRouteValue<T>(
this RedirectToRouteResult result,
string key,
IEnumerable<T> values)
{
if (result.RouteValues.Keys.Any(k => k.StartsWith(key + "[")))
throw new ArgumentException("Key already exists in collection");
if (values == null)
throw new ArgumentNullException("values cannot be null");
var valuesList = values.ToList();
for (int i = 0; i < valuesList.Count; i++)
{
result.RouteValues.Add(String.Format("{0}[{1}]", key, i), valuesList[i]);
}
return result;
}
次のように呼び出します。
return this.RedirectToAction("Index", "Home")
.WithRouteValue("id", 1)
.WithRouteValue("list", new[] { 1, 2, 3 });
私の頭に浮かんだ別の解決策:
string url = "/Controller/Action?iVal=5&str=" + string.Join("&str=", strArray);
これはダーティであり、使用する前にテストする必要がありますが、それでも機能するはずです。お役に立てれば。
nbinder と呼ばれるライブラリがあり、複雑なオブジェクトをルート/ URLに挿入するために使用できます。
それはこのように動作します:
using Unbound;
Unbinder u = new Unbinder();
string url = Url.RouteUrl("routeName", new RouteValueDictionary(u.Unbind(YourComplexObject)));
これはHelperExtension解決配列とIEnumerableプロパティのトラブルです。
public static class AjaxHelperExtensions
{
public static MvcHtmlString ActionLinkWithCollectionModel(this AjaxHelper ajaxHelper, string linkText, string actionName, object model, AjaxOptions ajaxOptions, IDictionary<string, object> htmlAttributes)
{
var rv = new RouteValueDictionary();
foreach (var property in model.GetType().GetProperties())
{
if (typeof(ICollection).IsAssignableFrom(property.PropertyType))
{
var s = ((IEnumerable<object>)property.GetValue(model));
if (s != null && s.Any())
{
var values = s.Select(p => p.ToString()).Where(p => !string.IsNullOrEmpty(p)).ToList();
for (var i = 0; i < values.Count(); i++)
rv.Add(string.Concat(property.Name, "[", i, "]"), values[i]);
}
}
else
{
var value = property.GetGetMethod().Invoke(model, null) == null ? "" : property.GetGetMethod().Invoke(model, null).ToString();
if (!string.IsNullOrEmpty(value))
rv.Add(property.Name, value);
}
}
return System.Web.Mvc.Ajax.AjaxExtensions.ActionLink(ajaxHelper, linkText, actionName, rv, ajaxOptions, htmlAttributes);
}
}