|  | /*
 * DISTRHO Plugin Toolkit (DPT)
 * Copyright (C) 2012-2013 Filipe Coelho <falktx@falktx.com>
 *
 * Permission to use, copy, modify, and/or distribute this software for any purpose with
 * or without fee is hereby granted, provided that the above copyright notice and this
 * permission notice appear in all copies.
 *
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
 * TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
 * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
 * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
 * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 */
#include "../App.hpp"
#include "../Widget.hpp"
#include "../Window.hpp"
#include <cassert>
#include <list>
#include "pugl/pugl.h"
#if DGL_OS_WINDOWS
# include "pugl/pugl_win.cpp"
#elif DGL_OS_MAC
extern "C" {
# include "pugl/pugl_osx_extended.h"
}
#elif DGL_OS_LINUX
extern "C" {
# include "pugl/pugl_x11.c"
}
#else
# error Unsupported platform
#endif
#define FOR_EACH_WIDGET(it) \
  for (std::list<Widget*>::iterator it = fWidgets.begin(); it != fWidgets.end(); ++it)
#define FOR_EACH_WIDGET_INV(rit) \
  for (std::list<Widget*>::reverse_iterator rit = fWidgets.rbegin(); rit != fWidgets.rend(); ++rit)
START_NAMESPACE_DGL
Window* dgl_lastUiParent = nullptr;
// -----------------------------------------------------------------------
// Window Private
class Window::PrivateData
{
public:
    PrivateData(App& app, Window* const self)
        : fApp(app),
          fSelf(self),
          fView(puglCreate(0, "Window", 100, 100, true, false)),
          fFirstInit(true),
          fVisible(false),
          fResizable(true),
#if DGL_OS_WINDOWS
          hwnd(0)
#elif DGL_OS_LINUX
          xDisplay(nullptr),
          xWindow(0)
#else
          _dummy('\0')
#endif
    {
        init();
    }
    PrivateData(App& app, Window* const self, Window& parent)
        : fApp(app),
          fSelf(self),
          fView(puglCreate(0, "Window", 100, 100, true, false)),
          fFirstInit(true),
          fVisible(false),
          fResizable(true),
          fModal(parent.pData),
#if DGL_OS_WINDOWS
          hwnd(0)
#elif DGL_OS_LINUX
          xDisplay(nullptr),
          xWindow(0)
#else
          _dummy('\0')
#endif
    {
        init();
#if DGL_OS_LINUX
        PuglInternals* const parentImpl = parent.pData->fView->impl;
        XSetTransientForHint(xDisplay, xWindow, parentImpl->win);
        XFlush(xDisplay);
#endif
    }
    PrivateData(App& app, Window* const self, const intptr_t parentId)
        : fApp(app),
          fSelf(self),
          fView(puglCreate(parentId, "Window", 100, 100, true, true)),
          fFirstInit(true),
          fVisible(true),
          fResizable(false),
#if DGL_OS_WINDOWS
          hwnd(0)
#elif DGL_OS_LINUX
          xDisplay(nullptr),
          xWindow(0)
#else
          _dummy('\0')
#endif
    {
        init();
        // starts visible
        fApp.oneShown();
        fFirstInit = false;
    }
    void init()
    {
        if (fView == nullptr)
            return;
        dgl_lastUiParent = fSelf;
        puglSetHandle(fView, this);
        puglSetDisplayFunc(fView, onDisplayCallback);
        puglSetKeyboardFunc(fView, onKeyboardCallback);
        puglSetMotionFunc(fView, onMotionCallback);
        puglSetMouseFunc(fView, onMouseCallback);
        puglSetScrollFunc(fView, onScrollCallback);
        puglSetSpecialFunc(fView, onSpecialCallback);
        puglSetReshapeFunc(fView, onReshapeCallback);
        puglSetCloseFunc(fView, onCloseCallback);
#if DGL_OS_WINDOWS
        PuglInternals* impl = fView->impl;
        hwnd = impl->hwnd;
#elif DGL_OS_LINUX
        PuglInternals* impl = fView->impl;
        xDisplay = impl->display;
        xWindow  = impl->win;
#endif
        fApp.addWindow(fSelf);
    }
    ~PrivateData()
    {
        //fOnModal = false;
        fWidgets.clear();
        if (fView != nullptr)
        {
            fApp.removeWindow(fSelf);
            puglDestroy(fView);
        }
    }
    // -------------------------------------------------------------------
    void close()
    {
        setVisible(false);
        if (! fFirstInit)
        {
            fApp.oneHidden();
            fFirstInit = true;
        }
    }
    void exec(const bool lockWait)
    {
        exec_init();
        if (lockWait)
        {
            while (fVisible && fModal.enabled)
            {
                // idle()
                puglProcessEvents(fView);
                if (fModal.parent != nullptr)
                    fModal.parent->idle();
                msleep(10);
            }
            exec_fini();
        }
        else
        {
            idle();
        }
    }
    // -------------------------------------------------------------------
    void focus()
    {
#if DGL_OS_WINDOWS
        SetForegroundWindow(hwnd);
        SetActiveWindow(hwnd);
        SetFocus(hwnd);
#elif DGL_OS_MAC
        puglImplFocus(fView);
#elif DGL_OS_LINUX
        XRaiseWindow(xDisplay, xWindow);
        XSetInputFocus(xDisplay, xWindow, RevertToPointerRoot, CurrentTime);
        XFlush(xDisplay);
#endif
    }
    void idle()
    {
        puglProcessEvents(fView);
        if (fVisible && fModal.enabled && fModal.parent != nullptr)
            fModal.parent->idle();
    }
    void repaint()
    {
        puglPostRedisplay(fView);
    }
    // -------------------------------------------------------------------
    bool isVisible() const noexcept
    {
        return fVisible;
    }
    void setVisible(const bool yesNo)
    {
        if (fVisible == yesNo)
            return;
        fVisible = yesNo;
#ifndef DGL_OS_MAC
        if (yesNo && fFirstInit)
            setSize(fView->width, fView->height, true);
#endif
#if DGL_OS_WINDOWS
        if (yesNo)
        {
            ShowWindow(hwnd, WS_VISIBLE);
            if (! fFirstInit)
                ShowWindow(hwnd, SW_RESTORE);
        }
        else
        {
            ShowWindow(hwnd, SW_HIDE);
        }
        UpdateWindow(hwnd);
#elif DGL_OS_MAC
        puglImplSetVisible(fView, yesNo);
#elif DGL_OS_LINUX
        if (yesNo)
            XMapRaised(xDisplay, xWindow);
        else
            XUnmapWindow(xDisplay, xWindow);
        XFlush(xDisplay);
#endif
        if (yesNo)
        {
            if (fFirstInit)
            {
                fApp.oneShown();
                fFirstInit = false;
            }
        }
        else if (fModal.enabled)
            exec_fini();
    }
    // -------------------------------------------------------------------
    bool isResizable() const noexcept
    {
        return fResizable;
    }
    void setResizable(const bool yesNo)
    {
        if (fResizable == yesNo)
            return;
        fResizable = yesNo;
#ifndef DGL_OS_MAC
        setSize(fView->width, fView->height, true);
#endif
    }
    // -------------------------------------------------------------------
#ifndef DGL_OS_MAC
    int getWidth() const noexcept
    {
        return fView->width;
    }
    int getHeight() const noexcept
    {
        return fView->height;
    }
    Size<int> getSize() const noexcept
    {
        return Size<int>(fView->width, fView->height);
    }
#endif
    void setSize(unsigned int width, unsigned int height, const bool forced = false)
    {
        if (width == 0)
            width = 1;
        if (height == 0)
            height = 1;
#ifndef DGL_OS_MAC
        fView->width  = width;
        fView->height = height;
        if (fView->width == (int)width && fView->height == (int)height && ! forced)
           return;
#endif
#if DGL_OS_WINDOWS
        int winFlags = WS_POPUPWINDOW | WS_CAPTION;
        if (fResizable)
            winFlags |= WS_SIZEBOX;
        RECT wr = { 0, 0, (long)width, (long)height };
        AdjustWindowRectEx(&wr, winFlags, FALSE, WS_EX_TOPMOST);
        SetWindowPos(hwnd, 0, 0, 0, wr.right-wr.left, wr.bottom-wr.top, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOOWNERZORDER|SWP_NOZORDER);
        UpdateWindow(hwnd);
#elif DGL_OS_MAC
        puglImplSetSize(fView, width, height);
#elif DGL_OS_LINUX
        XResizeWindow(xDisplay, xWindow, width, height);
        if (! fResizable)
        {
            XSizeHints sizeHints;
            memset(&sizeHints, 0, sizeof(sizeHints));
            sizeHints.flags      = PMinSize|PMaxSize;
            sizeHints.min_width  = width;
            sizeHints.min_height = height;
            sizeHints.max_width  = width;
            sizeHints.max_height = height;
            XSetNormalHints(xDisplay, xWindow, &sizeHints);
        }
        XFlush(xDisplay);
#endif
        repaint();
    }
    // -------------------------------------------------------------------
    void setTitle(const char* const title)
    {
#if DGL_OS_WINDOWS
        SetWindowTextA(hwnd, title);
#elif DGL_OS_MAC
        puglImplSetTitle(fView, title);
#elif DGL_OS_LINUX
        XStoreName(xDisplay, xWindow, title);
        XFlush(xDisplay);
#endif
    }
    App& getApp() const noexcept
    {
        return fApp;
    }
    int getModifiers() const
    {
        return puglGetModifiers(fView);
    }
    uint32_t getEventTimestamp() const
    {
        return puglGetEventTimestamp(fView);
    }
    intptr_t getWindowId() const
    {
        return puglGetNativeWindow(fView);
    }
    // -------------------------------------------------------------------
    void addWidget(Widget* const widget)
    {
        fWidgets.push_back(widget);
    }
    void removeWidget(Widget* const widget)
    {
        fWidgets.remove(widget);
    }
    // -------------------------------------------------------------------
    void exec_init()
    {
        fModal.enabled = true;
        assert(fModal.parent != nullptr);
        if (fModal.parent == nullptr)
            return setVisible(true);
        fModal.parent->fModal.childFocus = this;
#if DGL_OS_WINDOWS
        // Center this window
        PuglInternals* const parentImpl = fParent->fView->impl;
        RECT curRect;
        RECT parentRect;
        GetWindowRect(hwnd, &curRect);
        GetWindowRect(parentImpl->hwnd, &parentRect);
        int x = parentRect.left+(parentRect.right-curRect.right)/2;
        int y = parentRect.top +(parentRect.bottom-curRect.bottom)/2;
        SetWindowPos(hwnd, 0, x, y, 0, 0, SWP_NOACTIVATE|SWP_NOOWNERZORDER|SWP_NOSIZE|SWP_NOZORDER);
        UpdateWindow(hwnd);
#endif
        fModal.parent->setVisible(true);
        setVisible(true);
    }
    void exec_fini()
    {
        fModal.enabled = false;
        if (fModal.parent != nullptr)
            fModal.parent->fModal.childFocus = nullptr;
    }
    // -------------------------------------------------------------------
protected:
    void onDisplay()
    {
        glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
        FOR_EACH_WIDGET(it)
        {
            Widget* const widget(*it);
            if (widget->isVisible())
                widget->onDisplay();
        }
    }
    void onKeyboard(const bool press, const uint32_t key)
    {
        if (fModal.childFocus != nullptr)
            return fModal.childFocus->focus();
        FOR_EACH_WIDGET_INV(rit)
        {
            Widget* const widget(*rit);
            if (widget->isVisible() && widget->onKeyboard(press, key))
                break;
        }
    }
    void onMouse(const int button, const bool press, const int x, const int y)
    {
        if (fModal.childFocus != nullptr)
            return fModal.childFocus->focus();
        FOR_EACH_WIDGET_INV(rit)
        {
            Widget* const widget(*rit);
            if (widget->isVisible() && widget->onMouse(button, press, x, y))
                break;
        }
    }
    void onMotion(const int x, const int y)
    {
        if (fModal.childFocus != nullptr)
            return;
        FOR_EACH_WIDGET_INV(rit)
        {
            Widget* const widget(*rit);
            if (widget->isVisible() && widget->onMotion(x, y))
                break;
        }
    }
    void onScroll(const float dx, const float dy)
    {
        if (fModal.childFocus != nullptr)
            return;
        FOR_EACH_WIDGET_INV(rit)
        {
            Widget* const widget(*rit);
            if (widget->isVisible() && widget->onScroll(dx, dy))
                break;
        }
    }
    void onSpecial(const bool press, const Key key)
    {
        if (fModal.childFocus != nullptr)
            return;
        FOR_EACH_WIDGET_INV(rit)
        {
            Widget* const widget(*rit);
            if (widget->isVisible() && widget->onSpecial(press, key))
                break;
        }
    }
    void onReshape(const int width, const int height)
    {
        printf("resized: %i:%i\n", width, height);
        FOR_EACH_WIDGET(it)
        {
            Widget* const widget(*it);
            widget->onReshape(width, height);
        }
    }
    void onClose()
    {
        fModal.enabled = false;
        if (fModal.childFocus != nullptr)
            fModal.childFocus->onClose();
        FOR_EACH_WIDGET(it)
        {
            Widget* const widget(*it);
            widget->onClose();
        }
        close();
    }
    // -------------------------------------------------------------------
private:
    App&            fApp;
    Window*   const fSelf;
    PuglView* const fView;
    bool fFirstInit;
    bool fVisible;
    bool fResizable;
    std::list<Widget*> fWidgets;
    struct Modal {
        bool enabled;
        PrivateData* parent;
        PrivateData* childFocus;
        Modal()
            : enabled(false),
              parent(nullptr),
              childFocus(nullptr) {}
        Modal(PrivateData* const p)
            : enabled(false),
              parent(p),
              childFocus(nullptr) {}
        ~Modal()
        {
            assert(! enabled);
            assert(childFocus == nullptr);
        }
    } fModal;
#if DGL_OS_WINDOWS
    HWND     hwnd;
#elif DGL_OS_LINUX
    Display* xDisplay;
    ::Window xWindow;
#else
    char      _dummy;
#endif
    // -------------------------------------------------------------------
    // Callbacks
    #define handlePtr ((PrivateData*)puglGetHandle(view))
    static void onDisplayCallback(PuglView* view)
    {
        handlePtr->onDisplay();
    }
    static void onKeyboardCallback(PuglView* view, bool press, uint32_t key)
    {
        handlePtr->onKeyboard(press, key);
    }
    static void onMouseCallback(PuglView* view, int button, bool press, int x, int y)
    {
        handlePtr->onMouse(button, press, x, y);
    }
    static void onMotionCallback(PuglView* view, int x, int y)
    {
        handlePtr->onMotion(x, y);
    }
    static void onScrollCallback(PuglView* view, float dx, float dy)
    {
        handlePtr->onScroll(dx, dy);
    }
    static void onSpecialCallback(PuglView* view, bool press, PuglKey key)
    {
        handlePtr->onSpecial(press, static_cast<Key>(key));
    }
    static void onReshapeCallback(PuglView* view, int width, int height)
    {
        handlePtr->onReshape(width, height);
    }
    static void onCloseCallback(PuglView* view)
    {
        handlePtr->onClose();
    }
    #undef handlePtr
};
// -----------------------------------------------------------------------
// Window
Window::Window(App& app)
    : pData(new PrivateData(app, this))
{
}
Window::Window(App& app, Window& parent)
    : pData(new PrivateData(app, this, parent))
{
}
Window::Window(App& app, intptr_t parentId)
    : pData(new PrivateData(app, this, parentId))
{
}
Window::~Window()
{
    delete pData;
}
void Window::show()
{
    pData->setVisible(true);
}
void Window::hide()
{
    pData->setVisible(false);
}
void Window::close()
{
    pData->close();
}
void Window::exec(bool lockWait)
{
    pData->exec(lockWait);
}
void Window::focus()
{
    pData->focus();
}
void Window::idle()
{
    pData->idle();
}
void Window::repaint()
{
    pData->repaint();
}
bool Window::isVisible() const noexcept
{
    return pData->isVisible();
}
void Window::setVisible(bool yesNo)
{
    pData->setVisible(yesNo);
}
bool Window::isResizable() const noexcept
{
    return pData->isResizable();
}
void Window::setResizable(bool yesNo)
{
    pData->setResizable(yesNo);
}
#ifndef DGL_OS_MAC
int Window::getWidth() const noexcept
{
    return pData->getWidth();
}
int Window::getHeight() const noexcept
{
    return pData->getHeight();
}
Size<int> Window::getSize() const noexcept
{
    return pData->getSize();
}
#endif
void Window::setSize(unsigned int width, unsigned int height)
{
    pData->setSize(width, height);
}
void Window::setTitle(const char* title)
{
    pData->setTitle(title);
}
App& Window::getApp() const noexcept
{
    return pData->getApp();
}
int Window::getModifiers() const
{
    return pData->getModifiers();
}
uint32_t Window::getEventTimestamp() const
{
    return pData->getEventTimestamp();
}
intptr_t Window::getWindowId() const
{
    return pData->getWindowId();
}
void Window::addWidget(Widget* const widget)
{
    pData->addWidget(widget);
}
void Window::removeWidget(Widget* const widget)
{
    pData->removeWidget(widget);
}
// -----------------------------------------------------------------------
END_NAMESPACE_DGL
 |