草庐IT

c# - 如何编写自动缩放到系统字体和 dpi 设置的 WinForms 代码?

coder 2023-07-07 原文

简介:有很多评论说“WinForms 不能很好地自动缩放到 DPI/字体设置;切换到 WPF。”但是,我认为这是基于 .NET 1.1;看起来他们实际上在 .NET 2.0 中实现自动缩放方面做得很好。至少基于我们迄今为止的研究和测试。但是,如果你们中的一些人知道得更好,我们很乐意听取您的意见。 (请不要争论我们应该切换到 WPF ......现在这不是一个选择。)
问题:

  • WinForms 中的哪些内容不能正确自动缩放,因此应该避免?
  • 程序员在编写 WinForms 代码时应该遵循哪些设计准则,以便它能够很好地自动缩放?

  • 到目前为止,我们已经确定的设计指南:
    community wiki answer下面。
    其中有哪些是不正确的或不充分的?我们应该采用任何其他准则吗?还有其他需要避免的模式吗?对此的任何其他指导将不胜感激。

    最佳答案

    不支持正确缩放的控件:

  • LabelAutoSize = FalseFont继承的。显式设置 Font在控件上,因此它在“属性”窗口中以粗体显示。
  • ListView列宽不缩放。覆盖表单的 ScaleControl改为这样做。见 this answer
  • SplitContainerPanel1MinSize , Panel2MinSizeSplitterDistance属性(property)
  • TextBoxMultiLine = TrueFont继承的。显式设置 Font在控件上,因此它在“属性”窗口中以粗体显示。
  • ToolStripButton的形象。在表单的构造函数中:
  • 套装ToolStrip.AutoSize = False
  • 套装ToolStrip.ImageScalingSize根据 CreateGraphics.DpiX.DpiY
  • 套装ToolStrip.AutoSize = True如果需要。

  • 有时 AutoSize可以留在True但有时它无法在没有这些步骤的情况下调整大小。使用 .NET Framework 4.5.2 无需更改即可工作和 EnableWindowsFormsHighDpiAutoResizing .
  • TreeView的图像。套装ImageList.ImageSize根据 CreateGraphics.DpiX.DpiY .对于 StateImageList , 无需更改 .NET Framework 4.5.1EnableWindowsFormsHighDpiAutoResizing .
  • Form的大小。比例尺固定Form创建后手动。

  • 设计指南:
  • 所有 ContainerControls 必须设置为相同的 AutoScaleMode = Font .
    (字体将处理 DPI 更改和系统字体更改
    尺寸设置; DPI 将只处理 DPI 更改,而不处理对
    系统字体大小设置。)
  • 所有 ContainerControls 也必须设置为相同的 AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F); ,假设 96dpi(请参阅下一个项目符号)和 MS Sans Serif 的默认字体(请参阅第二个项目符号)。这是设计师自动添加的
    根据您打开设计器的 DPI...
    我们许多最古老的设计师文件。也许 Visual Studio .NET(
    VS 2005 之前的版本)没有正确添加。
  • 在 96dpi 中完成您所有的设计师工作(我们也许可以切换到
    120dpi;但互联网上的智慧说坚持96dpi;
    实验在那里;按照设计,这无关紧要,因为它只是更改了 AutoScaleDimensions设计者插入的行)。
    要将 Visual Studio 设置为在高分辨率显示器上以虚拟 96dpi 运行,
    找到它的 .exe 文件,右键单击以编辑属性,然后在兼容性下
    选择“覆盖高 DPI 缩放行为。缩放执行者:系统”。
  • 确保你永远不会在容器级别设置字体......只在
    如果您想要除 MS Sans Serif 之外的应用程序范围的默认字体,请在最基础 Form 的构造函数中使用叶控件或。 (在容器上设置字体似乎关闭
    该容器的自动缩放,因为它按字母顺序出现在 AutoScaleMode 和 AutoScaleDimensions 设置之后。)请注意,如果您确实更改了最基础 Form 的构造函数中的 Font,这将导致您的 AutoScaleDimensions 的计算方式与 6x13 不同;特别是,如果您更改为 Segoe UI(Win 10 默认字体),那么它将是 7x15……您需要触摸设计器中的每个表单,以便它可以重新计算该 .designer 文件中的所有尺寸,包括AutoScaleDimensions = new System.Drawing.SizeF(7F, 15F); .
  • 不要使用 anchor RightBottom anchor 定到一个用户控件......它的
    定位不会自动缩放;相反,放下面板或其他
    容器到您的 UserControl 并将您的其他控件 anchor 定到
    该小组;让面板使用 Dock Right , Bottom , 或 Fill在你的
    用户控件。
  • ResumeLayout 时,只有控件列表中的控件最后
    InitializeComponent被调用将自动缩放...如果你
    动态添加控件,则需要SuspendLayout();AutoScaleDimensions = new SizeF(6F, 13F); AutoScaleMode = AutoScaleMode.Font;ResumeLayout();在你添加它之前在那个控件上。和你的
    如果您不使用 Dock,也需要调整定位
    模式或布局管理器,如 FlowLayoutPanelTableLayoutPanel .
  • 派生自 ContainerControl 的基类应该离开 AutoScaleMode设置为 Inherit (在类 ContainerControl 中设置的默认值;但不是设计者设置的默认值)。如果您将其设置为其他任何内容,然后您的派生类尝试将其设置为 Font(应该如此),那么将其设置为 Font 的行为将清除AutoScaleDimensions的设计师设置,导致实际上关闭自动缩放! (本指南与之前的指南相结合意味着您永远无法在设计器中实例化基类……所有类都需要设计为基类或叶类!)
  • 避免使用 Form.MaxSize静态/在设计器中。 MinSizeMaxSize表格上的缩放比例不如其他所有内容。因此,如果您在 96dpi 中完成所有工作,那么在更高 DPI 时您的 MinSize不会引起问题,但可能没有您预期的那么严格,但是您的 MaxSize可能会限制您的 Size 的缩放,这可能会导致问题。如果你想要 MinSize == Size == MaxSize , 不要在 Designer 中这样做...在你的构造函数或 OnLoad 中这样做覆盖...设置两者 MinSizeMaxSize到您适当缩放的尺寸。
  • 特定 Panel 上的所有控件或 Container应该使用 anchor 定或对接。如果你混合它们,自动缩放由 Panel 完成经常会以微妙的奇怪方式行为不端。
  • 当它进行自动缩放时,它会尝试缩放整个表单......但是,如果在该过程中它遇到了屏幕尺寸的上限,那就是一个硬限制,然后可能会搞砸(剪辑)缩放。因此,您应该确保设计器中 100%/96dpi 的所有窗体的大小不大于 1024x720(对应于 1080p 屏幕上的 150% 或 4K 屏幕上的 Windows 推荐值 300%)。但是你需要减去巨大的 Win10 标题/标题栏......所以更像是 1000x680 最大尺寸......在设计器中它就像 994x642 ClientSize。 (因此,您可以在 ClientSize 上执行 FindAll References 以查找违规者。)
  • 关于c# - 如何编写自动缩放到系统字体和 dpi 设置的 WinForms 代码?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22735174/

    有关c# - 如何编写自动缩放到系统字体和 dpi 设置的 WinForms 代码?的更多相关文章

    1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

      我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

    2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

      总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

    3. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

      很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

    4. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

      关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

    5. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

      给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

    6. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

      我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

    7. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

      我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

    8. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

      如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

    9. ruby - 如何指定 Rack 处理程序 - 2

      Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

    10. ruby-on-rails - Rails 源代码 : initialize hash in a weird way? - 2

      在rails源中:https://github.com/rails/rails/blob/master/activesupport/lib/active_support/lazy_load_hooks.rb可以看到以下内容@load_hooks=Hash.new{|h,k|h[k]=[]}在IRB中,它只是初始化一个空哈希。和做有什么区别@load_hooks=Hash.new 最佳答案 查看rubydocumentationforHashnew→new_hashclicktotogglesourcenew(obj)→new_has

    随机推荐