Added version information.
[wine] / dlls / user / user_main.c
1 /*
2  * USER initialization code
3  *
4  * Copyright 2000 Alexandre Julliard
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  */
20
21 #include <string.h>
22 #include "windef.h"
23 #include "winbase.h"
24 #include "wingdi.h"
25 #include "winuser.h"
26 #include "winreg.h"
27 #include "wine/winbase16.h"
28 #include "wine/winuser16.h"
29
30 #include "controls.h"
31 #include "cursoricon.h"
32 #include "global.h"
33 #include "message.h"
34 #include "user.h"
35 #include "thread.h"
36 #include "win.h"
37 #include "wine/debug.h"
38
39 WINE_DEFAULT_DEBUG_CHANNEL(graphics);
40
41 USER_DRIVER USER_Driver;
42
43 WINE_LOOK TWEAK_WineLook = WIN31_LOOK;
44
45 WORD USER_HeapSel = 0;  /* USER heap selector */
46
47 extern HPALETTE (WINAPI *pfnGDISelectPalette)(HDC hdc, HPALETTE hpal, WORD bkgnd );
48 extern UINT (WINAPI *pfnGDIRealizePalette)(HDC hdc);
49
50 static HMODULE graphics_driver;
51 static DWORD exiting_thread_id;
52
53 extern void COMM_Init(void);
54 extern void WDML_NotifyThreadDetach(void);
55
56 #define GET_USER_FUNC(name) USER_Driver.p##name = (void*)GetProcAddress( graphics_driver, #name )
57
58 /* load the graphics driver */
59 static BOOL load_driver(void)
60 {
61     char buffer[MAX_PATH];
62     HKEY hkey;
63     DWORD type, count;
64
65     strcpy( buffer, "x11drv" );  /* default value */
66     if (!RegOpenKeyA( HKEY_LOCAL_MACHINE, "Software\\Wine\\Wine\\Config\\Wine", &hkey ))
67     {
68         count = sizeof(buffer);
69         RegQueryValueExA( hkey, "GraphicsDriver", 0, &type, buffer, &count );
70         RegCloseKey( hkey );
71     }
72
73     if (!(graphics_driver = LoadLibraryA( buffer )))
74     {
75         MESSAGE( "Could not load graphics driver '%s'\n", buffer );
76         return FALSE;
77     }
78
79     GET_USER_FUNC(InitKeyboard);
80     GET_USER_FUNC(VkKeyScan);
81     GET_USER_FUNC(MapVirtualKey);
82     GET_USER_FUNC(GetKeyNameText);
83     GET_USER_FUNC(ToUnicode);
84     GET_USER_FUNC(Beep);
85     GET_USER_FUNC(InitMouse);
86     GET_USER_FUNC(SetCursor);
87     GET_USER_FUNC(GetCursorPos);
88     GET_USER_FUNC(SetCursorPos);
89     GET_USER_FUNC(GetScreenSaveActive);
90     GET_USER_FUNC(SetScreenSaveActive);
91     GET_USER_FUNC(AcquireClipboard);
92     GET_USER_FUNC(ReleaseClipboard);
93     GET_USER_FUNC(SetClipboardData);
94     GET_USER_FUNC(GetClipboardData);
95     GET_USER_FUNC(IsClipboardFormatAvailable);
96     GET_USER_FUNC(RegisterClipboardFormat);
97     GET_USER_FUNC(GetClipboardFormatName);
98     GET_USER_FUNC(IsSelectionOwner);
99     GET_USER_FUNC(ResetSelectionOwner);
100     GET_USER_FUNC(ChangeDisplaySettingsExW);
101     GET_USER_FUNC(EnumDisplaySettingsExW);
102     GET_USER_FUNC(CreateWindow);
103     GET_USER_FUNC(DestroyWindow);
104     GET_USER_FUNC(GetDC);
105     GET_USER_FUNC(ForceWindowRaise);
106     GET_USER_FUNC(MsgWaitForMultipleObjectsEx);
107     GET_USER_FUNC(ReleaseDC);
108     GET_USER_FUNC(ScrollDC);
109     GET_USER_FUNC(ScrollWindowEx);
110     GET_USER_FUNC(SetFocus);
111     GET_USER_FUNC(SetParent);
112     GET_USER_FUNC(SetWindowPos);
113     GET_USER_FUNC(SetWindowRgn);
114     GET_USER_FUNC(SetWindowIcon);
115     GET_USER_FUNC(SetWindowStyle);
116     GET_USER_FUNC(SetWindowText);
117     GET_USER_FUNC(ShowWindow);
118     GET_USER_FUNC(SysCommandSizeMove);
119
120     return TRUE;
121 }
122
123
124 /***********************************************************************
125  *           controls_init
126  *
127  * Register the classes for the builtin controls
128  */
129 static void controls_init(void)
130 {
131     extern const struct builtin_class_descr BUTTON_builtin_class;
132     extern const struct builtin_class_descr COMBO_builtin_class;
133     extern const struct builtin_class_descr COMBOLBOX_builtin_class;
134     extern const struct builtin_class_descr DIALOG_builtin_class;
135     extern const struct builtin_class_descr DESKTOP_builtin_class;
136     extern const struct builtin_class_descr EDIT_builtin_class;
137     extern const struct builtin_class_descr ICONTITLE_builtin_class;
138     extern const struct builtin_class_descr LISTBOX_builtin_class;
139     extern const struct builtin_class_descr MDICLIENT_builtin_class;
140     extern const struct builtin_class_descr MENU_builtin_class;
141     extern const struct builtin_class_descr SCROLL_builtin_class;
142     extern const struct builtin_class_descr STATIC_builtin_class;
143
144     CLASS_RegisterBuiltinClass( &BUTTON_builtin_class );
145     CLASS_RegisterBuiltinClass( &COMBO_builtin_class );
146     CLASS_RegisterBuiltinClass( &COMBOLBOX_builtin_class );
147     CLASS_RegisterBuiltinClass( &DIALOG_builtin_class );
148     CLASS_RegisterBuiltinClass( &DESKTOP_builtin_class );
149     CLASS_RegisterBuiltinClass( &EDIT_builtin_class );
150     CLASS_RegisterBuiltinClass( &ICONTITLE_builtin_class );
151     CLASS_RegisterBuiltinClass( &LISTBOX_builtin_class );
152     CLASS_RegisterBuiltinClass( &MDICLIENT_builtin_class );
153     CLASS_RegisterBuiltinClass( &MENU_builtin_class );
154     CLASS_RegisterBuiltinClass( &SCROLL_builtin_class );
155     CLASS_RegisterBuiltinClass( &STATIC_builtin_class );
156 }
157
158
159 /***********************************************************************
160  *           palette_init
161  *
162  * Patch the function pointers in GDI for SelectPalette and RealizePalette
163  */
164 static void palette_init(void)
165 {
166     void **ptr;
167     HMODULE module = GetModuleHandleA( "gdi32" );
168     if (!module)
169     {
170         ERR( "cannot get GDI32 handle\n" );
171         return;
172     }
173     if ((ptr = (void**)GetProcAddress( module, "pfnSelectPalette" )))
174         pfnGDISelectPalette = InterlockedExchangePointer( ptr, SelectPalette );
175     else ERR( "cannot find pfnSelectPalette in GDI32\n" );
176     if ((ptr = (void**)GetProcAddress( module, "pfnRealizePalette" )))
177         pfnGDIRealizePalette = InterlockedExchangePointer( ptr, UserRealizePalette );
178     else ERR( "cannot find pfnRealizePalette in GDI32\n" );
179 }
180
181
182 /***********************************************************************
183  *           tweak_init
184  */
185 static void tweak_init(void)
186 {
187     static const char *OS = "Win3.1";
188     char buffer[80];
189     HKEY hkey;
190     DWORD type, count = sizeof(buffer);
191
192     if (RegOpenKeyA( HKEY_LOCAL_MACHINE, "Software\\Wine\\Wine\\Config\\Tweak.Layout", &hkey ))
193         return;
194     if (RegQueryValueExA( hkey, "WineLook", 0, &type, buffer, &count ))
195         strcpy( buffer, "Win31" );  /* default value */
196     RegCloseKey( hkey );
197
198     /* WIN31_LOOK is default */
199     if (!strncasecmp( buffer, "Win95", 5 ))
200     {
201         TWEAK_WineLook = WIN95_LOOK;
202         OS = "Win95";
203     }
204     else if (!strncasecmp( buffer, "Win98", 5 ))
205     {
206         TWEAK_WineLook = WIN98_LOOK;
207         OS = "Win98";
208     }
209     TRACE("Using %s look and feel.\n", OS);
210 }
211
212
213 /***********************************************************************
214  *           USER initialisation routine
215  */
216 static BOOL process_attach(void)
217 {
218     HINSTANCE16 instance;
219
220     /* Create USER heap */
221     if ((instance = LoadLibrary16( "USER.EXE" )) >= 32) USER_HeapSel = instance | 7;
222     else
223     {
224         USER_HeapSel = GlobalAlloc16( GMEM_FIXED, 65536 );
225         LocalInit16( USER_HeapSel, 32, 65534 );
226     }
227
228     /* Load the graphics driver */
229     tweak_init();
230     if (!load_driver()) return FALSE;
231
232     /* Initialize system colors and metrics */
233     SYSMETRICS_Init();
234     SYSCOLOR_Init();
235
236     /* Setup palette function pointers */
237     palette_init();
238
239     /* Initialize window procedures */
240     if (!WINPROC_Init()) return FALSE;
241
242     /* Initialize built-in window classes */
243     controls_init();
244
245     /* Initialize dialog manager */
246     if (!DIALOG_Init()) return FALSE;
247
248     /* Initialize menus */
249     if (!MENU_Init()) return FALSE;
250
251     /* Initialize message spying */
252     if (!SPY_Init()) return FALSE;
253
254     /* Create message queue of initial thread */
255     InitThreadInput16( 0, 0 );
256
257     /* Create desktop window */
258     if (!WIN_CreateDesktopWindow()) return FALSE;
259
260     /* Initialize keyboard driver */
261     if (USER_Driver.pInitKeyboard) USER_Driver.pInitKeyboard( InputKeyStateTable );
262
263     /* Initialize mouse driver */
264     if (USER_Driver.pInitMouse) USER_Driver.pInitMouse( InputKeyStateTable );
265
266     /* Initialize 16-bit serial communications */
267     COMM_Init();
268
269     return TRUE;
270 }
271
272
273 /**********************************************************************
274  *           USER_IsExitingThread
275  */
276 BOOL USER_IsExitingThread( DWORD tid )
277 {
278     return (tid == exiting_thread_id);
279 }
280
281
282 /**********************************************************************
283  *           thread_detach
284  */
285 static void thread_detach(void)
286 {
287     exiting_thread_id = GetCurrentThreadId();
288
289     WDML_NotifyThreadDetach();
290
291     TIMER_RemoveThreadTimers();
292     WIN_DestroyThreadWindows( GetDesktopWindow() );
293     QUEUE_DeleteMsgQueue();
294
295     if (!(NtCurrentTeb()->tibflags & TEBF_WIN32))
296     {
297         HMODULE16 hModule = GetExePtr( MapHModuleLS(0) );
298
299         /* FIXME: maybe destroy menus (Windows only complains about them
300          * but does nothing);
301          */
302         if (GetModuleUsage16( hModule ) <= 1)
303         {
304             /* ModuleUnload() in "Internals" */
305             /* HOOK_FreeModuleHooks( hModule ); */
306             CLASS_FreeModuleClasses( hModule );
307             CURSORICON_FreeModuleIcons( hModule );
308         }
309     }
310     exiting_thread_id = 0;
311 }
312
313
314 /***********************************************************************
315  *           UserClientDllInitialize  (USER32.@)
316  *
317  * USER dll initialisation routine
318  */
319 BOOL WINAPI UserClientDllInitialize( HINSTANCE inst, DWORD reason, LPVOID reserved )
320 {
321     BOOL ret = TRUE;
322     switch(reason)
323     {
324     case DLL_PROCESS_ATTACH:
325         ret = process_attach();
326         break;
327     case DLL_THREAD_DETACH:
328         thread_detach();
329         break;
330     }
331     return ret;
332 }