在电子中更改 BrowserWindow 的系统 z 顺序,可能吗?

Max*_*xim 3 javascript desktop-application electron

我需要控制主窗口的 z-order 可以这么说。总是在最前面不是我的情况。我想把我的窗口放在所有其他窗口和桌面之上。是否可以?是否有类似于 C++ SetWindowPos 的函数?或者也许是一些解决方法?

Ven*_*ryx 5

首先,您应该将窗口发送到 z-index 堆栈的后面/底部,然后禁用窗口的 z-index 更改。

我尝试在我的项目中使用它,并最终成功,如下所示。

但是,请注意我的一些代码注释可能不准确。我对 node-ffi 不是很了解,只能通过反复试验使其工作;我的代码注释只是我试图“理解”我观察到的行为的尝试。

import ffi from "ffi";
import ref from "ref";
import Struct from "ref-struct";

const HWND_BOTTOM = 1;
export let SetWindowPos_Flags = {
    NOSIZE: 0x0001,
    NOMOVE: 0x0002,
    NOACTIVATE: 0x0010,
}

export var user32 = new ffi.Library("user32", {
    // return, handle, handleInsertAfter, x, y, cx, cy, flags
    SetWindowPos: ["bool", ["int32", "int32", "int32", "int32", "int32", "int32", "uint32"]],
});

export function SendWindowToBack(handle: number) {
    user32.SetWindowPos(handle, HWND_BOTTOM, 0, 0, 0, 0, SetWindowPos_Flags.NOMOVE | SetWindowPos_Flags.NOSIZE | SetWindowPos_Flags.NOACTIVATE);
}

export const WM_WINDOWPOSCHANGING = 0x0046;
export const SWP_NOZORDER = 0x0004;
let WINDOWPOS = Struct({
    hwndInsertAfter: ffi.types.int32,
    hwnd: ffi.types.int32,
    x: ffi.types.int32,
    y: ffi.types.int32,
    cx: ffi.types.int32,
    cy: ffi.types.int32,
    flags: ffi.types.uint32,
});
export function AddHook(window: Electron.BrowserWindow, disableZIndexChanging = false) {
    window.hookWindowMessage(WM_WINDOWPOSCHANGING, (wParam: Buffer, lParam: Buffer)=> {
        // The "lParam" buffer holds the address to a place in unmanaged memory, which itself holds the address to the actual (unmanaged) struct-data.
        // Thus: js-pseudo-pointer (8-bytes; the lParam Buffer) -> c-pointer (8-bytes, unmanaged) -> struct-data (28-bytes, unmanaged)
        // However, the lParam buffer does not realize it is/could-be a "pseudo-pointer" -- all it knows is that it's holding some random numbers.
        // To access the unmanaged struct-data, we have to tell the js-side that the contents of lParam are actually the address explained above.
        // Then we'll be able to use the Buffer.deref() function to correctly obtain the final struct-data.

        // To do this, we:
        // 1) Create a new js-pseudo-pointer (using the modified Buffer class), whose contents/what-it-points-to is marked as of type "pointer to a pointer" (ie. the c-pointer entry above)
        let lParam2 = Buffer.alloc(8);
        lParam2["type"] = ref.refType(WINDOWPOS);

        // 2) Fill the js-pseudo-pointer with the actual address to that c-pointer (just copy this address from the unmarked lParam Buffer)
        lParam.copy(lParam2);

        // 3) Dereference our js-pseudo-pointer, retrieving the actual struct-data bytes
        let actualStructDataBuffer = lParam2["deref"]() as Buffer;

        // 4) Convert the struct-data bytes into a regular JavaScript object
        let windowPos = actualStructDataBuffer["deref"]() as {hwndInsertAfter: number, hwnd: number, x: number, y: number, cx: number, cy: number, flags: number};

        // 5) Modify the 7th integer (the flags field) in the (unmanaged) struct-data, to include the new SWP_NOZORDER flag
        if (disableZIndexChanging) {
            //lParam.flags |= SWP_NOZORDER;
            let newFlags = windowPos.flags | SWP_NOZORDER;
            actualStructDataBuffer.writeUInt32LE(newFlags, 6);
        }
    });
}

// This is the function you actually call from the rest of your program.
export function SendElectronWindowToBackAndKeepItThere(window: Electron.BrowserWindow) {
    let handleAsBuffer = window.getNativeWindowHandle();
    let handleAsNumber = ref.types.int64.get(handleAsBuffer, 0);
    SendWindowToBack(handleAsNumber);
    AddHook(window, true);
}
Run Code Online (Sandbox Code Playgroud)

  • 这太棒了,谢谢您的分享:) 对于任何可能尝试在最新电子版本(我使用的是 8.0.0)上使用它的人,您需要使用 `ffi-napi`、`ref-napi`、`ref-支持 Nodejs 12(最新电子版本使用)的 struct-napi`。 (2认同)
  • @Flaff 对代码做任何你想做的事情——考虑代码块公共域(不需要归属)。如果您需要特定的许可证,那么您也可以考虑使用 MIT 许可。(我在 GitHub 上指定的标准许可证) (2认同)
  • @Venryx这个[链接](https://gist.github.com/lowfront/528a658d25e4357c07c8ea4f1b2cb46c)是在Node.js 14版本和Electron 14版本上运行的示例。我是参考你的代码做出来的。谢谢。 (2认同)

Vad*_*gon 3

如果这不是您可以使用父/子窗口完成的事情,那么您可以SetWindowPos()通过调用node-ffi(或编写本机节点模块/插件)。拨打HWND电话。BrowserWindowgetNativeWindowHandle()我不知道如何在 macOS 或 Ubuntu 上本地执行此操作。