Атрибуты ListItems в DropDownList теряются при обратной передаче?
Сотрудник показал мне это:
У него есть DropDownList и кнопка на веб-странице. Вот код позади:
protected void Page_Load(object sender, EventArgs e)
{
if (!IsPostBack)
{
ListItem item = new ListItem("1");
item.Attributes.Add("title", "A");
ListItem item2 = new ListItem("2");
item2.Attributes.Add("title", "B");
DropDownList1.Items.AddRange(new[] {item, item2});
string s = DropDownList1.Items[0].Attributes["title"];
}
}
protected void Button1_Click(object sender, EventArgs e)
{
DropDownList1.Visible = !DropDownList1.Visible;
}
При загрузке страницы отображаются всплывающие подсказки элементов, но при первом обратном обращении атрибуты теряются. Почему это так, и есть ли какие-либо обходные пути?
Ответы
Ответ 1
У меня была та же проблема и я хотел внести вклад этот ресурс, где автор создал унаследованного потребителя ListItem для сохранения атрибутов ViewState. Надеюсь, это спасет кого-то время, которое я потратил впустую, пока не наткнулся на него.
protected override object SaveViewState()
{
// create object array for Item count + 1
object[] allStates = new object[this.Items.Count + 1];
// the +1 is to hold the base info
object baseState = base.SaveViewState();
allStates[0] = baseState;
Int32 i = 1;
// now loop through and save each Style attribute for the List
foreach (ListItem li in this.Items)
{
Int32 j = 0;
string[][] attributes = new string[li.Attributes.Count][];
foreach (string attribute in li.Attributes.Keys)
{
attributes[j++] = new string[] {attribute, li.Attributes[attribute]};
}
allStates[i++] = attributes;
}
return allStates;
}
protected override void LoadViewState(object savedState)
{
if (savedState != null)
{
object[] myState = (object[])savedState;
// restore base first
if (myState[0] != null)
base.LoadViewState(myState[0]);
Int32 i = 1;
foreach (ListItem li in this.Items)
{
// loop through and restore each style attribute
foreach (string[] attribute in (string[][])myState[i++])
{
li.Attributes[attribute[0]] = attribute[1];
}
}
}
}
Ответ 2
Спасибо, Ларами. Только то, что я искал. Он отлично сохраняет атрибуты.
Чтобы развернуть, ниже представлен файл класса, созданный с использованием кода Laramie для создания выпадающего списка в VS2008. Создайте класс в папке App_Code. После создания класса используйте эту строку на странице aspx для ее регистрации:
<%@ Register TagPrefix="aspNewControls" Namespace="NewControls"%>
Затем вы можете поместить элемент управления в свою веб-форму с помощью
<aspNewControls:NewDropDownList ID="ddlWhatever" runat="server">
</aspNewControls:NewDropDownList>
Хорошо, вот класс...
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Security.Permissions;
using System.Linq;
using System.Text;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;
namespace NewControls
{
[DefaultProperty("Text")]
[ToolboxData("<{0}:ServerControl1 runat=server></{0}:ServerControl1>")]
public class NewDropDownList : DropDownList
{
[Bindable(true)]
[Category("Appearance")]
[DefaultValue("")]
[Localizable(true)]
protected override object SaveViewState()
{
// create object array for Item count + 1
object[] allStates = new object[this.Items.Count + 1];
// the +1 is to hold the base info
object baseState = base.SaveViewState();
allStates[0] = baseState;
Int32 i = 1;
// now loop through and save each Style attribute for the List
foreach (ListItem li in this.Items)
{
Int32 j = 0;
string[][] attributes = new string[li.Attributes.Count][];
foreach (string attribute in li.Attributes.Keys)
{
attributes[j++] = new string[] { attribute, li.Attributes[attribute] };
}
allStates[i++] = attributes;
}
return allStates;
}
protected override void LoadViewState(object savedState)
{
if (savedState != null)
{
object[] myState = (object[])savedState;
// restore base first
if (myState[0] != null)
base.LoadViewState(myState[0]);
Int32 i = 1;
foreach (ListItem li in this.Items)
{
// loop through and restore each style attribute
foreach (string[] attribute in (string[][])myState[i++])
{
li.Attributes[attribute[0]] = attribute[1];
}
}
}
}
}
}
Ответ 3
Простым решением является добавление атрибутов всплывающей подсказки в событие pre-render
выпадающего списка. Любые изменения состояния должны выполняться в событии pre-render
.
пример кода:
protected void drpBrand_PreRender(object sender, EventArgs e)
{
foreach (ListItem _listItem in drpBrand.Items)
{
_listItem.Attributes.Add("title", _listItem.Text);
}
drpBrand.Attributes.Add("onmouseover", "this.title=this.options[this.selectedIndex].title");
}
Ответ 4
Если вы хотите загрузить список только при первой загрузке страницы, вам необходимо включить ViewState, чтобы элемент управления мог сериализовать его состояние и перезагрузить его, когда страница заходит назад.
Есть несколько мест, где ViewState может быть включен - проверьте <pages/>
node в файле web.config, а также в директиве <%@ page %>
в верхней части самого файла aspx для свойства EnableViewState
. Этот параметр должен быть true
для работы ViewState.
Если вы не хотите использовать ViewState, просто удалите if (!IsPostBack) { ... }
из кода, который добавляет ListItems
, и элементы будут воссозданы при каждой обратной передаче.
Изменить: Извиняюсь. Я неправильно понял ваш вопрос. Вы правы, что атрибуты не сохраняют обратную передачу, поскольку они не сериализованы в ViewState. Вы должны повторно добавить эти атрибуты при каждой обратной передаче.
Ответ 5
Одно простое решение. Вызовите функцию загрузки вниз по событию клика, на который вы запрашиваете пост.
Ответ 6
Типичные решения этой проблемы связаны с созданием новых элементов управления, которые не вполне возможны в обычных условиях. Существует простое, но тривиальное решение этой проблемы.
Проблема заключается в том, что ListItem
теряет свои атрибуты при обратной передаче. Однако сам список никогда не теряет никаких пользовательских атрибутов. Таким образом можно воспользоваться этим простым, но эффективным способом.
Шаги:
-
Сериализовать свои атрибуты с помощью кода в ответе выше (fooobar.com/questions/90793/...)
-
Сохраните его в пользовательском атрибуте ListControl (выпадающий список, контрольный список, что угодно).
-
В ответном сообщении верните пользовательский атрибут из ListControl и затем десериализуйте его как атрибуты.
Вот код, который я использовал для (сериализации) атрибутов (что мне нужно было сделать, это следить за тем, какие элементы списка были первоначально отображены как выбраны при извлечении из бэкэнд, а затем сохранить или удалить строки в соответствии с изменения, внесенные пользователем в пользовательском интерфейсе):
string[] selections = new string[Users.Items.Count];
for(int i = 0; i < Users.Items.Count; i++)
{
selections[i] = string.Format("{0};{1}", Users.Items[i].Value, Users.Items[i].Selected);
}
Users.Attributes["data-item-previous-states"] = string.Join("|", selections);
(выше, "Пользователи" - это элемент управления CheckboxList
).
В ответном сообщении (в моем случае кнопка Отправить кнопку Click), я использую приведенный ниже код, чтобы получить то же самое и сохранить их в словаре для последующей обработки:
Dictionary<Guid, bool> previousStates = new Dictionary<Guid, bool>();
string[] state = Users.Attributes["data-item-previous-states"].Split(new char[] {'|'}, StringSplitOptions.RemoveEmptyEntries);
foreach(string obj in state)
{
string[] kv = obj.Split(new char[] { ';' }, StringSplitOptions.None);
previousStates.Add(kv[0], kv[1]);
}
(PS: У меня есть библиотеки, которые выполняют обработку ошибок и преобразование данных, опуская их здесь для краткости).
Ответ 7
Вот код VB.Net решения, предложенного Ларами и уточненного gleapman.
Обновление: Код, который я разместил ниже, фактически предназначен для элемента управления ListBox. Просто измените наследование на DropDownList и переименуйте класс.
Imports System.Collections.Generic
Imports System.ComponentModel
Imports System.Security.Permissions
Imports System.Linq
Imports System.Text
Imports System.Web
Imports System.Web.UI
Imports System.Web.UI.WebControls
Namespace CustomControls
<DefaultProperty("Text")> _
<ToolboxData("<{0}:ServerControl1 runat=server></{0}:ServerControl1>")>
Public Class PersistentListBox
Inherits ListBox
<Bindable(True)> _
<Category("Appearance")> _
<DefaultValue("")> _
<Localizable(True)> _
Protected Overrides Function SaveViewState() As Object
' Create object array for Item count + 1
Dim allStates As Object() = New Object(Me.Items.Count + 1) {}
' The +1 is to hold the base info
Dim baseState As Object = MyBase.SaveViewState()
allStates(0) = baseState
Dim i As Int32 = 1
' Now loop through and save each attribute for the List
For Each li As ListItem In Me.Items
Dim j As Int32 = 0
Dim attributes As String()() = New String(li.Attributes.Count - 1)() {}
For Each attribute As String In li.Attributes.Keys
attributes(j) = New String() {attribute, li.Attributes(attribute)}
j += 1
Next
allStates(i) = attributes
i += 1
Next
Return allStates
End Function
Protected Overrides Sub LoadViewState(savedState As Object)
If savedState IsNot Nothing Then
Dim myState As Object() = DirectCast(savedState, Object())
' Restore base first
If myState(0) IsNot Nothing Then
MyBase.LoadViewState(myState(0))
End If
Dim i As Int32 = 0
For Each li As ListItem In Me.Items
' Loop through and restore each attribute
' NOTE: Ignore the first item as that is the base state and is represented by a Triplet struct
i += 1
For Each attribute As String() In DirectCast(myState(i), String()())
li.Attributes(attribute(0)) = attribute(1)
Next
Next
End If
End Sub
End Class
End Namespace
Ответ 8
@Sujay
Вы могли бы добавить текст с разделителем в виде двоеточия в атрибут выпадающего значения (например, стиль csv) и использовать String.Split(';'), чтобы получить 2 "значения" из одного значения, в качестве обходного пути, чтобы избежать неприятностей необходимо создать новый пользовательский элемент управления. Особенно, если у вас есть только несколько дополнительных атрибутов, и если он не слишком длинный. Вы также можете использовать значение JSON в атрибуте выпадающего значения, а затем проанализировать все, что вам нужно.
Ответ 9
//In the same block where the ddl is loaded (assuming the dataview is retrieved whether postback or not), search for the listitem and re-apply the attribute
if(IsPostBack)
foreach (DataRow dr in dvFacility.Table.Rows)
{
//search the listitem
ListItem li = ddl_FacilityFilter.Items.FindByValue(dr["FACILITY_CD"].ToString());
if (li!=null)
{
li.Attributes.Add("Title", dr["Facility_Description"].ToString());
}
} //end for each
Ответ 10
Простое решение без ViewState, создание нового серверного элемента управления или комплекса smth:
Создание:
public void AddItemList(DropDownList list, string text, string value, string group = null, string type = null)
{
var item = new ListItem(text, value);
if (!string.IsNullOrEmpty(group))
{
if (string.IsNullOrEmpty(type)) type = "group";
item.Attributes["data-" + type] = group;
}
list.Items.Add(item);
}
Обновление:
public void ChangeItemList(DropDownList list, string eq, string group = null, string type = null)
{
var listItem = list.Items.Cast<ListItem>().First(item => item.Value == eq);
if (!string.IsNullOrEmpty(group))
{
if (string.IsNullOrEmpty(type)) type = "group";
listItem.Attributes["data-" + type] = group;
}
}
Пример:
protected void Page_Load(object sender, EventArgs e)
{
if (!Page.IsPostBack)
{
using (var context = new WOContext())
{
context.Report_Types.ToList().ForEach(types => AddItemList(DropDownList1, types.Name, types.ID.ToString(), types.ReportBaseTypes.Name));
DropDownList1.DataBind();
}
}
else
{
using (var context = new WOContext())
{
context.Report_Types.ToList().ForEach(types => ChangeItemList(DropDownList1, types.ID.ToString(), types.ReportBaseTypes.Name));
}
}
}
Ответ 11
Мне удалось добиться этого с использованием переменных Session, в моем случае мой список не будет содержать много элементов, поэтому он работает очень хорошо, вот как я это сделал:
protected void Page_Load(object sender, EventArgs e)
{
if (!IsPostBack)
{
string[] elems;//Array with values to add to the list
for (int q = 0; q < elems.Length; q++)
{
ListItem li = new ListItem() { Value = "text", Text = "text" };
li.Attributes["data-image"] = elems[q];
myList.Items.Add(li);
HttpContext.Current.Session.Add("attr" + q, elems[q]);
}
}
else
{
for (int o = 0; o < webmenu.Items.Count; o++)
{
myList.Items[o].Attributes["data-image"] = HttpContext.Current.Session["attr" + o].ToString();
}
}
}
Когда страница загружается в первый раз, когда список заполняется, и я добавляю атрибут изображения, который теряется после обратной передачи: (поэтому в момент добавления элементов с его атрибутами я создаю одну переменную сеанса "attr" плюс число элемент, взятый из цикла "для" (он будет подобен attr0, attr1, attr2 и т.д.), и в них я сохраняю значение атрибута (путь к изображению в моем случае), когда происходит обратная передача ( внутри "else" ). Я просто зацикливаю список и добавляю атрибут, взятый из переменной Session, используя "int" цикла "for", который совпадает с загрузкой страницы (это происходит потому, что на этой странице я делаю не добавляйте элементы в список, просто выберите так, чтобы они всегда имели один и тот же индекс), и атрибуты снова установлены, я надеюсь, что это поможет кому-то в будущем, приветствия!