260 lines
9.4 KiB
C
260 lines
9.4 KiB
C
/* Unit test suite for Twain DSM functions
|
|
*
|
|
* Copyright 2009 Jeremy White, CodeWeavers, Inc.
|
|
*
|
|
* 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 <stdarg.h>
|
|
#include <assert.h>
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "wingdi.h"
|
|
#include "winerror.h"
|
|
#include "winuser.h"
|
|
#include "twain.h"
|
|
|
|
#include "wine/test.h"
|
|
|
|
static DSMENTRYPROC pDSM_Entry;
|
|
|
|
static BOOL dsm_RegisterWindowClasses(void)
|
|
{
|
|
WNDCLASSA cls;
|
|
|
|
cls.style = 0;
|
|
cls.lpfnWndProc = DefWindowProc;
|
|
cls.cbClsExtra = 0;
|
|
cls.cbWndExtra = 0;
|
|
cls.hInstance = GetModuleHandleA(0);
|
|
cls.hIcon = 0;
|
|
cls.hCursor = LoadCursorA(0, IDC_ARROW);
|
|
cls.hbrBackground = GetStockObject(WHITE_BRUSH);
|
|
cls.lpszMenuName = NULL;
|
|
cls.lpszClassName = "TWAIN_dsm_class";
|
|
if (!RegisterClassA(&cls)) return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void get_condition_code(TW_IDENTITY *appid, TW_IDENTITY *source, TW_STATUS *status)
|
|
{
|
|
TW_UINT16 rc;
|
|
rc = pDSM_Entry(appid, source, DG_CONTROL, DAT_STATUS, MSG_GET, status);
|
|
ok(rc == TWRC_SUCCESS, "Condition code not available, rc %d\n", rc);
|
|
}
|
|
|
|
static void test_single_source(TW_IDENTITY *appid, TW_IDENTITY *source)
|
|
{
|
|
TW_UINT16 rc;
|
|
TW_STATUS status;
|
|
TW_CAPABILITY cap;
|
|
UINT16 capabilities[CAP_CUSTOMBASE];
|
|
|
|
memset(&cap, 0, sizeof(cap));
|
|
cap.Cap = CAP_SUPPORTEDCAPS;
|
|
cap.ConType = TWON_DONTCARE16;
|
|
|
|
rc = pDSM_Entry(appid, source, DG_CONTROL, DAT_CAPABILITY, MSG_GET, &cap);
|
|
get_condition_code(appid, source, &status);
|
|
ok(rc == TWRC_SUCCESS || status.ConditionCode == TWCC_SUCCESS,
|
|
"Error obtaining CAP_SUPPORTEDCAPS\n");
|
|
|
|
memset(capabilities, 0, sizeof(capabilities));
|
|
if (rc == TWRC_SUCCESS && cap.ConType == TWON_ARRAY)
|
|
{
|
|
TW_ARRAY *a;
|
|
a = GlobalLock(cap.hContainer);
|
|
if (a)
|
|
{
|
|
if (a->ItemType == TWTY_UINT16)
|
|
{
|
|
int i;
|
|
UINT16 *u = (UINT16 *) a->ItemList;
|
|
trace("%d Capabilities:\n", a->NumItems);
|
|
for (i = 0; i < a->NumItems; i++)
|
|
if (u[i] < sizeof(capabilities) / sizeof(capabilities[0]))
|
|
{
|
|
capabilities[u[i]] = 1;
|
|
trace(" %d: 0x%x\n", i, u[i]);
|
|
}
|
|
}
|
|
GlobalUnlock(cap.hContainer);
|
|
}
|
|
}
|
|
|
|
/* For Twain 1.6, all sources must support: */
|
|
ok(capabilities[CAP_SUPPORTEDCAPS], "CAP_SUPPORTEDCAPS not supported\n");
|
|
todo_wine
|
|
ok(capabilities[CAP_XFERCOUNT], "CAP_XFERCOUNT not supported\n");
|
|
todo_wine
|
|
ok(capabilities[CAP_UICONTROLLABLE], "CAP_UICONTROLLABLE not supported\n");
|
|
|
|
if (source->SupportedGroups & DG_IMAGE)
|
|
{
|
|
/* For Twain 1.6:
|
|
Sources that supply image information must support DG_CONTROL / DAT_CAPABILITY /
|
|
MSG_GET, MSG_GETCURRENT, MSG_GETDEFAULT on:
|
|
*/
|
|
todo_wine
|
|
ok(capabilities[ICAP_COMPRESSION], "ICAP_COMPRESSION not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_PLANARCHUNKY], "ICAP_PLANARCHUNKY not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_PHYSICALHEIGHT], "ICAP_PHYSICALHEIGHT not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_PHYSICALWIDTH], "ICAP_PHYSICALWIDTH not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_PIXELFLAVOR], "ICAP_PIXELFLAVOR not supported\n");
|
|
|
|
/* For Twain 1.6:
|
|
Sources that supply image information must support DG_CONTROL / DAT_CAPABILITY /
|
|
MSG_GET, MSG_GETCURRENT, MSG_GETDEFAULT, MSG_RESET and MSG_SET on:
|
|
*/
|
|
todo_wine
|
|
ok(capabilities[ICAP_BITDEPTH], "ICAP_BITDEPTH not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_BITORDER], "ICAP_BITORDER not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_PIXELTYPE], "ICAP_PIXELTYPE not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_UNITS], "ICAP_UNITS not supported\n");
|
|
ok(capabilities[ICAP_XFERMECH], "ICAP_XFERMECH not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_XRESOLUTION], "ICAP_XRESOLUTION not supported\n");
|
|
todo_wine
|
|
ok(capabilities[ICAP_YRESOLUTION], "ICAP_YRESOLUTION not supported\n");
|
|
}
|
|
}
|
|
|
|
static void test_sources(TW_IDENTITY *appid)
|
|
{
|
|
TW_UINT16 rc;
|
|
TW_IDENTITY source;
|
|
TW_STATUS status;
|
|
int scannercount = 0;
|
|
|
|
memset(&source, 0, sizeof(source));
|
|
rc = pDSM_Entry(appid, NULL, DG_CONTROL, DAT_IDENTITY, MSG_GETFIRST, &source);
|
|
get_condition_code(appid, NULL, &status);
|
|
ok( (rc == TWRC_SUCCESS && status.ConditionCode == TWCC_SUCCESS) ||
|
|
(rc == TWRC_FAILURE && status.ConditionCode == TWCC_NODS),
|
|
"Get first invalid condition code, rc %d, cc %d\n", rc, status.ConditionCode);
|
|
|
|
while (rc == TWRC_SUCCESS)
|
|
{
|
|
scannercount++;
|
|
trace("[Scanner %d|Version %d.%d(%s)|Protocol %d.%d|SupportedGroups 0x%x|Manufacturer %s|Family %s|ProductName %s]\n",
|
|
scannercount,
|
|
source.Version.MajorNum, source.Version.MinorNum, source.Version.Info,
|
|
source.ProtocolMajor, source.ProtocolMinor, source.SupportedGroups,
|
|
source.Manufacturer, source.ProductFamily, source.ProductName);
|
|
memset(&source, 0, sizeof(source));
|
|
rc = pDSM_Entry(appid, NULL, DG_CONTROL, DAT_IDENTITY, MSG_GETNEXT, &source);
|
|
get_condition_code(appid, NULL, &status);
|
|
ok(rc == TWRC_SUCCESS || rc == TWRC_ENDOFLIST, "Get next source failed, rc %d, cc %d\n", rc, status.ConditionCode);
|
|
}
|
|
|
|
memset(&source, 0, sizeof(source));
|
|
rc = pDSM_Entry(appid, NULL, DG_CONTROL, DAT_IDENTITY, MSG_GETDEFAULT, &source);
|
|
get_condition_code(appid, NULL, &status);
|
|
ok( (rc == TWRC_SUCCESS && status.ConditionCode == TWCC_SUCCESS) ||
|
|
(rc == TWRC_FAILURE && status.ConditionCode == TWCC_NODS),
|
|
"Get default invalid condition code, rc %d, cc %d\n", rc, status.ConditionCode);
|
|
|
|
if (rc == TWRC_SUCCESS && status.ConditionCode == TWCC_SUCCESS)
|
|
{
|
|
rc = pDSM_Entry(appid, NULL, DG_CONTROL, DAT_IDENTITY, MSG_OPENDS, &source);
|
|
get_condition_code(appid, NULL, &status);
|
|
|
|
if (rc == TWRC_SUCCESS && status.ConditionCode == TWCC_SUCCESS)
|
|
{
|
|
rc = pDSM_Entry(appid, NULL, DG_CONTROL, DAT_IDENTITY, MSG_CLOSEDS, &source);
|
|
get_condition_code(appid, NULL, &status);
|
|
ok(rc == TWRC_SUCCESS, "Close DS Failed, rc %d, cc %d\n", rc, status.ConditionCode);
|
|
}
|
|
}
|
|
|
|
if (winetest_interactive)
|
|
{
|
|
trace("Interactive, so trying userselect\n");
|
|
memset(&source, 0, sizeof(source));
|
|
rc = pDSM_Entry(appid, NULL, DG_CONTROL, DAT_IDENTITY, MSG_USERSELECT, &source);
|
|
get_condition_code(appid, NULL, &status);
|
|
ok(rc == TWRC_SUCCESS || rc == TWRC_CANCEL, "Userselect failed, rc %d, cc %d\n", rc, status.ConditionCode);
|
|
|
|
if (rc == TWRC_SUCCESS && status.ConditionCode == TWCC_SUCCESS)
|
|
{
|
|
rc = pDSM_Entry(appid, NULL, DG_CONTROL, DAT_IDENTITY, MSG_OPENDS, &source);
|
|
get_condition_code(appid, NULL, &status);
|
|
if (rc == TWRC_SUCCESS && status.ConditionCode == TWCC_SUCCESS)
|
|
{
|
|
test_single_source(appid, &source);
|
|
rc = pDSM_Entry(appid, NULL, DG_CONTROL, DAT_IDENTITY, MSG_CLOSEDS, &source);
|
|
get_condition_code(appid, NULL, &status);
|
|
ok(rc == TWRC_SUCCESS, "Close DS Failed, rc %d, cc %d\n", rc, status.ConditionCode);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
START_TEST(dsm)
|
|
{
|
|
TW_IDENTITY appid;
|
|
TW_UINT16 rc;
|
|
HANDLE hwnd;
|
|
HMODULE htwain;
|
|
|
|
if (!dsm_RegisterWindowClasses()) assert(0);
|
|
|
|
htwain = LoadLibraryA("twain_32.dll");
|
|
if (! htwain)
|
|
{
|
|
skip("twain_32.dll not available, skipping tests\n");
|
|
return;
|
|
}
|
|
pDSM_Entry = (void*)GetProcAddress(htwain, "DSM_Entry");
|
|
ok(pDSM_Entry != NULL, "Unable to GetProcAddress DSM_Entry\n");
|
|
if (! pDSM_Entry)
|
|
{
|
|
skip("DSM_Entry not available, skipping tests\n");
|
|
return;
|
|
}
|
|
|
|
memset(&appid, 0, sizeof(appid));
|
|
appid.Version.Language = TWLG_ENGLISH_USA;
|
|
appid.Version.Country = TWCY_USA;
|
|
appid.ProtocolMajor = TWON_PROTOCOLMAJOR;
|
|
appid.ProtocolMinor = TWON_PROTOCOLMINOR;
|
|
appid.SupportedGroups = DG_CONTROL | DG_IMAGE;
|
|
|
|
hwnd = CreateWindow("TWAIN_dsm_class", "Twain Test", 0,
|
|
CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
|
|
NULL, NULL, GetModuleHandleA(0), NULL);
|
|
|
|
rc = pDSM_Entry(&appid, NULL, DG_CONTROL, DAT_PARENT, MSG_OPENDSM, (TW_MEMREF) &hwnd);
|
|
ok(rc == TWRC_SUCCESS, "MSG_OPENDSM returned %d\n", rc);
|
|
|
|
test_sources(&appid);
|
|
|
|
rc = pDSM_Entry(&appid, NULL, DG_CONTROL, DAT_PARENT, MSG_CLOSEDSM, (TW_MEMREF) &hwnd);
|
|
ok(rc == TWRC_SUCCESS, "MSG_CLOSEDSM returned %d\n", rc);
|
|
|
|
DestroyWindow(hwnd);
|
|
FreeLibrary(htwain);
|
|
}
|