ウィンドウをいつも最背面にして、壁紙のような機能をつくります。
WPFだとWindowクラスにウィンドウハンドルを取得するプロパティが無いので、自分で取得しなければだめそうです。あまりウィンドウハンドルを使うことはないのですが、今後また使うようなことがあるかもしれないので、残しておきます。
WPFでウィンドウハンドルを取得する
下記コードでWPFのウィンドウハンドルを取得できます。このハンドルを使ってウィンドウを制御します。
public IntPtr Handle
{
get
{
ver helper = new System.Windows.Interop.WindowsInteropHelper(this);
return helper.Handle;
}
}
WPFウィンドウを常時最背面にする
ウィンドウのZオーダーを細かく制御するにはWin32APIを使うことになりそうです。
FindWindow関数でProgram Managerのウィンドウハンドルを取得して、そのハンドルをSetParent関数で親ウィンドウのハンドルに設定することで最背面にしてみます。
using System;
using System.Windows;
namespace AShell2
{
/// <summary>
/// MainWindow.xaml の相互作用ロジック
/// </summary>
public partial class MainWindow : Window
{
[System.Runtime.InteropServices.DllImport("user32.dll", SetLastError = true)]
static extern IntPtr FindWindow(string lpClassName, string lpWindowName);
[System.Runtime.InteropServices.DllImport("user32.dll")]
static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent);
public MainWindow()
{
InitializeComponent();
}
private void Window_Loaded(object sender, RoutedEventArgs e)
{
IntPtr desktop = FindWindow(null, "Program Manager");
if (desktop != null)
{
var helper = new System.Windows.Interop.WindowInteropHelper(this);
SetParent(helper.Handle, desktop);
}
}
}
}
これで一応はうまく壁紙のように最背面に設定することができました。
しかし、サイネージで使おうと思ったらうまく動作しません。
いろいろと調べていたらProgram Magagerとは、Explorerのことのようです。すなわち、ShellがExplorerになっていないとちゃんと動かないようです。要はExplorer Shellを親ウィンドウとして登録しているだけのようです。
組み込み系のシステムでは、カスタムシェルを使って製品化するので、Explorerシェルはないのです。
ウィンドウメッセージをフックして背面に隠す
ウィンドウハンドルをSetParentする方法ではうまく動作しなかったので、ウィンドウメッセージをフックしてSetWindowsPosで裏に隠すことにしました。ちらつきがあるかなと思ったのですが、特にちらつきもなく動作しています。
using System;
using System.Windows;
using System.Runtime.InteropServices;
using System.Windows.Interop;
namespace AShell
{
/// <summary>
/// MainWindow.xaml の相互作用ロジック
/// </summary>
public partial class MainWindow : Window
{
#region Windows API
// ReSharper disable InconsistentNaming
private const int WM_WINDOWPOSCHANGING = 0x0046;
private const uint WM_WINDOWPOSCHANGED = 0x0047;
private const uint SWP_NOSIZE = 0x0001;
private const uint SWP_NOMOVE = 0x0002;
private const uint SWP_NOZORDER = 0x0004;
private const uint SWP_NOACTIVATE = 0x0010;
#endregion
private static readonly IntPtr HWND_BOTTOM = new IntPtr(1);
[DllImport("user32.dll")]
private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int x, int y, int cx, int cy,
uint uFlags);
public MainWindow()
{
InitializeComponent();
}
protected override void OnSourceInitialized(EventArgs e)
{
base.OnSourceInitialized(e);
WindowInteropHelper helper = new WindowInteropHelper(this);
HwndSource source = HwndSource.FromHwnd(helper.Handle);
source.AddHook(new HwndSourceHook(WndProc));
}
private IntPtr WndProc(IntPtr hWnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
{
if ((msg == WM_WINDOWPOSCHANGING) || (msg == WM_WINDOWPOSCHANGED))
{
SetWindowPos(hWnd, HWND_BOTTOM, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE);
}
return IntPtr.Zero;
}
}
}