繁体   English   中英

C#:获取完整的桌面大小?

[英]C#: Get complete desktop size?

如何找出整个桌面的大小? 不是“工作区域”也不是“屏幕分辨率”,两者都只指一个屏幕。 我想找出每个显示器只显示一部分的虚拟桌面的总宽度和高度。

您有两种选择:

  1. PresentationFramework.dll

     SystemParameters.VirtualScreenWidth SystemParameters.VirtualScreenHeight 
  2. System.Windows.Forms.dll中

     SystemInformation.VirtualScreen.Width SystemInformation.VirtualScreen.Height 

如果您正在开发WPF应用程序,请使用第一个选项

我认为现在是时候用一点LINQ把这个答案更新了,这样可以很容易地用一个表达式获得整个桌面大小。

Console.WriteLine(
    Screen.AllScreens.Select(screen=>screen.Bounds)
    .Aggregate(Rectangle.Union)
    .Size
);

我的原始答案如下:


我想你想要的是这样的:

int minx, miny, maxx, maxy;
minx = miny = int.MaxValue;
maxx = maxy = int.MinValue;

foreach(Screen screen in Screen.AllScreens){
    var bounds = screen.Bounds;
    minx = Math.Min(minx, bounds.X);
    miny = Math.Min(miny, bounds.Y);
    maxx = Math.Max(maxx, bounds.Right);
    maxy = Math.Max(maxy, bounds.Bottom);
}

Console.WriteLine("(width, height) = ({0}, {1})", maxx - minx, maxy - miny);

请记住,这并不能说明整个故事。 多个监视器可以交错排列,或者以非矩形形状排列。 因此,可能不是(minx,miny)和(maxx,maxy)之间的所有空间都是可见的。

编辑:

我刚刚意识到使用Rectangle.Union代码可能会更简单:

Rectangle rect = new Rectangle(int.MaxValue, int.MaxValue, int.MinValue, int.MinValue);

foreach(Screen screen in Screen.AllScreens)
    rect = Rectangle.Union(rect, screen.Bounds);

Console.WriteLine("(width, height) = ({0}, {1})", rect.Width, rect.Height);

校验:

SystemInformation.VirtualScreen.Width
SystemInformation.VirtualScreen.Height

这并没有回答这个问题,只是增加了对所有屏幕内窗口点(位置)的额外见解。

使用下面的代码来确定Point(例如,最后一个已知的窗口位置)是否在整个桌面的范围内。 如果没有,将窗口的位置重置为默认的pBaseLoc ;

代码不考虑TaskBar或其他工具栏,你自己在那里。

示例使用:将窗口位置从站A保存到数据库。 用户使用2个监视器登录到B站并将窗口移动到第2个监视器,注销保存新位置。 返回站A ,除非使用上述代码,否则不会显示窗口。

我进一步解决了将userID和工作站的IP(&winLoc)保存到给定应用程序的数据库或本地用户prefs文件,然后加载该工作站和应用程序的用户首选项。

Point pBaseLoc = new Point(40, 40)
int x = -500, y = 140;
Point pLoc = new Point(x, y);
bool bIsInsideBounds = false;

foreach (Screen s in Screen.AllScreens)
{
    bIsInsideBounds = s.Bounds.Contains(pLoc);
    if (bIsInsideBounds) { break; }
}//foreach (Screen s in Screen.AllScreens)

if (!bIsInsideBounds) { pLoc = pBaseLoc;  }

this.Location = pLoc;

要获得监视器的物理像素大小,您可以使用它。

static class DisplayTools
{
    [DllImport("gdi32.dll")]
    static extern int GetDeviceCaps(IntPtr hdc, int nIndex);

    private enum DeviceCap
    {
        Desktopvertres = 117,
        Desktophorzres = 118
    }

    public static Size GetPhysicalDisplaySize()
    {
        Graphics g = Graphics.FromHwnd(IntPtr.Zero);
        IntPtr desktop = g.GetHdc();

        int physicalScreenHeight = GetDeviceCaps(desktop, (int)DeviceCap.Desktopvertres);
        int physicalScreenWidth = GetDeviceCaps(desktop, (int)DeviceCap.Desktophorzres);

        return new Size(physicalScreenWidth, physicalScreenHeight);
    }


}

您可以使用System.Drawing的Bounds。

您可以创建这样的函数

public System.Windows.Form.Screen[] GetScreens(){
    Screen[] screens = Screen.AllScreens;
    return screens;
}

而且你可以在这样的变量中获得屏幕一,二等等:

System.Windows.Form.Screen[] screens = func.GetScreens();
System.Windows.Form.Screen screen1 = screens[0];

然后你可以得到屏幕的界限:

System.Drawing.Rectangle screen1Bounds = screen1.Bounds;

使用此代码,您将获得所有属性,如WidthHeight等。

此方法使用Left和Top的最低值以及Right和Bottom的最高值返回包含所有屏幕边界的矩形...

static Rectangle GetDesktopBounds() {
   var l = int.MaxValue;
   var t = int.MaxValue;
   var r = int.MinValue;
   var b = int.MinValue;
   foreach(var screen in Screen.AllScreens) {
      if(screen.Bounds.Left   < l) l = screen.Bounds.Left  ;
      if(screen.Bounds.Top    < t) t = screen.Bounds.Top   ;
      if(screen.Bounds.Right  > r) r = screen.Bounds.Right ;
      if(screen.Bounds.Bottom > b) b = screen.Bounds.Bottom;
   }
   return Rectangle.FromLTRB(l, t, r, b);
}

我认为获得“真实”屏幕大小的最佳方法是直接从视频控制器获取值。


    using System;
    using System.Management;
    using System.Windows.Forms;

    namespace MOS
    {
        public class ClassMOS
        {
            public static void Main()
            {
                try
                {
                    ManagementObjectSearcher searcher = 
                        new ManagementObjectSearcher("root\\CIMV2", 
                        "SELECT * FROM Win32_VideoController"); 

                    foreach (ManagementObject queryObj in searcher.Get())
                    {
                        Console.WriteLine("CurrentHorizontalResolution: {0}", queryObj["CurrentHorizontalResolution"]);
                        Console.WriteLine("-----------------------------------");
                        Console.WriteLine("CurrentVerticalResolution: {0}", queryObj["CurrentVerticalResolution"]);
                    }
                }
                catch (ManagementException e)
                {
                    MessageBox.Show("An error occurred while querying for WMI data: " + e.Message);
                }
            }
        }
}

这应该做的工作;)问候......

获取没有任何依赖关系的虚拟显示的大小

public enum SystemMetric
{
    VirtualScreenWidth = 78, // CXVIRTUALSCREEN 0x0000004E 
    VirtualScreenHeight = 79, // CYVIRTUALSCREEN 0x0000004F 
}

[DllImport("user32.dll")]
public static extern int GetSystemMetrics(SystemMetric metric);

public static Size GetVirtualDisplaySize()
{
    var width = GetSystemMetrics(SystemMetric.VirtualScreenWidth);
    var height = GetSystemMetrics(SystemMetric.VirtualScreenHeight);

    return new Size(width, height);
}

派生自 SM_XVIRTUALSCREEN 等的值是物理坐标,不等于控制面板中屏幕缩放 125%、150%、175% 的逻辑坐标。

您会发现许多窗口操作尚不可用或可能永远无法使用。 我使用 PInvoke.User32 nuget 包取得了最大的成功。

var screenX = PInvoke.User32.GetSystemMetrics(PInvoke.User32.SystemMetric.SM_CXSCREEN);
var screenY = PInvoke.User32.GetSystemMetrics(PInvoke.User32.SystemMetric.SM_CYSCREEN);

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM