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.

384 lines
17KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2013 - Raw Material Software Ltd.
  5. Permission is granted to use this software under the terms of either:
  6. a) the GPL v2 (or any later version)
  7. b) the Affero GPL v3
  8. Details of these licenses can be found at: www.gnu.org/licenses
  9. JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
  10. WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  11. A PARTICULAR PURPOSE. See the GNU General Public License for more details.
  12. ------------------------------------------------------------------------------
  13. To release a closed-source product which uses JUCE, commercial licenses are
  14. available: visit www.juce.com for more information.
  15. ==============================================================================
  16. */
  17. #ifndef __JUCE_COMPONENTPEER_JUCEHEADER__
  18. #define __JUCE_COMPONENTPEER_JUCEHEADER__
  19. #include "../components/juce_Component.h"
  20. #include "../mouse/juce_MouseCursor.h"
  21. #include "../keyboard/juce_TextInputTarget.h"
  22. class ComponentBoundsConstrainer;
  23. //==============================================================================
  24. /**
  25. The Component class uses a ComponentPeer internally to create and manage a real
  26. operating-system window.
  27. This is an abstract base class - the platform specific code contains implementations of
  28. it for the various platforms.
  29. User-code should very rarely need to have any involvement with this class.
  30. @see Component::createNewPeer
  31. */
  32. class JUCE_API ComponentPeer
  33. {
  34. public:
  35. //==============================================================================
  36. /** A combination of these flags is passed to the ComponentPeer constructor. */
  37. enum StyleFlags
  38. {
  39. windowAppearsOnTaskbar = (1 << 0), /**< Indicates that the window should have a corresponding
  40. entry on the taskbar (ignored on MacOSX) */
  41. windowIsTemporary = (1 << 1), /**< Indicates that the window is a temporary popup, like a menu,
  42. tooltip, etc. */
  43. windowIgnoresMouseClicks = (1 << 2), /**< Indicates that the window should let mouse clicks pass
  44. through it (may not be possible on some platforms). */
  45. windowHasTitleBar = (1 << 3), /**< Indicates that the window should have a normal OS-specific
  46. title bar and frame. if not specified, the window will be
  47. borderless. */
  48. windowIsResizable = (1 << 4), /**< Indicates that the window should have a resizable border. */
  49. windowHasMinimiseButton = (1 << 5), /**< Indicates that if the window has a title bar, it should have a
  50. minimise button on it. */
  51. windowHasMaximiseButton = (1 << 6), /**< Indicates that if the window has a title bar, it should have a
  52. maximise button on it. */
  53. windowHasCloseButton = (1 << 7), /**< Indicates that if the window has a title bar, it should have a
  54. close button on it. */
  55. windowHasDropShadow = (1 << 8), /**< Indicates that the window should have a drop-shadow (this may
  56. not be possible on all platforms). */
  57. windowRepaintedExplictly = (1 << 9), /**< Not intended for public use - this tells a window not to
  58. do its own repainting, but only to repaint when the
  59. performAnyPendingRepaintsNow() method is called. */
  60. windowIgnoresKeyPresses = (1 << 10), /**< Tells the window not to catch any keypresses. This can
  61. be used for things like plugin windows, to stop them interfering
  62. with the host's shortcut keys */
  63. windowIsSemiTransparent = (1 << 31) /**< Not intended for public use - makes a window transparent. */
  64. };
  65. //==============================================================================
  66. /** Creates a peer.
  67. The component is the one that we intend to represent, and the style flags are
  68. a combination of the values in the StyleFlags enum
  69. */
  70. ComponentPeer (Component& component, int styleFlags);
  71. /** Destructor. */
  72. virtual ~ComponentPeer();
  73. //==============================================================================
  74. /** Returns the component being represented by this peer. */
  75. Component& getComponent() noexcept { return component; }
  76. /** Returns the set of style flags that were set when the window was created.
  77. @see Component::addToDesktop
  78. */
  79. int getStyleFlags() const noexcept { return styleFlags; }
  80. /** Returns a unique ID for this peer.
  81. Each peer that is created is given a different ID.
  82. */
  83. uint32 getUniqueID() const noexcept { return uniqueID; }
  84. //==============================================================================
  85. /** Returns the raw handle to whatever kind of window is being used.
  86. On windows, this is probably a HWND, on the mac, it's likely to be a WindowRef,
  87. but rememeber there's no guarantees what you'll get back.
  88. */
  89. virtual void* getNativeHandle() const = 0;
  90. /** Shows or hides the window. */
  91. virtual void setVisible (bool shouldBeVisible) = 0;
  92. /** Changes the title of the window. */
  93. virtual void setTitle (const String& title) = 0;
  94. /** If this type of window is capable of indicating that the document in it has been
  95. edited, then this changes its status.
  96. For example in OSX, this changes the appearance of the close button.
  97. @returns true if the window has a mechanism for showing this, or false if not.
  98. */
  99. virtual bool setDocumentEditedStatus (bool edited);
  100. /** If this type of window is capable of indicating that it represents a file, then
  101. this lets you set the file.
  102. E.g. in OSX it'll show an icon for the file in the title bar.
  103. */
  104. virtual void setRepresentedFile (const File&);
  105. //==============================================================================
  106. /** Moves and resizes the window.
  107. If the native window is contained in another window, then the co-ordinates are
  108. relative to the parent window's origin, not the screen origin.
  109. This should result in a callback to handleMovedOrResized().
  110. */
  111. virtual void setBounds (const Rectangle<int>& newBounds, bool isNowFullScreen) = 0;
  112. /** Returns the current position and size of the window.
  113. If the native window is contained in another window, then the co-ordinates are
  114. relative to the parent window's origin, not the screen origin.
  115. */
  116. virtual Rectangle<int> getBounds() const = 0;
  117. /** Converts a position relative to the top-left of this component to screen co-ordinates. */
  118. virtual Point<int> localToGlobal (const Point<int>& relativePosition) = 0;
  119. /** Converts a rectangle relative to the top-left of this component to screen co-ordinates. */
  120. virtual Rectangle<int> localToGlobal (const Rectangle<int>& relativePosition);
  121. /** Converts a screen co-ordinate to a position relative to the top-left of this component. */
  122. virtual Point<int> globalToLocal (const Point<int>& screenPosition) = 0;
  123. /** Converts a screen area to a position relative to the top-left of this component. */
  124. virtual Rectangle<int> globalToLocal (const Rectangle<int>& screenPosition);
  125. /** Minimises the window. */
  126. virtual void setMinimised (bool shouldBeMinimised) = 0;
  127. /** True if the window is currently minimised. */
  128. virtual bool isMinimised() const = 0;
  129. /** Enable/disable fullscreen mode for the window. */
  130. virtual void setFullScreen (bool shouldBeFullScreen) = 0;
  131. /** True if the window is currently full-screen. */
  132. virtual bool isFullScreen() const = 0;
  133. /** Sets the size to restore to if fullscreen mode is turned off. */
  134. void setNonFullScreenBounds (const Rectangle<int>& newBounds) noexcept;
  135. /** Returns the size to restore to if fullscreen mode is turned off. */
  136. const Rectangle<int>& getNonFullScreenBounds() const noexcept;
  137. /** Attempts to change the icon associated with this window. */
  138. virtual void setIcon (const Image& newIcon) = 0;
  139. /** Sets a constrainer to use if the peer can resize itself.
  140. The constrainer won't be deleted by this object, so the caller must manage its lifetime.
  141. */
  142. void setConstrainer (ComponentBoundsConstrainer* newConstrainer) noexcept;
  143. /** Returns the current constrainer, if one has been set. */
  144. ComponentBoundsConstrainer* getConstrainer() const noexcept { return constrainer; }
  145. /** Checks if a point is in the window.
  146. Coordinates are relative to the top-left of this window. If trueIfInAChildWindow
  147. is false, then this returns false if the point is actually inside a child of this
  148. window.
  149. */
  150. virtual bool contains (const Point<int>& position, bool trueIfInAChildWindow) const = 0;
  151. /** Returns the size of the window frame that's around this window.
  152. Whether or not the window has a normal window frame depends on the flags
  153. that were set when the window was created by Component::addToDesktop()
  154. */
  155. virtual BorderSize<int> getFrameSize() const = 0;
  156. /** This is called when the window's bounds change.
  157. A peer implementation must call this when the window is moved and resized, so that
  158. this method can pass the message on to the component.
  159. */
  160. void handleMovedOrResized();
  161. /** This is called if the screen resolution changes.
  162. A peer implementation must call this if the monitor arrangement changes or the available
  163. screen size changes.
  164. */
  165. virtual void handleScreenSizeChange();
  166. //==============================================================================
  167. /** This is called to repaint the component into the given context. */
  168. void handlePaint (LowLevelGraphicsContext& contextToPaintTo);
  169. //==============================================================================
  170. /** Sets this window to either be always-on-top or normal.
  171. Some kinds of window might not be able to do this, so should return false.
  172. */
  173. virtual bool setAlwaysOnTop (bool alwaysOnTop) = 0;
  174. /** Brings the window to the top, optionally also giving it focus. */
  175. virtual void toFront (bool makeActive) = 0;
  176. /** Moves the window to be just behind another one. */
  177. virtual void toBehind (ComponentPeer* other) = 0;
  178. /** Called when the window is brought to the front, either by the OS or by a call
  179. to toFront().
  180. */
  181. void handleBroughtToFront();
  182. //==============================================================================
  183. /** True if the window has the keyboard focus. */
  184. virtual bool isFocused() const = 0;
  185. /** Tries to give the window keyboard focus. */
  186. virtual void grabFocus() = 0;
  187. /** Called when the window gains keyboard focus. */
  188. void handleFocusGain();
  189. /** Called when the window loses keyboard focus. */
  190. void handleFocusLoss();
  191. Component* getLastFocusedSubcomponent() const noexcept;
  192. /** Called when a key is pressed.
  193. For keycode info, see the KeyPress class.
  194. Returns true if the keystroke was used.
  195. */
  196. bool handleKeyPress (int keyCode, juce_wchar textCharacter);
  197. /** Called whenever a key is pressed or released.
  198. Returns true if the keystroke was used.
  199. */
  200. bool handleKeyUpOrDown (bool isKeyDown);
  201. /** Called whenever a modifier key is pressed or released. */
  202. void handleModifierKeysChange();
  203. //==============================================================================
  204. /** Tells the window that text input may be required at the given position.
  205. This may cause things like a virtual on-screen keyboard to appear, depending
  206. on the OS.
  207. */
  208. virtual void textInputRequired (const Point<int>& position) = 0;
  209. /** If there's some kind of OS input-method in progress, this should dismiss it. */
  210. virtual void dismissPendingTextInput();
  211. /** Returns the currently focused TextInputTarget, or null if none is found. */
  212. TextInputTarget* findCurrentTextInputTarget();
  213. //==============================================================================
  214. /** Invalidates a region of the window to be repainted asynchronously. */
  215. virtual void repaint (const Rectangle<int>& area) = 0;
  216. /** This can be called (from the message thread) to cause the immediate redrawing
  217. of any areas of this window that need repainting.
  218. You shouldn't ever really need to use this, it's mainly for special purposes
  219. like supporting audio plugins where the host's event loop is out of our control.
  220. */
  221. virtual void performAnyPendingRepaintsNow() = 0;
  222. /** Changes the window's transparency. */
  223. virtual void setAlpha (float newAlpha) = 0;
  224. //==============================================================================
  225. void handleMouseEvent (int touchIndex, const Point<int> positionWithinPeer, const ModifierKeys newMods, int64 time);
  226. void handleMouseWheel (int touchIndex, const Point<int> positionWithinPeer, int64 time, const MouseWheelDetails&);
  227. void handleMagnifyGesture (int touchIndex, const Point<int> positionWithinPeer, int64 time, float scaleFactor);
  228. void handleUserClosingWindow();
  229. struct DragInfo
  230. {
  231. StringArray files;
  232. String text;
  233. Point<int> position;
  234. bool isEmpty() const noexcept { return files.size() == 0 && text.isEmpty(); }
  235. void clear() noexcept { files.clear(); text = String::empty; }
  236. };
  237. bool handleDragMove (const DragInfo&);
  238. bool handleDragExit (const DragInfo&);
  239. bool handleDragDrop (const DragInfo&);
  240. //==============================================================================
  241. /** Resets the masking region.
  242. The subclass should call this every time it's about to call the handlePaint method.
  243. @see addMaskedRegion
  244. */
  245. void clearMaskedRegion();
  246. /** Adds a rectangle to the set of areas not to paint over.
  247. A component can call this on its peer during its paint() method, to signal
  248. that the painting code should ignore a given region. The reason
  249. for this is to stop embedded windows (such as OpenGL) getting painted over.
  250. The masked region is cleared each time before a paint happens, so a component
  251. will have to make sure it calls this every time it's painted.
  252. */
  253. void addMaskedRegion (const Rectangle<int>& area);
  254. //==============================================================================
  255. /** Returns the number of currently-active peers.
  256. @see getPeer
  257. */
  258. static int getNumPeers() noexcept;
  259. /** Returns one of the currently-active peers.
  260. @see getNumPeers
  261. */
  262. static ComponentPeer* getPeer (int index) noexcept;
  263. /** Returns the peer that's attached to the given component, or nullptr if there isn't one. */
  264. static ComponentPeer* getPeerFor (const Component*) noexcept;
  265. /** Checks if this peer object is valid.
  266. @see getNumPeers
  267. */
  268. static bool isValidPeer (const ComponentPeer* peer) noexcept;
  269. //==============================================================================
  270. virtual StringArray getAvailableRenderingEngines();
  271. virtual int getCurrentRenderingEngine() const;
  272. virtual void setCurrentRenderingEngine (int index);
  273. protected:
  274. //==============================================================================
  275. Component& component;
  276. const int styleFlags;
  277. RectangleList maskedRegion;
  278. Rectangle<int> lastNonFullscreenBounds;
  279. ComponentBoundsConstrainer* constrainer;
  280. static void updateCurrentModifiers() noexcept;
  281. private:
  282. //==============================================================================
  283. WeakReference<Component> lastFocusedComponent, dragAndDropTargetComponent;
  284. Component* lastDragAndDropCompUnderMouse;
  285. const uint32 uniqueID;
  286. bool fakeMouseMessageSent, isWindowMinimised;
  287. Component* getTargetForKeyPress();
  288. static MouseInputSource* getOrCreateMouseInputSource (int);
  289. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ComponentPeer)
  290. };
  291. #endif // __JUCE_COMPONENTPEER_JUCEHEADER__