| 
							- /*
 -   ==============================================================================
 - 
 -    This file is part of the JUCE library.
 -    Copyright (c) 2022 - Raw Material Software Limited
 - 
 -    JUCE is an open source library subject to commercial or open-source
 -    licensing.
 - 
 -    By using JUCE, you agree to the terms of both the JUCE 7 End-User License
 -    Agreement and JUCE Privacy Policy.
 - 
 -    End User License Agreement: www.juce.com/juce-7-licence
 -    Privacy Policy: www.juce.com/juce-privacy-policy
 - 
 -    Or: You may also use this code under the terms of the GPL v3 (see
 -    www.gnu.org/licenses).
 - 
 -    JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
 -    EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
 -    DISCLAIMED.
 - 
 -   ==============================================================================
 - */
 - 
 - namespace juce
 - {
 - 
 - struct XFreeDeleter
 - {
 -     void operator() (void* ptr) const
 -     {
 -         if (ptr != nullptr)
 -             X11Symbols::getInstance()->xFree (ptr);
 -     }
 - };
 - 
 - template <typename Data>
 - std::unique_ptr<Data, XFreeDeleter> makeXFreePtr (Data* raw) { return std::unique_ptr<Data, XFreeDeleter> (raw); }
 - 
 - //==============================================================================
 - // Defined in juce_Windowing_linux.cpp
 - void juce_LinuxAddRepaintListener (ComponentPeer*, Component* dummy);
 - void juce_LinuxRemoveRepaintListener (ComponentPeer*, Component* dummy);
 - 
 - class PeerListener : private ComponentMovementWatcher
 - {
 - public:
 -     PeerListener (Component& comp, Window embeddedWindow)
 -         : ComponentMovementWatcher (&comp),
 -           window (embeddedWindow),
 -           association (comp.getPeer(), window) {}
 - 
 - private:
 -     using ComponentMovementWatcher::componentMovedOrResized,
 -           ComponentMovementWatcher::componentVisibilityChanged;
 - 
 -     void componentMovedOrResized (bool, bool) override {}
 -     void componentVisibilityChanged() override {}
 - 
 -     void componentPeerChanged() override
 -     {
 -         // This should not be rewritten as a ternary expression or similar.
 -         // The old association must be destroyed before the new one is created.
 -         association = {};
 - 
 -         if (auto* comp = getComponent())
 -             association = ScopedWindowAssociation (comp->getPeer(), window);
 -     }
 - 
 -     Window window{};
 -     ScopedWindowAssociation association;
 - };
 - 
 - //==============================================================================
 - class OpenGLContext::NativeContext
 - {
 - private:
 -     struct DummyComponent  : public Component
 -     {
 -         DummyComponent (OpenGLContext::NativeContext& nativeParentContext)
 -             : native (nativeParentContext)
 -         {
 -         }
 - 
 -         void handleCommandMessage (int commandId) override
 -         {
 -             if (commandId == 0)
 -                 native.triggerRepaint();
 -         }
 - 
 -         OpenGLContext::NativeContext& native;
 -     };
 - 
 -     template <typename Traits>
 -     class ScopedGLXObject
 -     {
 -     public:
 -         using Type = typename Traits::Type;
 - 
 -         ScopedGLXObject() = default;
 - 
 -         explicit ScopedGLXObject (Type obj, ::Display* d)
 -             : object (obj), display (d) {}
 - 
 -         ScopedGLXObject (ScopedGLXObject&& other) noexcept
 -             : object (std::exchange (other.object, Type{})),
 -               display (std::exchange (other.display, nullptr)) {}
 - 
 -         ScopedGLXObject& operator= (ScopedGLXObject&& other) noexcept
 -         {
 -             ScopedGLXObject { std::move (other) }.swap (*this);
 -             return *this;
 -         }
 - 
 -         ~ScopedGLXObject() noexcept
 -         {
 -             if (object != Type{})
 -                 Traits::destroy (display, object);
 -         }
 - 
 -         Type get() const { return object; }
 - 
 -         void reset() noexcept
 -         {
 -             *this = ScopedGLXObject();
 -         }
 - 
 -         void swap (ScopedGLXObject& other) noexcept
 -         {
 -             std::swap (other.object, object);
 -             std::swap (other.display, display);
 -         }
 - 
 -         bool operator== (const ScopedGLXObject& other) const
 -         {
 -             const auto tie = [] (const auto& x) { return std::tie (x.object, x.display); };
 -             return tie (*this) == tie (other);
 -         }
 - 
 -         bool operator!= (const ScopedGLXObject& other) const
 -         {
 -             return ! operator== (other);
 -         }
 - 
 -     private:
 -         Type object{};
 -         ::Display* display{};
 -     };
 - 
 -     struct TraitsGLXContext
 -     {
 -         using Type = GLXContext;
 - 
 -         static void destroy (::Display* display, Type t)
 -         {
 -             glXDestroyContext (display, t);
 -         }
 -     };
 - 
 -     struct TraitsGLXWindow
 -     {
 -         using Type = GLXWindow;
 - 
 -         static void destroy (::Display* display, Type t)
 -         {
 -             glXDestroyWindow (display, t);
 -         }
 -     };
 - 
 -     using PtrGLXContext = ScopedGLXObject<TraitsGLXContext>;
 -     using PtrGLXWindow = ScopedGLXObject<TraitsGLXWindow>;
 - 
 - public:
 -     NativeContext (Component& comp,
 -                    const OpenGLPixelFormat& cPixelFormat,
 -                    void* shareContext,
 -                    bool useMultisamplingIn,
 -                    OpenGLVersion)
 -         : component (comp), contextToShareWith (shareContext), dummy (*this)
 -     {
 -         display = XWindowSystem::getInstance()->getDisplay();
 - 
 -         XWindowSystemUtilities::ScopedXLock xLock;
 - 
 -         X11Symbols::getInstance()->xSync (display, False);
 - 
 -         const std::vector<GLint> optionalAttribs
 -         {
 -             GLX_SAMPLE_BUFFERS, useMultisamplingIn ? 1 : 0,
 -             GLX_SAMPLES,        cPixelFormat.multisamplingLevel
 -         };
 - 
 -         if (! tryChooseVisual (cPixelFormat, optionalAttribs) && ! tryChooseVisual (cPixelFormat, {}))
 -             return;
 - 
 -         auto* peer = component.getPeer();
 -         jassert (peer != nullptr);
 - 
 -         auto windowH = (Window) peer->getNativeHandle();
 -         auto visual = glXGetVisualFromFBConfig (display, *bestConfig);
 -         auto colourMap = X11Symbols::getInstance()->xCreateColormap (display, windowH, visual->visual, AllocNone);
 - 
 -         XSetWindowAttributes swa;
 -         swa.colormap = colourMap;
 -         swa.border_pixel = 0;
 -         swa.event_mask = embeddedWindowEventMask;
 - 
 -         auto glBounds = component.getTopLevelComponent()->getLocalArea (&component, component.getLocalBounds());
 - 
 -         glBounds = Desktop::getInstance().getDisplays().logicalToPhysical (glBounds);
 - 
 -         embeddedWindow = X11Symbols::getInstance()->xCreateWindow (display, windowH,
 -                                                                    glBounds.getX(), glBounds.getY(),
 -                                                                    (unsigned int) jmax (1, glBounds.getWidth()),
 -                                                                    (unsigned int) jmax (1, glBounds.getHeight()),
 -                                                                    0, visual->depth,
 -                                                                    InputOutput,
 -                                                                    visual->visual,
 -                                                                    CWBorderPixel | CWColormap | CWEventMask,
 -                                                                    &swa);
 - 
 -         peerListener.emplace (component, embeddedWindow);
 - 
 -         X11Symbols::getInstance()->xMapWindow (display, embeddedWindow);
 -         X11Symbols::getInstance()->xFreeColormap (display, colourMap);
 - 
 -         X11Symbols::getInstance()->xSync (display, False);
 - 
 -         juce_LinuxAddRepaintListener (peer, &dummy);
 -     }
 - 
 -     ~NativeContext()
 -     {
 -         if (auto* peer = component.getPeer())
 -         {
 -             juce_LinuxRemoveRepaintListener (peer, &dummy);
 - 
 -             if (embeddedWindow != 0)
 -             {
 -                 XWindowSystemUtilities::ScopedXLock xLock;
 - 
 -                 X11Symbols::getInstance()->xUnmapWindow (display, embeddedWindow);
 -                 X11Symbols::getInstance()->xDestroyWindow (display, embeddedWindow);
 -                 X11Symbols::getInstance()->xSync (display, False);
 - 
 -                 XEvent event;
 -                 while (X11Symbols::getInstance()->xCheckWindowEvent (display,
 -                                                                      embeddedWindow,
 -                                                                      embeddedWindowEventMask,
 -                                                                      &event) == True)
 -                 {
 -                 }
 -             }
 -         }
 -     }
 - 
 -     InitResult initialiseOnRenderThread (OpenGLContext& c)
 -     {
 -         XWindowSystemUtilities::ScopedXLock xLock;
 - 
 -         const auto components = [&]() -> Optional<Version>
 -         {
 -             switch (c.versionRequired)
 -             {
 -                 case OpenGLVersion::openGL3_2: return Version { 3, 2 };
 -                 case OpenGLVersion::openGL4_1: return Version { 4, 1 };
 -                 case OpenGLVersion::openGL4_3: return Version { 4, 3 };
 - 
 -                 case OpenGLVersion::defaultGLVersion: break;
 -             }
 - 
 -             return {};
 -         }();
 - 
 -         if (components.hasValue())
 -         {
 -             using GLXCreateContextAttribsARB = GLXContext (*) (Display*, GLXFBConfig, GLXContext, Bool, const int*);
 - 
 -             if (const auto glXCreateContextAttribsARB = (GLXCreateContextAttribsARB) OpenGLHelpers::getExtensionFunction ("glXCreateContextAttribsARB"))
 -             {
 -                #if JUCE_DEBUG
 -                 constexpr auto contextFlags = GLX_CONTEXT_DEBUG_BIT_ARB;
 -                #else
 -                 constexpr auto contextFlags = 0;
 -                #endif
 - 
 -                 const int attribs[]
 -                 {
 -                     GLX_CONTEXT_MAJOR_VERSION_ARB, components->major,
 -                     GLX_CONTEXT_MINOR_VERSION_ARB, components->minor,
 -                     GLX_CONTEXT_PROFILE_MASK_ARB,  GLX_CONTEXT_CORE_PROFILE_BIT_ARB,
 -                     GLX_CONTEXT_FLAGS_ARB,         contextFlags,
 -                     None
 -                 };
 - 
 -                 renderContext = PtrGLXContext { glXCreateContextAttribsARB (display, *bestConfig, (GLXContext) contextToShareWith, GL_TRUE, attribs),
 -                                                 display };
 -             }
 -         }
 - 
 -         if (renderContext == PtrGLXContext{})
 -             renderContext = PtrGLXContext { glXCreateNewContext (display, *bestConfig, GLX_RGBA_TYPE, (GLXContext) contextToShareWith, GL_TRUE),
 -                                             display };
 - 
 -         if (renderContext == PtrGLXContext{})
 -             return InitResult::fatal;
 - 
 -         glxWindow = PtrGLXWindow { glXCreateWindow (display, *bestConfig, embeddedWindow, nullptr),
 -                                    display };
 -         c.makeActive();
 -         context = &c;
 -         return InitResult::success;
 -     }
 - 
 -     void shutdownOnRenderThread()
 -     {
 -         XWindowSystemUtilities::ScopedXLock xLock;
 -         context = nullptr;
 -         deactivateCurrentContext();
 -         renderContext.reset();
 -         glxWindow.reset();
 -     }
 - 
 -     bool makeActive() const noexcept
 -     {
 -         XWindowSystemUtilities::ScopedXLock xLock;
 -         return renderContext != PtrGLXContext{}
 -                  && glXMakeContextCurrent (display, glxWindow.get(), glxWindow.get(), renderContext.get());
 -     }
 - 
 -     bool isActive() const noexcept
 -     {
 -         XWindowSystemUtilities::ScopedXLock xLock;
 -         return glXGetCurrentContext() == renderContext.get() && renderContext != PtrGLXContext{};
 -     }
 - 
 -     static void deactivateCurrentContext()
 -     {
 -         if (auto* display = XWindowSystem::getInstance()->getDisplay())
 -         {
 -             XWindowSystemUtilities::ScopedXLock xLock;
 -             glXMakeCurrent (display, None, nullptr);
 -         }
 -     }
 - 
 -     void swapBuffers()
 -     {
 -         glXSwapBuffers (display, glxWindow.get());
 -     }
 - 
 -     void updateWindowPosition (Rectangle<int> newBounds)
 -     {
 -         bounds = newBounds;
 -         auto physicalBounds = Desktop::getInstance().getDisplays().logicalToPhysical (bounds);
 - 
 -         XWindowSystemUtilities::ScopedXLock xLock;
 -         X11Symbols::getInstance()->xMoveResizeWindow (display, embeddedWindow,
 -                                                       physicalBounds.getX(), physicalBounds.getY(),
 -                                                       (unsigned int) jmax (1, physicalBounds.getWidth()),
 -                                                       (unsigned int) jmax (1, physicalBounds.getHeight()));
 -     }
 - 
 -     bool setSwapInterval (int numFramesPerSwap)
 -     {
 -         if (numFramesPerSwap == swapFrames)
 -             return true;
 - 
 -         if (auto GLXSwapIntervalEXT
 -               = (PFNGLXSWAPINTERVALEXTPROC) OpenGLHelpers::getExtensionFunction ("glXSwapIntervalEXT"))
 -         {
 -             XWindowSystemUtilities::ScopedXLock xLock;
 -             swapFrames = numFramesPerSwap;
 -             GLXSwapIntervalEXT (display, glxWindow.get(), numFramesPerSwap);
 -             return true;
 -         }
 - 
 -         return false;
 -     }
 - 
 -     int getSwapInterval() const                 { return swapFrames; }
 -     bool createdOk() const noexcept             { return true; }
 -     void* getRawContext() const noexcept        { return renderContext.get(); }
 -     GLuint getFrameBufferID() const noexcept    { return 0; }
 - 
 -     void triggerRepaint()
 -     {
 -         if (context != nullptr)
 -             context->triggerRepaint();
 -     }
 - 
 -     struct Locker
 -     {
 -         explicit Locker (NativeContext& ctx) : lock (ctx.mutex) {}
 -         const ScopedLock lock;
 -     };
 - 
 - private:
 -     bool tryChooseVisual (const OpenGLPixelFormat& format, const std::vector<GLint>& optionalAttribs)
 -     {
 -         std::vector<GLint> allAttribs
 -         {
 -             GLX_RENDER_TYPE,      GLX_RGBA_BIT,
 -             GLX_DOUBLEBUFFER,     True,
 -             GLX_RED_SIZE,         format.redBits,
 -             GLX_GREEN_SIZE,       format.greenBits,
 -             GLX_BLUE_SIZE,        format.blueBits,
 -             GLX_ALPHA_SIZE,       format.alphaBits,
 -             GLX_DEPTH_SIZE,       format.depthBufferBits,
 -             GLX_STENCIL_SIZE,     format.stencilBufferBits,
 -             GLX_ACCUM_RED_SIZE,   format.accumulationBufferRedBits,
 -             GLX_ACCUM_GREEN_SIZE, format.accumulationBufferGreenBits,
 -             GLX_ACCUM_BLUE_SIZE,  format.accumulationBufferBlueBits,
 -             GLX_ACCUM_ALPHA_SIZE, format.accumulationBufferAlphaBits
 -         };
 - 
 -         allAttribs.insert (allAttribs.end(), optionalAttribs.begin(), optionalAttribs.end());
 - 
 -         allAttribs.push_back (None);
 - 
 -         int nElements = 0;
 -         bestConfig = makeXFreePtr (glXChooseFBConfig (display, X11Symbols::getInstance()->xDefaultScreen (display), allAttribs.data(), &nElements));
 - 
 -         return nElements != 0 && bestConfig != nullptr;
 -     }
 - 
 -     static constexpr int embeddedWindowEventMask = ExposureMask | StructureNotifyMask;
 - 
 -     CriticalSection mutex;
 -     Component& component;
 -     PtrGLXContext renderContext;
 -     PtrGLXWindow glxWindow;
 -     Window embeddedWindow = {};
 - 
 -     std::optional<PeerListener> peerListener;
 - 
 -     int swapFrames = 0;
 -     Rectangle<int> bounds;
 -     std::unique_ptr<GLXFBConfig, XFreeDeleter> bestConfig;
 -     void* contextToShareWith;
 - 
 -     OpenGLContext* context = nullptr;
 -     DummyComponent dummy;
 - 
 -     ::Display* display = nullptr;
 - 
 -     JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (NativeContext)
 - };
 - 
 - //==============================================================================
 - bool OpenGLHelpers::isContextActive()
 - {
 -     XWindowSystemUtilities::ScopedXLock xLock;
 -     return glXGetCurrentContext() != nullptr;
 - }
 - 
 - } // namespace juce
 
 
  |