web-dev-qa-db-ja.com

Windows10で複数のディスプレイの配置を自動化する

私は、すべての作業スペースに外部モニターを備えたオープンスペースのオフィスタイプで働いています。設定された作業スペースがないため、新しい外部モニター画面に接続するたびに、デフォルトで2つの画面が隣り合って開きます。私は、アレンジメントウィンドウで2つの画面が重なり合うモードでのみ作業します。

Windows 10で配置を自動化する方法はありますか?新しいモニターを接続してクイックバッチコマンドを実行し、配置を設定した場合はどうなりますか?

定期的に行うと3回程度のクリックで済むことは知っていますが、毎日発生すると煩わしくなります。

ありがとう!

3
kftb

これは、PowerShellを使用して実行できます。まず、「MoveScreens.ps1」というPowershellスクリプトを作成し、次のコードをコピーして貼り付けます。

Function Set-ScreenPosition { 
param ( 
[Parameter(Mandatory=$true, 
  Position = 0)] 
[int] 
$x, 
[Parameter(Mandatory=$true, 
  Position = 1)] 
[int] 
$y 
) 
$pinvokeCode = @" 
using System; 
using System.Runtime.InteropServices; 
namespace Mover
{

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
    public struct DEVMODE
    {
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 32)]
        public string dmDeviceName;

        public short dmSpecVersion;
        public short dmDriverVersion;
        public short dmSize;
        public short dmDriverExtra;
        public int dmFields;
        public int dmPositionX;
        public int dmPositionY;
        public int dmDisplayOrientation;
        public int dmDisplayFixedOutput;
        public short dmColor;
        public short dmDuplex;
        public short dmYResolution;
        public short dmTTOption;
        public short dmCollate;

        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 32)]
        public string dmFormName;

        public short dmLogPixels;
        public short dmBitsPerPel;
        public int dmPelsWidth;
        public int dmPelsHeight;
        public int dmDisplayFlags;
        public int dmDisplayFrequency;
        public int dmICMMethod;
        public int dmICMIntent;
        public int dmMediaType;
        public int dmDitherType;
        public int dmReserved1;
        public int dmReserved2;
        public int dmPanningWidth;
        public int dmPanningHeight;
    };
    public class NativeMethods
    {
        // PInvoke declaration for EnumDisplaySettings Win32 API
        [System.Runtime.InteropServices.DllImport("user32.dll")]
        public static extern int EnumDisplaySettings(string lpszDeviceName, int iModeNum, ref DEVMODE lpDevMode);

        // PInvoke declaration for ChangeDisplaySettings Win32 API
        [System.Runtime.InteropServices.DllImport("user32.dll")]
        public static extern int ChangeDisplaySettings(ref DEVMODE lpDevMode, int dwFlags);


        // constants
        public const int ENUM_CURRENT_SETTINGS = -1;
        public const int CDS_UPDATEREGISTRY = 0x01;
        public const int CDS_TEST = 0x02;
        public const int DISP_CHANGE_SUCCESSFUL = 0;
        public const int DISP_CHANGE_RESTART = 1;
        public const int DISP_CHANGE_FAILED = -1;



        public static DEVMODE CreateDevmode()
        {
            DEVMODE dm = new DEVMODE();
            dm.dmDeviceName = new String(new char[32]);
            dm.dmFormName = new String(new char[32]);
            dm.dmSize = (short)Marshal.SizeOf(dm);
            return dm;
        }
    }



  public class PrimaryScreenPosition
  { 
  static public string ChangePosition(int x, int y) 
  { 
  DEVMODE dm = CreateDevmode(); 
  if (0 != NativeMethods.EnumDisplaySettings(@"\\.\DISPLAY1", NativeMethods.ENUM_CURRENT_SETTINGS, ref dm)) 
  { 
    dm.dmPositionX = x;
    dm.dmPositionY = y;

  int iRet = NativeMethods.ChangeDisplaySettings(ref dm, NativeMethods.CDS_TEST); 
  if (iRet == NativeMethods.DISP_CHANGE_FAILED) 
  { 
  return "Unable To Process Your Request. Sorry For This Inconvenience."; 
  } 
  else 
  { 
  iRet = NativeMethods.ChangeDisplaySettings(ref dm, 0); 
  switch (iRet) 
  { 
  case NativeMethods.DISP_CHANGE_SUCCESSFUL: 
  { 
  return "Success"; 
  } 
  case NativeMethods.DISP_CHANGE_RESTART: 
  { 
  return "You Need To Reboot For The Change To Happen.\n If You Feel Any Problem After Rebooting Your Machine\nThen Try To Change Resolution In Safe Mode."; 
  } 
  default: 
  { 
  return "Failed To Change The Position"; 
  } 
  } 
  } 
  } 
  else 
  { 
  return "Failed To Change The Position."; 
  } 
  } 
  private static DEVMODE1 GetDevMode1() 
  { 
  DEVMODE1 dm = new DEVMODE1(); 
  dm.dmDeviceName = new String(new char[32]); 
  dm.dmFormName = new String(new char[32]); 
  dm.dmSize = (short)Marshal.SizeOf(dm); 
  return dm; 
  } 
  } 
} 
"@ 
Add-Type $pinvokeCode -ErrorAction SilentlyContinue 
[Resolution.PrmaryScreenPosition]::ChangePosition($x,$y) 
}

次に、上記のように実行する別のスクリプトを作成します(数値は、それぞれ2番目の画面の幅と高さです。明らかに、最初のスクリプトを配置する場所へのパスを更新する必要があります!):

.\C:\Temp\MoveScreeens.ps1
Set-ScreenPosition 1920 1080

次に、2番目のスクリプトを実行するだけで、プライマリが下部にある状態で画面がスタックされます。

警告:これは、マシンのスクリプト実行設定によっては機能しない場合があります。これに遭遇した場合は、適切な設定を有効にしてスクリプトを実行する必要があります。

PowerShell.exe -ExecutionPolicy Bypass -File .\MoveScreens_Stack.ps1

別の警告:このスクリプトは、モニターが水平に配置され、左側のディスプレイが#1であることを前提としています。それらが異なる配置にある場合(たとえば、垂直配置ではすでに)、奇妙なことが起こる可能性があります!

2
motosubatsu