/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /*************************************************************************** * window.cc * * Sun Oct 9 13:11:53 CEST 2011 * Copyright 2011 Bent Bisballe Nyeng * deva@aasimon.org ****************************************************************************/ /* * This file is part of DrumGizmo. * * DrumGizmo is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * DrumGizmo is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with DrumGizmo; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ #include "window.h" #include <cstring> #include "painter.h" #ifndef UI_PUGL #ifdef UI_X11 #include "nativewindow_x11.h" #endif // UI_X11 #ifdef UI_WIN32 #include "nativewindow_win32.h" #endif // UI_WIN32 #ifdef UI_COCOA #include "nativewindow_cocoa.h" #endif // UI_COCOA #else #include "nativewindow_pugl.h" #endif // !UI_PUGL namespace GUI { Window::Window(void* native_window) : Widget(nullptr) , wpixbuf(1, 1) { // Make sure we have a valid size when initialising the NativeWindow _width = wpixbuf.width; _height = wpixbuf.height; #ifndef UI_PUGL #ifdef UI_X11 native = new NativeWindowX11(native_window, *this); #endif // UI_X11 #ifdef UI_WIN32 native = new NativeWindowWin32(native_window, *this); #endif // UI_WIN32 #ifdef UI_COCOA native = new NativeWindowCocoa(native_window, *this); #endif // UI_COCOA #else // Use pugl native = new NativeWindowPugl(native_window, *this); #endif // !UI_PUGL eventhandler = new EventHandler(*native, *this); setVisible(true); // The root widget is always visible. } Window::~Window() { delete native; delete eventhandler; } void Window::setFixedSize(int w, int h) { native->setFixedSize(w, h); } void Window::setAlwaysOnTop(bool always_on_top) { native->setAlwaysOnTop(always_on_top); } void Window::setCaption(const std::string& caption) { native->setCaption(caption); } //! This overload the resize method on Widget and simply requests a window resize //! on the windowmanager/OS. The resized() method is called by the event handler //! once the window has been resized. void Window::resize(std::size_t width, std::size_t height) { native->resize(width, height); } //! This overload the move method on Widget and simply requests a window move //! on the windowmanager/OS. The moved() method is called by the event handler //! once the window has been moved. void Window::move(int x, int y) { native->move(x, y); } void Window::show() { Widget::show(); redraw(); native->show(); } void Window::hide() { native->hide(); Widget::hide(); } Window* Window::window() { return this; } Size Window::getNativeSize() { auto sz = native->getSize(); return {sz.first, sz.second}; } ImageCache& Window::getImageCache() { return image_cache; } EventHandler* Window::eventHandler() { return eventhandler; } Widget* Window::keyboardFocus() { return _keyboardFocus; } void Window::setKeyboardFocus(Widget* widget) { auto oldFocusWidget = _keyboardFocus; _keyboardFocus = widget; if(oldFocusWidget) { oldFocusWidget->redraw(); } if(_keyboardFocus) { _keyboardFocus->redraw(); } } Widget* Window::buttonDownFocus() { return _buttonDownFocus; } void Window::setButtonDownFocus(Widget* widget) { _buttonDownFocus = widget; native->grabMouse(widget != nullptr); } Widget* Window::mouseFocus() { return _mouseFocus; } void Window::setMouseFocus(Widget* widget) { _mouseFocus = widget; } void Window::needsRedraw() { needs_redraw = true; } void* Window::getNativeWindowHandle() const { return native->getNativeWindowHandle(); } Point Window::translateToScreen(const Point& point) { return native->translateToScreen(point); } std::size_t Window::translateToWindowX() { return 0; } std::size_t Window::translateToWindowY() { return 0; } //! Called by event handler when an windowmanager/OS window resize event has //! been received. Do not call this directly. void Window::resized(std::size_t width, std::size_t height) { auto size = native->getSize(); if((wpixbuf.width != size.first) || (wpixbuf.height != size.second)) { wpixbuf.realloc(size.first, size.second); Widget::resize(size.first, size.second); } updateBuffer(); } //! Called by event handler when an windowmanager/OS window move event has //! been received. Do not call this directly. void Window::moved(int x, int y) { // Make sure widget coordinates are updated. Widget::move(x, y); } bool Window::updateBuffer() { if(!native) { return false; } if(!needs_redraw) { // Nothing changed, don't update anything. return false; } bool has_dirty_rect{false}; Rect dirty_rect; auto pixel_buffers = getPixelBuffers(); for(auto& pixel_buffer : pixel_buffers) { if(pixel_buffer->dirty) { auto x1 = (std::size_t)pixel_buffer->x; auto x2 = (std::size_t)(pixel_buffer->x + pixel_buffer->width); auto y1 = (std::size_t)pixel_buffer->y; auto y2 = (std::size_t)(pixel_buffer->y + pixel_buffer->height); pixel_buffer->dirty = false; if(!has_dirty_rect) { // Insert this area: dirty_rect = {x1, y1, x2, y2}; has_dirty_rect = true; } else { // Expand existing area: auto x1_0 = dirty_rect.x1; auto y1_0 = dirty_rect.y1; auto x2_0 = dirty_rect.x2; auto y2_0 = dirty_rect.y2; dirty_rect = { (x1_0 < x1) ? x1_0 : x1, (y1_0 < y1) ? y1_0 : y1, (x2_0 > x2) ? x2_0 : x2, (y2_0 > y2) ? y2_0 : y2 }; } } if(pixel_buffer->has_last) { auto x1 = (std::size_t)pixel_buffer->last_x; auto x2 = (std::size_t)(pixel_buffer->last_x + pixel_buffer->last_width); auto y1 = (std::size_t)pixel_buffer->last_y; auto y2 = (std::size_t)(pixel_buffer->last_y + pixel_buffer->last_height); pixel_buffer->has_last = false; if(!has_dirty_rect) { // Insert this area: dirty_rect = {x1, y1, x2, y2}; has_dirty_rect = true; } else { // Expand existing area: auto x1_0 = dirty_rect.x1; auto y1_0 = dirty_rect.y1; auto x2_0 = dirty_rect.x2; auto y2_0 = dirty_rect.y2; dirty_rect = { (x1_0 < x1) ? x1_0 : x1, (y1_0 < y1) ? y1_0 : y1, (x2_0 > x2) ? x2_0 : x2, (y2_0 > y2) ? y2_0 : y2 }; } } } if(!has_dirty_rect) { return false; } for(auto& pixel_buffer : pixel_buffers) { if(!pixel_buffer->visible) { continue; } int update_width = pixel_buffer->width; int update_height = pixel_buffer->height; // Skip buffer if not inside window. if(((int)wpixbuf.width < pixel_buffer->x) || ((int)wpixbuf.height < pixel_buffer->y)) { continue; } if(update_width > ((int)wpixbuf.width - pixel_buffer->x)) { update_width = ((int)wpixbuf.width - pixel_buffer->x); } if(update_height > ((int)wpixbuf.height - pixel_buffer->y)) { update_height = ((int)wpixbuf.height - pixel_buffer->y); } std::uint8_t r, g, b, a; auto from_x = (int)dirty_rect.x1 - pixel_buffer->x; from_x = std::max(0, from_x); auto from_y = (int)dirty_rect.y1 - pixel_buffer->y; from_y = std::max(0, from_y); auto to_x = (int)dirty_rect.x2 - pixel_buffer->x; to_x = std::min(to_x, (int)update_width); auto to_y = (int)dirty_rect.y2 - pixel_buffer->y; to_y = std::min(to_y, (int)update_height); for(int y = from_y; y < to_y; y++) { for(int x = from_x; x < to_x; x++) { pixel_buffer->pixel(x, y, &r, &g, &b, &a); wpixbuf.setPixel(x + pixel_buffer->x, y + pixel_buffer->y, r, g, b, a); } } } dirty_rect.x2 = std::min(wpixbuf.width, dirty_rect.x2); dirty_rect.y2 = std::min(wpixbuf.height, dirty_rect.y2); // Make sure we don't try to paint a rect backwards. if(dirty_rect.x1 > dirty_rect.x2) { std::swap(dirty_rect.x1, dirty_rect.x2); } if(dirty_rect.y1 > dirty_rect.y2) { std::swap(dirty_rect.y1, dirty_rect.y2); } native->redraw(dirty_rect); needs_redraw = false; return true; } } // GUI::