/* * Window procedure callbacks * * Copyright 1995 Martin von Loewis * Copyright 1996 Alexandre Julliard * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "config.h" #include "wine/port.h" #include <assert.h> #include <stdarg.h> #include <string.h> #include "windef.h" #include "winbase.h" #include "controls.h" #include "win.h" #include "user_private.h" #include "wine/unicode.h" #include "wine/debug.h" WINE_DECLARE_DEBUG_CHANNEL(msg); WINE_DECLARE_DEBUG_CHANNEL(relay); WINE_DEFAULT_DEBUG_CHANNEL(win); typedef struct tagWINDOWPROC { WNDPROC procA; /* ASCII window proc */ WNDPROC procW; /* Unicode window proc */ } WINDOWPROC; #define MAX_WINPROCS 4096 #define MAX_WINPROC_RECURSION 64 #define WINPROC_PROC16 ((WINDOWPROC *)1) /* placeholder for 16-bit window procs */ static LRESULT WINAPI ButtonWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI ButtonWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI ComboWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI ComboWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); LRESULT WINAPI EditWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI EditWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI ListBoxWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI ListBoxWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI MDIClientWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI MDIClientWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI ScrollBarWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI ScrollBarWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI StaticWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static LRESULT WINAPI StaticWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); static WINDOWPROC winproc_array[MAX_WINPROCS] = { { ButtonWndProcA, ButtonWndProcW }, /* WINPROC_BUTTON */ { ComboWndProcA, ComboWndProcW }, /* WINPROC_COMBO */ { DefWindowProcA, DefWindowProcW }, /* WINPROC_DEFWND */ { DefDlgProcA, DefDlgProcW }, /* WINPROC_DIALOG */ { EditWndProcA, EditWndProcW }, /* WINPROC_EDIT */ { ListBoxWndProcA, ListBoxWndProcW }, /* WINPROC_LISTBOX */ { MDIClientWndProcA, MDIClientWndProcW }, /* WINPROC_MDICLIENT */ { ScrollBarWndProcA, ScrollBarWndProcW }, /* WINPROC_SCROLLBAR */ { StaticWndProcA, StaticWndProcW }, /* WINPROC_STATIC */ { NULL, DesktopWndProc }, /* WINPROC_DESKTOP */ { NULL, IconTitleWndProc }, /* WINPROC_ICONTITLE */ { NULL, PopupMenuWndProc }, /* WINPROC_MENU */ { NULL, MessageWndProc }, /* WINPROC_MESSAGE */ }; static UINT winproc_used = NB_BUILTIN_WINPROCS; static CRITICAL_SECTION winproc_cs; static CRITICAL_SECTION_DEBUG critsect_debug = { 0, 0, &winproc_cs, { &critsect_debug.ProcessLocksList, &critsect_debug.ProcessLocksList }, 0, 0, { (DWORD_PTR)(__FILE__ ": winproc_cs") } }; static CRITICAL_SECTION winproc_cs = { &critsect_debug, -1, 0, 0, 0, 0 }; static inline void *get_buffer( void *static_buffer, size_t size, size_t need ) { if (size >= need) return static_buffer; return HeapAlloc( GetProcessHeap(), 0, need ); } static inline void free_buffer( void *static_buffer, void *buffer ) { if (buffer != static_buffer) HeapFree( GetProcessHeap(), 0, buffer ); } /* find an existing winproc for a given function and type */ /* FIXME: probably should do something more clever than a linear search */ static inline WINDOWPROC *find_winproc( WNDPROC func, BOOL unicode ) { unsigned int i; for (i = 0; i < NB_BUILTIN_AW_WINPROCS; i++) { /* match either proc, some apps confuse A and W */ if (winproc_array[i].procA != func && winproc_array[i].procW != func) continue; return &winproc_array[i]; } for (i = NB_BUILTIN_AW_WINPROCS; i < winproc_used; i++) { if (!unicode && winproc_array[i].procA != func) continue; if (unicode && winproc_array[i].procW != func) continue; return &winproc_array[i]; } return NULL; } /* return the window proc for a given handle, or NULL for an invalid handle, * or WINPROC_PROC16 for a handle to a 16-bit proc. */ static inline WINDOWPROC *handle_to_proc( WNDPROC handle ) { UINT index = LOWORD(handle); if ((ULONG_PTR)handle >> 16 != WINPROC_HANDLE) return NULL; if (index >= MAX_WINPROCS) return WINPROC_PROC16; if (index >= winproc_used) return NULL; return &winproc_array[index]; } /* create a handle for a given window proc */ static inline WNDPROC proc_to_handle( WINDOWPROC *proc ) { return (WNDPROC)(ULONG_PTR)((proc - winproc_array) | (WINPROC_HANDLE << 16)); } /* allocate and initialize a new winproc */ static inline WINDOWPROC *alloc_winproc( WNDPROC func, BOOL unicode ) { WINDOWPROC *proc; /* check if the function is already a win proc */ if (!func) return NULL; if ((proc = handle_to_proc( func ))) return proc; EnterCriticalSection( &winproc_cs ); /* check if we already have a winproc for that function */ if (!(proc = find_winproc( func, unicode ))) { if (winproc_used < MAX_WINPROCS) { proc = &winproc_array[winproc_used++]; if (unicode) proc->procW = func; else proc->procA = func; TRACE( "allocated %p for %c %p (%d/%d used)\n", proc_to_handle(proc), unicode ? 'W' : 'A', func, winproc_used, MAX_WINPROCS ); } else FIXME( "too many winprocs, cannot allocate one for %p\n", func ); } else TRACE( "reusing %p for %p\n", proc_to_handle(proc), func ); LeaveCriticalSection( &winproc_cs ); return proc; } #ifdef __i386__ /* Some window procedures modify register they shouldn't, or are not * properly declared stdcall; so we need a small assembly wrapper to * call them. */ extern LRESULT WINPROC_wrapper( WNDPROC proc, HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ); __ASM_GLOBAL_FUNC( WINPROC_wrapper, "pushl %ebp\n\t" __ASM_CFI(".cfi_adjust_cfa_offset 4\n\t") __ASM_CFI(".cfi_rel_offset %ebp,0\n\t") "movl %esp,%ebp\n\t" __ASM_CFI(".cfi_def_cfa_register %ebp\n\t") "pushl %edi\n\t" __ASM_CFI(".cfi_rel_offset %edi,-4\n\t") "pushl %esi\n\t" __ASM_CFI(".cfi_rel_offset %esi,-8\n\t") "pushl %ebx\n\t" __ASM_CFI(".cfi_rel_offset %ebx,-12\n\t") "subl $12,%esp\n\t" "pushl 24(%ebp)\n\t" "pushl 20(%ebp)\n\t" "pushl 16(%ebp)\n\t" "pushl 12(%ebp)\n\t" "movl 8(%ebp),%eax\n\t" "call *%eax\n\t" "leal -12(%ebp),%esp\n\t" "popl %ebx\n\t" __ASM_CFI(".cfi_same_value %ebx\n\t") "popl %esi\n\t" __ASM_CFI(".cfi_same_value %esi\n\t") "popl %edi\n\t" __ASM_CFI(".cfi_same_value %edi\n\t") "leave\n\t" __ASM_CFI(".cfi_def_cfa %esp,4\n\t") __ASM_CFI(".cfi_same_value %ebp\n\t") "ret" ) #else static inline LRESULT WINPROC_wrapper( WNDPROC proc, HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return proc( hwnd, msg, wParam, lParam ); } #endif /* __i386__ */ static WPARAM map_wparam_char_WtoA( WPARAM wParam, DWORD len ) { WCHAR wch = wParam; BYTE ch[2]; RtlUnicodeToMultiByteN( (LPSTR)ch, len, &len, &wch, sizeof(wch) ); if (len == 2) return MAKEWPARAM( (ch[0] << 8) | ch[1], HIWORD(wParam) ); else return MAKEWPARAM( ch[0], HIWORD(wParam) ); } /* call a 32-bit window procedure */ static LRESULT call_window_proc( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp, LRESULT *result, void *arg ) { WNDPROC proc = arg; USER_CheckNotLock(); hwnd = WIN_GetFullHandle( hwnd ); if (TRACE_ON(relay)) DPRINTF( "%04x:Call window proc %p (hwnd=%p,msg=%s,wp=%08lx,lp=%08lx)\n", GetCurrentThreadId(), proc, hwnd, SPY_GetMsgName(msg, hwnd), wp, lp ); *result = WINPROC_wrapper( proc, hwnd, msg, wp, lp ); if (TRACE_ON(relay)) DPRINTF( "%04x:Ret window proc %p (hwnd=%p,msg=%s,wp=%08lx,lp=%08lx) retval=%08lx\n", GetCurrentThreadId(), proc, hwnd, SPY_GetMsgName(msg, hwnd), wp, lp, *result ); return *result; } /* call a 32-bit dialog procedure */ static LRESULT call_dialog_proc( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp, LRESULT *result, void *arg ) { WNDPROC proc = arg; LRESULT ret; USER_CheckNotLock(); hwnd = WIN_GetFullHandle( hwnd ); if (TRACE_ON(relay)) DPRINTF( "%04x:Call dialog proc %p (hwnd=%p,msg=%s,wp=%08lx,lp=%08lx)\n", GetCurrentThreadId(), proc, hwnd, SPY_GetMsgName(msg, hwnd), wp, lp ); ret = WINPROC_wrapper( proc, hwnd, msg, wp, lp ); *result = GetWindowLongPtrW( hwnd, DWLP_MSGRESULT ); if (TRACE_ON(relay)) DPRINTF( "%04x:Ret dialog proc %p (hwnd=%p,msg=%s,wp=%08lx,lp=%08lx) retval=%08lx result=%08lx\n", GetCurrentThreadId(), proc, hwnd, SPY_GetMsgName(msg, hwnd), wp, lp, ret, *result ); return ret; } /********************************************************************** * WINPROC_GetProc * * Get a window procedure pointer that can be passed to the Windows program. */ WNDPROC WINPROC_GetProc( WNDPROC proc, BOOL unicode ) { WINDOWPROC *ptr = handle_to_proc( proc ); if (!ptr || ptr == WINPROC_PROC16) return proc; if (unicode) { if (ptr->procW) return ptr->procW; return proc; } else { if (ptr->procA) return ptr->procA; return proc; } } /********************************************************************** * WINPROC_AllocProc * * Allocate a window procedure for a window or class. * * Note that allocated winprocs are never freed; the idea is that even if an app creates a * lot of windows, it will usually only have a limited number of window procedures, so the * array won't grow too large, and this way we avoid the need to track allocations per window. */ WNDPROC WINPROC_AllocProc( WNDPROC func, BOOL unicode ) { WINDOWPROC *proc; if (!(proc = alloc_winproc( func, unicode ))) return NULL; if (proc == WINPROC_PROC16) return func; return proc_to_handle( proc ); } /********************************************************************** * WINPROC_IsUnicode * * Return the window procedure type, or the default value if not a winproc handle. */ BOOL WINPROC_IsUnicode( WNDPROC proc, BOOL def_val ) { WINDOWPROC *ptr = handle_to_proc( proc ); if (!ptr) return def_val; if (ptr == WINPROC_PROC16) return FALSE; /* 16-bit is always A */ if (ptr->procA && ptr->procW) return def_val; /* can be both */ return (ptr->procW != NULL); } /********************************************************************** * WINPROC_TestLBForStr * * Return TRUE if the lparam is a string */ static inline BOOL WINPROC_TestLBForStr( HWND hwnd, UINT msg ) { DWORD style = GetWindowLongA( hwnd, GWL_STYLE ); if (msg <= CB_MSGMAX) return (!(style & (CBS_OWNERDRAWFIXED | CBS_OWNERDRAWVARIABLE)) || (style & CBS_HASSTRINGS)); else return (!(style & (LBS_OWNERDRAWFIXED | LBS_OWNERDRAWVARIABLE)) || (style & LBS_HASSTRINGS)); } /********************************************************************** * WINPROC_CallProcAtoW * * Call a window procedure, translating args from Ansi to Unicode. */ LRESULT WINPROC_CallProcAtoW( winproc_callback_t callback, HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, LRESULT *result, void *arg, enum wm_char_mapping mapping ) { LRESULT ret = 0; TRACE_(msg)("(hwnd=%p,msg=%s,wp=%08lx,lp=%08lx)\n", hwnd, SPY_GetMsgName(msg, hwnd), wParam, lParam); switch(msg) { case WM_NCCREATE: case WM_CREATE: { WCHAR *ptr, buffer[512]; CREATESTRUCTA *csA = (CREATESTRUCTA *)lParam; CREATESTRUCTW csW = *(CREATESTRUCTW *)csA; MDICREATESTRUCTW mdi_cs; DWORD name_lenA = 0, name_lenW = 0, class_lenA = 0, class_lenW = 0; if (!IS_INTRESOURCE(csA->lpszClass)) { class_lenA = strlen(csA->lpszClass) + 1; RtlMultiByteToUnicodeSize( &class_lenW, csA->lpszClass, class_lenA ); } if (!IS_INTRESOURCE(csA->lpszName)) { name_lenA = strlen(csA->lpszName) + 1; RtlMultiByteToUnicodeSize( &name_lenW, csA->lpszName, name_lenA ); } if (!(ptr = get_buffer( buffer, sizeof(buffer), class_lenW + name_lenW ))) break; if (class_lenW) { csW.lpszClass = ptr; RtlMultiByteToUnicodeN( ptr, class_lenW, NULL, csA->lpszClass, class_lenA ); } if (name_lenW) { csW.lpszName = ptr + class_lenW/sizeof(WCHAR); RtlMultiByteToUnicodeN( ptr + class_lenW/sizeof(WCHAR), name_lenW, NULL, csA->lpszName, name_lenA ); } if (GetWindowLongW(hwnd, GWL_EXSTYLE) & WS_EX_MDICHILD) { mdi_cs = *(MDICREATESTRUCTW *)csA->lpCreateParams; mdi_cs.szTitle = csW.lpszName; mdi_cs.szClass = csW.lpszClass; csW.lpCreateParams = &mdi_cs; } ret = callback( hwnd, msg, wParam, (LPARAM)&csW, result, arg ); free_buffer( buffer, ptr ); } break; case WM_MDICREATE: { WCHAR *ptr, buffer[512]; DWORD title_lenA = 0, title_lenW = 0, class_lenA = 0, class_lenW = 0; MDICREATESTRUCTA *csA = (MDICREATESTRUCTA *)lParam; MDICREATESTRUCTW csW; memcpy( &csW, csA, sizeof(csW) ); if (!IS_INTRESOURCE(csA->szTitle)) { title_lenA = strlen(csA->szTitle) + 1; RtlMultiByteToUnicodeSize( &title_lenW, csA->szTitle, title_lenA ); } if (!IS_INTRESOURCE(csA->szClass)) { class_lenA = strlen(csA->szClass) + 1; RtlMultiByteToUnicodeSize( &class_lenW, csA->szClass, class_lenA ); } if (!(ptr = get_buffer( buffer, sizeof(buffer), title_lenW + class_lenW ))) break; if (title_lenW) { csW.szTitle = ptr; RtlMultiByteToUnicodeN( ptr, title_lenW, NULL, csA->szTitle, title_lenA ); } if (class_lenW) { csW.szClass = ptr + title_lenW/sizeof(WCHAR); RtlMultiByteToUnicodeN( ptr + title_lenW/sizeof(WCHAR), class_lenW, NULL, csA->szClass, class_lenA ); } ret = callback( hwnd, msg, wParam, (LPARAM)&csW, result, arg ); free_buffer( buffer, ptr ); } break; case WM_GETTEXT: case WM_ASKCBFORMATNAME: { WCHAR *ptr, buffer[512]; LPSTR str = (LPSTR)lParam; DWORD len = wParam * sizeof(WCHAR); if (!(ptr = get_buffer( buffer, sizeof(buffer), len ))) break; ret = callback( hwnd, msg, wParam, (LPARAM)ptr, result, arg ); if (wParam) { len = 0; if (*result) RtlUnicodeToMultiByteN( str, wParam - 1, &len, ptr, strlenW(ptr) * sizeof(WCHAR) ); str[len] = 0; *result = len; } free_buffer( buffer, ptr ); } break; case LB_ADDSTRING: case LB_INSERTSTRING: case LB_FINDSTRING: case LB_FINDSTRINGEXACT: case LB_SELECTSTRING: case CB_ADDSTRING: case CB_INSERTSTRING: case CB_FINDSTRING: case CB_FINDSTRINGEXACT: case CB_SELECTSTRING: if (!lParam || !WINPROC_TestLBForStr( hwnd, msg )) { ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; } /* fall through */ case WM_SETTEXT: case WM_WININICHANGE: case WM_DEVMODECHANGE: case CB_DIR: case LB_DIR: case LB_ADDFILE: case EM_REPLACESEL: if (!lParam) ret = callback( hwnd, msg, wParam, lParam, result, arg ); else { WCHAR *ptr, buffer[512]; LPCSTR strA = (LPCSTR)lParam; DWORD lenW, lenA = strlen(strA) + 1; RtlMultiByteToUnicodeSize( &lenW, strA, lenA ); if ((ptr = get_buffer( buffer, sizeof(buffer), lenW ))) { RtlMultiByteToUnicodeN( ptr, lenW, NULL, strA, lenA ); ret = callback( hwnd, msg, wParam, (LPARAM)ptr, result, arg ); free_buffer( buffer, ptr ); } } break; case LB_GETTEXT: case CB_GETLBTEXT: if (lParam && WINPROC_TestLBForStr( hwnd, msg )) { WCHAR buffer[512]; /* FIXME: fixed sized buffer */ ret = callback( hwnd, msg, wParam, (LPARAM)buffer, result, arg ); if (*result >= 0) { DWORD len; RtlUnicodeToMultiByteN( (LPSTR)lParam, ~0u, &len, buffer, (strlenW(buffer) + 1) * sizeof(WCHAR) ); *result = len - 1; } } else ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; case EM_GETLINE: { WCHAR *ptr, buffer[512]; WORD len = *(WORD *)lParam; if (!(ptr = get_buffer( buffer, sizeof(buffer), len * sizeof(WCHAR) ))) break; *((WORD *)ptr) = len; /* store the length */ ret = callback( hwnd, msg, wParam, (LPARAM)ptr, result, arg ); if (*result) { DWORD reslen; RtlUnicodeToMultiByteN( (LPSTR)lParam, len, &reslen, ptr, *result * sizeof(WCHAR) ); if (reslen < len) ((LPSTR)lParam)[reslen] = 0; *result = reslen; } free_buffer( buffer, ptr ); } break; case WM_GETDLGCODE: if (lParam) { MSG newmsg = *(MSG *)lParam; if (map_wparam_AtoW( newmsg.message, &newmsg.wParam, WMCHAR_MAP_NOMAPPING )) ret = callback( hwnd, msg, wParam, (LPARAM)&newmsg, result, arg ); } else ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; case WM_CHARTOITEM: case WM_MENUCHAR: case WM_CHAR: case WM_DEADCHAR: case WM_SYSCHAR: case WM_SYSDEADCHAR: case EM_SETPASSWORDCHAR: case WM_IME_CHAR: if (map_wparam_AtoW( msg, &wParam, mapping )) ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; case WM_GETTEXTLENGTH: case CB_GETLBTEXTLEN: case LB_GETTEXTLEN: ret = callback( hwnd, msg, wParam, lParam, result, arg ); if (*result >= 0) { WCHAR *ptr, buffer[512]; LRESULT tmp; DWORD len = *result + 1; /* Determine respective GETTEXT message */ UINT msgGetText = (msg == WM_GETTEXTLENGTH) ? WM_GETTEXT : ((msg == CB_GETLBTEXTLEN) ? CB_GETLBTEXT : LB_GETTEXT); /* wParam differs between the messages */ WPARAM wp = (msg == WM_GETTEXTLENGTH) ? len : wParam; if (!(ptr = get_buffer( buffer, sizeof(buffer), len * sizeof(WCHAR) ))) break; if (callback == call_window_proc) /* FIXME: hack */ callback( hwnd, msgGetText, wp, (LPARAM)ptr, &tmp, arg ); else tmp = SendMessageW( hwnd, msgGetText, wp, (LPARAM)ptr ); RtlUnicodeToMultiByteSize( &len, ptr, tmp * sizeof(WCHAR) ); *result = len; free_buffer( buffer, ptr ); } break; case WM_PAINTCLIPBOARD: case WM_SIZECLIPBOARD: FIXME_(msg)( "message %s (0x%x) needs translation, please report\n", SPY_GetMsgName(msg, hwnd), msg ); break; default: ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; } return ret; } /********************************************************************** * WINPROC_CallProcWtoA * * Call a window procedure, translating args from Unicode to Ansi. */ static LRESULT WINPROC_CallProcWtoA( winproc_callback_t callback, HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, LRESULT *result, void *arg ) { LRESULT ret = 0; TRACE_(msg)("(hwnd=%p,msg=%s,wp=%08lx,lp=%08lx)\n", hwnd, SPY_GetMsgName(msg, hwnd), wParam, lParam); switch(msg) { case WM_NCCREATE: case WM_CREATE: { char buffer[1024], *cls; CREATESTRUCTW *csW = (CREATESTRUCTW *)lParam; CREATESTRUCTA csA = *(CREATESTRUCTA *)csW; MDICREATESTRUCTA mdi_cs; DWORD name_lenA = 0, name_lenW = 0, class_lenA = 0, class_lenW = 0; if (!IS_INTRESOURCE(csW->lpszClass)) { class_lenW = (strlenW(csW->lpszClass) + 1) * sizeof(WCHAR); RtlUnicodeToMultiByteSize(&class_lenA, csW->lpszClass, class_lenW); } if (!IS_INTRESOURCE(csW->lpszName)) { name_lenW = (strlenW(csW->lpszName) + 1) * sizeof(WCHAR); RtlUnicodeToMultiByteSize(&name_lenA, csW->lpszName, name_lenW); } if (!(cls = get_buffer( buffer, sizeof(buffer), class_lenA + name_lenA ))) break; if (class_lenA) { RtlUnicodeToMultiByteN(cls, class_lenA, NULL, csW->lpszClass, class_lenW); csA.lpszClass = cls; } if (name_lenA) { char *name = cls + class_lenA; RtlUnicodeToMultiByteN(name, name_lenA, NULL, csW->lpszName, name_lenW); csA.lpszName = name; } if (GetWindowLongW(hwnd, GWL_EXSTYLE) & WS_EX_MDICHILD) { mdi_cs = *(MDICREATESTRUCTA *)csW->lpCreateParams; mdi_cs.szTitle = csA.lpszName; mdi_cs.szClass = csA.lpszClass; csA.lpCreateParams = &mdi_cs; } ret = callback( hwnd, msg, wParam, (LPARAM)&csA, result, arg ); free_buffer( buffer, cls ); } break; case WM_GETTEXT: case WM_ASKCBFORMATNAME: { char *ptr, buffer[512]; DWORD len = wParam * 2; if (!(ptr = get_buffer( buffer, sizeof(buffer), len ))) break; ret = callback( hwnd, msg, wParam, (LPARAM)ptr, result, arg ); if (len) { if (*result) { RtlMultiByteToUnicodeN( (LPWSTR)lParam, wParam*sizeof(WCHAR), &len, ptr, strlen(ptr)+1 ); *result = len/sizeof(WCHAR) - 1; /* do not count terminating null */ } ((LPWSTR)lParam)[*result] = 0; } free_buffer( buffer, ptr ); } break; case LB_ADDSTRING: case LB_INSERTSTRING: case LB_FINDSTRING: case LB_FINDSTRINGEXACT: case LB_SELECTSTRING: case CB_ADDSTRING: case CB_INSERTSTRING: case CB_FINDSTRING: case CB_FINDSTRINGEXACT: case CB_SELECTSTRING: if (!lParam || !WINPROC_TestLBForStr( hwnd, msg )) { ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; } /* fall through */ case WM_SETTEXT: case WM_WININICHANGE: case WM_DEVMODECHANGE: case CB_DIR: case LB_DIR: case LB_ADDFILE: case EM_REPLACESEL: if (!lParam) ret = callback( hwnd, msg, wParam, lParam, result, arg ); else { char *ptr, buffer[512]; LPCWSTR strW = (LPCWSTR)lParam; DWORD lenA, lenW = (strlenW(strW) + 1) * sizeof(WCHAR); RtlUnicodeToMultiByteSize( &lenA, strW, lenW ); if ((ptr = get_buffer( buffer, sizeof(buffer), lenA ))) { RtlUnicodeToMultiByteN( ptr, lenA, NULL, strW, lenW ); ret = callback( hwnd, msg, wParam, (LPARAM)ptr, result, arg ); free_buffer( buffer, ptr ); } } break; case WM_MDICREATE: { char *ptr, buffer[1024]; DWORD title_lenA = 0, title_lenW = 0, class_lenA = 0, class_lenW = 0; MDICREATESTRUCTW *csW = (MDICREATESTRUCTW *)lParam; MDICREATESTRUCTA csA; memcpy( &csA, csW, sizeof(csA) ); if (!IS_INTRESOURCE(csW->szTitle)) { title_lenW = (strlenW(csW->szTitle) + 1) * sizeof(WCHAR); RtlUnicodeToMultiByteSize( &title_lenA, csW->szTitle, title_lenW ); } if (!IS_INTRESOURCE(csW->szClass)) { class_lenW = (strlenW(csW->szClass) + 1) * sizeof(WCHAR); RtlUnicodeToMultiByteSize( &class_lenA, csW->szClass, class_lenW ); } if (!(ptr = get_buffer( buffer, sizeof(buffer), title_lenA + class_lenA ))) break; if (title_lenA) { RtlUnicodeToMultiByteN( ptr, title_lenA, NULL, csW->szTitle, title_lenW ); csA.szTitle = ptr; } if (class_lenA) { RtlUnicodeToMultiByteN( ptr + title_lenA, class_lenA, NULL, csW->szClass, class_lenW ); csA.szClass = ptr + title_lenA; } ret = callback( hwnd, msg, wParam, (LPARAM)&csA, result, arg ); free_buffer( buffer, ptr ); } break; case LB_GETTEXT: case CB_GETLBTEXT: if (lParam && WINPROC_TestLBForStr( hwnd, msg )) { char buffer[512]; /* FIXME: fixed sized buffer */ ret = callback( hwnd, msg, wParam, (LPARAM)buffer, result, arg ); if (*result >= 0) { DWORD len; RtlMultiByteToUnicodeN( (LPWSTR)lParam, ~0u, &len, buffer, strlen(buffer) + 1 ); *result = len / sizeof(WCHAR) - 1; } } else ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; case EM_GETLINE: { char *ptr, buffer[512]; WORD len = *(WORD *)lParam; if (!(ptr = get_buffer( buffer, sizeof(buffer), len * 2 ))) break; *((WORD *)ptr) = len * 2; /* store the length */ ret = callback( hwnd, msg, wParam, (LPARAM)ptr, result, arg ); if (*result) { DWORD reslen; RtlMultiByteToUnicodeN( (LPWSTR)lParam, len*sizeof(WCHAR), &reslen, ptr, *result ); *result = reslen / sizeof(WCHAR); if (*result < len) ((LPWSTR)lParam)[*result] = 0; } free_buffer( buffer, ptr ); } break; case WM_GETDLGCODE: if (lParam) { MSG newmsg = *(MSG *)lParam; switch(newmsg.message) { case WM_CHAR: case WM_DEADCHAR: case WM_SYSCHAR: case WM_SYSDEADCHAR: newmsg.wParam = map_wparam_char_WtoA( newmsg.wParam, 1 ); break; case WM_IME_CHAR: newmsg.wParam = map_wparam_char_WtoA( newmsg.wParam, 2 ); break; } ret = callback( hwnd, msg, wParam, (LPARAM)&newmsg, result, arg ); } else ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; case WM_CHAR: { WCHAR wch = wParam; char ch[2]; DWORD len; RtlUnicodeToMultiByteN( ch, 2, &len, &wch, sizeof(wch) ); ret = callback( hwnd, msg, (BYTE)ch[0], lParam, result, arg ); if (len == 2) ret = callback( hwnd, msg, (BYTE)ch[1], lParam, result, arg ); } break; case WM_CHARTOITEM: case WM_MENUCHAR: case WM_DEADCHAR: case WM_SYSCHAR: case WM_SYSDEADCHAR: case EM_SETPASSWORDCHAR: ret = callback( hwnd, msg, map_wparam_char_WtoA(wParam,1), lParam, result, arg ); break; case WM_IME_CHAR: ret = callback( hwnd, msg, map_wparam_char_WtoA(wParam,2), lParam, result, arg ); break; case WM_PAINTCLIPBOARD: case WM_SIZECLIPBOARD: FIXME_(msg)( "message %s (%04x) needs translation, please report\n", SPY_GetMsgName(msg, hwnd), msg ); break; default: ret = callback( hwnd, msg, wParam, lParam, result, arg ); break; } return ret; } /********************************************************************** * WINPROC_call_window * * Call the window procedure of the specified window. */ BOOL WINPROC_call_window( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, LRESULT *result, BOOL unicode, enum wm_char_mapping mapping ) { struct user_thread_info *thread_info = get_user_thread_info(); WND *wndPtr; WNDPROC func; WINDOWPROC *proc; if (!(wndPtr = WIN_GetPtr( hwnd ))) return FALSE; if (wndPtr == WND_OTHER_PROCESS || wndPtr == WND_DESKTOP) return FALSE; if (wndPtr->tid != GetCurrentThreadId()) { WIN_ReleasePtr( wndPtr ); return FALSE; } func = wndPtr->winproc; proc = handle_to_proc( wndPtr->winproc ); WIN_ReleasePtr( wndPtr ); if (!proc) return TRUE; if (thread_info->recursion_count > MAX_WINPROC_RECURSION) return FALSE; thread_info->recursion_count++; if (unicode) { if (proc == WINPROC_PROC16) WINPROC_CallProcWtoA( wow_handlers.call_window_proc, hwnd, msg, wParam, lParam, result, func ); else if (proc->procW) call_window_proc( hwnd, msg, wParam, lParam, result, proc->procW ); else WINPROC_CallProcWtoA( call_window_proc, hwnd, msg, wParam, lParam, result, proc->procA ); } else { if (proc == WINPROC_PROC16) wow_handlers.call_window_proc( hwnd, msg, wParam, lParam, result, func ); else if (proc->procA) call_window_proc( hwnd, msg, wParam, lParam, result, proc->procA ); else WINPROC_CallProcAtoW( call_window_proc, hwnd, msg, wParam, lParam, result, proc->procW, mapping ); } thread_info->recursion_count--; return TRUE; } /********************************************************************** * CallWindowProcA (USER32.@) * * The CallWindowProc() function invokes the windows procedure _func_, * with _hwnd_ as the target window, the message specified by _msg_, and * the message parameters _wParam_ and _lParam_. * * Some kinds of argument conversion may be done, I'm not sure what. * * CallWindowProc() may be used for windows subclassing. Use * SetWindowLong() to set a new windows procedure for windows of the * subclass, and handle subclassed messages in the new windows * procedure. The new windows procedure may then use CallWindowProc() * with _func_ set to the parent class's windows procedure to dispatch * the message to the superclass. * * RETURNS * * The return value is message dependent. * * CONFORMANCE * * ECMA-234, Win32 */ LRESULT WINAPI CallWindowProcA( WNDPROC func, /* [in] window procedure */ HWND hwnd, /* [in] target window */ UINT msg, /* [in] message */ WPARAM wParam, /* [in] message dependent parameter */ LPARAM lParam /* [in] message dependent parameter */ ) { WINDOWPROC *proc; LRESULT result; if (!func) return 0; if (!(proc = handle_to_proc( func ))) call_window_proc( hwnd, msg, wParam, lParam, &result, func ); else if (proc == WINPROC_PROC16) wow_handlers.call_window_proc( hwnd, msg, wParam, lParam, &result, func ); else if (proc->procA) call_window_proc( hwnd, msg, wParam, lParam, &result, proc->procA ); else WINPROC_CallProcAtoW( call_window_proc, hwnd, msg, wParam, lParam, &result, proc->procW, WMCHAR_MAP_CALLWINDOWPROC ); return result; } /********************************************************************** * CallWindowProcW (USER32.@) * * See CallWindowProcA. */ LRESULT WINAPI CallWindowProcW( WNDPROC func, HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { WINDOWPROC *proc; LRESULT result; if (!func) return 0; if (!(proc = handle_to_proc( func ))) call_window_proc( hwnd, msg, wParam, lParam, &result, func ); else if (proc == WINPROC_PROC16) WINPROC_CallProcWtoA( wow_handlers.call_window_proc, hwnd, msg, wParam, lParam, &result, func ); else if (proc->procW) call_window_proc( hwnd, msg, wParam, lParam, &result, proc->procW ); else WINPROC_CallProcWtoA( call_window_proc, hwnd, msg, wParam, lParam, &result, proc->procA ); return result; } /********************************************************************** * WINPROC_CallDlgProcA */ INT_PTR WINPROC_CallDlgProcA( DLGPROC func, HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { WINDOWPROC *proc; LRESULT result; INT_PTR ret; if (!func) return 0; if (!(proc = handle_to_proc( func ))) ret = call_dialog_proc( hwnd, msg, wParam, lParam, &result, func ); else if (proc == WINPROC_PROC16) { ret = wow_handlers.call_dialog_proc( hwnd, msg, wParam, lParam, &result, func ); SetWindowLongPtrW( hwnd, DWLP_MSGRESULT, result ); } else if (proc->procW) { ret = WINPROC_CallProcAtoW( call_dialog_proc, hwnd, msg, wParam, lParam, &result, proc->procW, WMCHAR_MAP_CALLWINDOWPROC ); SetWindowLongPtrW( hwnd, DWLP_MSGRESULT, result ); } else ret = call_dialog_proc( hwnd, msg, wParam, lParam, &result, proc->procA ); return ret; } /********************************************************************** * WINPROC_CallDlgProcW */ INT_PTR WINPROC_CallDlgProcW( DLGPROC func, HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { WINDOWPROC *proc; LRESULT result; INT_PTR ret; if (!func) return 0; if (!(proc = handle_to_proc( func ))) ret = call_dialog_proc( hwnd, msg, wParam, lParam, &result, func ); else if (proc == WINPROC_PROC16) { ret = WINPROC_CallProcWtoA( wow_handlers.call_dialog_proc, hwnd, msg, wParam, lParam, &result, func ); SetWindowLongPtrW( hwnd, DWLP_MSGRESULT, result ); } else if (proc->procA) { ret = WINPROC_CallProcWtoA( call_dialog_proc, hwnd, msg, wParam, lParam, &result, proc->procA ); SetWindowLongPtrW( hwnd, DWLP_MSGRESULT, result ); } else ret = call_dialog_proc( hwnd, msg, wParam, lParam, &result, proc->procW ); return ret; } /*********************************************************************** * Window procedures for builtin classes */ static LRESULT WINAPI ButtonWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.button_proc( hwnd, msg, wParam, lParam, FALSE ); } static LRESULT WINAPI ButtonWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.button_proc( hwnd, msg, wParam, lParam, TRUE ); } static LRESULT WINAPI ComboWndProcA( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam ) { return wow_handlers.combo_proc( hwnd, message, wParam, lParam, FALSE ); } static LRESULT WINAPI ComboWndProcW( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam ) { return wow_handlers.combo_proc( hwnd, message, wParam, lParam, TRUE ); } LRESULT WINAPI EditWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.edit_proc( hwnd, msg, wParam, lParam, FALSE ); } static LRESULT WINAPI EditWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.edit_proc( hwnd, msg, wParam, lParam, TRUE ); } static LRESULT WINAPI ListBoxWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.listbox_proc( hwnd, msg, wParam, lParam, FALSE ); } static LRESULT WINAPI ListBoxWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.listbox_proc( hwnd, msg, wParam, lParam, TRUE ); } static LRESULT WINAPI MDIClientWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.mdiclient_proc( hwnd, msg, wParam, lParam, FALSE ); } static LRESULT WINAPI MDIClientWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.mdiclient_proc( hwnd, msg, wParam, lParam, TRUE ); } static LRESULT WINAPI ScrollBarWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.scrollbar_proc( hwnd, msg, wParam, lParam, FALSE ); } static LRESULT WINAPI ScrollBarWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.scrollbar_proc( hwnd, msg, wParam, lParam, TRUE ); } static LRESULT WINAPI StaticWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.static_proc( hwnd, msg, wParam, lParam, FALSE ); } static LRESULT WINAPI StaticWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) { return wow_handlers.static_proc( hwnd, msg, wParam, lParam, TRUE ); } static DWORD wait_message( DWORD count, CONST HANDLE *handles, DWORD timeout, DWORD mask, DWORD flags ) { DWORD ret = USER_Driver->pMsgWaitForMultipleObjectsEx( count, handles, timeout, mask, flags ); if (ret == WAIT_TIMEOUT && !count && !timeout) NtYieldExecution(); return ret; } /********************************************************************** * UserRegisterWowHandlers (USER32.@) * * NOTE: no attempt has been made to be compatible here, * the Windows function is most likely completely different. */ void WINAPI UserRegisterWowHandlers( const struct wow_handlers16 *new, struct wow_handlers32 *orig ) { orig->button_proc = ButtonWndProc_common; orig->combo_proc = ComboWndProc_common; orig->edit_proc = EditWndProc_common; orig->listbox_proc = ListBoxWndProc_common; orig->mdiclient_proc = MDIClientWndProc_common; orig->scrollbar_proc = ScrollBarWndProc_common; orig->static_proc = StaticWndProc_common; orig->wait_message = wait_message; orig->create_window = WIN_CreateWindowEx; orig->get_win_handle = WIN_GetFullHandle; orig->alloc_winproc = WINPROC_AllocProc; orig->get_dialog_info = DIALOG_get_info; orig->dialog_box_loop = DIALOG_DoDialogBox; orig->get_icon_param = get_icon_param; orig->set_icon_param = set_icon_param; wow_handlers = *new; } struct wow_handlers16 wow_handlers = { ButtonWndProc_common, ComboWndProc_common, EditWndProc_common, ListBoxWndProc_common, MDIClientWndProc_common, ScrollBarWndProc_common, StaticWndProc_common, wait_message, WIN_CreateWindowEx, NULL, /* call_window_proc */ NULL, /* call_dialog_proc */ NULL, /* free_icon_param */ };