#include typedef struct _stack_t { struct _stack_t *next; char text[1]; // this should be the length of string_size } stack_t; int popstring(char *str); // 0 on success, 1 on empty stack void pushstring(char *str); enum { INST_0, // $0 INST_1, // $1 INST_2, // $2 INST_3, // $3 INST_4, // $4 INST_5, // $5 INST_6, // $6 INST_7, // $7 INST_8, // $8 INST_9, // $9 INST_R0, // $R0 INST_R1, // $R1 INST_R2, // $R2 INST_R3, // $R3 INST_R4, // $R4 INST_R5, // $R5 INST_R6, // $R6 INST_R7, // $R7 INST_R8, // $R8 INST_R9, // $R9 INST_CMDLINE, // $CMDLINE INST_INSTDIR, // $INSTDIR INST_OUTDIR, // $OUTDIR INST_EXEDIR, // $EXEDIR INST_LANG, // $LANGUAGE __INST_LAST }; char *getuservariable(int varnum); HINSTANCE g_hInstance; HWND g_hwndParent; int g_stringsize; stack_t **g_stacktop; char *g_variables; HBITMAP g_hbm; int sleep_val; int g_rv; static LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { if (uMsg == WM_CREATE) { BITMAP bm; RECT vp; GetObject(g_hbm, sizeof(bm), (LPSTR)&bm); SystemParametersInfo(SPI_GETWORKAREA, 0, &vp, 0); SetWindowLong(hwnd,GWL_STYLE,0); SetWindowPos(hwnd,NULL, vp.left+(vp.right-vp.left-bm.bmWidth)/2, vp.top+(vp.bottom-vp.top-bm.bmHeight)/2, bm.bmWidth,bm.bmHeight, SWP_NOZORDER); ShowWindow(hwnd,SW_SHOW); SetTimer(hwnd,1,sleep_val,NULL); return 0; } if (uMsg == WM_PAINT) { PAINTSTRUCT ps; RECT r; HDC curdc=BeginPaint(hwnd,&ps); HDC hdc=CreateCompatibleDC(curdc); HBITMAP oldbm; GetClientRect(hwnd,&r); oldbm=(HBITMAP)SelectObject(hdc,g_hbm); BitBlt(curdc,r.left,r.top,r.right-r.left,r.bottom-r.top,hdc,0,0,SRCCOPY); SelectObject(hdc,oldbm); DeleteDC(hdc); EndPaint(hwnd,&ps); return 0; } if (uMsg == WM_CLOSE) return 0; if (uMsg == WM_DESTROY) { PostQuitMessage(0); return 0; } if (uMsg == WM_TIMER || uMsg == WM_LBUTTONDOWN) { g_rv=(uMsg == WM_LBUTTONDOWN); DestroyWindow(hwnd); } return DefWindowProc(hwnd,uMsg,wParam,lParam); } BOOL WINAPI _DllMainCRTStartup(HANDLE hInst, ULONG ul_reason_for_call, LPVOID lpReserved) { g_hInstance=hInst; return TRUE; } void __declspec(dllexport) show(HWND hwndParent, int string_size, char *variables, stack_t **stacktop) { g_hwndParent=hwndParent; g_stringsize=string_size; g_stacktop=stacktop; g_variables=variables; // do your stuff here { char fn[MAX_PATH]; char temp[64]; char *sleep=temp; popstring(fn); popstring(sleep); sleep_val=0; while (*sleep >= '0' && *sleep <= '9') { sleep_val*=10; sleep_val+=*sleep++-'0'; } if (fn[0] && sleep_val>0) { MSG msg; char classname[4]="_sp"; static WNDCLASS wc; wc.lpfnWndProc = WndProc; wc.hInstance = g_hInstance; wc.hCursor = LoadCursor(NULL,IDC_ARROW); wc.lpszClassName = classname; if (RegisterClass(&wc)) { char fn2[MAX_PATH]; lstrcpy(fn2,fn); lstrcat(fn,".bmp"); lstrcat(fn2,".wav"); g_hbm=LoadImage(NULL,fn,IMAGE_BITMAP,0,0,LR_CREATEDIBSECTION|LR_LOADFROMFILE); if (g_hbm) { BOOL s=0; HANDLE f=CreateFile(fn2,0,0,NULL,OPEN_EXISTING,0,NULL); if (f != INVALID_HANDLE_VALUE) { CloseHandle(f); s=PlaySound(fn2,NULL,SND_ASYNC|SND_FILENAME); } CreateWindowEx(WS_EX_TOOLWINDOW,classname,classname, 0,0,0,0,0,(HWND)hwndParent,NULL,g_hInstance,NULL); while (GetMessage(&msg,NULL,0,0)) { DispatchMessage(&msg); } if (s) PlaySound(NULL,0,0); DeleteObject(g_hbm); wsprintf(temp,"%d",g_rv); pushstring(temp); } } } } } // utility functions (not required but often useful) int popstring(char *str) { stack_t *th; if (!g_stacktop || !*g_stacktop) return 1; th=(*g_stacktop); lstrcpy(str,th->text); *g_stacktop = th->next; GlobalFree((HGLOBAL)th); return 0; } void pushstring(char *str) { stack_t *th; if (!g_stacktop) return; th=(stack_t*)GlobalAlloc(GPTR,sizeof(stack_t)+g_stringsize); lstrcpyn(th->text,str,g_stringsize); th->next=*g_stacktop; *g_stacktop=th; } char *getuservariable(int varnum) { if (varnum < 0 || varnum >= __INST_LAST) return NULL; return g_variables+varnum*g_stringsize; }