// WinButton.cc for Fluxbox Window Manager // Copyright (c) 2003 Henrik Kinnunen (fluxgen at users.sourceforge.net) // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. /// $Id$ #include "WinButton.hh" #include "App.hh" #include "Window.hh" #include "WinButtonTheme.hh" WinButton::WinButton(const FluxboxWindow &listen_to, WinButtonTheme &theme, Type buttontype, const FbTk::FbWindow &parent, int x, int y, unsigned int width, unsigned int height): FbTk::Button(parent, x, y, width, height), m_type(buttontype), m_listen_to(listen_to), m_theme(theme) { theme.reconfigSig().attach(this); } void WinButton::exposeEvent(XExposeEvent &event) { FbTk::Button::exposeEvent(event); drawType(false, false); } void WinButton::buttonReleaseEvent(XButtonEvent &event) { FbTk::Button::buttonReleaseEvent(event); } // clear is used to force this to clear the window (e.g. called from clear()) void WinButton::drawType(bool clear, bool no_trans) { bool used = false; // if it's odd and we're centring, we need to add one int oddW = width()%2; int oddH = height()%2; switch (m_type) { case MAXIMIZE: if (pressed() && m_theme.maximizePressedPixmap().pixmap().drawable() != 0) { FbTk::FbWindow::setBackgroundPixmap(m_theme. maximizePressedPixmap(). pixmap().drawable()); used = true; } else { // check focus if (!m_listen_to.isFocused()) { if (m_theme.maximizeUnfocusPixmap().pixmap().drawable() != 0) { // not focused FbTk::FbWindow::setBackgroundPixmap(m_theme. maximizeUnfocusPixmap(). pixmap().drawable()); used = true; } } else if (m_theme.maximizePixmap().pixmap().drawable() != 0) { FbTk::FbWindow::setBackgroundPixmap(m_theme. maximizePixmap(). pixmap().drawable()); used = true; } } if (used || clear) FbTk::FbWindow::clear(); // if no pixmap was used, use old style if (!used) { if (gc() == 0) // must have valid graphic context return; drawRectangle(gc(), 2, 2, width() - 5, height() - 5); drawLine(gc(), 2, 3, width() - 3, 3); } break; case MINIMIZE: if (pressed() && m_theme.iconifyPressedPixmap().pixmap().drawable() != 0) { FbTk::FbWindow::setBackgroundPixmap(m_theme. iconifyPressedPixmap(). pixmap().drawable()); used = true; } else { if (m_theme.iconifyPixmap().pixmap().drawable()){ // check focus if (!m_listen_to.isFocused()) { if (m_theme.iconifyUnfocusPixmap().pixmap().drawable() != 0) { // not focused FbTk::FbWindow::setBackgroundPixmap(m_theme. iconifyUnfocusPixmap(). pixmap().drawable()); used = true; } } else if (m_theme.iconifyPixmap().pixmap().drawable() != 0) { FbTk::FbWindow::setBackgroundPixmap(m_theme. iconifyPixmap(). pixmap().drawable()); used = true; } } } if (used || clear) { FbTk::FbWindow::clear(); } if (!used && gc() != 0) { // must have valid graphic context FbTk::FbWindow::drawRectangle(gc(), 2, height() - 5, width() - 5, 2); } break; case STICK: if (m_listen_to.isStuck() && !pressed()) { if ( m_theme.stuckPixmap().pixmap().drawable() && ! pressed()) { // we're using the same pixmap for pressed as in not stuck // check focus if (!m_listen_to.isFocused()) { if ( m_theme.stuckUnfocusPixmap().pixmap().drawable() != 0) { // not focused FbTk::FbWindow::setBackgroundPixmap(m_theme. stuckUnfocusPixmap(). pixmap().drawable()); used = true; } } else if (m_theme.stuckPixmap().pixmap().drawable() != 0) { // focused FbTk::FbWindow::setBackgroundPixmap(m_theme. stuckPixmap(). pixmap().drawable()); used = true; } } } else { // not stuck and pressed if (pressed()) { if (m_theme.stickPressedPixmap().pixmap().drawable() != 0) { FbTk::FbWindow::setBackgroundPixmap(m_theme. stickPressedPixmap(). pixmap().drawable()); used = true; } } else { // not pressed // check focus if (!m_listen_to.isFocused()) { if (m_theme.stickUnfocusPixmap().pixmap().drawable() != 0) { // not focused FbTk::FbWindow::setBackgroundPixmap(m_theme. stickUnfocusPixmap(). pixmap().drawable()); used = true; } } else if (m_theme.stickPixmap().pixmap().drawable()) { // focused FbTk::FbWindow::setBackgroundPixmap(m_theme. stickPixmap(). pixmap().drawable()); used = true; } } } if (used || clear) FbTk::FbWindow::clear(); if (!used && gc() != 0) { // width/4 != width/2, so we use /4*2 so that it's properly centred if (m_listen_to.isStuck()) { fillRectangle(gc(), width()/2 - width()/4, height()/2 - height()/4, width()/4*2 + oddW, height()/4*2 + oddH); } else { fillRectangle(gc(), width()/2 - width()/10, height()/2 - height()/10, width()/10*2 + oddW, height()/10*2 + oddH); } } break; case CLOSE: if (pressed()) { if (m_theme.closePressedPixmap().pixmap().drawable()) { FbTk::FbWindow::setBackgroundPixmap(m_theme. closePressedPixmap(). pixmap().drawable()); used = true; } } else { // not pressed // check focus if (!m_listen_to.isFocused()) { if (m_theme.closeUnfocusPixmap().pixmap().drawable() != 0) { // not focused FbTk::FbWindow::setBackgroundPixmap(m_theme. closeUnfocusPixmap(). pixmap().drawable()); used = true; } } else if (m_theme.closePixmap().pixmap().drawable() != 0) { // focused FbTk::FbWindow::setBackgroundPixmap(m_theme. closePixmap(). pixmap().drawable()); used = true; } } if (used || clear) FbTk::FbWindow::clear(); if (!used && gc() != 0) { // must have valid graphic context drawLine(gc(), 2, 2, width() - 3, height() - 3); // I can't figure out why this second one needs a y offset of 1????? // but it does - at least on my box: // XFree86 Version 4.2.1.1 (Debian 4.2.1-12.1 20031003005825) // (protocol Version 11, revision 0, vendor release 6600) // But not on mine? It's wonky. Put back to the same coords. // was width-2, 1 in the second drawline // Perhaps some X versions don't draw the endpoint? // Mine: // XFree86 Version 4.3.0.1 (Debian 4.3.0.dfsg.1-1 20040428170728) // (X Protocol Version 11, Revision 0, Release 6.6) drawLine(gc(), 2, height() - 3, width() - 3, 2); } break; case SHADE: if (pressed()) { if (m_theme.shadePressedPixmap().pixmap().drawable()) { FbTk::FbWindow::setBackgroundPixmap(m_theme. shadePressedPixmap(). pixmap().drawable()); used = true; } } else { // not pressed // check focus if (!m_listen_to.isFocused()) { if ( m_theme.shadeUnfocusPixmap().pixmap().drawable() != 0) { // not focused FbTk::FbWindow::setBackgroundPixmap(m_theme. shadeUnfocusPixmap(). pixmap().drawable()); used = true; } } else if (m_theme.shadePixmap().pixmap().drawable() != 0) { // focused FbTk::FbWindow::setBackgroundPixmap(m_theme. shadePixmap(). pixmap().drawable()); used = true; } } if (used || clear) FbTk::FbWindow::clear(); break; } // if ((used || clear) && !no_trans) updateTransparent(); } void WinButton::clear() { drawType(true, true); } void WinButton::update(FbTk::Subject *subj) { clear(); }