2012-11-20 17:21:08 +01:00
|
|
|
/*
|
|
|
|
* Unit tests for the pager control
|
|
|
|
*
|
|
|
|
* Copyright 2012 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 <windows.h>
|
|
|
|
#include <commctrl.h>
|
|
|
|
|
|
|
|
#include "wine/test.h"
|
|
|
|
#include "msg.h"
|
|
|
|
|
|
|
|
#define NUM_MSG_SEQUENCES 1
|
|
|
|
#define PAGER_SEQ_INDEX 0
|
|
|
|
|
2017-04-07 16:02:03 +02:00
|
|
|
static HWND parent_wnd, child1_wnd, child2_wnd;
|
2018-08-24 16:04:35 +02:00
|
|
|
static INT notify_format;
|
|
|
|
static BOOL notify_query_received;
|
2018-09-17 17:26:31 +02:00
|
|
|
static WCHAR test_w[] = {'t', 'e', 's', 't', 0};
|
|
|
|
static CHAR test_a[] = {'t', 'e', 's', 't', 0};
|
2018-09-24 09:12:02 +02:00
|
|
|
/* Double zero so that it's safe to cast it to WCHAR * */
|
|
|
|
static CHAR te_a[] = {'t', 'e', 0, 0};
|
2018-09-17 17:26:31 +02:00
|
|
|
static WCHAR empty_w[] = {0};
|
|
|
|
static CHAR empty_a[] = {0};
|
|
|
|
static CHAR large_a[] = "You should have received a copy of the GNU Lesser General Public License along with this ...";
|
2018-09-17 17:26:48 +02:00
|
|
|
static WCHAR large_w[] =
|
|
|
|
{
|
|
|
|
'Y', 'o', 'u', ' ', 's', 'h', 'o', 'u', 'l', 'd', ' ', 'h', 'a', 'v', 'e', ' ', 'r', 'e', 'c', 'e', 'i', 'v', 'e',
|
|
|
|
'd', ' ', 'a', ' ', 'c', 'o', 'p', 'y', ' ', 'o', 'f', ' ', 't', 'h', 'e', ' ', 'G', 'N', 'U', ' ', 'L', 'e', 's',
|
|
|
|
's', 'e', 'r', ' ', 'G', 'e', 'n', 'e', 'r', 'a', 'l', ' ', 'P', 'u', 'b', 'l', 'i', 'c', ' ', 'L', 'i', 'c', 'e',
|
|
|
|
'n', 's', 'e', ' ', 'a', 'l', 'o', 'n', 'g', ' ', 'w', 'i', 't', 'h', ' ', 't', 'h', 'i', 's', ' ', '.', '.', '.', 0
|
|
|
|
};
|
2018-09-17 17:26:31 +02:00
|
|
|
static WCHAR large_truncated_65_w[65] =
|
|
|
|
{
|
|
|
|
'Y', 'o', 'u', ' ', 's', 'h', 'o', 'u', 'l', 'd', ' ', 'h', 'a', 'v', 'e', ' ', 'r', 'e', 'c', 'e', 'i', 'v',
|
|
|
|
'e', 'd', ' ', 'a', ' ', 'c', 'o', 'p', 'y', ' ', 'o', 'f', ' ', 't', 'h', 'e', ' ', 'G', 'N', 'U', ' ', 'L',
|
|
|
|
'e', 's', 's', 'e', 'r', ' ', 'G', 'e', 'n', 'e', 'r', 'a', 'l', ' ', 'P', 'u', 'b', 'l', 'i', 'c', 0
|
|
|
|
};
|
2018-09-17 17:26:48 +02:00
|
|
|
static WCHAR large_truncated_80_w[80] =
|
|
|
|
{
|
|
|
|
'Y', 'o', 'u', ' ', 's', 'h', 'o', 'u', 'l', 'd', ' ', 'h', 'a', 'v', 'e', ' ', 'r', 'e', 'c', 'e',
|
|
|
|
'i', 'v', 'e', 'd', ' ', 'a', ' ', 'c', 'o', 'p', 'y', ' ', 'o', 'f', ' ', 't', 'h', 'e', ' ', 'G',
|
|
|
|
'N', 'U', ' ', 'L', 'e', 's', 's', 'e', 'r', ' ', 'G', 'e', 'n', 'e', 'r', 'a', 'l', ' ', 'P', 'u',
|
|
|
|
'b', 'l', 'i', 'c', ' ', 'L', 'i', 'c', 'e', 'n', 's', 'e', ' ', 'a', 'l', 'o', 'n', 'g', ' ', 'w'
|
|
|
|
};
|
2018-09-17 17:26:31 +02:00
|
|
|
static WCHAR buffer[64];
|
|
|
|
|
|
|
|
/* Text field conversion test behavior flags. */
|
|
|
|
enum test_conversion_flags
|
|
|
|
{
|
|
|
|
CONVERT_SEND = 0x01,
|
|
|
|
DONT_CONVERT_SEND = 0x02,
|
|
|
|
CONVERT_RECEIVE = 0x04,
|
|
|
|
DONT_CONVERT_RECEIVE = 0x08,
|
|
|
|
SEND_EMPTY_IF_NULL = 0x10,
|
2018-09-17 17:27:00 +02:00
|
|
|
DONT_SEND_EMPTY_IF_NULL = 0x20,
|
|
|
|
SET_NULL_IF_NO_MASK = 0x40,
|
|
|
|
ZERO_SEND = 0x80
|
2018-09-17 17:26:31 +02:00
|
|
|
};
|
|
|
|
|
2018-09-28 05:06:45 +02:00
|
|
|
enum handler_ids
|
|
|
|
{
|
|
|
|
TVITEM_NEW_HANDLER,
|
|
|
|
TVITEM_OLD_HANDLER
|
|
|
|
};
|
|
|
|
|
2018-09-17 17:26:31 +02:00
|
|
|
static struct notify_test_info
|
|
|
|
{
|
|
|
|
UINT unicode;
|
|
|
|
UINT ansi;
|
|
|
|
UINT_PTR id_from;
|
|
|
|
HWND hwnd_from;
|
|
|
|
/* Whether parent received notification */
|
|
|
|
BOOL received;
|
|
|
|
UINT test_id;
|
|
|
|
UINT sub_test_id;
|
2018-09-28 05:06:45 +02:00
|
|
|
UINT handler_id;
|
2018-09-17 17:26:31 +02:00
|
|
|
/* Text field conversion test behavior flag */
|
|
|
|
DWORD flags;
|
|
|
|
} notify_test_info;
|
|
|
|
|
|
|
|
struct notify_test_send
|
|
|
|
{
|
|
|
|
/* Data sent to pager */
|
|
|
|
WCHAR *send_text;
|
|
|
|
INT send_text_size;
|
|
|
|
INT send_text_max;
|
|
|
|
/* Data expected by parent of pager */
|
|
|
|
void *expect_text;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct notify_test_receive
|
|
|
|
{
|
|
|
|
/* Data sent to pager */
|
|
|
|
WCHAR *send_text;
|
|
|
|
INT send_text_size;
|
|
|
|
INT send_text_max;
|
|
|
|
/* Data for parent to write */
|
|
|
|
CHAR *write_pointer;
|
|
|
|
CHAR *write_text;
|
|
|
|
INT write_text_size;
|
|
|
|
INT write_text_max;
|
|
|
|
/* Data when message returned */
|
|
|
|
void *return_text;
|
|
|
|
INT return_text_max;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct generic_text_helper_para
|
|
|
|
{
|
|
|
|
void *ptr;
|
|
|
|
size_t size;
|
|
|
|
UINT *mask;
|
|
|
|
UINT required_mask;
|
|
|
|
WCHAR **text;
|
|
|
|
INT *text_max;
|
|
|
|
UINT code_unicode;
|
|
|
|
UINT code_ansi;
|
|
|
|
DWORD flags;
|
2018-09-28 05:06:45 +02:00
|
|
|
UINT handler_id;
|
2018-09-17 17:26:31 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static const struct notify_test_send test_convert_send_data[] =
|
|
|
|
{
|
|
|
|
{test_w, sizeof(test_w), ARRAY_SIZE(buffer), test_a}
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct notify_test_send test_dont_convert_send_data[] =
|
|
|
|
{
|
|
|
|
{test_w, sizeof(test_w), ARRAY_SIZE(buffer), test_w}
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct notify_test_receive test_convert_receive_data[] =
|
|
|
|
{
|
|
|
|
{empty_w, sizeof(empty_w), ARRAY_SIZE(buffer), NULL, test_a, sizeof(test_a), -1, test_w, ARRAY_SIZE(buffer)},
|
|
|
|
{empty_w, sizeof(empty_w), ARRAY_SIZE(buffer), test_a, NULL, 0, -1, test_w, ARRAY_SIZE(buffer)},
|
|
|
|
{NULL, sizeof(empty_w), ARRAY_SIZE(buffer), test_a, NULL, 0, -1, NULL, ARRAY_SIZE(buffer)},
|
|
|
|
{empty_w, sizeof(empty_w), ARRAY_SIZE(buffer), large_a, NULL, 0, -1, large_truncated_65_w, ARRAY_SIZE(buffer)},
|
|
|
|
{empty_w, sizeof(empty_w), ARRAY_SIZE(buffer), empty_a, 0, 0, 1, empty_w, 1},
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct notify_test_receive test_dont_convert_receive_data[] =
|
|
|
|
{
|
|
|
|
{empty_w, sizeof(empty_w), ARRAY_SIZE(buffer), NULL, test_a, sizeof(test_a), -1, test_a, ARRAY_SIZE(buffer)},
|
|
|
|
{empty_w, sizeof(empty_w), ARRAY_SIZE(buffer), test_a, NULL, 0, -1, test_a, ARRAY_SIZE(buffer)},
|
|
|
|
};
|
2017-04-07 16:02:02 +02:00
|
|
|
|
2018-09-17 17:26:48 +02:00
|
|
|
static const struct notify_test_tooltip
|
|
|
|
{
|
|
|
|
/* Data for parent to write */
|
|
|
|
CHAR *write_sztext;
|
|
|
|
INT write_sztext_size;
|
|
|
|
CHAR *write_lpsztext;
|
|
|
|
HMODULE write_hinst;
|
|
|
|
/* Data when message returned */
|
|
|
|
WCHAR *return_sztext;
|
|
|
|
INT return_sztext_size;
|
|
|
|
WCHAR *return_lpsztext;
|
|
|
|
HMODULE return_hinst;
|
|
|
|
/* Data expected by parent */
|
|
|
|
CHAR *expect_sztext;
|
|
|
|
/* Data send to parent */
|
|
|
|
WCHAR *send_sztext;
|
|
|
|
INT send_sztext_size;
|
|
|
|
WCHAR *send_lpsztext;
|
|
|
|
} test_tooltip_data[] =
|
|
|
|
{
|
|
|
|
{NULL, 0, NULL, NULL, empty_w, -1, empty_w},
|
|
|
|
{test_a, sizeof(test_a), NULL, NULL, test_w, -1, test_w},
|
|
|
|
{test_a, sizeof(test_a), test_a, NULL, test_w, -1, test_w},
|
|
|
|
{test_a, sizeof(test_a), (CHAR *)1, (HMODULE)0xdeadbeef, empty_w, -1, (WCHAR *)1, (HMODULE)0xdeadbeef},
|
|
|
|
{test_a, sizeof(test_a), test_a, (HMODULE)0xdeadbeef, test_w, -1, test_w, (HMODULE)0xdeadbeef},
|
|
|
|
{NULL, 0, test_a, NULL, test_w, -1, test_w},
|
|
|
|
{test_a, 2, test_a, NULL, test_w, -1, test_w},
|
|
|
|
{NULL, 0, NULL, NULL, test_w, -1, test_w, NULL, test_a, test_w, sizeof(test_w)},
|
|
|
|
{NULL, 0, NULL, NULL, empty_w, -1, empty_w, NULL, empty_a, NULL, 0, test_w},
|
|
|
|
{NULL, 0, large_a, NULL, large_truncated_80_w, sizeof(large_truncated_80_w), large_w}
|
|
|
|
};
|
|
|
|
|
2018-09-24 09:12:02 +02:00
|
|
|
static const struct notify_test_datetime_format
|
|
|
|
{
|
|
|
|
/* Data send to parent */
|
|
|
|
WCHAR *send_pszformat;
|
|
|
|
/* Data expected by parent */
|
|
|
|
CHAR *expect_pszformat;
|
|
|
|
/* Data for parent to write */
|
|
|
|
CHAR *write_szdisplay;
|
|
|
|
INT write_szdisplay_size;
|
|
|
|
CHAR *write_pszdisplay;
|
|
|
|
/* Data when message returned */
|
|
|
|
WCHAR *return_szdisplay;
|
|
|
|
INT return_szdisplay_size;
|
|
|
|
WCHAR *return_pszdisplay;
|
|
|
|
} test_datetime_format_data[] =
|
|
|
|
{
|
|
|
|
{test_w, test_a},
|
|
|
|
{NULL, NULL, NULL, 0, test_a, empty_w, -1, test_w},
|
|
|
|
{NULL, NULL, test_a, sizeof(test_a), NULL, test_w, -1, test_w},
|
|
|
|
{NULL, NULL, test_a, 2, test_a, (WCHAR *)te_a, -1, test_w},
|
|
|
|
{NULL, NULL, NULL, 0, large_a, NULL, 0, large_w}
|
|
|
|
};
|
|
|
|
|
2017-04-07 16:02:02 +02:00
|
|
|
#define CHILD1_ID 1
|
2017-04-07 16:02:03 +02:00
|
|
|
#define CHILD2_ID 2
|
2012-11-20 17:21:08 +01:00
|
|
|
|
2018-08-24 16:04:20 +02:00
|
|
|
static BOOL (WINAPI *pInitCommonControlsEx)(const INITCOMMONCONTROLSEX*);
|
2012-11-20 17:21:08 +01:00
|
|
|
static BOOL (WINAPI *pSetWindowSubclass)(HWND, SUBCLASSPROC, UINT_PTR, DWORD_PTR);
|
|
|
|
|
|
|
|
static struct msg_sequence *sequences[NUM_MSG_SEQUENCES];
|
|
|
|
|
|
|
|
static const struct message set_child_seq[] = {
|
|
|
|
{ PGM_SETCHILD, sent },
|
|
|
|
{ WM_WINDOWPOSCHANGING, sent },
|
|
|
|
{ WM_NCCALCSIZE, sent|wparam, TRUE },
|
|
|
|
{ WM_NOTIFY, sent|id|parent, 0, 0, PGN_CALCSIZE },
|
|
|
|
{ WM_WINDOWPOSCHANGED, sent },
|
2017-04-07 16:02:02 +02:00
|
|
|
{ WM_WINDOWPOSCHANGING, sent|id, 0, 0, CHILD1_ID },
|
2017-04-07 16:02:03 +02:00
|
|
|
{ WM_NCCALCSIZE, sent|wparam|id|optional, TRUE, 0, CHILD1_ID },
|
2017-04-07 16:02:02 +02:00
|
|
|
{ WM_CHILDACTIVATE, sent|id, 0, 0, CHILD1_ID },
|
|
|
|
{ WM_WINDOWPOSCHANGED, sent|id, 0, 0, CHILD1_ID },
|
2017-04-07 16:02:03 +02:00
|
|
|
{ WM_SIZE, sent|id|defwinproc|optional, 0, 0, CHILD1_ID },
|
|
|
|
{ 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
/* This differs from the above message list only in the child window that is
|
|
|
|
* expected to receive the child messages. No message is sent to the old child.
|
|
|
|
* Also child 2 is hidden while child 1 is visible. The pager does not make the
|
|
|
|
* hidden child visible. */
|
|
|
|
static const struct message switch_child_seq[] = {
|
|
|
|
{ PGM_SETCHILD, sent },
|
|
|
|
{ WM_WINDOWPOSCHANGING, sent },
|
|
|
|
{ WM_NCCALCSIZE, sent|wparam, TRUE },
|
|
|
|
{ WM_NOTIFY, sent|id|parent, 0, 0, PGN_CALCSIZE },
|
|
|
|
{ WM_WINDOWPOSCHANGED, sent },
|
|
|
|
{ WM_WINDOWPOSCHANGING, sent|id, 0, 0, CHILD2_ID },
|
|
|
|
{ WM_NCCALCSIZE, sent|wparam|id, TRUE, 0, CHILD2_ID },
|
|
|
|
{ WM_CHILDACTIVATE, sent|id, 0, 0, CHILD2_ID },
|
|
|
|
{ WM_WINDOWPOSCHANGED, sent|id, 0, 0, CHILD2_ID },
|
|
|
|
{ WM_SIZE, sent|id|defwinproc, 0, 0, CHILD2_ID },
|
2012-11-20 17:21:08 +01:00
|
|
|
{ 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct message set_pos_seq[] = {
|
|
|
|
{ PGM_SETPOS, sent },
|
|
|
|
{ WM_WINDOWPOSCHANGING, sent },
|
|
|
|
{ WM_NCCALCSIZE, sent|wparam, TRUE },
|
|
|
|
{ WM_NOTIFY, sent|id|parent, 0, 0, PGN_CALCSIZE },
|
|
|
|
{ WM_WINDOWPOSCHANGED, sent },
|
|
|
|
{ WM_MOVE, sent|optional },
|
2017-04-07 16:02:02 +02:00
|
|
|
/* The WM_SIZE handler sends WM_WINDOWPOSCHANGING, WM_CHILDACTIVATE
|
|
|
|
* and WM_WINDOWPOSCHANGED (which sends WM_MOVE) to the child.
|
|
|
|
* Another WM_WINDOWPOSCHANGING is sent afterwards.
|
|
|
|
*
|
|
|
|
* The 2nd WM_WINDOWPOSCHANGING is unconditional, but the comparison
|
|
|
|
* function is too simple to roll back an accepted message, so we have
|
|
|
|
* to mark the 2nd message optional. */
|
2012-11-20 17:21:08 +01:00
|
|
|
{ WM_SIZE, sent|optional },
|
2017-04-07 16:02:02 +02:00
|
|
|
{ WM_WINDOWPOSCHANGING, sent|id, 0, 0, CHILD1_ID }, /* Actually optional. */
|
|
|
|
{ WM_CHILDACTIVATE, sent|id, 0, 0, CHILD1_ID }, /* Actually optional. */
|
|
|
|
{ WM_WINDOWPOSCHANGED, sent|id|optional, TRUE, 0, CHILD1_ID},
|
|
|
|
{ WM_MOVE, sent|id|optional|defwinproc, 0, 0, CHILD1_ID },
|
|
|
|
{ WM_WINDOWPOSCHANGING, sent|id|optional, 0, 0, CHILD1_ID }, /* Actually not optional. */
|
|
|
|
{ WM_CHILDACTIVATE, sent|id|optional, 0, 0, CHILD1_ID }, /* Actually not optional. */
|
2012-11-20 17:21:08 +01:00
|
|
|
{ 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct message set_pos_empty_seq[] = {
|
|
|
|
{ PGM_SETPOS, sent },
|
|
|
|
{ 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
static LRESULT WINAPI parent_wnd_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
static LONG defwndproc_counter = 0;
|
|
|
|
LRESULT ret;
|
|
|
|
struct message msg;
|
|
|
|
|
|
|
|
/* log system messages, except for painting */
|
|
|
|
if (message < WM_USER &&
|
|
|
|
message != WM_PAINT &&
|
|
|
|
message != WM_ERASEBKGND &&
|
|
|
|
message != WM_NCPAINT &&
|
|
|
|
message != WM_NCHITTEST &&
|
|
|
|
message != WM_GETTEXT &&
|
|
|
|
message != WM_GETICON &&
|
|
|
|
message != WM_DEVICECHANGE)
|
|
|
|
{
|
|
|
|
msg.message = message;
|
|
|
|
msg.flags = sent|wparam|lparam|parent;
|
|
|
|
if (defwndproc_counter) msg.flags |= defwinproc;
|
|
|
|
msg.wParam = wParam;
|
|
|
|
msg.lParam = lParam;
|
|
|
|
if (message == WM_NOTIFY && lParam) msg.id = ((NMHDR*)lParam)->code;
|
|
|
|
add_message(sequences, PAGER_SEQ_INDEX, &msg);
|
|
|
|
}
|
|
|
|
|
2017-02-14 05:47:38 +01:00
|
|
|
if (message == WM_NOTIFY)
|
|
|
|
{
|
|
|
|
NMHDR *nmhdr = (NMHDR *)lParam;
|
|
|
|
|
|
|
|
switch (nmhdr->code)
|
|
|
|
{
|
|
|
|
case PGN_CALCSIZE:
|
|
|
|
{
|
|
|
|
NMPGCALCSIZE *nmpgcs = (NMPGCALCSIZE *)lParam;
|
|
|
|
DWORD style = GetWindowLongA(nmpgcs->hdr.hwndFrom, GWL_STYLE);
|
|
|
|
|
|
|
|
if (style & PGS_HORZ)
|
|
|
|
ok(nmpgcs->dwFlag == PGF_CALCWIDTH, "Unexpected flags %#x.\n", nmpgcs->dwFlag);
|
|
|
|
else
|
|
|
|
ok(nmpgcs->dwFlag == PGF_CALCHEIGHT, "Unexpected flags %#x.\n", nmpgcs->dwFlag);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-20 17:21:08 +01:00
|
|
|
defwndproc_counter++;
|
|
|
|
ret = DefWindowProcA(hwnd, message, wParam, lParam);
|
|
|
|
defwndproc_counter--;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL register_parent_wnd_class(void)
|
|
|
|
{
|
|
|
|
WNDCLASSA cls;
|
|
|
|
|
|
|
|
cls.style = 0;
|
|
|
|
cls.lpfnWndProc = parent_wnd_proc;
|
|
|
|
cls.cbClsExtra = 0;
|
|
|
|
cls.cbWndExtra = 0;
|
|
|
|
cls.hInstance = GetModuleHandleA(NULL);
|
|
|
|
cls.hIcon = 0;
|
2013-10-19 21:35:01 +02:00
|
|
|
cls.hCursor = LoadCursorA(0, (LPCSTR)IDC_ARROW);
|
2012-11-20 17:21:08 +01:00
|
|
|
cls.hbrBackground = GetStockObject(WHITE_BRUSH);
|
|
|
|
cls.lpszMenuName = NULL;
|
|
|
|
cls.lpszClassName = "Pager test parent class";
|
|
|
|
return RegisterClassA(&cls);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HWND create_parent_window(void)
|
|
|
|
{
|
|
|
|
if (!register_parent_wnd_class())
|
|
|
|
return NULL;
|
|
|
|
|
2013-10-19 21:35:01 +02:00
|
|
|
return CreateWindowA("Pager test parent class", "Pager test parent window",
|
2012-11-20 17:21:08 +01:00
|
|
|
WS_OVERLAPPED | WS_VISIBLE,
|
|
|
|
0, 0, 200, 200, 0, NULL, GetModuleHandleA(NULL), NULL );
|
|
|
|
}
|
|
|
|
|
|
|
|
static LRESULT WINAPI pager_subclass_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
WNDPROC oldproc = (WNDPROC)GetWindowLongPtrA(hwnd, GWLP_USERDATA);
|
2017-03-29 10:11:39 +02:00
|
|
|
struct message msg = { 0 };
|
2012-11-20 17:21:08 +01:00
|
|
|
|
|
|
|
msg.message = message;
|
|
|
|
msg.flags = sent|wparam|lparam;
|
|
|
|
msg.wParam = wParam;
|
|
|
|
msg.lParam = lParam;
|
|
|
|
add_message(sequences, PAGER_SEQ_INDEX, &msg);
|
|
|
|
return CallWindowProcA(oldproc, hwnd, message, wParam, lParam);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HWND create_pager_control( DWORD style )
|
|
|
|
{
|
|
|
|
WNDPROC oldproc;
|
|
|
|
HWND hwnd;
|
|
|
|
RECT rect;
|
|
|
|
|
|
|
|
GetClientRect( parent_wnd, &rect );
|
|
|
|
hwnd = CreateWindowA( WC_PAGESCROLLERA, "pager", WS_CHILD | WS_BORDER | WS_VISIBLE | style,
|
|
|
|
0, 0, 100, 100, parent_wnd, 0, GetModuleHandleA(0), 0 );
|
|
|
|
oldproc = (WNDPROC)SetWindowLongPtrA(hwnd, GWLP_WNDPROC, (LONG_PTR)pager_subclass_proc);
|
|
|
|
SetWindowLongPtrA(hwnd, GWLP_USERDATA, (LONG_PTR)oldproc);
|
|
|
|
return hwnd;
|
|
|
|
}
|
|
|
|
|
2017-04-07 16:02:02 +02:00
|
|
|
static LRESULT WINAPI child_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
static LONG defwndproc_counter;
|
2017-04-21 12:16:01 +02:00
|
|
|
struct message msg = { 0 };
|
2017-04-07 16:02:02 +02:00
|
|
|
LRESULT ret;
|
|
|
|
|
|
|
|
msg.message = message;
|
|
|
|
msg.flags = sent | wparam | lparam;
|
|
|
|
if (defwndproc_counter)
|
|
|
|
msg.flags |= defwinproc;
|
|
|
|
msg.wParam = wParam;
|
|
|
|
msg.lParam = lParam;
|
|
|
|
|
|
|
|
if (hwnd == child1_wnd)
|
|
|
|
msg.id = CHILD1_ID;
|
2017-04-07 16:02:03 +02:00
|
|
|
else if (hwnd == child2_wnd)
|
|
|
|
msg.id = CHILD2_ID;
|
2017-04-07 16:02:02 +02:00
|
|
|
else
|
|
|
|
msg.id = 0;
|
|
|
|
|
|
|
|
add_message(sequences, PAGER_SEQ_INDEX, &msg);
|
|
|
|
|
|
|
|
defwndproc_counter++;
|
|
|
|
ret = DefWindowProcA(hwnd, message, wParam, lParam);
|
|
|
|
defwndproc_counter--;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL register_child_wnd_class(void)
|
|
|
|
{
|
|
|
|
WNDCLASSA cls;
|
|
|
|
|
|
|
|
cls.style = 0;
|
|
|
|
cls.lpfnWndProc = child_proc;
|
|
|
|
cls.cbClsExtra = 0;
|
|
|
|
cls.cbWndExtra = 0;
|
|
|
|
cls.hInstance = GetModuleHandleA(NULL);
|
|
|
|
cls.hIcon = 0;
|
|
|
|
cls.hCursor = LoadCursorA(0, (LPCSTR)IDC_ARROW);
|
|
|
|
cls.hbrBackground = GetStockObject(WHITE_BRUSH);
|
|
|
|
cls.lpszMenuName = NULL;
|
|
|
|
cls.lpszClassName = "Pager test child class";
|
|
|
|
return RegisterClassA(&cls);
|
|
|
|
}
|
|
|
|
|
2012-11-20 17:21:08 +01:00
|
|
|
static void test_pager(void)
|
|
|
|
{
|
2017-04-07 16:02:02 +02:00
|
|
|
HWND pager;
|
2017-02-14 05:47:39 +01:00
|
|
|
RECT rect, rect2;
|
2012-11-20 17:21:08 +01:00
|
|
|
|
|
|
|
pager = create_pager_control( PGS_HORZ );
|
2018-08-24 16:04:20 +02:00
|
|
|
ok(pager != NULL, "Fail to create pager\n");
|
2017-04-07 16:02:02 +02:00
|
|
|
|
|
|
|
register_child_wnd_class();
|
|
|
|
|
|
|
|
child1_wnd = CreateWindowA( "Pager test child class", "button", WS_CHILD | WS_BORDER | WS_VISIBLE, 0, 0, 300, 300,
|
2012-11-20 17:21:08 +01:00
|
|
|
pager, 0, GetModuleHandleA(0), 0 );
|
2017-04-07 16:02:03 +02:00
|
|
|
child2_wnd = CreateWindowA("Pager test child class", "button", WS_CHILD | WS_BORDER, 0, 0, 300, 300,
|
|
|
|
pager, 0, GetModuleHandleA(0), 0);
|
2012-11-20 17:21:08 +01:00
|
|
|
|
|
|
|
flush_sequences( sequences, NUM_MSG_SEQUENCES );
|
2017-04-07 16:02:02 +02:00
|
|
|
SendMessageA( pager, PGM_SETCHILD, 0, (LPARAM)child1_wnd );
|
2017-04-21 12:16:02 +02:00
|
|
|
ok_sequence(sequences, PAGER_SEQ_INDEX, set_child_seq, "set child", FALSE);
|
2012-11-20 17:21:08 +01:00
|
|
|
GetWindowRect( pager, &rect );
|
|
|
|
ok( rect.right - rect.left == 100 && rect.bottom - rect.top == 100,
|
|
|
|
"pager resized %dx%d\n", rect.right - rect.left, rect.bottom - rect.top );
|
|
|
|
|
2017-04-07 16:02:03 +02:00
|
|
|
flush_sequences(sequences, NUM_MSG_SEQUENCES);
|
|
|
|
SendMessageA(pager, PGM_SETCHILD, 0, (LPARAM)child2_wnd);
|
2017-04-21 12:16:02 +02:00
|
|
|
ok_sequence(sequences, PAGER_SEQ_INDEX, switch_child_seq, "switch to invisible child", FALSE);
|
2017-04-07 16:02:03 +02:00
|
|
|
GetWindowRect(pager, &rect);
|
|
|
|
ok(rect.right - rect.left == 100 && rect.bottom - rect.top == 100,
|
|
|
|
"pager resized %dx%d\n", rect.right - rect.left, rect.bottom - rect.top);
|
2017-04-07 16:02:04 +02:00
|
|
|
ok(!IsWindowVisible(child2_wnd), "Child window 2 is visible\n");
|
2017-04-07 16:02:03 +02:00
|
|
|
|
|
|
|
flush_sequences(sequences, NUM_MSG_SEQUENCES);
|
|
|
|
SendMessageA(pager, PGM_SETCHILD, 0, (LPARAM)child1_wnd);
|
2017-04-21 12:16:02 +02:00
|
|
|
ok_sequence(sequences, PAGER_SEQ_INDEX, set_child_seq, "switch to visible child", FALSE);
|
2017-04-07 16:02:03 +02:00
|
|
|
GetWindowRect(pager, &rect);
|
|
|
|
ok(rect.right - rect.left == 100 && rect.bottom - rect.top == 100,
|
|
|
|
"pager resized %dx%d\n", rect.right - rect.left, rect.bottom - rect.top);
|
|
|
|
|
2012-11-20 17:21:08 +01:00
|
|
|
flush_sequences( sequences, NUM_MSG_SEQUENCES );
|
|
|
|
SendMessageA( pager, PGM_SETPOS, 0, 10 );
|
|
|
|
ok_sequence(sequences, PAGER_SEQ_INDEX, set_pos_seq, "set pos", TRUE);
|
|
|
|
GetWindowRect( pager, &rect );
|
|
|
|
ok( rect.right - rect.left == 100 && rect.bottom - rect.top == 100,
|
|
|
|
"pager resized %dx%d\n", rect.right - rect.left, rect.bottom - rect.top );
|
|
|
|
|
|
|
|
flush_sequences( sequences, NUM_MSG_SEQUENCES );
|
|
|
|
SendMessageA( pager, PGM_SETPOS, 0, 10 );
|
|
|
|
ok_sequence(sequences, PAGER_SEQ_INDEX, set_pos_empty_seq, "set pos empty", TRUE);
|
|
|
|
|
|
|
|
flush_sequences( sequences, NUM_MSG_SEQUENCES );
|
|
|
|
SendMessageA( pager, PGM_SETPOS, 0, 9 );
|
|
|
|
ok_sequence(sequences, PAGER_SEQ_INDEX, set_pos_seq, "set pos", TRUE);
|
|
|
|
|
|
|
|
DestroyWindow( pager );
|
2017-02-14 05:47:39 +01:00
|
|
|
|
|
|
|
/* Test if resizing works */
|
|
|
|
pager = create_pager_control( CCS_NORESIZE );
|
|
|
|
ok(pager != NULL, "failed to create pager control\n");
|
|
|
|
|
|
|
|
GetWindowRect( pager, &rect );
|
|
|
|
MoveWindow( pager, 0, 0, 200, 100, TRUE );
|
|
|
|
GetWindowRect( pager, &rect2 );
|
|
|
|
ok(rect2.right - rect2.left > rect.right - rect.left, "expected pager window to resize, %s\n",
|
|
|
|
wine_dbgstr_rect( &rect2 ));
|
|
|
|
|
|
|
|
DestroyWindow( pager );
|
|
|
|
|
|
|
|
pager = create_pager_control( CCS_NORESIZE | PGS_HORZ );
|
|
|
|
ok(pager != NULL, "failed to create pager control\n");
|
|
|
|
|
|
|
|
GetWindowRect( pager, &rect );
|
|
|
|
MoveWindow( pager, 0, 0, 100, 200, TRUE );
|
|
|
|
GetWindowRect( pager, &rect2 );
|
|
|
|
ok(rect2.bottom - rect2.top > rect.bottom - rect.top, "expected pager window to resize, %s\n",
|
|
|
|
wine_dbgstr_rect( &rect2 ));
|
|
|
|
|
|
|
|
DestroyWindow( pager );
|
2012-11-20 17:21:08 +01:00
|
|
|
}
|
|
|
|
|
2018-08-24 16:04:35 +02:00
|
|
|
static LRESULT WINAPI test_notifyformat_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
switch (message)
|
|
|
|
{
|
|
|
|
case WM_NOTIFYFORMAT:
|
|
|
|
if (lParam == NF_QUERY)
|
|
|
|
{
|
|
|
|
notify_query_received = TRUE;
|
|
|
|
return notify_format;
|
|
|
|
}
|
|
|
|
else if (lParam == NF_REQUERY)
|
|
|
|
return SendMessageA(GetParent(hwnd), WM_NOTIFYFORMAT, (WPARAM)hwnd, NF_QUERY);
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return notify_format == NFR_UNICODE ? DefWindowProcW(hwnd, message, wParam, lParam)
|
|
|
|
: DefWindowProcA(hwnd, message, wParam, lParam);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL register_notifyformat_class(void)
|
|
|
|
{
|
|
|
|
static const WCHAR class_w[] = {'P', 'a', 'g', 'e', 'r', ' ', 'n', 'o', 't', 'i', 'f', 'y', 'f',
|
|
|
|
'o', 'r', 'm', 'a', 't', ' ', 'c', 'l', 'a', 's', 's', 0};
|
|
|
|
WNDCLASSW cls = {0};
|
|
|
|
|
|
|
|
cls.lpfnWndProc = test_notifyformat_proc;
|
|
|
|
cls.hInstance = GetModuleHandleW(NULL);
|
|
|
|
cls.lpszClassName = class_w;
|
|
|
|
return RegisterClassW(&cls);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_wm_notifyformat(void)
|
|
|
|
{
|
|
|
|
static const WCHAR class_w[] = {'P', 'a', 'g', 'e', 'r', ' ', 'n', 'o', 't', 'i', 'f', 'y', 'f',
|
|
|
|
'o', 'r', 'm', 'a', 't', ' ', 'c', 'l', 'a', 's', 's', 0};
|
|
|
|
static const WCHAR parent_w[] = {'p', 'a', 'r', 'e', 'n', 't', 0};
|
|
|
|
static const WCHAR pager_w[] = {'p', 'a', 'g', 'e', 'r', 0};
|
|
|
|
static const WCHAR child_w[] = {'c', 'h', 'i', 'l', 'd', 0};
|
|
|
|
static const INT formats[] = {NFR_UNICODE, NFR_ANSI};
|
|
|
|
HWND parent, pager, child;
|
|
|
|
LRESULT ret;
|
|
|
|
INT i;
|
|
|
|
|
|
|
|
ok(register_notifyformat_class(), "Register test class failed, error 0x%08x\n", GetLastError());
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(formats); i++)
|
|
|
|
{
|
|
|
|
notify_format = formats[i];
|
|
|
|
parent = CreateWindowW(class_w, parent_w, WS_OVERLAPPED, 0, 0, 100, 100, 0, 0, GetModuleHandleW(0), 0);
|
|
|
|
ok(parent != NULL, "CreateWindow failed\n");
|
|
|
|
pager = CreateWindowW(WC_PAGESCROLLERW, pager_w, WS_CHILD, 0, 0, 100, 100, parent, 0, GetModuleHandleW(0), 0);
|
|
|
|
ok(pager != NULL, "CreateWindow failed\n");
|
|
|
|
child = CreateWindowW(class_w, child_w, WS_CHILD, 0, 0, 100, 100, pager, 0, GetModuleHandleW(0), 0);
|
|
|
|
ok(child != NULL, "CreateWindow failed\n");
|
|
|
|
SendMessageW(pager, PGM_SETCHILD, 0, (LPARAM)child);
|
|
|
|
|
|
|
|
/* Test parent */
|
|
|
|
notify_query_received = FALSE;
|
|
|
|
ret = SendMessageW(pager, WM_NOTIFYFORMAT, (WPARAM)parent, NF_REQUERY);
|
2018-08-24 16:04:49 +02:00
|
|
|
ok(ret == notify_format, "Expect %d, got %ld\n", notify_format, ret);
|
|
|
|
ok(notify_query_received, "Didn't receive notify\n");
|
2018-08-24 16:04:35 +02:00
|
|
|
|
|
|
|
/* Send NF_QUERY directly to parent */
|
|
|
|
notify_query_received = FALSE;
|
|
|
|
ret = SendMessageW(parent, WM_NOTIFYFORMAT, (WPARAM)pager, NF_QUERY);
|
|
|
|
ok(ret == notify_format, "Expect %d, got %ld\n", notify_format, ret);
|
|
|
|
ok(notify_query_received, "Didn't receive notify\n");
|
|
|
|
|
|
|
|
/* Pager send notifications to its parent regardless of wParam */
|
|
|
|
notify_query_received = FALSE;
|
|
|
|
ret = SendMessageW(pager, WM_NOTIFYFORMAT, (WPARAM)parent_wnd, NF_REQUERY);
|
2018-08-24 16:04:49 +02:00
|
|
|
ok(ret == notify_format, "Expect %d, got %ld\n", notify_format, ret);
|
|
|
|
ok(notify_query_received, "Didn't receive notify\n");
|
2018-08-24 16:04:35 +02:00
|
|
|
|
|
|
|
/* Pager always wants Unicode notifications from children */
|
|
|
|
ret = SendMessageW(child, WM_NOTIFYFORMAT, (WPARAM)pager, NF_REQUERY);
|
|
|
|
ok(ret == NFR_UNICODE, "Expect %d, got %ld\n", NFR_UNICODE, ret);
|
|
|
|
ret = SendMessageW(pager, WM_NOTIFYFORMAT, (WPARAM)child, NF_QUERY);
|
|
|
|
ok(ret == NFR_UNICODE, "Expect %d, got %ld\n", NFR_UNICODE, ret);
|
|
|
|
|
|
|
|
DestroyWindow(parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
UnregisterClassW(class_w, GetModuleHandleW(NULL));
|
|
|
|
}
|
|
|
|
|
2018-09-17 17:26:31 +02:00
|
|
|
static void notify_generic_text_handler(CHAR **text, INT *text_max)
|
|
|
|
{
|
|
|
|
const struct notify_test_send *send_data;
|
|
|
|
const struct notify_test_receive *receive_data;
|
|
|
|
|
|
|
|
switch (notify_test_info.test_id)
|
|
|
|
{
|
|
|
|
case CONVERT_SEND:
|
|
|
|
case DONT_CONVERT_SEND:
|
|
|
|
{
|
|
|
|
send_data = (notify_test_info.test_id == CONVERT_SEND ? test_convert_send_data : test_dont_convert_send_data)
|
|
|
|
+ notify_test_info.sub_test_id;
|
2018-09-17 17:27:00 +02:00
|
|
|
if (notify_test_info.flags & ZERO_SEND)
|
|
|
|
ok(!lstrcmpA(*text, empty_a), "Code 0x%08x test 0x%08x sub test %d expect empty text, got %s\n",
|
|
|
|
notify_test_info.unicode, notify_test_info.test_id, notify_test_info.sub_test_id, *text);
|
|
|
|
else if (notify_test_info.flags & CONVERT_SEND)
|
2018-09-17 17:26:31 +02:00
|
|
|
ok(!lstrcmpA(send_data->expect_text, *text), "Code 0x%08x test 0x%08x sub test %d expect %s, got %s\n",
|
|
|
|
notify_test_info.unicode, notify_test_info.test_id, notify_test_info.sub_test_id,
|
|
|
|
(CHAR *)send_data->expect_text, *text);
|
|
|
|
else
|
|
|
|
ok(!lstrcmpW((WCHAR *)send_data->expect_text, (WCHAR *)*text),
|
|
|
|
"Code 0x%08x test 0x%08x sub test %d expect %s, got %s\n", notify_test_info.unicode,
|
|
|
|
notify_test_info.test_id, notify_test_info.sub_test_id, wine_dbgstr_w((WCHAR *)send_data->expect_text),
|
|
|
|
wine_dbgstr_w((WCHAR *)*text));
|
|
|
|
if (text_max)
|
|
|
|
ok(*text_max == send_data->send_text_max, "Code 0x%08x test 0x%08x sub test %d expect %d, got %d\n",
|
|
|
|
notify_test_info.unicode, notify_test_info.test_id, notify_test_info.sub_test_id,
|
|
|
|
send_data->send_text_max, *text_max);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case CONVERT_RECEIVE:
|
|
|
|
case DONT_CONVERT_RECEIVE:
|
|
|
|
{
|
|
|
|
receive_data = (notify_test_info.test_id == CONVERT_RECEIVE ? test_convert_receive_data : test_dont_convert_receive_data)
|
|
|
|
+ notify_test_info.sub_test_id;
|
|
|
|
if (text_max)
|
|
|
|
ok(*text_max == receive_data->send_text_max, "Code 0x%08x test 0x%08x sub test %d expect %d, got %d\n",
|
|
|
|
notify_test_info.unicode, notify_test_info.test_id, notify_test_info.sub_test_id,
|
|
|
|
receive_data->send_text_max, *text_max);
|
|
|
|
|
|
|
|
if (receive_data->write_text)
|
|
|
|
memcpy(*text, receive_data->write_text, receive_data->write_text_size);
|
|
|
|
else
|
|
|
|
*text = receive_data->write_pointer;
|
|
|
|
if (text_max && receive_data->write_text_max != -1) *text_max = receive_data->write_text_max;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SEND_EMPTY_IF_NULL:
|
|
|
|
ok(!lstrcmpA(*text, empty_a), "Code 0x%08x test 0x%08x sub test %d expect empty text, got %s\n",
|
|
|
|
notify_test_info.unicode, notify_test_info.test_id, notify_test_info.sub_test_id, *text);
|
|
|
|
break;
|
|
|
|
case DONT_SEND_EMPTY_IF_NULL:
|
|
|
|
ok(!*text, "Code 0x%08x test 0x%08x sub test %d expect null text\n", notify_test_info.unicode,
|
|
|
|
notify_test_info.test_id, notify_test_info.sub_test_id);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-17 17:26:48 +02:00
|
|
|
static void notify_tooltip_handler(NMTTDISPINFOA *nm)
|
|
|
|
{
|
|
|
|
const struct notify_test_tooltip *data = test_tooltip_data + notify_test_info.sub_test_id;
|
|
|
|
ok(nm->lpszText == nm->szText, "Sub test %d expect %p, got %p\n", notify_test_info.sub_test_id, nm->szText,
|
|
|
|
nm->lpszText);
|
|
|
|
if (data->expect_sztext)
|
|
|
|
ok(!lstrcmpA(data->expect_sztext, nm->szText), "Sub test %d expect %s, got %s\n", notify_test_info.sub_test_id,
|
|
|
|
data->expect_sztext, nm->szText);
|
|
|
|
if (data->write_sztext) memcpy(nm->szText, data->write_sztext, data->write_sztext_size);
|
|
|
|
if (data->write_lpsztext) nm->lpszText = data->write_lpsztext;
|
|
|
|
if (data->write_hinst) nm->hinst = data->write_hinst;
|
|
|
|
}
|
|
|
|
|
2018-09-24 09:12:02 +02:00
|
|
|
static void notify_datetime_handler(NMDATETIMEFORMATA *nm)
|
|
|
|
{
|
|
|
|
const struct notify_test_datetime_format *data = test_datetime_format_data + notify_test_info.sub_test_id;
|
|
|
|
if (data->expect_pszformat)
|
|
|
|
ok(!lstrcmpA(data->expect_pszformat, nm->pszFormat), "Sub test %d expect %s, got %s\n",
|
|
|
|
notify_test_info.sub_test_id, data->expect_pszformat, nm->pszFormat);
|
|
|
|
ok(nm->pszDisplay == nm->szDisplay, "Test %d expect %p, got %p\n", notify_test_info.sub_test_id, nm->szDisplay,
|
|
|
|
nm->pszDisplay);
|
|
|
|
if (data->write_szdisplay) memcpy(nm->szDisplay, data->write_szdisplay, data->write_szdisplay_size);
|
|
|
|
if (data->write_pszdisplay) nm->pszDisplay = data->write_pszdisplay;
|
|
|
|
}
|
|
|
|
|
2018-09-17 17:26:31 +02:00
|
|
|
static LRESULT WINAPI test_notify_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
static const WCHAR test[] = {'t', 'e', 's', 't', 0};
|
|
|
|
switch (message)
|
|
|
|
{
|
|
|
|
case WM_NOTIFY:
|
|
|
|
{
|
|
|
|
NMHDR *hdr = (NMHDR *)lParam;
|
|
|
|
|
|
|
|
/* Not notifications we want to test */
|
|
|
|
if (!notify_test_info.unicode) break;
|
|
|
|
ok(!notify_test_info.received, "Extra notification received\n");
|
|
|
|
|
|
|
|
ok(wParam == notify_test_info.id_from, "Expect %ld, got %ld\n", notify_test_info.id_from, wParam);
|
|
|
|
ok(hdr->code == notify_test_info.ansi, "Expect 0x%08x, got 0x%08x\n", notify_test_info.ansi, hdr->code);
|
|
|
|
ok(hdr->idFrom == notify_test_info.id_from, "Expect %ld, got %ld\n", notify_test_info.id_from, wParam);
|
|
|
|
ok(hdr->hwndFrom == notify_test_info.hwnd_from, "Expect %p, got %p\n", notify_test_info.hwnd_from, hdr->hwndFrom);
|
|
|
|
|
|
|
|
if (hdr->code != notify_test_info.ansi)
|
|
|
|
{
|
|
|
|
skip("Notification code mismatch, skipping lParam check\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
switch (hdr->code)
|
|
|
|
{
|
2018-09-17 17:27:00 +02:00
|
|
|
/* ComboBoxEx */
|
|
|
|
case CBEN_INSERTITEM:
|
|
|
|
case CBEN_DELETEITEM:
|
|
|
|
{
|
|
|
|
NMCOMBOBOXEXW *nmcbe = (NMCOMBOBOXEXW *)hdr;
|
|
|
|
notify_generic_text_handler((CHAR **)&nmcbe->ceItem.pszText, NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case CBEN_DRAGBEGINA:
|
|
|
|
{
|
|
|
|
NMCBEDRAGBEGINA *nmcbedb = (NMCBEDRAGBEGINA *)hdr;
|
|
|
|
ok(!lstrcmpA(nmcbedb->szText, test_a), "Expect %s, got %s\n", nmcbedb->szText, test_a);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case CBEN_ENDEDITA:
|
|
|
|
{
|
|
|
|
NMCBEENDEDITA *nmcbeed = (NMCBEENDEDITA *)hdr;
|
|
|
|
ok(!lstrcmpA(nmcbeed->szText, test_a), "Expect %s, got %s\n", nmcbeed->szText, test_a);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case CBEN_GETDISPINFOA:
|
|
|
|
{
|
|
|
|
NMCOMBOBOXEXA *nmcbe = (NMCOMBOBOXEXA *)hdr;
|
|
|
|
notify_generic_text_handler(&nmcbe->ceItem.pszText, &nmcbe->ceItem.cchTextMax);
|
|
|
|
break;
|
|
|
|
}
|
2018-09-24 09:12:02 +02:00
|
|
|
/* Date and Time Picker */
|
|
|
|
case DTN_FORMATA:
|
|
|
|
{
|
|
|
|
notify_datetime_handler((NMDATETIMEFORMATA *)hdr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DTN_FORMATQUERYA:
|
|
|
|
{
|
|
|
|
NMDATETIMEFORMATQUERYA *nmdtfq = (NMDATETIMEFORMATQUERYA *)hdr;
|
|
|
|
notify_generic_text_handler((CHAR **)&nmdtfq->pszFormat, NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DTN_WMKEYDOWNA:
|
|
|
|
{
|
|
|
|
NMDATETIMEWMKEYDOWNA *nmdtkd = (NMDATETIMEWMKEYDOWNA *)hdr;
|
|
|
|
notify_generic_text_handler((CHAR **)&nmdtkd->pszFormat, NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DTN_USERSTRINGA:
|
|
|
|
{
|
|
|
|
NMDATETIMESTRINGA *nmdts = (NMDATETIMESTRINGA *)hdr;
|
|
|
|
notify_generic_text_handler((CHAR **)&nmdts->pszUserString, NULL);
|
|
|
|
break;
|
|
|
|
}
|
2018-09-28 05:06:28 +02:00
|
|
|
/* List View */
|
|
|
|
case LVN_BEGINLABELEDITA:
|
|
|
|
case LVN_ENDLABELEDITA:
|
|
|
|
case LVN_GETDISPINFOA:
|
|
|
|
case LVN_SETDISPINFOA:
|
|
|
|
{
|
|
|
|
NMLVDISPINFOA *nmlvdi = (NMLVDISPINFOA *)hdr;
|
|
|
|
notify_generic_text_handler(&nmlvdi->item.pszText, &nmlvdi->item.cchTextMax);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LVN_GETINFOTIPA:
|
|
|
|
{
|
|
|
|
NMLVGETINFOTIPA *nmlvgit = (NMLVGETINFOTIPA *)hdr;
|
|
|
|
notify_generic_text_handler(&nmlvgit->pszText, &nmlvgit->cchTextMax);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LVN_INCREMENTALSEARCHA:
|
|
|
|
case LVN_ODFINDITEMA:
|
|
|
|
{
|
|
|
|
NMLVFINDITEMA *nmlvfi = (NMLVFINDITEMA *)hdr;
|
|
|
|
notify_generic_text_handler((CHAR **)&nmlvfi->lvfi.psz, NULL);
|
|
|
|
break;
|
|
|
|
}
|
2018-09-17 17:26:31 +02:00
|
|
|
/* Toolbar */
|
|
|
|
case TBN_SAVE:
|
|
|
|
{
|
|
|
|
NMTBSAVE *nmtbs = (NMTBSAVE *)hdr;
|
|
|
|
notify_generic_text_handler((CHAR **)&nmtbs->tbButton.iString, NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TBN_RESTORE:
|
|
|
|
{
|
|
|
|
NMTBRESTORE *nmtbr = (NMTBRESTORE *)hdr;
|
|
|
|
notify_generic_text_handler((CHAR **)&nmtbr->tbButton.iString, NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TBN_GETBUTTONINFOA:
|
|
|
|
{
|
|
|
|
NMTOOLBARA *nmtb = (NMTOOLBARA *)hdr;
|
|
|
|
notify_generic_text_handler(&nmtb->pszText, &nmtb->cchText);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TBN_GETDISPINFOW:
|
|
|
|
{
|
|
|
|
NMTBDISPINFOW *nmtbdi = (NMTBDISPINFOW *)hdr;
|
|
|
|
notify_generic_text_handler((CHAR **)&nmtbdi->pszText, &nmtbdi->cchText);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TBN_GETINFOTIPA:
|
|
|
|
{
|
|
|
|
NMTBGETINFOTIPA *nmtbgit = (NMTBGETINFOTIPA *)hdr;
|
|
|
|
notify_generic_text_handler(&nmtbgit->pszText, &nmtbgit->cchTextMax);
|
|
|
|
break;
|
|
|
|
}
|
2018-09-17 17:26:48 +02:00
|
|
|
/* Tooltip */
|
|
|
|
case TTN_GETDISPINFOA:
|
|
|
|
{
|
|
|
|
notify_tooltip_handler((NMTTDISPINFOA *)hdr);
|
|
|
|
break;
|
|
|
|
}
|
2018-09-28 05:06:45 +02:00
|
|
|
/* Tree View */
|
|
|
|
case TVN_BEGINLABELEDITA:
|
|
|
|
case TVN_ENDLABELEDITA:
|
|
|
|
case TVN_GETDISPINFOA:
|
|
|
|
case TVN_SETDISPINFOA:
|
|
|
|
{
|
|
|
|
NMTVDISPINFOA *nmtvdi = (NMTVDISPINFOA *)hdr;
|
|
|
|
notify_generic_text_handler(&nmtvdi->item.pszText, &nmtvdi->item.cchTextMax);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TVN_GETINFOTIPA:
|
|
|
|
{
|
|
|
|
NMTVGETINFOTIPA *nmtvgit = (NMTVGETINFOTIPA *)hdr;
|
|
|
|
notify_generic_text_handler(&nmtvgit->pszText, &nmtvgit->cchTextMax);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TVN_SINGLEEXPAND:
|
|
|
|
case TVN_BEGINDRAGA:
|
|
|
|
case TVN_BEGINRDRAGA:
|
|
|
|
case TVN_ITEMEXPANDEDA:
|
|
|
|
case TVN_ITEMEXPANDINGA:
|
|
|
|
case TVN_DELETEITEMA:
|
|
|
|
case TVN_SELCHANGINGA:
|
|
|
|
case TVN_SELCHANGEDA:
|
|
|
|
{
|
|
|
|
NMTREEVIEWA *nmtv = (NMTREEVIEWA *)hdr;
|
|
|
|
if (notify_test_info.handler_id == TVITEM_NEW_HANDLER)
|
|
|
|
notify_generic_text_handler((CHAR **)&nmtv->itemNew.pszText, &nmtv->itemNew.cchTextMax);
|
|
|
|
else
|
|
|
|
notify_generic_text_handler((CHAR **)&nmtv->itemOld.pszText, &nmtv->itemOld.cchTextMax);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-09-17 17:26:31 +02:00
|
|
|
default:
|
|
|
|
ok(0, "Unexpected message 0x%08x\n", hdr->code);
|
|
|
|
}
|
|
|
|
notify_test_info.received = TRUE;
|
|
|
|
ok(!lstrcmpA(test_a, "test"), "test_a got modified\n");
|
|
|
|
ok(!lstrcmpW(test_w, test), "test_w got modified\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
case WM_NOTIFYFORMAT:
|
|
|
|
if (lParam == NF_QUERY) return NFR_ANSI;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return DefWindowProcA(hwnd, message, wParam, lParam);
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL register_test_notify_class(void)
|
|
|
|
{
|
|
|
|
WNDCLASSA cls = {0};
|
|
|
|
|
|
|
|
cls.lpfnWndProc = test_notify_proc;
|
|
|
|
cls.hInstance = GetModuleHandleA(NULL);
|
|
|
|
cls.lpszClassName = "Pager notify class";
|
|
|
|
return RegisterClassA(&cls);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void send_notify(HWND pager, UINT unicode, UINT ansi, LPARAM lParam, BOOL code_change)
|
|
|
|
{
|
|
|
|
NMHDR *hdr = (NMHDR *)lParam;
|
|
|
|
|
|
|
|
notify_test_info.unicode = unicode;
|
|
|
|
notify_test_info.id_from = 1;
|
|
|
|
notify_test_info.hwnd_from = child1_wnd;
|
|
|
|
notify_test_info.ansi = ansi;
|
|
|
|
notify_test_info.received = FALSE;
|
|
|
|
|
|
|
|
hdr->code = unicode;
|
|
|
|
hdr->idFrom = 1;
|
|
|
|
hdr->hwndFrom = child1_wnd;
|
|
|
|
|
|
|
|
SendMessageW(pager, WM_NOTIFY, hdr->idFrom, lParam);
|
|
|
|
ok(notify_test_info.received, "Expect notification received\n");
|
|
|
|
ok(hdr->code == code_change ? ansi : unicode, "Expect 0x%08x, got 0x%08x\n", hdr->code,
|
|
|
|
code_change ? ansi : unicode);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Send notify to test text field conversion. In parent proc notify_generic_text_handler() handles these messages */
|
|
|
|
static void test_notify_generic_text_helper(HWND pager, const struct generic_text_helper_para *para)
|
|
|
|
{
|
|
|
|
const struct notify_test_send *send_data;
|
|
|
|
const struct notify_test_receive *receive_data;
|
|
|
|
INT array_size;
|
|
|
|
INT i;
|
|
|
|
|
|
|
|
notify_test_info.flags = para->flags;
|
2018-09-28 05:06:45 +02:00
|
|
|
notify_test_info.handler_id = para->handler_id;
|
2018-09-17 17:26:31 +02:00
|
|
|
|
|
|
|
if (para->flags & (CONVERT_SEND | DONT_CONVERT_SEND))
|
|
|
|
{
|
|
|
|
if (para->flags & CONVERT_SEND)
|
|
|
|
{
|
|
|
|
notify_test_info.test_id = CONVERT_SEND;
|
|
|
|
send_data = test_convert_send_data;
|
|
|
|
array_size = ARRAY_SIZE(test_convert_send_data);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
notify_test_info.test_id = DONT_CONVERT_SEND;
|
|
|
|
send_data = test_dont_convert_send_data;
|
|
|
|
array_size = ARRAY_SIZE(test_dont_convert_send_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < array_size; i++)
|
|
|
|
{
|
|
|
|
const struct notify_test_send *data = send_data + i;
|
|
|
|
notify_test_info.sub_test_id = i;
|
|
|
|
|
|
|
|
memset(para->ptr, 0, para->size);
|
|
|
|
if (para->mask) *para->mask = para->required_mask;
|
|
|
|
if (data->send_text)
|
|
|
|
{
|
|
|
|
memcpy(buffer, data->send_text, data->send_text_size);
|
|
|
|
*para->text = buffer;
|
|
|
|
}
|
|
|
|
if (para->text_max) *para->text_max = data->send_text_max;
|
|
|
|
send_notify(pager, para->code_unicode, para->code_ansi, (LPARAM)para->ptr, TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (para->flags & (CONVERT_RECEIVE | DONT_CONVERT_RECEIVE))
|
|
|
|
{
|
|
|
|
if (para->flags & CONVERT_RECEIVE)
|
|
|
|
{
|
|
|
|
notify_test_info.test_id = CONVERT_RECEIVE;
|
|
|
|
receive_data = test_convert_receive_data;
|
|
|
|
array_size = ARRAY_SIZE(test_convert_receive_data);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
notify_test_info.test_id = DONT_CONVERT_RECEIVE;
|
|
|
|
receive_data = test_dont_convert_receive_data;
|
|
|
|
array_size = ARRAY_SIZE(test_dont_convert_receive_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < array_size; i++)
|
|
|
|
{
|
|
|
|
const struct notify_test_receive *data = receive_data + i;
|
|
|
|
notify_test_info.sub_test_id = i;
|
|
|
|
|
|
|
|
memset(para->ptr, 0, para->size);
|
|
|
|
if (para->mask) *para->mask = para->required_mask;
|
|
|
|
if (data->send_text)
|
|
|
|
{
|
|
|
|
memcpy(buffer, data->send_text, data->send_text_size);
|
|
|
|
*para->text = buffer;
|
|
|
|
}
|
|
|
|
if (para->text_max) *para->text_max = data->send_text_max;
|
|
|
|
send_notify(pager, para->code_unicode, para->code_ansi, (LPARAM)para->ptr, TRUE);
|
|
|
|
if (data->return_text)
|
|
|
|
{
|
|
|
|
if (para->flags & CONVERT_RECEIVE)
|
|
|
|
ok(!lstrcmpW(data->return_text, *para->text), "Code 0x%08x sub test %d expect %s, got %s\n",
|
|
|
|
para->code_unicode, i, wine_dbgstr_w((WCHAR *)data->return_text), wine_dbgstr_w(*para->text));
|
|
|
|
else
|
|
|
|
ok(!lstrcmpA(data->return_text, (CHAR *)*para->text), "Code 0x%08x sub test %d expect %s, got %s\n",
|
|
|
|
para->code_unicode, i, (CHAR *)data->return_text, (CHAR *)*para->text);
|
|
|
|
}
|
|
|
|
if (para->text_max)
|
|
|
|
ok(data->return_text_max == *para->text_max, "Code 0x%08x sub test %d expect %d, got %d\n",
|
|
|
|
para->code_unicode, i, data->return_text_max, *para->text_max);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Extra tests for other behavior flags that are not worth it to create their own test arrays */
|
|
|
|
memset(para->ptr, 0, para->size);
|
|
|
|
if (para->mask) *para->mask = para->required_mask;
|
|
|
|
if (para->text_max) *para->text_max = 1;
|
|
|
|
if (para->flags & SEND_EMPTY_IF_NULL)
|
|
|
|
notify_test_info.test_id = SEND_EMPTY_IF_NULL;
|
|
|
|
else
|
|
|
|
notify_test_info.test_id = DONT_SEND_EMPTY_IF_NULL;
|
|
|
|
send_notify(pager, para->code_unicode, para->code_ansi, (LPARAM)para->ptr, TRUE);
|
2018-09-17 17:27:00 +02:00
|
|
|
|
|
|
|
notify_test_info.test_id = SET_NULL_IF_NO_MASK;
|
|
|
|
memset(para->ptr, 0, para->size);
|
|
|
|
memset(buffer, 0, sizeof(buffer));
|
|
|
|
*para->text = buffer;
|
|
|
|
if (para->text_max) *para->text_max = ARRAY_SIZE(buffer);
|
|
|
|
send_notify(pager, para->code_unicode, para->code_ansi, (LPARAM)para->ptr, TRUE);
|
|
|
|
if(para->flags & SET_NULL_IF_NO_MASK)
|
|
|
|
ok(!*para->text, "Expect null text\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_wm_notify_comboboxex(HWND pager)
|
|
|
|
{
|
|
|
|
static NMCBEDRAGBEGINW nmcbedb;
|
|
|
|
static NMCBEENDEDITW nmcbeed;
|
|
|
|
|
|
|
|
/* CBEN_DRAGBEGIN */
|
|
|
|
memset(&nmcbedb, 0, sizeof(nmcbedb));
|
|
|
|
memcpy(nmcbedb.szText, test_w, sizeof(test_w));
|
|
|
|
send_notify(pager, CBEN_DRAGBEGINW, CBEN_DRAGBEGINA, (LPARAM)&nmcbedb, FALSE);
|
|
|
|
ok(!lstrcmpW(nmcbedb.szText, test_w), "Expect %s, got %s\n", wine_dbgstr_w(test_w), wine_dbgstr_w(nmcbedb.szText));
|
|
|
|
|
|
|
|
/* CBEN_ENDEDIT */
|
|
|
|
memset(&nmcbeed, 0, sizeof(nmcbeed));
|
|
|
|
memcpy(nmcbeed.szText, test_w, sizeof(test_w));
|
|
|
|
send_notify(pager, CBEN_ENDEDITW, CBEN_ENDEDITA, (LPARAM)&nmcbeed, FALSE);
|
|
|
|
ok(!lstrcmpW(nmcbeed.szText, test_w), "Expect %s, got %s\n", wine_dbgstr_w(test_w), wine_dbgstr_w(nmcbeed.szText));
|
2018-09-17 17:26:31 +02:00
|
|
|
}
|
|
|
|
|
2018-09-24 09:12:02 +02:00
|
|
|
static void test_wm_notify_datetime(HWND pager)
|
|
|
|
{
|
|
|
|
const struct notify_test_datetime_format *data;
|
|
|
|
NMDATETIMEFORMATW nmdtf;
|
|
|
|
INT i;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(test_datetime_format_data); i++)
|
|
|
|
{
|
|
|
|
data = test_datetime_format_data + i;
|
|
|
|
notify_test_info.sub_test_id = i;
|
|
|
|
|
|
|
|
memset(&nmdtf, 0, sizeof(nmdtf));
|
|
|
|
if(data->send_pszformat) nmdtf.pszFormat = data->send_pszformat;
|
|
|
|
nmdtf.pszDisplay = nmdtf.szDisplay;
|
|
|
|
send_notify(pager, DTN_FORMATW, DTN_FORMATA, (LPARAM)&nmdtf, TRUE);
|
|
|
|
if (data->return_szdisplay)
|
|
|
|
ok(!lstrcmpW(nmdtf.szDisplay, data->return_szdisplay), "Sub test %d expect %s, got %s\n", i,
|
|
|
|
wine_dbgstr_w(data->return_szdisplay), wine_dbgstr_w(nmdtf.szDisplay));
|
|
|
|
if (data->return_pszdisplay)
|
|
|
|
ok(!lstrcmpW(nmdtf.pszDisplay, data->return_pszdisplay), "Sub test %d expect %s, got %s\n", i,
|
|
|
|
wine_dbgstr_w(data->return_pszdisplay), wine_dbgstr_w(nmdtf.pszDisplay));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-17 17:26:48 +02:00
|
|
|
static void test_wm_notify_tooltip(HWND pager)
|
|
|
|
{
|
|
|
|
NMTTDISPINFOW nmttdi;
|
|
|
|
const struct notify_test_tooltip *data;
|
|
|
|
INT i;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(test_tooltip_data); i++)
|
|
|
|
{
|
|
|
|
data = test_tooltip_data + i;
|
|
|
|
notify_test_info.sub_test_id = i;
|
|
|
|
|
|
|
|
memset(&nmttdi, 0, sizeof(nmttdi));
|
|
|
|
if (data->send_sztext) memcpy(nmttdi.szText, data->send_sztext, data->send_sztext_size);
|
|
|
|
if (data->send_lpsztext) nmttdi.lpszText = data->send_lpsztext;
|
|
|
|
send_notify(pager, TTN_GETDISPINFOW, TTN_GETDISPINFOA, (LPARAM)&nmttdi, FALSE);
|
|
|
|
if (data->return_sztext)
|
|
|
|
{
|
|
|
|
if (data->return_sztext_size == -1)
|
|
|
|
ok(!lstrcmpW(nmttdi.szText, data->return_sztext), "Sub test %d expect %s, got %s\n", i,
|
|
|
|
wine_dbgstr_w(data->return_sztext), wine_dbgstr_w(nmttdi.szText));
|
|
|
|
else
|
|
|
|
ok(!memcmp(nmttdi.szText, data->return_sztext, data->return_sztext_size), "Wrong szText content\n");
|
|
|
|
}
|
|
|
|
if (data->return_lpsztext)
|
|
|
|
{
|
|
|
|
if (IS_INTRESOURCE(data->return_lpsztext))
|
|
|
|
ok(nmttdi.lpszText == data->return_lpsztext, "Sub test %d expect %s, got %s\n", i,
|
|
|
|
wine_dbgstr_w(data->return_lpsztext), wine_dbgstr_w(nmttdi.lpszText));
|
|
|
|
else
|
|
|
|
ok(!lstrcmpW(nmttdi.lpszText, data->return_lpsztext), "Test %d expect %s, got %s\n", i,
|
|
|
|
wine_dbgstr_w(data->return_lpsztext), wine_dbgstr_w(nmttdi.lpszText));
|
|
|
|
}
|
|
|
|
if (data->return_hinst)
|
|
|
|
ok(nmttdi.hinst == data->return_hinst, "Sub test %d expect %p, got %p\n", i, data->return_hinst,
|
|
|
|
nmttdi.hinst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-17 17:26:31 +02:00
|
|
|
static void test_wm_notify(void)
|
|
|
|
{
|
|
|
|
static const CHAR *class = "Pager notify class";
|
|
|
|
HWND parent, pager;
|
2018-09-17 17:27:00 +02:00
|
|
|
/* Combo Box Ex */
|
|
|
|
static NMCOMBOBOXEXW nmcbe;
|
2018-09-24 09:12:02 +02:00
|
|
|
/* Date and Time Picker */
|
|
|
|
static NMDATETIMEFORMATQUERYW nmdtfq;
|
|
|
|
static NMDATETIMEWMKEYDOWNW nmdtkd;
|
|
|
|
static NMDATETIMESTRINGW nmdts;
|
2018-09-28 05:06:28 +02:00
|
|
|
/* List View */
|
|
|
|
static NMLVDISPINFOW nmlvdi;
|
|
|
|
static NMLVGETINFOTIPW nmlvgit;
|
|
|
|
static NMLVFINDITEMW nmlvfi;
|
2018-09-17 17:26:31 +02:00
|
|
|
/* Tool Bar */
|
|
|
|
static NMTBRESTORE nmtbr;
|
|
|
|
static NMTBSAVE nmtbs;
|
|
|
|
static NMTOOLBARW nmtb;
|
|
|
|
static NMTBDISPINFOW nmtbdi;
|
|
|
|
static NMTBGETINFOTIPW nmtbgit;
|
2018-09-28 05:06:45 +02:00
|
|
|
/* Tree View */
|
|
|
|
static NMTVDISPINFOW nmtvdi;
|
|
|
|
static NMTVGETINFOTIPW nmtvgit;
|
|
|
|
static NMTREEVIEWW nmtv;
|
2018-09-17 17:26:31 +02:00
|
|
|
static const struct generic_text_helper_para paras[] =
|
|
|
|
{
|
2018-09-17 17:27:00 +02:00
|
|
|
/* Combo Box Ex */
|
|
|
|
{&nmcbe, sizeof(nmcbe), &nmcbe.ceItem.mask, CBEIF_TEXT, &nmcbe.ceItem.pszText, &nmcbe.ceItem.cchTextMax,
|
|
|
|
CBEN_INSERTITEM, CBEN_INSERTITEM, DONT_CONVERT_SEND | DONT_CONVERT_RECEIVE},
|
|
|
|
{&nmcbe, sizeof(nmcbe), &nmcbe.ceItem.mask, CBEIF_TEXT, &nmcbe.ceItem.pszText, &nmcbe.ceItem.cchTextMax,
|
|
|
|
CBEN_DELETEITEM, CBEN_DELETEITEM, DONT_CONVERT_SEND | DONT_CONVERT_RECEIVE},
|
|
|
|
{&nmcbe, sizeof(nmcbe), &nmcbe.ceItem.mask, CBEIF_TEXT, &nmcbe.ceItem.pszText, &nmcbe.ceItem.cchTextMax,
|
|
|
|
CBEN_GETDISPINFOW, CBEN_GETDISPINFOA, ZERO_SEND | SET_NULL_IF_NO_MASK | DONT_CONVERT_SEND | CONVERT_RECEIVE},
|
2018-09-24 09:12:02 +02:00
|
|
|
/* Date and Time Picker */
|
|
|
|
{&nmdtfq, sizeof(nmdtfq), NULL, 0, (WCHAR **)&nmdtfq.pszFormat, NULL, DTN_FORMATQUERYW, DTN_FORMATQUERYA,
|
|
|
|
CONVERT_SEND},
|
|
|
|
{&nmdtkd, sizeof(nmdtkd), NULL, 0, (WCHAR **)&nmdtkd.pszFormat, NULL, DTN_WMKEYDOWNW, DTN_WMKEYDOWNA,
|
|
|
|
CONVERT_SEND},
|
|
|
|
{&nmdts, sizeof(nmdts), NULL, 0, (WCHAR **)&nmdts.pszUserString, NULL, DTN_USERSTRINGW, DTN_USERSTRINGA,
|
|
|
|
CONVERT_SEND},
|
2018-09-28 05:06:28 +02:00
|
|
|
/* List View */
|
|
|
|
{&nmlvfi, sizeof(nmlvfi), &nmlvfi.lvfi.flags, LVFI_STRING, (WCHAR **)&nmlvfi.lvfi.psz, NULL,
|
|
|
|
LVN_INCREMENTALSEARCHW, LVN_INCREMENTALSEARCHA, CONVERT_SEND},
|
|
|
|
{&nmlvfi, sizeof(nmlvfi), &nmlvfi.lvfi.flags, LVFI_SUBSTRING, (WCHAR **)&nmlvfi.lvfi.psz, NULL, LVN_ODFINDITEMW,
|
|
|
|
LVN_ODFINDITEMA, CONVERT_SEND},
|
|
|
|
{&nmlvdi, sizeof(nmlvdi), &nmlvdi.item.mask, LVIF_TEXT, &nmlvdi.item.pszText, &nmlvdi.item.cchTextMax,
|
|
|
|
LVN_BEGINLABELEDITW, LVN_BEGINLABELEDITA, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmlvdi, sizeof(nmlvdi), &nmlvdi.item.mask, LVIF_TEXT, &nmlvdi.item.pszText, &nmlvdi.item.cchTextMax,
|
|
|
|
LVN_ENDLABELEDITW, LVN_ENDLABELEDITA, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmlvdi, sizeof(nmlvdi), &nmlvdi.item.mask, LVIF_TEXT, &nmlvdi.item.pszText, &nmlvdi.item.cchTextMax,
|
|
|
|
LVN_GETDISPINFOW, LVN_GETDISPINFOA, DONT_CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmlvdi, sizeof(nmlvdi), &nmlvdi.item.mask, LVIF_TEXT, &nmlvdi.item.pszText, &nmlvdi.item.cchTextMax,
|
|
|
|
LVN_SETDISPINFOW, LVN_SETDISPINFOA, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmlvgit, sizeof(nmlvgit), NULL, 0, &nmlvgit.pszText, &nmlvgit.cchTextMax, LVN_GETINFOTIPW, LVN_GETINFOTIPA,
|
|
|
|
CONVERT_SEND | CONVERT_RECEIVE},
|
2018-09-17 17:26:31 +02:00
|
|
|
/* Tool Bar */
|
|
|
|
{&nmtbs, sizeof(nmtbs), NULL, 0, (WCHAR **)&nmtbs.tbButton.iString, NULL, TBN_SAVE, TBN_SAVE,
|
|
|
|
DONT_CONVERT_SEND | DONT_CONVERT_RECEIVE},
|
|
|
|
{&nmtbr, sizeof(nmtbr), NULL, 0, (WCHAR **)&nmtbr.tbButton.iString, NULL, TBN_RESTORE, TBN_RESTORE,
|
|
|
|
DONT_CONVERT_SEND | DONT_CONVERT_RECEIVE},
|
|
|
|
{&nmtbdi, sizeof(nmtbdi), &nmtbdi.dwMask, TBNF_TEXT, &nmtbdi.pszText, &nmtbdi.cchText, TBN_GETDISPINFOW,
|
|
|
|
TBN_GETDISPINFOW, DONT_CONVERT_SEND | DONT_CONVERT_RECEIVE},
|
|
|
|
{&nmtb, sizeof(nmtb), NULL, 0, &nmtb.pszText, &nmtb.cchText, TBN_GETBUTTONINFOW, TBN_GETBUTTONINFOA,
|
|
|
|
SEND_EMPTY_IF_NULL | CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmtbgit, sizeof(nmtbgit), NULL, 0, &nmtbgit.pszText, &nmtbgit.cchTextMax, TBN_GETINFOTIPW, TBN_GETINFOTIPA,
|
2018-09-28 05:06:45 +02:00
|
|
|
DONT_CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
/* Tree View */
|
|
|
|
{&nmtvdi, sizeof(nmtvdi), &nmtvdi.item.mask, TVIF_TEXT, &nmtvdi.item.pszText, &nmtvdi.item.cchTextMax,
|
|
|
|
TVN_BEGINLABELEDITW, TVN_BEGINLABELEDITA, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmtvdi, sizeof(nmtvdi), &nmtvdi.item.mask, TVIF_TEXT, &nmtvdi.item.pszText, &nmtvdi.item.cchTextMax,
|
|
|
|
TVN_ENDLABELEDITW, TVN_ENDLABELEDITA, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmtvdi, sizeof(nmtvdi), &nmtvdi.item.mask, TVIF_TEXT, &nmtvdi.item.pszText, &nmtvdi.item.cchTextMax,
|
|
|
|
TVN_GETDISPINFOW, TVN_GETDISPINFOA, ZERO_SEND | DONT_CONVERT_SEND| CONVERT_RECEIVE},
|
|
|
|
{&nmtvdi, sizeof(nmtvdi), &nmtvdi.item.mask, TVIF_TEXT, &nmtvdi.item.pszText, &nmtvdi.item.cchTextMax,
|
|
|
|
TVN_SETDISPINFOW, TVN_SETDISPINFOA, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmtvgit, sizeof(nmtvgit), NULL, 0, &nmtvgit.pszText, &nmtvgit.cchTextMax, TVN_GETINFOTIPW, TVN_GETINFOTIPA,
|
|
|
|
DONT_CONVERT_SEND | CONVERT_RECEIVE},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemNew.mask, TVIF_TEXT, &nmtv.itemNew.pszText, &nmtv.itemNew.cchTextMax,
|
|
|
|
TVN_SINGLEEXPAND, TVN_SINGLEEXPAND, DONT_CONVERT_SEND | DONT_CONVERT_RECEIVE, TVITEM_NEW_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemOld.mask, TVIF_TEXT, &nmtv.itemOld.pszText, &nmtv.itemOld.cchTextMax,
|
|
|
|
TVN_SINGLEEXPAND, TVN_SINGLEEXPAND, DONT_CONVERT_SEND | DONT_CONVERT_RECEIVE, TVITEM_OLD_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemNew.mask, TVIF_TEXT, &nmtv.itemNew.pszText, &nmtv.itemNew.cchTextMax,
|
|
|
|
TVN_BEGINDRAGW, TVN_BEGINDRAGA, CONVERT_SEND, TVITEM_NEW_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemOld.mask, TVIF_TEXT, &nmtv.itemOld.pszText, &nmtv.itemOld.cchTextMax,
|
|
|
|
TVN_BEGINDRAGW, TVN_BEGINDRAGA, DONT_CONVERT_SEND, TVITEM_OLD_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemNew.mask, TVIF_TEXT, &nmtv.itemNew.pszText, &nmtv.itemNew.cchTextMax,
|
|
|
|
TVN_BEGINRDRAGW, TVN_BEGINRDRAGA, CONVERT_SEND, TVITEM_NEW_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemOld.mask, TVIF_TEXT, &nmtv.itemOld.pszText, &nmtv.itemOld.cchTextMax,
|
|
|
|
TVN_BEGINRDRAGW, TVN_BEGINRDRAGA, DONT_CONVERT_SEND, TVITEM_OLD_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemNew.mask, TVIF_TEXT, &nmtv.itemNew.pszText, &nmtv.itemNew.cchTextMax,
|
|
|
|
TVN_ITEMEXPANDEDW, TVN_ITEMEXPANDEDA, CONVERT_SEND, TVITEM_NEW_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemOld.mask, TVIF_TEXT, &nmtv.itemOld.pszText, &nmtv.itemOld.cchTextMax,
|
|
|
|
TVN_ITEMEXPANDEDW, TVN_ITEMEXPANDEDA, DONT_CONVERT_SEND, TVITEM_OLD_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemNew.mask, TVIF_TEXT, &nmtv.itemNew.pszText, &nmtv.itemNew.cchTextMax,
|
|
|
|
TVN_ITEMEXPANDINGW, TVN_ITEMEXPANDINGA, CONVERT_SEND, TVITEM_NEW_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemOld.mask, TVIF_TEXT, &nmtv.itemOld.pszText, &nmtv.itemOld.cchTextMax,
|
|
|
|
TVN_ITEMEXPANDINGW, TVN_ITEMEXPANDINGA, DONT_CONVERT_SEND, TVITEM_OLD_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemNew.mask, TVIF_TEXT, &nmtv.itemNew.pszText, &nmtv.itemNew.cchTextMax,
|
|
|
|
TVN_DELETEITEMW, TVN_DELETEITEMA, DONT_CONVERT_SEND, TVITEM_NEW_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemOld.mask, TVIF_TEXT, &nmtv.itemOld.pszText, &nmtv.itemOld.cchTextMax,
|
|
|
|
TVN_DELETEITEMW, TVN_DELETEITEMA, CONVERT_SEND, TVITEM_OLD_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemNew.mask, TVIF_TEXT, &nmtv.itemNew.pszText, &nmtv.itemNew.cchTextMax,
|
|
|
|
TVN_SELCHANGINGW, TVN_SELCHANGINGA, CONVERT_SEND, TVITEM_NEW_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemOld.mask, TVIF_TEXT, &nmtv.itemOld.pszText, &nmtv.itemOld.cchTextMax,
|
|
|
|
TVN_SELCHANGINGW, TVN_SELCHANGINGA, CONVERT_SEND, TVITEM_OLD_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemNew.mask, TVIF_TEXT, &nmtv.itemNew.pszText, &nmtv.itemNew.cchTextMax,
|
|
|
|
TVN_SELCHANGEDW, TVN_SELCHANGEDA, CONVERT_SEND, TVITEM_NEW_HANDLER},
|
|
|
|
{&nmtv, sizeof(nmtv), &nmtv.itemOld.mask, TVIF_TEXT, &nmtv.itemOld.pszText, &nmtv.itemOld.cchTextMax,
|
|
|
|
TVN_SELCHANGEDW, TVN_SELCHANGEDA, CONVERT_SEND, TVITEM_OLD_HANDLER}
|
2018-09-17 17:26:31 +02:00
|
|
|
};
|
|
|
|
INT i;
|
|
|
|
|
|
|
|
ok(register_test_notify_class(), "Register test class failed, error 0x%08x\n", GetLastError());
|
|
|
|
|
|
|
|
parent = CreateWindowA(class, "parent", WS_OVERLAPPED, 0, 0, 100, 100, 0, 0, GetModuleHandleA(0), 0);
|
|
|
|
ok(parent != NULL, "CreateWindow failed\n");
|
|
|
|
pager = CreateWindowA(WC_PAGESCROLLERA, "pager", WS_CHILD, 0, 0, 100, 100, parent, 0, GetModuleHandleA(0), 0);
|
|
|
|
ok(pager != NULL, "CreateWindow failed\n");
|
|
|
|
child1_wnd = CreateWindowA(class, "child", WS_CHILD, 0, 0, 100, 100, pager, (HMENU)1, GetModuleHandleA(0), 0);
|
|
|
|
ok(child1_wnd != NULL, "CreateWindow failed\n");
|
|
|
|
SendMessageW(pager, PGM_SETCHILD, 0, (LPARAM)child1_wnd);
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(paras); i++)
|
|
|
|
test_notify_generic_text_helper(pager, paras + i);
|
|
|
|
|
2018-09-17 17:26:48 +02:00
|
|
|
/* Tests for those that can't be covered by generic text test helper */
|
2018-09-17 17:27:00 +02:00
|
|
|
test_wm_notify_comboboxex(pager);
|
2018-09-24 09:12:02 +02:00
|
|
|
test_wm_notify_datetime(pager);
|
2018-09-17 17:26:48 +02:00
|
|
|
test_wm_notify_tooltip(pager);
|
|
|
|
|
2018-09-17 17:26:31 +02:00
|
|
|
DestroyWindow(parent);
|
|
|
|
UnregisterClassA(class, GetModuleHandleA(NULL));
|
|
|
|
}
|
|
|
|
|
2018-08-24 16:04:20 +02:00
|
|
|
static void init_functions(void)
|
2012-11-20 17:21:08 +01:00
|
|
|
{
|
2018-08-24 16:04:20 +02:00
|
|
|
HMODULE mod = LoadLibraryA("comctl32.dll");
|
|
|
|
|
|
|
|
#define X(f) p##f = (void*)GetProcAddress(mod, #f);
|
|
|
|
X(InitCommonControlsEx);
|
|
|
|
#undef X
|
2012-11-20 17:21:08 +01:00
|
|
|
|
|
|
|
pSetWindowSubclass = (void*)GetProcAddress(mod, (LPSTR)410);
|
2018-08-24 16:04:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
START_TEST(pager)
|
|
|
|
{
|
|
|
|
INITCOMMONCONTROLSEX iccex;
|
|
|
|
|
|
|
|
init_functions();
|
|
|
|
|
|
|
|
iccex.dwSize = sizeof(iccex);
|
|
|
|
iccex.dwICC = ICC_PAGESCROLLER_CLASS;
|
|
|
|
pInitCommonControlsEx(&iccex);
|
2012-11-20 17:21:08 +01:00
|
|
|
|
|
|
|
init_msg_sequences(sequences, NUM_MSG_SEQUENCES);
|
|
|
|
|
|
|
|
parent_wnd = create_parent_window();
|
|
|
|
ok(parent_wnd != NULL, "Failed to create parent window!\n");
|
|
|
|
|
|
|
|
test_pager();
|
2018-08-24 16:04:35 +02:00
|
|
|
test_wm_notifyformat();
|
2018-09-17 17:26:31 +02:00
|
|
|
test_wm_notify();
|
2018-08-24 16:04:20 +02:00
|
|
|
|
|
|
|
DestroyWindow(parent_wnd);
|
2012-11-20 17:21:08 +01:00
|
|
|
}
|