c# - 在每个循环迭代中创建对象

原文 标签 c# .net winforms menu tree-structure

我正在使用内置的Z80导航控件,以下是演示链接:
Z80 Navigation Menu
如果有人看到这个控件,它有一个对象来创建菜单,比如父菜单和子菜单。大致如下:

public List<NavBarItem> sampleDynamicNav; //List of navbar objects
public DemoItems()
{
    //Create object instance here and assign the parent as well child menus here
    sampleDynamicNav = new List<NavBarItem> {
    new NavBarItem {ID = 1, Text = "UserInfo", Icon = new ItemIcon {Default = SampleProject.Properties.Resources.nav_new_home, Hover = SampleProject.Properties.Resources.nav_new_home, Selected = SampleProject.Properties.Resources.nav_new_home}, ToolTip = "tooltip Main Menu", Height = 40,
        Icon = new ItemIcon {Default = SampleProject.Properties.Resources.nav_new_home, Hover = SampleProject.Properties.Resources.nav_new_home, Selected = SampleProject.Properties.Resources.nav_new_home }, ToolTip = "tooltip Desktop"},
        Childs = new List<NavBarItem> {
                    new NavBarItem {ID = 41, Text = "Add/Edit Users", Height = 30 },
                    new NavBarItem {ID = 42, ParentID = 1, Text = "Inactive User", Height = 30}
    };
}

如果我们静态地分配菜单,这非常简单。但我坚持这样做,当尝试动态添加它们时,我的意思是从数据库创建菜单,如下所示:
public DemoItems()
{
    foreach (var parent in GetParent("USER-0001"))
    {
          foreach (var child in GetChild(parent.MenuNo))
          {
            sampleDynamicNav = new List<NavBarItem> {
                 new NavBarItem {
                 ID = parent.MenuNo, Text = parent.MenuName, Icon = new ItemIcon {Default =  SampleProject.Properties.Resources.nav_new_home, Hover = SampleProject.Properties.Resources.nav_new_home, Selected = SampleProject.Properties.Resources.nav_new_home}, ToolTip = "tooltip Main Menu", Height = 40,
                 Childs = new List<NavBarItem> {
                                    new NavBarItem {ID = child.MenuNo, ParentID = parent.MenuNo, Text = child.MenuName, Height = 30 },
                           }
                     }
               };
          }
     }
}

使用上面的代码,它应该至少在导航栏中获得父菜单。现在,撇开子菜单不谈,它在导航栏中显示一个父菜单,如下所示:
Sample 1
但它应该如下所示,因为有两个父菜单,并使用foreach循环迭代列表(getParents()返回一个对象列表):
Sample 2
我不知道是否还需要为它做些什么,不知道是否可以循环浏览导航栏的子属性,如下所示:
foreach (var child in GetChild(parent.MenuNo))
{
   Childs = new List<NavBarItem> {
            new NavBarItem {ID = child.MenuNo, ParentID = parent.MenuNo, Text = child.MenuName, Height = 30 },
}

注意:当试图用循环来迭代子属性时,它会立即抛出错误。第二个内部循环工作,并取出子菜单,但也就是说,父菜单有两个子菜单,它一次返回1。我调试了列表,它会像往常一样返回双亲菜单,但不会显示在导航栏中。
getParents方法:
/**Get Menu Details - Starts**/
public IEnumerable<UserViewModel> GetParent(string empNo)
{
       List<UserViewModel> lstUser = new List<UserViewModel>();

       string query = "SELECT DISTINCT M.PARENT, M.MENUNO, M.MENUNAME FROM (SELECT DISTINCT M.MENUNO, M.MENUNAME, M.PARENT " +
                      "FROM USER_DETAILS U INNER JOIN USER_GROUP_DETAILS UG ON UG.EMPNO = U.EMPNO " +
                      "INNER JOIN ASSIGN_MENU_DETAILS AM ON AM.GROUPNO = UG.GROUPNO INNER JOIN MENU_DETAILS M " +
                      "ON M.MENUNO = AM.MENUNO WHERE U.EMPNO = '" + empNo + "' " +
                      "UNION ALL " +
                      "SELECT DISTINCT M.MENUNO, M.MENUNAME, " +
                      "M.PARENT FROM MENU_DETAILS M " +
                      "INNER JOIN MENU_DETAILS C " +
                      "ON C.PARENT = M.MENUNO) m WHERE M.PARENT = '0' ORDER BY M.PARENT";

        DataTable dt = SelectData(query);

        if (dt != null && dt.Rows.Count > 0)
        {
            foreach (DataRow dr in dt.Rows)
            {
                UserViewModel bo = new UserViewModel();
                bo.Parent = Convert.ToInt32(dr["PARENT"].ToString());
                bo.MenuNo = Convert.ToInt32(dr["MENUNO"].ToString());
                bo.MenuName = dr["MENUNAME"].ToString();

                lstUser.Add(bo);
            }
        }
   return lstUser;
}
/**Get Menu Details - Ends**/

最佳答案

您可以创建以下助手方法并使用它们创建一个List<NavBarItem接受任何类型的数据源作为输入,包括DataTableList<YourEntity>或任何其他IEnumerable<T>的方法。
因此,无论您拥有哪种数据存储,都可以使用以下方法。
它依赖于递归算法来创建树。要从任何类型的数据源创建树,您需要具有以下信息:
数据源
如何检测数据源中的项是否为根项
如何在数据源中查找项的子项
如何从数据源项创建树项。
以下方法通过询问上述信息创建NavBarItem层次结构列表:

private IEnumerable<NavBarItem> GetNavBarItems<T>(
    IEnumerable<T> source,
    Func<T, Boolean> isRoot,
    Func<T, IEnumerable<T>, IEnumerable<T>> getChilds,
    Func<T, NavBarItem> getItem)
{
    IEnumerable<T> roots = source.Where(x => isRoot(x));
    foreach (T root in roots)
        yield return ConvertEntityToNavBarItem(root, source, getChilds, getItem); ;
}

private NavBarItem ConvertEntityToNavBarItem<T>(
    T entity,
    IEnumerable<T> source,
    Func<T, IEnumerable<T>, IEnumerable<T>> getChilds,
    Func<T, NavBarItem> getItem)
{
    NavBarItem node = getItem(entity);
    var childs = getChilds(entity, source);
    foreach (T child in childs)
        node.Childs.Add(ConvertEntityToNavBarItem(child, source, getChilds, getItem));
    return node;
}

例子
我假设您已将数据加载到以下结构中:
var dt = new DataTable();
dt.Columns.Add("Id", typeof(int));
dt.Columns.Add("Name", typeof(string));
dt.Columns.Add("ParentId", typeof(int));

dt.Rows.Add(1, "Menu 1", DBNull.Value);
dt.Rows.Add(11, "Menu 1-1", 1);
dt.Rows.Add(111, "Menu 1-1-1", 11);
dt.Rows.Add(112, "Menu 1-1-2", 11);
dt.Rows.Add(12, "Menu 1-2", 1);
dt.Rows.Add(121, "Menu 1-2-1", 12);
dt.Rows.Add(122, "Menu 1-2-2", 12);
dt.Rows.Add(123, "Menu 1-2-3", 12);
dt.Rows.Add(124, "Menu 1-2-4", 12);
dt.Rows.Add(2, "Menu 2", DBNull.Value);
dt.Rows.Add(21, "Menu 2-1", 2);
dt.Rows.Add(211, "Menu 2-1-1", 21);

然后,要将其转换为List<NavBarItem>,可以使用以下代码:
var source = dt.AsEnumerable();
var list = GetNavBarItems(
        source,
        (r) => r.Field<int?>("ParentId") == null,
        (r, s) => s.Where(x => x.Field<int?>("ParentId") == r.Field<int?>("Id")),
        (r) => new NavBarItem()
        {
            ID = r.Field<int>("Id"),
            Text = r.Field<string>("Name"),
            ParentID = r.Field<int?>("ParentId")
        }).ToList();

因此,您将具有以下结构:
Menu 1
   Menu 1-1
       Menu 1-1-1
       Menu 1-1-2
   Menu 1-2
       Menu 1-2-1
       Menu 1-2-2
       Menu 1-2-3
       Menu 1-2-4
Menu 2
    Menu 2-1
        Menu 2-1-1

注意
对于那些不想安装包但想测试结构的用户,可以使用以下NavBarItem类:
public class NavBarItem
{
    public NavBarItem()
    {
        Childs = new List<NavBarItem>();
    }
    public int ID { get; set; }
    public int? ParentID { get; set; }
    public string Text { get; set; }
    public List<NavBarItem> Childs { get; set; }
    public override string ToString()
    {
        return Text;
    }
}

相关文章:

c# - Web请求如何将List <t>导出到Excel

c# - 结构中的C#数组

c# - TextBlock.GetBindingExpression返回NULL

c# - 在双屏设置中显示Winforms

c# - 在Page_Init期间强制HTTPS的最简单方法

c# - C#API返回HTML而不是JSON

.net - 在未安装Visual Studio的计算机上使用FUSLOGVW.EXE

c# - 如何在ASP.NET MVC中获取当前用户

c# - 如何从mysql表中删除单个值

c# - 如何使用只读保护打开Excel文件?