многоуровневое меню asp.net с привязкой к данным

В настоящее время я использую элемент управления меню asp.net для загрузки из родительских/дочерних элементов таблицы. Проблема, с которой я сталкиваюсь, заключается в том, что если у ребенка есть еще один ребенок. Мой код в этом смысле статичен, и я не могу найти лучшего или «того» способа сделать это. Я видел карту сайта в качестве источника данных, но мне не нужна карта сайта, и я чувствую, что это было бы излишним для того, чего мне нужно достичь.

foreach (ClassName option in list)
{
   MenuItem module = new MenuItem(option.Description.ToLower(), "", "", option.Url + "?option=" + option.Optionid);
   module.Selectable = true;
   navigation.Items.Add(module);
   //this is my second level
   foreach (ClassName child in listfromparent(option.Optionid))
   {
         MenuItem childmenu = new MenuItem(child.Description.ToLower(), "", "", child.Url + "?option=" + child.Optionid);
         module.ChildItems.Add(childmenu);
   }
 }

как вы можете видеть, это работает, но для 2 уровней :( и, конечно, я мог бы поместить еще один дочерний уровень внутри дочернего для создания 3-го уровня, но что, если есть 4-й, 5-й? Вот почему мне нужно, чтобы он сделал это сам. Я заметил В древовидном представлении есть onpopulate, но, видимо, в меню нет. Заранее спасибо.


person Andres    schedule 02.12.2011    source источник


Ответы (1)


Вот один из способов сделать это.

  • Представляйте отношения родитель/потомок в вашей таблице со списком смежности
  • Сопоставьте этот список смежности с древовидной структурой
  • Преобразуйте эту древовидную структуру в свою структуру пунктов меню.

Возможно, вы могли бы пропустить этот промежуточный шаг и сопоставить список смежности прямо с деревом элементов меню, возможно, с некоторыми методами расширения для элемента меню.

Но все таки...

Default.aspx

<%@ Page Language="C#" Inherits="MenuTreeDemo.Default" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html>
<head runat="server">
    <title>Default</title>
</head>
<body>
    <form id="form1" runat="server">
        <asp:Menu ID="MyMenu" runat="server" StaticDisplayLevels="3" />
    </form>
</body>
</html>

Default.aspx.cs

using System;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Data;
using System.Collections.Generic;

namespace MenuTreeDemo
{
    public partial class Default : System.Web.UI.Page
    {
        protected void Page_Load(object sender, EventArgs e)    
        {
            if (!IsPostBack)
            {
                MenuNode root = ConvertTableToTree(GetTreeTable());
                foreach (MenuNode topLevelNode in root.Children)
                {
                    MyMenu.Items.Add(topLevelNode.ToMenuItem()); // Visits all nodes in the tree.
                }
            }
        }


        // The menu tree as an adjacency list in a table. 
        static DataTable GetTreeTable()
        {
            DataTable table = new DataTable();
            table.Columns.Add("Id", typeof(int));
            table.Columns.Add("Description", typeof(string));
            table.Columns.Add("Url", typeof(string));
            table.Columns.Add("ParentId", typeof(int));

            table.Rows.Add(1, "TopMenu1", "/foo.html", 0);
            table.Rows.Add(2, "SubMenu1.1", "/baz.html", 1);
            table.Rows.Add(3, "SubMenu1.2", "/barry.html", 1);
            table.Rows.Add(4, "SubMenu1.2.1", "/skeet.html", 3);
            table.Rows.Add(5, "TopMenu2", "/bar.html", 0);
            table.Rows.Add(6, "TopMenu3", "/bar.html", 0);
            table.Rows.Add(7, "SubMenu3.1", "/ack.html", 6);

            return table;
        }


        // See e.g. http://stackoverflow.com/questions/2654627/most-efficient-way-of-creating-tree-from-adjacency-list
        // Assuming table is ordered.
        static MenuNode ConvertTableToTree(DataTable table)
        {
            var map = new Dictionary<int, MenuNode>();
            map[0] = new MenuNode() { Id = 0 }; // root node

            foreach (DataRow row in table.Rows)
            {
                int nodeId = int.Parse(row["Id"].ToString());
                int parentId = int.Parse(row["ParentId"].ToString());

                MenuNode newNode = MenuNodeFromDataRow(row);

                map[parentId].Children.Add(newNode);
                map[nodeId] = newNode;
            }

            return map[0]; // root node
        }


        static MenuNode MenuNodeFromDataRow(DataRow row)
        {
            int nodeId = int.Parse(row["Id"].ToString());
            int parentId = int.Parse(row["ParentId"].ToString());
            string description = row["Description"].ToString();
            string url = row["Url"].ToString();

            return new MenuNode() { Id=nodeId, ParentId=parentId, Description=description, Url=url };
        }
    }
}

MenuNode.cs

using System;
using System.Collections.Generic;
using System.Web.UI.WebControls;

namespace MenuTreeDemo
{
    public class MenuNode
    {
        public int Id { get; set; }
        public int ParentId { get; set; }
        public string Description { get; set; }
        public string Url { get; set; }
        public List<MenuNode> Children { get; set; }


        public MenuNode ()
        {
            Children = new List<MenuNode>();
        }


        // Will visit all descendants and turn them into menu items.
        public MenuItem ToMenuItem()
        {
            MenuItem item = new MenuItem(Description) { NavigateUrl=Url };
            foreach (MenuNode child in Children)
            {
                item.ChildItems.Add(child.ToMenuItem());
            }

            return item;
        }
    }
}
person ngm    schedule 03.12.2011
comment
Спасибо! это сработало отлично, я только что изменил его, чтобы он работал с моими текущими классами, а не с данными. Извините, это заняло у меня много времени, но у меня были длинные выходные, поэтому я просто попробовал это сегодня :) - person Andres; 07.12.2011
comment
Привет, небольшая помощь, мне нужно что-то подобное с тем же кодом snoupix .com/demo/css3-mega-dropdown-menu/demo.html возможно ли - person Developer; 02.01.2013