4
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

【.NET Framework】PresantationCore.dllを読み込むとウィンドウが縮小する問題

Posted at

.NET FrameworkのWindowsフォームアプリケーションで、ウィンドウの大きさを変えるようなコードを書いていないのにもかかわらず、突然小さくなってしまった経験はないでしょうか?
それはもしかすると、PresentationCore.dllの読み込みが原因になっているかもしれません。その場合、dpiAwareの設定を追加することで解決できます。

この記事では、発生条件再現コード修正方法原因を説明します。

直し方だけ知りたい方は、修正方法をお読みください。

縮小したときの見た目

以下のフォームは、ボタンを押すと不具合が発生するようになっています。
縮小前
押下前.png

縮小後
押下後.png

発生条件

qiita1.png

(Windows 11の場合)

  1. システム > ディスプレイの「拡大/縮小」を100%以外に設定する
  2. PresentationCore.dllで定義されているクラスや構造体を使用する(例:System.Windows.DataFormat

再現コード

再現コードでは、ボタンが配置されたフォームを生成します。ボタンを押すとウィンドウが縮小します。

using System;
using System.Windows.Forms;

namespace ShrinkFormExample
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            // 前提:PresentationCore.dllを参照に追加すること。
            new System.Windows.DataFormat("test", 0);
        }
    }
}

※コンパイルが面倒な場合は、PowerShellに以下のコードをペーストして実行してください。

Add-Type -Assembly System.Windows.Forms

# フォーム生成
$form = New-Object System.Windows.Forms.Form
$form.Size = New-Object System.Drawing.Size(640, 480)

# ボタン配置
$button = New-Object System.Windows.Forms.Button
$button.Text = "クリックで縮小"
$button.Location = New-Object System.Drawing.Point(100, 100)
$button.size = New-Object System.Drawing.Size(200, 100)
$form.Controls.Add($button)
$button.Add_Click({
    # ボタンを押した場合の処理
    # 以下のコードが実行されると、ウィンドウが縮小する。
    Add-Type -AssemblyName PresentationCore
    [System.Windows.DataFormat]::new("test", 0)
})
$form.ShowDialog()

修正方法

「app.manifestを追加する」か「SetProcessDpiAwareness(0)を実行する」ことで修正できます。

app.manifestの追加

「追加」>「新しい項目の追加」>「全般」にある「アプリケーションマニフェストファイル」を選択します。

image.png

その後、以下のコードを追加します。

<application xmlns="urn:schemas-microsoft-com:asm.v3">
    <windowsSettings>
      <dpiAware xmlns="http://schemas.microsoft.com/SMI/2005/WindowsSettings">true</dpiAware>
      <longPathAware xmlns="http://schemas.microsoft.com/SMI/2016/WindowsSettings">true</longPathAware>
    </windowsSettings>
</application>

SetProcessDpiAwarenessの実行

app.manifestが用意できない場合、Windows APIのSetProcessDpiAwareness関数をコンストラクタ等から呼び出すことで縮小の影響を無くせます。

[DllImport("Shcore.dll", EntryPoint = "SetProcessDpiAwareness")]
private static extern int SetProcessDpiAwareness(int value);

public Form()
{
    SetProcessDpiAwareness(0);
}

原因

PresentationCore.dllで定義されるクラスや構造体にアクセスするとModuleInitializerが実行されますが、その中の処理に問題があります。

GitHubにあがっている実装を読むと、ModuleInitializer.csの49行目に原因があります。
SetProcessDPIAware_Internalというメソッドが実行されるコードになっていますが、実体はWindows APIのSetProcessDPIAwareをDllImportしたものになっており、この関数が実行されるとウィンドウの縮小が発生します。
この挙動は.NET Frameworkだけでなく、ネイティブコードでも発生します。以下は、C++で再現したコードです。

// ShrinkFormExampleInWin32.cpp : Defines the entry point for the application.
//

#include "framework.h"
#include "ShrinkFormExampleInWin32.h"

#define MAX_LOADSTRING 100
constexpr int Button_ID = 1;

// Global Variables:
HINSTANCE hInst;                                // current instance
WCHAR szTitle[MAX_LOADSTRING];                  // The title bar text
WCHAR szWindowClass[MAX_LOADSTRING];            // the main window class name

// Forward declarations of functions included in this code module:
ATOM                MyRegisterClass(HINSTANCE hInstance);
BOOL                InitInstance(HINSTANCE, int);
LRESULT CALLBACK    WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK    About(HWND, UINT, WPARAM, LPARAM);

int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
                      _In_opt_ HINSTANCE hPrevInstance,
                      _In_ LPWSTR    lpCmdLine,
                      _In_ int       nCmdShow) {
    UNREFERENCED_PARAMETER(hPrevInstance);
    UNREFERENCED_PARAMETER(lpCmdLine);

    // TODO: Place code here.

    // Initialize global strings
    LoadStringW(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
    LoadStringW(hInstance, IDC_SHRINKFORMEXAMPLEINWIN32, szWindowClass, MAX_LOADSTRING);
    MyRegisterClass(hInstance);

    // Perform application initialization:
    if (!InitInstance(hInstance, nCmdShow)) {
        return FALSE;
    }

    HACCEL hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_SHRINKFORMEXAMPLEINWIN32));

    MSG msg;

    // Main message loop:
    while (GetMessage(&msg, nullptr, 0, 0)) {
        if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) {
            TranslateMessage(&msg);
            DispatchMessage(&msg);
        }
    }
    return (int)msg.wParam;
}

//
//  FUNCTION: MyRegisterClass()
//
//  PURPOSE: Registers the window class.
//
ATOM MyRegisterClass(HINSTANCE hInstance) {
    WNDCLASSEXW wcex;

    wcex.cbSize = sizeof(WNDCLASSEX);

    wcex.style = CS_HREDRAW | CS_VREDRAW;
    wcex.lpfnWndProc = WndProc;
    wcex.cbClsExtra = 0;
    wcex.cbWndExtra = 0;
    wcex.hInstance = hInstance;
    wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_SHRINKFORMEXAMPLEINWIN32));
    wcex.hCursor = LoadCursor(nullptr, IDC_ARROW);
    wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
    wcex.lpszMenuName = MAKEINTRESOURCEW(IDC_SHRINKFORMEXAMPLEINWIN32);
    wcex.lpszClassName = szWindowClass;
    wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));

    return RegisterClassExW(&wcex);
}

//
//   FUNCTION: InitInstance(HINSTANCE, int)
//
//   PURPOSE: Saves instance handle and creates main window
//
//   COMMENTS:
//
//        In this function, we save the instance handle in a global variable and
//        create and display the main program window.
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow) {
    hInst = hInstance; // Store instance handle in our global variable

    HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
                              CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr);

    if (!hWnd) {
        return FALSE;
    }

    CreateWindow(
        L"BUTTON",
        L"クリックで縮小",
        WS_CHILD | WS_VISIBLE,
        100, 50, 100, 30,
        hWnd,
        (HMENU)Button_ID,
        hInstance,
        NULL
    );

    ShowWindow(hWnd, nCmdShow);
    UpdateWindow(hWnd);

    return TRUE;
}

//
//  FUNCTION: WndProc(HWND, UINT, WPARAM, LPARAM)
//
//  PURPOSE: Processes messages for the main window.
//
//  WM_COMMAND  - process the application menu
//  WM_PAINT    - Paint the main window
//  WM_DESTROY  - post a quit message and return
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) {
    switch (message) {
        case WM_COMMAND:
        {
            if (LOWORD(wParam) == Button_ID) {
                // ボタンをクリックしたときの処理
                SetProcessDPIAware();
            } else {
                int wmId = LOWORD(wParam);
                // Parse the menu selections:
                switch (wmId) {
                    case IDM_ABOUT:
                        DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
                        break;
                    case IDM_EXIT:
                        DestroyWindow(hWnd);
                        break;
                    default:
                        return DefWindowProc(hWnd, message, wParam, lParam);
                }
            }
        }
        break;
        case WM_PAINT:
        {
            PAINTSTRUCT ps;
            HDC hdc = BeginPaint(hWnd, &ps);
            // TODO: Add any drawing code that uses hdc here...
            EndPaint(hWnd, &ps);
        }
        break;
        case WM_DESTROY:
            PostQuitMessage(0);
            break;
        default:
            return DefWindowProc(hWnd, message, wParam, lParam);
    }
    return 0;
}

// Message handler for about box.
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam) {
    UNREFERENCED_PARAMETER(lParam);
    switch (message) {
        case WM_INITDIALOG:
            return (INT_PTR)TRUE;
        case WM_COMMAND:
            if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL) {
                EndDialog(hDlg, LOWORD(wParam));
                return (INT_PTR)TRUE;
            }
            break;
    }
    return (INT_PTR)FALSE;
}

SetProcessDPIAwareとは

DPIスケールの設定を変更する関数です。DPIスケールとは、ディスプレイの解像度に応じてウィンドウやそのパーツを拡大縮小する機能です。これが行われると、文字がぼやけて見える場合があります。大雑把に書けば、この関数はその拡大・縮小を無効化するためのものです(つまり、今回のケースならぼやけないようにウィンドウの大きさを小さくします)。
こういった挙動をするため、アプリケーションが起動した直後に使われるべきですが、ユーザが操作している間に実行されてしまうことで変な挙動をしているように見えてしまいます。

おまけ

どういうケースで問題につながるのか?

本来、PresentationCore.dllはWPFアプリケーションで使われているライブラリで、新しくプロジェクトを作成すると必ず参照されています。(おそらく)WPF以外で使われることを想定して作られていないため、設定が不足すると意図しないタイミングでウィンドウが縮小します。

なので、そもそも「PresentationCore.dllをWindowsフォームアプリケーションで使わなければよい」と考えることができます。
ところが、場合によっては外部のライブラリがPresentationCore.dllを利用している場合があり、意図しないところで発生する可能性があります。

ケース:CommonOpenFileDialogクラス(Microsoft.WindowsAPICodePack)

このteratailの質問では、WindowsAPICodePackに実装されているCommonOpenFileDialogクラスのShowDialogメソッドを利用した結果、問題に遭遇しています。

このケースでは、CommonOpenFileDialog.ShowDialogメソッド内で実行されるCommonFileDialog.ApplyNativeSettingsメソッドに原因があります。ApplyNativeSettingsメソッドの686行目の処理が行われると、System.Windows.Application.Currentに対してNullチェックが行われます。ApplicationクラスはPresentationCore.dllで定義されているため、そのクラスが判定された時点でModuleInitializerが実行され、問題につながります。

この問題は、上記で挙げた解決策か、System.Windows.Forms.OpenFileDialogクラスを代わりに利用することで解決できます。

動作を再現したコードについて

自分のGitHubにプロジェクトファイルをアップロードしたので、良ければ参考にしてください。プロジェクトには以下の内容が含まれています。

  • 縮小が発生するフォームアプリケーション
  • 縮小の発生を修正したフォームアプリケーション
  • Windows APIを使った縮小例アプリ
  • WindowsAPICodePackを使って縮小させた例

参考ページ

私が今回の問題に当たった時、teratailとStack Overflowを参考に解決しました。ライブラリ内のどの部分が原因かはググっても見つからなかったので、独自に調査したうえで解決方法と一緒にこの記事にをまとめさせていただきました。

4
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?