/*********************************************************************** * fwindow.cpp - Intermediate base class for all window objects * * * * This file is part of the Final Cut widget toolkit * * * * Copyright 2015-2019 Markus Gans * * * * The Final Cut 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. * * * * The Final Cut 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 this program. If not, see * * . * ***********************************************************************/ #include #include "final/fapplication.h" #include "final/fevent.h" #include "final/fmenubar.h" #include "final/fstatusbar.h" #include "final/fwindow.h" namespace finalcut { // static attributes FWindow* FWindow::previous_window{nullptr}; //---------------------------------------------------------------------- // class FWindow //---------------------------------------------------------------------- // constructor and destructor //---------------------------------------------------------------------- FWindow::FWindow(FWidget* parent) : FWidget(parent) { setWindowWidget(); FRect geometry (getTermGeometry()); geometry.move(-1, -1); createArea (geometry, getShadow(), getVWin()); addWindow (this); } //---------------------------------------------------------------------- FWindow::~FWindow() // destructor { auto fapp = FApplication::getApplicationObject(); if ( previous_window == this ) previous_window = nullptr; if ( isAlwaysOnTop() ) deleteFromAlwaysOnTopList (this); // unset the global active window if ( this == FWindow::getActiveWindow() ) unsetActiveWindow(); delWindow (this); if ( ! fapp->isQuit() ) { const auto& t_geometry = getTermGeometryWithShadow(); restoreVTerm (t_geometry); } removeArea (getVWin()); } // public methods of FWindow //---------------------------------------------------------------------- FWidget* FWindow::getWindowFocusWidget() const { // returns the focused widget of this window return win_focus_widget; } //---------------------------------------------------------------------- bool FWindow::setWindowWidget (bool enable) { if ( isWindowWidget() == enable ) return true; setFlags().window_widget = enable; if ( enable ) setTermOffset(); else setParentOffset(); return enable; } //---------------------------------------------------------------------- void FWindow::setActiveWindow (FWindow* window) { // activate FWindow object window if ( ! getWindowList() || getWindowList()->empty() ) return; for (auto&& win : *getWindowList()) { if ( win == window ) { if ( ! window->isWindowActive() ) { window->activateWindow(); FEvent ev(fc::WindowActive_Event); FApplication::sendEvent(window, &ev); } } else { auto w = static_cast(win); if ( w->isWindowActive() ) { w->deactivateWindow(); FEvent ev(fc::WindowInactive_Event); FApplication::sendEvent(win, &ev); } } } } //---------------------------------------------------------------------- void FWindow::setWindowFocusWidget (FWidget* obj) { // set focus widget of this window win_focus_widget = obj; } //---------------------------------------------------------------------- bool FWindow::activateWindow (bool enable) { // activate/deactivate this window if ( enable ) { FWidget::setActiveWindow (this); setActiveArea (getVWin()); } return (window_active = enable); } //---------------------------------------------------------------------- void FWindow::unsetActiveWindow() { // unset the active FWindow object FWidget::setActiveWindow (nullptr); } //---------------------------------------------------------------------- bool FWindow::setResizeable (bool enable) { return (setFlags().resizeable = enable); } //---------------------------------------------------------------------- bool FWindow::setTransparentShadow (bool enable) { setFlags().shadow = setFlags().trans_shadow = enable; if ( enable ) setShadowSize (FSize(2, 1)); else setShadowSize (FSize(0, 0)); return enable; } //---------------------------------------------------------------------- bool FWindow::setShadow (bool enable) { if ( isMonochron() ) return false; if ( enable ) { setFlags().shadow = true; setFlags().trans_shadow = false; setShadowSize (FSize(1, 1)); } else { setFlags().shadow = false; setFlags().trans_shadow = false; setShadowSize (FSize(0, 0)); } return enable; } //---------------------------------------------------------------------- bool FWindow::setAlwaysOnTop (bool enable) { if ( isAlwaysOnTop() == enable ) return true; setFlags().always_on_top = enable; if ( enable ) { if ( getAlwaysOnTopList() ) { deleteFromAlwaysOnTopList (this); getAlwaysOnTopList()->push_back (this); } } else deleteFromAlwaysOnTopList (this); return enable; } //---------------------------------------------------------------------- bool FWindow::isWindowHidden() const { // returns the window hidden state if ( isVirtualWindow() ) return ! getVWin()->visible; else return false; } //---------------------------------------------------------------------- void FWindow::drawBorder() { if ( isNewFont() ) // Draw a newfont outer frame { FRect r(FPoint(1, 1), getSize()); print() << r.getUpperLeftPos() << fc::NF_border_corner_upper_left // ⎡ << FString(r.getWidth() - 2, fc::NF_border_line_upper) // ¯ << fc::NF_rev_border_corner_upper_right; // ⎤ for (int y = r.getY1() + 1; y < r.getY2(); y++) { print() << FPoint(r.getX1(), y) << fc::NF_border_line_left // border left ⎸ << FPoint(r.getX2(), y) << fc::NF_rev_border_line_right; // border right⎹ } print() << r.getLowerLeftPos() << fc::NF_border_corner_lower_left // ⎣ << FString(r.getWidth() - 2, fc::NF_border_line_bottom) // _ << fc::NF_rev_border_corner_lower_right; // ⎦ } else { FWidget::drawBorder(); } } //---------------------------------------------------------------------- void FWindow::show() { if ( isVirtualWindow() ) getVWin()->visible = true; FWidget::show(); } //---------------------------------------------------------------------- void FWindow::hide() { if ( isVirtualWindow() ) getVWin()->visible = false; FWidget::hide(); } //---------------------------------------------------------------------- void FWindow::setX (int x, bool adjust) { FWidget::setX (x, adjust); if ( isVirtualWindow() ) getVWin()->offset_left = getTermX() - 1; } //---------------------------------------------------------------------- void FWindow::setY (int y, bool adjust) { if ( y < 1 ) y = 1; FWidget::setY (y, adjust); if ( isVirtualWindow() ) getVWin()->offset_top = getTermY() - 1; } //---------------------------------------------------------------------- void FWindow::setPos (const FPoint& p, bool adjust) { FPoint pos(p); if ( pos.getY() < 1 ) pos.setY(1); FWidget::setPos (pos, adjust); if ( isVirtualWindow() ) { auto virtual_win = getVWin(); virtual_win->offset_left = getTermX() - 1; virtual_win->offset_top = getTermY() - 1; } } //---------------------------------------------------------------------- void FWindow::setWidth (std::size_t w, bool adjust) { std::size_t old_width = getWidth(); FWidget::setWidth (w, adjust); if ( isVirtualWindow() && getWidth() != old_width ) { FRect geometry (getTermGeometry()); geometry.move(-1, -1); resizeArea (geometry, getShadow(), getVWin()); } } //---------------------------------------------------------------------- void FWindow::setHeight (std::size_t h, bool adjust) { std::size_t old_height = getHeight(); FWidget::setHeight (h, adjust); if ( isVirtualWindow() && getHeight() != old_height ) { FRect geometry (getTermGeometry()); geometry.move(-1, -1); resizeArea (geometry, getShadow(), getVWin()); } } //---------------------------------------------------------------------- void FWindow::setSize (const FSize& size, bool adjust) { std::size_t old_width = getWidth(); std::size_t old_height = getHeight(); FWidget::setSize (size, adjust); if ( isVirtualWindow() && (getWidth() != old_width || getHeight() != old_height) ) { FRect geometry (getTermGeometry()); geometry.move(-1, -1); resizeArea (geometry, getShadow(), getVWin()); } } //---------------------------------------------------------------------- void FWindow::setGeometry ( const FPoint& p, const FSize& size, bool adjust) { // Sets the geometry of the widget int old_x = getX(); int old_y = getY(); FPoint pos(p); FSize old_size(getSize()); if ( pos.getY() < 1 ) pos.setY(1); FWidget::setGeometry (pos, size, adjust); if ( ! isVirtualWindow() ) return; if ( getSize() != old_size ) { FRect geometry (getTermGeometry()); geometry.move(-1, -1); resizeArea (geometry, getShadow(), getVWin()); } else { if ( getX() != old_x ) getVWin()->offset_left = getTermX() - 1; if ( getY() != old_y ) getVWin()->offset_top = getTermY() - 1; } } //---------------------------------------------------------------------- void FWindow::move (const FPoint& pos) { FWidget::move(pos); if ( isVirtualWindow() ) { auto virtual_win = getVWin(); virtual_win->offset_left = getTermX() - 1; virtual_win->offset_top = getTermY() - 1; } } //---------------------------------------------------------------------- FWindow* FWindow::getWindowWidgetAt (int x, int y) { // returns the window object to the corresponding coordinates if ( getWindowList() && ! getWindowList()->empty() ) { auto iter = getWindowList()->end(); auto begin = getWindowList()->begin(); do { --iter; if ( *iter ) { auto w = static_cast(*iter); if ( ! w->isWindowHidden() && w->getTermGeometry().contains(x, y) ) return w; } } while ( iter != begin ); } return 0; } //---------------------------------------------------------------------- void FWindow::addWindow (FWidget* obj) { // add the window object obj to the window list if ( getWindowList() ) getWindowList()->push_back(obj); processAlwaysOnTop(); } //---------------------------------------------------------------------- void FWindow::delWindow (FWidget* obj) { // delete the window object obj from the window list if ( ! getWindowList() || getWindowList()->empty() ) return; auto iter = getWindowList()->begin(); while ( iter != getWindowList()->end() ) { if ( (*iter) == obj ) { getWindowList()->erase(iter); return; } ++iter; } } //---------------------------------------------------------------------- FWindow* FWindow::getWindowWidget (const FWidget* obj) { // returns the window object to the given widget obj auto p_obj = obj->getParentWidget(); while ( ! obj->isWindowWidget() && p_obj ) { obj = p_obj; p_obj = p_obj->getParentWidget(); } if ( obj->isWindowWidget() ) return const_cast(reinterpret_cast(obj)); else return 0; } //---------------------------------------------------------------------- int FWindow::getWindowLayer (const FWidget* obj) { // returns the window layer from the widget obj const FWidget* window; if ( ! getWindowList() ) return -1; if ( getWindowList()->empty() ) return -1; if ( ! obj->isWindowWidget() ) { if ( (window = getWindowWidget(obj)) == 0 ) return -1; } else window = obj; auto iter = getWindowList()->begin(); auto end = getWindowList()->end(); while ( iter != end ) { if ( *iter == window ) break; ++iter; } return int(std::distance(getWindowList()->begin(), iter) + 1); } //---------------------------------------------------------------------- void FWindow::swapWindow (FWidget* obj1, FWidget* obj2) { // swaps the window layer between obj1 and obj2 if ( ! getWindowList() ) return; if ( getWindowList()->empty() ) return; if ( obj1->getFlags().modal ) return; if ( obj2->getFlags().modal ) return; auto iter = getWindowList()->begin(); auto end = getWindowList()->end(); auto iter1 = end; auto iter2 = end; while ( iter != end ) { if ( (*iter) == obj1 ) iter1 = iter; else if ( (*iter) == obj2 ) iter2 = iter; ++iter; } if ( iter1 != end && iter2 != end ) std::swap (iter1, iter2); } //---------------------------------------------------------------------- bool FWindow::raiseWindow (FWidget* obj) { // raises the window widget obj to the top if ( ! getWindowList() ) return false; if ( getWindowList()->empty() ) return false; if ( ! obj->isWindowWidget() ) return false; if ( getWindowList()->back() == obj ) return false; if ( getWindowList()->back()->getFlags().modal && ! obj->isMenuWidget() ) return false; auto iter = getWindowList()->begin(); while ( iter != getWindowList()->end() ) { if ( *iter == obj ) { getWindowList()->erase (iter); getWindowList()->push_back (obj); FEvent ev(fc::WindowRaised_Event); FApplication::sendEvent(obj, &ev); processAlwaysOnTop(); return true; } ++iter; } return false; } //---------------------------------------------------------------------- bool FWindow::lowerWindow (FWidget* obj) { // lowers the window widget obj to the bottom if ( ! getWindowList() ) return false; if ( getWindowList()->empty() ) return false; if ( ! obj->isWindowWidget() ) return false; if ( getWindowList()->front() == obj ) return false; if ( obj->getFlags().modal ) return false; auto iter = getWindowList()->begin(); while ( iter != getWindowList()->end() ) { if ( *iter == obj ) { getWindowList()->erase (iter); getWindowList()->insert (getWindowList()->begin(), obj); FEvent ev(fc::WindowLowered_Event); FApplication::sendEvent(obj, &ev); return true; } ++iter; } return false; } //---------------------------------------------------------------------- bool FWindow::zoomWindow() { if ( zoomed ) { zoomed = false; FRect oldGeometry (getTermGeometryWithShadow()); setGeometry (normalGeometry); restoreVTerm (oldGeometry); redraw(); } else { zoomed = true; // save the current geometry normalGeometry = getGeometry(); FRect oldGeometry (getTermGeometryWithShadow()); setGeometry (FPoint(1, 1), FSize(getMaxWidth(), getMaxHeight())); restoreVTerm (oldGeometry); redraw(); } return zoomed; } //---------------------------------------------------------------------- void FWindow::switchToPrevWindow (FWidget* widget) { // switch to previous window // Disable terminal updates to avoid flickering // when redrawing the focused widget if ( widget ) widget->updateTerminal (FVTerm::stop_refresh); bool is_activated = activatePrevWindow(); auto active_win = static_cast(getActiveWindow()); if ( ! is_activated ) { // no previous window -> looking for another window if ( getWindowList() && getWindowList()->size() > 1 ) { auto iter = getWindowList()->end(); auto begin = getWindowList()->begin(); do { --iter; auto w = static_cast(*iter); if ( w && w != active_win && ! (w->isWindowHidden() || w->isWindowActive()) && w != static_cast(getStatusBar()) && w != static_cast(getMenuBar()) ) { setActiveWindow(w); break; } } while ( iter != begin ); } } if ( active_win ) { auto focus = active_win->getWindowFocusWidget(); if ( ! active_win->isWindowActive() ) setActiveWindow(active_win); if ( focus) { focus->setFocus(); if ( ! focus->isWindowWidget() ) focus->redraw(); } } // Enable terminal updates again if ( widget ) widget->updateTerminal (FVTerm::continue_refresh); } //---------------------------------------------------------------------- bool FWindow::activatePrevWindow() { // activate the previous window auto w = previous_window; if ( w ) { if ( w->isWindowActive() ) return true; if ( ! w->isWindowHidden() ) { setActiveWindow(w); return true; } } return false; } //---------------------------------------------------------------------- void FWindow::setShadowSize (const FSize& size) { const FSize& old_shadow = getShadow(); FWidget::setShadowSize (size); const FSize& new_shadow = getShadow(); if ( isVirtualWindow() && (new_shadow != old_shadow) ) { auto geometry = getTermGeometry(); geometry.move(-1, -1); resizeArea (geometry, getShadow(), getVWin()); } } // protected methods of FWindow //---------------------------------------------------------------------- void FWindow::adjustSize() { int old_x = getTermX(); int old_y = getTermY(); FWidget::adjustSize(); if ( zoomed ) setGeometry (FPoint(1, 1), FSize(getMaxWidth(), getMaxHeight()), false); else if ( isVirtualWindow() ) { if ( getTermX() != old_x ) getVWin()->offset_left = getTermX() - 1; if ( getTermY() != old_y ) getVWin()->offset_top = getTermY() - 1; } } //---------------------------------------------------------------------- bool FWindow::event (FEvent* ev) { switch ( uInt(ev->type()) ) { case fc::WindowActive_Event: onWindowActive (ev); break; case fc::WindowInactive_Event: onWindowInactive (ev); break; case fc::WindowRaised_Event: onWindowRaised (ev); break; case fc::WindowLowered_Event: onWindowLowered (ev); break; default: return FWidget::event(ev); } return true; } //---------------------------------------------------------------------- void FWindow::onWindowActive (FEvent*) { } //---------------------------------------------------------------------- void FWindow::onWindowInactive (FEvent*) { } //---------------------------------------------------------------------- void FWindow::onWindowRaised (FEvent*) { } //---------------------------------------------------------------------- void FWindow::onWindowLowered (FEvent*) { } // private methods of FWindow //---------------------------------------------------------------------- void FWindow::deleteFromAlwaysOnTopList (FWidget* obj) { // delete the window object obj from the always-on-top list if ( ! getAlwaysOnTopList() || getAlwaysOnTopList()->empty() ) return; auto iter = getAlwaysOnTopList()->begin(); while ( iter != getAlwaysOnTopList()->end() ) { if ( *iter == obj ) { getAlwaysOnTopList()->erase (iter); return; } ++iter; } } //---------------------------------------------------------------------- void FWindow::processAlwaysOnTop() { // Raise all always-on-top windows if ( ! getAlwaysOnTopList() || getAlwaysOnTopList()->empty() ) return; auto iter = getAlwaysOnTopList()->begin(); while ( iter != getAlwaysOnTopList()->end() ) { delWindow (*iter); if ( getWindowList() ) getWindowList()->push_back(*iter); ++iter; } } } // namespace finalcut