The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

3009 lines
103KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library - "Jules' Utility Class Extensions"
  4. Copyright 2004-11 by Raw Material Software Ltd.
  5. ------------------------------------------------------------------------------
  6. JUCE can be redistributed and/or modified under the terms of the GNU General
  7. Public License (Version 2), as published by the Free Software Foundation.
  8. A copy of the license is included in the JUCE distribution, or can be found
  9. online at www.gnu.org/licenses.
  10. JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
  11. WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  12. A PARTICULAR PURPOSE. See the GNU General Public License for more details.
  13. ------------------------------------------------------------------------------
  14. To release a closed-source product which uses JUCE, commercial licenses are
  15. available: visit www.rawmaterialsoftware.com/juce for more information.
  16. ==============================================================================
  17. */
  18. #undef GetSystemMetrics // multimon overrides this for some reason and causes a mess..
  19. // these are in the windows SDK, but need to be repeated here for GCC..
  20. #ifndef GET_APPCOMMAND_LPARAM
  21. #define FAPPCOMMAND_MASK 0xF000
  22. #define GET_APPCOMMAND_LPARAM(lParam) ((short) (HIWORD (lParam) & ~FAPPCOMMAND_MASK))
  23. #define APPCOMMAND_MEDIA_NEXTTRACK 11
  24. #define APPCOMMAND_MEDIA_PREVIOUSTRACK 12
  25. #define APPCOMMAND_MEDIA_STOP 13
  26. #define APPCOMMAND_MEDIA_PLAY_PAUSE 14
  27. #endif
  28. extern void juce_repeatLastProcessPriority();
  29. extern void juce_CheckCurrentlyFocusedTopLevelWindow(); // in juce_TopLevelWindow.cpp
  30. extern bool juce_IsRunningInWine();
  31. typedef bool (*CheckEventBlockedByModalComps) (const MSG&);
  32. extern CheckEventBlockedByModalComps isEventBlockedByModalComps;
  33. static bool shouldDeactivateTitleBar = true;
  34. //==============================================================================
  35. typedef BOOL (WINAPI* UpdateLayeredWinFunc) (HWND, HDC, POINT*, SIZE*, HDC, POINT*, COLORREF, BLENDFUNCTION*, DWORD);
  36. static UpdateLayeredWinFunc updateLayeredWindow = 0;
  37. bool Desktop::canUseSemiTransparentWindows() noexcept
  38. {
  39. if (updateLayeredWindow == 0)
  40. {
  41. if (! juce_IsRunningInWine())
  42. {
  43. HMODULE user32Mod = GetModuleHandle (_T("user32.dll"));
  44. updateLayeredWindow = (UpdateLayeredWinFunc) GetProcAddress (user32Mod, "UpdateLayeredWindow");
  45. }
  46. }
  47. return updateLayeredWindow != 0;
  48. }
  49. Desktop::DisplayOrientation Desktop::getCurrentOrientation() const
  50. {
  51. return upright;
  52. }
  53. int64 getMouseEventTime()
  54. {
  55. static int64 eventTimeOffset = 0;
  56. static LONG lastMessageTime = 0;
  57. const LONG thisMessageTime = GetMessageTime();
  58. if (thisMessageTime < lastMessageTime || lastMessageTime == 0)
  59. {
  60. lastMessageTime = thisMessageTime;
  61. eventTimeOffset = Time::currentTimeMillis() - thisMessageTime;
  62. }
  63. return eventTimeOffset + thisMessageTime;
  64. }
  65. //==============================================================================
  66. const int extendedKeyModifier = 0x10000;
  67. const int KeyPress::spaceKey = VK_SPACE;
  68. const int KeyPress::returnKey = VK_RETURN;
  69. const int KeyPress::escapeKey = VK_ESCAPE;
  70. const int KeyPress::backspaceKey = VK_BACK;
  71. const int KeyPress::deleteKey = VK_DELETE | extendedKeyModifier;
  72. const int KeyPress::insertKey = VK_INSERT | extendedKeyModifier;
  73. const int KeyPress::tabKey = VK_TAB;
  74. const int KeyPress::leftKey = VK_LEFT | extendedKeyModifier;
  75. const int KeyPress::rightKey = VK_RIGHT | extendedKeyModifier;
  76. const int KeyPress::upKey = VK_UP | extendedKeyModifier;
  77. const int KeyPress::downKey = VK_DOWN | extendedKeyModifier;
  78. const int KeyPress::homeKey = VK_HOME | extendedKeyModifier;
  79. const int KeyPress::endKey = VK_END | extendedKeyModifier;
  80. const int KeyPress::pageUpKey = VK_PRIOR | extendedKeyModifier;
  81. const int KeyPress::pageDownKey = VK_NEXT | extendedKeyModifier;
  82. const int KeyPress::F1Key = VK_F1 | extendedKeyModifier;
  83. const int KeyPress::F2Key = VK_F2 | extendedKeyModifier;
  84. const int KeyPress::F3Key = VK_F3 | extendedKeyModifier;
  85. const int KeyPress::F4Key = VK_F4 | extendedKeyModifier;
  86. const int KeyPress::F5Key = VK_F5 | extendedKeyModifier;
  87. const int KeyPress::F6Key = VK_F6 | extendedKeyModifier;
  88. const int KeyPress::F7Key = VK_F7 | extendedKeyModifier;
  89. const int KeyPress::F8Key = VK_F8 | extendedKeyModifier;
  90. const int KeyPress::F9Key = VK_F9 | extendedKeyModifier;
  91. const int KeyPress::F10Key = VK_F10 | extendedKeyModifier;
  92. const int KeyPress::F11Key = VK_F11 | extendedKeyModifier;
  93. const int KeyPress::F12Key = VK_F12 | extendedKeyModifier;
  94. const int KeyPress::F13Key = VK_F13 | extendedKeyModifier;
  95. const int KeyPress::F14Key = VK_F14 | extendedKeyModifier;
  96. const int KeyPress::F15Key = VK_F15 | extendedKeyModifier;
  97. const int KeyPress::F16Key = VK_F16 | extendedKeyModifier;
  98. const int KeyPress::numberPad0 = VK_NUMPAD0 | extendedKeyModifier;
  99. const int KeyPress::numberPad1 = VK_NUMPAD1 | extendedKeyModifier;
  100. const int KeyPress::numberPad2 = VK_NUMPAD2 | extendedKeyModifier;
  101. const int KeyPress::numberPad3 = VK_NUMPAD3 | extendedKeyModifier;
  102. const int KeyPress::numberPad4 = VK_NUMPAD4 | extendedKeyModifier;
  103. const int KeyPress::numberPad5 = VK_NUMPAD5 | extendedKeyModifier;
  104. const int KeyPress::numberPad6 = VK_NUMPAD6 | extendedKeyModifier;
  105. const int KeyPress::numberPad7 = VK_NUMPAD7 | extendedKeyModifier;
  106. const int KeyPress::numberPad8 = VK_NUMPAD8 | extendedKeyModifier;
  107. const int KeyPress::numberPad9 = VK_NUMPAD9 | extendedKeyModifier;
  108. const int KeyPress::numberPadAdd = VK_ADD | extendedKeyModifier;
  109. const int KeyPress::numberPadSubtract = VK_SUBTRACT | extendedKeyModifier;
  110. const int KeyPress::numberPadMultiply = VK_MULTIPLY | extendedKeyModifier;
  111. const int KeyPress::numberPadDivide = VK_DIVIDE | extendedKeyModifier;
  112. const int KeyPress::numberPadSeparator = VK_SEPARATOR | extendedKeyModifier;
  113. const int KeyPress::numberPadDecimalPoint = VK_DECIMAL | extendedKeyModifier;
  114. const int KeyPress::numberPadEquals = 0x92 /*VK_OEM_NEC_EQUAL*/ | extendedKeyModifier;
  115. const int KeyPress::numberPadDelete = VK_DELETE | extendedKeyModifier;
  116. const int KeyPress::playKey = 0x30000;
  117. const int KeyPress::stopKey = 0x30001;
  118. const int KeyPress::fastForwardKey = 0x30002;
  119. const int KeyPress::rewindKey = 0x30003;
  120. //==============================================================================
  121. class WindowsBitmapImage : public Image::SharedImage
  122. {
  123. public:
  124. WindowsBitmapImage (const Image::PixelFormat format_,
  125. const int w, const int h, const bool clearImage)
  126. : Image::SharedImage (format_, w, h)
  127. {
  128. jassert (format_ == Image::RGB || format_ == Image::ARGB);
  129. pixelStride = (format_ == Image::RGB) ? 3 : 4;
  130. lineStride = -((w * pixelStride + 3) & ~3);
  131. zerostruct (bitmapInfo);
  132. bitmapInfo.bV4Size = sizeof (BITMAPV4HEADER);
  133. bitmapInfo.bV4Width = w;
  134. bitmapInfo.bV4Height = h;
  135. bitmapInfo.bV4Planes = 1;
  136. bitmapInfo.bV4CSType = 1;
  137. bitmapInfo.bV4BitCount = (unsigned short) (pixelStride * 8);
  138. if (format_ == Image::ARGB)
  139. {
  140. bitmapInfo.bV4AlphaMask = 0xff000000;
  141. bitmapInfo.bV4RedMask = 0xff0000;
  142. bitmapInfo.bV4GreenMask = 0xff00;
  143. bitmapInfo.bV4BlueMask = 0xff;
  144. bitmapInfo.bV4V4Compression = BI_BITFIELDS;
  145. }
  146. else
  147. {
  148. bitmapInfo.bV4V4Compression = BI_RGB;
  149. }
  150. HDC dc = GetDC (0);
  151. hdc = CreateCompatibleDC (dc);
  152. ReleaseDC (0, dc);
  153. SetMapMode (hdc, MM_TEXT);
  154. hBitmap = CreateDIBSection (hdc, (BITMAPINFO*) &(bitmapInfo), DIB_RGB_COLORS,
  155. (void**) &bitmapData, 0, 0);
  156. previousBitmap = SelectObject (hdc, hBitmap);
  157. if (format_ == Image::ARGB && clearImage)
  158. zeromem (bitmapData, (size_t) std::abs (h * lineStride));
  159. imageData = bitmapData - (lineStride * (h - 1));
  160. }
  161. ~WindowsBitmapImage()
  162. {
  163. SelectObject (hdc, previousBitmap); // Selecting the previous bitmap before deleting the DC avoids a warning in BoundsChecker
  164. DeleteDC (hdc);
  165. DeleteObject (hBitmap);
  166. }
  167. Image::ImageType getType() const { return Image::NativeImage; }
  168. LowLevelGraphicsContext* createLowLevelContext()
  169. {
  170. return new LowLevelGraphicsSoftwareRenderer (Image (this));
  171. }
  172. void initialiseBitmapData (Image::BitmapData& bitmap, int x, int y, Image::BitmapData::ReadWriteMode /*mode*/)
  173. {
  174. bitmap.data = imageData + x * pixelStride + y * lineStride;
  175. bitmap.pixelFormat = format;
  176. bitmap.lineStride = lineStride;
  177. bitmap.pixelStride = pixelStride;
  178. }
  179. Image::SharedImage* clone()
  180. {
  181. WindowsBitmapImage* im = new WindowsBitmapImage (format, width, height, false);
  182. for (int i = 0; i < height; ++i)
  183. memcpy (im->imageData + i * lineStride, imageData + i * lineStride, (size_t) lineStride);
  184. return im;
  185. }
  186. void blitToWindow (HWND hwnd, HDC dc, const bool transparent,
  187. const int x, const int y,
  188. const RectangleList& maskedRegion,
  189. const uint8 updateLayeredWindowAlpha) noexcept
  190. {
  191. static HDRAWDIB hdd = createDrawDIB();
  192. SetMapMode (dc, MM_TEXT);
  193. if (transparent)
  194. {
  195. if (! maskedRegion.isEmpty())
  196. {
  197. for (RectangleList::Iterator i (maskedRegion); i.next();)
  198. {
  199. const Rectangle<int>& r = *i.getRectangle();
  200. ExcludeClipRect (hdc, r.getX(), r.getY(), r.getRight(), r.getBottom());
  201. }
  202. }
  203. RECT windowBounds;
  204. GetWindowRect (hwnd, &windowBounds);
  205. POINT p = { -x, -y };
  206. POINT pos = { windowBounds.left, windowBounds.top };
  207. SIZE size = { windowBounds.right - windowBounds.left,
  208. windowBounds.bottom - windowBounds.top };
  209. BLENDFUNCTION bf;
  210. bf.AlphaFormat = 1 /*AC_SRC_ALPHA*/;
  211. bf.BlendFlags = 0;
  212. bf.BlendOp = AC_SRC_OVER;
  213. bf.SourceConstantAlpha = updateLayeredWindowAlpha;
  214. updateLayeredWindow (hwnd, 0, &pos, &size, hdc, &p, 0, &bf, 2 /*ULW_ALPHA*/);
  215. }
  216. else
  217. {
  218. int savedDC = 0;
  219. if (! maskedRegion.isEmpty())
  220. {
  221. savedDC = SaveDC (dc);
  222. for (RectangleList::Iterator i (maskedRegion); i.next();)
  223. {
  224. const Rectangle<int>& r = *i.getRectangle();
  225. ExcludeClipRect (dc, r.getX(), r.getY(), r.getRight(), r.getBottom());
  226. }
  227. }
  228. if (hdd == 0)
  229. {
  230. StretchDIBits (dc,
  231. x, y, width, height,
  232. 0, 0, width, height,
  233. bitmapData, (const BITMAPINFO*) &bitmapInfo,
  234. DIB_RGB_COLORS, SRCCOPY);
  235. }
  236. else
  237. {
  238. DrawDibDraw (hdd, dc, x, y, -1, -1,
  239. (BITMAPINFOHEADER*) &bitmapInfo, bitmapData,
  240. 0, 0, width, height, 0);
  241. }
  242. if (! maskedRegion.isEmpty())
  243. RestoreDC (dc, savedDC);
  244. }
  245. }
  246. HBITMAP hBitmap;
  247. HGDIOBJ previousBitmap;
  248. BITMAPV4HEADER bitmapInfo;
  249. HDC hdc;
  250. uint8* bitmapData;
  251. int pixelStride, lineStride;
  252. uint8* imageData;
  253. private:
  254. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (WindowsBitmapImage);
  255. static HDRAWDIB createDrawDIB() noexcept
  256. {
  257. HDC dc = GetDC (0);
  258. const int n = GetDeviceCaps (dc, BITSPIXEL);
  259. ReleaseDC (0, dc);
  260. // only open if we're not palettised
  261. return n > 8 ? DrawDibOpen() : 0;
  262. }
  263. };
  264. //==============================================================================
  265. namespace IconConverters
  266. {
  267. Image createImageFromHBITMAP (HBITMAP bitmap)
  268. {
  269. Image im;
  270. if (bitmap != 0)
  271. {
  272. BITMAP bm;
  273. if (GetObject (bitmap, sizeof (BITMAP), &bm)
  274. && bm.bmWidth > 0 && bm.bmHeight > 0)
  275. {
  276. HDC tempDC = GetDC (0);
  277. HDC dc = CreateCompatibleDC (tempDC);
  278. ReleaseDC (0, tempDC);
  279. SelectObject (dc, bitmap);
  280. im = Image (Image::ARGB, bm.bmWidth, bm.bmHeight, true);
  281. Image::BitmapData imageData (im, Image::BitmapData::writeOnly);
  282. for (int y = bm.bmHeight; --y >= 0;)
  283. {
  284. for (int x = bm.bmWidth; --x >= 0;)
  285. {
  286. COLORREF col = GetPixel (dc, x, y);
  287. imageData.setPixelColour (x, y, Colour ((uint8) GetRValue (col),
  288. (uint8) GetGValue (col),
  289. (uint8) GetBValue (col)));
  290. }
  291. }
  292. DeleteDC (dc);
  293. }
  294. }
  295. return im;
  296. }
  297. Image createImageFromHICON (HICON icon)
  298. {
  299. ICONINFO info;
  300. if (GetIconInfo (icon, &info))
  301. {
  302. Image mask (createImageFromHBITMAP (info.hbmMask));
  303. Image image (createImageFromHBITMAP (info.hbmColor));
  304. if (mask.isValid() && image.isValid())
  305. {
  306. for (int y = image.getHeight(); --y >= 0;)
  307. {
  308. for (int x = image.getWidth(); --x >= 0;)
  309. {
  310. const float brightness = mask.getPixelAt (x, y).getBrightness();
  311. if (brightness > 0.0f)
  312. image.multiplyAlphaAt (x, y, 1.0f - brightness);
  313. }
  314. }
  315. return image;
  316. }
  317. }
  318. return Image::null;
  319. }
  320. HICON createHICONFromImage (const Image& image, const BOOL isIcon, int hotspotX, int hotspotY)
  321. {
  322. WindowsBitmapImage* nativeBitmap = new WindowsBitmapImage (Image::ARGB, image.getWidth(), image.getHeight(), true);
  323. Image bitmap (nativeBitmap);
  324. {
  325. Graphics g (bitmap);
  326. g.drawImageAt (image, 0, 0);
  327. }
  328. HBITMAP mask = CreateBitmap (image.getWidth(), image.getHeight(), 1, 1, 0);
  329. ICONINFO info;
  330. info.fIcon = isIcon;
  331. info.xHotspot = (DWORD) hotspotX;
  332. info.yHotspot = (DWORD) hotspotY;
  333. info.hbmMask = mask;
  334. info.hbmColor = nativeBitmap->hBitmap;
  335. HICON hi = CreateIconIndirect (&info);
  336. DeleteObject (mask);
  337. return hi;
  338. }
  339. }
  340. //==============================================================================
  341. class HWNDComponentPeer : public ComponentPeer
  342. {
  343. public:
  344. enum RenderingEngineType
  345. {
  346. softwareRenderingEngine = 0,
  347. direct2DRenderingEngine
  348. };
  349. //==============================================================================
  350. HWNDComponentPeer (Component* const component,
  351. const int windowStyleFlags,
  352. HWND parentToAddTo_)
  353. : ComponentPeer (component, windowStyleFlags),
  354. dontRepaint (false),
  355. currentRenderingEngine (softwareRenderingEngine),
  356. fullScreen (false),
  357. isDragging (false),
  358. isMouseOver (false),
  359. hasCreatedCaret (false),
  360. constrainerIsResizing (false),
  361. currentWindowIcon (0),
  362. dropTarget (nullptr),
  363. parentToAddTo (parentToAddTo_),
  364. updateLayeredWindowAlpha (255)
  365. {
  366. callFunctionIfNotLocked (&createWindowCallback, this);
  367. setTitle (component->getName());
  368. if ((windowStyleFlags & windowHasDropShadow) != 0
  369. && Desktop::canUseSemiTransparentWindows())
  370. {
  371. shadower = component->getLookAndFeel().createDropShadowerForComponent (component);
  372. if (shadower != nullptr)
  373. shadower->setOwner (component);
  374. }
  375. }
  376. ~HWNDComponentPeer()
  377. {
  378. shadower = nullptr;
  379. // do this before the next bit to avoid messages arriving for this window
  380. // before it's destroyed
  381. JuceWindowIdentifier::setAsJUCEWindow (hwnd, false);
  382. callFunctionIfNotLocked (&destroyWindowCallback, (void*) hwnd);
  383. if (currentWindowIcon != 0)
  384. DestroyIcon (currentWindowIcon);
  385. if (dropTarget != nullptr)
  386. {
  387. dropTarget->Release();
  388. dropTarget = nullptr;
  389. }
  390. #if JUCE_DIRECT2D
  391. direct2DContext = nullptr;
  392. #endif
  393. }
  394. //==============================================================================
  395. void* getNativeHandle() const { return hwnd; }
  396. void setVisible (bool shouldBeVisible)
  397. {
  398. ShowWindow (hwnd, shouldBeVisible ? SW_SHOWNA : SW_HIDE);
  399. if (shouldBeVisible)
  400. InvalidateRect (hwnd, 0, 0);
  401. else
  402. lastPaintTime = 0;
  403. }
  404. void setTitle (const String& title)
  405. {
  406. SetWindowText (hwnd, title.toWideCharPointer());
  407. }
  408. void setPosition (int x, int y)
  409. {
  410. offsetWithinParent (x, y);
  411. SetWindowPos (hwnd, 0,
  412. x - windowBorder.getLeft(),
  413. y - windowBorder.getTop(),
  414. 0, 0,
  415. SWP_NOACTIVATE | SWP_NOSIZE | SWP_NOZORDER | SWP_NOOWNERZORDER);
  416. }
  417. void repaintNowIfTransparent()
  418. {
  419. if (isUsingUpdateLayeredWindow() && lastPaintTime > 0 && Time::getMillisecondCounter() > lastPaintTime + 30)
  420. handlePaintMessage();
  421. }
  422. void updateBorderSize()
  423. {
  424. WINDOWINFO info;
  425. info.cbSize = sizeof (info);
  426. if (GetWindowInfo (hwnd, &info))
  427. {
  428. windowBorder = BorderSize<int> (info.rcClient.top - info.rcWindow.top,
  429. info.rcClient.left - info.rcWindow.left,
  430. info.rcWindow.bottom - info.rcClient.bottom,
  431. info.rcWindow.right - info.rcClient.right);
  432. }
  433. #if JUCE_DIRECT2D
  434. if (direct2DContext != nullptr)
  435. direct2DContext->resized();
  436. #endif
  437. }
  438. void setSize (int w, int h)
  439. {
  440. SetWindowPos (hwnd, 0, 0, 0,
  441. w + windowBorder.getLeftAndRight(),
  442. h + windowBorder.getTopAndBottom(),
  443. SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOZORDER | SWP_NOOWNERZORDER);
  444. updateBorderSize();
  445. repaintNowIfTransparent();
  446. }
  447. void setBounds (int x, int y, int w, int h, bool isNowFullScreen)
  448. {
  449. fullScreen = isNowFullScreen;
  450. offsetWithinParent (x, y);
  451. SetWindowPos (hwnd, 0,
  452. x - windowBorder.getLeft(),
  453. y - windowBorder.getTop(),
  454. w + windowBorder.getLeftAndRight(),
  455. h + windowBorder.getTopAndBottom(),
  456. SWP_NOACTIVATE | SWP_NOZORDER | SWP_NOOWNERZORDER);
  457. updateBorderSize();
  458. repaintNowIfTransparent();
  459. }
  460. const Rectangle<int> getBounds() const
  461. {
  462. RECT r;
  463. GetWindowRect (hwnd, &r);
  464. Rectangle<int> bounds (r.left, r.top, r.right - r.left, r.bottom - r.top);
  465. HWND parentH = GetParent (hwnd);
  466. if (parentH != 0)
  467. {
  468. GetWindowRect (parentH, &r);
  469. bounds.translate (-r.left, -r.top);
  470. }
  471. return windowBorder.subtractedFrom (bounds);
  472. }
  473. const Point<int> getScreenPosition() const
  474. {
  475. RECT r;
  476. GetWindowRect (hwnd, &r);
  477. return Point<int> (r.left + windowBorder.getLeft(),
  478. r.top + windowBorder.getTop());
  479. }
  480. const Point<int> localToGlobal (const Point<int>& relativePosition)
  481. {
  482. return relativePosition + getScreenPosition();
  483. }
  484. const Point<int> globalToLocal (const Point<int>& screenPosition)
  485. {
  486. return screenPosition - getScreenPosition();
  487. }
  488. void setAlpha (float newAlpha)
  489. {
  490. const uint8 intAlpha = (uint8) jlimit (0, 255, (int) (newAlpha * 255.0f));
  491. if (component->isOpaque())
  492. {
  493. if (newAlpha < 1.0f)
  494. {
  495. SetWindowLong (hwnd, GWL_EXSTYLE, GetWindowLong (hwnd, GWL_EXSTYLE) | WS_EX_LAYERED);
  496. SetLayeredWindowAttributes (hwnd, RGB (0, 0, 0), intAlpha, LWA_ALPHA);
  497. }
  498. else
  499. {
  500. SetWindowLong (hwnd, GWL_EXSTYLE, GetWindowLong (hwnd, GWL_EXSTYLE) & ~WS_EX_LAYERED);
  501. RedrawWindow (hwnd, 0, 0, RDW_ERASE | RDW_INVALIDATE | RDW_FRAME | RDW_ALLCHILDREN);
  502. }
  503. }
  504. else
  505. {
  506. updateLayeredWindowAlpha = intAlpha;
  507. component->repaint();
  508. }
  509. }
  510. void setMinimised (bool shouldBeMinimised)
  511. {
  512. if (shouldBeMinimised != isMinimised())
  513. ShowWindow (hwnd, shouldBeMinimised ? SW_MINIMIZE : SW_SHOWNORMAL);
  514. }
  515. bool isMinimised() const
  516. {
  517. WINDOWPLACEMENT wp;
  518. wp.length = sizeof (WINDOWPLACEMENT);
  519. GetWindowPlacement (hwnd, &wp);
  520. return wp.showCmd == SW_SHOWMINIMIZED;
  521. }
  522. void setFullScreen (bool shouldBeFullScreen)
  523. {
  524. setMinimised (false);
  525. if (isFullScreen() != shouldBeFullScreen)
  526. {
  527. fullScreen = shouldBeFullScreen;
  528. const WeakReference<Component> deletionChecker (component);
  529. if (! fullScreen)
  530. {
  531. const Rectangle<int> boundsCopy (lastNonFullscreenBounds);
  532. if (hasTitleBar())
  533. ShowWindow (hwnd, SW_SHOWNORMAL);
  534. if (! boundsCopy.isEmpty())
  535. {
  536. setBounds (boundsCopy.getX(),
  537. boundsCopy.getY(),
  538. boundsCopy.getWidth(),
  539. boundsCopy.getHeight(),
  540. false);
  541. }
  542. }
  543. else
  544. {
  545. if (hasTitleBar())
  546. ShowWindow (hwnd, SW_SHOWMAXIMIZED);
  547. else
  548. SendMessageW (hwnd, WM_SETTINGCHANGE, 0, 0);
  549. }
  550. if (deletionChecker != nullptr)
  551. handleMovedOrResized();
  552. }
  553. }
  554. bool isFullScreen() const
  555. {
  556. if (! hasTitleBar())
  557. return fullScreen;
  558. WINDOWPLACEMENT wp;
  559. wp.length = sizeof (wp);
  560. GetWindowPlacement (hwnd, &wp);
  561. return wp.showCmd == SW_SHOWMAXIMIZED;
  562. }
  563. bool contains (const Point<int>& position, bool trueIfInAChildWindow) const
  564. {
  565. if (! (isPositiveAndBelow (position.getX(), component->getWidth())
  566. && isPositiveAndBelow (position.getY(), component->getHeight())))
  567. return false;
  568. RECT r;
  569. GetWindowRect (hwnd, &r);
  570. POINT p = { position.getX() + r.left + windowBorder.getLeft(),
  571. position.getY() + r.top + windowBorder.getTop() };
  572. HWND w = WindowFromPoint (p);
  573. return w == hwnd || (trueIfInAChildWindow && (IsChild (hwnd, w) != 0));
  574. }
  575. const BorderSize<int> getFrameSize() const
  576. {
  577. return windowBorder;
  578. }
  579. bool setAlwaysOnTop (bool alwaysOnTop)
  580. {
  581. const bool oldDeactivate = shouldDeactivateTitleBar;
  582. shouldDeactivateTitleBar = ((styleFlags & windowIsTemporary) == 0);
  583. SetWindowPos (hwnd, alwaysOnTop ? HWND_TOPMOST : HWND_NOTOPMOST,
  584. 0, 0, 0, 0,
  585. SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE | SWP_NOSENDCHANGING);
  586. shouldDeactivateTitleBar = oldDeactivate;
  587. if (shadower != nullptr)
  588. shadower->componentBroughtToFront (*component);
  589. return true;
  590. }
  591. void toFront (bool makeActive)
  592. {
  593. setMinimised (false);
  594. const bool oldDeactivate = shouldDeactivateTitleBar;
  595. shouldDeactivateTitleBar = ((styleFlags & windowIsTemporary) == 0);
  596. callFunctionIfNotLocked (makeActive ? &toFrontCallback1 : &toFrontCallback2, hwnd);
  597. shouldDeactivateTitleBar = oldDeactivate;
  598. if (! makeActive)
  599. {
  600. // in this case a broughttofront call won't have occured, so do it now..
  601. handleBroughtToFront();
  602. }
  603. }
  604. void toBehind (ComponentPeer* other)
  605. {
  606. HWNDComponentPeer* const otherPeer = dynamic_cast <HWNDComponentPeer*> (other);
  607. jassert (otherPeer != nullptr); // wrong type of window?
  608. if (otherPeer != nullptr)
  609. {
  610. setMinimised (false);
  611. // Must be careful not to try to put a topmost window behind a normal one, or Windows
  612. // promotes the normal one to be topmost!
  613. if (getComponent()->isAlwaysOnTop() == otherPeer->getComponent()->isAlwaysOnTop())
  614. SetWindowPos (hwnd, otherPeer->hwnd, 0, 0, 0, 0,
  615. SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE | SWP_NOSENDCHANGING);
  616. else if (otherPeer->getComponent()->isAlwaysOnTop())
  617. SetWindowPos (hwnd, HWND_TOP, 0, 0, 0, 0,
  618. SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE | SWP_NOSENDCHANGING);
  619. }
  620. }
  621. bool isFocused() const
  622. {
  623. return callFunctionIfNotLocked (&getFocusCallback, 0) == (void*) hwnd;
  624. }
  625. void grabFocus()
  626. {
  627. const bool oldDeactivate = shouldDeactivateTitleBar;
  628. shouldDeactivateTitleBar = ((styleFlags & windowIsTemporary) == 0);
  629. callFunctionIfNotLocked (&setFocusCallback, hwnd);
  630. shouldDeactivateTitleBar = oldDeactivate;
  631. }
  632. void textInputRequired (const Point<int>&)
  633. {
  634. if (! hasCreatedCaret)
  635. {
  636. hasCreatedCaret = true;
  637. CreateCaret (hwnd, (HBITMAP) 1, 0, 0);
  638. }
  639. ShowCaret (hwnd);
  640. SetCaretPos (0, 0);
  641. }
  642. void dismissPendingTextInput()
  643. {
  644. imeHandler.handleSetContext (hwnd, false);
  645. }
  646. void repaint (const Rectangle<int>& area)
  647. {
  648. const RECT r = { area.getX(), area.getY(), area.getRight(), area.getBottom() };
  649. InvalidateRect (hwnd, &r, FALSE);
  650. }
  651. void performAnyPendingRepaintsNow()
  652. {
  653. MSG m;
  654. if (component->isVisible()
  655. && (PeekMessage (&m, hwnd, WM_PAINT, WM_PAINT, PM_REMOVE) || isUsingUpdateLayeredWindow()))
  656. handlePaintMessage();
  657. }
  658. //==============================================================================
  659. static HWNDComponentPeer* getOwnerOfWindow (HWND h) noexcept
  660. {
  661. if (h != 0 && JuceWindowIdentifier::isJUCEWindow (h))
  662. return (HWNDComponentPeer*) (pointer_sized_int) GetWindowLongPtr (h, 8);
  663. return nullptr;
  664. }
  665. //==============================================================================
  666. bool isInside (HWND h) const noexcept
  667. {
  668. return GetAncestor (hwnd, GA_ROOT) == h;
  669. }
  670. //==============================================================================
  671. static bool isKeyDown (const int key) noexcept { return (GetAsyncKeyState (key) & 0x8000) != 0; }
  672. static void updateKeyModifiers() noexcept
  673. {
  674. int keyMods = 0;
  675. if (isKeyDown (VK_SHIFT)) keyMods |= ModifierKeys::shiftModifier;
  676. if (isKeyDown (VK_CONTROL)) keyMods |= ModifierKeys::ctrlModifier;
  677. if (isKeyDown (VK_MENU)) keyMods |= ModifierKeys::altModifier;
  678. if (isKeyDown (VK_RMENU)) keyMods &= ~(ModifierKeys::ctrlModifier | ModifierKeys::altModifier);
  679. currentModifiers = currentModifiers.withOnlyMouseButtons().withFlags (keyMods);
  680. }
  681. static void updateModifiersFromWParam (const WPARAM wParam)
  682. {
  683. int mouseMods = 0;
  684. if (wParam & MK_LBUTTON) mouseMods |= ModifierKeys::leftButtonModifier;
  685. if (wParam & MK_RBUTTON) mouseMods |= ModifierKeys::rightButtonModifier;
  686. if (wParam & MK_MBUTTON) mouseMods |= ModifierKeys::middleButtonModifier;
  687. currentModifiers = currentModifiers.withoutMouseButtons().withFlags (mouseMods);
  688. updateKeyModifiers();
  689. }
  690. //==============================================================================
  691. bool dontRepaint;
  692. static ModifierKeys currentModifiers;
  693. static ModifierKeys modifiersAtLastCallback;
  694. private:
  695. HWND hwnd, parentToAddTo;
  696. ScopedPointer<DropShadower> shadower;
  697. RenderingEngineType currentRenderingEngine;
  698. #if JUCE_DIRECT2D
  699. ScopedPointer<Direct2DLowLevelGraphicsContext> direct2DContext;
  700. #endif
  701. bool fullScreen, isDragging, isMouseOver, hasCreatedCaret, constrainerIsResizing;
  702. BorderSize<int> windowBorder;
  703. HICON currentWindowIcon;
  704. IDropTarget* dropTarget;
  705. uint8 updateLayeredWindowAlpha;
  706. //==============================================================================
  707. class TemporaryImage : public Timer
  708. {
  709. public:
  710. TemporaryImage() {}
  711. const Image& getImage (const bool transparent, const int w, const int h)
  712. {
  713. const Image::PixelFormat format = transparent ? Image::ARGB : Image::RGB;
  714. if ((! image.isValid()) || image.getWidth() < w || image.getHeight() < h || image.getFormat() != format)
  715. image = Image (new WindowsBitmapImage (format, (w + 31) & ~31, (h + 31) & ~31, false));
  716. startTimer (3000);
  717. return image;
  718. }
  719. void timerCallback()
  720. {
  721. stopTimer();
  722. image = Image::null;
  723. }
  724. private:
  725. Image image;
  726. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (TemporaryImage);
  727. };
  728. TemporaryImage offscreenImageGenerator;
  729. //==============================================================================
  730. class WindowClassHolder : public DeletedAtShutdown
  731. {
  732. public:
  733. WindowClassHolder()
  734. {
  735. // this name has to be different for each app/dll instance because otherwise poor old Windows can
  736. // get a bit confused (even despite it not being a process-global window class).
  737. String windowClassName ("JUCE_");
  738. windowClassName << (int) (Time::currentTimeMillis() & 0x7fffffff);
  739. HINSTANCE moduleHandle = (HINSTANCE) Process::getCurrentModuleInstanceHandle();
  740. TCHAR moduleFile [1024] = { 0 };
  741. GetModuleFileName (moduleHandle, moduleFile, 1024);
  742. WORD iconNum = 0;
  743. WNDCLASSEX wcex = { 0 };
  744. wcex.cbSize = sizeof (wcex);
  745. wcex.style = CS_OWNDC;
  746. wcex.lpfnWndProc = (WNDPROC) windowProc;
  747. wcex.lpszClassName = windowClassName.toWideCharPointer();
  748. wcex.cbWndExtra = 32;
  749. wcex.hInstance = moduleHandle;
  750. wcex.hIcon = ExtractAssociatedIcon (moduleHandle, moduleFile, &iconNum);
  751. iconNum = 1;
  752. wcex.hIconSm = ExtractAssociatedIcon (moduleHandle, moduleFile, &iconNum);
  753. atom = RegisterClassEx (&wcex);
  754. jassert (atom != 0);
  755. isEventBlockedByModalComps = checkEventBlockedByModalComps;
  756. }
  757. ~WindowClassHolder()
  758. {
  759. if (ComponentPeer::getNumPeers() == 0)
  760. UnregisterClass (getWindowClassName(), (HINSTANCE) Process::getCurrentModuleInstanceHandle());
  761. clearSingletonInstance();
  762. }
  763. LPCTSTR getWindowClassName() const noexcept { return (LPCTSTR) MAKELONG (atom, 0); }
  764. juce_DeclareSingleton_SingleThreaded_Minimal (WindowClassHolder);
  765. private:
  766. ATOM atom;
  767. static bool isHWNDBlockedByModalComponents (HWND h)
  768. {
  769. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  770. {
  771. Component* const c = Desktop::getInstance().getComponent (i);
  772. if (c != nullptr
  773. && (! c->isCurrentlyBlockedByAnotherModalComponent())
  774. && IsChild ((HWND) c->getWindowHandle(), h))
  775. return false;
  776. }
  777. return true;
  778. }
  779. static bool checkEventBlockedByModalComps (const MSG& m)
  780. {
  781. if (Component::getNumCurrentlyModalComponents() == 0 || JuceWindowIdentifier::isJUCEWindow (m.hwnd))
  782. return false;
  783. switch (m.message)
  784. {
  785. case WM_MOUSEMOVE:
  786. case WM_NCMOUSEMOVE:
  787. case 0x020A: /* WM_MOUSEWHEEL */
  788. case 0x020E: /* WM_MOUSEHWHEEL */
  789. case WM_KEYUP:
  790. case WM_SYSKEYUP:
  791. case WM_CHAR:
  792. case WM_APPCOMMAND:
  793. case WM_LBUTTONUP:
  794. case WM_MBUTTONUP:
  795. case WM_RBUTTONUP:
  796. case WM_MOUSEACTIVATE:
  797. case WM_NCMOUSEHOVER:
  798. case WM_MOUSEHOVER:
  799. return isHWNDBlockedByModalComponents (m.hwnd);
  800. case WM_NCLBUTTONDOWN:
  801. case WM_NCLBUTTONDBLCLK:
  802. case WM_NCRBUTTONDOWN:
  803. case WM_NCRBUTTONDBLCLK:
  804. case WM_NCMBUTTONDOWN:
  805. case WM_NCMBUTTONDBLCLK:
  806. case WM_LBUTTONDOWN:
  807. case WM_LBUTTONDBLCLK:
  808. case WM_MBUTTONDOWN:
  809. case WM_MBUTTONDBLCLK:
  810. case WM_RBUTTONDOWN:
  811. case WM_RBUTTONDBLCLK:
  812. case WM_KEYDOWN:
  813. case WM_SYSKEYDOWN:
  814. if (isHWNDBlockedByModalComponents (m.hwnd))
  815. {
  816. Component* const modal = Component::getCurrentlyModalComponent (0);
  817. if (modal != nullptr)
  818. modal->inputAttemptWhenModal();
  819. return true;
  820. }
  821. break;
  822. default:
  823. break;
  824. }
  825. return false;
  826. }
  827. JUCE_DECLARE_NON_COPYABLE (WindowClassHolder);
  828. };
  829. //==============================================================================
  830. static void* createWindowCallback (void* userData)
  831. {
  832. static_cast <HWNDComponentPeer*> (userData)->createWindow();
  833. return nullptr;
  834. }
  835. void createWindow()
  836. {
  837. DWORD exstyle = WS_EX_ACCEPTFILES;
  838. DWORD type = WS_CLIPSIBLINGS | WS_CLIPCHILDREN;
  839. if (hasTitleBar())
  840. {
  841. type |= WS_OVERLAPPED;
  842. if ((styleFlags & windowHasCloseButton) != 0)
  843. {
  844. type |= WS_SYSMENU;
  845. }
  846. else
  847. {
  848. // annoyingly, windows won't let you have a min/max button without a close button
  849. jassert ((styleFlags & (windowHasMinimiseButton | windowHasMaximiseButton)) == 0);
  850. }
  851. if ((styleFlags & windowIsResizable) != 0)
  852. type |= WS_THICKFRAME;
  853. }
  854. else if (parentToAddTo != 0)
  855. {
  856. type |= WS_CHILD;
  857. }
  858. else
  859. {
  860. type |= WS_POPUP | WS_SYSMENU;
  861. }
  862. if ((styleFlags & windowAppearsOnTaskbar) == 0)
  863. exstyle |= WS_EX_TOOLWINDOW;
  864. else
  865. exstyle |= WS_EX_APPWINDOW;
  866. if ((styleFlags & windowHasMinimiseButton) != 0) type |= WS_MINIMIZEBOX;
  867. if ((styleFlags & windowHasMaximiseButton) != 0) type |= WS_MAXIMIZEBOX;
  868. if ((styleFlags & windowIgnoresMouseClicks) != 0) exstyle |= WS_EX_TRANSPARENT;
  869. if ((styleFlags & windowIsSemiTransparent) != 0 && Desktop::canUseSemiTransparentWindows())
  870. exstyle |= WS_EX_LAYERED;
  871. hwnd = CreateWindowEx (exstyle, WindowClassHolder::getInstance()->getWindowClassName(),
  872. L"", type, 0, 0, 0, 0, parentToAddTo, 0,
  873. (HINSTANCE) Process::getCurrentModuleInstanceHandle(), 0);
  874. #if JUCE_DIRECT2D
  875. setCurrentRenderingEngine (1);
  876. #endif
  877. if (hwnd != 0)
  878. {
  879. SetWindowLongPtr (hwnd, 0, 0);
  880. SetWindowLongPtr (hwnd, 8, (LONG_PTR) this);
  881. JuceWindowIdentifier::setAsJUCEWindow (hwnd, true);
  882. if (dropTarget == nullptr)
  883. dropTarget = new JuceDropTarget (*this);
  884. RegisterDragDrop (hwnd, dropTarget);
  885. updateBorderSize();
  886. // Calling this function here is (for some reason) necessary to make Windows
  887. // correctly enable the menu items that we specify in the wm_initmenu message.
  888. GetSystemMenu (hwnd, false);
  889. const float alpha = component->getAlpha();
  890. if (alpha < 1.0f)
  891. setAlpha (alpha);
  892. }
  893. else
  894. {
  895. jassertfalse;
  896. }
  897. }
  898. static void* destroyWindowCallback (void* handle)
  899. {
  900. RevokeDragDrop ((HWND) handle);
  901. DestroyWindow ((HWND) handle);
  902. return nullptr;
  903. }
  904. static void* toFrontCallback1 (void* h)
  905. {
  906. SetForegroundWindow ((HWND) h);
  907. return nullptr;
  908. }
  909. static void* toFrontCallback2 (void* h)
  910. {
  911. SetWindowPos ((HWND) h, HWND_TOP, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE | SWP_NOSENDCHANGING);
  912. return nullptr;
  913. }
  914. static void* setFocusCallback (void* h)
  915. {
  916. SetFocus ((HWND) h);
  917. return nullptr;
  918. }
  919. static void* getFocusCallback (void*)
  920. {
  921. return GetFocus();
  922. }
  923. void offsetWithinParent (int& x, int& y) const
  924. {
  925. if (isUsingUpdateLayeredWindow())
  926. {
  927. HWND parentHwnd = GetParent (hwnd);
  928. if (parentHwnd != 0)
  929. {
  930. RECT parentRect;
  931. GetWindowRect (parentHwnd, &parentRect);
  932. x += parentRect.left;
  933. y += parentRect.top;
  934. }
  935. }
  936. }
  937. bool isUsingUpdateLayeredWindow() const
  938. {
  939. return ! component->isOpaque();
  940. }
  941. inline bool hasTitleBar() const noexcept { return (styleFlags & windowHasTitleBar) != 0; }
  942. void setIcon (const Image& newIcon)
  943. {
  944. HICON hicon = IconConverters::createHICONFromImage (newIcon, TRUE, 0, 0);
  945. if (hicon != 0)
  946. {
  947. SendMessage (hwnd, WM_SETICON, ICON_BIG, (LPARAM) hicon);
  948. SendMessage (hwnd, WM_SETICON, ICON_SMALL, (LPARAM) hicon);
  949. if (currentWindowIcon != 0)
  950. DestroyIcon (currentWindowIcon);
  951. currentWindowIcon = hicon;
  952. }
  953. }
  954. //==============================================================================
  955. void handlePaintMessage()
  956. {
  957. #if JUCE_DIRECT2D
  958. if (direct2DContext != nullptr)
  959. {
  960. RECT r;
  961. if (GetUpdateRect (hwnd, &r, false))
  962. {
  963. direct2DContext->start();
  964. direct2DContext->clipToRectangle (Rectangle<int> (r.left, r.top, r.right - r.left, r.bottom - r.top));
  965. handlePaint (*direct2DContext);
  966. direct2DContext->end();
  967. }
  968. }
  969. else
  970. #endif
  971. {
  972. HRGN rgn = CreateRectRgn (0, 0, 0, 0);
  973. const int regionType = GetUpdateRgn (hwnd, rgn, false);
  974. PAINTSTRUCT paintStruct;
  975. HDC dc = BeginPaint (hwnd, &paintStruct); // Note this can immediately generate a WM_NCPAINT
  976. // message and become re-entrant, but that's OK
  977. // if something in a paint handler calls, e.g. a message box, this can become reentrant and
  978. // corrupt the image it's using to paint into, so do a check here.
  979. static bool reentrant = false;
  980. if (reentrant)
  981. {
  982. DeleteObject (rgn);
  983. EndPaint (hwnd, &paintStruct);
  984. return;
  985. }
  986. const ScopedValueSetter<bool> setter (reentrant, true, false);
  987. // this is the rectangle to update..
  988. int x = paintStruct.rcPaint.left;
  989. int y = paintStruct.rcPaint.top;
  990. int w = paintStruct.rcPaint.right - x;
  991. int h = paintStruct.rcPaint.bottom - y;
  992. const bool transparent = isUsingUpdateLayeredWindow();
  993. if (transparent)
  994. {
  995. // it's not possible to have a transparent window with a title bar at the moment!
  996. jassert (! hasTitleBar());
  997. RECT r;
  998. GetWindowRect (hwnd, &r);
  999. x = y = 0;
  1000. w = r.right - r.left;
  1001. h = r.bottom - r.top;
  1002. }
  1003. if (w > 0 && h > 0)
  1004. {
  1005. clearMaskedRegion();
  1006. Image offscreenImage (offscreenImageGenerator.getImage (transparent, w, h));
  1007. RectangleList contextClip;
  1008. const Rectangle<int> clipBounds (0, 0, w, h);
  1009. bool needToPaintAll = true;
  1010. if (regionType == COMPLEXREGION && ! transparent)
  1011. {
  1012. HRGN clipRgn = CreateRectRgnIndirect (&paintStruct.rcPaint);
  1013. CombineRgn (rgn, rgn, clipRgn, RGN_AND);
  1014. DeleteObject (clipRgn);
  1015. char rgnData [8192];
  1016. const DWORD res = GetRegionData (rgn, sizeof (rgnData), (RGNDATA*) rgnData);
  1017. if (res > 0 && res <= sizeof (rgnData))
  1018. {
  1019. const RGNDATAHEADER* const hdr = &(((const RGNDATA*) rgnData)->rdh);
  1020. if (hdr->iType == RDH_RECTANGLES
  1021. && hdr->rcBound.right - hdr->rcBound.left >= w
  1022. && hdr->rcBound.bottom - hdr->rcBound.top >= h)
  1023. {
  1024. needToPaintAll = false;
  1025. const RECT* rects = (const RECT*) (rgnData + sizeof (RGNDATAHEADER));
  1026. for (int i = (int) ((RGNDATA*) rgnData)->rdh.nCount; --i >= 0;)
  1027. {
  1028. if (rects->right <= x + w && rects->bottom <= y + h)
  1029. {
  1030. const int cx = jmax (x, (int) rects->left);
  1031. contextClip.addWithoutMerging (Rectangle<int> (cx - x, rects->top - y, rects->right - cx, rects->bottom - rects->top)
  1032. .getIntersection (clipBounds));
  1033. }
  1034. else
  1035. {
  1036. needToPaintAll = true;
  1037. break;
  1038. }
  1039. ++rects;
  1040. }
  1041. }
  1042. }
  1043. }
  1044. if (needToPaintAll)
  1045. {
  1046. contextClip.clear();
  1047. contextClip.addWithoutMerging (Rectangle<int> (w, h));
  1048. }
  1049. if (transparent)
  1050. {
  1051. RectangleList::Iterator i (contextClip);
  1052. while (i.next())
  1053. offscreenImage.clear (*i.getRectangle());
  1054. }
  1055. // if the component's not opaque, this won't draw properly unless the platform can support this
  1056. jassert (Desktop::canUseSemiTransparentWindows() || component->isOpaque());
  1057. updateCurrentModifiers();
  1058. LowLevelGraphicsSoftwareRenderer context (offscreenImage, -x, -y, contextClip);
  1059. handlePaint (context);
  1060. if (! dontRepaint)
  1061. static_cast <WindowsBitmapImage*> (offscreenImage.getSharedImage())
  1062. ->blitToWindow (hwnd, dc, transparent, x, y, maskedRegion, updateLayeredWindowAlpha);
  1063. }
  1064. DeleteObject (rgn);
  1065. EndPaint (hwnd, &paintStruct);
  1066. }
  1067. #ifndef JUCE_GCC
  1068. _fpreset(); // because some graphics cards can unmask FP exceptions
  1069. #endif
  1070. lastPaintTime = Time::getMillisecondCounter();
  1071. }
  1072. //==============================================================================
  1073. void doMouseEvent (const Point<int>& position)
  1074. {
  1075. handleMouseEvent (0, position, currentModifiers, getMouseEventTime());
  1076. }
  1077. StringArray getAvailableRenderingEngines()
  1078. {
  1079. StringArray s (ComponentPeer::getAvailableRenderingEngines());
  1080. #if JUCE_DIRECT2D
  1081. if (SystemStats::getOperatingSystemType() >= SystemStats::Windows7)
  1082. s.add ("Direct2D");
  1083. #endif
  1084. return s;
  1085. }
  1086. int getCurrentRenderingEngine() const { return currentRenderingEngine; }
  1087. #if JUCE_DIRECT2D
  1088. void updateDirect2DContext()
  1089. {
  1090. if (currentRenderingEngine != direct2DRenderingEngine)
  1091. direct2DContext = 0;
  1092. else if (direct2DContext == 0)
  1093. direct2DContext = new Direct2DLowLevelGraphicsContext (hwnd);
  1094. }
  1095. #endif
  1096. void setCurrentRenderingEngine (int index)
  1097. {
  1098. (void) index;
  1099. #if JUCE_DIRECT2D
  1100. if (getAvailableRenderingEngines().size() > 1)
  1101. {
  1102. currentRenderingEngine = index == 1 ? direct2DRenderingEngine : softwareRenderingEngine;
  1103. updateDirect2DContext();
  1104. repaint (component->getLocalBounds());
  1105. }
  1106. #endif
  1107. }
  1108. static int getMinTimeBetweenMouseMoves()
  1109. {
  1110. if (SystemStats::getOperatingSystemType() >= SystemStats::WinVista)
  1111. return 0;
  1112. return 1000 / 60; // Throttling the incoming mouse-events seems to still be needed in XP..
  1113. }
  1114. void doMouseMove (const Point<int>& position)
  1115. {
  1116. if (! isMouseOver)
  1117. {
  1118. isMouseOver = true;
  1119. ModifierKeys::getCurrentModifiersRealtime(); // (This avoids a rare stuck-button problem when focus is lost unexpectedly)
  1120. updateKeyModifiers();
  1121. TRACKMOUSEEVENT tme;
  1122. tme.cbSize = sizeof (tme);
  1123. tme.dwFlags = TME_LEAVE;
  1124. tme.hwndTrack = hwnd;
  1125. tme.dwHoverTime = 0;
  1126. if (! TrackMouseEvent (&tme))
  1127. jassertfalse;
  1128. Desktop::getInstance().getMainMouseSource().forceMouseCursorUpdate();
  1129. }
  1130. else if (! isDragging)
  1131. {
  1132. if (! contains (position, false))
  1133. return;
  1134. }
  1135. static uint32 lastMouseTime = 0;
  1136. static int minTimeBetweenMouses = getMinTimeBetweenMouseMoves();
  1137. const uint32 now = Time::getMillisecondCounter();
  1138. if (now >= lastMouseTime + minTimeBetweenMouses)
  1139. {
  1140. lastMouseTime = now;
  1141. doMouseEvent (position);
  1142. }
  1143. }
  1144. void doMouseDown (const Point<int>& position, const WPARAM wParam)
  1145. {
  1146. if (GetCapture() != hwnd)
  1147. SetCapture (hwnd);
  1148. doMouseMove (position);
  1149. updateModifiersFromWParam (wParam);
  1150. isDragging = true;
  1151. doMouseEvent (position);
  1152. }
  1153. void doMouseUp (const Point<int>& position, const WPARAM wParam)
  1154. {
  1155. updateModifiersFromWParam (wParam);
  1156. isDragging = false;
  1157. // release the mouse capture if the user has released all buttons
  1158. if ((wParam & (MK_LBUTTON | MK_RBUTTON | MK_MBUTTON)) == 0 && hwnd == GetCapture())
  1159. ReleaseCapture();
  1160. doMouseEvent (position);
  1161. }
  1162. void doCaptureChanged()
  1163. {
  1164. if (constrainerIsResizing)
  1165. {
  1166. if (constrainer != nullptr)
  1167. constrainer->resizeEnd();
  1168. constrainerIsResizing = false;
  1169. }
  1170. if (isDragging)
  1171. doMouseUp (getCurrentMousePos(), (WPARAM) 0);
  1172. }
  1173. void doMouseExit()
  1174. {
  1175. isMouseOver = false;
  1176. doMouseEvent (getCurrentMousePos());
  1177. }
  1178. void doMouseWheel (const Point<int>& globalPos, const WPARAM wParam, const bool isVertical)
  1179. {
  1180. updateKeyModifiers();
  1181. const float amount = jlimit (-1000.0f, 1000.0f, 0.75f * (short) HIWORD (wParam));
  1182. // Because Windows stupidly sends all wheel events to the window with the keyboard
  1183. // focus, we have to redirect them here according to the mouse pos..
  1184. POINT p = { globalPos.getX(), globalPos.getY() };
  1185. HWNDComponentPeer* peer = getOwnerOfWindow (WindowFromPoint (p));
  1186. if (peer == nullptr)
  1187. peer = this;
  1188. peer->handleMouseWheel (0, peer->globalToLocal (globalPos), getMouseEventTime(),
  1189. isVertical ? 0.0f : -amount,
  1190. isVertical ? amount : 0.0f);
  1191. }
  1192. //==============================================================================
  1193. void sendModifierKeyChangeIfNeeded()
  1194. {
  1195. if (modifiersAtLastCallback != currentModifiers)
  1196. {
  1197. modifiersAtLastCallback = currentModifiers;
  1198. handleModifierKeysChange();
  1199. }
  1200. }
  1201. bool doKeyUp (const WPARAM key)
  1202. {
  1203. updateKeyModifiers();
  1204. switch (key)
  1205. {
  1206. case VK_SHIFT:
  1207. case VK_CONTROL:
  1208. case VK_MENU:
  1209. case VK_CAPITAL:
  1210. case VK_LWIN:
  1211. case VK_RWIN:
  1212. case VK_APPS:
  1213. case VK_NUMLOCK:
  1214. case VK_SCROLL:
  1215. case VK_LSHIFT:
  1216. case VK_RSHIFT:
  1217. case VK_LCONTROL:
  1218. case VK_LMENU:
  1219. case VK_RCONTROL:
  1220. case VK_RMENU:
  1221. sendModifierKeyChangeIfNeeded();
  1222. }
  1223. return handleKeyUpOrDown (false)
  1224. || Component::getCurrentlyModalComponent() != nullptr;
  1225. }
  1226. bool doKeyDown (const WPARAM key)
  1227. {
  1228. updateKeyModifiers();
  1229. bool used = false;
  1230. switch (key)
  1231. {
  1232. case VK_SHIFT:
  1233. case VK_LSHIFT:
  1234. case VK_RSHIFT:
  1235. case VK_CONTROL:
  1236. case VK_LCONTROL:
  1237. case VK_RCONTROL:
  1238. case VK_MENU:
  1239. case VK_LMENU:
  1240. case VK_RMENU:
  1241. case VK_LWIN:
  1242. case VK_RWIN:
  1243. case VK_CAPITAL:
  1244. case VK_NUMLOCK:
  1245. case VK_SCROLL:
  1246. case VK_APPS:
  1247. sendModifierKeyChangeIfNeeded();
  1248. break;
  1249. case VK_LEFT:
  1250. case VK_RIGHT:
  1251. case VK_UP:
  1252. case VK_DOWN:
  1253. case VK_PRIOR:
  1254. case VK_NEXT:
  1255. case VK_HOME:
  1256. case VK_END:
  1257. case VK_DELETE:
  1258. case VK_INSERT:
  1259. case VK_F1:
  1260. case VK_F2:
  1261. case VK_F3:
  1262. case VK_F4:
  1263. case VK_F5:
  1264. case VK_F6:
  1265. case VK_F7:
  1266. case VK_F8:
  1267. case VK_F9:
  1268. case VK_F10:
  1269. case VK_F11:
  1270. case VK_F12:
  1271. case VK_F13:
  1272. case VK_F14:
  1273. case VK_F15:
  1274. case VK_F16:
  1275. used = handleKeyUpOrDown (true);
  1276. used = handleKeyPress (extendedKeyModifier | (int) key, 0) || used;
  1277. break;
  1278. case VK_ADD:
  1279. case VK_SUBTRACT:
  1280. case VK_MULTIPLY:
  1281. case VK_DIVIDE:
  1282. case VK_SEPARATOR:
  1283. case VK_DECIMAL:
  1284. used = handleKeyUpOrDown (true);
  1285. break;
  1286. default:
  1287. used = handleKeyUpOrDown (true);
  1288. {
  1289. MSG msg;
  1290. if (! PeekMessage (&msg, hwnd, WM_CHAR, WM_DEADCHAR, PM_NOREMOVE))
  1291. {
  1292. // if there isn't a WM_CHAR or WM_DEADCHAR message pending, we need to
  1293. // manually generate the key-press event that matches this key-down.
  1294. const UINT keyChar = MapVirtualKey ((UINT) key, 2);
  1295. used = handleKeyPress ((int) LOWORD (keyChar), 0) || used;
  1296. }
  1297. }
  1298. break;
  1299. }
  1300. return used || (Component::getCurrentlyModalComponent() != nullptr);
  1301. }
  1302. bool doKeyChar (int key, const LPARAM flags)
  1303. {
  1304. updateKeyModifiers();
  1305. juce_wchar textChar = (juce_wchar) key;
  1306. const int virtualScanCode = (flags >> 16) & 0xff;
  1307. if (key >= '0' && key <= '9')
  1308. {
  1309. switch (virtualScanCode) // check for a numeric keypad scan-code
  1310. {
  1311. case 0x52:
  1312. case 0x4f:
  1313. case 0x50:
  1314. case 0x51:
  1315. case 0x4b:
  1316. case 0x4c:
  1317. case 0x4d:
  1318. case 0x47:
  1319. case 0x48:
  1320. case 0x49:
  1321. key = (key - '0') + KeyPress::numberPad0;
  1322. break;
  1323. default:
  1324. break;
  1325. }
  1326. }
  1327. else
  1328. {
  1329. // convert the scan code to an unmodified character code..
  1330. const UINT virtualKey = MapVirtualKey ((UINT) virtualScanCode, 1);
  1331. UINT keyChar = MapVirtualKey (virtualKey, 2);
  1332. keyChar = LOWORD (keyChar);
  1333. if (keyChar != 0)
  1334. key = (int) keyChar;
  1335. // avoid sending junk text characters for some control-key combinations
  1336. if (textChar < ' ' && currentModifiers.testFlags (ModifierKeys::ctrlModifier | ModifierKeys::altModifier))
  1337. textChar = 0;
  1338. }
  1339. return handleKeyPress (key, textChar);
  1340. }
  1341. void forwardMessageToParent (UINT message, WPARAM wParam, LPARAM lParam) const
  1342. {
  1343. HWND parentH = GetParent (hwnd);
  1344. if (parentH != 0)
  1345. PostMessage (parentH, message, wParam, lParam);
  1346. }
  1347. bool doAppCommand (const LPARAM lParam)
  1348. {
  1349. int key = 0;
  1350. switch (GET_APPCOMMAND_LPARAM (lParam))
  1351. {
  1352. case APPCOMMAND_MEDIA_PLAY_PAUSE: key = KeyPress::playKey; break;
  1353. case APPCOMMAND_MEDIA_STOP: key = KeyPress::stopKey; break;
  1354. case APPCOMMAND_MEDIA_NEXTTRACK: key = KeyPress::fastForwardKey; break;
  1355. case APPCOMMAND_MEDIA_PREVIOUSTRACK: key = KeyPress::rewindKey; break;
  1356. default: break;
  1357. }
  1358. if (key != 0)
  1359. {
  1360. updateKeyModifiers();
  1361. if (hwnd == GetActiveWindow())
  1362. {
  1363. handleKeyPress (key, 0);
  1364. return true;
  1365. }
  1366. }
  1367. return false;
  1368. }
  1369. bool isConstrainedNativeWindow() const
  1370. {
  1371. return constrainer != nullptr
  1372. && (styleFlags & (windowHasTitleBar | windowIsResizable)) == (windowHasTitleBar | windowIsResizable);
  1373. }
  1374. LRESULT handleSizeConstraining (RECT* const r, const WPARAM wParam)
  1375. {
  1376. if (isConstrainedNativeWindow())
  1377. {
  1378. Rectangle<int> pos (r->left, r->top, r->right - r->left, r->bottom - r->top);
  1379. constrainer->checkBounds (pos, windowBorder.addedTo (component->getBounds()),
  1380. Desktop::getInstance().getAllMonitorDisplayAreas().getBounds(),
  1381. wParam == WMSZ_TOP || wParam == WMSZ_TOPLEFT || wParam == WMSZ_TOPRIGHT,
  1382. wParam == WMSZ_LEFT || wParam == WMSZ_TOPLEFT || wParam == WMSZ_BOTTOMLEFT,
  1383. wParam == WMSZ_BOTTOM || wParam == WMSZ_BOTTOMLEFT || wParam == WMSZ_BOTTOMRIGHT,
  1384. wParam == WMSZ_RIGHT || wParam == WMSZ_TOPRIGHT || wParam == WMSZ_BOTTOMRIGHT);
  1385. r->left = pos.getX();
  1386. r->top = pos.getY();
  1387. r->right = pos.getRight();
  1388. r->bottom = pos.getBottom();
  1389. }
  1390. return TRUE;
  1391. }
  1392. LRESULT handlePositionChanging (WINDOWPOS* const wp)
  1393. {
  1394. if (isConstrainedNativeWindow())
  1395. {
  1396. if ((wp->flags & (SWP_NOMOVE | SWP_NOSIZE)) != (SWP_NOMOVE | SWP_NOSIZE)
  1397. && ! Component::isMouseButtonDownAnywhere())
  1398. {
  1399. Rectangle<int> pos (wp->x, wp->y, wp->cx, wp->cy);
  1400. const Rectangle<int> current (windowBorder.addedTo (component->getBounds()));
  1401. constrainer->checkBounds (pos, current,
  1402. Desktop::getInstance().getAllMonitorDisplayAreas().getBounds(),
  1403. pos.getY() != current.getY() && pos.getBottom() == current.getBottom(),
  1404. pos.getX() != current.getX() && pos.getRight() == current.getRight(),
  1405. pos.getY() == current.getY() && pos.getBottom() != current.getBottom(),
  1406. pos.getX() == current.getX() && pos.getRight() != current.getRight());
  1407. wp->x = pos.getX();
  1408. wp->y = pos.getY();
  1409. wp->cx = pos.getWidth();
  1410. wp->cy = pos.getHeight();
  1411. }
  1412. }
  1413. return 0;
  1414. }
  1415. void handleAppActivation (const WPARAM wParam)
  1416. {
  1417. modifiersAtLastCallback = -1;
  1418. updateKeyModifiers();
  1419. if (isMinimised())
  1420. {
  1421. component->repaint();
  1422. handleMovedOrResized();
  1423. if (! ComponentPeer::isValidPeer (this))
  1424. return;
  1425. }
  1426. Component* underMouse = component->getComponentAt (component->getMouseXYRelative());
  1427. if (underMouse == nullptr)
  1428. underMouse = component;
  1429. if (underMouse->isCurrentlyBlockedByAnotherModalComponent())
  1430. {
  1431. if (LOWORD (wParam) == WA_CLICKACTIVE)
  1432. Component::getCurrentlyModalComponent()->inputAttemptWhenModal();
  1433. else
  1434. ModalComponentManager::getInstance()->bringModalComponentsToFront();
  1435. }
  1436. else
  1437. {
  1438. handleBroughtToFront();
  1439. }
  1440. }
  1441. void handleLeftClickInNCArea (WPARAM wParam)
  1442. {
  1443. if (! sendInputAttemptWhenModalMessage())
  1444. {
  1445. switch (wParam)
  1446. {
  1447. case HTBOTTOM:
  1448. case HTBOTTOMLEFT:
  1449. case HTBOTTOMRIGHT:
  1450. case HTGROWBOX:
  1451. case HTLEFT:
  1452. case HTRIGHT:
  1453. case HTTOP:
  1454. case HTTOPLEFT:
  1455. case HTTOPRIGHT:
  1456. if (isConstrainedNativeWindow())
  1457. {
  1458. constrainerIsResizing = true;
  1459. constrainer->resizeStart();
  1460. }
  1461. break;
  1462. default:
  1463. break;
  1464. }
  1465. }
  1466. }
  1467. void initialiseSysMenu (HMENU menu) const
  1468. {
  1469. if (! hasTitleBar())
  1470. {
  1471. if (isFullScreen())
  1472. {
  1473. EnableMenuItem (menu, SC_RESTORE, MF_BYCOMMAND | MF_ENABLED);
  1474. EnableMenuItem (menu, SC_MOVE, MF_BYCOMMAND | MF_GRAYED);
  1475. }
  1476. else if (! isMinimised())
  1477. {
  1478. EnableMenuItem (menu, SC_MAXIMIZE, MF_BYCOMMAND | MF_GRAYED);
  1479. }
  1480. }
  1481. }
  1482. //==============================================================================
  1483. class JuceDropTarget : public ComBaseClassHelper <IDropTarget>
  1484. {
  1485. public:
  1486. JuceDropTarget (HWNDComponentPeer& owner_)
  1487. : owner (owner_)
  1488. {
  1489. }
  1490. JUCE_COMRESULT DragEnter (IDataObject* pDataObject, DWORD grfKeyState, POINTL mousePos, DWORD* pdwEffect)
  1491. {
  1492. updateFileList (pDataObject);
  1493. return DragOver (grfKeyState, mousePos, pdwEffect);
  1494. }
  1495. JUCE_COMRESULT DragLeave()
  1496. {
  1497. owner.handleFileDragExit (files);
  1498. return S_OK;
  1499. }
  1500. JUCE_COMRESULT DragOver (DWORD /*grfKeyState*/, POINTL mousePos, DWORD* pdwEffect)
  1501. {
  1502. const bool wasWanted = owner.handleFileDragMove (files, getMousePos (mousePos));
  1503. *pdwEffect = wasWanted ? (DWORD) DROPEFFECT_COPY : (DWORD) DROPEFFECT_NONE;
  1504. return S_OK;
  1505. }
  1506. JUCE_COMRESULT Drop (IDataObject* pDataObject, DWORD /*grfKeyState*/, POINTL mousePos, DWORD* pdwEffect)
  1507. {
  1508. updateFileList (pDataObject);
  1509. const bool wasWanted = owner.handleFileDragDrop (files, getMousePos (mousePos));
  1510. *pdwEffect = wasWanted ? (DWORD) DROPEFFECT_COPY : (DWORD) DROPEFFECT_NONE;
  1511. return S_OK;
  1512. }
  1513. private:
  1514. HWNDComponentPeer& owner;
  1515. StringArray files;
  1516. Point<int> getMousePos (const POINTL& mousePos) const
  1517. {
  1518. return owner.globalToLocal (Point<int> (mousePos.x, mousePos.y));
  1519. }
  1520. template <typename CharType>
  1521. void parseFileList (const CharType* names, const SIZE_T totalLen)
  1522. {
  1523. unsigned int i = 0;
  1524. for (;;)
  1525. {
  1526. unsigned int len = 0;
  1527. while (i + len < totalLen && names [i + len] != 0)
  1528. ++len;
  1529. if (len == 0)
  1530. break;
  1531. files.add (String (names + i, len));
  1532. i += len + 1;
  1533. }
  1534. }
  1535. void updateFileList (IDataObject* const pDataObject)
  1536. {
  1537. files.clear();
  1538. FORMATETC format = { CF_HDROP, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
  1539. STGMEDIUM medium = { TYMED_HGLOBAL, { 0 }, 0 };
  1540. if (pDataObject->GetData (&format, &medium) == S_OK)
  1541. {
  1542. const SIZE_T totalLen = GlobalSize (medium.hGlobal);
  1543. const LPDROPFILES dropFiles = (const LPDROPFILES) GlobalLock (medium.hGlobal);
  1544. const void* const names = addBytesToPointer (dropFiles, sizeof (DROPFILES));
  1545. if (dropFiles->fWide)
  1546. parseFileList (static_cast <const WCHAR*> (names), totalLen);
  1547. else
  1548. parseFileList (static_cast <const char*> (names), totalLen);
  1549. GlobalUnlock (medium.hGlobal);
  1550. }
  1551. }
  1552. JUCE_DECLARE_NON_COPYABLE (JuceDropTarget);
  1553. };
  1554. void doSettingChange()
  1555. {
  1556. Desktop::getInstance().refreshMonitorSizes();
  1557. if (fullScreen && ! isMinimised())
  1558. {
  1559. const Rectangle<int> r (component->getParentMonitorArea());
  1560. SetWindowPos (hwnd, 0, r.getX(), r.getY(), r.getWidth(), r.getHeight(),
  1561. SWP_NOACTIVATE | SWP_NOOWNERZORDER | SWP_NOZORDER | SWP_NOSENDCHANGING);
  1562. }
  1563. }
  1564. //==============================================================================
  1565. public:
  1566. static LRESULT CALLBACK windowProc (HWND h, UINT message, WPARAM wParam, LPARAM lParam)
  1567. {
  1568. HWNDComponentPeer* const peer = getOwnerOfWindow (h);
  1569. if (peer != nullptr)
  1570. {
  1571. jassert (isValidPeer (peer));
  1572. return peer->peerWindowProc (h, message, wParam, lParam);
  1573. }
  1574. return DefWindowProcW (h, message, wParam, lParam);
  1575. }
  1576. private:
  1577. static void* callFunctionIfNotLocked (MessageCallbackFunction* callback, void* userData)
  1578. {
  1579. if (MessageManager::getInstance()->currentThreadHasLockedMessageManager())
  1580. return callback (userData);
  1581. else
  1582. return MessageManager::getInstance()->callFunctionOnMessageThread (callback, userData);
  1583. }
  1584. static Point<int> getPointFromLParam (LPARAM lParam) noexcept
  1585. {
  1586. return Point<int> (GET_X_LPARAM (lParam), GET_Y_LPARAM (lParam));
  1587. }
  1588. static Point<int> getCurrentMousePosGlobal() noexcept
  1589. {
  1590. const DWORD mp = GetMessagePos();
  1591. return Point<int> (GET_X_LPARAM (mp),
  1592. GET_Y_LPARAM (mp));
  1593. }
  1594. Point<int> getCurrentMousePos() noexcept
  1595. {
  1596. return globalToLocal (getCurrentMousePosGlobal());
  1597. }
  1598. LRESULT peerWindowProc (HWND h, UINT message, WPARAM wParam, LPARAM lParam)
  1599. {
  1600. switch (message)
  1601. {
  1602. //==============================================================================
  1603. case WM_NCHITTEST:
  1604. if ((styleFlags & windowIgnoresMouseClicks) != 0)
  1605. return HTTRANSPARENT;
  1606. else if (! hasTitleBar())
  1607. return HTCLIENT;
  1608. break;
  1609. //==============================================================================
  1610. case WM_PAINT:
  1611. handlePaintMessage();
  1612. return 0;
  1613. case WM_NCPAINT:
  1614. if (wParam != 1) // (1 = a repaint of the entire NC region)
  1615. handlePaintMessage(); // this must be done, even with native titlebars, or there are rendering artifacts.
  1616. if (hasTitleBar())
  1617. break; // let the DefWindowProc handle drawing the frame.
  1618. return 0;
  1619. case WM_ERASEBKGND:
  1620. case WM_NCCALCSIZE:
  1621. if (hasTitleBar())
  1622. break;
  1623. return 1;
  1624. //==============================================================================
  1625. case WM_MOUSEMOVE: doMouseMove (getPointFromLParam (lParam)); return 0;
  1626. case WM_MOUSELEAVE: doMouseExit(); return 0;
  1627. case WM_LBUTTONDOWN:
  1628. case WM_MBUTTONDOWN:
  1629. case WM_RBUTTONDOWN: doMouseDown (getPointFromLParam (lParam), wParam); return 0;
  1630. case WM_LBUTTONUP:
  1631. case WM_MBUTTONUP:
  1632. case WM_RBUTTONUP: doMouseUp (getPointFromLParam (lParam), wParam); return 0;
  1633. case WM_CAPTURECHANGED: doCaptureChanged(); return 0;
  1634. case WM_NCMOUSEMOVE:
  1635. if (hasTitleBar())
  1636. break;
  1637. return 0;
  1638. case 0x020A: /* WM_MOUSEWHEEL */
  1639. case 0x020E: /* WM_MOUSEHWHEEL */
  1640. doMouseWheel (getCurrentMousePosGlobal(), wParam, message == 0x020A);
  1641. return 0;
  1642. //==============================================================================
  1643. case WM_SIZING: return handleSizeConstraining ((RECT*) lParam, wParam);
  1644. case WM_WINDOWPOSCHANGING: return handlePositionChanging ((WINDOWPOS*) lParam);
  1645. case WM_WINDOWPOSCHANGED:
  1646. {
  1647. const Point<int> pos (getCurrentMousePos());
  1648. if (contains (pos, false))
  1649. doMouseEvent (pos);
  1650. }
  1651. handleMovedOrResized();
  1652. if (dontRepaint)
  1653. break; // needed for non-accelerated openGL windows to draw themselves correctly..
  1654. return 0;
  1655. //==============================================================================
  1656. case WM_KEYDOWN:
  1657. case WM_SYSKEYDOWN:
  1658. if (doKeyDown (wParam))
  1659. return 0;
  1660. forwardMessageToParent (message, wParam, lParam);
  1661. break;
  1662. case WM_KEYUP:
  1663. case WM_SYSKEYUP:
  1664. if (doKeyUp (wParam))
  1665. return 0;
  1666. forwardMessageToParent (message, wParam, lParam);
  1667. break;
  1668. case WM_CHAR:
  1669. if (doKeyChar ((int) wParam, lParam))
  1670. return 0;
  1671. forwardMessageToParent (message, wParam, lParam);
  1672. break;
  1673. case WM_APPCOMMAND:
  1674. if (doAppCommand (lParam))
  1675. return TRUE;
  1676. break;
  1677. //==============================================================================
  1678. case WM_SETFOCUS:
  1679. updateKeyModifiers();
  1680. handleFocusGain();
  1681. break;
  1682. case WM_KILLFOCUS:
  1683. if (hasCreatedCaret)
  1684. {
  1685. hasCreatedCaret = false;
  1686. DestroyCaret();
  1687. }
  1688. handleFocusLoss();
  1689. break;
  1690. case WM_ACTIVATEAPP:
  1691. // Windows does weird things to process priority when you swap apps,
  1692. // so this forces an update when the app is brought to the front
  1693. if (wParam != FALSE)
  1694. juce_repeatLastProcessPriority();
  1695. else
  1696. Desktop::getInstance().setKioskModeComponent (nullptr); // turn kiosk mode off if we lose focus
  1697. juce_CheckCurrentlyFocusedTopLevelWindow();
  1698. modifiersAtLastCallback = -1;
  1699. return 0;
  1700. case WM_ACTIVATE:
  1701. if (LOWORD (wParam) == WA_ACTIVE || LOWORD (wParam) == WA_CLICKACTIVE)
  1702. {
  1703. handleAppActivation (wParam);
  1704. return 0;
  1705. }
  1706. break;
  1707. case WM_NCACTIVATE:
  1708. // while a temporary window is being shown, prevent Windows from deactivating the
  1709. // title bars of our main windows.
  1710. if (wParam == 0 && ! shouldDeactivateTitleBar)
  1711. wParam = TRUE; // change this and let it get passed to the DefWindowProc.
  1712. break;
  1713. case WM_MOUSEACTIVATE:
  1714. if (! component->getMouseClickGrabsKeyboardFocus())
  1715. return MA_NOACTIVATE;
  1716. break;
  1717. case WM_SHOWWINDOW:
  1718. if (wParam != 0)
  1719. handleBroughtToFront();
  1720. break;
  1721. case WM_CLOSE:
  1722. if (! component->isCurrentlyBlockedByAnotherModalComponent())
  1723. handleUserClosingWindow();
  1724. return 0;
  1725. case WM_QUERYENDSESSION:
  1726. if (JUCEApplication::getInstance() != nullptr)
  1727. {
  1728. JUCEApplication::getInstance()->systemRequestedQuit();
  1729. return MessageManager::getInstance()->hasStopMessageBeenSent();
  1730. }
  1731. return TRUE;
  1732. case WM_SYNCPAINT:
  1733. return 0;
  1734. case WM_DISPLAYCHANGE:
  1735. InvalidateRect (h, 0, 0);
  1736. // intentional fall-through...
  1737. case WM_SETTINGCHANGE: // note the fall-through in the previous case!
  1738. doSettingChange();
  1739. break;
  1740. case WM_INITMENU:
  1741. initialiseSysMenu ((HMENU) wParam);
  1742. break;
  1743. case WM_SYSCOMMAND:
  1744. switch (wParam & 0xfff0)
  1745. {
  1746. case SC_CLOSE:
  1747. if (sendInputAttemptWhenModalMessage())
  1748. return 0;
  1749. if (hasTitleBar())
  1750. {
  1751. PostMessage (h, WM_CLOSE, 0, 0);
  1752. return 0;
  1753. }
  1754. break;
  1755. case SC_KEYMENU:
  1756. // (NB mustn't call sendInputAttemptWhenModalMessage() here because of very obscure
  1757. // situations that can arise if a modal loop is started from an alt-key keypress).
  1758. if (hasTitleBar() && h == GetCapture())
  1759. ReleaseCapture();
  1760. break;
  1761. case SC_MAXIMIZE:
  1762. if (! sendInputAttemptWhenModalMessage())
  1763. setFullScreen (true);
  1764. return 0;
  1765. case SC_MINIMIZE:
  1766. if (sendInputAttemptWhenModalMessage())
  1767. return 0;
  1768. if (! hasTitleBar())
  1769. {
  1770. setMinimised (true);
  1771. return 0;
  1772. }
  1773. break;
  1774. case SC_RESTORE:
  1775. if (sendInputAttemptWhenModalMessage())
  1776. return 0;
  1777. if (hasTitleBar())
  1778. {
  1779. if (isFullScreen())
  1780. {
  1781. setFullScreen (false);
  1782. return 0;
  1783. }
  1784. }
  1785. else
  1786. {
  1787. if (isMinimised())
  1788. setMinimised (false);
  1789. else if (isFullScreen())
  1790. setFullScreen (false);
  1791. return 0;
  1792. }
  1793. break;
  1794. }
  1795. break;
  1796. case WM_NCLBUTTONDOWN:
  1797. handleLeftClickInNCArea (wParam);
  1798. break;
  1799. case WM_NCRBUTTONDOWN:
  1800. case WM_NCMBUTTONDOWN:
  1801. sendInputAttemptWhenModalMessage();
  1802. break;
  1803. case WM_IME_SETCONTEXT:
  1804. imeHandler.handleSetContext (h, wParam == TRUE);
  1805. lParam &= ~ISC_SHOWUICOMPOSITIONWINDOW;
  1806. break;
  1807. case WM_IME_STARTCOMPOSITION: imeHandler.handleStartComposition (*this); return 0;
  1808. case WM_IME_ENDCOMPOSITION: imeHandler.handleEndComposition (*this, h); break;
  1809. case WM_IME_COMPOSITION: imeHandler.handleComposition (*this, h, lParam); return 0;
  1810. case WM_GETDLGCODE:
  1811. return DLGC_WANTALLKEYS;
  1812. default:
  1813. break;
  1814. }
  1815. return DefWindowProcW (h, message, wParam, lParam);
  1816. }
  1817. bool sendInputAttemptWhenModalMessage()
  1818. {
  1819. if (component->isCurrentlyBlockedByAnotherModalComponent())
  1820. {
  1821. Component* const current = Component::getCurrentlyModalComponent();
  1822. if (current != nullptr)
  1823. current->inputAttemptWhenModal();
  1824. return true;
  1825. }
  1826. return false;
  1827. }
  1828. //==============================================================================
  1829. class IMEHandler
  1830. {
  1831. public:
  1832. IMEHandler()
  1833. {
  1834. reset();
  1835. }
  1836. void handleSetContext (HWND hWnd, const bool windowIsActive)
  1837. {
  1838. if (compositionInProgress && ! windowIsActive)
  1839. {
  1840. compositionInProgress = false;
  1841. HIMC hImc = ImmGetContext (hWnd);
  1842. if (hImc != 0)
  1843. {
  1844. ImmNotifyIME (hImc, NI_COMPOSITIONSTR, CPS_COMPLETE, 0);
  1845. ImmReleaseContext (hWnd, hImc);
  1846. }
  1847. }
  1848. }
  1849. void handleStartComposition (ComponentPeer& owner)
  1850. {
  1851. reset();
  1852. TextInputTarget* const target = owner.findCurrentTextInputTarget();
  1853. if (target != nullptr)
  1854. target->insertTextAtCaret (String::empty);
  1855. }
  1856. void handleEndComposition (ComponentPeer& owner, HWND hWnd)
  1857. {
  1858. if (compositionInProgress)
  1859. {
  1860. // If this occurs, the user has cancelled the composition, so clear their changes..
  1861. TextInputTarget* const target = owner.findCurrentTextInputTarget();
  1862. if (target != nullptr)
  1863. {
  1864. target->setHighlightedRegion (compositionRange);
  1865. target->insertTextAtCaret (String::empty);
  1866. compositionRange.setLength (0);
  1867. target->setHighlightedRegion (Range<int>::emptyRange (compositionRange.getEnd()));
  1868. target->setTemporaryUnderlining (Array<Range<int> >());
  1869. }
  1870. HIMC hImc = ImmGetContext (hWnd);
  1871. if (hImc != 0)
  1872. {
  1873. ImmNotifyIME (hImc, NI_CLOSECANDIDATE, 0, 0);
  1874. ImmReleaseContext (hWnd, hImc);
  1875. }
  1876. }
  1877. reset();
  1878. }
  1879. void handleComposition (ComponentPeer& owner, HWND hWnd, const LPARAM lParam)
  1880. {
  1881. TextInputTarget* const target = owner.findCurrentTextInputTarget();
  1882. HIMC hImc = ImmGetContext (hWnd);
  1883. if (target == nullptr || hImc == 0)
  1884. return;
  1885. if (compositionRange.getStart() < 0)
  1886. compositionRange = Range<int>::emptyRange (target->getHighlightedRegion().getStart());
  1887. if ((lParam & GCS_RESULTSTR) != 0) // (composition has finished)
  1888. {
  1889. replaceCurrentSelection (target, getCompositionString (hImc, GCS_RESULTSTR),
  1890. Range<int>::emptyRange (-1));
  1891. reset();
  1892. target->setTemporaryUnderlining (Array<Range<int> >());
  1893. }
  1894. else if ((lParam & GCS_COMPSTR) != 0) // (composition is still in-progress)
  1895. {
  1896. replaceCurrentSelection (target, getCompositionString (hImc, GCS_COMPSTR),
  1897. getCompositionSelection (hImc, lParam));
  1898. target->setTemporaryUnderlining (getCompositionUnderlines (hImc, lParam));
  1899. compositionInProgress = true;
  1900. }
  1901. moveCandidateWindowToLeftAlignWithSelection (hImc, owner, target);
  1902. ImmReleaseContext (hWnd, hImc);
  1903. }
  1904. private:
  1905. //==============================================================================
  1906. Range<int> compositionRange; // The range being modified in the TextInputTarget
  1907. bool compositionInProgress;
  1908. //==============================================================================
  1909. void reset()
  1910. {
  1911. compositionRange = Range<int>::emptyRange (-1);
  1912. compositionInProgress = false;
  1913. }
  1914. String getCompositionString (HIMC hImc, const DWORD type) const
  1915. {
  1916. jassert (hImc != 0);
  1917. const int stringSizeBytes = ImmGetCompositionString (hImc, type, 0, 0);
  1918. if (stringSizeBytes > 0)
  1919. {
  1920. HeapBlock<TCHAR> buffer;
  1921. buffer.calloc (stringSizeBytes / sizeof (TCHAR) + 1);
  1922. ImmGetCompositionString (hImc, type, buffer, (DWORD) stringSizeBytes);
  1923. return String (buffer);
  1924. }
  1925. return String::empty;
  1926. }
  1927. int getCompositionCaretPos (HIMC hImc, LPARAM lParam, const String& currentIMEString) const
  1928. {
  1929. jassert (hImc != 0);
  1930. if ((lParam & CS_NOMOVECARET) != 0)
  1931. return compositionRange.getStart();
  1932. if ((lParam & GCS_CURSORPOS) != 0)
  1933. {
  1934. const int localCaretPos = ImmGetCompositionString (hImc, GCS_CURSORPOS, 0, 0);
  1935. return compositionRange.getStart() + jmax (0, localCaretPos);
  1936. }
  1937. return compositionRange.getStart() + currentIMEString.length();
  1938. }
  1939. // Get selected/highlighted range while doing composition:
  1940. // returned range is relative to beginning of TextInputTarget, not composition string
  1941. Range<int> getCompositionSelection (HIMC hImc, LPARAM lParam) const
  1942. {
  1943. jassert (hImc != 0);
  1944. int selectionStart = 0;
  1945. int selectionEnd = 0;
  1946. if ((lParam & GCS_COMPATTR) != 0)
  1947. {
  1948. // Get size of attributes array:
  1949. const int attributeSizeBytes = ImmGetCompositionString (hImc, GCS_COMPATTR, 0, 0);
  1950. if (attributeSizeBytes > 0)
  1951. {
  1952. // Get attributes (8 bit flag per character):
  1953. HeapBlock<char> attributes ((size_t) attributeSizeBytes);
  1954. ImmGetCompositionString (hImc, GCS_COMPATTR, attributes, (DWORD) attributeSizeBytes);
  1955. selectionStart = 0;
  1956. for (selectionStart = 0; selectionStart < attributeSizeBytes; ++selectionStart)
  1957. if (attributes[selectionStart] == ATTR_TARGET_CONVERTED || attributes[selectionStart] == ATTR_TARGET_NOTCONVERTED)
  1958. break;
  1959. for (selectionEnd = selectionStart; selectionEnd < attributeSizeBytes; ++selectionEnd)
  1960. if (attributes [selectionEnd] != ATTR_TARGET_CONVERTED && attributes[selectionEnd] != ATTR_TARGET_NOTCONVERTED)
  1961. break;
  1962. }
  1963. }
  1964. return Range<int> (selectionStart, selectionEnd) + compositionRange.getStart();
  1965. }
  1966. void replaceCurrentSelection (TextInputTarget* const target, const String& newContent, Range<int> newSelection)
  1967. {
  1968. if (compositionInProgress)
  1969. target->setHighlightedRegion (compositionRange);
  1970. target->insertTextAtCaret (newContent);
  1971. compositionRange.setLength (newContent.length());
  1972. if (newSelection.getStart() < 0)
  1973. newSelection = Range<int>::emptyRange (compositionRange.getEnd());
  1974. target->setHighlightedRegion (newSelection);
  1975. }
  1976. Array<Range<int> > getCompositionUnderlines (HIMC hImc, LPARAM lParam) const
  1977. {
  1978. Array<Range<int> > result;
  1979. if (hImc != 0 && (lParam & GCS_COMPCLAUSE) != 0)
  1980. {
  1981. const int clauseDataSizeBytes = ImmGetCompositionString (hImc, GCS_COMPCLAUSE, 0, 0);
  1982. if (clauseDataSizeBytes > 0)
  1983. {
  1984. const size_t numItems = clauseDataSizeBytes / sizeof (uint32);
  1985. HeapBlock<uint32> clauseData (numItems);
  1986. if (ImmGetCompositionString (hImc, GCS_COMPCLAUSE, clauseData, (DWORD) clauseDataSizeBytes) > 0)
  1987. for (size_t i = 0; i + 1 < numItems; ++i)
  1988. result.add (Range<int> ((int) clauseData [i], (int) clauseData [i + 1]) + compositionRange.getStart());
  1989. }
  1990. }
  1991. return result;
  1992. }
  1993. void moveCandidateWindowToLeftAlignWithSelection (HIMC hImc, ComponentPeer& peer, TextInputTarget* target) const
  1994. {
  1995. Component* const targetComp = dynamic_cast <Component*> (target);
  1996. if (targetComp != nullptr)
  1997. {
  1998. const Rectangle<int> area (peer.getComponent()
  1999. ->getLocalArea (targetComp, target->getCaretRectangle()));
  2000. CANDIDATEFORM pos = { 0, CFS_CANDIDATEPOS, { area.getX(), area.getBottom() }, { 0, 0, 0, 0 } };
  2001. ImmSetCandidateWindow (hImc, &pos);
  2002. }
  2003. }
  2004. JUCE_DECLARE_NON_COPYABLE (IMEHandler);
  2005. };
  2006. IMEHandler imeHandler;
  2007. //==============================================================================
  2008. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (HWNDComponentPeer);
  2009. };
  2010. ModifierKeys HWNDComponentPeer::currentModifiers;
  2011. ModifierKeys HWNDComponentPeer::modifiersAtLastCallback;
  2012. ComponentPeer* Component::createNewPeer (int styleFlags, void* nativeWindowToAttachTo)
  2013. {
  2014. return new HWNDComponentPeer (this, styleFlags, (HWND) nativeWindowToAttachTo);
  2015. }
  2016. ComponentPeer* createNonRepaintingEmbeddedWindowsPeer (Component* component, void* parent)
  2017. {
  2018. HWNDComponentPeer* const p = new HWNDComponentPeer (component, ComponentPeer::windowIgnoresMouseClicks, (HWND) parent);
  2019. p->dontRepaint = true;
  2020. return p;
  2021. }
  2022. juce_ImplementSingleton_SingleThreaded (HWNDComponentPeer::WindowClassHolder);
  2023. //==============================================================================
  2024. void ModifierKeys::updateCurrentModifiers() noexcept
  2025. {
  2026. currentModifiers = HWNDComponentPeer::currentModifiers;
  2027. }
  2028. ModifierKeys ModifierKeys::getCurrentModifiersRealtime() noexcept
  2029. {
  2030. HWNDComponentPeer::updateKeyModifiers();
  2031. int mouseMods = 0;
  2032. if (HWNDComponentPeer::isKeyDown (VK_LBUTTON)) mouseMods |= ModifierKeys::leftButtonModifier;
  2033. if (HWNDComponentPeer::isKeyDown (VK_RBUTTON)) mouseMods |= ModifierKeys::rightButtonModifier;
  2034. if (HWNDComponentPeer::isKeyDown (VK_MBUTTON)) mouseMods |= ModifierKeys::middleButtonModifier;
  2035. HWNDComponentPeer::currentModifiers
  2036. = HWNDComponentPeer::currentModifiers.withoutMouseButtons().withFlags (mouseMods);
  2037. return HWNDComponentPeer::currentModifiers;
  2038. }
  2039. //==============================================================================
  2040. bool KeyPress::isKeyCurrentlyDown (const int keyCode)
  2041. {
  2042. SHORT k = (SHORT) keyCode;
  2043. if ((keyCode & extendedKeyModifier) == 0
  2044. && (k >= (SHORT) 'a' && k <= (SHORT) 'z'))
  2045. k += (SHORT) 'A' - (SHORT) 'a';
  2046. const SHORT translatedValues[] = { (SHORT) ',', VK_OEM_COMMA,
  2047. (SHORT) '+', VK_OEM_PLUS,
  2048. (SHORT) '-', VK_OEM_MINUS,
  2049. (SHORT) '.', VK_OEM_PERIOD,
  2050. (SHORT) ';', VK_OEM_1,
  2051. (SHORT) ':', VK_OEM_1,
  2052. (SHORT) '/', VK_OEM_2,
  2053. (SHORT) '?', VK_OEM_2,
  2054. (SHORT) '[', VK_OEM_4,
  2055. (SHORT) ']', VK_OEM_6 };
  2056. for (int i = 0; i < numElementsInArray (translatedValues); i += 2)
  2057. if (k == translatedValues [i])
  2058. k = translatedValues [i + 1];
  2059. return HWNDComponentPeer::isKeyDown (k);
  2060. }
  2061. //==============================================================================
  2062. bool Process::isForegroundProcess()
  2063. {
  2064. HWND fg = GetForegroundWindow();
  2065. if (fg == 0)
  2066. return true;
  2067. // when running as a plugin in IE8, the browser UI runs in a different process to the plugin, so
  2068. // process ID isn't a reliable way to check if the foreground window belongs to us - instead, we
  2069. // have to see if any of our windows are children of the foreground window
  2070. fg = GetAncestor (fg, GA_ROOT);
  2071. for (int i = ComponentPeer::getNumPeers(); --i >= 0;)
  2072. {
  2073. HWNDComponentPeer* const wp = dynamic_cast <HWNDComponentPeer*> (ComponentPeer::getPeer (i));
  2074. if (wp != nullptr && wp->isInside (fg))
  2075. return true;
  2076. }
  2077. return false;
  2078. }
  2079. //==============================================================================
  2080. class WindowsMessageBox : public AsyncUpdater
  2081. {
  2082. public:
  2083. WindowsMessageBox (AlertWindow::AlertIconType iconType,
  2084. const String& title_, const String& message_,
  2085. Component* associatedComponent,
  2086. UINT extraFlags,
  2087. ModalComponentManager::Callback* callback_,
  2088. const bool runAsync)
  2089. : flags (extraFlags | getMessageBoxFlags (iconType)),
  2090. owner (getWindowForMessageBox (associatedComponent)),
  2091. title (title_), message (message_), callback (callback_)
  2092. {
  2093. if (runAsync)
  2094. triggerAsyncUpdate();
  2095. }
  2096. int getResult() const
  2097. {
  2098. const int r = MessageBox (owner, message.toWideCharPointer(), title.toWideCharPointer(), flags);
  2099. return (r == IDYES || r == IDOK) ? 1 : (r == IDNO ? 2 : 0);
  2100. }
  2101. void handleAsyncUpdate()
  2102. {
  2103. const int result = getResult();
  2104. if (callback != nullptr)
  2105. callback->modalStateFinished (result);
  2106. delete this;
  2107. }
  2108. private:
  2109. UINT flags;
  2110. HWND owner;
  2111. String title, message;
  2112. ModalComponentManager::Callback* callback;
  2113. static UINT getMessageBoxFlags (AlertWindow::AlertIconType iconType) noexcept
  2114. {
  2115. UINT flags = MB_TASKMODAL | MB_SETFOREGROUND;
  2116. switch (iconType)
  2117. {
  2118. case AlertWindow::QuestionIcon: flags |= MB_ICONQUESTION; break;
  2119. case AlertWindow::WarningIcon: flags |= MB_ICONWARNING; break;
  2120. case AlertWindow::InfoIcon: flags |= MB_ICONINFORMATION; break;
  2121. default: break;
  2122. }
  2123. return flags;
  2124. }
  2125. static HWND getWindowForMessageBox (Component* associatedComponent)
  2126. {
  2127. return associatedComponent != nullptr ? (HWND) associatedComponent->getWindowHandle() : 0;
  2128. }
  2129. };
  2130. void JUCE_CALLTYPE NativeMessageBox::showMessageBox (AlertWindow::AlertIconType iconType,
  2131. const String& title, const String& message,
  2132. Component* associatedComponent)
  2133. {
  2134. WindowsMessageBox box (iconType, title, message, associatedComponent, MB_OK, 0, false);
  2135. (void) box.getResult();
  2136. }
  2137. void JUCE_CALLTYPE NativeMessageBox::showMessageBoxAsync (AlertWindow::AlertIconType iconType,
  2138. const String& title, const String& message,
  2139. Component* associatedComponent)
  2140. {
  2141. new WindowsMessageBox (iconType, title, message, associatedComponent, MB_OK, 0, true);
  2142. }
  2143. bool JUCE_CALLTYPE NativeMessageBox::showOkCancelBox (AlertWindow::AlertIconType iconType,
  2144. const String& title, const String& message,
  2145. Component* associatedComponent,
  2146. ModalComponentManager::Callback* callback)
  2147. {
  2148. ScopedPointer<WindowsMessageBox> mb (new WindowsMessageBox (iconType, title, message, associatedComponent,
  2149. MB_OKCANCEL, callback, callback != nullptr));
  2150. if (callback == nullptr)
  2151. return mb->getResult() != 0;
  2152. mb.release();
  2153. return false;
  2154. }
  2155. int JUCE_CALLTYPE NativeMessageBox::showYesNoCancelBox (AlertWindow::AlertIconType iconType,
  2156. const String& title, const String& message,
  2157. Component* associatedComponent,
  2158. ModalComponentManager::Callback* callback)
  2159. {
  2160. ScopedPointer<WindowsMessageBox> mb (new WindowsMessageBox (iconType, title, message, associatedComponent,
  2161. MB_YESNOCANCEL, callback, callback != nullptr));
  2162. if (callback == nullptr)
  2163. return mb->getResult();
  2164. mb.release();
  2165. return 0;
  2166. }
  2167. //==============================================================================
  2168. void Desktop::createMouseInputSources()
  2169. {
  2170. mouseSources.add (new MouseInputSource (0, true));
  2171. }
  2172. Point<int> MouseInputSource::getCurrentMousePosition()
  2173. {
  2174. POINT mousePos;
  2175. GetCursorPos (&mousePos);
  2176. return Point<int> (mousePos.x, mousePos.y);
  2177. }
  2178. void Desktop::setMousePosition (const Point<int>& newPosition)
  2179. {
  2180. SetCursorPos (newPosition.getX(), newPosition.getY());
  2181. }
  2182. //==============================================================================
  2183. Image::SharedImage* Image::SharedImage::createNativeImage (PixelFormat format, int width, int height, bool clearImage)
  2184. {
  2185. return createSoftwareImage (format, width, height, clearImage);
  2186. }
  2187. //==============================================================================
  2188. class ScreenSaverDefeater : public Timer
  2189. {
  2190. public:
  2191. ScreenSaverDefeater()
  2192. {
  2193. startTimer (10000);
  2194. timerCallback();
  2195. }
  2196. void timerCallback()
  2197. {
  2198. if (Process::isForegroundProcess())
  2199. {
  2200. // simulate a shift key getting pressed..
  2201. INPUT input[2];
  2202. input[0].type = INPUT_KEYBOARD;
  2203. input[0].ki.wVk = VK_SHIFT;
  2204. input[0].ki.dwFlags = 0;
  2205. input[0].ki.dwExtraInfo = 0;
  2206. input[1].type = INPUT_KEYBOARD;
  2207. input[1].ki.wVk = VK_SHIFT;
  2208. input[1].ki.dwFlags = KEYEVENTF_KEYUP;
  2209. input[1].ki.dwExtraInfo = 0;
  2210. SendInput (2, input, sizeof (INPUT));
  2211. }
  2212. }
  2213. };
  2214. static ScopedPointer<ScreenSaverDefeater> screenSaverDefeater;
  2215. void Desktop::setScreenSaverEnabled (const bool isEnabled)
  2216. {
  2217. if (isEnabled)
  2218. screenSaverDefeater = nullptr;
  2219. else if (screenSaverDefeater == nullptr)
  2220. screenSaverDefeater = new ScreenSaverDefeater();
  2221. }
  2222. bool Desktop::isScreenSaverEnabled()
  2223. {
  2224. return screenSaverDefeater == nullptr;
  2225. }
  2226. /* (The code below is the "correct" way to disable the screen saver, but it
  2227. completely fails on winXP when the saver is password-protected...)
  2228. static bool juce_screenSaverEnabled = true;
  2229. void Desktop::setScreenSaverEnabled (const bool isEnabled) noexcept
  2230. {
  2231. juce_screenSaverEnabled = isEnabled;
  2232. SetThreadExecutionState (isEnabled ? ES_CONTINUOUS
  2233. : (ES_DISPLAY_REQUIRED | ES_CONTINUOUS));
  2234. }
  2235. bool Desktop::isScreenSaverEnabled() noexcept
  2236. {
  2237. return juce_screenSaverEnabled;
  2238. }
  2239. */
  2240. //==============================================================================
  2241. void LookAndFeel::playAlertSound()
  2242. {
  2243. MessageBeep (MB_OK);
  2244. }
  2245. //==============================================================================
  2246. void SystemClipboard::copyTextToClipboard (const String& text)
  2247. {
  2248. if (OpenClipboard (0) != 0)
  2249. {
  2250. if (EmptyClipboard() != 0)
  2251. {
  2252. const size_t bytesNeeded = CharPointer_UTF16::getBytesRequiredFor (text.getCharPointer()) + 4;
  2253. if (bytesNeeded > 0)
  2254. {
  2255. HGLOBAL bufH = GlobalAlloc (GMEM_MOVEABLE | GMEM_DDESHARE | GMEM_ZEROINIT, bytesNeeded + sizeof (WCHAR));
  2256. if (bufH != 0)
  2257. {
  2258. WCHAR* const data = static_cast <WCHAR*> (GlobalLock (bufH));
  2259. if (data != nullptr)
  2260. {
  2261. text.copyToUTF16 (data, (int) bytesNeeded);
  2262. GlobalUnlock (bufH);
  2263. SetClipboardData (CF_UNICODETEXT, bufH);
  2264. }
  2265. }
  2266. }
  2267. }
  2268. CloseClipboard();
  2269. }
  2270. }
  2271. String SystemClipboard::getTextFromClipboard()
  2272. {
  2273. String result;
  2274. if (OpenClipboard (0) != 0)
  2275. {
  2276. HANDLE bufH = GetClipboardData (CF_UNICODETEXT);
  2277. if (bufH != 0)
  2278. {
  2279. const WCHAR* const data = (const WCHAR*) GlobalLock (bufH);
  2280. if (data != nullptr)
  2281. {
  2282. result = String (data, (size_t) (GlobalSize (bufH) / sizeof (WCHAR)));
  2283. GlobalUnlock (bufH);
  2284. }
  2285. }
  2286. CloseClipboard();
  2287. }
  2288. return result;
  2289. }
  2290. //==============================================================================
  2291. void Desktop::setKioskComponent (Component* kioskModeComponent, bool enableOrDisable, bool /*allowMenusAndBars*/)
  2292. {
  2293. if (enableOrDisable)
  2294. kioskModeComponent->setBounds (Desktop::getInstance().getMainMonitorArea (false));
  2295. }
  2296. //==============================================================================
  2297. static BOOL CALLBACK enumMonitorsProc (HMONITOR, HDC, LPRECT r, LPARAM userInfo)
  2298. {
  2299. Array <Rectangle<int> >* const monitorCoords = (Array <Rectangle<int> >*) userInfo;
  2300. monitorCoords->add (Rectangle<int> (r->left, r->top, r->right - r->left, r->bottom - r->top));
  2301. return TRUE;
  2302. }
  2303. void Desktop::getCurrentMonitorPositions (Array <Rectangle<int> >& monitorCoords, const bool clipToWorkArea)
  2304. {
  2305. EnumDisplayMonitors (0, 0, &enumMonitorsProc, (LPARAM) &monitorCoords);
  2306. // make sure the first in the list is the main monitor
  2307. for (int i = 1; i < monitorCoords.size(); ++i)
  2308. if (monitorCoords[i].getX() == 0 && monitorCoords[i].getY() == 0)
  2309. monitorCoords.swap (i, 0);
  2310. if (monitorCoords.size() == 0)
  2311. {
  2312. RECT r;
  2313. GetWindowRect (GetDesktopWindow(), &r);
  2314. monitorCoords.add (Rectangle<int> (r.left, r.top, r.right - r.left, r.bottom - r.top));
  2315. }
  2316. if (clipToWorkArea)
  2317. {
  2318. // clip the main monitor to the active non-taskbar area
  2319. RECT r;
  2320. SystemParametersInfo (SPI_GETWORKAREA, 0, &r, 0);
  2321. Rectangle<int>& screen = monitorCoords.getReference (0);
  2322. screen.setPosition (jmax (screen.getX(), (int) r.left),
  2323. jmax (screen.getY(), (int) r.top));
  2324. screen.setSize (jmin (screen.getRight(), (int) r.right) - screen.getX(),
  2325. jmin (screen.getBottom(), (int) r.bottom) - screen.getY());
  2326. }
  2327. }
  2328. //==============================================================================
  2329. static HICON extractFileHICON (const File& file)
  2330. {
  2331. WORD iconNum = 0;
  2332. WCHAR name [MAX_PATH * 2];
  2333. file.getFullPathName().copyToUTF16 (name, sizeof (name));
  2334. return ExtractAssociatedIcon ((HINSTANCE) Process::getCurrentModuleInstanceHandle(),
  2335. name, &iconNum);
  2336. }
  2337. Image juce_createIconForFile (const File& file)
  2338. {
  2339. Image image;
  2340. HICON icon = extractFileHICON (file);
  2341. if (icon != 0)
  2342. {
  2343. image = IconConverters::createImageFromHICON (icon);
  2344. DestroyIcon (icon);
  2345. }
  2346. return image;
  2347. }
  2348. //==============================================================================
  2349. void* MouseCursor::createMouseCursorFromImage (const Image& image, int hotspotX, int hotspotY)
  2350. {
  2351. const int maxW = GetSystemMetrics (SM_CXCURSOR);
  2352. const int maxH = GetSystemMetrics (SM_CYCURSOR);
  2353. Image im (image);
  2354. if (im.getWidth() > maxW || im.getHeight() > maxH)
  2355. {
  2356. im = im.rescaled (maxW, maxH);
  2357. hotspotX = (hotspotX * maxW) / image.getWidth();
  2358. hotspotY = (hotspotY * maxH) / image.getHeight();
  2359. }
  2360. return IconConverters::createHICONFromImage (im, FALSE, hotspotX, hotspotY);
  2361. }
  2362. void MouseCursor::deleteMouseCursor (void* const cursorHandle, const bool isStandard)
  2363. {
  2364. if (cursorHandle != nullptr && ! isStandard)
  2365. DestroyCursor ((HCURSOR) cursorHandle);
  2366. }
  2367. enum
  2368. {
  2369. hiddenMouseCursorHandle = 32500 // (arbitrary non-zero value to mark this type of cursor)
  2370. };
  2371. void* MouseCursor::createStandardMouseCursor (const MouseCursor::StandardCursorType type)
  2372. {
  2373. LPCTSTR cursorName = IDC_ARROW;
  2374. switch (type)
  2375. {
  2376. case NormalCursor: break;
  2377. case NoCursor: return (void*) hiddenMouseCursorHandle;
  2378. case WaitCursor: cursorName = IDC_WAIT; break;
  2379. case IBeamCursor: cursorName = IDC_IBEAM; break;
  2380. case PointingHandCursor: cursorName = MAKEINTRESOURCE(32649); break;
  2381. case CrosshairCursor: cursorName = IDC_CROSS; break;
  2382. case CopyingCursor: break; // can't seem to find one of these in the system list..
  2383. case LeftRightResizeCursor:
  2384. case LeftEdgeResizeCursor:
  2385. case RightEdgeResizeCursor: cursorName = IDC_SIZEWE; break;
  2386. case UpDownResizeCursor:
  2387. case TopEdgeResizeCursor:
  2388. case BottomEdgeResizeCursor: cursorName = IDC_SIZENS; break;
  2389. case TopLeftCornerResizeCursor:
  2390. case BottomRightCornerResizeCursor: cursorName = IDC_SIZENWSE; break;
  2391. case TopRightCornerResizeCursor:
  2392. case BottomLeftCornerResizeCursor: cursorName = IDC_SIZENESW; break;
  2393. case UpDownLeftRightResizeCursor: cursorName = IDC_SIZEALL; break;
  2394. case DraggingHandCursor:
  2395. {
  2396. static void* dragHandCursor = nullptr;
  2397. if (dragHandCursor == nullptr)
  2398. {
  2399. static const unsigned char dragHandData[] =
  2400. { 71,73,70,56,57,97,16,0,16,0,145,2,0,0,0,0,255,255,255,0,0,0,0,0,0,33,249,4,1,0,0,2,0,44,0,0,0,0,16,0,
  2401. 16,0,0,2,52,148,47,0,200,185,16,130,90,12,74,139,107,84,123,39,132,117,151,116,132,146,248,60,209,138,
  2402. 98,22,203,114,34,236,37,52,77,217,247,154,191,119,110,240,193,128,193,95,163,56,60,234,98,135,2,0,59 };
  2403. dragHandCursor = createMouseCursorFromImage (ImageFileFormat::loadFrom (dragHandData, sizeof (dragHandData)), 8, 7);
  2404. }
  2405. return dragHandCursor;
  2406. }
  2407. default:
  2408. jassertfalse; break;
  2409. }
  2410. HCURSOR cursorH = LoadCursor (0, cursorName);
  2411. if (cursorH == 0)
  2412. cursorH = LoadCursor (0, IDC_ARROW);
  2413. return cursorH;
  2414. }
  2415. //==============================================================================
  2416. void MouseCursor::showInWindow (ComponentPeer*) const
  2417. {
  2418. HCURSOR c = (HCURSOR) getHandle();
  2419. if (c == 0)
  2420. c = LoadCursor (0, IDC_ARROW);
  2421. else if (c == (HCURSOR) hiddenMouseCursorHandle)
  2422. c = 0;
  2423. SetCursor (c);
  2424. }
  2425. void MouseCursor::showInAllWindows() const
  2426. {
  2427. showInWindow (nullptr);
  2428. }