本文將告訴大家如何在 dotnet 裡面的後臺執行緒向自己程序內的視窗傳送訊息
核心是透過 XSendEvent 傳送訊息,傳送訊息想要有反應需要另開 XOpenDisplay 獲取 display 物件,最後再將其關閉才能傳送出去
核心程式碼如下
_ = Task.Run(async () =>
{
while (true)
{
await Task.Delay(TimeSpan.FromSeconds(1));
var display1 = XOpenDisplay(IntPtr.Zero);
try
{
var @event = new XEvent
{
ClientMessageEvent =
{
type = XEventName.ClientMessage,
send_event = true,
window = handle,
message_type = 0,
format = 32,
ptr1 = 0,
ptr2 = 0,
ptr3 = 0,
ptr4 = 0,
}
};
XSendEvent(display1, handle, false, 0, ref @event);
}
finally
{
XCloseDisplay(display1);
}
}
});
以上的 handle 是一個 X11 視窗指標,程式碼如下
var display = XOpenDisplay(IntPtr.Zero);
var screen = XDefaultScreen(display);
var handle = XCreateWindow(display, rootWindow, 0, 0, xDisplayWidth, xDisplayHeight, 5,
32,
(int)CreateWindowArgs.InputOutput,
visual,
(nuint)valueMask, ref xSetWindowAttributes);
如果在 Task.Run 後臺執行緒裡面,使用的是外面的 display 物件,則傳送失敗
以上程式碼放在 github 和 gitee 上,可以使用如下命令列拉取程式碼
先建立一個空資料夾,接著使用命令列 cd 命令進入此空資料夾,在命令列裡面輸入以下程式碼,即可獲取到本文的程式碼
git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin ee9c8da351838b0ec3b8ab577a6c9904e024517d
以上使用的是 gitee 的源,如果 gitee 不能訪問,請替換為 github 的源。請在命令列繼續輸入以下程式碼,將 gitee 源換成 github 源進行拉取程式碼
git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin ee9c8da351838b0ec3b8ab577a6c9904e024517d
獲取程式碼之後,進入 DikalehebeekaJaqunicobo 資料夾,即可獲取到原始碼
更多 X11 開發請參閱 部落格導航
再經過更多的測試和閱讀大佬們的示例程式碼,發現只需帶上 XFlush 即可,更改之後的程式碼如下
_ = Task.Run(async () =>
{
while (true)
{
await Task.Delay(TimeSpan.FromSeconds(1));
var @event = new XEvent
{
ClientMessageEvent =
{
type = XEventName.ClientMessage,
send_event = true,
window = handle,
message_type = 0,
format = 32,
ptr1 = 0,
ptr2 = 0,
ptr3 = 0,
ptr4 = 0,
}
};
XSendEvent(display, handle, false, 0, ref @event);
XFlush(display);
}
});
這裡由於需要進行多執行緒共用一個 display 物件,根據 X11 文件,需要新增 XInitThreads 方法才能確保安全
完全的程式碼如下
// See https://aka.ms/new-console-template for more information
using CPF.Linux;
using System;
using System.Diagnostics;
using System.Runtime;
using static CPF.Linux.XLib;
XInitThreads();
var display = XOpenDisplay(IntPtr.Zero);
var screen = XDefaultScreen(display);
var rootWindow = XDefaultRootWindow(display);
XMatchVisualInfo(display, screen, 32, 4, out var info);
var visual = info.visual;
var valueMask =
//SetWindowValuemask.BackPixmap
0
| SetWindowValuemask.BackPixel
| SetWindowValuemask.BorderPixel
| SetWindowValuemask.BitGravity
| SetWindowValuemask.WinGravity
| SetWindowValuemask.BackingStore
| SetWindowValuemask.ColorMap
//| SetWindowValuemask.OverrideRedirect
;
var xSetWindowAttributes = new XSetWindowAttributes
{
backing_store = 1,
bit_gravity = Gravity.NorthWestGravity,
win_gravity = Gravity.NorthWestGravity,
//override_redirect = true, // 設定視窗的override_redirect屬性為True,以避免視窗管理器的干預
colormap = XCreateColormap(display, rootWindow, visual, 0),
border_pixel = 0,
background_pixel = 0,
};
var xDisplayWidth = XDisplayWidth(display, screen) / 2;
var xDisplayHeight = XDisplayHeight(display, screen) / 2;
var handle = XCreateWindow(display, rootWindow, 0, 0, xDisplayWidth, xDisplayHeight, 5,
32,
(int)CreateWindowArgs.InputOutput,
visual,
(nuint)valueMask, ref xSetWindowAttributes);
XEventMask ignoredMask = XEventMask.SubstructureRedirectMask | XEventMask.ResizeRedirectMask |
XEventMask.PointerMotionHintMask;
var mask = new IntPtr(0xffffff ^ (int)ignoredMask);
XSelectInput(display, handle, mask);
XMapWindow(display, handle);
XFlush(display);
var white = XWhitePixel(display, screen);
var black = XBlackPixel(display, screen);
var gc = XCreateGC(display, handle, 0, 0);
XSetForeground(display, gc, white);
XSync(display, false);
_ = Task.Run(async () =>
{
while (true)
{
await Task.Delay(TimeSpan.FromSeconds(1));
var @event = new XEvent
{
ClientMessageEvent =
{
type = XEventName.ClientMessage,
send_event = true,
window = handle,
message_type = 0,
format = 32,
ptr1 = 0,
ptr2 = 0,
ptr3 = 0,
ptr4 = 0,
}
};
XSendEvent(display, handle, false, 0, ref @event);
XFlush(display);
}
});
while (true)
{
var xNextEvent = XNextEvent(display, out var @event);
if (xNextEvent != 0)
{
Console.WriteLine($"xNextEvent {xNextEvent}");
break;
}
if (@event.type == XEventName.Expose)
{
XDrawLine(display, handle, gc, 0, 0, 100, 100);
}
Console.WriteLine(@event.type);
}
Console.WriteLine("Hello, World!");
以上程式碼放在 github 和 gitee 上,可以使用如下命令列拉取程式碼
先建立一個空資料夾,接著使用命令列 cd 命令進入此空資料夾,在命令列裡面輸入以下程式碼,即可獲取到本文的程式碼
git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin c32c47812df8064445019dd9295867da802643ba
以上使用的是 gitee 的源,如果 gitee 不能訪問,請替換為 github 的源。請在命令列繼續輸入以下程式碼,將 gitee 源換成 github 源進行拉取程式碼
git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin c32c47812df8064445019dd9295867da802643ba
獲取程式碼之後,進入 DikalehebeekaJaqunicobo 資料夾,即可獲取到原始碼
更多 X11 開發請參閱 部落格導航