簡體   English   中英

如何修改沒有窗口句柄的窗口類?

[英]How do I modify a window class without a window handle?

我目前正在嘗試使用JNI將一些WIN32API函數轉換為Java。 這些功能之一是RegisterClassEx 通常,我認為您會為注冊的每個窗口類指定一個不同的回調函數,但是由於我將回調也轉換為Java,因此這是行不通的。

因此,當前的計划是將一個jobject (定義為_jobject* )添加到窗口類,並在回調中使用它。 問題是:您只能使用HWND更改附加到窗口類的數據。 就我所知,MSDN文檔未指定僅可以使用窗口類ATOM或名稱來修改窗口類的函數。

因此,我的問題是:是否有一種方法可以更改窗口類(使用類似SetClassLongPtr的東西),而不必使用有效的HWND?

Java方面 (我最終將添加一個執行我實際需要完成的公共功能):

public class c_winjni implements i_jnisystem {
    public interface i_wnd_proc {
        public int wnd_proc(long il_hwnd, int im_message, long im_wparam, long im_lparam);
    }

    private class c_wndclassex {
        public int im_style = 0;
        public i_wnd_proc ds_wnd_proc = null;
        public int im_cls_extra = 0;
        public int im_wnd_extra = 0;
        public long il_instance = 0L;
        public long il_icon = 0L;
        public long il_small_icon = 0L;
        public long il_cursor = 0L;
        public long il_background = 0L;
        public String str_menu_name = null;
        public String str_class_name = null;
    }

    private static native short registerClassEx(c_wndclassex ds_wcx, int[] imr_error);
}

C ++方面

LRESULT CALLBACK default_window_callback_proc(HWND ds_hwnd, UINT im_message, WPARAM im_w_param, LPARAM im_l_param) {
    return DefWindowProc(ds_hwnd, im_message, im_w_param, im_l_param);
}

/*
 * Class:     c_winjni
 * Method:    registerClassEx
 * Signature: (Lc_winjni/c_wndclassex;[I)S
 */
JNIEXPORT_EX jshort JNICALL Java_c_1winjni_registerClassEx
    (JNIEnv *ads_env, jclass /*jds_class*/, jobject jds_wcx, jintArray jimr_error)
JNI_CPPEXCEPTION_TRAP_BEGIN {
    c_jnienv jds_env(ads_env);

    jint *aim_error = NULL;
    if (jimr_error && jds_env.get_array_length(jimr_error) > 0) {
        aim_error = jds_env.get_array_elements(jimr_error, NULL);
    }

    WNDCLASSEX ds_wcx;
    ds_wcx.cbSize = sizeof(WNDCLASSEX);
    ds_wcx.style = jds_env.get_int_field(jds_wcx, "im_style");

    // Imagine I'm checking whether field ds_wnd_proc in object jds_wcx is null.
    // If it is, use the default callback (as shown below).
    // If it isn't, set ds_wcx.lpfnWndProc to some other callback that reads
    // custom class data and calls a Java function of the object attached to the window class.
    ds_wcx.lpfnWndProc = default_window_callback_proc;

    ds_wcx.cbClsExtra = jds_env.get_int_field(jds_wcx, "im_cls_extra") + sizeof(LONG_PTR);
    ds_wcx.cbWndExtra = jds_env.get_int_field(jds_wcx, "im_wnd_extra");
    ds_wcx.hInstance = (HINSTANCE) jds_env.get_long_field(jds_wcx, "il_instance");
    ds_wcx.hIcon = (HICON) jds_env.get_long_field(jds_wcx, "il_icon");
    ds_wcx.hIconSm = (HICON) jds_env.get_long_field(jds_wcx, "il_small_icon");
    ds_wcx.hCursor = (HCURSOR) jds_env.get_long_field(jds_wcx, "il_cursor");
    ds_wcx.hbrBackground = (HBRUSH) jds_env.get_long_field(jds_wcx, "il_background");
    ct_jstring<TCHAR, 256> str_menu_name(ads_env, (jstring) jds_env.get_string_field(jds_wcx, "str_menu_name"));
    ds_wcx.lpszMenuName = str_menu_name.get_data();
    ct_jstring<TCHAR, 256> str_class_name(ads_env, (jstring) jds_env.get_string_field(jds_wcx, "str_class_name"));
    ds_wcx.lpszClassName = str_class_name.get_data();

    jshort result = RegisterClassEx(&ds_wcx);
    if (result == NULL && aim_error) {
        *aim_error = GetLastError();
    }

    // commit changes and invalidate pointer
    if (aim_error) {
        jds_env.release_array_elements(jimr_error, aim_error, 0);
    }
    return result;
} JNI_CPPEXCEPTION_TRAP_END2(ads_env, 0)

您問題的簡單答案是只能使用SetClassLongPtr修改額外的類內存。 為此,您需要一個有效的窗口句柄。

有趣的問題... WNDCLASSEX是一個類,而HWND是一個窗口的句柄。 您可能將第一個視為OO類,將第二個視為對象(實例)的“指針”或“引用”。 然而,要修改該類,您似乎必須遍歷該類的一個實例……很奇怪:)我可以想象這是最常見的情況(具有HWND)(順便說一句,為什么您沒有HWND? )

一個想法:使用ATOM創建一個窗口(隱藏)並將返回的HWND用作SetClassLongPtr的“引用”是否可以SetClassLongPtr

為了獲得注冊類ATOM的有效HWND,可以使用FindWindow函數

通過為lpWindowName參數傳遞NULL ,您將獲得與該類匹配的第一個窗口的句柄。 當然,如果不存在窗口實例,您將失敗。 但是在這種情況下,最好假設相應的窗口類尚未注冊。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM