邵阳网首页 > 百科 > 使用C#和DesignSurface实现简单的窗体设计器
使用C#和DesignSurface实现简单的窗体设计器

TIPS:本文共有 14525 个字,阅读大概需要 30 分钟。

C#中的DesignSurface是一个强大的工具,可以帮助开发者轻松实现窗体设计器功能。本文将介绍如何利用DesignSurface简单地实现窗体设计器,包括创建设计表面、添加控件、设置属性等操作。通过这个简单的示例,读者可以了解如何使用DesignSurface来动态创建和修改窗体设计,为他们的应用程序提供更加灵活和自定义的界面设计功能。

更新时间:02月25日 16:09:06 作者:JackWang-CUMT

我要评论

这篇文章主要介绍了C#利用DesignSurface如何实现简单窗体设计器的相关资料,文中通过图文及示例代码介绍的很详细,对大家具有一定的参考价值,需要的朋友们下面来一起学习学习吧。

ponentModel.Design.DesignSurface是为设计组件提供一个用户界面,通过它可以实现一个简单的窗体设计器。

在构建之前,我们需要引入System.Design.dll,否则会出现找不到DesignSurface的错误。

?

1

2

3

4

5

6

7

8

9

10

private void Form1_Load(object sender, EventArgs e)

{

//引用System.Deisgn.dll

DesignSurface ds = new DesignSurface();

//开始加载窗体

ds.BeginLoad(typeof(Form));

Control designerContorl = (Control)ds.View;

designerContorl.Dock = DockStyle.Fill;

this.Controls.Add(designerContorl);

}

private void Form1_Load(object sender, EventArgs e) { //引用System.Deisgn.dll DesignSurface ds = new DesignSurface(); //开始加载窗体 ds.BeginLoad(typeof(Form)); Control designerContorl = (Control)ds.View; designerContorl.Dock = DockStyle.Fill; this.Controls.Add(designerContorl); }

运行后出现简单的一个UI设计器

但是该设计器并不能实现控件拖放和UI设计器,以及控件的属性配置。

为了支持从源代码加载初始化窗体,需要对源码中的相关方法进行解析,这里我们 CodeDomDesignerLoader来实现定制化业务,CodeDomDesignerLoader是提供用于实现基于 CodeDOM 的设计器加载程序的基类。

继承它的类需要重写CodeCompileUnit Parse()方法,来实现加载窗体:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

protected override CodeCompileUnit Parse()

{

#region 源文件读取

var sw = new StreamReader(@"E:\FrmUser.cs");

var sw_designer = new StreamReader(@"E:\FrmUser.Designer.cs");

string formCodeCS = sw.ReadToEnd();

string formCodeDesigner = sw_designer.ReadToEnd();

List<string> source = new List<string>();

source.Add(formCodeCS);

source.Add(formCodeDesigner);

#endregion

//Rolsyn解析C#

var rootDesigner = Source2CodeDom.Parse(formCodeDesigner);

codeDesingerCompileUnit = Source2CodeDom.GetDesignerCodeComplieUnit(rootDesigner);

var rootCS = Source2CodeDom.Parse(formCodeCS);

codeCSCompileUnit = Source2CodeDom.GetCodeComplieUnit(rootCS);

//MergeFormSource

string mergeS = Source2CodeDom.MergeFormSource(formCodeDesigner, formCodeCS);

codeMergeCompileUnit = Source2CodeDom.GetMergeDesignerCodeComplieUnit(mergeS);

return codeMergeCompileUnit;

protected override CodeCompileUnit Parse() { #region 源文件读取 var sw = new StreamReader(@"E:\FrmUser.cs"); var sw_designer = new StreamReader(@"E:\FrmUser.Designer.cs"); string formCodeCS = sw.ReadToEnd(); string formCodeDesigner = sw_designer.ReadToEnd(); List<string> source = new List<string>(); source.Add(formCodeCS); source.Add(formCodeDesigner); #endregion //Rolsyn解析C# var rootDesigner = Source2CodeDom.Parse(formCodeDesigner); codeDesingerCompileUnit = Source2CodeDom.GetDesignerCodeComplieUnit(rootDesigner); var rootCS = Source2CodeDom.Parse(formCodeCS); codeCSCompileUnit = Source2CodeDom.GetCodeComplieUnit(rootCS); //MergeFormSource string mergeS = Source2CodeDom.MergeFormSource(formCodeDesigner, formCodeCS); codeMergeCompileUnit = Source2CodeDom.GetMergeDesignerCodeComplieUnit(mergeS); return codeMergeCompileUnit;

解析的方法如下,但是此解析只是用于代码的生成,并不能用户UI界面的显示:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

public static CodeCompileUnit GetDesignerCodeComplieUnit2(CompilationUnitSyntax root)

{

CodeCompileUnit ccu = new CodeCompileUnit();

var firstMember = root.Members[0];

var namespaceDeclration = (NamespaceDeclarationSyntax)firstMember;

var designClassDeclaration = (ClassDeclarationSyntax)namespaceDeclration.Members[0];

var myDesignerClass = new CodeTypeDeclaration(designClassDeclaration.Identifier.ToString());

var initializeComponent = new CodeMemberMethod();

var ns = new CodeNamespace(namespaceDeclration.Name.ToString());

foreach (var m in designClassDeclaration.Members)

{

if (m is ConstructorDeclarationSyntax)

{

var ctor = ((ConstructorDeclarationSyntax)m);

var codeBody = ctor.Body.ToString();

codeBody = codeBody.Trim().TrimStart("{").TrimEnd("}").Trim().TrimEnd(";");

CodeSnippetExpression csbody = new CodeSnippetExpression(codeBody);

CodeExpressionStatement stmt = new CodeExpressionStatement(csbody);

//Add the expression statements to the method.

// InitializeComponent

var cctor = new CodeConstructor();

cctor.Name = ctor.Identifier.ToString();

//var cmm = new CodeMemberMethod();

//cmm.Name = ctor.Identifier.ToString();

//cmm.Attributes = GetCtoRAttrMapping(ctor);

//cmm.ReturnType = new CodeTypeReference(typeof(void));

cctor.Statements.Add(stmt);

myDesignerClass.Members.Add(cctor);

}

if (m is FieldDeclarationSyntax)

{

var F = ((FieldDeclarationSyntax)m);

var type = F.Declaration.Type;

foreach (var variable in F.Declaration.Variables)

{

var field = new CodeMemberField();

field.Name = variable.Identifier.ToString();

field.Type = new CodeTypeReference(type.ToString());

field.Attributes = GetFieldAttrMapping(F);

//field.InitExpression = new CodePrimitiveExpression(null);

myDesignerClass.Members.Add(field);

}

}

if (m is MethodDeclarationSyntax)

{

var node = m as MethodDeclarationSyntax;

#region xml comments

var xmlTrivia = node.GetLeadingTrivia()

.Select(i => i.GetStructure())

.OfType<DocumentationCommentTriviaSyntax>()

.FirstOrDefault();

#endregion

var method = (MethodDeclarationSyntax)m;

var cmm = new CodeMemberMethod();

cmm.Name = method.Identifier.ToString();

///XML注释

string[] comments = xmlTrivia.ToString().Split("\r\n".ToCharArray());

foreach (string text in comments)

{

if (text.Trim() != "")

{

ments.Add(new CodeCommentStatement(text.Trim().TrimStart("///".ToCharArray()).Trim(), true));

}

}

if (cmm.Name == "InitializeComponent")

{

//region

CodeRegionDirective codeRegion = new CodeRegionDirective(CodeRegionMode.Start, "Windows 窗体设计器生成的代码");

CodeRegionDirective codeEndRegion = new CodeRegionDirective(CodeRegionMode.End, "");

cmm.StartDirectives.Add(codeRegion);

cmm.EndDirectives.Add(codeEndRegion);

}

//MemberAttributes.Family is protected

//cmm.Attributes = MemberAttributes.Override | MemberAttributes.Family;

cmm.Attributes = GetMethodAttrMapping(method);

cmm.ReturnType = new CodeTypeReference(method.ReturnType.ToString());

foreach (var p in method.ParameterList.Parameters)

{

CodeParameterDeclarationExpression cpd = new CodeParameterDeclarationExpression();

cpd.Name = p.Identifier.ToString();

cpd.Type = new CodeTypeReference(p.Type.ToString());

cmm.Parameters.Add(cpd);

}

//包含方法{};,会重复生成{};

string codeBody = method.Body.ToString();

codeBody = codeBody.Trim().TrimStart("{").TrimEnd("}").Trim().TrimEnd(";");

if (codeBody != "")

{

CodeSnippetExpression csbody = new CodeSnippetExpression(codeBody);

CodeExpressionStatement stmt = new CodeExpressionStatement(csbody);

//Add the expression statements to the method.

cmm.Statements.Add(stmt);

}

myDesignerClass.Members.Add(cmm);

}

if (m is MemberDeclarationSyntax)

{

}

}

ccu.Namespaces.Add(ns);

//Partial Class

myDesignerClass.IsPartial = true;

ns.Types.Add(myDesignerClass);

return ccu;

}

public static CodeCompileUnit GetDesignerCodeComplieUnit2(CompilationUnitSyntax root) { CodeCompileUnit ccu = new CodeCompileUnit(); var firstMember = root.Members[0]; var namespaceDeclration = (NamespaceDeclarationSyntax)firstMember; var designClassDeclaration = (ClassDeclarationSyntax)namespaceDeclration.Members[0]; var myDesignerClass = new CodeTypeDeclaration(designClassDeclaration.Identifier.ToString()); var initializeComponent = new CodeMemberMethod(); var ns = new CodeNamespace(namespaceDeclration.Name.ToString()); foreach (var m in designClassDeclaration.Members) { if (m is ConstructorDeclarationSyntax) { var ctor = ((ConstructorDeclarationSyntax)m); var codeBody = ctor.Body.ToString(); codeBody = codeBody.Trim().TrimStart("{").TrimEnd("}").Trim().TrimEnd(";"); CodeSnippetExpression csbody = new CodeSnippetExpression(codeBody); CodeExpressionStatement stmt = new CodeExpressionStatement(csbody); //Add the expression statements to the method. // InitializeComponent var cctor = new CodeConstructor(); cctor.Name = ctor.Identifier.ToString(); //var cmm = new CodeMemberMethod(); //cmm.Name = ctor.Identifier.ToString(); //cmm.Attributes = GetCtoRAttrMapping(ctor); //cmm.ReturnType = new CodeTypeReference(typeof(void)); cctor.Statements.Add(stmt); myDesignerClass.Members.Add(cctor); } if (m is FieldDeclarationSyntax) { var F = ((FieldDeclarationSyntax)m); var type = F.Declaration.Type; foreach (var variable in F.Declaration.Variables) { var field = new CodeMemberField(); field.Name = variable.Identifier.ToString(); field.Type = new CodeTypeReference(type.ToString()); field.Attributes = GetFieldAttrMapping(F); //field.InitExpression = new CodePrimitiveExpression(null); myDesignerClass.Members.Add(field); } } if (m is MethodDeclarationSyntax) { var node = m as MethodDeclarationSyntax; #region xml comments var xmlTrivia = node.GetLeadingTrivia() .Select(i => i.GetStructure()) .OfType<DocumentationCommentTriviaSyntax>() .FirstOrDefault(); #endregion var method = (MethodDeclarationSyntax)m; var cmm = new CodeMemberMethod(); cmm.Name = method.Identifier.ToString(); ///XML注释 string[] comments = xmlTrivia.ToString().Split("\r\n".ToCharArray()); foreach (string text in comments) { if (text.Trim() != "") { ments.Add(new CodeCommentStatement(text.Trim().TrimStart("///".ToCharArray()).Trim(), true)); } } if (cmm.Name == "InitializeComponent") { //region CodeRegionDirective codeRegion = new CodeRegionDirective(CodeRegionMode.Start, "Windows 窗体设计器生成的代码"); CodeRegionDirective codeEndRegion = new CodeRegionDirective(CodeRegionMode.End, ""); cmm.StartDirectives.Add(codeRegion); cmm.EndDirectives.Add(codeEndRegion); } //MemberAttributes.Family is protected //cmm.Attributes = MemberAttributes.Override | MemberAttributes.Family; cmm.Attributes = GetMethodAttrMapping(method); cmm.ReturnType = new CodeTypeReference(method.ReturnType.ToString()); foreach (var p in method.ParameterList.Parameters) { CodeParameterDeclarationExpression cpd = new CodeParameterDeclarationExpression(); cpd.Name = p.Identifier.ToString(); cpd.Type = new CodeTypeReference(p.Type.ToString()); cmm.Parameters.Add(cpd); } //包含方法{};,会重复生成{}; string codeBody = method.Body.ToString(); codeBody = codeBody.Trim().TrimStart("{").TrimEnd("}").Trim().TrimEnd(";"); if (codeBody != "") { CodeSnippetExpression csbody = new CodeSnippetExpression(codeBody); CodeExpressionStatement stmt = new CodeExpressionStatement(csbody); //Add the expression statements to the method. cmm.Statements.Add(stmt); } myDesignerClass.Members.Add(cmm); } if (m is MemberDeclarationSyntax) { } } ccu.Namespaces.Add(ns); //Partial Class myDesignerClass.IsPartial = true; ns.Types.Add(myDesignerClass); return ccu; }

窗体的显示,需要逐句进行C#解析,特别是InitializeComponent()方法。

.CS Code其实最简单的就是读取源代码,然后返回就可以了。当设计器添加控件或者绑定事件时,可以通过文本操作进行代码完善。

?

1

2

3

4

5

6

//直接返回代码,最简单

public string GetTextCSCode()

{

Flush();

return __CSTextCode;

}

//直接返回代码,最简单 public string GetTextCSCode() { Flush(); return __CSTextCode; }

CodeDomHostLoader类中有OnComponentRename,在设计器重命名组件时候响应,这里可以修复后台.cs中的控件引用

但此设计器还有很多不完善的地方,后期有时间再完善吧。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流。

public static CodeCompileUnit GetDesignerCodeComplieUnit2(CompilationUnitSyntax root)

{

CodeCompileUnit ccu = new CodeCompileUnit();

var firstMember = root.Members[0];

var namespaceDeclration = (NamespaceDeclarationSyntax)firstMember;

var designClassDeclaration = (ClassDeclarationSyntax)namespaceDeclration.Members[0];

var myDesignerClass = new CodeTypeDeclaration(designClassDeclaration.Identifier.ToString());

var initializeComponent = new CodeMemberMethod();

var ns = new CodeNamespace(namespaceDeclration.Name.ToString());

foreach (var m in designClassDeclaration.Members)

{

if (m is ConstructorDeclarationSyntax)

{

var ctor = ((ConstructorDeclarationSyntax)m);

var codeBody = ctor.Body.ToString();

codeBody = codeBody.Trim().TrimStart("{").TrimEnd("}").Trim().TrimEnd(";");

CodeSnippetExpression csbody = new CodeSnippetExpression(codeBody);

CodeExpressionStatement stmt = new CodeExpressionStatement(csbody);

//Add the expression statements to the method.

// InitializeComponent

var cctor = new CodeConstructor();

cctor.Name = ctor.Identifier.ToString();

//var cmm = new CodeMemberMethod();

//cmm.Name = ctor.Identifier.ToString();

//cmm.Attributes = GetCtoRAttrMapping(ctor);

//cmm.ReturnType = new CodeTypeReference(typeof(void));

cctor.Statements.Add(stmt);

myDesignerClass.Members.Add(cctor);

}

if (m is FieldDeclarationSyntax)

{

var F = ((FieldDeclarationSyntax)m);

var type = F.Declaration.Type;

foreach (var variable in F.Declaration.Variables)

{

var field = new CodeMemberField();

field.Name = variable.Identifier.ToString();

field.Type = new CodeTypeReference(type.ToString());

field.Attributes = GetFieldAttrMapping(F);

//field.InitExpression = new CodePrimitiveExpression(null);

myDesignerClass.Members.Add(field);

}

}

if (m is MethodDeclarationSyntax)

{

var node = m as MethodDeclarationSyntax;

#region xml comments

var xmlTrivia = node.GetLeadingTrivia()

.Select(i => i.GetStructure())

.OfType<DocumentationCommentTriviaSyntax>()

.FirstOrDefault();

#endregion

var method = (MethodDeclarationSyntax)m;

var cmm = new CodeMemberMethod();

cmm.Name = method.Identifier.ToString();

///XML注释

string[] comments = xmlTrivia.ToString().Split("\r\n".ToCharArray());

foreach (string text in comments)

{

if (text.Trim() != "")

{

ments.Add(new CodeCommentStatement(text.Trim().TrimStart("///".ToCharArray()).Trim(), true));

}

}

if (cmm.Name == "InitializeComponent")

{

//region

CodeRegionDirective codeRegion = new CodeRegionDirective(CodeRegionMode.Start, "Windows 窗体设计器生成的代码");

CodeRegionDirective codeEndRegion = new CodeRegionDirective(CodeRegionMode.End, "");

cmm.StartDirectives.Add(codeRegion);

cmm.EndDirectives.Add(codeEndRegion);

}

//MemberAttributes.Family is protected

//cmm.Attributes = MemberAttributes.Override | MemberAttributes.Family;

cmm.Attributes = GetMethodAttrMapping(method);

cmm.ReturnType = new CodeTypeReference(method.ReturnType.ToString());

foreach (var p in method.ParameterList.Parameters)

{

CodeParameterDeclarationExpression cpd = new CodeParameterDeclarationExpression();

cpd.Name = p.Identifier.ToString();

cpd.Type = new CodeTypeReference(p.Type.ToString());

cmm.Parameters.Add(cpd);

}

//包含方法{};,会重复生成{};

string codeBody = method.Body.ToString();

codeBody = codeBody.Trim().TrimStart("{").TrimEnd("}").Trim().TrimEnd(";");

if (codeBody != "")

{

CodeSnippetExpression csbody = new CodeSnippetExpression(codeBody);

CodeExpressionStatement stmt = new CodeExpressionStatement(csbody);

//Add the expression statements to the method.

cmm.Statements.Add(stmt);

}

myDesignerClass.Members.Add(cmm);

}

if (m is MemberDeclarationSyntax)

{

}

}

ccu.Namespaces.Add(ns);

//Partial Class

myDesignerClass.IsPartial = true;

ns.Types.Add(myDesignerClass);

return ccu;

}

小编关于《使用C#和DesignSurface实现简单的窗体设计器》就分享到这儿了,如果内容对你有帮助,记得点赞加收藏哦!

网友评论
网友评论仅供其表达个人看法,并不表明网站立场。
显示评论内容(4)
  1. 上官若雪2024-02-24 17:10上官若雪[上海市网友]43.254.203.188
    这个方法听起来很有趣,C#和DesignSurface让窗体设计更加灵活,期待看到更多相关的分享和教程。
    顶0踩0
  2. 恋童2024-02-24 16:52恋童[新疆网友]103.49.199.128
    C#一直是我的最爱,能和DesignSurface一起使用简直太完美了,相信会为窗体设计带来新的可能性。
    顶4踩0
  3. 绿韵竹舍2024-02-24 16:33绿韵竹舍[北京市网友]203.57.157.154
    好厉害啊,想象一下用C#和DesignSurface设计窗体会有多方便,期待看到更多成果。
    顶25踩0
  4. 几度2024-02-24 16:15几度[山西省网友]43.249.170.171
    这个想法很棒,C#和DesignSurface结合起来可以让窗体设计变得更加直观和高效。
    顶0踩0
推荐阅读
湘ICP备19021678号
Copyright © 2023 邵阳网 i0739.com