Vs2005的win form动态生成菜单 (转)

 这几天想用vs2005做点东西。要用到windows form 的MainMenu控件。于是想做成动态生成并且动态梆定事件,在网上找了一下没有发现什么好的方法。于是自己来研究一下。以前是做web form的东西,对windows form还真有点陌生的感觉。还好以前用Delphi也做过类似人功能有一点印象。就是用递归去创建菜单。好了不多说了,下面看代码。

 

        private   void  frmMain_Load( object  sender, EventArgs e)
        
{
            CreateMenu();
        }


        
/// 
        
/// 动态创建菜单
        
/// 

         private   void  CreateMenu()
        
{
            
//定义一个主菜单
            MenuStrip mainMenu = new MenuStrip();
            DataSet ds 
= new DataSet();
            
//从XML中读取数据。数据结构后面详细讲一下。
            ds.ReadXml(@"....Menu.xml");
            DataView dv 
= ds.Tables[0].DefaultView;
            
//通过DataView来过滤数据首先得到最顶层的菜单
            dv.RowFilter = "ParentItemID=0";
            
for(int i=0; i<dv.Count;i++)
            
{
                
//创建一个菜单项
                ToolStripMenuItem topMenu = new ToolStripMenuItem();
                
//给菜单赋Text值。也就是在界面上看到的值。
                topMenu.Text = dv[i]["Text"].ToString();
               
//如果是有下级菜单则通过CreateSubMenu方法来创建下级菜单
                if (Convert.ToInt16(dv[i]["IsModule"]) == 1)
                
{
                    
//以ref的方式将顶层菜单传递参数,因为他可以在赋值后再回传。--也许还有更好的方法^_^.
                    CreateSubMenu(ref topMenu, Convert.ToInt32(dv[i]["ItemID"]),ds.Tables[0]);
                }

                
//显示应用程序中已打开的 MDI 子窗体列表的菜单项
                mainMenu.MdiWindowListItem = topMenu;
                
//将递归附加好的菜单加到菜单根项上。
                mainMenu.Items.Add(topMenu);
            }

            mainMenu.Dock 
= DockStyle.Top;
            
//将窗体的MainMenuStrip梆定为mainMenu.
            this.MainMenuStrip = mainMenu;
            
//这句很重要。如果不写这句菜单将不会出现在主窗体中。
            this.Controls.Add(mainMenu);
        }


        
//  
         /// 创建子菜单
        
/// 
        
/// 父菜单项
        
/// 父菜单的ID
        
/// 所有菜单数据集

         private   void  CreateSubMenu( ref  ToolStripMenuItem topMenu, int  ItemID, DataTable dt)
        
{
            DataView dv 
= new DataView(dt);
            
//过滤出当前父菜单下在所有子菜单数据(仅为下一层的)
            dv.RowFilter = "ParentItemID=" + ItemID.ToString();
            
            
for (int i = 0; i < dv.Count; i++)
            
{
                
//创建子菜单项
                ToolStripMenuItem subMenu = new ToolStripMenuItem();
                subMenu.Text 
= dv[i]["Text"].ToString();
                
//如果还有子菜单则继续递归加载。
                if (Convert.ToInt16(dv[i]["IsModule"]) == 1)
                
{
                    
//递归调用
                    CreateSubMenu(ref subMenu, Convert.ToInt32(dv[i]["ItemID"]), dt);
                }

                
else
                
{
                    
//扩展属性可以加任何想要的值。这里用formName属性来加载窗体。
                    subMenu.Tag = dv[i]["FormName"].ToString();
                    
//给没有子菜单的菜单项加事件。
                    subMenu.Click += new EventHandler(subMenu_Click);
                }

                
if (dv[i]["ImageName"].ToString().Length > 0)
                
{
                    
//设置菜单项前面的图票为16X16的图片文件。
                    Image img = Image.FromFile(@"....Image"+dv[i]["ImageName"].ToString());
                    subMenu.Image 
= img;
                    subMenu.Image.Tag 
= dv[i]["ImageName"].ToString();
                }

                
//将菜单加到顶层菜单下。
                topMenu.DropDownItems.Add(subMenu);
            }

        }


        
/// 
        
/// 菜单单击事件
        
/// 
        
/// 
        
/// 

         void  subMenu_Click( object  sender, EventArgs e)
        
{
            
//tag属性在这里有用到。
            string formName = ((ToolStripMenuItem)sender).Tag.ToString();
            CreateFormInstance(formName);
        }


        
/// 
        
/// 创建form实例。
        
/// 
        
/// form的类名

         private   void  CreateFormInstance( string  formName)
        
{
            
bool flag = false;
            
//遍历主窗口上的所有子菜单
            for(int i=0;i<this.MdiChildren.Length; i++)
            
{
                
//如果所点的窗口被打开则重新激活
                if (this.MdiChildren[i].Tag.ToString().ToLower() == formName.ToLower())
                
{
                    
this.MdiChildren[i].Activate();
                    
this.MdiChildren[i].Show();
                    
this.MdiChildren[i].WindowState = FormWindowState.Normal;
                    flag 
= true;
                    
break;
                }

            }

            
if (!flag)
            
{
                
//如果不存在则用反射创建form窗体实例。
                Assembly asm = Assembly.Load("Fastyou.BookShop.Win");//程序集名
                object frmObj = asm.CreateInstance("Fastyou.BookShop.Win." + formName);//程序集+form的类名。
                Form frms = (Form)frmObj;
                
//tag属性要重新写一次,否则在第二次的时候取不到。原因还不清楚。有知道的望告知。
                frms.Tag = formName.ToString();
                frms.MdiParent 
= this;
                frms.Show();
            }

        }
 

 

下面来说明一下XML文件。
Text:为显示在窗体上的文字,
ItemID:为菜单的ID,
ParentItemID:为此菜单的父菜单ID如果是顶级菜单则为0
FormName:为单击菜单所要显示的窗体类名如果是还有下级菜单的则不要写。
IsModule:是否还有子菜单1有,0没有
Level:菜单所在的层数。目前还没用到。
ImageName:菜单前的图标为16x16的图片文件。
在这里还可以根据自己的需要加其它的属性,也可以写成以数据库的方式。
这是我目前用到的功能。我会继续发表我做这个小软件的心得体会。希望对大家有所帮助。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Vue3中,可以使用`v-for`指令动态生成表单,以下是一个示例: ```html <template> <el-form :model="form" :rules="rules" ref="form" label-width="80px"> <el-form-item v-for="(item, index) in formItems" :key="index" :label="item.label" :prop="item.prop"> <el-input v-if="item.type === 'input'" v-model="form[item.prop]"></el-input> <el-select v-if="item.type === 'select'" v-model="form[item.prop]"> <el-option v-for="(option, optionIndex) in item.options" :key="optionIndex" :label="option.label" :value="option.value"></el-option> </el-select> <!-- 其他表单元素 --> </el-form-item> <el-form-item> <el-button type="primary" @click="submitForm">提交</el-button> </el-form-item> </el-form> </template> <script> export default { data() { return { form: { // 表单数据 }, rules: { // 表单验证规则 }, formItems: [ { label: '姓名', prop: 'name', type: 'input' }, { label: '年龄', prop: 'age', type: 'input' }, { label: '性别', prop: 'gender', type: 'select', options: [{ label: '男', value: 'male' }, { label: '女', value: 'female' }] }, // 其他表单项 ] } }, methods: { submitForm() { this.$refs.form.validate(valid => { if (valid) { // 表单验证通过,提交数据 } else { // 表单验证失败 return false } }) } } } </script> ``` 在上面的示例中,`formItems`数组定义了表单项的属性,通过在`v-for`指令中遍历`formItems`数组,动态生成表单元素。`v-if`指令用于根据表单项的类型显示不同的表单元素。需要注意的是,每个表单元素的`v-model`指令应该绑定到`form`对象的相应属性上,这样可以方便地获取表单数据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值