reactos操作系统实现(162)

前端之家收集整理的这篇文章主要介绍了reactos操作系统实现(162)前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

co_IntCreateWindowEx函数主要用创建一个显示的窗口,具体实现代码如下:

#001 HWND APIENTRY

#002 co_IntCreateWindowEx(DWORD dwExStyle,

#003 PUNICODE_STRING ClassName,

#004 PUNICODE_STRING WindowName,

#005 DWORD dwStyle,

#006 LONG x,

#007 LONG y,

#008 LONG nWidth,

#009 LONG nHeight,

#010 HWND hWndParent,

#011 HMENU hMenu,

#012 HINSTANCE hInstance,

#013 LPVOID lpParam,

#014 DWORD dwShowMode,

#015 BOOL bUnicodeWindow)

#016 {

#017 PWINSTATION_OBJECT WinSta;

#018 PWINDOW Wnd = NULL;

#019 PWINDOWCLASS *ClassLink,Class = NULL;

#020 RTL_ATOM ClassAtom;

#021 PWINDOW_OBJECT Window = NULL;

#022 PWINDOW_OBJECT ParentWindow = NULL,OwnerWindow;

#023 HWND ParentWindowHandle;

#024 HWND OwnerWindowHandle;

#025 PMENU_OBJECT SystemMenu;

#026 HWND hWnd;

#027 POINT Pos;

#028 SIZE Size;

#029 PW32THREADINFO ti = NULL;

#030 #if 0

#031

#032 POINT MaxSize,MaxPos,MinTrack,MaxTrack;

#033 #else

#034

#035 POINT MaxPos;

#036 #endif

#037 CREATESTRUCTW Cs;

#038 CBT_CREATEWNDW CbtCreate;

#039 LRESULT Result;

#040 BOOL MenuChanged;

#041 DECLARE_RETURN(HWND);

#042 BOOL HasOwner;

#043 USER_REFERENCE_ENTRY ParentRef,Ref;

#044 PTHREADINFO pti;

#045

获取当前运行的线程。

#046 pti = PsGetCurrentThreadWin32Thread();

获取桌面窗口作为父窗口。

#047 ParentWindowHandle = pti->Desktop->DesktopWindow;

#048 OwnerWindowHandle = NULL;

#049

是否为创建消息窗口。

#050 if (hWndParent == HWND_MESSAGE)

#051 {

#052 /*

#053 * native ole32.OleInitialize uses HWND_MESSAGE to create the

#054 * message window (style: WS_POPUP|WS_DISABLED)

#055 */

#056 DPRINT1("FIXME - Parent is HWND_MESSAGE/n");

#057 // ParentWindowHandle set already.

#058 }

是否已经父窗口。

#059 else if (hWndParent)

#060 {

#061 if ((dwStyle & (WS_CHILD | WS_POPUP)) != WS_CHILD)

#062 { //temp hack

#063 PWINDOW_OBJECT Par = UserGetWindowObject(hWndParent),Root;

#064 if (Par && (Root = UserGetAncestor(Par,GA_ROOT)))

#065 OwnerWindowHandle = Root->hSelf;

#066 }

#067 else

#068 ParentWindowHandle = hWndParent;

#069 }

如果又设置为子窗口属性,但又没有父窗口,就提示出错返回。

#070 else if ((dwStyle & (WS_CHILD | WS_POPUP)) == WS_CHILD)

#071 {

#072 SetLastWin32Error(ERROR_TLW_WITH_WSCHILD);

#073 RETURN( (HWND)0); /* WS_CHILD needs a parent,but WS_POPUP doesn't */

#074 }

#075

获取窗口对象

#076 // if (NULL != ParentWindowHandle)

#077 // {

#078 ParentWindow = UserGetWindowObject(ParentWindowHandle);

#079

#080 if (ParentWindow) UserRefObjectCo(ParentWindow,&ParentRef);

#081 // }

#082 // else

#083 // {

#084 // ParentWindow = NULL;

#085 // }

#086

#087 /* FIXME: parent must belong to the current process */

#088

检查WINDOWS的线程信息。

#089 /* Check the window station. */

#090 ti = GetW32ThreadInfo();

#091 if (ti == NULL || pti->Desktop == NULL)

#092 {

#093 DPRINT1("Thread is not attached to a desktop! Cannot create window!/n");

#094 RETURN( (HWND)0);

#095 }

#096

#097 /* Check the class. */

#098

检查要创建的窗口类是否已经注册好。

#099 ClassAtom = IntGetClassAtom(ClassName,

#100 hInstance,

#101 ti->kpi,

#102 &Class,

#103 &ClassLink);

#104

#105 if (ClassAtom == (RTL_ATOM)0)

#106 {

#107 if (IS_ATOM(ClassName->Buffer))

#108 {

#109 DPRINT1("Class 0x%p not found/n",(DWORD_PTR) ClassName->Buffer);

#110 }

#111 else

#112 {

#113 DPRINT1("Class /"%wZ/" not found/n",ClassName);

#114 }

#115

#116 SetLastWin32Error(ERROR_CANNOT_FIND_WND_CLASS);

#117 RETURN((HWND)0);

#118 }

#119

增加已经注册好的窗口类引用计数。

#120 Class = IntReferenceClass(Class,

#121 ClassLink,

#122 pti->Desktop);

#123 if (Class == NULL)

#124 {

#125 DPRINT1("Failed to reference window class!/n");

#126 RETURN(NULL);

#127 }

#128

获取桌面工作站。

#129 WinSta = pti->Desktop->WindowStation;

#130

#131 //FIXME: Reference thread/desktop instead

#132 ObReferenceObjectByPointer(WinSta,KernelMode,ExWindowStationObjectType,0);

#133

调用函数UserCreateObject创建窗口对象

#134 /* Create the window object. */

#135 Window = (PWINDOW_OBJECT)

#136 UserCreateObject(gHandleTable,(PHANDLE)&hWnd,

#137 otWindow,sizeof(WINDOW_OBJECT));

#138 if (Window)

#139 {

分配窗口所占用的堆空间。

#140 Window->Wnd = DesktopHeapAlloc(pti->Desktop,

#141 sizeof(WINDOW) + Class->WndExtra);

#142 if (!Window->Wnd)

#143 goto AllocErr;

#144 RtlZeroMemory(Window->Wnd,

#145 sizeof(WINDOW) + Class->WndExtra);

#146 Window->Wnd->hdr.Handle = hWnd; /* FIXME: Remove hack */

#147 Wnd = Window->Wnd;

#148

#149 Wnd->ti = ti;

#150 Wnd->pi = ti->kpi;

#151 Wnd->pdesktop = pti->Desktop;

#152 Wnd->hWndLastActive = hWnd;

#153 }

#154

检查是否创建窗口对象成功。

#155 DPRINT("Created object with handle %X/n",hWnd);

#156 if (!Window)

#157 {

#158 AllocErr:

#159 ObDereferenceObject(WinSta);

#160 SetLastNtError(STATUS_INSUFFICIENT_RESOURCES);

#161 RETURN( (HWND)0);

#162 }

#163

增加引用窗口计数。

#164 UserRefObjectCo(Window,&Ref);

#165

#166 ObDereferenceObject(WinSta);

#167

如果还没有桌面窗口,说明当前创建的窗口就是桌面窗口。

#168 if (NULL == pti->Desktop->DesktopWindow)

#169 {

#170 /* If there is no desktop window yet,we must be creating it */

#171 pti->Desktop->DesktopWindow = hWnd;

#172 pti->Desktop->DesktopInfo->Wnd = Wnd;

#173 }

#174

填写窗口描述结构。

#175 /*

#176 * Fill out the structure describing it.

#177 */

#178 Window->ti = ti;

#179 Wnd->Class = Class;

#180 Class = NULL;

#181

#182 Window->SystemMenu = (HMENU)0;

#183 Wnd->ContextHelpId = 0;

#184 Wnd->IDMenu = 0;

#185 Wnd->Instance = hInstance;

#186 Window->hSelf = hWnd;

#187

设置窗口消息队列。

#188 Window->MessageQueue = pti->MessageQueue;

#189 IntReferenceMessageQueue(Window->MessageQueue);

#190 Window->Parent = ParentWindow;

#191 Wnd->Parent = ParentWindow ? ParentWindow->Wnd : NULL;

#192 if (Wnd->Parent != NULL && hWndParent != 0)

#193 {

#194 Wnd->HideFocus = Wnd->Parent->HideFocus;

#195 Wnd->HideAccel = Wnd->Parent->HideAccel;

#196 }

#197

设置属于那一个窗口。

#198 if((OwnerWindow = UserGetWindowObject(OwnerWindowHandle)))

#199 {

#200 Window->hOwner = OwnerWindowHandle;

#201 Wnd->Owner = OwnerWindow->Wnd;

#202 HasOwner = TRUE;

#203 }

#204 else

#205 {

#206 Window->hOwner = NULL;

#207 Wnd->Owner = NULL;

#208 HasOwner = FALSE;

#209 }

#210

窗口用户数据设置为0.

#211 Wnd->UserData = 0;

#212

#213 Wnd->IsSystem = Wnd->Class->System;

#214

#215 /* BugBoy Comments: Comment below say that System classes are always created as UNICODE.

#216 In windows,creating a window with the ANSI version of CreateWindow sets the window

#217 to ansi as verified by testing with IsUnicodeWindow API.

#218

#219 No where can I see in code or through testing does the window change back to ANSI

#220 after being created as UNICODE in ROS. I didnt do more testing to see what problems this would cause.*/

#221 // See NtUserDefSetText! We convert to Unicode all the time and never use Mix. (jt)

根据是否系统里定义的窗口类来选择是否使用UNICODE的窗口。

#222 if (Wnd->Class->System)

#223 {

#224 /* NOTE: Always create a unicode window for system classes! */

#225 Wnd->Unicode = TRUE;

#226 Wnd->WndProc = Wnd->Class->WndProc;

#227 Wnd->WndProcExtra = Wnd->Class->WndProcExtra;

#228 }

#229 else

#230 {

#231 Wnd->Unicode = Wnd->Class->Unicode;

#232 Wnd->WndProc = Wnd->Class->WndProc;

#233 Wnd->CallProc = NULL;

#234 }

#235

设置当前窗口的线程。

#236 Window->OwnerThread = PsGetCurrentThread();

#237 Window->FirstChild = NULL;

#238 Window->LastChild = NULL;

#239 Window->PrevSibling = NULL;

#240 Window->NextSibling = NULL;

#241 Wnd->ExtraDataSize = Wnd->Class->WndExtra;

#242

#243 InitializeListHead(&Wnd->PropListHead);

#244 InitializeListHead(&Window->WndObjListHead);

#245

设置窗口的名称

#246 if (NULL != WindowName->Buffer && WindowName->Length > 0)

#247 {

#248 Wnd->WindowName.Buffer = DesktopHeapAlloc(Wnd->pdesktop,

#249 WindowName->Length + sizeof(UNICODE_NULL));

#250 if (Wnd->WindowName.Buffer == NULL)

#251 {

#252 SetLastNtError(STATUS_INSUFFICIENT_RESOURCES);

#253 RETURN( (HWND)0);

#254 }

#255

#256 Wnd->WindowName.Buffer[WindowName->Length / sizeof(WCHAR)] = L'/0';

#257 _SEH2_TRY

#258 {

#259 RtlCopyMemory(Wnd->WindowName.Buffer,

#260 WindowName->Buffer,

#261 WindowName->Length);

#262 Wnd->WindowName.Length = WindowName->Length;

#263 }

#264 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)

#265 {

#266 WindowName->Length = 0;

#267 Wnd->WindowName.Buffer[0] = L'/0';

#268 }

#269 _SEH2_END;

#270 }

#271

检查是否创建对话框窗口。

#272 /*

#273 * This has been tested for WS_CHILD | WS_VISIBLE. It has not been

#274 * tested for WS_POPUP

#275 */

#276 if ((dwExStyle & WS_EX_DLGMODALFRAME) ||

#277 ((!(dwExStyle & WS_EX_STATICEDGE)) &&

#278 (dwStyle & (WS_DLGFRAME | WS_THICKFRAME))))

#279 dwExStyle |= WS_EX_WINDOWEDGE;

#280 else

#281 dwExStyle &= ~WS_EX_WINDOWEDGE;

#282

检查不是子窗口的属性是否对。

#283 /* Correct the window style. */

#284 if (!(dwStyle & WS_CHILD))

#285 {

#286 dwStyle |= WS_CLIPSIBLINGS;

#287 DPRINT("3: Style is now %lx/n",dwStyle);

#288 if (!(dwStyle & WS_POPUP))

#289 {

#290 dwStyle |= WS_CAPTION;

#291 Window->Flags |= WINDOWOBJECT_NEED_SIZE;

#292 DPRINT("4: Style is now %lx/n",dwStyle);

#293 }

#294 }

#295

创建窗口系统菜单

#296 /* create system menu */

#297 if((dwStyle & WS_SYSMENU) )//&& (dwStyle & WS_CAPTION) == WS_CAPTION)

#298 {

#299 SystemMenu = IntGetSystemMenu(Window,TRUE,TRUE);

#300 if(SystemMenu)

#301 {

#302 Window->SystemMenu = SystemMenu->MenuInfo.Self;

#303 IntReleaseMenuObject(SystemMenu);

#304 }

#305 }

#306

设置窗口菜单

#307 /* Set the window menu */

#308 if ((dwStyle & (WS_CHILD | WS_POPUP)) != WS_CHILD)

#309 {

#310 if (hMenu)

#311 IntSetMenu(Window,hMenu,&MenuChanged);

#312 else

#313 {

#314 hMenu = Wnd->Class->hMenu;

#315 if (hMenu) IntSetMenu(Window,&MenuChanged);

#316 }

#317 }

#318 else

#319 Wnd->IDMenu = (UINT) hMenu;

#320

把窗口添加到线程队列里。

#321 /* Insert the window into the thread's window list. */

#322 InsertTailList (&pti->WindowListHead,&Window->ThreadListEntry);

#323

为窗口分配设备DC,可以分配窗口类的DC,也可以是窗口的DC。这里调用驱动程序来创建一个设备。

#324 /* Handle "CS_CLASSDC",it is tested first. */

#325 if ((Wnd->Class->Style & CS_CLASSDC) && !(Wnd->Class->Dce)) // One DCE per class to have CLASS.

#326 Wnd->Class->Dce = DceAllocDCE(Window,DCE_CLASS_DC);

#327 /* Allocate a DCE for this window. */

#328 else if ( Wnd->Class->Style & CS_OWNDC)

#329 Window->Dce = DceAllocDCE(Window,DCE_WINDOW_DC);

#330

设置窗口的位置和大小。

#331 Pos.x = x;

#332 Pos.y = y;

#333 Size.cx = nWidth;

#334 Size.cy = nHeight;

#335

#336 Wnd->ExStyle = dwExStyle;

#337 Wnd->Style = dwStyle & ~WS_VISIBLE;

#338

设置窗口HOOK

#339 /* call hook */

#340 Cs.lpCreateParams = lpParam;

#341 Cs.hInstance = hInstance;

#342 Cs.hMenu = hMenu;

#343 Cs.hwndParent = hWndParent; //Pass the original Parent handle!

#344 Cs.cx = Size.cx;

#345 Cs.cy = Size.cy;

#346 Cs.x = Pos.x;

#347 Cs.y = Pos.y;

#348 Cs.style = Wnd->Style;

#349 Cs.lpszName = (LPCWSTR) WindowName;

#350 Cs.lpszClass = (LPCWSTR) ClassName;

#351 Cs.dwExStyle = dwExStyle;

#352 CbtCreate.lpcs = &Cs;

#353 CbtCreate.hwndInsertAfter = HWND_TOP;

#354 if (co_HOOK_CallHooks(WH_CBT,HCBT_CREATEWND,(WPARAM) hWnd,(LPARAM) &CbtCreate))

#355 {

#356 /* FIXME - Delete window object and remove it from the thread windows list */

#357 /* FIXME - delete allocated DCE */

#358 DPRINT1("CBT-hook returned !0/n");

#359 RETURN( (HWND) NULL);

#360 }

#361

#362 x = Cs.x;

#363 y = Cs.y;

#364 nWidth = Cs.cx;

#365 nHeight = Cs.cy;

#366

计算重叠窗口位置和大小。

#367 /* default positioning for overlapped windows */

#368 if(!(Wnd->Style & (WS_POPUP | WS_CHILD)))

#369 {

#370 RECT rc,WorkArea;

#371 PRTL_USER_PROCESS_PARAMETERS ProcessParams;

#372 BOOL CalculatedDefPosSize = FALSE;

#373

#374 IntGetDesktopWorkArea(((PTHREADINFO)Window->OwnerThread->Tcb.Win32Thread)->Desktop,&WorkArea);

#375

#376 rc = WorkArea;

#377 ProcessParams = PsGetCurrentProcess()->Peb->ProcessParameters;

#378

#379 if(x == CW_USEDEFAULT || x == CW_USEDEFAULT16)

#380 {

#381 CalculatedDefPosSize = IntCalcDefPosSize(ParentWindow,Window,&rc,TRUE);

#382

#383 if(ProcessParams->WindowFlags & STARTF_USEPOSITION)

#384 {

#385 ProcessParams->WindowFlags &= ~STARTF_USEPOSITION;

#386 Pos.x = WorkArea.left + ProcessParams->StartingX;

#387 Pos.y = WorkArea.top + ProcessParams->StartingY;

#388 }

#389 else

#390 {

#391 Pos.x = rc.left;

#392 Pos.y = rc.top;

#393 }

#394

#395 /*

#396 According to wine,the ShowMode is set to y if x == CW_USEDEFAULT(16) and

#397 y is something else. and Quote!

#398 */

#399

#400 /* Never believe Microsoft's documentation... CreateWindowEx doc says

#401 * that if an overlapped window is created with WS_VISIBLE style bit

#402 * set and the x parameter is set to CW_USEDEFAULT,the system ignores

#403 * the y parameter. However,disassembling NT implementation (WIN32K.SYS)

#404 * reveals that

#405 *

#406 * 1) not only it checks for CW_USEDEFAULT but also for CW_USEDEFAULT16

#407 * 2) it does not ignore the y parameter as the docs claim; instead,it

#408 * uses it as second parameter to ShowWindow() unless y is either

#409 * CW_USEDEFAULT or CW_USEDEFAULT16.

#410 *

#411 * The fact that we didn't do 2) caused bogus windows pop up when wine

#412 * was running apps that were using this obscure feature. Example -

#413 * calc.exe that comes with Win98 (only Win98,it's different from

#414 * the one that comes with Win95 and NT)

#415 */

#416 if(y != CW_USEDEFAULT && y != CW_USEDEFAULT16)

#417 {

#418 dwShowMode = y;

#419 }

#420 }

#421 if(nWidth == CW_USEDEFAULT || nWidth == CW_USEDEFAULT16)

#422 {

#423 if(!CalculatedDefPosSize)

#424 {

#425 IntCalcDefPosSize(ParentWindow,FALSE);

#426 }

#427 if(ProcessParams->WindowFlags & STARTF_USESIZE)

#428 {

#429 ProcessParams->WindowFlags &= ~STARTF_USESIZE;

#430 Size.cx = ProcessParams->CountX;

#431 Size.cy = ProcessParams->CountY;

#432 }

#433 else

#434 {

#435 Size.cx = rc.right - rc.left;

#436 Size.cy = rc.bottom - rc.top;

#437 }

#438

#439 /* move the window if necessary */

#440 if(Pos.x > rc.left)

#441 Pos.x = max(rc.left,0);

#442 if(Pos.y > rc.top)

#443 Pos.y = max(rc.top,0);

#444 }

#445 }

#446 else

#447 {

#448 /* if CW_USEDEFAULT(16) is set for non-overlapped windows,both values are set to zero) */

#449 if(x == CW_USEDEFAULT || x == CW_USEDEFAULT16)

#450 {

#451 Pos.x = 0;

#452 Pos.y = 0;

#453 }

#454 if(nWidth == CW_USEDEFAULT || nWidth == CW_USEDEFAULT16)

#455 {

#456 Size.cx = 0;

#457 Size.cy = 0;

#458 }

#459 }

#460

初始化窗口显示的大小。

#461 /* Initialize the window dimensions. */

#462 Wnd->WindowRect.left = Pos.x;

#463 Wnd->WindowRect.top = Pos.y;

#464 Wnd->WindowRect.right = Pos.x + Size.cx;

#465 Wnd->WindowRect.bottom = Pos.y + Size.cy;

#466 if (0 != (Wnd->Style & WS_CHILD) && ParentWindow)

#467 {

#468 IntGdioffsetRect(&(Wnd->WindowRect),ParentWindow->Wnd->ClientRect.left,

#469 ParentWindow->Wnd->ClientRect.top);

#470 }

#471 Wnd->ClientRect = Wnd->WindowRect;

#472

#473 /*

#474 * Get the size and position of the window.

#475 */

#476 if ((dwStyle & WS_THICKFRAME) || !(dwStyle & (WS_POPUP | WS_CHILD)))

#477 {

#478 POINT MaxSize,MaxTrack;

#479

#480 /* WinPosGetMinMaxInfo sends the WM_GETMINMAXINFO message */

#481 co_WinPosGetMinMaxInfo(Window,&MaxSize,&MaxPos,&MinTrack,

#482 &MaxTrack);

#483 if (MaxSize.x < Size.cx)

#484 Size.cx = MaxSize.x;

#485 if (MaxSize.y < Size.cy)

#486 Size.cy = MaxSize.y;

#487 if (Size.cx < MinTrack.x )

#488 Size.cx = MinTrack.x;

#489 if (Size.cy < MinTrack.y )

#490 Size.cy = MinTrack.y;

#491 if (Size.cx < 0)

#492 Size.cx = 0;

#493 if (Size.cy < 0)

#494 Size.cy = 0;

#495 }

#496

#497 Wnd->WindowRect.left = Pos.x;

#498 Wnd->WindowRect.top = Pos.y;

#499 Wnd->WindowRect.right = Pos.x + Size.cx;

#500 Wnd->WindowRect.bottom = Pos.y + Size.cy;

#501 if (0 != (Wnd->Style & WS_CHILD) && ParentWindow)

#502 {

#503 IntGdioffsetRect(&(Wnd->WindowRect),

#504 ParentWindow->Wnd->ClientRect.top);

#505 }

窗口客户区的大小。

#506 Wnd->ClientRect = Wnd->WindowRect;

#507

#508 /* FIXME: Initialize the window menu. */

#509

#510 /* Send a NCCREATE message. */

#511 Cs.cx = Size.cx;

#512 Cs.cy = Size.cy;

#513 Cs.x = Pos.x;

#514 Cs.y = Pos.y;

#515

#516 DPRINT("[win32k.window] IntCreateWindowEx style %d,exstyle %d,parent %d/n",Cs.style,Cs.dwExStyle,Cs.hwndParent);

#517 DPRINT("IntCreateWindowEx(): (%d,%d-%d,%d)/n",x,y,Size.cx,Size.cy);

#518 DPRINT("IntCreateWindowEx(): About to send NCCREATE message./n");

发送创建窗口的WM_NCCREATE消息。

#519 Result = co_IntSendMessage(Window->hSelf,WM_NCCREATE,(LPARAM) &Cs);

#520 if (!Result)

#521 {

#522 /* FIXME: Cleanup. */

#523 DPRINT1("IntCreateWindowEx(): NCCREATE message Failed. No cleanup performed!/n");

#524 RETURN((HWND)0);

#525 }

#526

计算非窗户客户端的大小。

#527 /* Calculate the non-client size. */

#528 MaxPos.x = Window->Wnd->WindowRect.left;

#529 MaxPos.y = Window->Wnd->WindowRect.top;

#530

#531

#532 DPRINT("IntCreateWindowEx(): About to get non-client size./n");

#533 /* WinPosGetNonClientSize SENDS THE WM_NCCALCSIZE message */

#534 Result = co_WinPosGetNonClientSize(Window,

#535 &Window->Wnd->WindowRect,

#536 &Window->Wnd->ClientRect);

#537

#538 IntGdioffsetRect(&Window->Wnd->WindowRect,

#539 MaxPos.x - Window->Wnd->WindowRect.left,

#540 MaxPos.y - Window->Wnd->WindowRect.top);

#541

#542

如果父窗口不为空,说明本窗口是子窗口。

#543 if (NULL != ParentWindow)

#544 {

把本窗口连接到父窗口列表里。

#545 /* link the window into the parent's child list */

#546 if ((dwStyle & (WS_CHILD|WS_MAXIMIZE)) == WS_CHILD)

#547 {

#548 PWINDOW_OBJECT PrevSibling;

#549

#550 PrevSibling = ParentWindow->LastChild;

#551

#552 /* link window as bottom sibling */

#553 IntLinkWindow(Window,ParentWindow,PrevSibling /*prev sibling*/);

#554 }

#555 else

#556 {

#557 /* link window as top sibling (but after topmost siblings) */

#558 PWINDOW_OBJECT InsertAfter,Sibling;

#559 if (!(dwExStyle & WS_EX_TOPMOST))

#560 {

#561 InsertAfter = NULL;

#562 Sibling = ParentWindow->FirstChild;

#563 while (Sibling && (Sibling->Wnd->ExStyle & WS_EX_TOPMOST))

#564 {

#565 InsertAfter = Sibling;

#566 Sibling = Sibling->NextSibling;

#567 }

#568 }

#569 else

#570 {

#571 InsertAfter = NULL;

#572 }

#573

#574 IntLinkWindow(Window,InsertAfter /* prev sibling */);

#575

#576 }

#577 }

#578

发送创建窗口的WM_CREATE消息。

#579 /* Send the WM_CREATE message. */

#580 DPRINT("IntCreateWindowEx(): about to send CREATE message./n");

#581 Result = co_IntSendMessage(Window->hSelf,WM_CREATE,(LPARAM) &Cs);

#582

#583 if (Result == (LRESULT)-1)

#584 {

#585 /* FIXME: Cleanup. */

#586 DPRINT1("IntCreateWindowEx(): send CREATE message Failed. No cleanup performed!/n");

#587 IntUnlinkWindow(Window);

#588 RETURN((HWND)0);

#589 }

#590 #if 0

#591 Result = IntNotifyWinEvent(EVENT_OBJECT_CREATE,OBJID_WINDOW,0);

#592

#593 if (Result == (LRESULT)-1)

#594 {

#595 /* FIXME: Cleanup. */

#596 DPRINT1("IntCreateWindowEx(): event CREATE hook Failed. No cleanup performed!/n");

#597 IntUnlinkWindow(Window);

#598 RETURN((HWND)0);

#599 }

#600 #endif

发送窗口设置大小的WM_SIZE消息。

#601 /* Send move and size messages. */

#602 if (!(Window->Flags & WINDOWOBJECT_NEED_SIZE))

#603 {

#604 LONG lParam;

#605

#606 DPRINT("IntCreateWindow(): About to send WM_SIZE/n");

#607

#608 if ((Window->Wnd->ClientRect.right - Window->Wnd->ClientRect.left) < 0 ||

#609 (Window->Wnd->ClientRect.bottom - Window->Wnd->ClientRect.top) < 0)

#610 {

#611 DPRINT("Sending bogus WM_SIZE/n");

#612 }

#613

#614 lParam = MAKE_LONG(Window->Wnd->ClientRect.right -

#615 Window->Wnd->ClientRect.left,

#616 Window->Wnd->ClientRect.bottom -

#617 Window->Wnd->ClientRect.top);

#618 co_IntSendMessage(Window->hSelf,WM_SIZE,SIZE_RESTORED,

#619 lParam);

#620

#621 DPRINT("IntCreateWindow(): About to send WM_MOVE/n");

#622

发送窗口移动到某位置的WM_MOVE消息。

#623 if (0 != (Wnd->Style & WS_CHILD) && ParentWindow)

#624 {

#625 lParam = MAKE_LONG(Wnd->ClientRect.left - ParentWindow->Wnd->ClientRect.left,

#626 Wnd->ClientRect.top - ParentWindow->Wnd->ClientRect.top);

#627 }

#628 else

#629 {

#630 lParam = MAKE_LONG(Wnd->ClientRect.left,

#631 Wnd->ClientRect.top);

#632 }

#633

#634 co_IntSendMessage(Window->hSelf,WM_MOVE,lParam);

#635

#636 /* Call WNDOBJ change procs */

#637 IntEngWindowChanged(Window,WOC_RGN_CLIENT);

#638 }

#639

设置窗口最大化或最小化显示

#640 /* Show or maybe minimize or maximize the window. */

#641 if (Wnd->Style & (WS_MINIMIZE | WS_MAXIMIZE))

#642 {

#643 RECT NewPos;

#644 UINT16 SwFlag;

#645

#646 SwFlag = (Wnd->Style & WS_MINIMIZE) ? SW_MINIMIZE :

#647 SW_MAXIMIZE;

#648

#649 co_WinPosMinMaximize(Window,SwFlag,&NewPos);

#650

#651 SwFlag = ((Wnd->Style & WS_CHILD) || UserGetActiveWindow()) ?

#652 SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED :

#653 SWP_NOZORDER | SWP_FRAMECHANGED;

#654

#655 DPRINT("IntCreateWindow(): About to minimize/maximize/n");

#656 DPRINT("%d,%d %dx%d/n",NewPos.left,NewPos.top,NewPos.right,NewPos.bottom);

#657 co_WinPosSetWindowPos(Window,

#658 NewPos.right,NewPos.bottom,SwFlag);

#659 }

#660

如果是创建子窗口,通知父窗口已经创建了一个子窗口。

#661 /* Notify the parent window of a new child. */

#662 if ((Wnd->Style & WS_CHILD) &&

#663 (!(Wnd->ExStyle & WS_EX_NOPARENTNOTIFY)) && ParentWindow)

#664 {

#665 DPRINT("IntCreateWindow(): About to notify parent/n");

#666 co_IntSendMessage(ParentWindow->hSelf,

#667 WM_PARENTNOTIFY,

#668 MAKEWPARAM(WM_CREATE,Wnd->IDMenu),

#669 (LPARAM)Window->hSelf);

#670 }

#671

#672 if ((!hWndParent) && (!HasOwner))

#673 {

#674 DPRINT("Sending CREATED notify/n");

#675 co_IntShellHookNotify(HSHELL_WINDOWCREATED,(LPARAM)hWnd);

#676 }

#677 else

#678 {

#679 DPRINT("Not sending CREATED notify,%x %d/n",HasOwner);

#680 }

#681

设置窗口的滚动条。

#682 /* Initialize and show the window's scrollbars */

#683 if (Wnd->Style & WS_VSCROLL)

#684 {

#685 co_UserShowScrollBar(Window,SB_VERT,TRUE);

#686 }

#687 if (Wnd->Style & WS_HSCROLL)

#688 {

#689 co_UserShowScrollBar(Window,SB_HORZ,TRUE);

#690 }

#691

设置窗口显示方式。

#692 if (dwStyle & WS_VISIBLE)

#693 {

#694 if (Wnd->Style & WS_MAXIMIZE)

#695 dwShowMode = SW_SHOW;

#696 else if (Wnd->Style & WS_MINIMIZE)

#697 dwShowMode = SW_SHOWMINIMIZED;

#698

#699 DPRINT("IntCreateWindow(): About to show window/n");

#700 co_WinPosShowWindow(Window,dwShowMode);

#701

#702 if (Wnd->ExStyle & WS_EX_MDICHILD)

#703 {

#704 co_IntSendMessage(ParentWindow->hSelf,WM_MDIREFRESHMENU,0);

#705 /* ShowWindow won't activate child windows */

#706 co_WinPosSetWindowPos(Window,HWND_TOP,SWP_SHOWWINDOW | SWP_NOMOVE | SWP_NOSIZE);

#707 }

#708 }

#709

#710 /* BugBoy Comments: if the window being created is a edit control,ATOM 0xC007,

#711 then my testing shows that windows (2k and XP) creates a CallProc for it immediately

#712 Dont understand why it does this. */

#713 if (ClassAtom == 0XC007)

#714 {

#715 PCALLPROC CallProc;

#716 //CallProc = CreateCallProc(NULL,Wnd->WndProc,bUnicodeWindow,Wnd->ti->kpi);

#717 CallProc = CreateCallProc(NULL,Wnd->Unicode,Wnd->ti->kpi);

#718

#719 if (!CallProc)

#720 {

#721 SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);

#722 DPRINT1("Warning: Unable to create CallProc for edit control. Control may not operate correctly! hwnd %x/n",hWnd);

#723 }

#724 else

#725 {

#726 UserAddCallProcToClass(Wnd->Class,CallProc);

#727 Wnd->CallProc = CallProc;

#728 Wnd->IsSystem = FALSE;

#729 }

#730 }

#731

返回创建窗口的句柄。

#732 DPRINT("IntCreateWindow(): = %X/n",hWnd);

#733 DPRINT("WindowObject->SystemMenu = 0x%x/n",Window->SystemMenu);

#734 RETURN(hWnd);

#735

下面是清除失败时分配的资源。

#736 CLEANUP:

#737 if (!_ret_ && Window && Window->Wnd && ti)

#738 UserFreeWindowInfo(ti,Window);

#739 if (Window)

#740 {

#741 UserDerefObjectCo(Window);

#742 UserDereferenceObject(Window);

#743 }

#744 if (ParentWindow) UserDerefObjectCo(ParentWindow);

#745 if (!_ret_ && ti != NULL)

#746 {

#747 if (Class != NULL)

#748 {

#749 IntDereferenceClass(Class,

#750 ti->Desktop,

#751 ti->kpi);

#752 }

#753 }

#754 END_CLEANUP;

#755 }

#756

猜你在找的React相关文章