| /* |
| * Window properties |
| * |
| * Copyright 1995, 1996, 2001 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 <stdarg.h> |
| #include <string.h> |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "wownt32.h" |
| #include "wine/unicode.h" |
| #include "wine/winuser16.h" |
| #include "wine/server.h" |
| |
| /* size of buffer needed to store an atom string */ |
| #define ATOM_BUFFER_SIZE 256 |
| |
| |
| /*********************************************************************** |
| * get_properties |
| * |
| * Retrieve the list of properties of a given window. |
| * Returned buffer must be freed by caller. |
| */ |
| static property_data_t *get_properties( HWND hwnd, int *count ) |
| { |
| property_data_t *data; |
| int total = 32; |
| |
| while (total) |
| { |
| int res = 0; |
| if (!(data = HeapAlloc( GetProcessHeap(), 0, total * sizeof(*data) ))) break; |
| *count = 0; |
| SERVER_START_REQ( get_window_properties ) |
| { |
| req->window = wine_server_user_handle( hwnd ); |
| wine_server_set_reply( req, data, total * sizeof(*data) ); |
| if (!wine_server_call( req )) res = reply->total; |
| } |
| SERVER_END_REQ; |
| if (res && res <= total) |
| { |
| *count = res; |
| return data; |
| } |
| HeapFree( GetProcessHeap(), 0, data ); |
| total = res; /* restart with larger buffer */ |
| } |
| return NULL; |
| } |
| |
| |
| /*********************************************************************** |
| * EnumPropsA_relay |
| * |
| * relay to call the EnumProps callback function from EnumPropsEx |
| */ |
| static BOOL CALLBACK EnumPropsA_relay( HWND hwnd, LPSTR str, HANDLE handle, ULONG_PTR lparam ) |
| { |
| PROPENUMPROCA func = (PROPENUMPROCA)lparam; |
| return func( hwnd, str, handle ); |
| } |
| |
| |
| /*********************************************************************** |
| * EnumPropsW_relay |
| * |
| * relay to call the EnumProps callback function from EnumPropsEx |
| */ |
| static BOOL CALLBACK EnumPropsW_relay( HWND hwnd, LPWSTR str, HANDLE handle, ULONG_PTR lparam ) |
| { |
| PROPENUMPROCW func = (PROPENUMPROCW)lparam; |
| return func( hwnd, str, handle ); |
| } |
| |
| |
| /*********************************************************************** |
| * EnumPropsA (USER32.@) |
| */ |
| INT WINAPI EnumPropsA( HWND hwnd, PROPENUMPROCA func ) |
| { |
| return EnumPropsExA( hwnd, EnumPropsA_relay, (LPARAM)func ); |
| } |
| |
| |
| /*********************************************************************** |
| * EnumPropsW (USER32.@) |
| */ |
| INT WINAPI EnumPropsW( HWND hwnd, PROPENUMPROCW func ) |
| { |
| return EnumPropsExW( hwnd, EnumPropsW_relay, (LPARAM)func ); |
| } |
| |
| |
| /*********************************************************************** |
| * GetPropA (USER32.@) |
| */ |
| HANDLE WINAPI GetPropA( HWND hwnd, LPCSTR str ) |
| { |
| WCHAR buffer[ATOM_BUFFER_SIZE]; |
| |
| if (!HIWORD(str)) return GetPropW( hwnd, (LPCWSTR)str ); |
| if (!MultiByteToWideChar( CP_ACP, 0, str, -1, buffer, ATOM_BUFFER_SIZE )) return 0; |
| return GetPropW( hwnd, buffer ); |
| } |
| |
| |
| /*********************************************************************** |
| * GetPropW (USER32.@) |
| */ |
| HANDLE WINAPI GetPropW( HWND hwnd, LPCWSTR str ) |
| { |
| ULONG_PTR ret = 0; |
| |
| SERVER_START_REQ( get_window_property ) |
| { |
| req->window = wine_server_user_handle( hwnd ); |
| if (!HIWORD(str)) req->atom = LOWORD(str); |
| else wine_server_add_data( req, str, strlenW(str) * sizeof(WCHAR) ); |
| if (!wine_server_call_err( req )) ret = reply->data; |
| } |
| SERVER_END_REQ; |
| return (HANDLE)ret; |
| } |
| |
| |
| /*********************************************************************** |
| * SetPropA (USER32.@) |
| */ |
| BOOL WINAPI SetPropA( HWND hwnd, LPCSTR str, HANDLE handle ) |
| { |
| WCHAR buffer[ATOM_BUFFER_SIZE]; |
| |
| if (!HIWORD(str)) return SetPropW( hwnd, (LPCWSTR)str, handle ); |
| if (!MultiByteToWideChar( CP_ACP, 0, str, -1, buffer, ATOM_BUFFER_SIZE )) return FALSE; |
| return SetPropW( hwnd, buffer, handle ); |
| } |
| |
| |
| /*********************************************************************** |
| * SetPropW (USER32.@) |
| */ |
| BOOL WINAPI SetPropW( HWND hwnd, LPCWSTR str, HANDLE handle ) |
| { |
| BOOL ret; |
| |
| SERVER_START_REQ( set_window_property ) |
| { |
| req->window = wine_server_user_handle( hwnd ); |
| req->data = (ULONG_PTR)handle; |
| if (!HIWORD(str)) req->atom = LOWORD(str); |
| else wine_server_add_data( req, str, strlenW(str) * sizeof(WCHAR) ); |
| ret = !wine_server_call_err( req ); |
| } |
| SERVER_END_REQ; |
| return ret; |
| } |
| |
| |
| /*********************************************************************** |
| * RemovePropA (USER32.@) |
| */ |
| HANDLE WINAPI RemovePropA( HWND hwnd, LPCSTR str ) |
| { |
| WCHAR buffer[ATOM_BUFFER_SIZE]; |
| |
| if (!HIWORD(str)) return RemovePropW( hwnd, (LPCWSTR)str ); |
| if (!MultiByteToWideChar( CP_ACP, 0, str, -1, buffer, ATOM_BUFFER_SIZE )) return 0; |
| return RemovePropW( hwnd, buffer ); |
| } |
| |
| |
| /*********************************************************************** |
| * RemovePropW (USER32.@) |
| */ |
| HANDLE WINAPI RemovePropW( HWND hwnd, LPCWSTR str ) |
| { |
| ULONG_PTR ret = 0; |
| |
| SERVER_START_REQ( remove_window_property ) |
| { |
| req->window = wine_server_user_handle( hwnd ); |
| if (!HIWORD(str)) req->atom = LOWORD(str); |
| else wine_server_add_data( req, str, strlenW(str) * sizeof(WCHAR) ); |
| if (!wine_server_call_err( req )) ret = reply->data; |
| } |
| SERVER_END_REQ; |
| |
| return (HANDLE)ret; |
| } |
| |
| |
| /*********************************************************************** |
| * EnumPropsExA (USER32.@) |
| */ |
| INT WINAPI EnumPropsExA(HWND hwnd, PROPENUMPROCEXA func, LPARAM lParam) |
| { |
| int ret = -1, i, count; |
| property_data_t *list = get_properties( hwnd, &count ); |
| |
| if (list) |
| { |
| for (i = 0; i < count; i++) |
| { |
| char string[ATOM_BUFFER_SIZE]; |
| if (!GlobalGetAtomNameA( list[i].atom, string, ATOM_BUFFER_SIZE )) continue; |
| if (!(ret = func( hwnd, string, (HANDLE)(ULONG_PTR)list[i].data, lParam ))) break; |
| } |
| HeapFree( GetProcessHeap(), 0, list ); |
| } |
| return ret; |
| } |
| |
| |
| /*********************************************************************** |
| * EnumPropsExW (USER32.@) |
| */ |
| INT WINAPI EnumPropsExW(HWND hwnd, PROPENUMPROCEXW func, LPARAM lParam) |
| { |
| int ret = -1, i, count; |
| property_data_t *list = get_properties( hwnd, &count ); |
| |
| if (list) |
| { |
| for (i = 0; i < count; i++) |
| { |
| WCHAR string[ATOM_BUFFER_SIZE]; |
| if (!GlobalGetAtomNameW( list[i].atom, string, ATOM_BUFFER_SIZE )) continue; |
| if (!(ret = func( hwnd, string, (HANDLE)(ULONG_PTR)list[i].data, lParam ))) break; |
| } |
| HeapFree( GetProcessHeap(), 0, list ); |
| } |
| return ret; |
| } |
| |
| |
| /*********************************************************************** |
| * EnumProps (USER.27) |
| */ |
| INT16 WINAPI EnumProps16( HWND16 hwnd, PROPENUMPROC16 func ) |
| { |
| int ret = -1, i, count; |
| property_data_t *list = get_properties( HWND_32(hwnd), &count ); |
| |
| if (list) |
| { |
| char string[ATOM_BUFFER_SIZE]; |
| SEGPTR segptr = MapLS( string ); |
| WORD args[4]; |
| DWORD result; |
| |
| for (i = 0; i < count; i++) |
| { |
| if (list[i].string) /* it was a string originally */ |
| { |
| if (!GlobalGetAtomNameA( list[i].atom, string, ATOM_BUFFER_SIZE )) continue; |
| args[3] = hwnd; |
| args[2] = SELECTOROF(segptr); |
| args[1] = OFFSETOF(segptr); |
| args[0] = LOWORD(list[i].data); |
| } |
| else |
| { |
| args[3] = hwnd; |
| args[2] = 0; |
| args[1] = list[i].atom; |
| args[0] = LOWORD(list[i].data); |
| } |
| WOWCallback16Ex( (DWORD)func, WCB16_PASCAL, sizeof(args), args, &result ); |
| if (!(ret = LOWORD(result))) break; |
| } |
| UnMapLS( segptr ); |
| HeapFree( GetProcessHeap(), 0, list ); |
| } |
| return ret; |
| } |