1 /************************************************************** 2 * 3 * Licensed to the Apache Software Foundation (ASF) under one 4 * or more contributor license agreements. See the NOTICE file 5 * distributed with this work for additional information 6 * regarding copyright ownership. The ASF licenses this file 7 * to you under the Apache License, Version 2.0 (the 8 * "License"); you may not use this file except in compliance 9 * with the License. You may obtain a copy of the License at 10 * 11 * http://www.apache.org/licenses/LICENSE-2.0 12 * 13 * Unless required by applicable law or agreed to in writing, 14 * software distributed under the License is distributed on an 15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16 * KIND, either express or implied. See the License for the 17 * specific language governing permissions and limitations 18 * under the License. 19 * 20 *************************************************************/ 21 22 23 24 // MARKER(update_precomp.py): autogen include statement, do not remove 25 #include "precompiled_vcl.hxx" 26 27 // i72022: ad-hoc to forcibly enable reconversion 28 #if WINVER < 0x0500 29 #undef WINVER 30 #define WINVER 0x0500 31 #endif 32 33 #include <com/sun/star/lang/XMultiServiceFactory.hpp> 34 #include <com/sun/star/container/XIndexAccess.hpp> 35 #include <com/sun/star/beans/XPropertySet.hpp> 36 #include <com/sun/star/awt/Rectangle.hpp> 37 #include <comphelper/processfactory.hxx> 38 #include <unotools/misccfg.hxx> 39 40 #include <string.h> 41 #include <limits.h> 42 43 #include <stdio.h> 44 45 #include <tools/svwin.h> 46 #ifdef __MINGW32__ 47 #include <excpt.h> 48 #endif 49 50 #include <rtl/string.h> 51 #include <rtl/ustring.h> 52 53 #include <osl/module.h> 54 55 #include <tools/debug.hxx> 56 57 #include <vcl/sysdata.hxx> 58 #include <vcl/timer.hxx> 59 #include <vcl/settings.hxx> 60 #include <vcl/keycodes.hxx> 61 #include <vcl/window.hxx> 62 #include <vcl/wrkwin.hxx> 63 #include <vcl/svapp.hxx> 64 #include <vcl/impdel.hxx> 65 66 // Warning in SDK header 67 #if defined(_MSC_VER) && (_MSC_VER > 1400) 68 #pragma warning( disable: 4242 4244 ) 69 #endif 70 #include <win/wincomp.hxx> 71 #include <win/salids.hrc> 72 #include <win/saldata.hxx> 73 #include <win/salinst.h> 74 #include <win/salbmp.h> 75 #include <win/salgdi.h> 76 #include <win/salsys.h> 77 #include <win/salframe.h> 78 #include <win/salvd.h> 79 #include <win/salmenu.h> 80 #include <win/salobj.h> 81 #include <win/saltimer.h> 82 83 #include <impbmp.hxx> 84 #include <window.h> 85 #include <sallayout.hxx> 86 87 #define COMPILE_MULTIMON_STUBS 88 #include <multimon.h> 89 #include <vector> 90 #ifdef __MINGW32__ 91 #include <algorithm> 92 using ::std::max; 93 #endif 94 95 #include <com/sun/star/uno/Exception.hdl> 96 97 #include <time.h> 98 99 using ::rtl::OUString; 100 using namespace ::com::sun::star; 101 using namespace ::com::sun::star::uno; 102 using namespace ::com::sun::star::lang; 103 using namespace ::com::sun::star::container; 104 using namespace ::com::sun::star::beans; 105 106 // The following defines are newly added in Longhorn 107 #ifndef WM_MOUSEHWHEEL 108 # define WM_MOUSEHWHEEL 0x020E 109 #endif 110 #ifndef SPI_GETWHEELSCROLLCHARS 111 # define SPI_GETWHEELSCROLLCHARS 0x006C 112 #endif 113 #ifndef SPI_SETWHEELSCROLLCHARS 114 # define SPI_SETWHEELSCROLLCHARS 0x006D 115 #endif 116 117 118 119 #if OSL_DEBUG_LEVEL > 1 120 void MyOutputDebugString( char *s) { OutputDebugString( s ); } 121 #endif 122 123 // misssing prototypes and constants for LayeredWindows 124 extern "C" { 125 //WINUSERAPI sal_Bool WINAPI SetLayeredWindowAttributes(HWND,COLORREF,BYTE,DWORD); 126 typedef sal_Bool ( WINAPI * SetLayeredWindowAttributes_Proc_T ) (HWND,COLORREF,BYTE,DWORD); 127 static SetLayeredWindowAttributes_Proc_T lpfnSetLayeredWindowAttributes; 128 }; 129 130 // ======================================================================= 131 132 const unsigned int WM_USER_SYSTEM_WINDOW_ACTIVATED = RegisterWindowMessageA("SYSTEM_WINDOW_ACTIVATED"); 133 134 sal_Bool WinSalFrame::mbInReparent = FALSE; 135 136 // ======================================================================= 137 138 // Wegen Fehler in Windows-Headerfiles 139 #ifndef IMN_OPENCANDIDATE 140 #define IMN_OPENCANDIDATE 0x0005 141 #endif 142 #ifndef IMN_CLOSECANDIDATE 143 #define IMN_CLOSECANDIDATE 0x0004 144 #endif 145 146 #ifndef WM_THEMECHANGED 147 #define WM_THEMECHANGED 0x031A 148 #endif 149 150 // Macros for support of WM_UNICHAR & Keyman 6.0 151 #define Uni_UTF32ToSurrogate1(ch) (((unsigned long) (ch) - 0x10000) / 0x400 + 0xD800) 152 #define Uni_UTF32ToSurrogate2(ch) (((unsigned long) (ch) - 0x10000) % 0x400 + 0xDC00) 153 #define Uni_SupplementaryPlanesStart 0x10000 154 155 // ======================================================================= 156 157 static void UpdateFrameGeometry( HWND hWnd, WinSalFrame* pFrame ); 158 static void SetMaximizedFrameGeometry( HWND hWnd, WinSalFrame* pFrame, RECT* pParentRect = NULL ); 159 160 static void ImplSaveFrameState( WinSalFrame* pFrame ) 161 { 162 // Position, Groesse und Status fuer GetWindowState() merken 163 if ( !pFrame->mbFullScreen ) 164 { 165 sal_Bool bVisible = (GetWindowStyle( pFrame->mhWnd ) & WS_VISIBLE) != 0; 166 if ( IsIconic( pFrame->mhWnd ) ) 167 { 168 pFrame->maState.mnState |= SAL_FRAMESTATE_MINIMIZED; 169 if ( bVisible ) 170 pFrame->mnShowState = SW_SHOWMAXIMIZED; 171 } 172 else if ( IsZoomed( pFrame->mhWnd ) ) 173 { 174 pFrame->maState.mnState &= ~SAL_FRAMESTATE_MINIMIZED; 175 pFrame->maState.mnState |= SAL_FRAMESTATE_MAXIMIZED; 176 if ( bVisible ) 177 pFrame->mnShowState = SW_SHOWMAXIMIZED; 178 pFrame->mbRestoreMaximize = TRUE; 179 180 WINDOWPLACEMENT aPlacement; 181 aPlacement.length = sizeof(aPlacement); 182 if( GetWindowPlacement( pFrame->mhWnd, &aPlacement ) ) 183 { 184 RECT aRect = aPlacement.rcNormalPosition; 185 RECT aRect2 = aRect; 186 AdjustWindowRectEx( &aRect2, GetWindowStyle( pFrame->mhWnd ), 187 FALSE, GetWindowExStyle( pFrame->mhWnd ) ); 188 long nTopDeco = abs( aRect.top - aRect2.top ); 189 long nLeftDeco = abs( aRect.left - aRect2.left ); 190 long nBottomDeco = abs( aRect.bottom - aRect2.bottom ); 191 long nRightDeco = abs( aRect.right - aRect2.right ); 192 193 pFrame->maState.mnX = aRect.left + nLeftDeco; 194 pFrame->maState.mnY = aRect.top + nTopDeco; 195 pFrame->maState.mnWidth = aRect.right - aRect.left - nLeftDeco - nRightDeco; 196 pFrame->maState.mnHeight = aRect.bottom - aRect.top - nTopDeco - nBottomDeco; 197 } 198 } 199 else 200 { 201 RECT aRect; 202 GetWindowRect( pFrame->mhWnd, &aRect ); 203 204 // to be consistent with Unix, the frame state is without(!) decoration 205 RECT aRect2 = aRect; 206 AdjustWindowRectEx( &aRect2, GetWindowStyle( pFrame->mhWnd ), 207 FALSE, GetWindowExStyle( pFrame->mhWnd ) ); 208 long nTopDeco = abs( aRect.top - aRect2.top ); 209 long nLeftDeco = abs( aRect.left - aRect2.left ); 210 long nBottomDeco = abs( aRect.bottom - aRect2.bottom ); 211 long nRightDeco = abs( aRect.right - aRect2.right ); 212 213 pFrame->maState.mnState &= ~(SAL_FRAMESTATE_MINIMIZED | SAL_FRAMESTATE_MAXIMIZED); 214 // subtract decoration 215 pFrame->maState.mnX = aRect.left+nLeftDeco; 216 pFrame->maState.mnY = aRect.top+nTopDeco; 217 pFrame->maState.mnWidth = aRect.right-aRect.left-nLeftDeco-nRightDeco; 218 pFrame->maState.mnHeight = aRect.bottom-aRect.top-nTopDeco-nBottomDeco; 219 if ( bVisible ) 220 pFrame->mnShowState = SW_SHOWNORMAL; 221 pFrame->mbRestoreMaximize = FALSE; 222 } 223 } 224 } 225 226 // ----------------------------------------------------------------------- 227 228 // if pParentRect is set, the workarea of the monitor that contains pParentRect is returned 229 void ImplSalGetWorkArea( HWND hWnd, RECT *pRect, const RECT *pParentRect ) 230 { 231 static int winVerChecked = 0; 232 static int winVerOk = 0; 233 234 // check if we or our parent is fullscreen, then the taskbar should be ignored 235 bool bIgnoreTaskbar = false; 236 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 237 if( pFrame ) 238 { 239 Window *pWin = pFrame->GetWindow(); 240 while( pWin ) 241 { 242 WorkWindow *pWorkWin = (pWin->GetType() == WINDOW_WORKWINDOW) ? (WorkWindow *) pWin : NULL; 243 if( pWorkWin && pWorkWin->ImplGetWindowImpl()->mbReallyVisible && pWorkWin->IsFullScreenMode() ) 244 { 245 bIgnoreTaskbar = true; 246 break; 247 } 248 else 249 pWin = pWin->ImplGetWindowImpl()->mpParent; 250 } 251 } 252 253 if( !winVerChecked ) 254 { 255 winVerChecked = 1; 256 winVerOk = 1; 257 258 // multi monitor calls not available on Win95/NT 259 if ( aSalShlData.maVersionInfo.dwPlatformId == VER_PLATFORM_WIN32_NT ) 260 { 261 if ( aSalShlData.maVersionInfo.dwMajorVersion <= 4 ) 262 winVerOk = 0; // NT 263 } 264 else if( aSalShlData.maVersionInfo.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS ) 265 { 266 if ( aSalShlData.maVersionInfo.dwMajorVersion == 4 && aSalShlData.maVersionInfo.dwMinorVersion == 0 ) 267 winVerOk = 0; // Win95 268 } 269 } 270 271 // calculates the work area taking multiple monitors into account 272 if( winVerOk ) 273 { 274 static int nMonitors = GetSystemMetrics( SM_CMONITORS ); 275 if( nMonitors == 1 ) 276 { 277 if( bIgnoreTaskbar ) 278 { 279 pRect->left = pRect->top = 0; 280 pRect->right = GetSystemMetrics( SM_CXSCREEN ); 281 pRect->bottom = GetSystemMetrics( SM_CYSCREEN ); 282 } 283 else 284 SystemParametersInfo( SPI_GETWORKAREA, 0, pRect, 0 ); 285 } 286 else 287 { 288 if( pParentRect != NULL ) 289 { 290 // return the size of the monitor where pParentRect lives 291 HMONITOR hMonitor; 292 MONITORINFO mi; 293 294 // get the nearest monitor to the passed rect. 295 hMonitor = MonitorFromRect(pParentRect, MONITOR_DEFAULTTONEAREST); 296 297 // get the work area or entire monitor rect. 298 mi.cbSize = sizeof(mi); 299 GetMonitorInfo(hMonitor, &mi); 300 if( !bIgnoreTaskbar ) 301 *pRect = mi.rcWork; 302 else 303 *pRect = mi.rcMonitor; 304 } 305 else 306 { 307 // return the union of all monitors 308 pRect->left = GetSystemMetrics( SM_XVIRTUALSCREEN ); 309 pRect->top = GetSystemMetrics( SM_YVIRTUALSCREEN ); 310 pRect->right = pRect->left + GetSystemMetrics( SM_CXVIRTUALSCREEN ); 311 pRect->bottom = pRect->top + GetSystemMetrics( SM_CYVIRTUALSCREEN ); 312 313 // virtualscreen does not take taskbar into account, so use the corresponding 314 // diffs between screen and workarea from the default screen 315 // however, this is still not perfect: the taskbar might not be on the primary screen 316 if( !bIgnoreTaskbar ) 317 { 318 RECT wRect, scrRect; 319 SystemParametersInfo( SPI_GETWORKAREA, 0, &wRect, 0 ); 320 scrRect.left = 0; 321 scrRect.top = 0; 322 scrRect.right = GetSystemMetrics( SM_CXSCREEN ); 323 scrRect.bottom = GetSystemMetrics( SM_CYSCREEN ); 324 325 pRect->left += wRect.left; 326 pRect->top += wRect.top; 327 pRect->right -= scrRect.right - wRect.right; 328 pRect->bottom -= scrRect.bottom - wRect.bottom; 329 } 330 } 331 } 332 } 333 else 334 { 335 if( bIgnoreTaskbar ) 336 { 337 pRect->left = pRect->top = 0; 338 pRect->right = GetSystemMetrics( SM_CXSCREEN ); 339 pRect->bottom = GetSystemMetrics( SM_CYSCREEN ); 340 } 341 else 342 SystemParametersInfo( SPI_GETWORKAREA, 0, pRect, 0 ); 343 } 344 } 345 346 // ======================================================================= 347 348 SalFrame* ImplSalCreateFrame( WinSalInstance* pInst, 349 HWND hWndParent, sal_uLong nSalFrameStyle ) 350 { 351 WinSalFrame* pFrame = new WinSalFrame; 352 HWND hWnd; 353 DWORD nSysStyle = 0; 354 DWORD nExSysStyle = 0; 355 sal_Bool bSubFrame = FALSE; 356 357 if( getenv( "SAL_SYNCHRONIZE" ) ) // no buffering of drawing commands 358 GdiSetBatchLimit( 1 ); 359 360 static int bLayeredAPI = -1; 361 if( bLayeredAPI == -1 ) 362 { 363 bLayeredAPI = 0; 364 // check for W2k and XP 365 if ( aSalShlData.maVersionInfo.dwPlatformId == VER_PLATFORM_WIN32_NT && aSalShlData.maVersionInfo.dwMajorVersion >= 5 ) 366 { 367 oslModule pLib = osl_loadAsciiModule( "user32", SAL_LOADMODULE_DEFAULT ); 368 oslGenericFunction pFunc = NULL; 369 if( pLib ) 370 pFunc = osl_getAsciiFunctionSymbol( pLib, "SetLayeredWindowAttributes" ); 371 372 lpfnSetLayeredWindowAttributes = ( SetLayeredWindowAttributes_Proc_T ) pFunc; 373 374 bLayeredAPI = pFunc ? 1 : 0; 375 } 376 } 377 static const char* pEnvTransparentFloats = getenv("SAL_TRANSPARENT_FLOATS" ); 378 379 // determine creation data 380 if ( nSalFrameStyle & (SAL_FRAME_STYLE_PLUG | SAL_FRAME_STYLE_SYSTEMCHILD) ) 381 { 382 nSysStyle |= WS_CHILD; 383 if( nSalFrameStyle & SAL_FRAME_STYLE_SYSTEMCHILD ) 384 nSysStyle |= WS_CLIPSIBLINGS; 385 } 386 else 387 { 388 // #i87402# commenting out WS_CLIPCHILDREN 389 // this breaks SAL_FRAME_STYLE_SYSTEMCHILD handling, which is not 390 // used currently. Probably SAL_FRAME_STYLE_SYSTEMCHILD should be 391 // removed again. 392 393 // nSysStyle |= WS_CLIPCHILDREN; 394 if ( hWndParent ) 395 { 396 nSysStyle |= WS_POPUP; 397 bSubFrame = TRUE; 398 pFrame->mbNoIcon = TRUE; 399 } 400 else 401 { 402 // Only with WS_OVRLAPPED we get a useful default position/size 403 if ( (nSalFrameStyle & (SAL_FRAME_STYLE_SIZEABLE | SAL_FRAME_STYLE_MOVEABLE)) == 404 (SAL_FRAME_STYLE_SIZEABLE | SAL_FRAME_STYLE_MOVEABLE) ) 405 nSysStyle |= WS_OVERLAPPED; 406 else 407 { 408 nSysStyle |= WS_POPUP; 409 if ( !(nSalFrameStyle & SAL_FRAME_STYLE_MOVEABLE) ) 410 nExSysStyle |= WS_EX_TOOLWINDOW; // avoid taskbar appearance, for eg splash screen 411 } 412 } 413 414 if ( nSalFrameStyle & SAL_FRAME_STYLE_MOVEABLE ) 415 { 416 pFrame->mbCaption = TRUE; 417 nSysStyle |= WS_SYSMENU | WS_CAPTION; 418 if ( !hWndParent ) 419 nSysStyle |= WS_SYSMENU | WS_MINIMIZEBOX; 420 else 421 nExSysStyle |= WS_EX_DLGMODALFRAME; 422 423 if ( nSalFrameStyle & SAL_FRAME_STYLE_SIZEABLE ) 424 { 425 pFrame->mbSizeBorder = TRUE; 426 nSysStyle |= WS_THICKFRAME; 427 if ( !hWndParent ) 428 nSysStyle |= WS_MAXIMIZEBOX; 429 } 430 else 431 pFrame->mbFixBorder = TRUE; 432 433 if ( nSalFrameStyle & SAL_FRAME_STYLE_DEFAULT ) 434 nExSysStyle |= WS_EX_APPWINDOW; 435 } 436 if( nSalFrameStyle & SAL_FRAME_STYLE_TOOLWINDOW 437 // #100656# toolwindows lead to bad alt-tab behaviour, if they have the focus 438 // you must press it twice to leave the application 439 // so toolwindows are only used for non sizeable windows 440 // which are typically small, so a small caption makes sense 441 442 // #103578# looked too bad - above changes reverted 443 /* && !(nSalFrameStyle & SAL_FRAME_STYLE_SIZEABLE) */ ) 444 { 445 pFrame->mbNoIcon = TRUE; 446 nExSysStyle |= WS_EX_TOOLWINDOW; 447 if ( pEnvTransparentFloats && bLayeredAPI == 1 /*&& !(nSalFrameStyle & SAL_FRAME_STYLE_MOVEABLE) */) 448 nExSysStyle |= WS_EX_LAYERED; 449 } 450 } 451 if ( nSalFrameStyle & SAL_FRAME_STYLE_FLOAT ) 452 { 453 nExSysStyle |= WS_EX_TOOLWINDOW; 454 pFrame->mbFloatWin = TRUE; 455 456 if ( (bLayeredAPI == 1) && (pEnvTransparentFloats /* does not work remote! || (nSalFrameStyle & SAL_FRAME_STYLE_FLOAT_FOCUSABLE) */ ) ) 457 nExSysStyle |= WS_EX_LAYERED; 458 459 } 460 if( (nSalFrameStyle & SAL_FRAME_STYLE_TOOLTIP) || (nSalFrameStyle & SAL_FRAME_STYLE_FLOAT_FOCUSABLE) ) 461 nExSysStyle |= WS_EX_TOPMOST; 462 463 // init frame data 464 pFrame->mnStyle = nSalFrameStyle; 465 466 // determine show style 467 pFrame->mnShowState = SW_SHOWNORMAL; 468 if ( (nSysStyle & (WS_POPUP | WS_MAXIMIZEBOX | WS_THICKFRAME)) == (WS_MAXIMIZEBOX | WS_THICKFRAME) ) 469 { 470 if ( GetSystemMetrics( SM_CXSCREEN ) <= 1024 ) 471 pFrame->mnShowState = SW_SHOWMAXIMIZED; 472 else 473 { 474 if ( nSalFrameStyle & SAL_FRAME_STYLE_DEFAULT ) 475 { 476 SalData* pSalData = GetSalData(); 477 pFrame->mnShowState = pSalData->mnCmdShow; 478 if ( (pFrame->mnShowState != SW_SHOWMINIMIZED) && 479 (pFrame->mnShowState != SW_MINIMIZE) && 480 (pFrame->mnShowState != SW_SHOWMINNOACTIVE) ) 481 { 482 if ( (pFrame->mnShowState == SW_SHOWMAXIMIZED) || 483 (pFrame->mnShowState == SW_MAXIMIZE) ) 484 pFrame->mbOverwriteState = FALSE; 485 pFrame->mnShowState = SW_SHOWMAXIMIZED; 486 } 487 else 488 pFrame->mbOverwriteState = FALSE; 489 } 490 else 491 { 492 // Document Windows are also maximized, if the current Document Window 493 // is also maximized 494 HWND hWnd = GetForegroundWindow(); 495 if ( hWnd && IsMaximized( hWnd ) && 496 (GetWindowInstance( hWnd ) == pInst->mhInst) && 497 ((GetWindowStyle( hWnd ) & (WS_POPUP | WS_MAXIMIZEBOX | WS_THICKFRAME)) == (WS_MAXIMIZEBOX | WS_THICKFRAME)) ) 498 pFrame->mnShowState = SW_SHOWMAXIMIZED; 499 } 500 } 501 } 502 503 // create frame 504 if( true/*aSalShlData.mbWNT*/ ) 505 { 506 LPCWSTR pClassName; 507 if ( bSubFrame ) 508 { 509 if ( nSalFrameStyle & (SAL_FRAME_STYLE_MOVEABLE|SAL_FRAME_STYLE_NOSHADOW) ) // check if shadow not wanted 510 pClassName = SAL_SUBFRAME_CLASSNAMEW; 511 else 512 pClassName = SAL_TMPSUBFRAME_CLASSNAMEW; // undecorated floaters will get shadow on XP 513 } 514 else 515 { 516 if ( nSalFrameStyle & SAL_FRAME_STYLE_MOVEABLE ) 517 pClassName = SAL_FRAME_CLASSNAMEW; 518 else 519 pClassName = SAL_TMPSUBFRAME_CLASSNAMEW; 520 } 521 hWnd = CreateWindowExW( nExSysStyle, pClassName, L"", nSysStyle, 522 CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, 523 hWndParent, 0, pInst->mhInst, (void*)pFrame ); 524 if( !hWnd ) 525 ImplWriteLastError( GetLastError(), "CreateWindowEx" ); 526 #if OSL_DEBUG_LEVEL > 1 527 // set transparency value 528 if( bLayeredAPI == 1 && GetWindowExStyle( hWnd ) & WS_EX_LAYERED ) 529 lpfnSetLayeredWindowAttributes( hWnd, 0, 230, 0x00000002 /*LWA_ALPHA*/ ); 530 #endif 531 } 532 if ( !hWnd ) 533 { 534 delete pFrame; 535 return NULL; 536 } 537 538 // If we have an Window with an Caption Bar and without 539 // an MaximizeBox, we change the SystemMenu 540 if ( (nSysStyle & (WS_CAPTION | WS_MAXIMIZEBOX)) == (WS_CAPTION) ) 541 { 542 HMENU hSysMenu = GetSystemMenu( hWnd, FALSE ); 543 if ( hSysMenu ) 544 { 545 if ( !(nSysStyle & (WS_MINIMIZEBOX | WS_MAXIMIZEBOX)) ) 546 DeleteMenu( hSysMenu, SC_RESTORE, MF_BYCOMMAND ); 547 else 548 EnableMenuItem( hSysMenu, SC_RESTORE, MF_BYCOMMAND | MF_GRAYED | MF_DISABLED ); 549 if ( !(nSysStyle & WS_MINIMIZEBOX) ) 550 DeleteMenu( hSysMenu, SC_MINIMIZE, MF_BYCOMMAND ); 551 if ( !(nSysStyle & WS_MAXIMIZEBOX) ) 552 DeleteMenu( hSysMenu, SC_MAXIMIZE, MF_BYCOMMAND ); 553 if ( !(nSysStyle & WS_THICKFRAME) ) 554 DeleteMenu( hSysMenu, SC_SIZE, MF_BYCOMMAND ); 555 } 556 } 557 if ( (nSysStyle & WS_SYSMENU) && !(nSalFrameStyle & SAL_FRAME_STYLE_CLOSEABLE) ) 558 { 559 HMENU hSysMenu = GetSystemMenu( hWnd, FALSE ); 560 if ( hSysMenu ) 561 EnableMenuItem( hSysMenu, SC_CLOSE, MF_BYCOMMAND | MF_GRAYED | MF_DISABLED ); 562 } 563 564 // reset input context 565 pFrame->mhDefIMEContext = ImmAssociateContext( hWnd, 0 ); 566 567 // determine output size and state 568 RECT aRect; 569 GetClientRect( hWnd, &aRect ); 570 pFrame->mnWidth = aRect.right; 571 pFrame->mnHeight = aRect.bottom; 572 ImplSaveFrameState( pFrame ); 573 pFrame->mbDefPos = TRUE; 574 575 UpdateFrameGeometry( hWnd, pFrame ); 576 577 if( pFrame->mnShowState == SW_SHOWMAXIMIZED ) 578 { 579 // #96084 set a useful internal window size because 580 // the window will not be maximized (and the size updated) before show() 581 582 SetMaximizedFrameGeometry( hWnd, pFrame ); 583 } 584 585 return pFrame; 586 } 587 588 // helper that only creates the HWND 589 // to allow for easy reparenting of system windows, (i.e. destroy and create new) 590 HWND ImplSalReCreateHWND( HWND hWndParent, HWND oldhWnd, sal_Bool bAsChild ) 591 { 592 HINSTANCE hInstance = GetSalData()->mhInst; 593 ULONG nSysStyle = GetWindowLong( oldhWnd, GWL_STYLE ); 594 ULONG nExSysStyle = GetWindowLong( oldhWnd, GWL_EXSTYLE ); 595 596 if( bAsChild ) 597 { 598 nSysStyle = WS_CHILD; 599 nExSysStyle = 0; 600 } 601 602 LPCWSTR pClassName = SAL_SUBFRAME_CLASSNAMEW; 603 HWND hWnd = CreateWindowExW( nExSysStyle, pClassName, L"", nSysStyle, 604 CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, 605 hWndParent, 0, hInstance, (void*)GetWindowPtr( oldhWnd ) ); 606 return hWnd; 607 } 608 609 // ======================================================================= 610 611 // Uebersetzungstabelle von System-Keycodes in StarView-Keycodes 612 #define KEY_TAB_SIZE 146 613 614 static sal_uInt16 aImplTranslateKeyTab[KEY_TAB_SIZE] = 615 { 616 // StarView-Code System-Code Index 617 0, // 0 618 0, // VK_LBUTTON 1 619 0, // VK_RBUTTON 2 620 0, // VK_CANCEL 3 621 0, // VK_MBUTTON 4 622 0, // 5 623 0, // 6 624 0, // 7 625 KEY_BACKSPACE, // VK_BACK 8 626 KEY_TAB, // VK_TAB 9 627 0, // 10 628 0, // 11 629 0, // VK_CLEAR 12 630 KEY_RETURN, // VK_RETURN 13 631 0, // 14 632 0, // 15 633 0, // VK_SHIFT 16 634 0, // VK_CONTROL 17 635 0, // VK_MENU 18 636 0, // VK_PAUSE 19 637 0, // VK_CAPITAL 20 638 0, // VK_HANGUL 21 639 0, // 22 640 0, // 23 641 0, // 24 642 KEY_HANGUL_HANJA, // VK_HANJA 25 643 0, // 26 644 KEY_ESCAPE, // VK_ESCAPE 27 645 0, // 28 646 0, // 29 647 0, // 30 648 0, // 31 649 KEY_SPACE, // VK_SPACE 32 650 KEY_PAGEUP, // VK_PRIOR 33 651 KEY_PAGEDOWN, // VK_NEXT 34 652 KEY_END, // VK_END 35 653 KEY_HOME, // VK_HOME 36 654 KEY_LEFT, // VK_LEFT 37 655 KEY_UP, // VK_UP 38 656 KEY_RIGHT, // VK_RIGHT 39 657 KEY_DOWN, // VK_DOWN 40 658 0, // VK_SELECT 41 659 0, // VK_PRINT 42 660 0, // VK_EXECUTE 43 661 0, // VK_SNAPSHOT 44 662 KEY_INSERT, // VK_INSERT 45 663 KEY_DELETE, // VK_DELETE 46 664 KEY_HELP, // VK_HELP 47 665 KEY_0, // 48 666 KEY_1, // 49 667 KEY_2, // 50 668 KEY_3, // 51 669 KEY_4, // 52 670 KEY_5, // 53 671 KEY_6, // 54 672 KEY_7, // 55 673 KEY_8, // 56 674 KEY_9, // 57 675 0, // 58 676 0, // 59 677 0, // 60 678 0, // 61 679 0, // 62 680 0, // 63 681 0, // 64 682 KEY_A, // 65 683 KEY_B, // 66 684 KEY_C, // 67 685 KEY_D, // 68 686 KEY_E, // 69 687 KEY_F, // 70 688 KEY_G, // 71 689 KEY_H, // 72 690 KEY_I, // 73 691 KEY_J, // 74 692 KEY_K, // 75 693 KEY_L, // 76 694 KEY_M, // 77 695 KEY_N, // 78 696 KEY_O, // 79 697 KEY_P, // 80 698 KEY_Q, // 81 699 KEY_R, // 82 700 KEY_S, // 83 701 KEY_T, // 84 702 KEY_U, // 85 703 KEY_V, // 86 704 KEY_W, // 87 705 KEY_X, // 88 706 KEY_Y, // 89 707 KEY_Z, // 90 708 0, // VK_LWIN 91 709 0, // VK_RWIN 92 710 KEY_CONTEXTMENU, // VK_APPS 93 711 0, // 94 712 0, // 95 713 KEY_0, // VK_NUMPAD0 96 714 KEY_1, // VK_NUMPAD1 97 715 KEY_2, // VK_NUMPAD2 98 716 KEY_3, // VK_NUMPAD3 99 717 KEY_4, // VK_NUMPAD4 100 718 KEY_5, // VK_NUMPAD5 101 719 KEY_6, // VK_NUMPAD6 102 720 KEY_7, // VK_NUMPAD7 103 721 KEY_8, // VK_NUMPAD8 104 722 KEY_9, // VK_NUMPAD9 105 723 KEY_MULTIPLY, // VK_MULTIPLY 106 724 KEY_ADD, // VK_ADD 107 725 KEY_DECIMAL, // VK_SEPARATOR 108 726 KEY_SUBTRACT, // VK_SUBTRACT 109 727 KEY_DECIMAL, // VK_DECIMAL 110 728 KEY_DIVIDE, // VK_DIVIDE 111 729 KEY_F1, // VK_F1 112 730 KEY_F2, // VK_F2 113 731 KEY_F3, // VK_F3 114 732 KEY_F4, // VK_F4 115 733 KEY_F5, // VK_F5 116 734 KEY_F6, // VK_F6 117 735 KEY_F7, // VK_F7 118 736 KEY_F8, // VK_F8 119 737 KEY_F9, // VK_F9 120 738 KEY_F10, // VK_F10 121 739 KEY_F11, // VK_F11 122 740 KEY_F12, // VK_F12 123 741 KEY_F13, // VK_F13 124 742 KEY_F14, // VK_F14 125 743 KEY_F15, // VK_F15 126 744 KEY_F16, // VK_F16 127 745 KEY_F17, // VK_F17 128 746 KEY_F18, // VK_F18 129 747 KEY_F19, // VK_F19 130 748 KEY_F20, // VK_F20 131 749 KEY_F21, // VK_F21 132 750 KEY_F22, // VK_F22 133 751 KEY_F23, // VK_F23 134 752 KEY_F24, // VK_F24 135 753 0, // 136 754 0, // 137 755 0, // 138 756 0, // 139 757 0, // 140 758 0, // 141 759 0, // 142 760 0, // 143 761 0, // NUMLOCK 144 762 0 // SCROLLLOCK 145 763 }; 764 765 // ======================================================================= 766 767 static UINT ImplSalGetWheelScrollLines() 768 { 769 UINT nScrLines = 0; 770 HWND hWndMsWheel = WIN_FindWindow( MSH_WHEELMODULE_CLASS, MSH_WHEELMODULE_TITLE ); 771 if ( hWndMsWheel ) 772 { 773 UINT nGetScrollLinesMsgId = RegisterWindowMessage( MSH_SCROLL_LINES ); 774 nScrLines = (UINT)ImplSendMessage( hWndMsWheel, nGetScrollLinesMsgId, 0, 0 ); 775 } 776 777 if ( !nScrLines ) 778 if( !SystemParametersInfo( SPI_GETWHEELSCROLLLINES, 0, &nScrLines, 0 ) ) 779 nScrLines = 0 ; 780 781 if ( !nScrLines ) 782 nScrLines = 3; 783 784 return nScrLines; 785 } 786 787 // ----------------------------------------------------------------------- 788 789 static UINT ImplSalGetWheelScrollChars() 790 { 791 UINT nScrChars = 0; 792 if( !SystemParametersInfo( SPI_GETWHEELSCROLLCHARS, 0, &nScrChars, 0 ) ) 793 { 794 // Depending on Windows version, use proper default or 1 (when 795 // driver emulates hscroll) 796 if( VER_PLATFORM_WIN32_NT == aSalShlData.maVersionInfo.dwPlatformId && 797 aSalShlData.maVersionInfo.dwMajorVersion < 6 ) 798 { 799 // Windows 2000 & WinXP : emulating driver, use step size 800 // of 1 801 return 1; 802 } 803 else 804 { 805 // Longhorn or above: use proper default value of 3 806 return 3; 807 } 808 } 809 810 // system settings successfully read 811 return nScrChars; 812 } 813 814 // ----------------------------------------------------------------------- 815 816 static void ImplSalAddBorder( const WinSalFrame* pFrame, int& width, int& height ) 817 { 818 // transform client size into window size 819 RECT aWinRect; 820 aWinRect.left = 0; 821 aWinRect.right = width-1; 822 aWinRect.top = 0; 823 aWinRect.bottom = height-1; 824 AdjustWindowRectEx( &aWinRect, GetWindowStyle( pFrame->mhWnd ), 825 FALSE, GetWindowExStyle( pFrame->mhWnd ) ); 826 width = aWinRect.right - aWinRect.left + 1; 827 height = aWinRect.bottom - aWinRect.top + 1; 828 } 829 830 // ----------------------------------------------------------------------- 831 832 static void ImplSalCalcFullScreenSize( const WinSalFrame* pFrame, 833 int& rX, int& rY, int& rDX, int& rDY ) 834 { 835 // set window to screen size 836 int nFrameX; 837 int nFrameY; 838 int nCaptionY; 839 int nScreenX = 0; 840 int nScreenY = 0; 841 int nScreenDX = 0; 842 int nScreenDY = 0; 843 844 if ( pFrame->mbSizeBorder ) 845 { 846 nFrameX = GetSystemMetrics( SM_CXSIZEFRAME ); 847 nFrameY = GetSystemMetrics( SM_CYSIZEFRAME ); 848 } 849 else if ( pFrame->mbFixBorder ) 850 { 851 nFrameX = GetSystemMetrics( SM_CXFIXEDFRAME ); 852 nFrameY = GetSystemMetrics( SM_CYFIXEDFRAME ); 853 } 854 else if ( pFrame->mbBorder ) 855 { 856 nFrameX = GetSystemMetrics( SM_CXBORDER ); 857 nFrameY = GetSystemMetrics( SM_CYBORDER ); 858 } 859 else 860 { 861 nFrameX = 0; 862 nFrameY = 0; 863 } 864 if ( pFrame->mbCaption ) 865 nCaptionY = GetSystemMetrics( SM_CYCAPTION ); 866 else 867 nCaptionY = 0; 868 869 try 870 { 871 uno::Reference< XMultiServiceFactory > xFactory( ::comphelper::getProcessServiceFactory(), UNO_QUERY_THROW ); 872 uno::Reference< XIndexAccess > xMultiMon( xFactory->createInstance(OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.awt.DisplayAccess" ) ) ), UNO_QUERY_THROW ); 873 sal_Int32 nMonitors = xMultiMon->getCount(); 874 if( (pFrame->mnDisplay >= 0) && (pFrame->mnDisplay < nMonitors) ) 875 { 876 uno::Reference< XPropertySet > xMonitor( xMultiMon->getByIndex( pFrame->mnDisplay ), UNO_QUERY_THROW ); 877 com::sun::star::awt::Rectangle aRect; 878 if( xMonitor->getPropertyValue( OUString( RTL_CONSTASCII_USTRINGPARAM( "ScreenArea" ) ) ) >>= aRect ) 879 { 880 nScreenX = aRect.X; 881 nScreenY = aRect.Y; 882 nScreenDX = aRect.Width+1; // difference between java/awt convention and vcl 883 nScreenDY = aRect.Height+1; // difference between java/awt convention and vcl 884 } 885 } 886 else 887 { 888 Rectangle aCombined; 889 uno::Reference< XPropertySet > xMonitor( xMultiMon->getByIndex( 0 ), UNO_QUERY_THROW ); 890 com::sun::star::awt::Rectangle aRect; 891 if( xMonitor->getPropertyValue( OUString( RTL_CONSTASCII_USTRINGPARAM( "ScreenArea" ) ) ) >>= aRect ) 892 { 893 aCombined.Left() = aRect.X; 894 aCombined.Top() = aRect.Y; 895 aCombined.Right() = aRect.X + aRect.Width; 896 aCombined.Bottom() = aRect.Y + aRect.Height; 897 for( sal_Int32 i = 1 ; i < nMonitors ; i++ ) 898 { 899 xMonitor = uno::Reference< XPropertySet >( xMultiMon->getByIndex(i), UNO_QUERY_THROW ); 900 if( xMonitor->getPropertyValue( OUString( RTL_CONSTASCII_USTRINGPARAM( "ScreenArea" ) ) ) >>= aRect ) 901 { 902 aCombined.Union( Rectangle( aRect.X, aRect.Y, aRect.X+aRect.Width, aRect.Y+aRect.Height ) ); 903 } 904 } 905 } 906 nScreenX = aCombined.Left(); 907 nScreenY = aCombined.Top(); 908 nScreenDX = aCombined.GetWidth(); 909 nScreenDY = aCombined.GetHeight(); 910 } 911 } 912 catch( Exception& ) 913 { 914 } 915 916 if( !nScreenDX || !nScreenDY ) 917 { 918 nScreenDX = GetSystemMetrics( SM_CXSCREEN ); 919 nScreenDY = GetSystemMetrics( SM_CYSCREEN ); 920 } 921 922 rX = nScreenX -nFrameX; 923 rY = nScreenY -(nFrameY+nCaptionY); 924 rDX = nScreenDX+(nFrameX*2); 925 rDY = nScreenDY+(nFrameY*2)+nCaptionY; 926 } 927 928 // ----------------------------------------------------------------------- 929 930 static void ImplSalFrameFullScreenPos( WinSalFrame* pFrame, sal_Bool bAlways = FALSE ) 931 { 932 if ( bAlways || !IsIconic( pFrame->mhWnd ) ) 933 { 934 // set window to screen size 935 int nX; 936 int nY; 937 int nWidth; 938 int nHeight; 939 ImplSalCalcFullScreenSize( pFrame, nX, nY, nWidth, nHeight ); 940 SetWindowPos( pFrame->mhWnd, 0, 941 nX, nY, nWidth, nHeight, 942 SWP_NOZORDER | SWP_NOACTIVATE ); 943 } 944 } 945 946 // ----------------------------------------------------------------------- 947 948 WinSalFrame::WinSalFrame() 949 { 950 SalData* pSalData = GetSalData(); 951 952 mhWnd = 0; 953 mhCursor = LoadCursor( 0, IDC_ARROW ); 954 mhDefIMEContext = 0; 955 mpGraphics = NULL; 956 mpGraphics2 = NULL; 957 mnShowState = SW_SHOWNORMAL; 958 mnWidth = 0; 959 mnHeight = 0; 960 mnMinWidth = 0; 961 mnMinHeight = 0; 962 mnMaxWidth = SHRT_MAX; 963 mnMaxHeight = SHRT_MAX; 964 mnInputLang = 0; 965 mnInputCodePage = 0; 966 mbGraphics = FALSE; 967 mbCaption = FALSE; 968 mbBorder = FALSE; 969 mbFixBorder = FALSE; 970 mbSizeBorder = FALSE; 971 mbFullScreen = FALSE; 972 mbPresentation = FALSE; 973 mbInShow = FALSE; 974 mbRestoreMaximize = FALSE; 975 mbInMoveMsg = FALSE; 976 mbInSizeMsg = FALSE; 977 mbFullScreenToolWin = FALSE; 978 mbDefPos = TRUE; 979 mbOverwriteState = TRUE; 980 mbIME = FALSE; 981 mbHandleIME = FALSE; 982 mbSpezIME = FALSE; 983 mbAtCursorIME = FALSE; 984 mbCandidateMode = FALSE; 985 mbFloatWin = FALSE; 986 mbNoIcon = FALSE; 987 mSelectedhMenu = 0; 988 mLastActivatedhMenu = 0; 989 mpClipRgnData = NULL; 990 mbFirstClipRect = TRUE; 991 mpNextClipRect = NULL; 992 mnDisplay = 0; 993 994 memset( &maState, 0, sizeof( SalFrameState ) ); 995 maSysData.nSize = sizeof( SystemEnvData ); 996 997 memset( &maGeometry, 0, sizeof( maGeometry ) ); 998 999 // Daten ermitteln, wenn erster Frame angelegt wird 1000 if ( !pSalData->mpFirstFrame ) 1001 { 1002 if ( !aSalShlData.mnWheelMsgId ) 1003 aSalShlData.mnWheelMsgId = RegisterWindowMessage( MSH_MOUSEWHEEL ); 1004 if ( !aSalShlData.mnWheelScrollLines ) 1005 aSalShlData.mnWheelScrollLines = ImplSalGetWheelScrollLines(); 1006 if ( !aSalShlData.mnWheelScrollChars ) 1007 aSalShlData.mnWheelScrollChars = ImplSalGetWheelScrollChars(); 1008 } 1009 1010 // insert frame in framelist 1011 mpNextFrame = pSalData->mpFirstFrame; 1012 pSalData->mpFirstFrame = this; 1013 } 1014 1015 // ----------------------------------------------------------------------- 1016 void WinSalFrame::updateScreenNumber() 1017 { 1018 if( mnDisplay == -1 ) // spans all monitors 1019 return; 1020 WinSalSystem* pSys = static_cast<WinSalSystem*>(ImplGetSalSystem()); 1021 if( pSys ) 1022 { 1023 const std::vector<WinSalSystem::DisplayMonitor>& rMonitors = 1024 pSys->getMonitors(); 1025 Point aPoint( maGeometry.nX, maGeometry.nY ); 1026 size_t nMon = rMonitors.size(); 1027 for( size_t i = 0; i < nMon; i++ ) 1028 { 1029 if( rMonitors[i].m_aArea.IsInside( aPoint ) ) 1030 { 1031 mnDisplay = static_cast<sal_Int32>(i); 1032 maGeometry.nScreenNumber = static_cast<unsigned int>(i); 1033 } 1034 } 1035 } 1036 } 1037 1038 // ----------------------------------------------------------------------- 1039 1040 WinSalFrame::~WinSalFrame() 1041 { 1042 SalData* pSalData = GetSalData(); 1043 1044 if( mpClipRgnData ) 1045 delete [] (BYTE*)mpClipRgnData; 1046 1047 // remove frame from framelist 1048 WinSalFrame** ppFrame = &pSalData->mpFirstFrame; 1049 for(; (*ppFrame != this) && *ppFrame; ppFrame = &(*ppFrame)->mpNextFrame ); 1050 if( *ppFrame ) 1051 *ppFrame = mpNextFrame; 1052 mpNextFrame = NULL; 1053 1054 // Release Cache DC 1055 if ( mpGraphics2 && 1056 mpGraphics2->mhDC ) 1057 ReleaseGraphics( mpGraphics2 ); 1058 1059 // destroy saved DC 1060 if ( mpGraphics ) 1061 { 1062 if ( mpGraphics->mhDefPal ) 1063 SelectPalette( mpGraphics->mhDC, mpGraphics->mhDefPal, TRUE ); 1064 ImplSalDeInitGraphics( mpGraphics ); 1065 ReleaseDC( mhWnd, mpGraphics->mhDC ); 1066 delete mpGraphics; 1067 mpGraphics = NULL; 1068 } 1069 1070 if ( mhWnd ) 1071 { 1072 // reset mouse leave data 1073 if ( pSalData->mhWantLeaveMsg == mhWnd ) 1074 { 1075 pSalData->mhWantLeaveMsg = 0; 1076 if ( pSalData->mpMouseLeaveTimer ) 1077 { 1078 delete pSalData->mpMouseLeaveTimer; 1079 pSalData->mpMouseLeaveTimer = NULL; 1080 } 1081 } 1082 1083 // destroy system frame 1084 if ( !DestroyWindow( mhWnd ) ) 1085 SetWindowPtr( mhWnd, 0 ); 1086 1087 mhWnd = 0; 1088 } 1089 } 1090 1091 // ----------------------------------------------------------------------- 1092 1093 SalGraphics* WinSalFrame::GetGraphics() 1094 { 1095 if ( mbGraphics ) 1096 return NULL; 1097 1098 // Other threads get an own DC, because Windows modify in the 1099 // other case our DC (changing clip region), when they send a 1100 // WM_ERASEBACKGROUND message 1101 SalData* pSalData = GetSalData(); 1102 if ( pSalData->mnAppThreadId != GetCurrentThreadId() ) 1103 { 1104 // We use only three CacheDC's for all threads, because W9x is limited 1105 // to max. 5 Cache DC's per thread 1106 if ( pSalData->mnCacheDCInUse >= 3 ) 1107 return NULL; 1108 1109 if ( !mpGraphics2 ) 1110 { 1111 mpGraphics2 = new WinSalGraphics; 1112 mpGraphics2->mhDC = 0; 1113 mpGraphics2->mhWnd = mhWnd; 1114 mpGraphics2->mbPrinter = FALSE; 1115 mpGraphics2->mbVirDev = FALSE; 1116 mpGraphics2->mbWindow = TRUE; 1117 mpGraphics2->mbScreen = TRUE; 1118 } 1119 1120 HDC hDC = (HDC)ImplSendMessage( pSalData->mpFirstInstance->mhComWnd, 1121 SAL_MSG_GETDC, 1122 (WPARAM)mhWnd, 0 ); 1123 if ( hDC ) 1124 { 1125 mpGraphics2->mhDC = hDC; 1126 if ( pSalData->mhDitherPal ) 1127 { 1128 mpGraphics2->mhDefPal = SelectPalette( hDC, pSalData->mhDitherPal, TRUE ); 1129 RealizePalette( hDC ); 1130 } 1131 ImplSalInitGraphics( mpGraphics2 ); 1132 mbGraphics = TRUE; 1133 1134 pSalData->mnCacheDCInUse++; 1135 return mpGraphics2; 1136 } 1137 else 1138 return NULL; 1139 } 1140 else 1141 { 1142 if ( !mpGraphics ) 1143 { 1144 HDC hDC = GetDC( mhWnd ); 1145 if ( hDC ) 1146 { 1147 mpGraphics = new WinSalGraphics; 1148 mpGraphics->mhDC = hDC; 1149 mpGraphics->mhWnd = mhWnd; 1150 mpGraphics->mbPrinter = FALSE; 1151 mpGraphics->mbVirDev = FALSE; 1152 mpGraphics->mbWindow = TRUE; 1153 mpGraphics->mbScreen = TRUE; 1154 if ( pSalData->mhDitherPal ) 1155 { 1156 mpGraphics->mhDefPal = SelectPalette( hDC, pSalData->mhDitherPal, TRUE ); 1157 RealizePalette( hDC ); 1158 } 1159 ImplSalInitGraphics( mpGraphics ); 1160 mbGraphics = TRUE; 1161 } 1162 } 1163 else 1164 mbGraphics = TRUE; 1165 1166 return mpGraphics; 1167 } 1168 } 1169 1170 // ----------------------------------------------------------------------- 1171 1172 void WinSalFrame::ReleaseGraphics( SalGraphics* pGraphics ) 1173 { 1174 if ( mpGraphics2 == pGraphics ) 1175 { 1176 if ( mpGraphics2->mhDC ) 1177 { 1178 SalData* pSalData = GetSalData(); 1179 if ( mpGraphics2->mhDefPal ) 1180 SelectPalette( mpGraphics2->mhDC, mpGraphics2->mhDefPal, TRUE ); 1181 ImplSalDeInitGraphics( mpGraphics2 ); 1182 ImplSendMessage( pSalData->mpFirstInstance->mhComWnd, 1183 SAL_MSG_RELEASEDC, 1184 (WPARAM)mhWnd, 1185 (LPARAM)mpGraphics2->mhDC ); 1186 mpGraphics2->mhDC = 0; 1187 pSalData->mnCacheDCInUse--; 1188 } 1189 } 1190 1191 mbGraphics = FALSE; 1192 } 1193 1194 // ----------------------------------------------------------------------- 1195 1196 sal_Bool WinSalFrame::PostEvent( void* pData ) 1197 { 1198 return (sal_Bool)ImplPostMessage( mhWnd, SAL_MSG_USEREVENT, 0, (LPARAM)pData ); 1199 } 1200 1201 // ----------------------------------------------------------------------- 1202 1203 void WinSalFrame::SetTitle( const XubString& rTitle ) 1204 { 1205 DBG_ASSERT( sizeof( WCHAR ) == sizeof( xub_Unicode ), "WinSalFrame::SetTitle(): WCHAR != sal_Unicode" ); 1206 1207 if ( !SetWindowTextW( mhWnd, reinterpret_cast<LPCWSTR>(rTitle.GetBuffer()) ) ) 1208 { 1209 ByteString aAnsiTitle = ImplSalGetWinAnsiString( rTitle ); 1210 SetWindowTextA( mhWnd, aAnsiTitle.GetBuffer() ); 1211 } 1212 } 1213 1214 // ----------------------------------------------------------------------- 1215 1216 void WinSalFrame::SetIcon( sal_uInt16 nIcon ) 1217 { 1218 // If we have a window without an Icon (for example a dialog), ignore this call 1219 if ( mbNoIcon ) 1220 return; 1221 1222 // 0 means default (class) icon 1223 HICON hIcon = NULL, hSmIcon = NULL; 1224 if ( !nIcon ) 1225 nIcon = 1; 1226 1227 ImplLoadSalIcon( nIcon, hIcon, hSmIcon ); 1228 1229 DBG_ASSERT( hIcon , "WinSalFrame::SetIcon(): Could not load large icon !" ); 1230 DBG_ASSERT( hSmIcon , "WinSalFrame::SetIcon(): Could not load small icon !" ); 1231 1232 ImplSendMessage( mhWnd, WM_SETICON, ICON_BIG, (LPARAM)hIcon ); 1233 ImplSendMessage( mhWnd, WM_SETICON, ICON_SMALL, (LPARAM)hSmIcon ); 1234 } 1235 1236 // ----------------------------------------------------------------------- 1237 1238 void WinSalFrame::SetMenu( SalMenu* pSalMenu ) 1239 { 1240 WinSalMenu* pWMenu = static_cast<WinSalMenu*>(pSalMenu); 1241 if( pSalMenu && pWMenu->mbMenuBar ) 1242 ::SetMenu( mhWnd, pWMenu->mhMenu ); 1243 } 1244 1245 void WinSalFrame::DrawMenuBar() 1246 { 1247 ::DrawMenuBar( mhWnd ); 1248 } 1249 1250 // ----------------------------------------------------------------------- 1251 HWND ImplGetParentHwnd( HWND hWnd ) 1252 { 1253 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 1254 if( !pFrame || !pFrame->GetWindow()) 1255 return ::GetParent( hWnd ); 1256 Window *pRealParent = pFrame->GetWindow()->ImplGetWindowImpl()->mpRealParent; 1257 if( pRealParent ) 1258 return static_cast<WinSalFrame*>(pRealParent->ImplGetWindowImpl()->mpFrame)->mhWnd; 1259 else 1260 return ::GetParent( hWnd ); 1261 1262 } 1263 1264 // ----------------------------------------------------------------------- 1265 1266 SalFrame* WinSalFrame::GetParent() const 1267 { 1268 return GetWindowPtr( ImplGetParentHwnd( mhWnd ) ); 1269 } 1270 1271 // ----------------------------------------------------------------------- 1272 1273 static void ImplSalShow( HWND hWnd, sal_Bool bVisible, sal_Bool bNoActivate ) 1274 { 1275 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 1276 if ( !pFrame ) 1277 return; 1278 1279 if ( bVisible ) 1280 { 1281 pFrame->mbDefPos = FALSE; 1282 pFrame->mbOverwriteState = TRUE; 1283 pFrame->mbInShow = TRUE; 1284 1285 // #i4715, save position 1286 RECT aRectPreMatrox, aRectPostMatrox; 1287 GetWindowRect( hWnd, &aRectPreMatrox ); 1288 1289 vcl::DeletionListener aDogTag( pFrame ); 1290 if( bNoActivate ) 1291 ShowWindow( hWnd, SW_SHOWNOACTIVATE ); 1292 else 1293 ShowWindow( hWnd, pFrame->mnShowState ); 1294 if( aDogTag.isDeleted() ) 1295 return; 1296 1297 if ( aSalShlData.mbWXP && pFrame->mbFloatWin && !(pFrame->mnStyle & SAL_FRAME_STYLE_NOSHADOW)) 1298 { 1299 // erase the window immediately to improve XP shadow effect 1300 // otherwise the shadow may appears long time before the rest of the window 1301 // especially when accessibility is on 1302 HDC dc = GetDC( hWnd ); 1303 RECT aRect; 1304 GetClientRect( hWnd, &aRect ); 1305 FillRect( dc, &aRect, (HBRUSH) (COLOR_MENU+1) ); // choose the menucolor, because its mostly noticeable for menues 1306 ReleaseDC( hWnd, dc ); 1307 } 1308 1309 // #i4715, matrox centerpopup might have changed our position 1310 // reposition popups without caption (menues, dropdowns, tooltips) 1311 GetWindowRect( hWnd, &aRectPostMatrox ); 1312 if( (GetWindowStyle( hWnd ) & WS_POPUP) && 1313 !pFrame->mbCaption && 1314 (aRectPreMatrox.left != aRectPostMatrox.left || aRectPreMatrox.top != aRectPostMatrox.top) ) 1315 SetWindowPos( hWnd, 0, aRectPreMatrox.left, aRectPreMatrox.top, 0, 0, SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOSIZE ); 1316 1317 if( aDogTag.isDeleted() ) 1318 return; 1319 Window *pClientWin = pFrame->GetWindow()->ImplGetClientWindow(); 1320 if ( pFrame->mbFloatWin || ( pClientWin && (pClientWin->GetStyle() & WB_SYSTEMFLOATWIN) ) ) 1321 pFrame->mnShowState = SW_SHOWNOACTIVATE; 1322 else 1323 pFrame->mnShowState = SW_SHOW; 1324 // Damit Taskleiste unter W98 auch gleich ausgeblendet wird 1325 if ( pFrame->mbPresentation ) 1326 { 1327 HWND hWndParent = ::GetParent( hWnd ); 1328 if ( hWndParent ) 1329 SetForegroundWindow( hWndParent ); 1330 SetForegroundWindow( hWnd ); 1331 } 1332 1333 pFrame->mbInShow = FALSE; 1334 pFrame->updateScreenNumber(); 1335 1336 // Direct Paint only, if we get the SolarMutx 1337 if ( ImplSalYieldMutexTryToAcquire() ) 1338 { 1339 UpdateWindow( hWnd ); 1340 ImplSalYieldMutexRelease(); 1341 } 1342 } 1343 else 1344 { 1345 // See also Bug #91813# and #68467# 1346 if ( pFrame->mbFullScreen && 1347 pFrame->mbPresentation && 1348 (aSalShlData.mnVersion < 500) && 1349 !::GetParent( hWnd ) ) 1350 { 1351 // Damit im Impress-Player in der Taskleiste nicht durch 1352 // einen Windows-Fehler hin- und wieder mal ein leerer 1353 // Button stehen bleibt, muessen wir hier die Taskleiste 1354 // etwas austricksen. Denn wenn wir im FullScreenMode sind 1355 // und das Fenster hiden kommt Windows anscheinend etwas aus 1356 // dem tritt und somit minimieren wir das Fenster damit es 1357 // nicht flackert 1358 ANIMATIONINFO aInfo; 1359 aInfo.cbSize = sizeof( aInfo ); 1360 SystemParametersInfo( SPI_GETANIMATION, 0, &aInfo, 0 ); 1361 if ( aInfo.iMinAnimate ) 1362 { 1363 int nOldAni = aInfo.iMinAnimate; 1364 aInfo.iMinAnimate = 0; 1365 SystemParametersInfo( SPI_SETANIMATION, 0, &aInfo, 0 ); 1366 ShowWindow( pFrame->mhWnd, SW_SHOWMINNOACTIVE ); 1367 aInfo.iMinAnimate = nOldAni; 1368 SystemParametersInfo( SPI_SETANIMATION, 0, &aInfo, 0 ); 1369 } 1370 else 1371 ShowWindow( hWnd, SW_SHOWMINNOACTIVE ); 1372 ShowWindow( hWnd, SW_HIDE ); 1373 } 1374 else 1375 ShowWindow( hWnd, SW_HIDE ); 1376 } 1377 } 1378 1379 // ----------------------------------------------------------------------- 1380 1381 1382 void WinSalFrame::SetExtendedFrameStyle( SalExtStyle ) 1383 { 1384 } 1385 1386 // ----------------------------------------------------------------------- 1387 1388 void WinSalFrame::Show( sal_Bool bVisible, sal_Bool bNoActivate ) 1389 { 1390 // Post this Message to the window, because this only works 1391 // in the thread of the window, which has create this window. 1392 // We post this message to avoid deadlocks 1393 if ( GetSalData()->mnAppThreadId != GetCurrentThreadId() ) 1394 ImplPostMessage( mhWnd, SAL_MSG_SHOW, bVisible, bNoActivate ); 1395 else 1396 ImplSalShow( mhWnd, bVisible, bNoActivate ); 1397 } 1398 1399 // ----------------------------------------------------------------------- 1400 1401 void WinSalFrame::Enable( sal_Bool bEnable ) 1402 { 1403 EnableWindow( mhWnd, bEnable ); 1404 } 1405 1406 // ----------------------------------------------------------------------- 1407 1408 void WinSalFrame::SetMinClientSize( long nWidth, long nHeight ) 1409 { 1410 mnMinWidth = nWidth; 1411 mnMinHeight = nHeight; 1412 } 1413 1414 void WinSalFrame::SetMaxClientSize( long nWidth, long nHeight ) 1415 { 1416 mnMaxWidth = nWidth; 1417 mnMaxHeight = nHeight; 1418 } 1419 1420 // ----------------------------------------------------------------------- 1421 1422 void WinSalFrame::SetPosSize( long nX, long nY, long nWidth, long nHeight, 1423 sal_uInt16 nFlags ) 1424 { 1425 sal_Bool bVisible = (GetWindowStyle( mhWnd ) & WS_VISIBLE) != 0; 1426 if ( !bVisible ) 1427 { 1428 Window *pClientWin = GetWindow()->ImplGetClientWindow(); 1429 if ( mbFloatWin || ( pClientWin && (pClientWin->GetStyle() & WB_SYSTEMFLOATWIN) ) ) 1430 mnShowState = SW_SHOWNOACTIVATE; 1431 else 1432 mnShowState = SW_SHOWNORMAL; 1433 } 1434 else 1435 { 1436 if ( IsIconic( mhWnd ) || IsZoomed( mhWnd ) ) 1437 ShowWindow( mhWnd, SW_RESTORE ); 1438 } 1439 1440 sal_uInt16 nEvent = 0; 1441 UINT nPosSize = 0; 1442 RECT aClientRect, aWindowRect; 1443 GetClientRect( mhWnd, &aClientRect ); // x,y always 0,0, but width and height without border 1444 GetWindowRect( mhWnd, &aWindowRect ); // x,y in screen coordinates, width and height with border 1445 1446 if ( !(nFlags & (SAL_FRAME_POSSIZE_X | SAL_FRAME_POSSIZE_Y)) ) 1447 nPosSize |= SWP_NOMOVE; 1448 else 1449 { 1450 //DBG_ASSERT( nX && nY, " Windowposition of (0,0) requested!" ); 1451 nEvent = SALEVENT_MOVE; 1452 } 1453 if ( !(nFlags & (SAL_FRAME_POSSIZE_WIDTH | SAL_FRAME_POSSIZE_HEIGHT)) ) 1454 nPosSize |= SWP_NOSIZE; 1455 else 1456 nEvent = (nEvent == SALEVENT_MOVE) ? SALEVENT_MOVERESIZE : SALEVENT_RESIZE; 1457 1458 if ( !(nFlags & SAL_FRAME_POSSIZE_X) ) 1459 nX = aWindowRect.left; 1460 if ( !(nFlags & SAL_FRAME_POSSIZE_Y) ) 1461 nY = aWindowRect.top; 1462 if ( !(nFlags & SAL_FRAME_POSSIZE_WIDTH) ) 1463 nWidth = aClientRect.right-aClientRect.left; 1464 if ( !(nFlags & SAL_FRAME_POSSIZE_HEIGHT) ) 1465 nHeight = aClientRect.bottom-aClientRect.top; 1466 1467 // Calculate window size including the border 1468 RECT aWinRect; 1469 aWinRect.left = 0; 1470 aWinRect.right = (int)nWidth-1; 1471 aWinRect.top = 0; 1472 aWinRect.bottom = (int)nHeight-1; 1473 AdjustWindowRectEx( &aWinRect, GetWindowStyle( mhWnd ), 1474 FALSE, GetWindowExStyle( mhWnd ) ); 1475 nWidth = aWinRect.right - aWinRect.left + 1; 1476 nHeight = aWinRect.bottom - aWinRect.top + 1; 1477 1478 if ( !(nPosSize & SWP_NOMOVE) && ::GetParent( mhWnd ) ) 1479 { 1480 // --- RTL --- (mirror window pos) 1481 RECT aParentRect; 1482 GetClientRect( ImplGetParentHwnd( mhWnd ), &aParentRect ); 1483 if( Application::GetSettings().GetLayoutRTL() ) 1484 nX = (aParentRect.right - aParentRect.left) - nWidth-1 - nX; 1485 1486 //#110386#, do not transform coordinates for system child windows 1487 if( !(GetWindowStyle( mhWnd ) & WS_CHILD) ) 1488 { 1489 POINT aPt; 1490 aPt.x = nX; 1491 aPt.y = nY; 1492 1493 HWND parentHwnd = ImplGetParentHwnd( mhWnd ); 1494 WinSalFrame* pParentFrame = GetWindowPtr( parentHwnd ); 1495 if ( pParentFrame && pParentFrame->mnShowState == SW_SHOWMAXIMIZED ) 1496 { 1497 // #i42485#: parent will be shown maximized in which case 1498 // a ClientToScreen uses the wrong coordinates (i.e. those from the restore pos) 1499 // so use the (already updated) frame geometry for the transformation 1500 aPt.x += pParentFrame->maGeometry.nX; 1501 aPt.y += pParentFrame->maGeometry.nY; 1502 } 1503 else 1504 ClientToScreen( parentHwnd, &aPt ); 1505 1506 nX = aPt.x; 1507 nY = aPt.y; 1508 } 1509 } 1510 1511 // #i3338# to be conformant to UNIX we must position the client window, ie without the decoration 1512 // #i43250# if the position was read from the system (GetWindowRect(), see above), it must not be modified 1513 if ( nFlags & SAL_FRAME_POSSIZE_X ) 1514 nX += aWinRect.left; 1515 if ( nFlags & SAL_FRAME_POSSIZE_Y ) 1516 nY += aWinRect.top; 1517 1518 int nScreenX; 1519 int nScreenY; 1520 int nScreenWidth; 1521 int nScreenHeight; 1522 1523 1524 RECT aRect; 1525 ImplSalGetWorkArea( mhWnd, &aRect, NULL ); 1526 nScreenX = aRect.left; 1527 nScreenY = aRect.top; 1528 nScreenWidth = aRect.right-aRect.left; 1529 nScreenHeight = aRect.bottom-aRect.top; 1530 1531 if ( mbDefPos && (nPosSize & SWP_NOMOVE)) // we got no positioning request, so choose default position 1532 { 1533 // center window 1534 1535 HWND hWndParent = ::GetParent( mhWnd ); 1536 // Search for TopLevel Frame 1537 while ( hWndParent && (GetWindowStyle( hWndParent ) & WS_CHILD) ) 1538 hWndParent = ::GetParent( hWndParent ); 1539 // if the Window has a Parent, than center the window to 1540 // the parent, in the other case to the screen 1541 if ( hWndParent && !IsIconic( hWndParent ) && 1542 (GetWindowStyle( hWndParent ) & WS_VISIBLE) ) 1543 { 1544 RECT aParentRect; 1545 GetWindowRect( hWndParent, &aParentRect ); 1546 int nParentWidth = aParentRect.right-aParentRect.left; 1547 int nParentHeight = aParentRect.bottom-aParentRect.top; 1548 1549 // We don't center, when Parent is smaller than our window 1550 if ( (nParentWidth-GetSystemMetrics( SM_CXFIXEDFRAME ) <= nWidth) && 1551 (nParentHeight-GetSystemMetrics( SM_CYFIXEDFRAME ) <= nHeight) ) 1552 { 1553 int nOff = GetSystemMetrics( SM_CYSIZEFRAME ) + GetSystemMetrics( SM_CYCAPTION ); 1554 nX = aParentRect.left+nOff; 1555 nY = aParentRect.top+nOff; 1556 } 1557 else 1558 { 1559 nX = (nParentWidth-nWidth)/2 + aParentRect.left; 1560 nY = (nParentHeight-nHeight)/2 + aParentRect.top; 1561 } 1562 } 1563 else 1564 { 1565 POINT pt; 1566 GetCursorPos( &pt ); 1567 RECT aRect; 1568 aRect.left = pt.x; 1569 aRect.top = pt.y; 1570 aRect.right = pt.x+2; 1571 aRect.bottom = pt.y+2; 1572 1573 // dualmonitor support: 1574 // Get screensize of the monitor whith the mouse pointer 1575 ImplSalGetWorkArea( mhWnd, &aRect, &aRect ); 1576 1577 nX = ((aRect.right-aRect.left)-nWidth)/2 + aRect.left; 1578 nY = ((aRect.bottom-aRect.top)-nHeight)/2 + aRect.top; 1579 } 1580 1581 1582 //if ( bVisible ) 1583 // mbDefPos = FALSE; 1584 1585 mbDefPos = FALSE; // center only once 1586 nPosSize &= ~SWP_NOMOVE; // activate positioning 1587 nEvent = SALEVENT_MOVERESIZE; 1588 } 1589 1590 1591 // Adjust Window in the screen 1592 sal_Bool bCheckOffScreen = TRUE; 1593 1594 // but don't do this for floaters or ownerdraw windows that are currently moved interactively 1595 if( (mnStyle & SAL_FRAME_STYLE_FLOAT) && !(mnStyle & SAL_FRAME_STYLE_OWNERDRAWDECORATION) ) 1596 bCheckOffScreen = FALSE; 1597 1598 if( mnStyle & SAL_FRAME_STYLE_OWNERDRAWDECORATION ) 1599 { 1600 // may be the window is currently being moved (mouse is captured), then no check is required 1601 if( mhWnd == ::GetCapture() ) 1602 bCheckOffScreen = FALSE; 1603 else 1604 bCheckOffScreen = TRUE; 1605 } 1606 1607 if( bCheckOffScreen ) 1608 { 1609 if ( nX+nWidth > nScreenX+nScreenWidth ) 1610 nX = (nScreenX+nScreenWidth) - nWidth; 1611 if ( nY+nHeight > nScreenY+nScreenHeight ) 1612 nY = (nScreenY+nScreenHeight) - nHeight; 1613 if ( nX < nScreenX ) 1614 nX = nScreenX; 1615 if ( nY < nScreenY ) 1616 nY = nScreenY; 1617 } 1618 1619 UINT nPosFlags = SWP_NOACTIVATE | SWP_NOOWNERZORDER | nPosSize; 1620 // bring floating windows always to top 1621 if( !(mnStyle & SAL_FRAME_STYLE_FLOAT) ) 1622 nPosFlags |= SWP_NOZORDER; // do not change z-order 1623 1624 SetWindowPos( mhWnd, HWND_TOP, nX, nY, (int)nWidth, (int)nHeight, nPosFlags ); 1625 1626 UpdateFrameGeometry( mhWnd, this ); 1627 1628 // Notification -- really ??? 1629 if( nEvent ) 1630 CallCallback( nEvent, NULL ); 1631 } 1632 1633 // ----------------------------------------------------------------------- 1634 1635 static void ImplSetParentFrame( WinSalFrame* pThis, HWND hNewParentWnd, sal_Bool bAsChild ) 1636 { 1637 // save hwnd, will be overwritten in WM_CREATE during createwindow 1638 HWND hWndOld = pThis->mhWnd; 1639 HWND hWndOldParent = ::GetParent( hWndOld ); 1640 SalData* pSalData = GetSalData(); 1641 1642 if( hNewParentWnd == hWndOldParent ) 1643 return; 1644 1645 ::std::vector< WinSalFrame* > children; 1646 ::std::vector< WinSalObject* > systemChildren; 1647 1648 // search child windows 1649 WinSalFrame *pFrame = pSalData->mpFirstFrame; 1650 while( pFrame ) 1651 { 1652 HWND hWndParent = ::GetParent( pFrame->mhWnd ); 1653 if( pThis->mhWnd == hWndParent ) 1654 children.push_back( pFrame ); 1655 pFrame = pFrame->mpNextFrame; 1656 } 1657 1658 // search system child windows (plugins etc.) 1659 WinSalObject *pObject = pSalData->mpFirstObject; 1660 while( pObject ) 1661 { 1662 HWND hWndParent = ::GetParent( pObject->mhWnd ); 1663 if( pThis->mhWnd == hWndParent ) 1664 systemChildren.push_back( pObject ); 1665 pObject = pObject->mpNextObject; 1666 } 1667 1668 sal_Bool bNeedGraphics = pThis->mbGraphics; 1669 sal_Bool bNeedCacheDC = FALSE; 1670 1671 HFONT hFont = NULL; 1672 HPEN hPen = NULL; 1673 HBRUSH hBrush = NULL; 1674 1675 #if OSL_DEBUG_LEVEL > 0 1676 int oldCount = pSalData->mnCacheDCInUse; 1677 (void)oldCount; 1678 #endif 1679 1680 // Release Cache DC 1681 if ( pThis->mpGraphics2 && 1682 pThis->mpGraphics2->mhDC ) 1683 { 1684 // save current gdi objects before hdc is gone 1685 hFont = (HFONT) GetCurrentObject( pThis->mpGraphics2->mhDC, OBJ_FONT); 1686 hPen = (HPEN) GetCurrentObject( pThis->mpGraphics2->mhDC, OBJ_PEN); 1687 hBrush = (HBRUSH) GetCurrentObject( pThis->mpGraphics2->mhDC, OBJ_BRUSH); 1688 pThis->ReleaseGraphics( pThis->mpGraphics2 ); 1689 1690 // recreate cache dc only if it was destroyed 1691 bNeedCacheDC = TRUE; 1692 } 1693 1694 // destroy saved DC 1695 if ( pThis->mpGraphics ) 1696 { 1697 if ( pThis->mpGraphics->mhDefPal ) 1698 SelectPalette( pThis->mpGraphics->mhDC, pThis->mpGraphics->mhDefPal, TRUE ); 1699 ImplSalDeInitGraphics( pThis->mpGraphics ); 1700 ReleaseDC( pThis->mhWnd, pThis->mpGraphics->mhDC ); 1701 } 1702 1703 // create a new hwnd with the same styles 1704 HWND hWndParent = hNewParentWnd; 1705 // forward to main thread 1706 HWND hWnd = (HWND) ImplSendMessage( pSalData->mpFirstInstance->mhComWnd, 1707 bAsChild ? SAL_MSG_RECREATECHILDHWND : SAL_MSG_RECREATEHWND, 1708 (WPARAM) hWndParent, (LPARAM)pThis->mhWnd ); 1709 1710 // succeeded ? 1711 DBG_ASSERT( IsWindow( hWnd ), "WinSalFrame::SetParent not successful"); 1712 1713 // recreate DCs 1714 if( bNeedGraphics ) 1715 { 1716 if( pThis->mpGraphics2 ) 1717 { 1718 pThis->mpGraphics2->mhWnd = hWnd; 1719 1720 if( bNeedCacheDC ) 1721 { 1722 // re-create cached DC 1723 HDC hDC = (HDC)ImplSendMessage( pSalData->mpFirstInstance->mhComWnd, 1724 SAL_MSG_GETDC, 1725 (WPARAM) hWnd, 0 ); 1726 if ( hDC ) 1727 { 1728 pThis->mpGraphics2->mhDC = hDC; 1729 if ( pSalData->mhDitherPal ) 1730 { 1731 pThis->mpGraphics2->mhDefPal = SelectPalette( hDC, pSalData->mhDitherPal, TRUE ); 1732 RealizePalette( hDC ); 1733 } 1734 ImplSalInitGraphics( pThis->mpGraphics2 ); 1735 1736 // re-select saved gdi objects 1737 if( hFont ) 1738 SelectObject( hDC, hFont ); 1739 if( hPen ) 1740 SelectObject( hDC, hPen ); 1741 if( hBrush ) 1742 SelectObject( hDC, hBrush ); 1743 1744 pThis->mbGraphics = TRUE; 1745 1746 pSalData->mnCacheDCInUse++; 1747 1748 DBG_ASSERT( oldCount == pSalData->mnCacheDCInUse, "WinSalFrame::SetParent() hDC count corrupted"); 1749 } 1750 } 1751 } 1752 1753 if( pThis->mpGraphics ) 1754 { 1755 // re-create DC 1756 pThis->mpGraphics->mhWnd = hWnd; 1757 pThis->mpGraphics->mhDC = GetDC( hWnd ); 1758 if ( GetSalData()->mhDitherPal ) 1759 { 1760 pThis->mpGraphics->mhDefPal = SelectPalette( pThis->mpGraphics->mhDC, GetSalData()->mhDitherPal, TRUE ); 1761 RealizePalette( pThis->mpGraphics->mhDC ); 1762 } 1763 ImplSalInitGraphics( pThis->mpGraphics ); 1764 pThis->mbGraphics = TRUE; 1765 } 1766 } 1767 1768 1769 // TODO: add SetParent() call for SalObjects 1770 DBG_ASSERT( systemChildren.empty(), "WinSalFrame::SetParent() parent of living system child window will be destroyed!"); 1771 1772 // reparent children before old parent is destroyed 1773 for( ::std::vector< WinSalFrame* >::iterator iChild = children.begin(); iChild != children.end(); iChild++ ) 1774 ImplSetParentFrame( *iChild, hWnd, FALSE ); 1775 1776 children.clear(); 1777 systemChildren.clear(); 1778 1779 // Now destroy original HWND in the thread where it was created. 1780 ImplSendMessage( GetSalData()->mpFirstInstance->mhComWnd, 1781 SAL_MSG_DESTROYHWND, (WPARAM) 0, (LPARAM)hWndOld); 1782 } 1783 1784 // ----------------------------------------------------------------------- 1785 1786 void WinSalFrame::SetParent( SalFrame* pNewParent ) 1787 { 1788 WinSalFrame::mbInReparent = TRUE; 1789 ImplSetParentFrame( this, static_cast<WinSalFrame*>(pNewParent)->mhWnd, FALSE ); 1790 WinSalFrame::mbInReparent = FALSE; 1791 } 1792 1793 bool WinSalFrame::SetPluginParent( SystemParentData* pNewParent ) 1794 { 1795 if ( pNewParent->hWnd == 0 ) 1796 { 1797 pNewParent->hWnd = GetDesktopWindow(); 1798 } 1799 1800 WinSalFrame::mbInReparent = TRUE; 1801 ImplSetParentFrame( this, pNewParent->hWnd, TRUE ); 1802 WinSalFrame::mbInReparent = FALSE; 1803 return true; 1804 } 1805 1806 1807 // ----------------------------------------------------------------------- 1808 1809 void WinSalFrame::GetWorkArea( Rectangle &rRect ) 1810 { 1811 RECT aRect; 1812 ImplSalGetWorkArea( mhWnd, &aRect, NULL ); 1813 rRect.nLeft = aRect.left; 1814 rRect.nRight = aRect.right-1; 1815 rRect.nTop = aRect.top; 1816 rRect.nBottom = aRect.bottom-1; 1817 } 1818 1819 // ----------------------------------------------------------------------- 1820 1821 void WinSalFrame::GetClientSize( long& rWidth, long& rHeight ) 1822 { 1823 rWidth = maGeometry.nWidth; 1824 rHeight = maGeometry.nHeight; 1825 } 1826 1827 // ----------------------------------------------------------------------- 1828 1829 void WinSalFrame::SetWindowState( const SalFrameState* pState ) 1830 { 1831 // Wir testen, ob das Fenster ueberhaupt auf den Bildschirm passt, damit 1832 // nicht wenn die Bildschirm-Aufloesung geaendert wurde, das Fenster aus 1833 // diesem herausragt 1834 int nX; 1835 int nY; 1836 int nWidth; 1837 int nHeight; 1838 int nScreenX; 1839 int nScreenY; 1840 int nScreenWidth; 1841 int nScreenHeight; 1842 1843 RECT aRect; 1844 ImplSalGetWorkArea( mhWnd, &aRect, NULL ); 1845 // #102500# allow some overlap, the window could have been made a little larger than the physical screen 1846 nScreenX = aRect.left-10; 1847 nScreenY = aRect.top-10; 1848 nScreenWidth = aRect.right-aRect.left+20; 1849 nScreenHeight = aRect.bottom-aRect.top+20; 1850 1851 UINT nPosSize = 0; 1852 RECT aWinRect; 1853 GetWindowRect( mhWnd, &aWinRect ); 1854 1855 // to be consistent with Unix, the frame state is without(!) decoration 1856 // ->add the decoration 1857 RECT aRect2 = aWinRect; 1858 AdjustWindowRectEx( &aRect2, GetWindowStyle( mhWnd ), 1859 FALSE, GetWindowExStyle( mhWnd ) ); 1860 long nTopDeco = abs( aWinRect.top - aRect2.top ); 1861 long nLeftDeco = abs( aWinRect.left - aRect2.left ); 1862 long nBottomDeco = abs( aWinRect.bottom - aRect2.bottom ); 1863 long nRightDeco = abs( aWinRect.right - aRect2.right ); 1864 1865 // Fenster-Position/Groesse in den Bildschirm einpassen 1866 if ( !(pState->mnMask & (SAL_FRAMESTATE_MASK_X | SAL_FRAMESTATE_MASK_Y)) ) 1867 nPosSize |= SWP_NOMOVE; 1868 if ( !(pState->mnMask & (SAL_FRAMESTATE_MASK_WIDTH | SAL_FRAMESTATE_MASK_HEIGHT)) ) 1869 nPosSize |= SWP_NOSIZE; 1870 if ( pState->mnMask & SAL_FRAMESTATE_MASK_X ) 1871 nX = (int)pState->mnX - nLeftDeco; 1872 else 1873 nX = aWinRect.left; 1874 if ( pState->mnMask & SAL_FRAMESTATE_MASK_Y ) 1875 nY = (int)pState->mnY - nTopDeco; 1876 else 1877 nY = aWinRect.top; 1878 if ( pState->mnMask & SAL_FRAMESTATE_MASK_WIDTH ) 1879 nWidth = (int)pState->mnWidth + nLeftDeco + nRightDeco; 1880 else 1881 nWidth = aWinRect.right-aWinRect.left; 1882 if ( pState->mnMask & SAL_FRAMESTATE_MASK_HEIGHT ) 1883 nHeight = (int)pState->mnHeight + nTopDeco + nBottomDeco; 1884 else 1885 nHeight = aWinRect.bottom-aWinRect.top; 1886 1887 // Adjust Window in the screen: 1888 // if it does not fit into the screen do nothing, ie default pos/size will be used 1889 // if there is an overlap with the screen border move the window while keeping its size 1890 1891 if( nWidth > nScreenWidth || nHeight > nScreenHeight ) 1892 nPosSize |= (SWP_NOMOVE | SWP_NOSIZE); 1893 1894 if ( nX+nWidth > nScreenX+nScreenWidth ) 1895 nX = (nScreenX+nScreenWidth) - nWidth; 1896 if ( nY+nHeight > nScreenY+nScreenHeight ) 1897 nY = (nScreenY+nScreenHeight) - nHeight; 1898 if ( nX < nScreenX ) 1899 nX = nScreenX; 1900 if ( nY < nScreenY ) 1901 nY = nScreenY; 1902 1903 // Restore-Position setzen 1904 WINDOWPLACEMENT aPlacement; 1905 aPlacement.length = sizeof( aPlacement ); 1906 GetWindowPlacement( mhWnd, &aPlacement ); 1907 1908 // Status setzen 1909 sal_Bool bVisible = (GetWindowStyle( mhWnd ) & WS_VISIBLE) != 0; 1910 sal_Bool bUpdateHiddenFramePos = FALSE; 1911 if ( !bVisible ) 1912 { 1913 aPlacement.showCmd = SW_HIDE; 1914 1915 if ( mbOverwriteState ) 1916 { 1917 if ( pState->mnMask & SAL_FRAMESTATE_MASK_STATE ) 1918 { 1919 if ( pState->mnState & SAL_FRAMESTATE_MINIMIZED ) 1920 mnShowState = SW_SHOWMINIMIZED; 1921 else if ( pState->mnState & SAL_FRAMESTATE_MAXIMIZED ) 1922 { 1923 mnShowState = SW_SHOWMAXIMIZED; 1924 bUpdateHiddenFramePos = TRUE; 1925 } 1926 else if ( pState->mnState & SAL_FRAMESTATE_NORMAL ) 1927 mnShowState = SW_SHOWNORMAL; 1928 } 1929 } 1930 } 1931 else 1932 { 1933 if ( pState->mnMask & SAL_FRAMESTATE_MASK_STATE ) 1934 { 1935 if ( pState->mnState & SAL_FRAMESTATE_MINIMIZED ) 1936 { 1937 if ( pState->mnState & SAL_FRAMESTATE_MAXIMIZED ) 1938 aPlacement.flags |= WPF_RESTORETOMAXIMIZED; 1939 aPlacement.showCmd = SW_SHOWMINIMIZED; 1940 } 1941 else if ( pState->mnState & SAL_FRAMESTATE_MAXIMIZED ) 1942 aPlacement.showCmd = SW_SHOWMAXIMIZED; 1943 else if ( pState->mnState & SAL_FRAMESTATE_NORMAL ) 1944 aPlacement.showCmd = SW_RESTORE; 1945 } 1946 } 1947 1948 // if a window is neither minimized nor maximized or need not be 1949 // positioned visibly (that is in visible state), do not use 1950 // SetWindowPlacement since it calculates including the TaskBar 1951 if ( !IsIconic( mhWnd ) && !IsZoomed( mhWnd ) && 1952 (!bVisible || (aPlacement.showCmd == SW_RESTORE)) ) 1953 { 1954 if( bUpdateHiddenFramePos ) 1955 { 1956 RECT aStateRect; 1957 aStateRect.left = nX; 1958 aStateRect.top = nY; 1959 aStateRect.right = nX+nWidth; 1960 aStateRect.bottom = nY+nHeight; 1961 // #96084 set a useful internal window size because 1962 // the window will not be maximized (and the size updated) before show() 1963 SetMaximizedFrameGeometry( mhWnd, this, &aStateRect ); 1964 SetWindowPos( mhWnd, 0, 1965 maGeometry.nX, maGeometry.nY, maGeometry.nWidth, maGeometry.nHeight, 1966 SWP_NOZORDER | SWP_NOACTIVATE | nPosSize ); 1967 } 1968 else 1969 SetWindowPos( mhWnd, 0, 1970 nX, nY, nWidth, nHeight, 1971 SWP_NOZORDER | SWP_NOACTIVATE | nPosSize ); 1972 } 1973 else 1974 { 1975 if( !(nPosSize & (SWP_NOMOVE|SWP_NOSIZE)) ) 1976 { 1977 aPlacement.rcNormalPosition.left = nX-nScreenX; 1978 aPlacement.rcNormalPosition.top = nY-nScreenY; 1979 aPlacement.rcNormalPosition.right = nX+nWidth-nScreenX; 1980 aPlacement.rcNormalPosition.bottom = nY+nHeight-nScreenY; 1981 } 1982 SetWindowPlacement( mhWnd, &aPlacement ); 1983 } 1984 1985 if( !(nPosSize & SWP_NOMOVE) ) 1986 mbDefPos = FALSE; // window was positioned 1987 } 1988 1989 // ----------------------------------------------------------------------- 1990 1991 sal_Bool WinSalFrame::GetWindowState( SalFrameState* pState ) 1992 { 1993 if ( maState.mnWidth && maState.mnHeight ) 1994 { 1995 *pState = maState; 1996 // #94144# allow Minimize again, should be masked out when read from configuration 1997 // 91625 - Don't save minimize 1998 //if ( !(pState->mnState & SAL_FRAMESTATE_MAXIMIZED) ) 1999 if ( !(pState->mnState & (SAL_FRAMESTATE_MINIMIZED | SAL_FRAMESTATE_MAXIMIZED)) ) 2000 pState->mnState |= SAL_FRAMESTATE_NORMAL; 2001 return TRUE; 2002 } 2003 2004 return FALSE; 2005 } 2006 2007 // ----------------------------------------------------------------------- 2008 2009 void WinSalFrame::SetScreenNumber( unsigned int nNewScreen ) 2010 { 2011 WinSalSystem* pSys = static_cast<WinSalSystem*>(ImplGetSalSystem()); 2012 if( pSys ) 2013 { 2014 const std::vector<WinSalSystem::DisplayMonitor>& rMonitors = 2015 pSys->getMonitors(); 2016 size_t nMon = rMonitors.size(); 2017 if( nNewScreen < nMon ) 2018 { 2019 Point aOldMonPos, aNewMonPos( rMonitors[nNewScreen].m_aArea.TopLeft() ); 2020 Point aCurPos( maGeometry.nX, maGeometry.nY ); 2021 for( size_t i = 0; i < nMon; i++ ) 2022 { 2023 if( rMonitors[i].m_aArea.IsInside( aCurPos ) ) 2024 { 2025 aOldMonPos = rMonitors[i].m_aArea.TopLeft(); 2026 break; 2027 } 2028 } 2029 mnDisplay = nNewScreen; 2030 maGeometry.nScreenNumber = nNewScreen; 2031 SetPosSize( aNewMonPos.X() + (maGeometry.nX - aOldMonPos.X()), 2032 aNewMonPos.Y() + (maGeometry.nY - aOldMonPos.Y()), 2033 0, 0, 2034 SAL_FRAME_POSSIZE_X | SAL_FRAME_POSSIZE_Y ); 2035 } 2036 } 2037 } 2038 2039 // ----------------------------------------------------------------------- 2040 2041 void WinSalFrame::ShowFullScreen( sal_Bool bFullScreen, sal_Int32 nDisplay ) 2042 { 2043 if ( (mbFullScreen == bFullScreen) && (!bFullScreen || (mnDisplay == nDisplay)) ) 2044 return; 2045 2046 mbFullScreen = bFullScreen; 2047 mnDisplay = nDisplay; 2048 2049 if ( bFullScreen ) 2050 { 2051 // Damit Taskleiste von Windows ausgeblendet wird 2052 DWORD nExStyle = GetWindowExStyle( mhWnd ); 2053 if ( nExStyle & WS_EX_TOOLWINDOW ) 2054 { 2055 mbFullScreenToolWin = TRUE; 2056 nExStyle &= ~WS_EX_TOOLWINDOW; 2057 SetWindowExStyle( mhWnd, nExStyle ); 2058 } 2059 // save old position 2060 GetWindowRect( mhWnd, &maFullScreenRect ); 2061 2062 // save show state 2063 mnFullScreenShowState = mnShowState; 2064 if ( !(GetWindowStyle( mhWnd ) & WS_VISIBLE) ) 2065 mnShowState = SW_SHOW; 2066 2067 // set window to screen size 2068 ImplSalFrameFullScreenPos( this, TRUE ); 2069 } 2070 else 2071 { 2072 // wenn ShowState wieder hergestellt werden muss, hiden wir zuerst 2073 // das Fenster, damit es nicht so sehr flackert 2074 sal_Bool bVisible = (GetWindowStyle( mhWnd ) & WS_VISIBLE) != 0; 2075 if ( bVisible && (mnShowState != mnFullScreenShowState) ) 2076 ShowWindow( mhWnd, SW_HIDE ); 2077 2078 if ( mbFullScreenToolWin ) 2079 SetWindowExStyle( mhWnd, GetWindowExStyle( mhWnd ) | WS_EX_TOOLWINDOW ); 2080 mbFullScreenToolWin = FALSE; 2081 2082 SetWindowPos( mhWnd, 0, 2083 maFullScreenRect.left, 2084 maFullScreenRect.top, 2085 maFullScreenRect.right-maFullScreenRect.left, 2086 maFullScreenRect.bottom-maFullScreenRect.top, 2087 SWP_NOZORDER | SWP_NOACTIVATE ); 2088 2089 // restore show state 2090 if ( mnShowState != mnFullScreenShowState ) 2091 { 2092 mnShowState = mnFullScreenShowState; 2093 if ( bVisible ) 2094 { 2095 mbInShow = TRUE; 2096 ShowWindow( mhWnd, mnShowState ); 2097 mbInShow = FALSE; 2098 UpdateWindow( mhWnd ); 2099 } 2100 } 2101 } 2102 } 2103 2104 // ----------------------------------------------------------------------- 2105 2106 void WinSalFrame::StartPresentation( sal_Bool bStart ) 2107 { 2108 if ( mbPresentation == bStart ) 2109 return; 2110 2111 mbPresentation = bStart; 2112 2113 SalData* pSalData = GetSalData(); 2114 if ( bStart ) 2115 { 2116 if ( !pSalData->mpSageEnableProc ) 2117 { 2118 if ( pSalData->mnSageStatus != DISABLE_AGENT ) 2119 { 2120 OFSTRUCT aOS; 2121 OpenFile( "SAGE.DLL", &aOS, OF_EXIST ); 2122 2123 if ( !aOS.nErrCode ) 2124 { 2125 oslModule mhSageInst = osl_loadAsciiModule( aOS.szPathName, SAL_LOADMODULE_DEFAULT ); 2126 pSalData->mpSageEnableProc = (SysAgt_Enable_PROC)osl_getAsciiFunctionSymbol( mhSageInst, "System_Agent_Enable" ); 2127 } 2128 else 2129 pSalData->mnSageStatus = DISABLE_AGENT; 2130 } 2131 } 2132 2133 if ( pSalData->mpSageEnableProc ) 2134 { 2135 pSalData->mnSageStatus = pSalData->mpSageEnableProc( GET_AGENT_STATUS ); 2136 if ( pSalData->mnSageStatus == ENABLE_AGENT ) 2137 pSalData->mpSageEnableProc( DISABLE_AGENT ); 2138 } 2139 2140 // Bildschirmschoner ausschalten, wenn Praesentation laueft 2141 SystemParametersInfo( SPI_GETSCREENSAVEACTIVE, 0, 2142 &(pSalData->mbScrSvrEnabled), 0 ); 2143 if ( pSalData->mbScrSvrEnabled ) 2144 SystemParametersInfo( SPI_SETSCREENSAVEACTIVE, FALSE, 0, 0 ); 2145 } 2146 else 2147 { 2148 // Bildschirmschoner wieder einschalten 2149 if ( pSalData->mbScrSvrEnabled ) 2150 SystemParametersInfo( SPI_SETSCREENSAVEACTIVE, pSalData->mbScrSvrEnabled, 0, 0 ); 2151 2152 // Systemagenten wieder aktivieren 2153 if ( pSalData->mnSageStatus == ENABLE_AGENT ) 2154 pSalData->mpSageEnableProc( pSalData->mnSageStatus ); 2155 } 2156 } 2157 2158 // ----------------------------------------------------------------------- 2159 2160 void WinSalFrame::SetAlwaysOnTop( sal_Bool bOnTop ) 2161 { 2162 HWND hWnd; 2163 if ( bOnTop ) 2164 hWnd = HWND_TOPMOST; 2165 else 2166 hWnd = HWND_NOTOPMOST; 2167 SetWindowPos( mhWnd, hWnd, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE ); 2168 } 2169 2170 // ----------------------------------------------------------------------- 2171 2172 static void ImplSalToTop( HWND hWnd, sal_uInt16 nFlags ) 2173 { 2174 WinSalFrame* pToTopFrame = GetWindowPtr( hWnd ); 2175 if( pToTopFrame && (pToTopFrame->mnStyle & SAL_FRAME_STYLE_SYSTEMCHILD) != 0 ) 2176 BringWindowToTop( hWnd ); 2177 2178 if ( nFlags & SAL_FRAME_TOTOP_FOREGROUNDTASK ) 2179 { 2180 // This magic code is necessary to connect the input focus of the 2181 // current window thread and the thread which owns the window that 2182 // should be the new foreground window. 2183 HWND hCurrWnd = GetForegroundWindow(); 2184 DWORD myThreadID = GetCurrentThreadId(); 2185 DWORD currThreadID = GetWindowThreadProcessId(hCurrWnd,NULL); 2186 AttachThreadInput(myThreadID, currThreadID,TRUE); 2187 SetForegroundWindow(hWnd); 2188 AttachThreadInput(myThreadID,currThreadID,FALSE); 2189 } 2190 2191 if ( nFlags & SAL_FRAME_TOTOP_RESTOREWHENMIN ) 2192 { 2193 HWND hIconicWnd = hWnd; 2194 while ( hIconicWnd ) 2195 { 2196 if ( IsIconic( hIconicWnd ) ) 2197 { 2198 WinSalFrame* pFrame = GetWindowPtr( hIconicWnd ); 2199 if ( pFrame ) 2200 { 2201 if ( GetWindowPtr( hWnd )->mbRestoreMaximize ) 2202 ShowWindow( hIconicWnd, SW_MAXIMIZE ); 2203 else 2204 ShowWindow( hIconicWnd, SW_RESTORE ); 2205 } 2206 else 2207 ShowWindow( hIconicWnd, SW_RESTORE ); 2208 } 2209 2210 hIconicWnd = ::GetParent( hIconicWnd ); 2211 } 2212 } 2213 2214 if ( !IsIconic( hWnd ) && IsWindowVisible( hWnd ) ) 2215 { 2216 SetFocus( hWnd ); 2217 2218 // Windows behauptet oefters mal, das man den Focus hat, obwohl 2219 // man diesen nicht hat. Wenn dies der Fall ist, dann versuchen 2220 // wir diesen auch ganz richtig zu bekommen. 2221 if ( ::GetFocus() == hWnd ) 2222 SetForegroundWindow( hWnd ); 2223 } 2224 } 2225 2226 // ----------------------------------------------------------------------- 2227 2228 void WinSalFrame::ToTop( sal_uInt16 nFlags ) 2229 { 2230 nFlags &= ~SAL_FRAME_TOTOP_GRABFOCUS; // this flag is not needed on win32 2231 // Post this Message to the window, because this only works 2232 // in the thread of the window, which has create this window. 2233 // We post this message to avoid deadlocks 2234 if ( GetSalData()->mnAppThreadId != GetCurrentThreadId() ) 2235 ImplPostMessage( mhWnd, SAL_MSG_TOTOP, nFlags, 0 ); 2236 else 2237 ImplSalToTop( mhWnd, nFlags ); 2238 } 2239 2240 // ----------------------------------------------------------------------- 2241 2242 void WinSalFrame::SetPointer( PointerStyle ePointerStyle ) 2243 { 2244 struct ImplPtrData 2245 { 2246 HCURSOR mhCursor; 2247 LPCSTR mnSysId; 2248 UINT mnOwnId; 2249 }; 2250 2251 static ImplPtrData aImplPtrTab[POINTER_COUNT] = 2252 { 2253 { 0, IDC_ARROW, 0 }, // POINTER_ARROW 2254 { 0, 0, SAL_RESID_POINTER_NULL }, // POINTER_NULL 2255 { 0, IDC_WAIT, 0 }, // POINTER_WAIT 2256 { 0, IDC_IBEAM, 0 }, // POINTER_TEXT 2257 { 0, IDC_HELP, 0 }, // POINTER_HELP 2258 { 0, 0, SAL_RESID_POINTER_CROSS }, // POINTER_CROSS 2259 { 0, 0, SAL_RESID_POINTER_MOVE }, // POINTER_MOVE 2260 { 0, IDC_SIZENS, 0 }, // POINTER_NSIZE 2261 { 0, IDC_SIZENS, 0 }, // POINTER_SSIZE 2262 { 0, IDC_SIZEWE, 0 }, // POINTER_WSIZE 2263 { 0, IDC_SIZEWE, 0 }, // POINTER_ESIZE 2264 { 0, IDC_SIZENWSE, 0 }, // POINTER_NWSIZE 2265 { 0, IDC_SIZENESW, 0 }, // POINTER_NESIZE 2266 { 0, IDC_SIZENESW, 0 }, // POINTER_SWSIZE 2267 { 0, IDC_SIZENWSE, 0 }, // POINTER_SESIZE 2268 { 0, IDC_SIZENS, 0 }, // POINTER_WINDOW_NSIZE 2269 { 0, IDC_SIZENS, 0 }, // POINTER_WINDOW_SSIZE 2270 { 0, IDC_SIZEWE, 0 }, // POINTER_WINDOW_WSIZE 2271 { 0, IDC_SIZEWE, 0 }, // POINTER_WINDOW_ESIZE 2272 { 0, IDC_SIZENWSE, 0 }, // POINTER_WINDOW_NWSIZE 2273 { 0, IDC_SIZENESW, 0 }, // POINTER_WINDOW_NESIZE 2274 { 0, IDC_SIZENESW, 0 }, // POINTER_WINDOW_SWSIZE 2275 { 0, IDC_SIZENWSE, 0 }, // POINTER_WINDOW_SESIZE 2276 { 0, 0, SAL_RESID_POINTER_HSPLIT }, // POINTER_HSPLIT 2277 { 0, 0, SAL_RESID_POINTER_VSPLIT }, // POINTER_VSPLIT 2278 { 0, 0, SAL_RESID_POINTER_HSIZEBAR }, // POINTER_HSIZEBAR 2279 { 0, 0, SAL_RESID_POINTER_VSIZEBAR }, // POINTER_VSIZEBAR 2280 { 0, 0, SAL_RESID_POINTER_HAND }, // POINTER_HAND 2281 { 0, 0, SAL_RESID_POINTER_REFHAND }, // POINTER_REFHAND 2282 { 0, 0, SAL_RESID_POINTER_PEN }, // POINTER_PEN 2283 { 0, 0, SAL_RESID_POINTER_MAGNIFY }, // POINTER_MAGNIFY 2284 { 0, 0, SAL_RESID_POINTER_FILL }, // POINTER_FILL 2285 { 0, 0, SAL_RESID_POINTER_ROTATE }, // POINTER_ROTATE 2286 { 0, 0, SAL_RESID_POINTER_HSHEAR }, // POINTER_HSHEAR 2287 { 0, 0, SAL_RESID_POINTER_VSHEAR }, // POINTER_VSHEAR 2288 { 0, 0, SAL_RESID_POINTER_MIRROR }, // POINTER_MIRROR 2289 { 0, 0, SAL_RESID_POINTER_CROOK }, // POINTER_CROOK 2290 { 0, 0, SAL_RESID_POINTER_CROP }, // POINTER_CROP 2291 { 0, 0, SAL_RESID_POINTER_MOVEPOINT }, // POINTER_MOVEPOINT 2292 { 0, 0, SAL_RESID_POINTER_MOVEBEZIERWEIGHT }, // POINTER_MOVEBEZIERWEIGHT 2293 { 0, 0, SAL_RESID_POINTER_MOVEDATA }, // POINTER_MOVEDATA 2294 { 0, 0, SAL_RESID_POINTER_COPYDATA }, // POINTER_COPYDATA 2295 { 0, 0, SAL_RESID_POINTER_LINKDATA }, // POINTER_LINKDATA 2296 { 0, 0, SAL_RESID_POINTER_MOVEDATALINK }, // POINTER_MOVEDATALINK 2297 { 0, 0, SAL_RESID_POINTER_COPYDATALINK }, // POINTER_COPYDATALINK 2298 { 0, 0, SAL_RESID_POINTER_MOVEFILE }, // POINTER_MOVEFILE 2299 { 0, 0, SAL_RESID_POINTER_COPYFILE }, // POINTER_COPYFILE 2300 { 0, 0, SAL_RESID_POINTER_LINKFILE }, // POINTER_LINKFILE 2301 { 0, 0, SAL_RESID_POINTER_MOVEFILELINK }, // POINTER_MOVEFILELINK 2302 { 0, 0, SAL_RESID_POINTER_COPYFILELINK }, // POINTER_COPYFILELINK 2303 { 0, 0, SAL_RESID_POINTER_MOVEFILES }, // POINTER_MOVEFILES 2304 { 0, 0, SAL_RESID_POINTER_COPYFILES }, // POINTER_COPYFILES 2305 { 0, 0, SAL_RESID_POINTER_NOTALLOWED }, // POINTER_NOTALLOWED 2306 { 0, 0, SAL_RESID_POINTER_DRAW_LINE }, // POINTER_DRAW_LINE 2307 { 0, 0, SAL_RESID_POINTER_DRAW_RECT }, // POINTER_DRAW_RECT 2308 { 0, 0, SAL_RESID_POINTER_DRAW_POLYGON }, // POINTER_DRAW_POLYGON 2309 { 0, 0, SAL_RESID_POINTER_DRAW_BEZIER }, // POINTER_DRAW_BEZIER 2310 { 0, 0, SAL_RESID_POINTER_DRAW_ARC }, // POINTER_DRAW_ARC 2311 { 0, 0, SAL_RESID_POINTER_DRAW_PIE }, // POINTER_DRAW_PIE 2312 { 0, 0, SAL_RESID_POINTER_DRAW_CIRCLECUT }, // POINTER_DRAW_CIRCLECUT 2313 { 0, 0, SAL_RESID_POINTER_DRAW_ELLIPSE }, // POINTER_DRAW_ELLIPSE 2314 { 0, 0, SAL_RESID_POINTER_DRAW_FREEHAND }, // POINTER_DRAW_FREEHAND 2315 { 0, 0, SAL_RESID_POINTER_DRAW_CONNECT }, // POINTER_DRAW_CONNECT 2316 { 0, 0, SAL_RESID_POINTER_DRAW_TEXT }, // POINTER_DRAW_TEXT 2317 { 0, 0, SAL_RESID_POINTER_DRAW_CAPTION }, // POINTER_DRAW_CAPTION 2318 { 0, 0, SAL_RESID_POINTER_CHART }, // POINTER_CHART 2319 { 0, 0, SAL_RESID_POINTER_DETECTIVE }, // POINTER_DETECTIVE 2320 { 0, 0, SAL_RESID_POINTER_PIVOT_COL }, // POINTER_PIVOT_COL 2321 { 0, 0, SAL_RESID_POINTER_PIVOT_ROW }, // POINTER_PIVOT_ROW 2322 { 0, 0, SAL_RESID_POINTER_PIVOT_FIELD }, // POINTER_PIVOT_FIELD 2323 { 0, 0, SAL_RESID_POINTER_CHAIN }, // POINTER_CHAIN 2324 { 0, 0, SAL_RESID_POINTER_CHAIN_NOTALLOWED }, // POINTER_CHAIN_NOTALLOWED 2325 { 0, 0, SAL_RESID_POINTER_TIMEEVENT_MOVE }, // POINTER_TIMEEVENT_MOVE 2326 { 0, 0, SAL_RESID_POINTER_TIMEEVENT_SIZE }, // POINTER_TIMEEVENT_SIZE 2327 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_N }, // POINTER_AUTOSCROLL_N 2328 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_S }, // POINTER_AUTOSCROLL_S 2329 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_W }, // POINTER_AUTOSCROLL_W 2330 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_E }, // POINTER_AUTOSCROLL_E 2331 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_NW }, // POINTER_AUTOSCROLL_NW 2332 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_NE }, // POINTER_AUTOSCROLL_NE 2333 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_SW }, // POINTER_AUTOSCROLL_SW 2334 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_SE }, // POINTER_AUTOSCROLL_SE 2335 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_NS }, // POINTER_AUTOSCROLL_NS 2336 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_WE }, // POINTER_AUTOSCROLL_WE 2337 { 0, 0, SAL_RESID_POINTER_AUTOSCROLL_NSWE }, // POINTER_AUTOSCROLL_NSWE 2338 { 0, 0, SAL_RESID_POINTER_AIRBRUSH }, // POINTER_AIRBRUSH 2339 { 0, 0, SAL_RESID_POINTER_TEXT_VERTICAL }, // POINTER_TEXT_VERTICAL 2340 { 0, 0, SAL_RESID_POINTER_PIVOT_DELETE }, // POINTER_PIVOT_DELETE 2341 2342 // --> FME 2004-07-30 #i32329# Enhanced table selection 2343 { 0, 0, SAL_RESID_POINTER_TAB_SELECT_S }, // POINTER_TAB_SELECT_S 2344 { 0, 0, SAL_RESID_POINTER_TAB_SELECT_E }, // POINTER_TAB_SELECT_E 2345 { 0, 0, SAL_RESID_POINTER_TAB_SELECT_SE }, // POINTER_TAB_SELECT_SE 2346 { 0, 0, SAL_RESID_POINTER_TAB_SELECT_W }, // POINTER_TAB_SELECT_W 2347 { 0, 0, SAL_RESID_POINTER_TAB_SELECT_SW }, // POINTER_TAB_SELECT_SW 2348 // <-- 2349 2350 // --> FME 2004-08-16 #i20119# Paintbrush tool 2351 { 0, 0, SAL_RESID_POINTER_PAINTBRUSH } // POINTER_PAINTBRUSH 2352 // <-- 2353 2354 }; 2355 2356 #if POINTER_COUNT != 94 2357 #error New Pointer must be defined! 2358 #endif 2359 2360 // Mousepointer loaded ? 2361 if ( !aImplPtrTab[ePointerStyle].mhCursor ) 2362 { 2363 if ( aImplPtrTab[ePointerStyle].mnOwnId ) 2364 aImplPtrTab[ePointerStyle].mhCursor = ImplLoadSalCursor( aImplPtrTab[ePointerStyle].mnOwnId ); 2365 else 2366 aImplPtrTab[ePointerStyle].mhCursor = LoadCursor( 0, aImplPtrTab[ePointerStyle].mnSysId ); 2367 } 2368 2369 // Unterscheidet sich der Mauspointer, dann den neuen setzen 2370 if ( mhCursor != aImplPtrTab[ePointerStyle].mhCursor ) 2371 { 2372 mhCursor = aImplPtrTab[ePointerStyle].mhCursor; 2373 SetCursor( mhCursor ); 2374 } 2375 } 2376 2377 // ----------------------------------------------------------------------- 2378 2379 void WinSalFrame::CaptureMouse( sal_Bool bCapture ) 2380 { 2381 // Send this Message to the window, because CaptureMouse() only work 2382 // in the thread of the window, which has create this window 2383 int nMsg; 2384 if ( bCapture ) 2385 nMsg = SAL_MSG_CAPTUREMOUSE; 2386 else 2387 nMsg = SAL_MSG_RELEASEMOUSE; 2388 ImplSendMessage( mhWnd, nMsg, 0, 0 ); 2389 } 2390 2391 // ----------------------------------------------------------------------- 2392 2393 void WinSalFrame::SetPointerPos( long nX, long nY ) 2394 { 2395 POINT aPt; 2396 aPt.x = (int)nX; 2397 aPt.y = (int)nY; 2398 ClientToScreen( mhWnd, &aPt ); 2399 SetCursorPos( aPt.x, aPt.y ); 2400 } 2401 2402 // ----------------------------------------------------------------------- 2403 2404 void WinSalFrame::Flush() 2405 { 2406 GdiFlush(); 2407 } 2408 2409 // ----------------------------------------------------------------------- 2410 2411 void WinSalFrame::Sync() 2412 { 2413 GdiFlush(); 2414 } 2415 2416 // ----------------------------------------------------------------------- 2417 2418 static void ImplSalFrameSetInputContext( HWND hWnd, const SalInputContext* pContext ) 2419 { 2420 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 2421 sal_Bool bIME = (pContext->mnOptions & SAL_INPUTCONTEXT_TEXT) != 0; 2422 if ( bIME ) 2423 { 2424 if ( !pFrame->mbIME ) 2425 { 2426 pFrame->mbIME = TRUE; 2427 2428 if ( pFrame->mhDefIMEContext ) 2429 { 2430 ImmAssociateContext( pFrame->mhWnd, pFrame->mhDefIMEContext ); 2431 UINT nImeProps = ImmGetProperty( GetKeyboardLayout( 0 ), IGP_PROPERTY ); 2432 pFrame->mbSpezIME = (nImeProps & IME_PROP_SPECIAL_UI) != 0; 2433 pFrame->mbAtCursorIME = (nImeProps & IME_PROP_AT_CARET) != 0; 2434 pFrame->mbHandleIME = !pFrame->mbSpezIME; 2435 } 2436 } 2437 2438 // When the application can't handle IME messages, then the 2439 // System should handle the IME handling 2440 if ( !(pContext->mnOptions & SAL_INPUTCONTEXT_EXTTEXTINPUT) ) 2441 pFrame->mbHandleIME = FALSE; 2442 2443 // Set the Font for IME Handling 2444 if ( pContext->mpFont ) 2445 { 2446 HIMC hIMC = ImmGetContext( pFrame->mhWnd ); 2447 if ( hIMC ) 2448 { 2449 LOGFONTW aLogFont; 2450 HDC hDC = GetDC( pFrame->mhWnd ); 2451 // In case of vertical writing, always append a '@' to the 2452 // Windows font name, not only if such a Windows font really is 2453 // available (bTestVerticalAvail == false in the below call): 2454 // The Windows IME's candidates window seems to always use a 2455 // font that has all necessary glyphs, not necessarily the one 2456 // specified by this font name; but it seems to decide whether 2457 // to use that font's horizontal or vertical variant based on a 2458 // '@' in front of this font name. 2459 ImplGetLogFontFromFontSelect( hDC, pContext->mpFont, aLogFont, 2460 false ); 2461 ReleaseDC( pFrame->mhWnd, hDC ); 2462 ImmSetCompositionFontW( hIMC, &aLogFont ); 2463 ImmReleaseContext( pFrame->mhWnd, hIMC ); 2464 } 2465 } 2466 } 2467 else 2468 { 2469 if ( pFrame->mbIME ) 2470 { 2471 pFrame->mbIME = FALSE; 2472 pFrame->mbHandleIME = FALSE; 2473 ImmAssociateContext( pFrame->mhWnd, 0 ); 2474 } 2475 } 2476 } 2477 2478 // ----------------------------------------------------------------------- 2479 2480 void WinSalFrame::SetInputContext( SalInputContext* pContext ) 2481 { 2482 // Must be called in the main thread! 2483 ImplSendMessage( mhWnd, SAL_MSG_SETINPUTCONTEXT, 0, (LPARAM)(void*)pContext ); 2484 } 2485 2486 // ----------------------------------------------------------------------- 2487 2488 static void ImplSalFrameEndExtTextInput( HWND hWnd, sal_uInt16 nFlags ) 2489 { 2490 HIMC hIMC = ImmGetContext( hWnd ); 2491 if ( hIMC ) 2492 { 2493 DWORD nIndex; 2494 if ( nFlags & SAL_FRAME_ENDEXTTEXTINPUT_COMPLETE ) 2495 nIndex = CPS_COMPLETE; 2496 else 2497 nIndex = CPS_CANCEL; 2498 2499 ImmNotifyIME( hIMC, NI_COMPOSITIONSTR, nIndex, 0 ); 2500 ImmReleaseContext( hWnd, hIMC ); 2501 } 2502 } 2503 2504 // ----------------------------------------------------------------------- 2505 2506 void WinSalFrame::EndExtTextInput( sal_uInt16 nFlags ) 2507 { 2508 // Must be called in the main thread! 2509 ImplSendMessage( mhWnd, SAL_MSG_ENDEXTTEXTINPUT, (WPARAM)nFlags, 0 ); 2510 } 2511 2512 // ----------------------------------------------------------------------- 2513 2514 static void ImplGetKeyNameText( LONG lParam, sal_Unicode* pBuf, 2515 UINT& rCount, UINT nMaxSize, 2516 const sal_Char* pReplace ) 2517 { 2518 DBG_ASSERT( sizeof( WCHAR ) == sizeof( xub_Unicode ), "WinSalFrame::ImplGetKeyNameTextW(): WCHAR != sal_Unicode" ); 2519 2520 static const int nMaxKeyLen = 350; 2521 WCHAR aKeyBuf[ nMaxKeyLen ]; 2522 int nKeyLen = 0; 2523 if ( lParam ) 2524 { 2525 if ( true/*aSalShlData.mbWNT*/ ) 2526 { 2527 nKeyLen = GetKeyNameTextW( lParam, aKeyBuf, nMaxKeyLen ); 2528 DBG_ASSERT( nKeyLen <= nMaxKeyLen, "Invalid key name length!" ); 2529 if( nKeyLen > nMaxKeyLen ) 2530 nKeyLen = 0; 2531 else if( nKeyLen > 0 ) 2532 { 2533 // Capitalize just the first letter of key names 2534 CharLowerBuffW( aKeyBuf, nKeyLen ); 2535 2536 bool bUpper = true; 2537 for( WCHAR *pW=aKeyBuf, *pE=pW+nKeyLen; pW < pE; ++pW ) 2538 { 2539 if( bUpper ) 2540 CharUpperBuffW( pW, 1 ); 2541 bUpper = (*pW=='+') || (*pW=='-') || (*pW==' ') || (*pW=='.'); 2542 } 2543 } 2544 } 2545 } 2546 2547 if ( (nKeyLen > 0) || pReplace ) 2548 { 2549 if( (rCount > 0) && (rCount < nMaxSize) ) 2550 { 2551 pBuf[rCount] = '+'; 2552 rCount++; 2553 } 2554 2555 if( nKeyLen > 0 ) 2556 { 2557 if( nKeyLen + rCount > nMaxSize ) 2558 nKeyLen = nMaxSize - rCount; 2559 memcpy( pBuf+rCount, aKeyBuf, nKeyLen*sizeof( sal_Unicode ) ); 2560 rCount += nKeyLen; 2561 } 2562 else // fall back to provided default name 2563 { 2564 while( *pReplace && (rCount < nMaxSize) ) 2565 { 2566 pBuf[rCount] = *pReplace; 2567 rCount++; 2568 pReplace++; 2569 } 2570 } 2571 } 2572 else 2573 rCount = 0; 2574 } 2575 2576 // ----------------------------------------------------------------------- 2577 2578 XubString WinSalFrame::GetKeyName( sal_uInt16 nKeyCode ) 2579 { 2580 static const int nMaxKeyLen = 350; 2581 sal_Unicode aKeyBuf[ nMaxKeyLen ]; 2582 UINT nKeyBufLen = 0; 2583 UINT nSysCode = 0; 2584 2585 if ( nKeyCode & KEY_MOD1 ) 2586 { 2587 nSysCode = MapVirtualKey( VK_CONTROL, 0 ); 2588 nSysCode = (nSysCode << 16) | (((sal_uLong)1) << 25); 2589 ImplGetKeyNameText( nSysCode, aKeyBuf, nKeyBufLen, nMaxKeyLen, "Ctrl" ); 2590 } 2591 2592 if ( nKeyCode & KEY_MOD2 ) 2593 { 2594 nSysCode = MapVirtualKey( VK_MENU, 0 ); 2595 nSysCode = (nSysCode << 16) | (((sal_uLong)1) << 25); 2596 ImplGetKeyNameText( nSysCode, aKeyBuf, nKeyBufLen, nMaxKeyLen, "Alt" ); 2597 } 2598 2599 if ( nKeyCode & KEY_SHIFT ) 2600 { 2601 nSysCode = MapVirtualKey( VK_SHIFT, 0 ); 2602 nSysCode = (nSysCode << 16) | (((sal_uLong)1) << 25); 2603 ImplGetKeyNameText( nSysCode, aKeyBuf, nKeyBufLen, nMaxKeyLen, "Shift" ); 2604 } 2605 2606 sal_uInt16 nCode = nKeyCode & 0x0FFF; 2607 sal_uLong nSysCode2 = 0; 2608 sal_Char* pReplace = NULL; 2609 sal_Unicode cSVCode = 0; 2610 sal_Char aFBuf[4]; 2611 nSysCode = 0; 2612 2613 if ( (nCode >= KEY_0) && (nCode <= KEY_9) ) 2614 cSVCode = '0' + (nCode - KEY_0); 2615 else if ( (nCode >= KEY_A) && (nCode <= KEY_Z) ) 2616 cSVCode = 'A' + (nCode - KEY_A); 2617 else if ( (nCode >= KEY_F1) && (nCode <= KEY_F26) ) 2618 { 2619 nSysCode = VK_F1 + (nCode - KEY_F1); 2620 aFBuf[0] = 'F'; 2621 if ( (nCode >= KEY_F1) && (nCode <= KEY_F9) ) 2622 { 2623 aFBuf[1] = sal::static_int_cast<sal_Char>('1' + (nCode - KEY_F1)); 2624 aFBuf[2] = 0; 2625 } 2626 else if ( (nCode >= KEY_F10) && (nCode <= KEY_F19) ) 2627 { 2628 aFBuf[1] = '1'; 2629 aFBuf[2] = sal::static_int_cast<sal_Char>('0' + (nCode - KEY_F10)); 2630 aFBuf[3] = 0; 2631 } 2632 else 2633 { 2634 aFBuf[1] = '2'; 2635 aFBuf[2] = sal::static_int_cast<sal_Char>('0' + (nCode - KEY_F20)); 2636 aFBuf[3] = 0; 2637 } 2638 pReplace = aFBuf; 2639 } 2640 else 2641 { 2642 switch ( nCode ) 2643 { 2644 case KEY_DOWN: 2645 nSysCode = VK_DOWN; 2646 nSysCode2 = (((sal_uLong)1) << 24); 2647 pReplace = "Down"; 2648 break; 2649 case KEY_UP: 2650 nSysCode = VK_UP; 2651 nSysCode2 = (((sal_uLong)1) << 24); 2652 pReplace = "Up"; 2653 break; 2654 case KEY_LEFT: 2655 nSysCode = VK_LEFT; 2656 nSysCode2 = (((sal_uLong)1) << 24); 2657 pReplace = "Left"; 2658 break; 2659 case KEY_RIGHT: 2660 nSysCode = VK_RIGHT; 2661 nSysCode2 = (((sal_uLong)1) << 24); 2662 pReplace = "Right"; 2663 break; 2664 case KEY_HOME: 2665 nSysCode = VK_HOME; 2666 nSysCode2 = (((sal_uLong)1) << 24); 2667 pReplace = "Home"; 2668 break; 2669 case KEY_END: 2670 nSysCode = VK_END; 2671 nSysCode2 = (((sal_uLong)1) << 24); 2672 pReplace = "End"; 2673 break; 2674 case KEY_PAGEUP: 2675 nSysCode = VK_PRIOR; 2676 nSysCode2 = (((sal_uLong)1) << 24); 2677 pReplace = "Page Up"; 2678 break; 2679 case KEY_PAGEDOWN: 2680 nSysCode = VK_NEXT; 2681 nSysCode2 = (((sal_uLong)1) << 24); 2682 pReplace = "Page Down"; 2683 break; 2684 case KEY_RETURN: 2685 nSysCode = VK_RETURN; 2686 pReplace = "Enter"; 2687 break; 2688 case KEY_ESCAPE: 2689 nSysCode = VK_ESCAPE; 2690 pReplace = "Escape"; 2691 break; 2692 case KEY_TAB: 2693 nSysCode = VK_TAB; 2694 pReplace = "Tab"; 2695 break; 2696 case KEY_BACKSPACE: 2697 nSysCode = VK_BACK; 2698 pReplace = "Backspace"; 2699 break; 2700 case KEY_SPACE: 2701 nSysCode = VK_SPACE; 2702 pReplace = "Space"; 2703 break; 2704 case KEY_INSERT: 2705 nSysCode = VK_INSERT; 2706 nSysCode2 = (((sal_uLong)1) << 24); 2707 pReplace = "Insert"; 2708 break; 2709 case KEY_DELETE: 2710 nSysCode = VK_DELETE; 2711 nSysCode2 = (((sal_uLong)1) << 24); 2712 pReplace = "Delete"; 2713 break; 2714 2715 case KEY_ADD: 2716 cSVCode = '+'; 2717 break; 2718 case KEY_SUBTRACT: 2719 cSVCode = '-'; 2720 break; 2721 case KEY_MULTIPLY: 2722 cSVCode = '*'; 2723 break; 2724 case KEY_DIVIDE: 2725 cSVCode = '/'; 2726 break; 2727 case KEY_POINT: 2728 cSVCode = '.'; 2729 break; 2730 case KEY_COMMA: 2731 cSVCode = ','; 2732 break; 2733 case KEY_LESS: 2734 cSVCode = '<'; 2735 break; 2736 case KEY_GREATER: 2737 cSVCode = '>'; 2738 break; 2739 case KEY_EQUAL: 2740 cSVCode = '='; 2741 break; 2742 } 2743 } 2744 2745 if ( nSysCode ) 2746 { 2747 nSysCode = MapVirtualKey( (UINT)nSysCode, 0 ); 2748 if ( nSysCode ) 2749 nSysCode = (nSysCode << 16) | nSysCode2; 2750 ImplGetKeyNameText( nSysCode, aKeyBuf, nKeyBufLen, nMaxKeyLen, pReplace ); 2751 } 2752 else 2753 { 2754 if ( cSVCode ) 2755 { 2756 if ( nKeyBufLen > 0 ) 2757 aKeyBuf[ nKeyBufLen++ ] = '+'; 2758 if( nKeyBufLen < nMaxKeyLen ) 2759 aKeyBuf[ nKeyBufLen++ ] = cSVCode; 2760 } 2761 } 2762 2763 if( !nKeyBufLen ) 2764 return XubString(); 2765 2766 return XubString( aKeyBuf, sal::static_int_cast< sal_uInt16 >(nKeyBufLen) ); 2767 } 2768 2769 // ----------------------------------------------------------------------- 2770 2771 XubString WinSalFrame::GetSymbolKeyName( const XubString&, sal_uInt16 nKeyCode ) 2772 { 2773 return GetKeyName( nKeyCode ); 2774 } 2775 2776 // ----------------------------------------------------------------------- 2777 2778 inline Color ImplWinColorToSal( COLORREF nColor ) 2779 { 2780 return Color( GetRValue( nColor ), GetGValue( nColor ), GetBValue( nColor ) ); 2781 } 2782 2783 // ----------------------------------------------------------------------- 2784 2785 static void ImplSalUpdateStyleFontA( HDC hDC, const LOGFONTA& rLogFont, Font& rFont ) 2786 { 2787 ImplSalLogFontToFontA( hDC, rLogFont, rFont ); 2788 2789 // On Windows 9x, Windows NT we get sometimes very small sizes 2790 // (for example for the small Caption height). 2791 // So if it is MS Sans Serif, a none scalable font we use 2792 // 8 Point as the minimum control height, in all other cases 2793 // 6 Point is the smallest one 2794 if ( rFont.GetHeight() < 8 ) 2795 { 2796 if ( rtl_str_compareIgnoreAsciiCase( rLogFont.lfFaceName, "MS Sans Serif" ) == 0 ) 2797 rFont.SetHeight( 8 ); 2798 else if ( rFont.GetHeight() < 6 ) 2799 rFont.SetHeight( 6 ); 2800 } 2801 } 2802 2803 // ----------------------------------------------------------------------- 2804 2805 static void ImplSalUpdateStyleFontW( HDC hDC, const LOGFONTW& rLogFont, Font& rFont ) 2806 { 2807 ImplSalLogFontToFontW( hDC, rLogFont, rFont ); 2808 2809 // On Windows 9x, Windows NT we get sometimes very small sizes 2810 // (for example for the small Caption height). 2811 // So if it is MS Sans Serif, a none scalable font we use 2812 // 8 Point as the minimum control height, in all other cases 2813 // 6 Point is the smallest one 2814 if ( rFont.GetHeight() < 8 ) 2815 { 2816 if ( rtl_ustr_compareIgnoreAsciiCase( reinterpret_cast<const sal_Unicode*>(rLogFont.lfFaceName), reinterpret_cast<const sal_Unicode*>(L"MS Sans Serif") ) == 0 ) 2817 rFont.SetHeight( 8 ); 2818 else if ( rFont.GetHeight() < 6 ) 2819 rFont.SetHeight( 6 ); 2820 } 2821 } 2822 2823 // ----------------------------------------------------------------------- 2824 2825 static long ImplA2I( const BYTE* pStr ) 2826 { 2827 long n = 0; 2828 int nSign = 1; 2829 2830 if ( *pStr == '-' ) 2831 { 2832 nSign = -1; 2833 pStr++; 2834 } 2835 2836 while( (*pStr >= 48) && (*pStr <= 57) ) 2837 { 2838 n *= 10; 2839 n += ((*pStr) - 48); 2840 pStr++; 2841 } 2842 2843 n *= nSign; 2844 2845 return n; 2846 } 2847 2848 // ----------------------------------------------------------------------- 2849 static HRESULT WINAPI backwardCompatibleDwmIsCompositionEnabled( BOOL* pOut ) 2850 { 2851 *pOut = FALSE; 2852 return S_OK; 2853 } 2854 2855 static BOOL ImplDwmIsCompositionEnabled() 2856 { 2857 SalData* pSalData = GetSalData(); 2858 if( ! pSalData->mpDwmIsCompositionEnabled ) 2859 { 2860 pSalData->maDwmLib = osl_loadAsciiModule( "Dwmapi.dll", SAL_LOADMODULE_DEFAULT ); 2861 if( pSalData->maDwmLib ) 2862 pSalData->mpDwmIsCompositionEnabled = (DwmIsCompositionEnabled_ptr)osl_getAsciiFunctionSymbol( pSalData->maDwmLib, "DwmIsCompositionEnabled" ); 2863 if( ! pSalData->mpDwmIsCompositionEnabled ) // something failed 2864 pSalData->mpDwmIsCompositionEnabled = backwardCompatibleDwmIsCompositionEnabled; 2865 } 2866 BOOL aResult = FALSE; 2867 HRESULT nError = pSalData->mpDwmIsCompositionEnabled( &aResult ); 2868 return nError == S_OK && aResult; 2869 } 2870 2871 2872 void WinSalFrame::UpdateSettings( AllSettings& rSettings ) 2873 { 2874 MouseSettings aMouseSettings = rSettings.GetMouseSettings(); 2875 aMouseSettings.SetDoubleClickTime( GetDoubleClickTime() ); 2876 aMouseSettings.SetDoubleClickWidth( GetSystemMetrics( SM_CXDOUBLECLK ) ); 2877 aMouseSettings.SetDoubleClickHeight( GetSystemMetrics( SM_CYDOUBLECLK ) ); 2878 long nDragWidth = GetSystemMetrics( SM_CXDRAG ); 2879 long nDragHeight = GetSystemMetrics( SM_CYDRAG ); 2880 if ( nDragWidth ) 2881 aMouseSettings.SetStartDragWidth( nDragWidth ); 2882 if ( nDragHeight ) 2883 aMouseSettings.SetStartDragHeight( nDragHeight ); 2884 HKEY hRegKey; 2885 if ( RegOpenKey( HKEY_CURRENT_USER, 2886 "Control Panel\\Desktop", 2887 &hRegKey ) == ERROR_SUCCESS ) 2888 { 2889 BYTE aValueBuf[10]; 2890 DWORD nValueSize = sizeof( aValueBuf ); 2891 DWORD nType; 2892 if ( RegQueryValueEx( hRegKey, "MenuShowDelay", 0, 2893 &nType, aValueBuf, &nValueSize ) == ERROR_SUCCESS ) 2894 { 2895 if ( nType == REG_SZ ) 2896 aMouseSettings.SetMenuDelay( (sal_uLong)ImplA2I( aValueBuf ) ); 2897 } 2898 2899 RegCloseKey( hRegKey ); 2900 } 2901 2902 StyleSettings aStyleSettings = rSettings.GetStyleSettings(); 2903 // TODO: once those options vanish: just set bCompBorder to TRUE 2904 // to have the system colors read 2905 aStyleSettings.SetScrollBarSize( GetSystemMetrics( SM_CXVSCROLL ) ); 2906 aStyleSettings.SetSpinSize( GetSystemMetrics( SM_CXVSCROLL ) ); 2907 aStyleSettings.SetCursorBlinkTime( GetCaretBlinkTime() ); 2908 aStyleSettings.SetFloatTitleHeight( GetSystemMetrics( SM_CYSMCAPTION ) ); 2909 aStyleSettings.SetTitleHeight( GetSystemMetrics( SM_CYCAPTION ) ); 2910 aStyleSettings.SetActiveBorderColor( ImplWinColorToSal( GetSysColor( COLOR_ACTIVEBORDER ) ) ); 2911 aStyleSettings.SetDeactiveBorderColor( ImplWinColorToSal( GetSysColor( COLOR_INACTIVEBORDER ) ) ); 2912 if ( aSalShlData.mnVersion >= 410 ) 2913 { 2914 aStyleSettings.SetActiveColor2( ImplWinColorToSal( GetSysColor( COLOR_GRADIENTACTIVECAPTION ) ) ); 2915 aStyleSettings.SetDeactiveColor( ImplWinColorToSal( GetSysColor( COLOR_GRADIENTINACTIVECAPTION ) ) ); 2916 } 2917 aStyleSettings.SetFaceColor( ImplWinColorToSal( GetSysColor( COLOR_3DFACE ) ) ); 2918 aStyleSettings.SetInactiveTabColor( aStyleSettings.GetFaceColor() ); 2919 aStyleSettings.SetLightColor( ImplWinColorToSal( GetSysColor( COLOR_3DHILIGHT ) ) ); 2920 aStyleSettings.SetLightBorderColor( ImplWinColorToSal( GetSysColor( COLOR_3DLIGHT ) ) ); 2921 aStyleSettings.SetShadowColor( ImplWinColorToSal( GetSysColor( COLOR_3DSHADOW ) ) ); 2922 aStyleSettings.SetDarkShadowColor( ImplWinColorToSal( GetSysColor( COLOR_3DDKSHADOW ) ) ); 2923 aStyleSettings.SetWorkspaceColor( ImplWinColorToSal( GetSysColor( COLOR_APPWORKSPACE ) ) ); 2924 aStyleSettings.SetHelpColor( ImplWinColorToSal( GetSysColor( COLOR_INFOBK ) ) ); 2925 aStyleSettings.SetHelpTextColor( ImplWinColorToSal( GetSysColor( COLOR_INFOTEXT ) ) ); 2926 aStyleSettings.SetDialogColor( aStyleSettings.GetFaceColor() ); 2927 aStyleSettings.SetDialogTextColor( aStyleSettings.GetButtonTextColor() ); 2928 aStyleSettings.SetButtonTextColor( ImplWinColorToSal( GetSysColor( COLOR_BTNTEXT ) ) ); 2929 aStyleSettings.SetButtonRolloverTextColor( aStyleSettings.GetButtonTextColor() ); 2930 aStyleSettings.SetRadioCheckTextColor( ImplWinColorToSal( GetSysColor( COLOR_WINDOWTEXT ) ) ); 2931 aStyleSettings.SetGroupTextColor( aStyleSettings.GetRadioCheckTextColor() ); 2932 aStyleSettings.SetLabelTextColor( aStyleSettings.GetRadioCheckTextColor() ); 2933 aStyleSettings.SetInfoTextColor( aStyleSettings.GetRadioCheckTextColor() ); 2934 aStyleSettings.SetWindowColor( ImplWinColorToSal( GetSysColor( COLOR_WINDOW ) ) ); 2935 aStyleSettings.SetActiveTabColor( aStyleSettings.GetWindowColor() ); 2936 aStyleSettings.SetWindowTextColor( ImplWinColorToSal( GetSysColor( COLOR_WINDOWTEXT ) ) ); 2937 aStyleSettings.SetFieldColor( aStyleSettings.GetWindowColor() ); 2938 aStyleSettings.SetFieldTextColor( aStyleSettings.GetWindowTextColor() ); 2939 aStyleSettings.SetFieldRolloverTextColor( aStyleSettings.GetFieldTextColor() ); 2940 aStyleSettings.SetHighlightColor( ImplWinColorToSal( GetSysColor( COLOR_HIGHLIGHT ) ) ); 2941 aStyleSettings.SetHighlightTextColor( ImplWinColorToSal( GetSysColor( COLOR_HIGHLIGHTTEXT ) ) ); 2942 aStyleSettings.SetMenuHighlightColor( aStyleSettings.GetHighlightColor() ); 2943 aStyleSettings.SetMenuHighlightTextColor( aStyleSettings.GetHighlightTextColor() ); 2944 2945 ImplSVData* pSVData = ImplGetSVData(); 2946 pSVData->maNWFData.mnMenuFormatExtraBorder = 0; 2947 pSVData->maNWFData.maMenuBarHighlightTextColor = Color( COL_TRANSPARENT ); 2948 GetSalData()->mbThemeMenuSupport = FALSE; 2949 aStyleSettings.SetMenuColor( ImplWinColorToSal( GetSysColor( COLOR_MENU ) ) ); 2950 aStyleSettings.SetMenuBarColor( aStyleSettings.GetMenuColor() ); 2951 aStyleSettings.SetMenuBorderColor( aStyleSettings.GetLightBorderColor() ); // overriden below for flat menus 2952 aStyleSettings.SetUseFlatBorders( FALSE ); 2953 aStyleSettings.SetUseFlatMenues( FALSE ); 2954 aStyleSettings.SetMenuTextColor( ImplWinColorToSal( GetSysColor( COLOR_MENUTEXT ) ) ); 2955 aStyleSettings.SetMenuBarTextColor( ImplWinColorToSal( GetSysColor( COLOR_MENUTEXT ) ) ); 2956 aStyleSettings.SetActiveColor( ImplWinColorToSal( GetSysColor( COLOR_ACTIVECAPTION ) ) ); 2957 aStyleSettings.SetActiveTextColor( ImplWinColorToSal( GetSysColor( COLOR_CAPTIONTEXT ) ) ); 2958 aStyleSettings.SetDeactiveColor( ImplWinColorToSal( GetSysColor( COLOR_INACTIVECAPTION ) ) ); 2959 aStyleSettings.SetDeactiveTextColor( ImplWinColorToSal( GetSysColor( COLOR_INACTIVECAPTIONTEXT ) ) ); 2960 if ( aSalShlData.mbWXP ) 2961 { 2962 // only xp supports a different menu bar color 2963 long bFlatMenues = 0; 2964 SystemParametersInfo( SPI_GETFLATMENU, 0, &bFlatMenues, 0); 2965 if( bFlatMenues ) 2966 { 2967 aStyleSettings.SetUseFlatMenues( TRUE ); 2968 aStyleSettings.SetMenuBarColor( ImplWinColorToSal( GetSysColor( COLOR_MENUBAR ) ) ); 2969 aStyleSettings.SetMenuHighlightColor( ImplWinColorToSal( GetSysColor( COLOR_MENUHILIGHT ) ) ); 2970 aStyleSettings.SetMenuBorderColor( ImplWinColorToSal( GetSysColor( COLOR_3DSHADOW ) ) ); 2971 2972 // flat borders for our controls etc. as well in this mode (ie, no 3d borders) 2973 // this is not active in the classic style appearance 2974 aStyleSettings.SetUseFlatBorders( TRUE ); 2975 } 2976 } 2977 // check if vista or newer runs 2978 // in Aero theme (and similar ?) the menu text color does not change 2979 // for selected items; also on WinXP and earlier menus are not themed 2980 if( aSalShlData.maVersionInfo.dwMajorVersion >= 6 && 2981 ImplDwmIsCompositionEnabled() 2982 ) 2983 { 2984 // in aero menuitem highlight text is drawn in the same color as normal 2985 aStyleSettings.SetMenuHighlightTextColor( aStyleSettings.GetMenuTextColor() ); 2986 pSVData->maNWFData.mnMenuFormatExtraBorder = 2; 2987 pSVData->maNWFData.maMenuBarHighlightTextColor = aStyleSettings.GetMenuTextColor(); 2988 GetSalData()->mbThemeMenuSupport = TRUE; 2989 } 2990 // Bei hellgrau geben wir die Farbe vor, damit es besser aussieht 2991 if ( aStyleSettings.GetFaceColor() == COL_LIGHTGRAY ) 2992 aStyleSettings.SetCheckedColor( Color( 0xCC, 0xCC, 0xCC ) ); 2993 else 2994 { 2995 // Checked-Color berechnen 2996 Color aColor1 = aStyleSettings.GetFaceColor(); 2997 Color aColor2 = aStyleSettings.GetLightColor(); 2998 BYTE nRed = (BYTE)(((sal_uInt16)aColor1.GetRed() + (sal_uInt16)aColor2.GetRed())/2); 2999 BYTE nGreen = (BYTE)(((sal_uInt16)aColor1.GetGreen() + (sal_uInt16)aColor2.GetGreen())/2); 3000 BYTE nBlue = (BYTE)(((sal_uInt16)aColor1.GetBlue() + (sal_uInt16)aColor2.GetBlue())/2); 3001 aStyleSettings.SetCheckedColor( Color( nRed, nGreen, nBlue ) ); 3002 } 3003 3004 // caret width 3005 DWORD nCaretWidth = 2; 3006 if( SystemParametersInfo( SPI_GETCARETWIDTH, 0, &nCaretWidth, 0 ) ) 3007 aStyleSettings.SetCursorSize( nCaretWidth ); 3008 3009 // High contrast 3010 HIGHCONTRAST hc; 3011 hc.cbSize = sizeof( HIGHCONTRAST ); 3012 if( SystemParametersInfo( SPI_GETHIGHCONTRAST, hc.cbSize, &hc, 0) && (hc.dwFlags & HCF_HIGHCONTRASTON) ) 3013 aStyleSettings.SetHighContrastMode( 1 ); 3014 else 3015 aStyleSettings.SetHighContrastMode( 0 ); 3016 3017 3018 // Query Fonts 3019 Font aMenuFont = aStyleSettings.GetMenuFont(); 3020 Font aTitleFont = aStyleSettings.GetTitleFont(); 3021 Font aFloatTitleFont = aStyleSettings.GetFloatTitleFont(); 3022 Font aHelpFont = aStyleSettings.GetHelpFont(); 3023 Font aAppFont = aStyleSettings.GetAppFont(); 3024 Font aIconFont = aStyleSettings.GetIconFont(); 3025 HDC hDC = GetDC( 0 ); 3026 if( true/*aSalShlData.mbWNT*/ ) 3027 { 3028 NONCLIENTMETRICSW aNonClientMetrics; 3029 aNonClientMetrics.cbSize = sizeof( aNonClientMetrics ); 3030 if ( SystemParametersInfoW( SPI_GETNONCLIENTMETRICS, sizeof( aNonClientMetrics ), &aNonClientMetrics, 0 ) ) 3031 { 3032 ImplSalUpdateStyleFontW( hDC, aNonClientMetrics.lfMenuFont, aMenuFont ); 3033 ImplSalUpdateStyleFontW( hDC, aNonClientMetrics.lfCaptionFont, aTitleFont ); 3034 ImplSalUpdateStyleFontW( hDC, aNonClientMetrics.lfSmCaptionFont, aFloatTitleFont ); 3035 ImplSalUpdateStyleFontW( hDC, aNonClientMetrics.lfStatusFont, aHelpFont ); 3036 ImplSalUpdateStyleFontW( hDC, aNonClientMetrics.lfMessageFont, aAppFont ); 3037 3038 LOGFONTW aLogFont; 3039 if ( SystemParametersInfoW( SPI_GETICONTITLELOGFONT, 0, &aLogFont, 0 ) ) 3040 ImplSalUpdateStyleFontW( hDC, aLogFont, aIconFont ); 3041 } 3042 } 3043 3044 // get screen font resolution to calculate toolbox item size 3045 long nDPIY = GetDeviceCaps( hDC, LOGPIXELSY ); 3046 3047 ReleaseDC( 0, hDC ); 3048 3049 long nHeightPx = aMenuFont.GetHeight() * nDPIY / 72; 3050 aStyleSettings.SetToolbarIconSize( (((nHeightPx-1)*2) >= 28) ? STYLE_TOOLBAR_ICONSIZE_LARGE : STYLE_TOOLBAR_ICONSIZE_SMALL ); 3051 3052 aStyleSettings.SetMenuFont( aMenuFont ); 3053 aStyleSettings.SetTitleFont( aTitleFont ); 3054 aStyleSettings.SetFloatTitleFont( aFloatTitleFont ); 3055 aStyleSettings.SetHelpFont( aHelpFont ); 3056 aStyleSettings.SetIconFont( aIconFont ); 3057 // We prefer Arial in the russian version, because MS Sans Serif 3058 // is to wide for the dialogs 3059 if ( rSettings.GetLanguage() == LANGUAGE_RUSSIAN ) 3060 { 3061 XubString aFontName = aAppFont.GetName(); 3062 XubString aFirstName = aFontName.GetToken( 0, ';' ); 3063 if ( aFirstName.EqualsIgnoreCaseAscii( "MS Sans Serif" ) ) 3064 { 3065 aFontName.InsertAscii( "Arial;", 0 ); 3066 aAppFont.SetName( aFontName ); 3067 } 3068 } 3069 aStyleSettings.SetAppFont( aAppFont ); 3070 aStyleSettings.SetGroupFont( aAppFont ); 3071 aStyleSettings.SetLabelFont( aAppFont ); 3072 aStyleSettings.SetRadioCheckFont( aAppFont ); 3073 aStyleSettings.SetPushButtonFont( aAppFont ); 3074 aStyleSettings.SetFieldFont( aAppFont ); 3075 if ( aAppFont.GetWeight() > WEIGHT_NORMAL ) 3076 aAppFont.SetWeight( WEIGHT_NORMAL ); 3077 aStyleSettings.SetInfoFont( aAppFont ); 3078 aStyleSettings.SetToolFont( aAppFont ); 3079 3080 BOOL bDragFull; 3081 if ( SystemParametersInfo( SPI_GETDRAGFULLWINDOWS, 0, &bDragFull, 0 ) ) 3082 { 3083 sal_uLong nDragFullOptions = aStyleSettings.GetDragFullOptions(); 3084 if ( bDragFull ) 3085 nDragFullOptions |= DRAGFULL_OPTION_WINDOWMOVE | DRAGFULL_OPTION_WINDOWSIZE | DRAGFULL_OPTION_DOCKING | DRAGFULL_OPTION_SPLIT; 3086 else 3087 nDragFullOptions &= ~(DRAGFULL_OPTION_WINDOWMOVE | DRAGFULL_OPTION_WINDOWSIZE | DRAGFULL_OPTION_DOCKING | DRAGFULL_OPTION_SPLIT); 3088 aStyleSettings.SetDragFullOptions( nDragFullOptions ); 3089 } 3090 3091 aStyleSettings.SetIconHorzSpace( GetSystemMetrics( SM_CXICONSPACING ) ); 3092 aStyleSettings.SetIconVertSpace( GetSystemMetrics( SM_CYICONSPACING ) ); 3093 if ( RegOpenKey( HKEY_CURRENT_USER, 3094 "Control Panel\\International\\Calendars\\TwoDigitYearMax", 3095 &hRegKey ) == ERROR_SUCCESS ) 3096 { 3097 BYTE aValueBuf[10]; 3098 DWORD nValue; 3099 DWORD nValueSize = sizeof( aValueBuf ); 3100 DWORD nType; 3101 if ( RegQueryValueEx( hRegKey, "1", 0, 3102 &nType, aValueBuf, &nValueSize ) == ERROR_SUCCESS ) 3103 { 3104 if ( nType == REG_SZ ) 3105 { 3106 nValue = (sal_uLong)ImplA2I( aValueBuf ); 3107 if ( (nValue > 1000) && (nValue < 10000) ) 3108 { 3109 MiscSettings aMiscSettings = rSettings.GetMiscSettings(); 3110 utl::MiscCfg().SetYear2000( (sal_Int32)(nValue-99) ); 3111 rSettings.SetMiscSettings( aMiscSettings ); 3112 } 3113 } 3114 } 3115 3116 RegCloseKey( hRegKey ); 3117 } 3118 3119 rSettings.SetMouseSettings( aMouseSettings ); 3120 rSettings.SetStyleSettings( aStyleSettings ); 3121 } 3122 3123 // ----------------------------------------------------------------------- 3124 3125 SalBitmap* WinSalFrame::SnapShot() 3126 { 3127 WinSalBitmap* pSalBitmap = NULL; 3128 3129 RECT aRect; 3130 GetWindowRect( mhWnd, &aRect ); 3131 3132 int nDX = aRect.right-aRect.left; 3133 int nDY = aRect.bottom-aRect.top; 3134 HDC hDC = GetWindowDC( mhWnd ); 3135 HBITMAP hBmpBitmap = CreateCompatibleBitmap( hDC, nDX, nDY ); 3136 HDC hBmpDC = ImplGetCachedDC( CACHED_HDC_1, hBmpBitmap ); 3137 sal_Bool bRet; 3138 3139 bRet = BitBlt( hBmpDC, 0, 0, nDX, nDY, hDC, 0, 0, SRCCOPY ) ? TRUE : FALSE; 3140 ImplReleaseCachedDC( CACHED_HDC_1 ); 3141 3142 if ( bRet ) 3143 { 3144 pSalBitmap = new WinSalBitmap; 3145 3146 if ( !pSalBitmap->Create( hBmpBitmap, FALSE, FALSE ) ) 3147 { 3148 delete pSalBitmap; 3149 pSalBitmap = NULL; 3150 } 3151 } 3152 3153 return pSalBitmap; 3154 } 3155 3156 // ----------------------------------------------------------------------- 3157 3158 const SystemEnvData* WinSalFrame::GetSystemData() const 3159 { 3160 return &maSysData; 3161 } 3162 3163 // ----------------------------------------------------------------------- 3164 3165 void WinSalFrame::Beep( SoundType eSoundType ) 3166 { 3167 static UINT aImplSoundTab[5] = 3168 { 3169 0, // SOUND_DEFAULT 3170 MB_ICONASTERISK, // SOUND_INFO 3171 MB_ICONEXCLAMATION, // SOUND_WARNING 3172 MB_ICONHAND, // SOUND_ERROR 3173 MB_ICONQUESTION // SOUND_QUERY 3174 }; 3175 3176 if( eSoundType != SOUND_DISABLE ) // don't beep on disable 3177 MessageBeep( aImplSoundTab[eSoundType] ); 3178 } 3179 3180 // ----------------------------------------------------------------------- 3181 3182 SalFrame::SalPointerState WinSalFrame::GetPointerState() 3183 { 3184 SalPointerState aState; 3185 aState.mnState = 0; 3186 3187 if ( GetKeyState( VK_LBUTTON ) & 0x8000 ) 3188 aState.mnState |= MOUSE_LEFT; 3189 if ( GetKeyState( VK_MBUTTON ) & 0x8000 ) 3190 aState.mnState |= MOUSE_MIDDLE; 3191 if ( GetKeyState( VK_RBUTTON ) & 0x8000 ) 3192 aState.mnState |= MOUSE_RIGHT; 3193 if ( GetKeyState( VK_SHIFT ) & 0x8000 ) 3194 aState.mnState |= KEY_SHIFT; 3195 if ( GetKeyState( VK_CONTROL ) & 0x8000 ) 3196 aState.mnState |= KEY_MOD1; 3197 if ( GetKeyState( VK_MENU ) & 0x8000 ) 3198 aState.mnState |= KEY_MOD2; 3199 3200 POINT pt; 3201 GetCursorPos( &pt ); 3202 3203 aState.maPos = Point( pt.x - maGeometry.nX, pt.y - maGeometry.nY ); 3204 return aState; 3205 } 3206 3207 // ----------------------------------------------------------------------- 3208 3209 void WinSalFrame::SetBackgroundBitmap( SalBitmap* ) 3210 { 3211 } 3212 3213 // ----------------------------------------------------------------------- 3214 3215 void WinSalFrame::ResetClipRegion() 3216 { 3217 SetWindowRgn( mhWnd, 0, TRUE ); 3218 } 3219 3220 // ----------------------------------------------------------------------- 3221 3222 void WinSalFrame::BeginSetClipRegion( sal_uLong nRects ) 3223 { 3224 if( mpClipRgnData ) 3225 delete [] (BYTE*)mpClipRgnData; 3226 sal_uLong nRectBufSize = sizeof(RECT)*nRects; 3227 mpClipRgnData = (RGNDATA*)new BYTE[sizeof(RGNDATA)-1+nRectBufSize]; 3228 mpClipRgnData->rdh.dwSize = sizeof( RGNDATAHEADER ); 3229 mpClipRgnData->rdh.iType = RDH_RECTANGLES; 3230 mpClipRgnData->rdh.nCount = nRects; 3231 mpClipRgnData->rdh.nRgnSize = nRectBufSize; 3232 SetRectEmpty( &(mpClipRgnData->rdh.rcBound) ); 3233 mpNextClipRect = (RECT*)(&(mpClipRgnData->Buffer)); 3234 mbFirstClipRect = TRUE; 3235 } 3236 3237 // ----------------------------------------------------------------------- 3238 3239 void WinSalFrame::UnionClipRegion( long nX, long nY, long nWidth, long nHeight ) 3240 { 3241 if( ! mpClipRgnData ) 3242 return; 3243 3244 RECT* pRect = mpNextClipRect; 3245 RECT* pBoundRect = &(mpClipRgnData->rdh.rcBound); 3246 long nRight = nX + nWidth; 3247 long nBottom = nY + nHeight; 3248 3249 if ( mbFirstClipRect ) 3250 { 3251 pBoundRect->left = nX; 3252 pBoundRect->top = nY; 3253 pBoundRect->right = nRight; 3254 pBoundRect->bottom = nBottom; 3255 mbFirstClipRect = FALSE; 3256 } 3257 else 3258 { 3259 if ( nX < pBoundRect->left ) 3260 pBoundRect->left = (int)nX; 3261 3262 if ( nY < pBoundRect->top ) 3263 pBoundRect->top = (int)nY; 3264 3265 if ( nRight > pBoundRect->right ) 3266 pBoundRect->right = (int)nRight; 3267 3268 if ( nBottom > pBoundRect->bottom ) 3269 pBoundRect->bottom = (int)nBottom; 3270 } 3271 3272 pRect->left = (int)nX; 3273 pRect->top = (int)nY; 3274 pRect->right = (int)nRight; 3275 pRect->bottom = (int)nBottom; 3276 if( (mpNextClipRect - (RECT*)(&mpClipRgnData->Buffer)) < (int)mpClipRgnData->rdh.nCount ) 3277 mpNextClipRect++; 3278 } 3279 3280 // ----------------------------------------------------------------------- 3281 3282 void WinSalFrame::EndSetClipRegion() 3283 { 3284 if( ! mpClipRgnData ) 3285 return; 3286 3287 HRGN hRegion; 3288 3289 // create region from accumulated rectangles 3290 if ( mpClipRgnData->rdh.nCount == 1 ) 3291 { 3292 RECT* pRect = &(mpClipRgnData->rdh.rcBound); 3293 hRegion = CreateRectRgn( pRect->left, pRect->top, 3294 pRect->right, pRect->bottom ); 3295 } 3296 else 3297 { 3298 sal_uLong nSize = mpClipRgnData->rdh.nRgnSize+sizeof(RGNDATAHEADER); 3299 hRegion = ExtCreateRegion( NULL, nSize, mpClipRgnData ); 3300 } 3301 delete [] (BYTE*)mpClipRgnData; 3302 mpClipRgnData = NULL; 3303 3304 DBG_ASSERT( hRegion, "WinSalFrame::EndSetClipRegion() - Can't create ClipRegion" ); 3305 if( hRegion ) 3306 { 3307 RECT aWindowRect; 3308 GetWindowRect( mhWnd, &aWindowRect ); 3309 POINT aPt; 3310 aPt.x=0; 3311 aPt.y=0; 3312 ClientToScreen( mhWnd, &aPt ); 3313 OffsetRgn( hRegion, aPt.x - aWindowRect.left, aPt.y - aWindowRect.top ); 3314 3315 if( SetWindowRgn( mhWnd, hRegion, TRUE ) == 0 ) 3316 DeleteObject( hRegion ); 3317 } 3318 } 3319 3320 // ----------------------------------------------------------------------- 3321 3322 static long ImplHandleMouseMsg( HWND hWnd, UINT nMsg, 3323 WPARAM wParam, LPARAM lParam ) 3324 { 3325 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 3326 if ( !pFrame ) 3327 return 0; 3328 3329 if( nMsg == WM_LBUTTONDOWN || nMsg == WM_MBUTTONDOWN || nMsg == WM_RBUTTONDOWN ) 3330 { 3331 // #103168# post again if async focus has not arrived yet 3332 // hopefully we will not receive the corresponding button up before this 3333 // button down arrives again 3334 Window *pWin = pFrame->GetWindow(); 3335 if( pWin && pWin->ImplGetWindowImpl()->mpFrameData->mnFocusId ) 3336 { 3337 ImplPostMessage( hWnd, nMsg, wParam, lParam ); 3338 return 1; 3339 } 3340 } 3341 SalMouseEvent aMouseEvt; 3342 long nRet; 3343 sal_uInt16 nEvent = 0; 3344 sal_Bool bCall = TRUE; 3345 3346 aMouseEvt.mnX = (short)LOWORD( lParam ); 3347 aMouseEvt.mnY = (short)HIWORD( lParam ); 3348 aMouseEvt.mnCode = 0; 3349 aMouseEvt.mnTime = GetMessageTime(); 3350 3351 // Wegen (Logitech-)MouseTreiber ueber GetKeyState() gehen, die auf 3352 // mittlerer Maustaste Doppelklick simulieren und den KeyStatus nicht 3353 // beruecksichtigen 3354 3355 if ( GetKeyState( VK_LBUTTON ) & 0x8000 ) 3356 aMouseEvt.mnCode |= MOUSE_LEFT; 3357 if ( GetKeyState( VK_MBUTTON ) & 0x8000 ) 3358 aMouseEvt.mnCode |= MOUSE_MIDDLE; 3359 if ( GetKeyState( VK_RBUTTON ) & 0x8000 ) 3360 aMouseEvt.mnCode |= MOUSE_RIGHT; 3361 if ( GetKeyState( VK_SHIFT ) & 0x8000 ) 3362 aMouseEvt.mnCode |= KEY_SHIFT; 3363 if ( GetKeyState( VK_CONTROL ) & 0x8000 ) 3364 aMouseEvt.mnCode |= KEY_MOD1; 3365 if ( GetKeyState( VK_MENU ) & 0x8000 ) 3366 aMouseEvt.mnCode |= KEY_MOD2; 3367 3368 switch ( nMsg ) 3369 { 3370 case WM_MOUSEMOVE: 3371 { 3372 // Da bei Druecken von Modifier-Tasten die MouseEvents 3373 // nicht zusammengefast werden (da diese durch KeyEvents 3374 // unterbrochen werden), machen wir dieses hier selber 3375 if ( aMouseEvt.mnCode & (KEY_SHIFT | KEY_MOD1 | KEY_MOD2) ) 3376 { 3377 MSG aTempMsg; 3378 if ( ImplPeekMessage( &aTempMsg, hWnd, WM_MOUSEFIRST, WM_MOUSELAST, PM_NOREMOVE | PM_NOYIELD ) ) 3379 { 3380 if ( (aTempMsg.message == WM_MOUSEMOVE) && 3381 (aTempMsg.wParam == wParam) ) 3382 return 1; 3383 } 3384 } 3385 3386 SalData* pSalData = GetSalData(); 3387 // Test for MouseLeave 3388 if ( pSalData->mhWantLeaveMsg && (pSalData->mhWantLeaveMsg != hWnd) ) 3389 ImplSendMessage( pSalData->mhWantLeaveMsg, SAL_MSG_MOUSELEAVE, 0, GetMessagePos() ); 3390 3391 pSalData->mhWantLeaveMsg = hWnd; 3392 // Start MouseLeave-Timer 3393 if ( !pSalData->mpMouseLeaveTimer ) 3394 { 3395 pSalData->mpMouseLeaveTimer = new AutoTimer; 3396 pSalData->mpMouseLeaveTimer->SetTimeout( SAL_MOUSELEAVE_TIMEOUT ); 3397 pSalData->mpMouseLeaveTimer->Start(); 3398 // We dont need to set a timeout handler, because we test 3399 // for mouseleave in the timeout callback 3400 } 3401 aMouseEvt.mnButton = 0; 3402 nEvent = SALEVENT_MOUSEMOVE; 3403 } 3404 break; 3405 3406 case WM_NCMOUSEMOVE: 3407 case SAL_MSG_MOUSELEAVE: 3408 { 3409 SalData* pSalData = GetSalData(); 3410 if ( pSalData->mhWantLeaveMsg == hWnd ) 3411 { 3412 pSalData->mhWantLeaveMsg = 0; 3413 if ( pSalData->mpMouseLeaveTimer ) 3414 { 3415 delete pSalData->mpMouseLeaveTimer; 3416 pSalData->mpMouseLeaveTimer = NULL; 3417 } 3418 // Mouse-Coordinaates are relativ to the screen 3419 POINT aPt; 3420 aPt.x = (short)LOWORD( lParam ); 3421 aPt.y = (short)HIWORD( lParam ); 3422 ScreenToClient( hWnd, &aPt ); 3423 aMouseEvt.mnX = aPt.x; 3424 aMouseEvt.mnY = aPt.y; 3425 aMouseEvt.mnButton = 0; 3426 nEvent = SALEVENT_MOUSELEAVE; 3427 } 3428 else 3429 bCall = FALSE; 3430 } 3431 break; 3432 3433 case WM_LBUTTONDOWN: 3434 aMouseEvt.mnButton = MOUSE_LEFT; 3435 nEvent = SALEVENT_MOUSEBUTTONDOWN; 3436 break; 3437 3438 case WM_MBUTTONDOWN: 3439 aMouseEvt.mnButton = MOUSE_MIDDLE; 3440 nEvent = SALEVENT_MOUSEBUTTONDOWN; 3441 break; 3442 3443 case WM_RBUTTONDOWN: 3444 aMouseEvt.mnButton = MOUSE_RIGHT; 3445 nEvent = SALEVENT_MOUSEBUTTONDOWN; 3446 break; 3447 3448 case WM_LBUTTONUP: 3449 aMouseEvt.mnButton = MOUSE_LEFT; 3450 nEvent = SALEVENT_MOUSEBUTTONUP; 3451 break; 3452 3453 case WM_MBUTTONUP: 3454 aMouseEvt.mnButton = MOUSE_MIDDLE; 3455 nEvent = SALEVENT_MOUSEBUTTONUP; 3456 break; 3457 3458 case WM_RBUTTONUP: 3459 aMouseEvt.mnButton = MOUSE_RIGHT; 3460 nEvent = SALEVENT_MOUSEBUTTONUP; 3461 break; 3462 } 3463 3464 // check if this window was destroyed - this might happen if we are the help window 3465 // and sent a mouse leave message to the application which killed the help window, ie ourself 3466 if( !IsWindow( hWnd ) ) 3467 return 0; 3468 3469 if ( bCall ) 3470 { 3471 if ( nEvent == SALEVENT_MOUSEBUTTONDOWN ) 3472 UpdateWindow( hWnd ); 3473 3474 // --- RTL --- (mirror mouse pos) 3475 if( Application::GetSettings().GetLayoutRTL() ) 3476 aMouseEvt.mnX = pFrame->maGeometry.nWidth-1-aMouseEvt.mnX; 3477 3478 nRet = pFrame->CallCallback( nEvent, &aMouseEvt ); 3479 if ( nMsg == WM_MOUSEMOVE ) 3480 SetCursor( pFrame->mhCursor ); 3481 } 3482 else 3483 nRet = 0; 3484 3485 return nRet; 3486 } 3487 3488 // ----------------------------------------------------------------------- 3489 3490 static long ImplHandleMouseActivateMsg( HWND hWnd ) 3491 { 3492 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 3493 if ( !pFrame ) 3494 return 0; 3495 3496 if ( pFrame->mbFloatWin ) 3497 return TRUE; 3498 3499 SalMouseActivateEvent aMouseActivateEvt; 3500 POINT aPt; 3501 GetCursorPos( &aPt ); 3502 ScreenToClient( hWnd, &aPt ); 3503 aMouseActivateEvt.mnX = aPt.x; 3504 aMouseActivateEvt.mnY = aPt.y; 3505 return pFrame->CallCallback( SALEVENT_MOUSEACTIVATE, &aMouseActivateEvt ); 3506 } 3507 3508 // ----------------------------------------------------------------------- 3509 3510 static long ImplHandleWheelMsg( HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam ) 3511 { 3512 DBG_ASSERT( nMsg == WM_MOUSEWHEEL || 3513 nMsg == WM_MOUSEHWHEEL, 3514 "ImplHandleWheelMsg() called with no wheel mouse event" ); 3515 3516 ImplSalYieldMutexAcquireWithWait(); 3517 3518 long nRet = 0; 3519 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 3520 if ( pFrame ) 3521 { 3522 WORD nWinModCode = LOWORD( wParam ); 3523 POINT aWinPt; 3524 aWinPt.x = (short)LOWORD( lParam ); 3525 aWinPt.y = (short)HIWORD( lParam ); 3526 ScreenToClient( hWnd, &aWinPt ); 3527 3528 SalWheelMouseEvent aWheelEvt; 3529 aWheelEvt.mnTime = GetMessageTime(); 3530 aWheelEvt.mnX = aWinPt.x; 3531 aWheelEvt.mnY = aWinPt.y; 3532 aWheelEvt.mnCode = 0; 3533 aWheelEvt.mnDelta = (short)HIWORD( wParam ); 3534 aWheelEvt.mnNotchDelta = aWheelEvt.mnDelta/WHEEL_DELTA; 3535 if( aWheelEvt.mnNotchDelta == 0 ) 3536 { 3537 if( aWheelEvt.mnDelta > 0 ) 3538 aWheelEvt.mnNotchDelta = 1; 3539 else if( aWheelEvt.mnDelta < 0 ) 3540 aWheelEvt.mnNotchDelta = -1; 3541 } 3542 3543 if( nMsg == WM_MOUSEWHEEL ) 3544 { 3545 if ( aSalShlData.mnWheelScrollLines == WHEEL_PAGESCROLL ) 3546 aWheelEvt.mnScrollLines = SAL_WHEELMOUSE_EVENT_PAGESCROLL; 3547 else 3548 aWheelEvt.mnScrollLines = aSalShlData.mnWheelScrollLines; 3549 aWheelEvt.mbHorz = FALSE; 3550 } 3551 else 3552 { 3553 aWheelEvt.mnScrollLines = aSalShlData.mnWheelScrollChars; 3554 aWheelEvt.mbHorz = TRUE; 3555 } 3556 3557 if ( nWinModCode & MK_SHIFT ) 3558 aWheelEvt.mnCode |= KEY_SHIFT; 3559 if ( nWinModCode & MK_CONTROL ) 3560 aWheelEvt.mnCode |= KEY_MOD1; 3561 if ( GetKeyState( VK_MENU ) & 0x8000 ) 3562 aWheelEvt.mnCode |= KEY_MOD2; 3563 3564 // --- RTL --- (mirror mouse pos) 3565 if( Application::GetSettings().GetLayoutRTL() ) 3566 aWheelEvt.mnX = pFrame->maGeometry.nWidth-1-aWheelEvt.mnX; 3567 3568 nRet = pFrame->CallCallback( SALEVENT_WHEELMOUSE, &aWheelEvt ); 3569 } 3570 3571 ImplSalYieldMutexRelease(); 3572 3573 return nRet; 3574 } 3575 3576 // ----------------------------------------------------------------------- 3577 3578 static sal_uInt16 ImplSalGetKeyCode( WPARAM wParam ) 3579 { 3580 sal_uInt16 nKeyCode; 3581 3582 // convert KeyCode 3583 if ( wParam < KEY_TAB_SIZE ) 3584 nKeyCode = aImplTranslateKeyTab[wParam]; 3585 else 3586 { 3587 SalData* pSalData = GetSalData(); 3588 std::map< UINT, sal_uInt16 >::const_iterator it = pSalData->maVKMap.find( (UINT)wParam ); 3589 if( it != pSalData->maVKMap.end() ) 3590 nKeyCode = it->second; 3591 else 3592 nKeyCode = 0; 3593 } 3594 3595 return nKeyCode; 3596 } 3597 3598 // ----------------------------------------------------------------------- 3599 3600 static UINT ImplStrToNum( const sal_Char* pStr ) 3601 { 3602 sal_uInt16 n = 0; 3603 3604 // Solange es sich um eine Ziffer handelt, String umwandeln 3605 while( (*pStr >= 48) && (*pStr <= 57) ) 3606 { 3607 n *= 10; 3608 n += ((*pStr) - 48); 3609 pStr++; 3610 } 3611 3612 return n; 3613 } 3614 3615 // ----------------------------------------------------------------------- 3616 3617 static void ImplUpdateInputLang( WinSalFrame* pFrame ) 3618 { 3619 sal_Bool bLanguageChange = FALSE; 3620 UINT nLang = LOWORD( GetKeyboardLayout( 0 ) ); 3621 if ( nLang && nLang != pFrame->mnInputLang ) 3622 { 3623 // keep input lang up-to-date 3624 pFrame->mnInputLang = nLang; 3625 bLanguageChange = TRUE; 3626 } 3627 3628 // If we are on Windows NT we use Unicode FrameProcs and so we 3629 // get Unicode charcodes directly from Windows 3630 // no need to set up a code page 3631 return; 3632 } 3633 3634 3635 static sal_Unicode ImplGetCharCode( WinSalFrame* pFrame, WPARAM nCharCode ) 3636 { 3637 ImplUpdateInputLang( pFrame ); 3638 3639 // If we are on Windows NT we use Unicode FrameProcs and so we 3640 // get Unicode charcodes directly from Windows 3641 return (sal_Unicode)nCharCode; 3642 } 3643 3644 // ----------------------------------------------------------------------- 3645 3646 LanguageType WinSalFrame::GetInputLanguage() 3647 { 3648 if( !mnInputLang ) 3649 ImplUpdateInputLang( this ); 3650 3651 if( !mnInputLang ) 3652 return LANGUAGE_DONTKNOW; 3653 else 3654 return (LanguageType) mnInputLang; 3655 } 3656 3657 // ----------------------------------------------------------------------- 3658 3659 sal_Bool WinSalFrame::MapUnicodeToKeyCode( sal_Unicode aUnicode, LanguageType aLangType, KeyCode& rKeyCode ) 3660 { 3661 sal_Bool bRet = FALSE; 3662 HKL hkl = 0; 3663 3664 // just use the passed language identifier, do not try to load additional keyboard support 3665 hkl = (HKL) aLangType; 3666 3667 if( hkl ) 3668 { 3669 SHORT scan = VkKeyScanExW( aUnicode, hkl ); 3670 if( LOWORD(scan) == 0xFFFF ) 3671 // keyboard not loaded or key cannot be mapped 3672 bRet = FALSE; 3673 else 3674 { 3675 BYTE vkeycode = LOBYTE(scan); 3676 BYTE shiftstate = HIBYTE(scan); 3677 3678 // Last argument is set to FALSE, because there's no decission made 3679 // yet which key should be assigned to MOD3 modifier on Windows. 3680 // Windows key - user's can be confused, because it should display 3681 // Windows menu (applies to both left/right key) 3682 // Menu key - this key is used to display context menu 3683 // AltGr key - probably it has no sense 3684 rKeyCode = KeyCode( ImplSalGetKeyCode( vkeycode ), 3685 (shiftstate & 0x01) ? TRUE : FALSE, // shift 3686 (shiftstate & 0x02) ? TRUE : FALSE, // ctrl 3687 (shiftstate & 0x04) ? TRUE : FALSE, // alt 3688 FALSE ); 3689 bRet = TRUE; 3690 } 3691 } 3692 3693 return bRet; 3694 } 3695 3696 // ----------------------------------------------------------------------- 3697 3698 static long ImplHandleKeyMsg( HWND hWnd, UINT nMsg, 3699 WPARAM wParam, LPARAM lParam, LRESULT& rResult ) 3700 { 3701 static sal_Bool bIgnoreCharMsg = FALSE; 3702 static WPARAM nDeadChar = 0; 3703 static WPARAM nLastVKChar = 0; 3704 static sal_uInt16 nLastChar = 0; 3705 static sal_uInt16 nLastModKeyCode = 0; 3706 static bool bWaitForModKeyRelease = false; 3707 sal_uInt16 nRepeat = LOWORD( lParam )-1; 3708 sal_uInt16 nModCode = 0; 3709 3710 // Key wurde evtl. durch SysChild an uns weitergeleitet und 3711 // darf somit dann nicht doppelt verarbeitet werden 3712 GetSalData()->mnSalObjWantKeyEvt = 0; 3713 3714 if ( nMsg == WM_DEADCHAR ) 3715 { 3716 nDeadChar = wParam; 3717 return 0; 3718 } 3719 3720 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 3721 if ( !pFrame ) 3722 return 0; 3723 3724 // Wir restaurieren den Background-Modus bei jeder Texteingabe, 3725 // da einige Tools wie RichWin uns diesen hin- und wieder umsetzen 3726 if ( pFrame->mpGraphics && 3727 pFrame->mpGraphics->mhDC ) 3728 SetBkMode( pFrame->mpGraphics->mhDC, TRANSPARENT ); 3729 3730 // determine modifiers 3731 if ( GetKeyState( VK_SHIFT ) & 0x8000 ) 3732 nModCode |= KEY_SHIFT; 3733 if ( GetKeyState( VK_CONTROL ) & 0x8000 ) 3734 nModCode |= KEY_MOD1; 3735 if ( GetKeyState( VK_MENU ) & 0x8000 ) 3736 nModCode |= KEY_MOD2; 3737 3738 if ( (nMsg == WM_CHAR) || (nMsg == WM_SYSCHAR) ) 3739 { 3740 nDeadChar = 0; 3741 3742 if ( bIgnoreCharMsg ) 3743 { 3744 bIgnoreCharMsg = FALSE; 3745 // #101635# if zero is returned here for WM_SYSCHAR (ALT+<key>) Windows will beep 3746 // becaus this 'hotkey' was not processed -> better return 1 3747 // except for Alt-SPACE which should always open the sysmenu (#104616#) 3748 3749 // also return zero if a system menubar is available that might process this hotkey 3750 // this also applies to the OLE inplace embedding where we are a child window 3751 if( (GetWindowStyle( hWnd ) & WS_CHILD) || GetMenu( hWnd ) || (wParam == 0x20) ) 3752 return 0; 3753 else 3754 return 1; 3755 } 3756 3757 // Backspace ignorieren wir als eigenstaendige Taste, 3758 // damit wir keine Probleme in Kombination mit einem 3759 // DeadKey bekommen 3760 if ( wParam == 0x08 ) // BACKSPACE 3761 return 0; 3762 3763 // Hier kommen nur "freifliegende" WM_CHAR Message an, die durch 3764 // eintippen einer ALT-NUMPAD Kombination erzeugt wurden 3765 SalKeyEvent aKeyEvt; 3766 3767 if ( (wParam >= '0') && (wParam <= '9') ) 3768 aKeyEvt.mnCode = sal::static_int_cast<sal_uInt16>(KEYGROUP_NUM + wParam - '0'); 3769 else if ( (wParam >= 'A') && (wParam <= 'Z') ) 3770 aKeyEvt.mnCode = sal::static_int_cast<sal_uInt16>(KEYGROUP_ALPHA + wParam - 'A'); 3771 else if ( (wParam >= 'a') && (wParam <= 'z') ) 3772 aKeyEvt.mnCode = sal::static_int_cast<sal_uInt16>(KEYGROUP_ALPHA + wParam - 'a'); 3773 else if ( wParam == 0x0D ) // RETURN 3774 aKeyEvt.mnCode = KEY_RETURN; 3775 else if ( wParam == 0x1B ) // ESCAPE 3776 aKeyEvt.mnCode = KEY_ESCAPE; 3777 else if ( wParam == 0x09 ) // TAB 3778 aKeyEvt.mnCode = KEY_TAB; 3779 else if ( wParam == 0x20 ) // SPACE 3780 aKeyEvt.mnCode = KEY_SPACE; 3781 else 3782 aKeyEvt.mnCode = 0; 3783 3784 aKeyEvt.mnTime = GetMessageTime(); 3785 aKeyEvt.mnCode |= nModCode; 3786 aKeyEvt.mnCharCode = ImplGetCharCode( pFrame, wParam ); 3787 aKeyEvt.mnRepeat = nRepeat; 3788 nLastChar = 0; 3789 nLastVKChar = 0; 3790 long nRet = pFrame->CallCallback( SALEVENT_KEYINPUT, &aKeyEvt ); 3791 pFrame->CallCallback( SALEVENT_KEYUP, &aKeyEvt ); 3792 return nRet; 3793 } 3794 // #i11583#, MCD, 2003-01-13, Support for WM_UNICHAR & Keyman 6.0; addition begins 3795 else if( nMsg == WM_UNICHAR ) 3796 { 3797 // If Windows is asking if we accept WM_UNICHAR, return TRUE 3798 if(wParam == UNICODE_NOCHAR) 3799 { 3800 rResult = TRUE; // ssa: this will actually return TRUE to windows 3801 return 1; // ...but this will only avoid calling the defwindowproc 3802 } 3803 3804 SalKeyEvent aKeyEvt; 3805 aKeyEvt.mnCode = nModCode; // Or should it be 0? - as this is always a character returned 3806 aKeyEvt.mnTime = GetMessageTime(); 3807 aKeyEvt.mnRepeat = 0; 3808 3809 if( wParam >= Uni_SupplementaryPlanesStart ) 3810 { 3811 // character is supplementary char in UTF-32 format - must be converted to UTF-16 supplementary pair 3812 // sal_Unicode ch = (sal_Unicode) Uni_UTF32ToSurrogate1(wParam); 3813 nLastChar = 0; 3814 nLastVKChar = 0; 3815 pFrame->CallCallback( SALEVENT_KEYINPUT, &aKeyEvt ); 3816 pFrame->CallCallback( SALEVENT_KEYUP, &aKeyEvt ); 3817 wParam = (sal_Unicode) Uni_UTF32ToSurrogate2( wParam ); 3818 } 3819 3820 aKeyEvt.mnCharCode = (sal_Unicode) wParam; 3821 3822 nLastChar = 0; 3823 nLastVKChar = 0; 3824 long nRet = pFrame->CallCallback( SALEVENT_KEYINPUT, &aKeyEvt ); 3825 pFrame->CallCallback( SALEVENT_KEYUP, &aKeyEvt ); 3826 3827 return nRet; 3828 } 3829 // MCD, 2003-01-13, Support for WM_UNICHAR & Keyman 6.0; addition ends 3830 else 3831 { 3832 // Bei Shift, Control und Menu schicken wir einen KeyModChange-Event 3833 if ( (wParam == VK_SHIFT) || (wParam == VK_CONTROL) || (wParam == VK_MENU) ) 3834 { 3835 SalKeyModEvent aModEvt; 3836 aModEvt.mnTime = GetMessageTime(); 3837 aModEvt.mnCode = nModCode; 3838 aModEvt.mnModKeyCode = 0; // no command events will be sent if this member is 0 3839 3840 sal_uInt16 tmpCode = 0; 3841 if( GetKeyState( VK_LSHIFT ) & 0x8000 ) 3842 tmpCode |= MODKEY_LSHIFT; 3843 if( GetKeyState( VK_RSHIFT ) & 0x8000 ) 3844 tmpCode |= MODKEY_RSHIFT; 3845 if( GetKeyState( VK_LCONTROL ) & 0x8000 ) 3846 tmpCode |= MODKEY_LMOD1; 3847 if( GetKeyState( VK_RCONTROL ) & 0x8000 ) 3848 tmpCode |= MODKEY_RMOD1; 3849 if( GetKeyState( VK_LMENU ) & 0x8000 ) 3850 tmpCode |= MODKEY_LMOD2; 3851 if( GetKeyState( VK_RMENU ) & 0x8000 ) 3852 tmpCode |= MODKEY_RMOD2; 3853 3854 if( tmpCode < nLastModKeyCode ) 3855 { 3856 aModEvt.mnModKeyCode = nLastModKeyCode; 3857 nLastModKeyCode = 0; 3858 bWaitForModKeyRelease = true; 3859 } 3860 else 3861 { 3862 if( !bWaitForModKeyRelease ) 3863 nLastModKeyCode = tmpCode; 3864 } 3865 3866 if( !tmpCode ) 3867 bWaitForModKeyRelease = false; 3868 3869 return pFrame->CallCallback( SALEVENT_KEYMODCHANGE, &aModEvt ); 3870 } 3871 else 3872 { 3873 SalKeyEvent aKeyEvt; 3874 sal_uInt16 nEvent; 3875 MSG aCharMsg; 3876 BOOL bCharPeek = FALSE; 3877 UINT nCharMsg = WM_CHAR; 3878 sal_Bool bKeyUp = (nMsg == WM_KEYUP) || (nMsg == WM_SYSKEYUP); 3879 3880 nLastModKeyCode = 0; // make sure no modkey messages are sent if they belong to a hotkey (see above) 3881 aKeyEvt.mnCharCode = 0; 3882 aKeyEvt.mnCode = 0; 3883 3884 aKeyEvt.mnCode = ImplSalGetKeyCode( wParam ); 3885 if ( !bKeyUp ) 3886 { 3887 // check for charcode 3888 // Mit Hilfe von PeekMessage holen wir uns jetzt die 3889 // zugehoerige WM_CHAR Message, wenn vorhanden. 3890 // Diese WM_CHAR Message steht immer am Anfang der 3891 // Messagequeue. Ausserdem ist sichergestellt, dass immer 3892 // nur eine WM_CHAR Message in der Queue steht. 3893 bCharPeek = ImplPeekMessage( &aCharMsg, hWnd, 3894 WM_CHAR, WM_CHAR, PM_NOREMOVE | PM_NOYIELD ); 3895 if ( bCharPeek && (nDeadChar == aCharMsg.wParam) ) 3896 { 3897 bCharPeek = FALSE; 3898 nDeadChar = 0; 3899 3900 if ( wParam == VK_BACK ) 3901 { 3902 ImplPeekMessage( &aCharMsg, hWnd, 3903 nCharMsg, nCharMsg, PM_REMOVE | PM_NOYIELD ); 3904 return 0; 3905 } 3906 } 3907 else 3908 { 3909 if ( !bCharPeek ) 3910 { 3911 bCharPeek = ImplPeekMessage( &aCharMsg, hWnd, 3912 WM_SYSCHAR, WM_SYSCHAR, PM_NOREMOVE | PM_NOYIELD ); 3913 nCharMsg = WM_SYSCHAR; 3914 } 3915 } 3916 if ( bCharPeek ) 3917 aKeyEvt.mnCharCode = ImplGetCharCode( pFrame, aCharMsg.wParam ); 3918 else 3919 aKeyEvt.mnCharCode = 0; 3920 3921 nLastChar = aKeyEvt.mnCharCode; 3922 nLastVKChar = wParam; 3923 } 3924 else 3925 { 3926 if ( wParam == nLastVKChar ) 3927 { 3928 aKeyEvt.mnCharCode = nLastChar; 3929 nLastChar = 0; 3930 nLastVKChar = 0; 3931 } 3932 } 3933 3934 if ( aKeyEvt.mnCode || aKeyEvt.mnCharCode ) 3935 { 3936 if ( bKeyUp ) 3937 nEvent = SALEVENT_KEYUP; 3938 else 3939 nEvent = SALEVENT_KEYINPUT; 3940 3941 aKeyEvt.mnTime = GetMessageTime(); 3942 aKeyEvt.mnCode |= nModCode; 3943 aKeyEvt.mnRepeat = nRepeat; 3944 3945 if( (nModCode & (KEY_MOD1|KEY_MOD2)) == (KEY_MOD1|KEY_MOD2) && 3946 aKeyEvt.mnCharCode ) 3947 { 3948 // this is actually AltGr and should not be handled as Alt 3949 aKeyEvt.mnCode &= ~(KEY_MOD1|KEY_MOD2); 3950 } 3951 3952 bIgnoreCharMsg = bCharPeek ? TRUE : FALSE; 3953 long nRet = pFrame->CallCallback( nEvent, &aKeyEvt ); 3954 // independent part only reacts on keyup but Windows does not send 3955 // keyup for VK_HANJA 3956 if( aKeyEvt.mnCode == KEY_HANGUL_HANJA ) 3957 nRet = pFrame->CallCallback( SALEVENT_KEYUP, &aKeyEvt ); 3958 3959 bIgnoreCharMsg = FALSE; 3960 3961 // char-message, than remove or ignore 3962 if ( bCharPeek ) 3963 { 3964 nDeadChar = 0; 3965 if ( nRet ) 3966 { 3967 ImplPeekMessage( &aCharMsg, hWnd, 3968 nCharMsg, nCharMsg, PM_REMOVE | PM_NOYIELD ); 3969 } 3970 else 3971 bIgnoreCharMsg = TRUE; 3972 } 3973 3974 return nRet; 3975 } 3976 else 3977 return 0; 3978 } 3979 } 3980 } 3981 3982 // ----------------------------------------------------------------------- 3983 3984 long ImplHandleSalObjKeyMsg( HWND hWnd, UINT nMsg, 3985 WPARAM wParam, LPARAM lParam ) 3986 { 3987 if ( (nMsg == WM_KEYDOWN) || (nMsg == WM_KEYUP) ) 3988 { 3989 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 3990 if ( !pFrame ) 3991 return 0; 3992 3993 sal_uInt16 nRepeat = LOWORD( lParam )-1; 3994 sal_uInt16 nModCode = 0; 3995 3996 // determine modifiers 3997 if ( GetKeyState( VK_SHIFT ) & 0x8000 ) 3998 nModCode |= KEY_SHIFT; 3999 if ( GetKeyState( VK_CONTROL ) & 0x8000 ) 4000 nModCode |= KEY_MOD1; 4001 if ( GetKeyState( VK_MENU ) & 0x8000 ) 4002 nModCode |= KEY_MOD2; 4003 4004 if ( (wParam != VK_SHIFT) && (wParam != VK_CONTROL) && (wParam != VK_MENU) ) 4005 { 4006 SalKeyEvent aKeyEvt; 4007 sal_uInt16 nEvent; 4008 sal_Bool bKeyUp = (nMsg == WM_KEYUP) || (nMsg == WM_SYSKEYUP); 4009 4010 // convert KeyCode 4011 aKeyEvt.mnCode = ImplSalGetKeyCode( wParam ); 4012 aKeyEvt.mnCharCode = 0; 4013 4014 if ( aKeyEvt.mnCode ) 4015 { 4016 if ( bKeyUp ) 4017 nEvent = SALEVENT_KEYUP; 4018 else 4019 nEvent = SALEVENT_KEYINPUT; 4020 4021 aKeyEvt.mnTime = GetMessageTime(); 4022 aKeyEvt.mnCode |= nModCode; 4023 aKeyEvt.mnRepeat = nRepeat; 4024 long nRet = pFrame->CallCallback( nEvent, &aKeyEvt ); 4025 return nRet; 4026 } 4027 else 4028 return 0; 4029 } 4030 } 4031 4032 return 0; 4033 } 4034 4035 // ----------------------------------------------------------------------- 4036 4037 long ImplHandleSalObjSysCharMsg( HWND hWnd, WPARAM wParam, LPARAM lParam ) 4038 { 4039 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4040 if ( !pFrame ) 4041 return 0; 4042 4043 sal_uInt16 nRepeat = LOWORD( lParam )-1; 4044 sal_uInt16 nModCode = 0; 4045 sal_uInt16 cKeyCode = (sal_uInt16)wParam; 4046 4047 // determine modifiers 4048 if ( GetKeyState( VK_SHIFT ) & 0x8000 ) 4049 nModCode |= KEY_SHIFT; 4050 if ( GetKeyState( VK_CONTROL ) & 0x8000 ) 4051 nModCode |= KEY_MOD1; 4052 nModCode |= KEY_MOD2; 4053 4054 // KeyEvent zusammenbauen 4055 SalKeyEvent aKeyEvt; 4056 aKeyEvt.mnTime = GetMessageTime(); 4057 if ( (cKeyCode >= 48) && (cKeyCode <= 57) ) 4058 aKeyEvt.mnCode = KEY_0+(cKeyCode-48); 4059 else if ( (cKeyCode >= 65) && (cKeyCode <= 90) ) 4060 aKeyEvt.mnCode = KEY_A+(cKeyCode-65); 4061 else if ( (cKeyCode >= 97) && (cKeyCode <= 122) ) 4062 aKeyEvt.mnCode = KEY_A+(cKeyCode-97); 4063 else 4064 aKeyEvt.mnCode = 0; 4065 aKeyEvt.mnCode |= nModCode; 4066 aKeyEvt.mnCharCode = ImplGetCharCode( pFrame, cKeyCode ); 4067 aKeyEvt.mnRepeat = nRepeat; 4068 long nRet = pFrame->CallCallback( SALEVENT_KEYINPUT, &aKeyEvt ); 4069 pFrame->CallCallback( SALEVENT_KEYUP, &aKeyEvt ); 4070 return nRet; 4071 } 4072 4073 // ----------------------------------------------------------------------- 4074 4075 static bool ImplHandlePaintMsg( HWND hWnd ) 4076 { 4077 sal_Bool bMutex = FALSE; 4078 if ( ImplSalYieldMutexTryToAcquire() ) 4079 bMutex = TRUE; 4080 4081 // if we don't get the mutex, we can also change the clip region, 4082 // because other threads doesn't use the mutex from the main 4083 // thread --> see GetGraphics() 4084 4085 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4086 if ( pFrame ) 4087 { 4088 // Clip-Region muss zurueckgesetzt werden, da wir sonst kein 4089 // ordentliches Bounding-Rectangle bekommen 4090 if ( pFrame->mpGraphics && pFrame->mpGraphics->mhRegion ) 4091 SelectClipRgn( pFrame->mpGraphics->mhDC, 0 ); 4092 4093 // Laut Window-Doku soll man erst abfragen, ob ueberhaupt eine 4094 // Paint-Region anliegt 4095 if ( GetUpdateRect( hWnd, NULL, FALSE ) ) 4096 { 4097 // Call BeginPaint/EndPaint to query the rect and send 4098 // this Notofication to rect 4099 RECT aUpdateRect; 4100 PAINTSTRUCT aPs; 4101 BeginPaint( hWnd, &aPs ); 4102 CopyRect( &aUpdateRect, &aPs.rcPaint ); 4103 4104 // Paint 4105 // ClipRegion wieder herstellen 4106 if ( pFrame->mpGraphics && pFrame->mpGraphics->mhRegion ) 4107 { 4108 SelectClipRgn( pFrame->mpGraphics->mhDC, 4109 pFrame->mpGraphics->mhRegion ); 4110 } 4111 4112 if ( bMutex ) 4113 { 4114 SalPaintEvent aPEvt( aUpdateRect.left, aUpdateRect.top, aUpdateRect.right-aUpdateRect.left, aUpdateRect.bottom-aUpdateRect.top, pFrame->mbPresentation ); 4115 pFrame->CallCallback( SALEVENT_PAINT, &aPEvt ); 4116 } 4117 else 4118 { 4119 RECT* pRect = new RECT; 4120 CopyRect( pRect, &aUpdateRect ); 4121 ImplPostMessage( hWnd, SAL_MSG_POSTPAINT, (WPARAM)pRect, 0 ); 4122 } 4123 EndPaint( hWnd, &aPs ); 4124 } 4125 else 4126 { 4127 // ClipRegion wieder herstellen 4128 if ( pFrame->mpGraphics && pFrame->mpGraphics->mhRegion ) 4129 { 4130 SelectClipRgn( pFrame->mpGraphics->mhDC, 4131 pFrame->mpGraphics->mhRegion ); 4132 } 4133 } 4134 } 4135 4136 if ( bMutex ) 4137 ImplSalYieldMutexRelease(); 4138 4139 return bMutex ? true : false; 4140 } 4141 4142 // ----------------------------------------------------------------------- 4143 4144 static void ImplHandlePaintMsg2( HWND hWnd, RECT* pRect ) 4145 { 4146 // Paint 4147 if ( ImplSalYieldMutexTryToAcquire() ) 4148 { 4149 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4150 if ( pFrame ) 4151 { 4152 SalPaintEvent aPEvt( pRect->left, pRect->top, pRect->right-pRect->left, pRect->bottom-pRect->top ); 4153 pFrame->CallCallback( SALEVENT_PAINT, &aPEvt ); 4154 } 4155 ImplSalYieldMutexRelease(); 4156 delete pRect; 4157 } 4158 else 4159 ImplPostMessage( hWnd, SAL_MSG_POSTPAINT, (WPARAM)pRect, 0 ); 4160 } 4161 4162 // ----------------------------------------------------------------------- 4163 4164 static void SetMaximizedFrameGeometry( HWND hWnd, WinSalFrame* pFrame, RECT* pParentRect ) 4165 { 4166 // calculate and set frame geometry of a maximized window - useful if the window is still hidden 4167 4168 // dualmonitor support: 4169 // Get screensize of the monitor whith the mouse pointer 4170 4171 RECT aRectMouse; 4172 if( ! pParentRect ) 4173 { 4174 POINT pt; 4175 GetCursorPos( &pt ); 4176 aRectMouse.left = pt.x; 4177 aRectMouse.top = pt.y; 4178 aRectMouse.right = pt.x+2; 4179 aRectMouse.bottom = pt.y+2; 4180 pParentRect = &aRectMouse; 4181 } 4182 4183 RECT aRect; 4184 ImplSalGetWorkArea( hWnd, &aRect, pParentRect ); 4185 4186 // a maximized window has no other borders than the caption 4187 pFrame->maGeometry.nLeftDecoration = pFrame->maGeometry.nRightDecoration = pFrame->maGeometry.nBottomDecoration = 0; 4188 pFrame->maGeometry.nTopDecoration = pFrame->mbCaption ? GetSystemMetrics( SM_CYCAPTION ) : 0; 4189 4190 aRect.top += pFrame->maGeometry.nTopDecoration; 4191 pFrame->maGeometry.nX = aRect.left; 4192 pFrame->maGeometry.nY = aRect.top; 4193 pFrame->maGeometry.nWidth = aRect.right - aRect.left; 4194 pFrame->maGeometry.nHeight = aRect.bottom - aRect.top; 4195 } 4196 4197 static void UpdateFrameGeometry( HWND hWnd, WinSalFrame* pFrame ) 4198 { 4199 if( !pFrame ) 4200 return; 4201 4202 RECT aRect; 4203 GetWindowRect( hWnd, &aRect ); 4204 memset(&pFrame->maGeometry, 0, sizeof(SalFrameGeometry) ); 4205 4206 if ( IsIconic( hWnd ) ) 4207 return; 4208 4209 POINT aPt; 4210 aPt.x=0; 4211 aPt.y=0; 4212 ClientToScreen(hWnd, &aPt); 4213 int cx = aPt.x - aRect.left; 4214 pFrame->maGeometry.nTopDecoration = aPt.y - aRect.top; 4215 4216 pFrame->maGeometry.nLeftDecoration = cx; 4217 pFrame->maGeometry.nRightDecoration = cx; 4218 4219 pFrame->maGeometry.nX = aPt.x; 4220 pFrame->maGeometry.nY = aPt.y; 4221 4222 RECT aInnerRect; 4223 GetClientRect( hWnd, &aInnerRect ); 4224 if( aInnerRect.right ) 4225 { 4226 // improve right decoration 4227 aPt.x=aInnerRect.right; 4228 aPt.y=aInnerRect.top; 4229 ClientToScreen(hWnd, &aPt); 4230 pFrame->maGeometry.nRightDecoration = aRect.right - aPt.x; 4231 } 4232 if( aInnerRect.bottom ) // may be zero if window was not shown yet 4233 pFrame->maGeometry.nBottomDecoration += aRect.bottom - aPt.y - aInnerRect.bottom; 4234 else 4235 // bottom border is typically the same as left/right 4236 pFrame->maGeometry.nBottomDecoration = pFrame->maGeometry.nLeftDecoration; 4237 4238 int nWidth = aRect.right - aRect.left 4239 - pFrame->maGeometry.nRightDecoration - pFrame->maGeometry.nLeftDecoration; 4240 int nHeight = aRect.bottom - aRect.top 4241 - pFrame->maGeometry.nBottomDecoration - pFrame->maGeometry.nTopDecoration; 4242 // clamp to zero 4243 pFrame->maGeometry.nHeight = nHeight < 0 ? 0 : nHeight; 4244 pFrame->maGeometry.nWidth = nWidth < 0 ? 0 : nWidth; 4245 pFrame->updateScreenNumber(); 4246 } 4247 4248 // ----------------------------------------------------------------------- 4249 4250 static void ImplCallMoveHdl( HWND hWnd ) 4251 { 4252 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4253 if ( pFrame ) 4254 { 4255 pFrame->CallCallback( SALEVENT_MOVE, 0 ); 4256 // Um doppelte Paints von VCL und SAL zu vermeiden 4257 //if ( IsWindowVisible( hWnd ) && !pFrame->mbInShow ) 4258 // UpdateWindow( hWnd ); 4259 } 4260 } 4261 4262 // ----------------------------------------------------------------------- 4263 4264 static void ImplCallClosePopupsHdl( HWND hWnd ) 4265 { 4266 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4267 if ( pFrame ) 4268 { 4269 pFrame->CallCallback( SALEVENT_CLOSEPOPUPS, 0 ); 4270 } 4271 } 4272 4273 // ----------------------------------------------------------------------- 4274 4275 static void ImplHandleMoveMsg( HWND hWnd ) 4276 { 4277 if ( ImplSalYieldMutexTryToAcquire() ) 4278 { 4279 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4280 if ( pFrame ) 4281 { 4282 UpdateFrameGeometry( hWnd, pFrame ); 4283 4284 if ( GetWindowStyle( hWnd ) & WS_VISIBLE ) 4285 pFrame->mbDefPos = FALSE; 4286 4287 // Gegen moegliche Rekursionen sichern 4288 if ( !pFrame->mbInMoveMsg ) 4289 { 4290 // Fenster im FullScreenModus wieder einpassen 4291 pFrame->mbInMoveMsg = TRUE; 4292 if ( pFrame->mbFullScreen ) 4293 ImplSalFrameFullScreenPos( pFrame ); 4294 pFrame->mbInMoveMsg = FALSE; 4295 } 4296 4297 // Status merken 4298 ImplSaveFrameState( pFrame ); 4299 4300 // Call Hdl 4301 //#93851 if we call this handler, VCL floating windows are not updated correctly 4302 ImplCallMoveHdl( hWnd ); 4303 4304 } 4305 4306 ImplSalYieldMutexRelease(); 4307 } 4308 else 4309 ImplPostMessage( hWnd, SAL_MSG_POSTMOVE, 0, 0 ); 4310 } 4311 4312 // ----------------------------------------------------------------------- 4313 4314 static void ImplCallSizeHdl( HWND hWnd ) 4315 { 4316 // Da Windows diese Messages auch senden kann, muss hier auch die 4317 // Solar-Semaphore beruecksichtigt werden 4318 if ( ImplSalYieldMutexTryToAcquire() ) 4319 { 4320 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4321 if ( pFrame ) 4322 { 4323 pFrame->CallCallback( SALEVENT_RESIZE, 0 ); 4324 // Um doppelte Paints von VCL und SAL zu vermeiden 4325 if ( IsWindowVisible( hWnd ) && !pFrame->mbInShow ) 4326 UpdateWindow( hWnd ); 4327 } 4328 4329 ImplSalYieldMutexRelease(); 4330 } 4331 else 4332 ImplPostMessage( hWnd, SAL_MSG_POSTCALLSIZE, 0, 0 ); 4333 } 4334 4335 // ----------------------------------------------------------------------- 4336 4337 static void ImplHandleSizeMsg( HWND hWnd, WPARAM wParam, LPARAM lParam ) 4338 { 4339 if ( (wParam != SIZE_MAXSHOW) && (wParam != SIZE_MAXHIDE) ) 4340 { 4341 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4342 if ( pFrame ) 4343 { 4344 UpdateFrameGeometry( hWnd, pFrame ); 4345 4346 pFrame->mnWidth = (int)LOWORD(lParam); 4347 pFrame->mnHeight = (int)HIWORD(lParam); 4348 // Status merken 4349 ImplSaveFrameState( pFrame ); 4350 // Call Hdl 4351 ImplCallSizeHdl( hWnd ); 4352 } 4353 } 4354 } 4355 4356 // ----------------------------------------------------------------------- 4357 4358 static void ImplHandleFocusMsg( HWND hWnd ) 4359 { 4360 if ( ImplSalYieldMutexTryToAcquire() ) 4361 { 4362 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4363 if ( pFrame && !WinSalFrame::mbInReparent ) 4364 { 4365 // Query the actual status 4366 if ( ::GetFocus() == hWnd ) 4367 { 4368 if ( IsWindowVisible( hWnd ) && !pFrame->mbInShow ) 4369 UpdateWindow( hWnd ); 4370 4371 // Feststellen, ob wir IME unterstuetzen 4372 if ( pFrame->mbIME && pFrame->mhDefIMEContext ) 4373 { 4374 UINT nImeProps = ImmGetProperty( GetKeyboardLayout( 0 ), IGP_PROPERTY ); 4375 4376 pFrame->mbSpezIME = (nImeProps & IME_PROP_SPECIAL_UI) != 0; 4377 pFrame->mbAtCursorIME = (nImeProps & IME_PROP_AT_CARET) != 0; 4378 pFrame->mbHandleIME = !pFrame->mbSpezIME; 4379 } 4380 4381 pFrame->CallCallback( SALEVENT_GETFOCUS, 0 ); 4382 } 4383 else 4384 { 4385 pFrame->CallCallback( SALEVENT_LOSEFOCUS, 0 ); 4386 } 4387 } 4388 4389 ImplSalYieldMutexRelease(); 4390 } 4391 else 4392 ImplPostMessage( hWnd, SAL_MSG_POSTFOCUS, 0, 0 ); 4393 } 4394 4395 // ----------------------------------------------------------------------- 4396 4397 static void ImplHandleCloseMsg( HWND hWnd ) 4398 { 4399 if ( ImplSalYieldMutexTryToAcquire() ) 4400 { 4401 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4402 if ( pFrame ) 4403 { 4404 pFrame->CallCallback( SALEVENT_CLOSE, 0 ); 4405 } 4406 4407 ImplSalYieldMutexRelease(); 4408 } 4409 else 4410 ImplPostMessage( hWnd, WM_CLOSE, 0, 0 ); 4411 } 4412 4413 // ----------------------------------------------------------------------- 4414 4415 static long ImplHandleShutDownMsg( HWND hWnd ) 4416 { 4417 ImplSalYieldMutexAcquireWithWait(); 4418 long nRet = 0; 4419 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4420 if ( pFrame ) 4421 { 4422 nRet = pFrame->CallCallback( SALEVENT_SHUTDOWN, 0 ); 4423 } 4424 ImplSalYieldMutexRelease(); 4425 return nRet; 4426 } 4427 4428 // ----------------------------------------------------------------------- 4429 4430 static void ImplHandleSettingsChangeMsg( HWND hWnd, UINT nMsg, 4431 WPARAM wParam, LPARAM lParam ) 4432 { 4433 sal_uInt16 nSalEvent = SALEVENT_SETTINGSCHANGED; 4434 4435 if ( nMsg == WM_DEVMODECHANGE ) 4436 nSalEvent = SALEVENT_PRINTERCHANGED; 4437 else if ( nMsg == WM_DISPLAYCHANGE ) 4438 nSalEvent = SALEVENT_DISPLAYCHANGED; 4439 else if ( nMsg == WM_FONTCHANGE ) 4440 nSalEvent = SALEVENT_FONTCHANGED; 4441 else if ( nMsg == WM_TIMECHANGE ) 4442 nSalEvent = SALEVENT_DATETIMECHANGED; 4443 else if ( nMsg == WM_WININICHANGE ) 4444 { 4445 if ( lParam ) 4446 { 4447 if ( ImplSalWICompareAscii( (const wchar_t*)lParam, "devices" ) == 0 ) 4448 nSalEvent = SALEVENT_PRINTERCHANGED; 4449 } 4450 } 4451 4452 if ( nMsg == WM_SETTINGCHANGE ) 4453 { 4454 if ( wParam == SPI_SETWHEELSCROLLLINES ) 4455 aSalShlData.mnWheelScrollLines = ImplSalGetWheelScrollLines(); 4456 else if( wParam == SPI_SETWHEELSCROLLCHARS ) 4457 aSalShlData.mnWheelScrollChars = ImplSalGetWheelScrollChars(); 4458 } 4459 4460 if ( WM_SYSCOLORCHANGE == nMsg && GetSalData()->mhDitherPal ) 4461 ImplUpdateSysColorEntries(); 4462 4463 ImplSalYieldMutexAcquireWithWait(); 4464 4465 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4466 if ( pFrame ) 4467 { 4468 if ( (nMsg == WM_DISPLAYCHANGE) || (nMsg == WM_WININICHANGE) ) 4469 { 4470 if ( pFrame->mbFullScreen ) 4471 ImplSalFrameFullScreenPos( pFrame ); 4472 } 4473 4474 pFrame->CallCallback( nSalEvent, 0 ); 4475 } 4476 4477 ImplSalYieldMutexRelease(); 4478 } 4479 4480 // ----------------------------------------------------------------------- 4481 4482 static void ImplHandleUserEvent( HWND hWnd, LPARAM lParam ) 4483 { 4484 ImplSalYieldMutexAcquireWithWait(); 4485 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4486 if ( pFrame ) 4487 { 4488 pFrame->CallCallback( SALEVENT_USEREVENT, (void*)lParam ); 4489 } 4490 ImplSalYieldMutexRelease(); 4491 } 4492 4493 // ----------------------------------------------------------------------- 4494 4495 static void ImplHandleForcePalette( HWND hWnd ) 4496 { 4497 SalData* pSalData = GetSalData(); 4498 HPALETTE hPal = pSalData->mhDitherPal; 4499 if ( hPal ) 4500 { 4501 if ( !ImplSalYieldMutexTryToAcquire() ) 4502 { 4503 ImplPostMessage( hWnd, SAL_MSG_FORCEPALETTE, 0, 0 ); 4504 return; 4505 } 4506 4507 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4508 if ( pFrame && pFrame->mpGraphics ) 4509 { 4510 WinSalGraphics* pGraphics = pFrame->mpGraphics; 4511 if ( pGraphics && pGraphics->mhDefPal ) 4512 { 4513 SelectPalette( pGraphics->mhDC, hPal, FALSE ); 4514 if ( RealizePalette( pGraphics->mhDC ) ) 4515 { 4516 InvalidateRect( hWnd, NULL, FALSE ); 4517 UpdateWindow( hWnd ); 4518 pFrame->CallCallback( SALEVENT_DISPLAYCHANGED, 0 ); 4519 } 4520 } 4521 } 4522 4523 ImplSalYieldMutexRelease(); 4524 } 4525 } 4526 4527 // ----------------------------------------------------------------------- 4528 4529 static LRESULT ImplHandlePalette( sal_Bool bFrame, HWND hWnd, UINT nMsg, 4530 WPARAM wParam, LPARAM lParam, int& rDef ) 4531 { 4532 SalData* pSalData = GetSalData(); 4533 HPALETTE hPal = pSalData->mhDitherPal; 4534 if ( !hPal ) 4535 return 0; 4536 4537 rDef = FALSE; 4538 if ( pSalData->mbInPalChange ) 4539 return 0; 4540 4541 if ( (nMsg == WM_PALETTECHANGED) || (nMsg == SAL_MSG_POSTPALCHANGED) ) 4542 { 4543 if ( (HWND)wParam == hWnd ) 4544 return 0; 4545 } 4546 4547 sal_Bool bReleaseMutex = FALSE; 4548 if ( (nMsg == WM_QUERYNEWPALETTE) || (nMsg == WM_PALETTECHANGED) ) 4549 { 4550 // Da Windows diese Messages auch sendet, muss hier auch die 4551 // Solar-Semaphore beruecksichtigt werden 4552 if ( ImplSalYieldMutexTryToAcquire() ) 4553 bReleaseMutex = TRUE; 4554 else if ( nMsg == WM_QUERYNEWPALETTE ) 4555 ImplPostMessage( hWnd, SAL_MSG_POSTQUERYNEWPAL, wParam, lParam ); 4556 else /* ( nMsg == WM_PALETTECHANGED ) */ 4557 ImplPostMessage( hWnd, SAL_MSG_POSTPALCHANGED, wParam, lParam ); 4558 } 4559 4560 WinSalVirtualDevice*pTempVD; 4561 WinSalFrame* pTempFrame; 4562 WinSalGraphics* pGraphics; 4563 HDC hDC; 4564 HPALETTE hOldPal; 4565 UINT nCols; 4566 sal_Bool bStdDC; 4567 sal_Bool bUpdate; 4568 4569 pSalData->mbInPalChange = TRUE; 4570 4571 // Alle Paletten in VirDevs und Frames zuruecksetzen 4572 pTempVD = pSalData->mpFirstVD; 4573 while ( pTempVD ) 4574 { 4575 pGraphics = pTempVD->mpGraphics; 4576 if ( pGraphics->mhDefPal ) 4577 { 4578 SelectPalette( pGraphics->mhDC, 4579 pGraphics->mhDefPal, 4580 TRUE ); 4581 } 4582 pTempVD = pTempVD->mpNext; 4583 } 4584 pTempFrame = pSalData->mpFirstFrame; 4585 while ( pTempFrame ) 4586 { 4587 pGraphics = pTempFrame->mpGraphics; 4588 if ( pGraphics && pGraphics->mhDefPal ) 4589 { 4590 SelectPalette( pGraphics->mhDC, 4591 pGraphics->mhDefPal, 4592 TRUE ); 4593 } 4594 pTempFrame = pTempFrame->mpNextFrame; 4595 } 4596 4597 // Palette neu realizen 4598 WinSalFrame* pFrame = NULL; 4599 if ( bFrame ) 4600 pFrame = GetWindowPtr( hWnd ); 4601 if ( pFrame && pFrame->mpGraphics ) 4602 { 4603 hDC = pFrame->mpGraphics->mhDC; 4604 bStdDC = TRUE; 4605 } 4606 else 4607 { 4608 hDC = GetDC( hWnd ); 4609 bStdDC = FALSE; 4610 } 4611 UnrealizeObject( hPal ); 4612 hOldPal = SelectPalette( hDC, hPal, TRUE ); 4613 nCols = RealizePalette( hDC ); 4614 bUpdate = nCols != 0; 4615 if ( !bStdDC ) 4616 { 4617 SelectPalette( hDC, hOldPal, TRUE ); 4618 ReleaseDC( hWnd, hDC ); 4619 } 4620 4621 // Alle Paletten in VirDevs und Frames neu setzen 4622 pTempVD = pSalData->mpFirstVD; 4623 while ( pTempVD ) 4624 { 4625 pGraphics = pTempVD->mpGraphics; 4626 if ( pGraphics->mhDefPal ) 4627 { 4628 SelectPalette( pGraphics->mhDC, hPal, TRUE ); 4629 RealizePalette( pGraphics->mhDC ); 4630 } 4631 pTempVD = pTempVD->mpNext; 4632 } 4633 pTempFrame = pSalData->mpFirstFrame; 4634 while ( pTempFrame ) 4635 { 4636 if ( pTempFrame != pFrame ) 4637 { 4638 pGraphics = pTempFrame->mpGraphics; 4639 if ( pGraphics && pGraphics->mhDefPal ) 4640 { 4641 SelectPalette( pGraphics->mhDC, hPal, TRUE ); 4642 if ( RealizePalette( pGraphics->mhDC ) ) 4643 bUpdate = TRUE; 4644 } 4645 } 4646 pTempFrame = pTempFrame->mpNextFrame; 4647 } 4648 4649 // Wenn sich Farben geaendert haben, dann die Fenster updaten 4650 if ( bUpdate ) 4651 { 4652 pTempFrame = pSalData->mpFirstFrame; 4653 while ( pTempFrame ) 4654 { 4655 pGraphics = pTempFrame->mpGraphics; 4656 if ( pGraphics && pGraphics->mhDefPal ) 4657 { 4658 InvalidateRect( pTempFrame->mhWnd, NULL, FALSE ); 4659 UpdateWindow( pTempFrame->mhWnd ); 4660 pTempFrame->CallCallback( SALEVENT_DISPLAYCHANGED, 0 ); 4661 } 4662 pTempFrame = pTempFrame->mpNextFrame; 4663 } 4664 } 4665 4666 pSalData->mbInPalChange = FALSE; 4667 4668 if ( bReleaseMutex ) 4669 ImplSalYieldMutexRelease(); 4670 4671 if ( nMsg == WM_PALETTECHANGED ) 4672 return 0; 4673 else 4674 return nCols; 4675 } 4676 4677 // ----------------------------------------------------------------------- 4678 4679 static int ImplHandleMinMax( HWND hWnd, LPARAM lParam ) 4680 { 4681 int bRet = FALSE; 4682 4683 if ( ImplSalYieldMutexTryToAcquire() ) 4684 { 4685 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 4686 if ( pFrame ) 4687 { 4688 MINMAXINFO* pMinMax = (MINMAXINFO*)lParam; 4689 4690 if ( pFrame->mbFullScreen ) 4691 { 4692 int nX; 4693 int nY; 4694 int nDX; 4695 int nDY; 4696 ImplSalCalcFullScreenSize( pFrame, nX, nY, nDX, nDY ); 4697 4698 if ( pMinMax->ptMaxPosition.x > nX ) 4699 pMinMax->ptMaxPosition.x = nX; 4700 if ( pMinMax->ptMaxPosition.y > nY ) 4701 pMinMax->ptMaxPosition.y = nY; 4702 4703 if ( pMinMax->ptMaxSize.x < nDX ) 4704 pMinMax->ptMaxSize.x = nDX; 4705 if ( pMinMax->ptMaxSize.y < nDY ) 4706 pMinMax->ptMaxSize.y = nDY; 4707 if ( pMinMax->ptMaxTrackSize.x < nDX ) 4708 pMinMax->ptMaxTrackSize.x = nDX; 4709 if ( pMinMax->ptMaxTrackSize.y < nDY ) 4710 pMinMax->ptMaxTrackSize.y = nDY; 4711 4712 pMinMax->ptMinTrackSize.x = nDX; 4713 pMinMax->ptMinTrackSize.y = nDY; 4714 4715 bRet = TRUE; 4716 } 4717 4718 if ( pFrame->mnMinWidth || pFrame->mnMinHeight ) 4719 { 4720 int nWidth = pFrame->mnMinWidth; 4721 int nHeight = pFrame->mnMinHeight; 4722 4723 ImplSalAddBorder( pFrame, nWidth, nHeight ); 4724 4725 if ( pMinMax->ptMinTrackSize.x < nWidth ) 4726 pMinMax->ptMinTrackSize.x = nWidth; 4727 if ( pMinMax->ptMinTrackSize.y < nHeight ) 4728 pMinMax->ptMinTrackSize.y = nHeight; 4729 } 4730 4731 if ( pFrame->mnMaxWidth || pFrame->mnMaxHeight ) 4732 { 4733 int nWidth = pFrame->mnMaxWidth; 4734 int nHeight = pFrame->mnMaxHeight; 4735 4736 ImplSalAddBorder( pFrame, nWidth, nHeight ); 4737 4738 if( nWidth > 0 && nHeight > 0 ) // protect against int overflow due to INT_MAX initialisation 4739 { 4740 if ( pMinMax->ptMaxTrackSize.x > nWidth ) 4741 pMinMax->ptMaxTrackSize.x = nWidth; 4742 if ( pMinMax->ptMaxTrackSize.y > nHeight ) 4743 pMinMax->ptMaxTrackSize.y = nHeight; 4744 } 4745 } 4746 } 4747 4748 ImplSalYieldMutexRelease(); 4749 } 4750 4751 return bRet; 4752 } 4753 4754 // ----------------------------------------------------------------------- 4755 4756 // retrieves the SalMenuItem pointer from a hMenu 4757 // the pointer is stored in every item, so if no position 4758 // is specified we just use the first item (ie, pos=0) 4759 // if bByPosition is FALSE then nPos denotes a menu id instead of a position 4760 static WinSalMenuItem* ImplGetSalMenuItem( HMENU hMenu, UINT nPos, sal_Bool bByPosition=TRUE ) 4761 { 4762 DWORD err=0; 4763 4764 MENUITEMINFOW mi; 4765 memset(&mi, 0, sizeof(mi)); 4766 mi.cbSize = sizeof( mi ); 4767 mi.fMask = MIIM_DATA; 4768 if( !GetMenuItemInfoW( hMenu, nPos, bByPosition, &mi) ) 4769 err = GetLastError(); 4770 4771 return (WinSalMenuItem *) mi.dwItemData; 4772 } 4773 4774 // returns the index of the currently selected item if any or -1 4775 static int ImplGetSelectedIndex( HMENU hMenu ) 4776 { 4777 DWORD err=0; 4778 4779 MENUITEMINFOW mi; 4780 memset(&mi, 0, sizeof(mi)); 4781 mi.cbSize = sizeof( mi ); 4782 mi.fMask = MIIM_STATE; 4783 int n = GetMenuItemCount( hMenu ); 4784 if( n != -1 ) 4785 { 4786 for(int i=0; i<n; i++ ) 4787 { 4788 if( !GetMenuItemInfoW( hMenu, i, TRUE, &mi) ) 4789 err = GetLastError(); 4790 else 4791 { 4792 if( mi.fState & MFS_HILITE ) 4793 return i; 4794 } 4795 } 4796 } 4797 return -1; 4798 } 4799 4800 static int ImplMenuChar( HWND, WPARAM wParam, LPARAM lParam ) 4801 { 4802 int nRet = MNC_IGNORE; 4803 HMENU hMenu = (HMENU) lParam; 4804 String aMnemonic; 4805 aMnemonic.AssignAscii("&"); 4806 aMnemonic.Append( (sal_Unicode) LOWORD(wParam) ); 4807 aMnemonic.ToLowerAscii(); // we only have ascii mnemonics 4808 4809 // search the mnemonic in the current menu 4810 int nItemCount = GetMenuItemCount( hMenu ); 4811 int nFound = 0; 4812 int idxFound = -1; 4813 int idxSelected = ImplGetSelectedIndex( hMenu ); 4814 int idx = idxSelected != -1 ? idxSelected+1 : 0; // if duplicate mnemonics cycle through menu 4815 for( int i=0; i< nItemCount; i++, idx++ ) 4816 { 4817 WinSalMenuItem* pSalMenuItem = ImplGetSalMenuItem( hMenu, idx % nItemCount ); 4818 if( !pSalMenuItem ) 4819 continue; 4820 String aStr = pSalMenuItem->mText; 4821 aStr.ToLowerAscii(); 4822 if( aStr.Search( aMnemonic ) != STRING_NOTFOUND) 4823 { 4824 if( idxFound == -1 ) 4825 idxFound = idx % nItemCount; 4826 if( nFound++ ) 4827 break; // duplicate found 4828 } 4829 } 4830 if( nFound == 1 ) 4831 nRet = MAKELRESULT( idxFound, MNC_EXECUTE ); 4832 else 4833 // duplicate mnemonics, just select the next occurence 4834 nRet = MAKELRESULT( idxFound, MNC_SELECT ); 4835 4836 return nRet; 4837 } 4838 4839 static int ImplMeasureItem( HWND hWnd, WPARAM wParam, LPARAM lParam ) 4840 { 4841 int nRet = 0; 4842 if( !wParam ) 4843 { 4844 // request was sent by a menu 4845 nRet = 1; 4846 MEASUREITEMSTRUCT *pMI = (LPMEASUREITEMSTRUCT) lParam; 4847 if( pMI->CtlType != ODT_MENU ) 4848 return 0; 4849 4850 WinSalMenuItem *pSalMenuItem = (WinSalMenuItem *) pMI->itemData; 4851 if( !pSalMenuItem ) 4852 return 0; 4853 4854 HDC hdc = GetDC( hWnd ); 4855 SIZE strSize; 4856 4857 NONCLIENTMETRICS ncm; 4858 memset( &ncm, 0, sizeof(ncm) ); 4859 ncm.cbSize = sizeof( ncm ); 4860 SystemParametersInfo( SPI_GETNONCLIENTMETRICS, 0, (PVOID) &ncm, 0 ); 4861 4862 // Assume every menu item can be default and printed bold 4863 //ncm.lfMenuFont.lfWeight = FW_BOLD; 4864 4865 HFONT hfntOld = (HFONT) SelectObject(hdc, (HFONT) CreateFontIndirect( &ncm.lfMenuFont )); 4866 4867 // menu text and accelerator 4868 String aStr(pSalMenuItem->mText.GetBuffer() ); 4869 if( pSalMenuItem->mAccelText.Len() ) 4870 { 4871 aStr.AppendAscii(" "); 4872 aStr.Append( pSalMenuItem->mAccelText ); 4873 } 4874 GetTextExtentPoint32W( hdc, (LPWSTR) aStr.GetBuffer(), 4875 aStr.Len(), &strSize ); 4876 4877 // image 4878 Size bmpSize( 16, 16 ); 4879 //if( !!pSalMenuItem->maBitmap ) 4880 // bmpSize = pSalMenuItem->maBitmap.GetSizePixel(); 4881 4882 // checkmark 4883 Size checkSize( GetSystemMetrics( SM_CXMENUCHECK ), GetSystemMetrics( SM_CYMENUCHECK ) ); 4884 4885 pMI->itemWidth = checkSize.Width() + 3 + bmpSize.Width() + 3 + strSize.cx; 4886 pMI->itemHeight = Max( Max( checkSize.Height(), bmpSize.Height() ), strSize.cy ); 4887 pMI->itemHeight += 4; 4888 4889 DeleteObject( SelectObject(hdc, hfntOld) ); 4890 ReleaseDC( hWnd, hdc ); 4891 } 4892 4893 return nRet; 4894 } 4895 4896 static int ImplDrawItem(HWND, WPARAM wParam, LPARAM lParam ) 4897 { 4898 int nRet = 0; 4899 DWORD err = 0; 4900 if( !wParam ) 4901 { 4902 // request was sent by a menu 4903 nRet = 1; 4904 DRAWITEMSTRUCT *pDI = (LPDRAWITEMSTRUCT) lParam; 4905 if( pDI->CtlType != ODT_MENU ) 4906 return 0; 4907 4908 WinSalMenuItem *pSalMenuItem = (WinSalMenuItem *) pDI->itemData; 4909 if( !pSalMenuItem ) 4910 return 0; 4911 4912 COLORREF clrPrevText, clrPrevBkgnd; 4913 HFONT hfntOld; 4914 HBRUSH hbrOld; 4915 sal_Bool fChecked = (pDI->itemState & ODS_CHECKED) ? TRUE : FALSE; 4916 sal_Bool fSelected = (pDI->itemState & ODS_SELECTED) ? TRUE : FALSE; 4917 sal_Bool fDisabled = (pDI->itemState & (ODS_DISABLED | ODS_GRAYED)) ? TRUE : FALSE; 4918 4919 // Set the appropriate foreground and background colors. 4920 RECT aRect = pDI->rcItem; 4921 4922 clrPrevBkgnd = SetBkColor( pDI->hDC, GetSysColor( COLOR_MENU ) ); 4923 4924 if ( fDisabled ) 4925 clrPrevText = SetTextColor( pDI->hDC, GetSysColor( COLOR_GRAYTEXT ) ); 4926 else 4927 clrPrevText = SetTextColor( pDI->hDC, GetSysColor( fSelected ? COLOR_HIGHLIGHTTEXT : COLOR_MENUTEXT ) ); 4928 4929 DWORD colBackground = GetSysColor( fSelected ? COLOR_HIGHLIGHT : COLOR_MENU ); 4930 if ( fSelected ) 4931 clrPrevBkgnd = SetBkColor( pDI->hDC, colBackground ); 4932 else 4933 clrPrevBkgnd = SetBkColor( pDI->hDC, colBackground ); 4934 4935 hbrOld = (HBRUSH)SelectObject( pDI->hDC, CreateSolidBrush( GetBkColor( pDI->hDC ) ) ); 4936 4937 // Fill background 4938 if(!PatBlt( pDI->hDC, aRect.left, aRect.top, aRect.right-aRect.left, aRect.bottom-aRect.top, PATCOPY )) 4939 err = GetLastError(); 4940 4941 int lineHeight = aRect.bottom-aRect.top; 4942 4943 int x = aRect.left; 4944 int y = aRect.top; 4945 4946 int checkWidth = GetSystemMetrics( SM_CXMENUCHECK ); 4947 int checkHeight = GetSystemMetrics( SM_CYMENUCHECK ); 4948 if( fChecked ) 4949 { 4950 RECT r; 4951 r.left = 0; 4952 r.top = 0; 4953 r.right = checkWidth; 4954 r.bottom = checkWidth; 4955 HDC memDC = CreateCompatibleDC( pDI->hDC ); 4956 HBITMAP memBmp = CreateCompatibleBitmap( pDI->hDC, checkWidth, checkHeight ); 4957 HBITMAP hOldBmp = (HBITMAP) SelectObject( memDC, memBmp ); 4958 DrawFrameControl( memDC, &r, DFC_MENU, DFCS_MENUCHECK ); 4959 BitBlt( pDI->hDC, x, y+(lineHeight-checkHeight)/2, checkWidth, checkHeight, memDC, 0, 0, SRCAND ); 4960 DeleteObject( SelectObject( memDC, hOldBmp ) ); 4961 DeleteDC( memDC ); 4962 } 4963 x += checkWidth+3; 4964 4965 //Size bmpSize = aBitmap.GetSizePixel(); 4966 Size bmpSize(16, 16); 4967 if( !!pSalMenuItem->maBitmap ) 4968 { 4969 Bitmap aBitmap( pSalMenuItem->maBitmap ); 4970 4971 // set transparent pixels to background color 4972 if( fDisabled ) 4973 colBackground = RGB(255,255,255); 4974 aBitmap.Replace( Color( COL_LIGHTMAGENTA ), 4975 Color( GetRValue(colBackground),GetGValue(colBackground),GetBValue(colBackground) ), 0); 4976 4977 WinSalBitmap* pSalBmp = static_cast<WinSalBitmap*>(aBitmap.ImplGetImpBitmap()->ImplGetSalBitmap()); 4978 HGLOBAL hDrawDIB = pSalBmp->ImplGethDIB(); 4979 4980 if( hDrawDIB ) 4981 { 4982 PBITMAPINFO pBI = (PBITMAPINFO) GlobalLock( hDrawDIB ); 4983 PBITMAPINFOHEADER pBIH = (PBITMAPINFOHEADER) pBI; 4984 PBYTE pBits = (PBYTE) pBI + *(DWORD*) pBI + 4985 pSalBmp->ImplGetDIBColorCount( hDrawDIB ) * sizeof( RGBQUAD ); 4986 4987 HBITMAP hBmp = CreateDIBitmap( pDI->hDC, pBIH, CBM_INIT, pBits, pBI, DIB_RGB_COLORS ); 4988 GlobalUnlock( hDrawDIB ); 4989 4990 HBRUSH hbrIcon = CreateSolidBrush( GetSysColor( COLOR_GRAYTEXT ) ); 4991 DrawStateW( pDI->hDC, (HBRUSH)hbrIcon, (DRAWSTATEPROC)NULL, (LPARAM)hBmp, (WPARAM)0, 4992 x, y+(lineHeight-bmpSize.Height())/2, bmpSize.Width(), bmpSize.Height(), 4993 DST_BITMAP | (fDisabled ? (fSelected ? DSS_MONO : DSS_DISABLED) : DSS_NORMAL) ); 4994 4995 DeleteObject( hbrIcon ); 4996 DeleteObject( hBmp ); 4997 } 4998 4999 } 5000 x += bmpSize.Width() + 3; 5001 aRect.left = x; 5002 5003 NONCLIENTMETRICS ncm; 5004 memset( &ncm, 0, sizeof(ncm) ); 5005 ncm.cbSize = sizeof( ncm ); 5006 SystemParametersInfo( SPI_GETNONCLIENTMETRICS, 0, (PVOID) &ncm, 0 ); 5007 5008 // Print default menu entry with bold font 5009 //if ( pDI->itemState & ODS_DEFAULT ) 5010 // ncm.lfMenuFont.lfWeight = FW_BOLD; 5011 5012 hfntOld = (HFONT) SelectObject(pDI->hDC, (HFONT) CreateFontIndirect( &ncm.lfMenuFont )); 5013 5014 SIZE strSize; 5015 String aStr( pSalMenuItem->mText.GetBuffer() ); 5016 GetTextExtentPoint32W( pDI->hDC, (LPWSTR) aStr.GetBuffer(), 5017 aStr.Len(), &strSize ); 5018 5019 if(!DrawStateW( pDI->hDC, (HBRUSH)NULL, (DRAWSTATEPROC)NULL, 5020 (LPARAM)(LPWSTR) aStr.GetBuffer(), 5021 (WPARAM)0, aRect.left, aRect.top + (lineHeight - strSize.cy)/2, 0, 0, 5022 DST_PREFIXTEXT | (fDisabled && !fSelected ? DSS_DISABLED : DSS_NORMAL) ) ) 5023 err = GetLastError(); 5024 5025 if( pSalMenuItem->mAccelText.Len() ) 5026 { 5027 SIZE strSizeA; 5028 aStr = pSalMenuItem->mAccelText; 5029 GetTextExtentPoint32W( pDI->hDC, (LPWSTR) aStr.GetBuffer(), 5030 aStr.Len(), &strSizeA ); 5031 TEXTMETRIC tm; 5032 GetTextMetrics( pDI->hDC, &tm ); 5033 5034 // position the accelerator string to the right but leave space for the 5035 // (potential) submenu arrow (tm.tmMaxCharWidth) 5036 if(!DrawStateW( pDI->hDC, (HBRUSH)NULL, (DRAWSTATEPROC)NULL, 5037 (LPARAM)(LPWSTR) aStr.GetBuffer(), 5038 (WPARAM)0, aRect.right-strSizeA.cx-tm.tmMaxCharWidth, aRect.top + (lineHeight - strSizeA.cy)/2, 0, 0, 5039 DST_TEXT | (fDisabled && !fSelected ? DSS_DISABLED : DSS_NORMAL) ) ) 5040 err = GetLastError(); 5041 } 5042 5043 // Restore the original font and colors. 5044 DeleteObject( SelectObject( pDI->hDC, hbrOld ) ); 5045 DeleteObject( SelectObject( pDI->hDC, hfntOld) ); 5046 SetTextColor(pDI->hDC, clrPrevText); 5047 SetBkColor(pDI->hDC, clrPrevBkgnd); 5048 } 5049 return nRet; 5050 } 5051 5052 static int ImplHandleMenuActivate( HWND hWnd, WPARAM wParam, LPARAM ) 5053 { 5054 // Menu activation 5055 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5056 if ( !pFrame ) 5057 return 0; 5058 5059 HMENU hMenu = (HMENU) wParam; 5060 // WORD nPos = LOWORD (lParam); 5061 // sal_Bool bWindowMenu = (sal_Bool) HIWORD(lParam); 5062 5063 // Send activate and deactivate together, so we have not keep track of opened menues 5064 // this will be enough to have the menues updated correctly 5065 SalMenuEvent aMenuEvt; 5066 WinSalMenuItem *pSalMenuItem = ImplGetSalMenuItem( hMenu, 0 ); 5067 if( pSalMenuItem ) 5068 aMenuEvt.mpMenu = pSalMenuItem->mpMenu; 5069 else 5070 aMenuEvt.mpMenu = NULL; 5071 5072 long nRet = pFrame->CallCallback( SALEVENT_MENUACTIVATE, &aMenuEvt ); 5073 if( nRet ) 5074 nRet = pFrame->CallCallback( SALEVENT_MENUDEACTIVATE, &aMenuEvt ); 5075 if( nRet ) 5076 pFrame->mLastActivatedhMenu = hMenu; 5077 5078 return (nRet!=0); 5079 } 5080 5081 static int ImplHandleMenuSelect( HWND hWnd, WPARAM wParam, LPARAM lParam ) 5082 { 5083 // Menu selection 5084 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5085 if ( !pFrame ) 5086 return 0; 5087 5088 WORD nId = LOWORD(wParam); // menu item or submenu index 5089 WORD nFlags = HIWORD(wParam); 5090 HMENU hMenu = (HMENU) lParam; 5091 5092 // check if we have to process the message 5093 if( !GetSalData()->IsKnownMenuHandle( hMenu ) ) 5094 return 0; 5095 5096 sal_Bool bByPosition = FALSE; 5097 if( nFlags & MF_POPUP ) 5098 bByPosition = TRUE; 5099 5100 long nRet = 0; 5101 if ( hMenu && !pFrame->mLastActivatedhMenu ) 5102 { 5103 // we never activated a menu (ie, no WM_INITMENUPOPUP has occured yet) 5104 // which means this must be the menubar -> send activation/deactivation 5105 SalMenuEvent aMenuEvt; 5106 WinSalMenuItem *pSalMenuItem = ImplGetSalMenuItem( hMenu, nId, bByPosition ); 5107 if( pSalMenuItem ) 5108 aMenuEvt.mpMenu = pSalMenuItem->mpMenu; 5109 else 5110 aMenuEvt.mpMenu = NULL; 5111 5112 nRet = pFrame->CallCallback( SALEVENT_MENUACTIVATE, &aMenuEvt ); 5113 if( nRet ) 5114 nRet = pFrame->CallCallback( SALEVENT_MENUDEACTIVATE, &aMenuEvt ); 5115 if( nRet ) 5116 pFrame->mLastActivatedhMenu = hMenu; 5117 } 5118 5119 if( !hMenu && nFlags == 0xFFFF ) 5120 { 5121 // all menus are closed, reset activation logic 5122 pFrame->mLastActivatedhMenu = NULL; 5123 } 5124 5125 if( hMenu ) 5126 { 5127 // hMenu must be saved, as it is not passed in WM_COMMAND which always occurs after a selection 5128 // if a menu is closed due to a command selection then hMenu is NULL, but WM_COMMAND comes later 5129 // so we must not overwrite it in this case 5130 pFrame->mSelectedhMenu = hMenu; 5131 5132 // send highlight event 5133 if( nFlags & MF_POPUP ) 5134 { 5135 // submenu selected 5136 // wParam now carries an index instead of an id -> retrieve id 5137 MENUITEMINFOW mi; 5138 memset(&mi, 0, sizeof(mi)); 5139 mi.cbSize = sizeof( mi ); 5140 mi.fMask = MIIM_ID; 5141 if( GetMenuItemInfoW( hMenu, LOWORD(wParam), TRUE, &mi) ) 5142 nId = sal::static_int_cast<WORD>(mi.wID); 5143 } 5144 5145 SalMenuEvent aMenuEvt; 5146 aMenuEvt.mnId = nId; 5147 WinSalMenuItem *pSalMenuItem = ImplGetSalMenuItem( hMenu, nId, FALSE ); 5148 if( pSalMenuItem ) 5149 aMenuEvt.mpMenu = pSalMenuItem->mpMenu; 5150 else 5151 aMenuEvt.mpMenu = NULL; 5152 5153 nRet = pFrame->CallCallback( SALEVENT_MENUHIGHLIGHT, &aMenuEvt ); 5154 } 5155 5156 return (nRet != 0); 5157 } 5158 5159 static int ImplHandleCommand( HWND hWnd, WPARAM wParam, LPARAM ) 5160 { 5161 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5162 if ( !pFrame ) 5163 return 0; 5164 5165 long nRet = 0; 5166 if( !HIWORD(wParam) ) 5167 { 5168 // Menu command 5169 WORD nId = LOWORD(wParam); 5170 if( nId ) // zero for separators 5171 { 5172 SalMenuEvent aMenuEvt; 5173 aMenuEvt.mnId = nId; 5174 WinSalMenuItem *pSalMenuItem = ImplGetSalMenuItem( pFrame->mSelectedhMenu, nId, FALSE ); 5175 if( pSalMenuItem ) 5176 aMenuEvt.mpMenu = pSalMenuItem->mpMenu; 5177 else 5178 aMenuEvt.mpMenu = NULL; 5179 5180 nRet = pFrame->CallCallback( SALEVENT_MENUCOMMAND, &aMenuEvt ); 5181 } 5182 } 5183 return (nRet != 0); 5184 } 5185 5186 static int ImplHandleSysCommand( HWND hWnd, WPARAM wParam, LPARAM lParam ) 5187 { 5188 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5189 if ( !pFrame ) 5190 return 0; 5191 5192 WPARAM nCommand = wParam & 0xFFF0; 5193 5194 if ( pFrame->mbFullScreen ) 5195 { 5196 BOOL bMaximize = IsZoomed( pFrame->mhWnd ); 5197 BOOL bMinimize = IsIconic( pFrame->mhWnd ); 5198 if ( (nCommand == SC_SIZE) || 5199 (!bMinimize && (nCommand == SC_MOVE)) || 5200 (!bMaximize && (nCommand == SC_MAXIMIZE)) || 5201 (bMaximize && (nCommand == SC_RESTORE)) ) 5202 { 5203 MessageBeep( 0 ); 5204 return TRUE; 5205 } 5206 } 5207 5208 if ( nCommand == SC_KEYMENU ) 5209 { 5210 // do not process SC_KEYMENU if we have a native menu 5211 // Windows should handle this 5212 if( GetMenu( hWnd ) ) 5213 return FALSE; 5214 5215 // Hier verarbeiten wir nur KeyMenu-Events fuer Alt um 5216 // den MenuBar zu aktivieren, oder wenn ein SysChild-Fenster 5217 // den Focus hat, da diese Alt+Tasten-Kombinationen nur 5218 // ueber diesen Event verarbeitet werden 5219 if ( !LOWORD( lParam ) ) 5220 { 5221 // Nur ausloesen, wenn keine weitere Taste gedrueckt ist. Im 5222 // Gegensatz zur Doku wird in der X-Koordinaate der CharCode 5223 // geliefert, der zusaetzlich gedrueckt ist 5224 // Also 32 fuer Space, 99 fuer c, 100 fuer d, ... 5225 // Da dies nicht dokumentiert ist, fragen wir vorsichtshalber 5226 // auch den Status der Space-Taste ab 5227 if ( GetKeyState( VK_SPACE ) & 0x8000 ) 5228 return 0; 5229 5230 // Damit nicht bei Alt+Maustaste auch der MenuBar aktiviert wird 5231 if ( (GetKeyState( VK_LBUTTON ) & 0x8000) || 5232 (GetKeyState( VK_RBUTTON ) & 0x8000) || 5233 (GetKeyState( VK_MBUTTON ) & 0x8000) || 5234 (GetKeyState( VK_SHIFT ) & 0x8000) ) 5235 return 1; 5236 5237 SalKeyEvent aKeyEvt; 5238 aKeyEvt.mnTime = GetMessageTime(); 5239 aKeyEvt.mnCode = KEY_MENU; 5240 aKeyEvt.mnCharCode = 0; 5241 aKeyEvt.mnRepeat = 0; 5242 long nRet = pFrame->CallCallback( SALEVENT_KEYINPUT, &aKeyEvt ); 5243 pFrame->CallCallback( SALEVENT_KEYUP, &aKeyEvt ); 5244 return (nRet != 0); 5245 } 5246 else 5247 { 5248 // Testen, ob ein SysChild den Focus hat 5249 HWND hFocusWnd = ::GetFocus(); 5250 if ( hFocusWnd && ImplFindSalObject( hFocusWnd ) ) 5251 { 5252 char cKeyCode = (char)(unsigned char)LOWORD( lParam ); 5253 // LowerCase 5254 if ( (cKeyCode >= 65) && (cKeyCode <= 90) ) 5255 cKeyCode += 32; 5256 // Wir nehmen nur 0-9 und A-Z, alle anderen Tasten muessen durch 5257 // den Hook vom SalObj verarbeitet werden 5258 if ( ((cKeyCode >= 48) && (cKeyCode <= 57)) || 5259 ((cKeyCode >= 97) && (cKeyCode <= 122)) ) 5260 { 5261 sal_uInt16 nModCode = 0; 5262 if ( GetKeyState( VK_SHIFT ) & 0x8000 ) 5263 nModCode |= KEY_SHIFT; 5264 if ( GetKeyState( VK_CONTROL ) & 0x8000 ) 5265 nModCode |= KEY_MOD1; 5266 nModCode |= KEY_MOD2; 5267 5268 SalKeyEvent aKeyEvt; 5269 aKeyEvt.mnTime = GetMessageTime(); 5270 if ( (cKeyCode >= 48) && (cKeyCode <= 57) ) 5271 aKeyEvt.mnCode = KEY_0+(cKeyCode-48); 5272 else 5273 aKeyEvt.mnCode = KEY_A+(cKeyCode-97); 5274 aKeyEvt.mnCode |= nModCode; 5275 aKeyEvt.mnCharCode = cKeyCode; 5276 aKeyEvt.mnRepeat = 0; 5277 long nRet = pFrame->CallCallback( SALEVENT_KEYINPUT, &aKeyEvt ); 5278 pFrame->CallCallback( SALEVENT_KEYUP, &aKeyEvt ); 5279 return (nRet != 0); 5280 } 5281 } 5282 } 5283 } 5284 5285 return FALSE; 5286 } 5287 5288 // ----------------------------------------------------------------------- 5289 5290 static void ImplHandleInputLangChange( HWND hWnd, WPARAM, LPARAM lParam ) 5291 { 5292 ImplSalYieldMutexAcquireWithWait(); 5293 5294 // Feststellen, ob wir IME unterstuetzen 5295 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5296 if ( pFrame && pFrame->mbIME && pFrame->mhDefIMEContext ) 5297 { 5298 HKL hKL = (HKL)lParam; 5299 UINT nImeProps = ImmGetProperty( hKL, IGP_PROPERTY ); 5300 5301 pFrame->mbSpezIME = (nImeProps & IME_PROP_SPECIAL_UI) != 0; 5302 pFrame->mbAtCursorIME = (nImeProps & IME_PROP_AT_CARET) != 0; 5303 pFrame->mbHandleIME = !pFrame->mbSpezIME; 5304 } 5305 5306 // trigger input language and codepage update 5307 UINT nLang = pFrame->mnInputLang; 5308 ImplUpdateInputLang( pFrame ); 5309 5310 // notify change 5311 if( nLang != pFrame->mnInputLang ) 5312 pFrame->CallCallback( SALEVENT_INPUTLANGUAGECHANGE, 0 ); 5313 5314 ImplSalYieldMutexRelease(); 5315 } 5316 5317 // ----------------------------------------------------------------------- 5318 5319 static void ImplUpdateIMECursorPos( WinSalFrame* pFrame, HIMC hIMC ) 5320 { 5321 COMPOSITIONFORM aForm; 5322 memset( &aForm, 0, sizeof( aForm ) ); 5323 5324 // Cursor-Position ermitteln und aus der die Default-Position fuer 5325 // das Composition-Fenster berechnen 5326 SalExtTextInputPosEvent aPosEvt; 5327 pFrame->CallCallback( SALEVENT_EXTTEXTINPUTPOS, (void*)&aPosEvt ); 5328 if ( (aPosEvt.mnX == -1) && (aPosEvt.mnY == -1) ) 5329 aForm.dwStyle |= CFS_DEFAULT; 5330 else 5331 { 5332 aForm.dwStyle |= CFS_POINT; 5333 aForm.ptCurrentPos.x = aPosEvt.mnX; 5334 aForm.ptCurrentPos.y = aPosEvt.mnY; 5335 } 5336 ImmSetCompositionWindow( hIMC, &aForm ); 5337 5338 // Because not all IME's use this values, we create 5339 // a Windows caret to force the Position from the IME 5340 if ( GetFocus() == pFrame->mhWnd ) 5341 { 5342 CreateCaret( pFrame->mhWnd, 0, 5343 aPosEvt.mnWidth, aPosEvt.mnHeight ); 5344 SetCaretPos( aPosEvt.mnX, aPosEvt.mnY ); 5345 } 5346 } 5347 5348 // ----------------------------------------------------------------------- 5349 5350 static sal_Bool ImplHandleIMEStartComposition( HWND hWnd ) 5351 { 5352 sal_Bool bDef = TRUE; 5353 5354 ImplSalYieldMutexAcquireWithWait(); 5355 5356 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5357 if ( pFrame ) 5358 { 5359 HIMC hIMC = ImmGetContext( hWnd ); 5360 if ( hIMC ) 5361 { 5362 ImplUpdateIMECursorPos( pFrame, hIMC ); 5363 ImmReleaseContext( hWnd, hIMC ); 5364 } 5365 5366 if ( pFrame->mbHandleIME ) 5367 { 5368 if ( pFrame->mbAtCursorIME ) 5369 bDef = FALSE; 5370 } 5371 } 5372 5373 ImplSalYieldMutexRelease(); 5374 5375 return bDef; 5376 } 5377 5378 // ----------------------------------------------------------------------- 5379 5380 static sal_Bool ImplHandleIMECompositionInput( WinSalFrame* pFrame, 5381 HIMC hIMC, LPARAM lParam ) 5382 { 5383 sal_Bool bDef = TRUE; 5384 5385 // Init Event 5386 SalExtTextInputEvent aEvt; 5387 aEvt.mnTime = GetMessageTime(); 5388 aEvt.mpTextAttr = NULL; 5389 aEvt.mnCursorPos = 0; 5390 aEvt.mnDeltaStart = 0; 5391 aEvt.mbOnlyCursor = FALSE; 5392 aEvt.mnCursorFlags = 0; 5393 5394 // If we get a result string, then we handle this input 5395 if ( lParam & GCS_RESULTSTR ) 5396 { 5397 bDef = FALSE; 5398 5399 LONG nTextLen = ImmGetCompositionStringW( hIMC, GCS_RESULTSTR, 0, 0 ) / sizeof( WCHAR ); 5400 if ( nTextLen >= 0 ) 5401 { 5402 WCHAR* pTextBuf = new WCHAR[nTextLen]; 5403 ImmGetCompositionStringW( hIMC, GCS_RESULTSTR, pTextBuf, nTextLen*sizeof( WCHAR ) ); 5404 aEvt.maText = XubString( reinterpret_cast<const xub_Unicode*>(pTextBuf), (xub_StrLen)nTextLen ); 5405 delete [] pTextBuf; 5406 } 5407 5408 aEvt.mnCursorPos = aEvt.maText.Len(); 5409 pFrame->CallCallback( SALEVENT_EXTTEXTINPUT, (void*)&aEvt ); 5410 pFrame->CallCallback( SALEVENT_ENDEXTTEXTINPUT, (void*)NULL ); 5411 ImplUpdateIMECursorPos( pFrame, hIMC ); 5412 } 5413 5414 // If the IME doesn't support OnSpot input, then there is nothing to do 5415 if ( !pFrame->mbAtCursorIME ) 5416 return !bDef; 5417 5418 // If we get new Composition data, then we handle this new input 5419 if ( (lParam & (GCS_COMPSTR | GCS_COMPATTR)) || 5420 ((lParam & GCS_CURSORPOS) && !(lParam & GCS_RESULTSTR)) ) 5421 { 5422 bDef = FALSE; 5423 5424 sal_uInt16* pSalAttrAry = NULL; 5425 LONG nTextLen = ImmGetCompositionStringW( hIMC, GCS_COMPSTR, 0, 0 ) / sizeof( WCHAR ); 5426 if ( nTextLen > 0 ) 5427 { 5428 WCHAR* pTextBuf = new WCHAR[nTextLen]; 5429 ImmGetCompositionStringW( hIMC, GCS_COMPSTR, pTextBuf, nTextLen*sizeof( WCHAR ) ); 5430 aEvt.maText = XubString( reinterpret_cast<const xub_Unicode*>(pTextBuf), (xub_StrLen)nTextLen ); 5431 delete [] pTextBuf; 5432 5433 BYTE* pAttrBuf = NULL; 5434 LONG nAttrLen = ImmGetCompositionStringW( hIMC, GCS_COMPATTR, 0, 0 ); 5435 if ( nAttrLen > 0 ) 5436 { 5437 pAttrBuf = new BYTE[nAttrLen]; 5438 ImmGetCompositionStringW( hIMC, GCS_COMPATTR, pAttrBuf, nAttrLen ); 5439 } 5440 5441 if ( pAttrBuf ) 5442 { 5443 xub_StrLen nTextLen = aEvt.maText.Len(); 5444 pSalAttrAry = new sal_uInt16[nTextLen]; 5445 memset( pSalAttrAry, 0, nTextLen*sizeof( sal_uInt16 ) ); 5446 for ( xub_StrLen i = 0; (i < nTextLen) && (i < nAttrLen); i++ ) 5447 { 5448 BYTE nWinAttr = pAttrBuf[i]; 5449 sal_uInt16 nSalAttr; 5450 if ( nWinAttr == ATTR_TARGET_CONVERTED ) 5451 { 5452 nSalAttr = SAL_EXTTEXTINPUT_ATTR_BOLDUNDERLINE; 5453 aEvt.mnCursorFlags |= SAL_EXTTEXTINPUT_CURSOR_INVISIBLE; 5454 } 5455 else if ( nWinAttr == ATTR_CONVERTED ) 5456 nSalAttr = SAL_EXTTEXTINPUT_ATTR_DASHDOTUNDERLINE; 5457 else if ( nWinAttr == ATTR_TARGET_NOTCONVERTED ) 5458 nSalAttr = SAL_EXTTEXTINPUT_ATTR_HIGHLIGHT; 5459 else if ( nWinAttr == ATTR_INPUT_ERROR ) 5460 nSalAttr = SAL_EXTTEXTINPUT_ATTR_REDTEXT | SAL_EXTTEXTINPUT_ATTR_DOTTEDUNDERLINE; 5461 else /* ( nWinAttr == ATTR_INPUT ) */ 5462 nSalAttr = SAL_EXTTEXTINPUT_ATTR_DOTTEDUNDERLINE; 5463 pSalAttrAry[i] = nSalAttr; 5464 } 5465 5466 aEvt.mpTextAttr = pSalAttrAry; 5467 delete [] pAttrBuf; 5468 } 5469 } 5470 5471 // Only when we get new composition data, we must send this event 5472 if ( (nTextLen > 0) || !(lParam & GCS_RESULTSTR) ) 5473 { 5474 // End the mode, if the last character is deleted 5475 if ( !nTextLen && !pFrame->mbCandidateMode ) 5476 { 5477 pFrame->CallCallback( SALEVENT_EXTTEXTINPUT, (void*)&aEvt ); 5478 pFrame->CallCallback( SALEVENT_ENDEXTTEXTINPUT, (void*)NULL ); 5479 } 5480 else 5481 { 5482 // Because Cursor-Position and DeltaStart never updated 5483 // from the korean input engine, we must handle this here 5484 if ( lParam & CS_INSERTCHAR ) 5485 { 5486 aEvt.mnCursorPos = nTextLen; 5487 if ( aEvt.mnCursorPos && (lParam & CS_NOMOVECARET) ) 5488 aEvt.mnCursorPos--; 5489 } 5490 else 5491 aEvt.mnCursorPos = LOWORD( ImmGetCompositionStringW( hIMC, GCS_CURSORPOS, 0, 0 ) ); 5492 5493 if ( pFrame->mbCandidateMode ) 5494 aEvt.mnCursorFlags |= SAL_EXTTEXTINPUT_CURSOR_INVISIBLE; 5495 if ( lParam & CS_NOMOVECARET ) 5496 aEvt.mnCursorFlags |= SAL_EXTTEXTINPUT_CURSOR_OVERWRITE; 5497 5498 pFrame->CallCallback( SALEVENT_EXTTEXTINPUT, (void*)&aEvt ); 5499 } 5500 ImplUpdateIMECursorPos( pFrame, hIMC ); 5501 } 5502 5503 if ( pSalAttrAry ) 5504 delete [] pSalAttrAry; 5505 } 5506 5507 return !bDef; 5508 } 5509 5510 // ----------------------------------------------------------------------- 5511 5512 static sal_Bool ImplHandleIMEComposition( HWND hWnd, LPARAM lParam ) 5513 { 5514 sal_Bool bDef = TRUE; 5515 ImplSalYieldMutexAcquireWithWait(); 5516 5517 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5518 if ( pFrame && (!lParam || (lParam & GCS_RESULTSTR)) ) 5519 { 5520 // Wir restaurieren den Background-Modus bei jeder Texteingabe, 5521 // da einige Tools wie RichWin uns diesen hin- und wieder umsetzen 5522 if ( pFrame->mpGraphics && 5523 pFrame->mpGraphics->mhDC ) 5524 SetBkMode( pFrame->mpGraphics->mhDC, TRANSPARENT ); 5525 } 5526 5527 if ( pFrame && pFrame->mbHandleIME ) 5528 { 5529 if ( !lParam ) 5530 { 5531 SalExtTextInputEvent aEvt; 5532 aEvt.mnTime = GetMessageTime(); 5533 aEvt.mpTextAttr = NULL; 5534 aEvt.mnCursorPos = 0; 5535 aEvt.mnDeltaStart = 0; 5536 aEvt.mbOnlyCursor = FALSE; 5537 aEvt.mnCursorFlags = 0; 5538 pFrame->CallCallback( SALEVENT_EXTTEXTINPUT, (void*)&aEvt ); 5539 pFrame->CallCallback( SALEVENT_ENDEXTTEXTINPUT, (void*)NULL ); 5540 } 5541 else if ( lParam & (GCS_RESULTSTR | GCS_COMPSTR | GCS_COMPATTR | GCS_CURSORPOS) ) 5542 { 5543 HIMC hIMC = ImmGetContext( hWnd ); 5544 if ( hIMC ) 5545 { 5546 if ( ImplHandleIMECompositionInput( pFrame, hIMC, lParam ) ) 5547 bDef = FALSE; 5548 5549 ImmReleaseContext( hWnd, hIMC ); 5550 } 5551 } 5552 } 5553 5554 ImplSalYieldMutexRelease(); 5555 return bDef; 5556 } 5557 5558 // ----------------------------------------------------------------------- 5559 5560 static sal_Bool ImplHandleIMEEndComposition( HWND hWnd ) 5561 { 5562 sal_Bool bDef = TRUE; 5563 5564 ImplSalYieldMutexAcquireWithWait(); 5565 5566 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5567 if ( pFrame && pFrame->mbHandleIME ) 5568 { 5569 if ( pFrame->mbAtCursorIME ) 5570 bDef = FALSE; 5571 } 5572 5573 ImplSalYieldMutexRelease(); 5574 5575 return bDef; 5576 } 5577 5578 // ----------------------------------------------------------------------- 5579 5580 static boolean ImplHandleAppCommand( HWND hWnd, LPARAM lParam ) 5581 { 5582 sal_Int16 nCommand = 0; 5583 switch( GET_APPCOMMAND_LPARAM(lParam) ) 5584 { 5585 case APPCOMMAND_MEDIA_CHANNEL_DOWN: nCommand = MEDIA_COMMAND_CHANNEL_DOWN; break; 5586 case APPCOMMAND_MEDIA_CHANNEL_UP: nCommand = MEDIA_COMMAND_CHANNEL_UP; break; 5587 case APPCOMMAND_MEDIA_NEXTTRACK: nCommand = MEDIA_COMMAND_NEXTTRACK; break; 5588 case APPCOMMAND_MEDIA_PAUSE: nCommand = MEDIA_COMMAND_PAUSE; break; 5589 case APPCOMMAND_MEDIA_PLAY: nCommand = MEDIA_COMMAND_PLAY; break; 5590 case APPCOMMAND_MEDIA_PLAY_PAUSE: nCommand = MEDIA_COMMAND_PLAY_PAUSE; break; 5591 case APPCOMMAND_MEDIA_PREVIOUSTRACK: nCommand = MEDIA_COMMAND_PREVIOUSTRACK; break; 5592 case APPCOMMAND_MEDIA_RECORD: nCommand = MEDIA_COMMAND_RECORD; break; 5593 case APPCOMMAND_MEDIA_REWIND: nCommand = MEDIA_COMMAND_REWIND; break; 5594 case APPCOMMAND_MEDIA_STOP: nCommand = MEDIA_COMMAND_STOP; break; 5595 case APPCOMMAND_MIC_ON_OFF_TOGGLE: nCommand = MEDIA_COMMAND_MIC_ON_OFF_TOGGLE; break; 5596 case APPCOMMAND_MICROPHONE_VOLUME_DOWN: nCommand = MEDIA_COMMAND_MICROPHONE_VOLUME_DOWN; break; 5597 case APPCOMMAND_MICROPHONE_VOLUME_MUTE: nCommand = MEDIA_COMMAND_MICROPHONE_VOLUME_MUTE; break; 5598 case APPCOMMAND_MICROPHONE_VOLUME_UP: nCommand = MEDIA_COMMAND_MICROPHONE_VOLUME_UP; break; 5599 case APPCOMMAND_VOLUME_DOWN: nCommand = MEDIA_COMMAND_VOLUME_DOWN; break; 5600 case APPCOMMAND_VOLUME_MUTE: nCommand = MEDIA_COMMAND_VOLUME_MUTE; break; 5601 case APPCOMMAND_VOLUME_UP: nCommand = MEDIA_COMMAND_VOLUME_UP; break; 5602 break; 5603 default: 5604 return false; 5605 } 5606 5607 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5608 Window *pWindow = pFrame ? pFrame->GetWindow() : NULL; 5609 5610 if( pWindow ) 5611 { 5612 const Point aPoint; 5613 CommandEvent aCEvt( aPoint, COMMAND_MEDIA, FALSE, &nCommand ); 5614 NotifyEvent aNCmdEvt( EVENT_COMMAND, pWindow, &aCEvt ); 5615 5616 if ( !ImplCallPreNotify( aNCmdEvt ) ) 5617 { 5618 pWindow->Command( aCEvt ); 5619 return true; 5620 } 5621 } 5622 5623 return false; 5624 } 5625 5626 5627 static void ImplHandleIMENotify( HWND hWnd, WPARAM wParam ) 5628 { 5629 if ( wParam == (WPARAM)IMN_OPENCANDIDATE ) 5630 { 5631 ImplSalYieldMutexAcquireWithWait(); 5632 5633 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5634 if ( pFrame && pFrame->mbHandleIME && 5635 pFrame->mbAtCursorIME ) 5636 { 5637 // Wir wollen den Cursor hiden 5638 pFrame->mbCandidateMode = TRUE; 5639 ImplHandleIMEComposition( hWnd, GCS_CURSORPOS ); 5640 5641 HWND hWnd = pFrame->mhWnd; 5642 HIMC hIMC = ImmGetContext( hWnd ); 5643 if ( hIMC ) 5644 { 5645 LONG nBufLen = ImmGetCompositionStringW( hIMC, GCS_COMPSTR, 0, 0 ); 5646 if ( nBufLen >= 1 ) 5647 { 5648 SalExtTextInputPosEvent aPosEvt; 5649 pFrame->CallCallback( SALEVENT_EXTTEXTINPUTPOS, (void*)&aPosEvt ); 5650 5651 // Vertical !!! 5652 CANDIDATEFORM aForm; 5653 aForm.dwIndex = 0; 5654 aForm.dwStyle = CFS_EXCLUDE; 5655 aForm.ptCurrentPos.x = aPosEvt.mnX; 5656 aForm.ptCurrentPos.y = aPosEvt.mnY+1; 5657 aForm.rcArea.left = aPosEvt.mnX; 5658 aForm.rcArea.top = aPosEvt.mnY; 5659 aForm.rcArea.right = aForm.rcArea.left+aPosEvt.mnExtWidth+1; 5660 aForm.rcArea.bottom = aForm.rcArea.top+aPosEvt.mnHeight+1; 5661 ImmSetCandidateWindow( hIMC, &aForm ); 5662 } 5663 5664 ImmReleaseContext( hWnd, hIMC ); 5665 } 5666 } 5667 5668 ImplSalYieldMutexRelease(); 5669 } 5670 else if ( wParam == (WPARAM)IMN_CLOSECANDIDATE ) 5671 { 5672 ImplSalYieldMutexAcquireWithWait(); 5673 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5674 if ( pFrame ) 5675 pFrame->mbCandidateMode = FALSE; 5676 ImplSalYieldMutexRelease(); 5677 } 5678 } 5679 5680 // ----------------------------------------------------------------------- 5681 #if WINVER >= 0x0500 5682 5683 static LRESULT ImplHandleIMEReconvertString( HWND hWnd, LPARAM lParam ) 5684 { 5685 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5686 LPRECONVERTSTRING pReconvertString = (LPRECONVERTSTRING) lParam; 5687 LRESULT nRet = 0; 5688 SalSurroundingTextRequestEvent aEvt; 5689 aEvt.maText = UniString(); 5690 aEvt.mnStart = aEvt.mnEnd = 0; 5691 5692 UINT nImeProps = ImmGetProperty( GetKeyboardLayout( 0 ), IGP_SETCOMPSTR ); 5693 if( (nImeProps & SCS_CAP_SETRECONVERTSTRING) == 0 ) 5694 { 5695 // This IME does not support reconversion. 5696 return 0; 5697 } 5698 5699 if( !pReconvertString ) 5700 { 5701 // The first call for reconversion. 5702 pFrame->CallCallback( SALEVENT_STARTRECONVERSION, (void*)NULL ); 5703 5704 // Retrieve the surrounding text from the focused control. 5705 pFrame->CallCallback( SALEVENT_SURROUNDINGTEXTREQUEST, (void*)&aEvt ); 5706 5707 if( aEvt.maText.Len() == 0 ) 5708 { 5709 return 0; 5710 } 5711 5712 nRet = sizeof(RECONVERTSTRING) + (aEvt.maText.Len() + 1) * sizeof(WCHAR); 5713 } 5714 else 5715 { 5716 // The second call for reconversion. 5717 5718 // Retrieve the surrounding text from the focused control. 5719 pFrame->CallCallback( SALEVENT_SURROUNDINGTEXTREQUEST, (void*)&aEvt ); 5720 nRet = sizeof(RECONVERTSTRING) + (aEvt.maText.Len() + 1) * sizeof(WCHAR); 5721 5722 pReconvertString->dwStrOffset = sizeof(RECONVERTSTRING); 5723 pReconvertString->dwStrLen = aEvt.maText.Len(); 5724 pReconvertString->dwCompStrOffset = aEvt.mnStart * sizeof(WCHAR); 5725 pReconvertString->dwCompStrLen = aEvt.mnEnd - aEvt.mnStart; 5726 pReconvertString->dwTargetStrOffset = pReconvertString->dwCompStrOffset; 5727 pReconvertString->dwTargetStrLen = pReconvertString->dwCompStrLen; 5728 5729 memcpy( (LPWSTR)(pReconvertString + 1), aEvt.maText.GetBuffer(), (aEvt.maText.Len() + 1) * sizeof(WCHAR) ); 5730 } 5731 5732 // just return the required size of buffer to reconvert. 5733 return nRet; 5734 } 5735 5736 // ----------------------------------------------------------------------- 5737 5738 static LRESULT ImplHandleIMEConfirmReconvertString( HWND hWnd, LPARAM lParam ) 5739 { 5740 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 5741 LPRECONVERTSTRING pReconvertString = (LPRECONVERTSTRING) lParam; 5742 SalSurroundingTextRequestEvent aEvt; 5743 aEvt.maText = UniString(); 5744 aEvt.mnStart = aEvt.mnEnd = 0; 5745 5746 pFrame->CallCallback( SALEVENT_SURROUNDINGTEXTREQUEST, (void*)&aEvt ); 5747 5748 sal_uLong nTmpStart = pReconvertString->dwCompStrOffset / sizeof(WCHAR); 5749 sal_uLong nTmpEnd = nTmpStart + pReconvertString->dwCompStrLen; 5750 5751 if( nTmpStart != aEvt.mnStart || nTmpEnd != aEvt.mnEnd ) 5752 { 5753 SalSurroundingTextSelectionChangeEvent aSelEvt; 5754 aSelEvt.mnStart = nTmpStart; 5755 aSelEvt.mnEnd = nTmpEnd; 5756 5757 pFrame->CallCallback( SALEVENT_SURROUNDINGTEXTSELECTIONCHANGE, (void*)&aSelEvt ); 5758 } 5759 5760 return TRUE; 5761 } 5762 5763 #endif // WINVER >= 0x0500 5764 5765 // ----------------------------------------------------------------------- 5766 5767 void SalTestMouseLeave() 5768 { 5769 SalData* pSalData = GetSalData(); 5770 5771 if ( pSalData->mhWantLeaveMsg && !::GetCapture() ) 5772 { 5773 POINT aPt; 5774 GetCursorPos( &aPt ); 5775 if ( pSalData->mhWantLeaveMsg != WindowFromPoint( aPt ) ) 5776 ImplSendMessage( pSalData->mhWantLeaveMsg, SAL_MSG_MOUSELEAVE, 0, MAKELPARAM( aPt.x, aPt.y ) ); 5777 } 5778 } 5779 5780 // ----------------------------------------------------------------------- 5781 5782 static int ImplSalWheelMousePos( HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam , 5783 LRESULT& rResult ) 5784 { 5785 POINT aPt; 5786 POINT aScreenPt; 5787 aScreenPt.x = (short)LOWORD( lParam ); 5788 aScreenPt.y = (short)HIWORD( lParam ); 5789 // Child-Fenster suchen, welches an der entsprechenden 5790 // Position liegt 5791 HWND hChildWnd; 5792 HWND hWheelWnd = hWnd; 5793 do 5794 { 5795 hChildWnd = hWheelWnd; 5796 aPt = aScreenPt; 5797 ScreenToClient( hChildWnd, &aPt ); 5798 hWheelWnd = ChildWindowFromPointEx( hChildWnd, aPt, CWP_SKIPINVISIBLE | CWP_SKIPTRANSPARENT ); 5799 } 5800 while ( hWheelWnd && (hWheelWnd != hChildWnd) ); 5801 if ( hWheelWnd && (hWheelWnd != hWnd) && 5802 (hWheelWnd != ::GetFocus()) && IsWindowEnabled( hWheelWnd ) ) 5803 { 5804 rResult = ImplSendMessage( hWheelWnd, nMsg, wParam, lParam ); 5805 return FALSE; 5806 } 5807 5808 return TRUE; 5809 } 5810 5811 // ----------------------------------------------------------------------- 5812 5813 LRESULT CALLBACK SalFrameWndProc( HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam, int& rDef ) 5814 { 5815 LRESULT nRet = 0; 5816 static int bInWheelMsg = FALSE; 5817 static int bInQueryEnd = FALSE; 5818 5819 // By WM_CRETAE we connect the frame with the window handle 5820 if ( nMsg == WM_CREATE ) 5821 { 5822 // Window-Instanz am Windowhandle speichern 5823 // Can also be used for the W-Version, because the struct 5824 // to access lpCreateParams is the same structure 5825 CREATESTRUCTA* pStruct = (CREATESTRUCTA*)lParam; 5826 WinSalFrame* pFrame = (WinSalFrame*)pStruct->lpCreateParams; 5827 if ( pFrame != 0 ) 5828 { 5829 SetWindowPtr( hWnd, pFrame ); 5830 // HWND schon hier setzen, da schon auf den Instanzdaten 5831 // gearbeitet werden kann, wenn Messages waehrend 5832 // CreateWindow() gesendet werden 5833 pFrame->mhWnd = hWnd; 5834 pFrame->maSysData.hWnd = hWnd; 5835 } 5836 return 0; 5837 } 5838 5839 ImplSVData* pSVData = ImplGetSVData(); 5840 // #i72707# TODO: the mbDeInit check will not be needed 5841 // once all windows that are not properly closed on exit got fixed 5842 if( pSVData->mbDeInit ) 5843 return 0; 5844 5845 if ( WM_USER_SYSTEM_WINDOW_ACTIVATED == nMsg ) 5846 { 5847 if (pSVData->mpIntroWindow) 5848 pSVData->mpIntroWindow->Hide(); 5849 5850 return 0; 5851 } 5852 5853 bool bCheckTimers = false; 5854 5855 switch( nMsg ) 5856 { 5857 case WM_MOUSEMOVE: 5858 case WM_LBUTTONDOWN: 5859 case WM_MBUTTONDOWN: 5860 case WM_RBUTTONDOWN: 5861 case WM_LBUTTONUP: 5862 case WM_MBUTTONUP: 5863 case WM_RBUTTONUP: 5864 case WM_NCMOUSEMOVE: 5865 case SAL_MSG_MOUSELEAVE: 5866 ImplSalYieldMutexAcquireWithWait(); 5867 rDef = !ImplHandleMouseMsg( hWnd, nMsg, wParam, lParam ); 5868 ImplSalYieldMutexRelease(); 5869 break; 5870 5871 case WM_NCLBUTTONDOWN: 5872 case WM_NCMBUTTONDOWN: 5873 case WM_NCRBUTTONDOWN: 5874 ImplSalYieldMutexAcquireWithWait(); 5875 ImplCallClosePopupsHdl( hWnd ); // close popups... 5876 ImplSalYieldMutexRelease(); 5877 break; 5878 5879 case WM_MOUSEACTIVATE: 5880 if ( LOWORD( lParam ) == HTCLIENT ) 5881 { 5882 ImplSalYieldMutexAcquireWithWait(); 5883 nRet = ImplHandleMouseActivateMsg( hWnd ); 5884 ImplSalYieldMutexRelease(); 5885 if ( nRet ) 5886 { 5887 nRet = MA_NOACTIVATE; 5888 rDef = FALSE; 5889 } 5890 } 5891 break; 5892 5893 case WM_KEYDOWN: 5894 case WM_KEYUP: 5895 case WM_DEADCHAR: 5896 case WM_CHAR: 5897 case WM_UNICHAR: // MCD, 2003-01-13, Support for WM_UNICHAR & Keyman 6.0 5898 case WM_SYSKEYDOWN: 5899 case WM_SYSKEYUP: 5900 case WM_SYSCHAR: 5901 ImplSalYieldMutexAcquireWithWait(); 5902 rDef = !ImplHandleKeyMsg( hWnd, nMsg, wParam, lParam, nRet ); 5903 ImplSalYieldMutexRelease(); 5904 break; 5905 5906 case WM_MOUSEWHEEL: 5907 // FALLTHROUGH intended 5908 case WM_MOUSEHWHEEL: 5909 // Gegen Rekursion absichern, falls wir vom IE oder dem externen 5910 // Fenster die Message wieder zurueckbekommen 5911 if ( !bInWheelMsg ) 5912 { 5913 bInWheelMsg++; 5914 rDef = !ImplHandleWheelMsg( hWnd, nMsg, wParam, lParam ); 5915 // Wenn wir die Message nicht ausgewertet haben, schauen wir 5916 // noch einmal nach, ob dort ein geplugtes Fenster steht, 5917 // welches wir dann benachrichtigen 5918 if ( rDef ) 5919 rDef = ImplSalWheelMousePos( hWnd, nMsg, wParam, lParam, nRet ); 5920 bInWheelMsg--; 5921 } 5922 break; 5923 5924 case WM_COMMAND: 5925 ImplSalYieldMutexAcquireWithWait(); 5926 rDef = !ImplHandleCommand( hWnd, wParam, lParam ); 5927 ImplSalYieldMutexRelease(); 5928 break; 5929 5930 case WM_INITMENUPOPUP: 5931 ImplSalYieldMutexAcquireWithWait(); 5932 rDef = !ImplHandleMenuActivate( hWnd, wParam, lParam ); 5933 ImplSalYieldMutexRelease(); 5934 break; 5935 5936 case WM_MENUSELECT: 5937 ImplSalYieldMutexAcquireWithWait(); 5938 rDef = !ImplHandleMenuSelect( hWnd, wParam, lParam ); 5939 ImplSalYieldMutexRelease(); 5940 break; 5941 5942 case WM_SYSCOMMAND: 5943 ImplSalYieldMutexAcquireWithWait(); 5944 nRet = ImplHandleSysCommand( hWnd, wParam, lParam ); 5945 ImplSalYieldMutexRelease(); 5946 if ( nRet ) 5947 rDef = FALSE; 5948 break; 5949 5950 case WM_MENUCHAR: 5951 nRet = ImplMenuChar( hWnd, wParam, lParam ); 5952 if( nRet ) 5953 rDef = FALSE; 5954 break; 5955 5956 case WM_MEASUREITEM: 5957 nRet = ImplMeasureItem(hWnd, wParam, lParam); 5958 if( nRet ) 5959 rDef = FALSE; 5960 break; 5961 5962 case WM_DRAWITEM: 5963 nRet = ImplDrawItem(hWnd, wParam, lParam); 5964 if( nRet ) 5965 rDef = FALSE; 5966 break; 5967 5968 case WM_MOVE: 5969 case SAL_MSG_POSTMOVE: 5970 ImplHandleMoveMsg( hWnd ); 5971 rDef = FALSE; 5972 break; 5973 case WM_SIZE: 5974 ImplHandleSizeMsg( hWnd, wParam, lParam ); 5975 rDef = FALSE; 5976 break; 5977 case SAL_MSG_POSTCALLSIZE: 5978 ImplCallSizeHdl( hWnd ); 5979 rDef = FALSE; 5980 break; 5981 5982 case WM_GETMINMAXINFO: 5983 if ( ImplHandleMinMax( hWnd, lParam ) ) 5984 rDef = FALSE; 5985 break; 5986 5987 case WM_ERASEBKGND: 5988 nRet = 1; 5989 rDef = FALSE; 5990 break; 5991 case WM_PAINT: 5992 bCheckTimers = ImplHandlePaintMsg( hWnd ); 5993 rDef = FALSE; 5994 break; 5995 case SAL_MSG_POSTPAINT: 5996 ImplHandlePaintMsg2( hWnd, (RECT*)wParam ); 5997 bCheckTimers = true; 5998 rDef = FALSE; 5999 break; 6000 6001 case SAL_MSG_FORCEPALETTE: 6002 ImplHandleForcePalette( hWnd ); 6003 rDef = FALSE; 6004 break; 6005 6006 case WM_QUERYNEWPALETTE: 6007 case SAL_MSG_POSTQUERYNEWPAL: 6008 nRet = ImplHandlePalette( TRUE, hWnd, nMsg, wParam, lParam, rDef ); 6009 break; 6010 6011 case WM_ACTIVATE: 6012 // Wenn wir aktiviert werden, dann wollen wir auch unsere 6013 // Palette setzen. Wir machen dieses in Activate, 6014 // damit andere externe Child-Fenster auch unsere Palette 6015 // ueberschreiben koennen. So wird unsere jedenfalls nur einmal 6016 // gesetzt und nicht immer rekursiv, da an allen anderen Stellen 6017 // diese nur als Background-Palette gesetzt wird 6018 if ( LOWORD( wParam ) != WA_INACTIVE ) 6019 ImplSendMessage( hWnd, SAL_MSG_FORCEPALETTE, 0, 0 ); 6020 break; 6021 6022 case WM_ENABLE: 6023 // #95133# a system dialog is opened/closed, using our app window as parent 6024 { 6025 WinSalFrame* pFrame = GetWindowPtr( hWnd ); 6026 Window *pWin = NULL; 6027 if( pFrame ) 6028 pWin = pFrame->GetWindow(); 6029 6030 if( !wParam ) 6031 { 6032 ImplSVData* pSVData = ImplGetSVData(); 6033 pSVData->maAppData.mnModalMode++; 6034 6035 // #106431#, hide SplashScreen 6036 if( pSVData->mpIntroWindow ) 6037 pSVData->mpIntroWindow->Hide(); 6038 6039 if( pWin ) 6040 { 6041 pWin->EnableInput( FALSE, TRUE, TRUE, NULL ); 6042 pWin->ImplIncModalCount(); // #106303# support frame based modal count 6043 } 6044 } 6045 else 6046 { 6047 ImplGetSVData()->maAppData.mnModalMode--; 6048 if( pWin ) 6049 { 6050 pWin->EnableInput( TRUE, TRUE, TRUE, NULL ); 6051 pWin->ImplDecModalCount(); // #106303# support frame based modal count 6052 } 6053 } 6054 } 6055 break; 6056 6057 case WM_KILLFOCUS: 6058 DestroyCaret(); 6059 case WM_SETFOCUS: 6060 case SAL_MSG_POSTFOCUS: 6061 ImplHandleFocusMsg( hWnd ); 6062 rDef = FALSE; 6063 break; 6064 6065 case WM_CLOSE: 6066 ImplHandleCloseMsg( hWnd ); 6067 rDef = FALSE; 6068 break; 6069 6070 case WM_QUERYENDSESSION: 6071 if( !bInQueryEnd ) 6072 { 6073 // handle queryendsession only once 6074 bInQueryEnd = TRUE; 6075 nRet = !ImplHandleShutDownMsg( hWnd ); 6076 rDef = FALSE; 6077 6078 // Issue #16314#: ImplHandleShutDownMsg causes a PostMessage in case of allowing shutdown. 6079 // This posted message was never processed and cause Windows XP to hang after log off 6080 // if there are multiple sessions and the current session wasn't the first one started. 6081 // So if shutdown is allowed we assume that a post message was done and retrieve all 6082 // messages in the message queue and dispatch them before we return control to the system. 6083 6084 if ( nRet ) 6085 { 6086 MSG msg; 6087 6088 while( PeekMessage( &msg, NULL, 0, 0, PM_REMOVE ) ) 6089 { 6090 DispatchMessage( &msg ); 6091 } 6092 } 6093 } 6094 else 6095 { 6096 ImplSalYieldMutexAcquireWithWait(); 6097 ImplSalYieldMutexRelease(); 6098 rDef = TRUE; 6099 } 6100 break; 6101 6102 case WM_ENDSESSION: 6103 if( !wParam ) 6104 bInQueryEnd = FALSE; // no shutdown: allow query again 6105 nRet = FALSE; 6106 rDef = FALSE; 6107 break; 6108 6109 case WM_DISPLAYCHANGE: 6110 case WM_SETTINGCHANGE: 6111 case WM_DEVMODECHANGE: 6112 case WM_FONTCHANGE: 6113 case WM_SYSCOLORCHANGE: 6114 case WM_TIMECHANGE: 6115 ImplHandleSettingsChangeMsg( hWnd, nMsg, wParam, lParam ); 6116 break; 6117 6118 case WM_THEMECHANGED: 6119 GetSalData()->mbThemeChanged = TRUE; 6120 break; 6121 6122 case SAL_MSG_USEREVENT: 6123 ImplHandleUserEvent( hWnd, lParam ); 6124 rDef = FALSE; 6125 break; 6126 6127 case SAL_MSG_CAPTUREMOUSE: 6128 SetCapture( hWnd ); 6129 rDef = FALSE; 6130 break; 6131 case SAL_MSG_RELEASEMOUSE: 6132 if ( ::GetCapture() == hWnd ) 6133 ReleaseCapture(); 6134 rDef = FALSE; 6135 break; 6136 case SAL_MSG_TOTOP: 6137 ImplSalToTop( hWnd, (sal_uInt16)wParam ); 6138 rDef = FALSE; 6139 break; 6140 case SAL_MSG_SHOW: 6141 ImplSalShow( hWnd, (sal_Bool)wParam, (sal_Bool)lParam ); 6142 rDef = FALSE; 6143 break; 6144 case SAL_MSG_SETINPUTCONTEXT: 6145 ImplSalFrameSetInputContext( hWnd, (const SalInputContext*)(void*)lParam ); 6146 rDef = FALSE; 6147 break; 6148 case SAL_MSG_ENDEXTTEXTINPUT: 6149 ImplSalFrameEndExtTextInput( hWnd, (sal_uInt16)(sal_uLong)(void*)wParam ); 6150 rDef = FALSE; 6151 break; 6152 6153 case WM_INPUTLANGCHANGE: 6154 ImplHandleInputLangChange( hWnd, wParam, lParam ); 6155 break; 6156 6157 case WM_IME_CHAR: 6158 // #103487#, some IMEs (eg, those that do not work onspot) 6159 // may send WM_IME_CHAR instead of WM_IME_COMPOSITION 6160 // we just handle it like a WM_CHAR message - seems to work fine 6161 ImplSalYieldMutexAcquireWithWait(); 6162 rDef = !ImplHandleKeyMsg( hWnd, WM_CHAR, wParam, lParam, nRet ); 6163 ImplSalYieldMutexRelease(); 6164 break; 6165 6166 case WM_IME_STARTCOMPOSITION: 6167 rDef = ImplHandleIMEStartComposition( hWnd ); 6168 break; 6169 6170 case WM_IME_COMPOSITION: 6171 rDef = ImplHandleIMEComposition( hWnd, lParam ); 6172 break; 6173 6174 case WM_IME_ENDCOMPOSITION: 6175 rDef = ImplHandleIMEEndComposition( hWnd ); 6176 break; 6177 6178 case WM_IME_NOTIFY: 6179 ImplHandleIMENotify( hWnd, wParam ); 6180 break; 6181 case WM_APPCOMMAND: 6182 if( ImplHandleAppCommand( hWnd, lParam ) ) 6183 { 6184 rDef = false; 6185 nRet = 1; 6186 } 6187 break; 6188 #if WINVER >= 0x0500 6189 case WM_IME_REQUEST: 6190 if ( PtrToInt( wParam ) == IMR_RECONVERTSTRING ) 6191 { 6192 nRet = ImplHandleIMEReconvertString( hWnd, lParam ); 6193 rDef = FALSE; 6194 } 6195 else if( PtrToInt( wParam ) == IMR_CONFIRMRECONVERTSTRING ) 6196 { 6197 nRet = ImplHandleIMEConfirmReconvertString( hWnd, lParam ); 6198 rDef = FALSE; 6199 } 6200 break; 6201 #endif // WINVER >= 0x0500 6202 } 6203 6204 // WheelMouse-Message abfangen 6205 if ( rDef && (nMsg == aSalShlData.mnWheelMsgId) && aSalShlData.mnWheelMsgId ) 6206 { 6207 // Gegen Rekursion absichern, falls wir vom IE oder dem externen 6208 // Fenster die Message wieder zurueckbekommen 6209 if ( !bInWheelMsg ) 6210 { 6211 bInWheelMsg++; 6212 // Zuerst wollen wir die Message dispatchen und dann darf auch 6213 // das SystemWindow drankommen 6214 WORD nKeyState = 0; 6215 if ( GetKeyState( VK_SHIFT ) & 0x8000 ) 6216 nKeyState |= MK_SHIFT; 6217 if ( GetKeyState( VK_CONTROL ) & 0x8000 ) 6218 nKeyState |= MK_CONTROL; 6219 // Mutex handling is inside from this call 6220 rDef = !ImplHandleWheelMsg( hWnd, 6221 WM_MOUSEWHEEL, 6222 MAKEWPARAM( nKeyState, (WORD)wParam ), 6223 lParam ); 6224 if ( rDef ) 6225 { 6226 HWND hWheelWnd = ::GetFocus(); 6227 if ( hWheelWnd && (hWheelWnd != hWnd) ) 6228 { 6229 nRet = ImplSendMessage( hWheelWnd, nMsg, wParam, lParam ); 6230 rDef = FALSE; 6231 } 6232 else 6233 rDef = ImplSalWheelMousePos( hWnd, nMsg, wParam, lParam, nRet ); 6234 } 6235 bInWheelMsg--; 6236 } 6237 } 6238 6239 if( bCheckTimers ) 6240 { 6241 SalData* pSalData = GetSalData(); 6242 if( pSalData->mnNextTimerTime ) 6243 { 6244 DWORD nCurTime = GetTickCount(); 6245 if( pSalData->mnNextTimerTime < nCurTime ) 6246 { 6247 MSG aMsg; 6248 if( ! ImplPeekMessage( &aMsg, 0, WM_PAINT, WM_PAINT, PM_NOREMOVE | PM_NOYIELD ) ) 6249 ImplPostMessage( pSalData->mpFirstInstance->mhComWnd, SAL_MSG_POSTTIMER, 0, nCurTime ); 6250 } 6251 } 6252 } 6253 6254 return nRet; 6255 } 6256 6257 LRESULT CALLBACK SalFrameWndProcA( HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam ) 6258 { 6259 int bDef = TRUE; 6260 LRESULT nRet = 0; 6261 #ifdef __MINGW32__ 6262 jmp_buf jmpbuf; 6263 __SEHandler han; 6264 if (__builtin_setjmp(jmpbuf) == 0) 6265 { 6266 han.Set(jmpbuf, NULL, (__SEHandler::PF)EXCEPTION_EXECUTE_HANDLER); 6267 #else 6268 __try 6269 { 6270 #endif 6271 nRet = SalFrameWndProc( hWnd, nMsg, wParam, lParam, bDef ); 6272 } 6273 #ifdef __MINGW32__ 6274 han.Reset(); 6275 #else 6276 __except(WinSalInstance::WorkaroundExceptionHandlingInUSER32Lib(GetExceptionCode(), GetExceptionInformation())) 6277 { 6278 } 6279 #endif 6280 if ( bDef ) 6281 nRet = DefWindowProcA( hWnd, nMsg, wParam, lParam ); 6282 return nRet; 6283 } 6284 6285 LRESULT CALLBACK SalFrameWndProcW( HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam ) 6286 { 6287 int bDef = TRUE; 6288 LRESULT nRet = 0; 6289 #ifdef __MINGW32__ 6290 jmp_buf jmpbuf; 6291 __SEHandler han; 6292 if (__builtin_setjmp(jmpbuf) == 0) 6293 { 6294 han.Set(jmpbuf, NULL, (__SEHandler::PF)EXCEPTION_EXECUTE_HANDLER); 6295 #else 6296 __try 6297 { 6298 #endif 6299 nRet = SalFrameWndProc( hWnd, nMsg, wParam, lParam, bDef ); 6300 } 6301 #ifdef __MINGW32__ 6302 han.Reset(); 6303 #else 6304 __except(WinSalInstance::WorkaroundExceptionHandlingInUSER32Lib(GetExceptionCode(), GetExceptionInformation())) 6305 { 6306 } 6307 #endif 6308 6309 if ( bDef ) 6310 nRet = DefWindowProcW( hWnd, nMsg, wParam, lParam ); 6311 return nRet; 6312 } 6313 6314 // ----------------------------------------------------------------------- 6315 6316 sal_Bool ImplHandleGlobalMsg( HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam, LRESULT& rlResult ) 6317 { 6318 // handle all messages concerning all frames so they get processed only once 6319 // Must work for Unicode and none Unicode 6320 sal_Bool bResult = FALSE; 6321 if ( (nMsg == WM_PALETTECHANGED) || (nMsg == SAL_MSG_POSTPALCHANGED) ) 6322 { 6323 int bDef = TRUE; 6324 rlResult = ImplHandlePalette( FALSE, hWnd, nMsg, wParam, lParam, bDef ); 6325 bResult = (bDef != 0); 6326 } 6327 else if( nMsg == WM_DISPLAYCHANGE ) 6328 { 6329 WinSalSystem* pSys = static_cast<WinSalSystem*>(ImplGetSalSystem()); 6330 if( pSys ) 6331 pSys->clearMonitors(); 6332 bResult = (pSys != NULL); 6333 } 6334 return bResult; 6335 } 6336 6337 // ----------------------------------------------------------------------- 6338 6339 sal_Bool ImplWriteLastError( DWORD lastError, const char *szApiCall ) 6340 { 6341 static int first=1; 6342 // if VCL_LOGFILE_ENABLED is set, Win32 API error messages can be written 6343 // to %TMP%/vcl.log or %TEMP%/vcl.log 6344 static char *logEnabled = getenv("VCL_LOGFILE_ENABLED"); 6345 if( logEnabled ) 6346 { 6347 sal_Bool bSuccess = FALSE; 6348 static char *szTmp = getenv("TMP"); 6349 if( !szTmp || !*szTmp ) 6350 szTmp = getenv("TEMP"); 6351 if( szTmp && *szTmp ) 6352 { 6353 char fname[5000]; 6354 strcpy( fname, szTmp ); 6355 if( fname[strlen(fname) - 1] != '\\' ) 6356 strcat( fname, "\\"); 6357 strcat( fname, "vcl.log" ); 6358 FILE *fp = fopen( fname, "a" ); // always append 6359 if( fp ) 6360 { 6361 if( first ) 6362 { 6363 first = 0; 6364 fprintf( fp, "Process ID: %d (0x%x)\n", GetCurrentProcessId(), GetCurrentProcessId() ); 6365 } 6366 time_t aclock; 6367 time( &aclock ); // Get time in seconds 6368 struct tm *newtime = localtime( &aclock ); // Convert time to struct tm form 6369 fprintf( fp, asctime( newtime ) ); // print time stamp 6370 6371 fprintf( fp, "%s returned %u (0x%x)\n", szApiCall, lastError, lastError ); 6372 bSuccess = TRUE; // may be FormatMessage fails but we wrote at least the error code 6373 6374 LPVOID lpMsgBuf; 6375 if (FormatMessageA( 6376 FORMAT_MESSAGE_ALLOCATE_BUFFER | 6377 FORMAT_MESSAGE_FROM_SYSTEM | 6378 FORMAT_MESSAGE_IGNORE_INSERTS, 6379 NULL, 6380 lastError, 6381 MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language 6382 (LPSTR) &lpMsgBuf, 6383 0, 6384 NULL )) 6385 { 6386 fprintf( fp, " %s\n", (LPSTR)lpMsgBuf ); 6387 LocalFree( lpMsgBuf ); 6388 } 6389 6390 fclose( fp ); 6391 } 6392 } 6393 return bSuccess; 6394 } 6395 else 6396 return TRUE; 6397 } 6398 6399 // ----------------------------------------------------------------------- 6400 6401