You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdeartwork/twin-styles/icewm/icewm.cpp

1705 lines
49 KiB

/*
$Id$
Gallium-IceWM themeable KWin client
Copyright 2001
Karol Szwed <gallium@kde.org>
http://gallium.n3.net/
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This program 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
General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; see the file COPYING. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
-----------------------------------------------------------------------------
This client loads most icewm 1.0.X pixmap themes, without taking into account
specific font settings for clients, or coloured mouse cursors. Titlebar
fonts can be changed via the kde control center. Bi-colour mouse cursors
may be added in future if requested by users, as well as theme font support.
Any styles using inbuilt icewm titlebar drawing without using pixmaps (e.g.
Warp4, win95 etc.) are not fully supported, and may cause drawing errors,
as these themes use in-built icewm drawing mechanisms.
When a pixmap theme is not present (or a corrupt one is present) then very
plain title decorations are painted instead, so that users don't see
non-painted window areas where possible ;)
At a later date, frame shaping may be added if really requested, and an
update to support the latest icewm 1.1.X theme format may be made.
*/
#include <tdeconfig.h>
#include <kstandarddirs.h>
#include <tdeglobal.h>
#include <tdelocale.h>
#include <kdrawutil.h>
#include <tqapplication.h>
#include <tqlabel.h>
#include <tqdrawutil.h>
#include <tqdatetime.h>
#include <tqbitmap.h>
#include <tqcursor.h>
#include <tqstring.h>
#include <tqtooltip.h>
#include <tqregexp.h>
#include "icewm.h"
namespace IceWM {
////////////////////////////////////////////////////////////////////////////////////////////
// Here's the global pixmap stuff - as memory efficient as it can be :)
////////////////////////////////////////////////////////////////////////////////////////////
// IceWM frame pixmaps
TQPixmap* frameTL[] = {NULL, NULL};
TQPixmap* frameT [] = {NULL, NULL};
TQPixmap* frameTR[] = {NULL, NULL};
TQPixmap* frameL [] = {NULL, NULL};
TQPixmap* frameR [] = {NULL, NULL};
TQPixmap* frameBL[] = {NULL, NULL};
TQPixmap* frameB [] = {NULL, NULL};
TQPixmap* frameBR[] = {NULL, NULL};
// Button pixmaps
TQPixmap* closePix[] = {NULL, NULL};
TQPixmap* depthPix[] = {NULL, NULL};
TQPixmap* maximizePix[] = {NULL, NULL};
TQPixmap* minimizePix[] = {NULL, NULL};
TQPixmap* restorePix[] = {NULL, NULL};
TQPixmap* hidePix[] = {NULL, NULL};
TQPixmap* rollupPix[] = {NULL, NULL};
TQPixmap* rolldownPix[] = {NULL, NULL};
TQPixmap* menuButtonPix[] = {NULL, NULL};
// Titlebar pixmaps
TQPixmap* titleJ[] = {NULL, NULL};
TQPixmap* titleL[] = {NULL, NULL};
TQPixmap* titleS[] = {NULL, NULL};
TQPixmap* titleP[] = {NULL, NULL};
TQPixmap* titleT[] = {NULL, NULL};
TQPixmap* titleM[] = {NULL, NULL};
TQPixmap* titleB[] = {NULL, NULL};
TQPixmap* titleR[] = {NULL, NULL};
TQPixmap* titleQ[] = {NULL, NULL};
ThemeHandler* clientHandler;
TQString* titleButtonsLeft;
TQString* titleButtonsRight;
TQColor* colorActiveBorder;
TQColor* colorInActiveBorder;
TQColor* colorActiveButton;
TQColor* colorInActiveButton;
TQColor* colorActiveTitleBarText;
TQColor* colorInActiveTitleBarText;
TQColor* colorActiveTitleBar;
TQColor* colorInActiveTitleBar;
TQColor* colorActiveTitleTextShadow;
TQColor* colorInActiveTitleTextShadow;
int cornerSizeX;
int cornerSizeY;
int titleBarHeight;
int borderSizeX;
int borderSizeY;
bool validframe = false;
bool useActiveShadow = false;
bool useInActiveShadow = false;
// KControl Settings - Read from twinicewmrc config file or icewm theme
bool themeTitleTextColors = true; // Allow theme to set colors.
// kcontrol will have no effect
bool titleBarOnTop = true; // Titlebars can be below windows too :)
bool showMenuButtonIcon = false; // Draw a mini icon over the menu pixmap.
bool customButtonPositions = false; // Let the theme dictate the btn pos.
bool titleBarCentered = true;
enum styles {OTHER, WARP3, WARP4, MOTIF, WIN95, NICE} themeLook;
////////////////////////////////////////////////////////////////////////////////////////////
// General utility functions
////////////////////////////////////////////////////////////////////////////////////////////
// Returns true if both active and inactive pixmaps are valid, and not null
bool validPixmaps( TQPixmap* p[] )
{
return ( p[Active] && ( !p[Active]->isNull() ) &&
p[InActive] && ( !p[InActive]->isNull() ) );
}
////////////////////////////////////////////////////////////////////////////////////////////
// ThemeHandler class
//
// This class allows us to free dynamic memory upon being reset, or unloaded
// from twin, so we don't leak big images everywhere, and handles the theme
// initialisation / destruction in general.
////////////////////////////////////////////////////////////////////////////////////////////
ThemeHandler::ThemeHandler()
{
TDEGlobal::locale()->insertCatalogue("twin_clients");
TDEGlobal::locale()->insertCatalogue("twin_art_clients");
initialized = false;
// Prevent having globals objects (use pointers to objects)
titleButtonsLeft = new TQString();
titleButtonsRight = new TQString();
colorActiveBorder = new TQColor();
colorInActiveBorder = new TQColor();
colorActiveButton = new TQColor();
colorInActiveButton = new TQColor();
colorActiveTitleBarText = new TQColor();
colorInActiveTitleBarText = new TQColor();
colorActiveTitleBar = new TQColor();
colorInActiveTitleBar = new TQColor();
colorActiveTitleTextShadow = new TQColor();
colorInActiveTitleTextShadow = new TQColor();
// Initialize
readConfig();
initTheme();
validframe = isFrameValid();
initialized = true;
}
ThemeHandler::~ThemeHandler()
{
if (initialized)
freePixmaps();
delete colorInActiveTitleTextShadow;
delete colorActiveTitleTextShadow;
delete colorInActiveBorder;
delete colorActiveTitleBarText;
delete colorInActiveTitleBarText;
delete colorActiveTitleBar;
delete colorInActiveTitleBar;
delete colorActiveBorder;
delete colorActiveButton;
delete colorInActiveButton;
delete titleButtonsRight;
delete titleButtonsLeft;
}
KDecoration* ThemeHandler::createDecoration( KDecorationBridge* bridge )
{
return new IceWMClient( bridge, this );
}
// Converts TDE style button strings to icewm style button strings
void ThemeHandler::convertButtons( TQString& s )
{
s.replace( TQRegExp("_"), ""); // Spacer (ignored)
s.replace( TQRegExp("H"), ""); // Help (ignored)
s.replace( TQRegExp("M"), "s"); // Sysmenu
s.replace( TQRegExp("S"), "d"); // Sticky/OnAllDesktops
s.replace( TQRegExp("I"), "i"); // Minimize
s.replace( TQRegExp("A"), "m"); // Maximize
s.replace( TQRegExp("X"), "x"); // Close
}
// Reverses all characters in a TQString
TQString ThemeHandler::reverseString( TQString s )
{
if (s.length() <= 1)
return s;
TQString tmpStr;
for(int i = s.length()-1; i >= 0; i--)
{
tmpStr += s[(unsigned int)i];
}
return tmpStr;
}
// This function reads the twinicewmrc config file
void ThemeHandler::readConfig()
{
TDEConfig conf("twinicewmrc");
conf.setGroup("General");
themeName = conf.readEntry("CurrentTheme");
themeTitleTextColors = conf.readBoolEntry("ThemeTitleTextColors", true);
showMenuButtonIcon = conf.readBoolEntry("ShowMenuButtonIcon", false);
titleBarOnTop = conf.readBoolEntry("TitleBarOnTop", true);
customButtonPositions = KDecoration::options()->customButtonPositions();
if (customButtonPositions)
{
*titleButtonsLeft = KDecoration::options()->titleButtonsLeft();
*titleButtonsRight = KDecoration::options()->titleButtonsRight();
// Convert TDE to icewm style buttons
convertButtons( *titleButtonsLeft );
convertButtons( *titleButtonsRight );
}
// Provide a default theme alias
if (themeName == "default")
themeName = "";
}
// This creates the dynamic pixmaps upon loading the style
// into the pixmap buffers above, and configures the dimensioning stuff.
void ThemeHandler::initTheme()
{
// Add a slash if required
if ( !themeName.isEmpty() )
themeName += "/";
// We use tdeconfig to read icewm config files...
// this is easy since icewm uses key=value pairs!
TDEConfig config( locate("data", TQString("twin/icewm-themes/") +
themeName + TQString("default.theme")) );
// Load specifics, or use IceWM defaults instead.
borderSizeX = config.readNumEntry("BorderSizeX", 6);
borderSizeY = config.readNumEntry("BorderSizeY", 6);
cornerSizeX = config.readNumEntry("CornerSizeX", 24);
cornerSizeY = config.readNumEntry("CornerSizeY", 24);
titleBarCentered = (bool) config.readNumEntry("TitleBarCentered", 0);
// Check if readConfig() hasn't overridden this value...
if (!showMenuButtonIcon)
showMenuButtonIcon = (bool) config.readNumEntry("ShowMenuButtonIcon", 0);
titleBarHeight = config.readNumEntry("TitleBarHeight", 20);
if (!customButtonPositions)
{
// Read in the button configuration, stripping any quotes
// Ignore on all desktops 'd' on the left buttons
// (some themes look bad with it on by default)
*titleButtonsLeft = config.readEntry("TitleButtonsLeft", "s");
*titleButtonsLeft = titleButtonsLeft->replace( TQRegExp(TQString("\"")), "");
*titleButtonsRight = config.readEntry("TitleButtonsRight", "xmir");
*titleButtonsRight = titleButtonsRight->replace( TQRegExp(TQString("\"")), "");
// I have no idea why the right side buttons in icewm are reversed
*titleButtonsRight = reverseString( *titleButtonsRight );
}
// Read the default border and text colours from the config file
// And use IceWM defaults if not found
TQString s;
s = config.readEntry("Look", "other");
if (s=="motif") themeLook = MOTIF;
else if (s=="warp3") themeLook = WARP3;
else if (s=="warp4") themeLook = WARP4;
else if (s=="win95") themeLook = WIN95;
else if (s=="nice") themeLook = NICE;
else themeLook = OTHER;
s = config.readEntry("ColorActiveBorder", "#C0C0C0");
*colorActiveBorder = decodeColor( s );
s = config.readEntry("ColorNormalBorder", "#C0C0C0");
*colorInActiveBorder = decodeColor( s );
s = config.readEntry("ColorActiveButton", "#C0C0C0");
*colorActiveButton = decodeColor( s );
s = config.readEntry("ColorNormalButton", "#C0C0C0");
*colorInActiveButton = decodeColor( s );
// Use these as a last resort
s = config.readEntry("ColorActiveTitleBar", "#0000A0");
*colorActiveTitleBar = decodeColor( s );
s = config.readEntry("ColorNormalTitleBar", "#808080");
*colorInActiveTitleBar = decodeColor( s );
// Read titlebar text colours
s = config.readEntry("ColorActiveTitleBarText", "#FFFFFF");
*colorActiveTitleBarText = decodeColor( s );
s = config.readEntry("ColorNormalTitleBarText", "#000000");
*colorInActiveTitleBarText = decodeColor( s );
// Use title text shadows only with theme title text colors
if ( themeTitleTextColors )
{
s = config.readEntry("ColorActiveTitleBarShadow");
if (!s.isEmpty())
{
*colorActiveTitleTextShadow = decodeColor( s );
useActiveShadow = true;
} else
useActiveShadow = false;
s = config.readEntry("ColorNormalTitleBarShadow");
if (!s.isEmpty())
{
*colorInActiveTitleTextShadow = decodeColor( s );
useInActiveShadow = true;
} else
useInActiveShadow = false;
} else
{
useActiveShadow = false;
useInActiveShadow = false;
}
// Stretch pixmaps for speed, where required
setPixmap( titleJ, "title", "J.xpm" );
setPixmap( titleL, "title", "L.xpm" );
setPixmap( titleS, "title", "S.xpm", true );
setPixmap( titleP, "title", "P.xpm" );
setPixmap( titleT, "title", "T.xpm", true );
setPixmap( titleM, "title", "M.xpm" );
setPixmap( titleB, "title", "B.xpm", true );
setPixmap( titleR, "title", "R.xpm" );
setPixmap( titleQ, "title", "Q.xpm" );
setPixmapButton( closePix, "close", ".xpm" );
setPixmapButton( depthPix, "depth", ".xpm" );
setPixmapButton( maximizePix, "maximize", ".xpm" );
setPixmapButton( minimizePix, "minimize", ".xpm" );
setPixmapButton( restorePix, "restore", ".xpm" );
setPixmapButton( hidePix, "hide", ".xpm" );
setPixmapButton( rollupPix, "rollup", ".xpm" );
setPixmapButton( rolldownPix, "rolldown", ".xpm" );
setPixmapButton( menuButtonPix,"menuButton",".xpm" );
// Top
setPixmap( frameTL, "frame", "TL.xpm" );
setPixmap( frameT, "frame", "T.xpm", true );
setPixmap( frameTR, "frame", "TR.xpm" );
// Sides
setPixmap( frameL, "frame", "L.xpm", true,TQt::Vertical );
setPixmap( frameR, "frame", "R.xpm", true,TQt::Vertical );
// Bottom
setPixmap( frameBL, "frame", "BL.xpm" );
setPixmap( frameB, "frame", "B.xpm", true );
setPixmap( frameBR, "frame", "BR.xpm" );
// Make sure border sizes are at least reasonable...
if (borderSizeX < 0)
borderSizeX = 0;
if (borderSizeY < 0)
borderSizeY = 0;
// ...and titleBarHeight as well
if (titleBarHeight < 0)
titleBarHeight = 0;
// This is a work-around for some themes
if (!titleT[Active])
titleT[Active] = duplicateValidPixmap( Active );
if (!titleB[Active])
titleB[Active] = duplicateValidPixmap( Active );
if (titleL[Active] && !titleL[InActive])
titleL[InActive] = duplicateValidPixmap( InActive, titleL[Active]->width() );
if (titleS[Active] && !titleS[InActive])
titleS[InActive] = duplicateValidPixmap( InActive, titleS[Active]->width() );
if (titleP[Active] && !titleP[InActive])
titleP[InActive] = duplicateValidPixmap( InActive, titleP[Active]->width() );
if (titleT[Active] && !titleT[InActive])
titleT[InActive] = duplicateValidPixmap( InActive, titleT[Active]->width() );
if (titleM[Active] && !titleM[InActive])
titleM[InActive] = duplicateValidPixmap( InActive, titleM[Active]->width() );
if (titleB[Active] && !titleB[InActive])
titleB[InActive] = duplicateValidPixmap( InActive, titleB[Active]->width() );
if (titleR[Active] && !titleR[InActive])
titleR[InActive] = duplicateValidPixmap( InActive, titleR[Active]->width() );
}
TQPixmap* ThemeHandler::duplicateValidPixmap( bool act, int size )
{
TQPixmap* p1 = NULL;
// Use the stretch or title pixmaps instead
if ( titleS[act] )
p1 = new TQPixmap( *titleS[act] );
else if ( titleB[act] )
p1 = new TQPixmap( *titleB[act] );
else if ( titleT[act] )
p1 = new TQPixmap( *titleT[act] );
// Stretch if required
if ( (size != -1) && p1 && (!p1->isNull()) )
p1 = stretchPixmap( p1, true, size );
return p1;
}
// Frees all memory used by pixmaps.
void ThemeHandler::freePixmaps()
{
freePixmapGroup( frameTL );
freePixmapGroup( frameT );
freePixmapGroup( frameTR );
freePixmapGroup( frameL );
freePixmapGroup( frameR );
freePixmapGroup( frameBL );
freePixmapGroup( frameB );
freePixmapGroup( frameBR );
freePixmapGroup( closePix );
freePixmapGroup( depthPix );
freePixmapGroup( maximizePix );
freePixmapGroup( minimizePix );
freePixmapGroup( restorePix );
freePixmapGroup( hidePix );
freePixmapGroup( rollupPix );
freePixmapGroup( rolldownPix );
freePixmapGroup( menuButtonPix );
freePixmapGroup( titleJ );
freePixmapGroup( titleL );
freePixmapGroup( titleS );
freePixmapGroup( titleP );
freePixmapGroup( titleT );
freePixmapGroup( titleM );
freePixmapGroup( titleB );
freePixmapGroup( titleR );
freePixmapGroup( titleQ );
}
// Frees a dynamic pixmap group from the heap.
void ThemeHandler::freePixmapGroup( TQPixmap* p[] )
{
if (p)
{
if (p[Active]) delete p[Active];
if (p[InActive]) delete p[InActive];
p[Active] = NULL;
p[InActive] = NULL;
} else
tqWarning("twin-icewm: freePixmapGroup - invalid TQPixmap** 'p'\n");
}
// Converts icewm colors #C0C0C0 or rgb:C0/C0/C0 to TQColors
TQColor ThemeHandler::decodeColor( TQString& s )
{
// Make rgb:C0/C0/C0, or #C0/C0/C0 -> C0C0C0
s.replace( TQRegExp("r"), "");
s.replace( TQRegExp("g"), "");
s.replace( TQRegExp("b"), "");
s.replace( TQRegExp("#"), "");
s.replace( TQRegExp("/"), "");
s.replace( TQRegExp(":"), "");
s.replace( TQRegExp("\\"), "");
s.replace( TQRegExp("\""), "");
// Wierd error - return grey
if (s.length() != 6)
return TQColor( 0xC0, 0xC0, 0xC0 );
// TQt makes this conversion very easy
return TQColor( TQString("#") + s );
}
// Stretches tiny pixmaps vertically or horizontally, taking into account
// repetition in patterns, so as not to make them mismatched
TQPixmap* ThemeHandler::stretchPixmap( TQPixmap* src, bool stretchHoriz, int stretchSize )
{
if (!src) return NULL;
if (src->isNull()) return NULL;
int s_inc, size;
// If its the right size already, just return
if (stretchSize == -1)
{
if (stretchHoriz)
s_inc = src->width();
else
s_inc = src->height();
size = s_inc;
if (size >= 100)
return src;
// Stretch an appropriate amount - taking care of pattern repetition
while( size < 100 )
size += s_inc;
} else
size = stretchSize;
TQPixmap* p = new TQPixmap();
if ( stretchHoriz )
p->resize( size, src->height() );
else
p->resize( src->width(), size );
TQPainter pnt( p );
if ( stretchHoriz )
pnt.drawTiledPixmap( 0, 0, size, src->height(), *src);
else
pnt.drawTiledPixmap( 0, 0, src->width(), size, *src);
pnt.end();
delete src;
return p;
}
static void draw3DRect(TQPainter &pnt, TQColor &col, int x, int y, int w, int h, bool up) {
TQColor light = col.light(135);
TQColor dark = col.dark(140);
pnt.setPen(up ? light : dark);
pnt.drawLine(x, y, x+w, y);
pnt.drawLine(x, y, x, y+h);
pnt.setPen(up ? dark : light);
pnt.drawLine(x, y+h, x+w, y+h);
pnt.drawLine(x+w, y, x+w, y+h);
pnt.setPen(col);
pnt.drawPoint(x+w, y);
pnt.drawPoint(x, y+h);
}
void ThemeHandler::setPixmapButton( TQPixmap* p[], TQString s1, TQString s2)
{
if ( p[Active] )
tqWarning("twin-icewm: setPixmap - should be null (1)\n");
if ( p[InActive] )
tqWarning("twin-icewm: setPixmap - should be null (2)\n");
TQString str = locate("appdata", TQString("icewm-themes/")
+ themeName + s1 + "A" + s2);
if (str.isEmpty())
str = locate("appdata", TQString("icewm-themes/")
+ themeName + s1 + s2);
TQPixmap *qp = new TQPixmap(str);
TQColor cActive = themeLook == WIN95 ? *colorActiveTitleBar : *colorActiveButton;
TQColor cInActive = themeLook == WIN95 ? *colorInActiveTitleBar : *colorInActiveButton;
if (!qp->isNull() && themeLook > 0) {
int w = qp->width();
if (themeLook > 0 && titleBarHeight > w) w = titleBarHeight;
p[Active] = new TQPixmap(w, 2*titleBarHeight );
p[Active] -> fill(cActive);
TQPainter pnt( p[Active] );
int offX = (w - qp->width())/2;
int offY = (titleBarHeight - qp->height())/2;
if (offY < 0) offY = 0;
if (themeLook == WIN95) {
draw3DRect(pnt, *colorActiveButton, offX-1, offY-1,
qp->width()+1, qp->height()+1, true);
draw3DRect(pnt, *colorActiveButton, offX-1, offY-1 + titleBarHeight,
qp->width()+1, qp->height()+1, false);
} else if (themeLook != WARP4) {
draw3DRect(pnt, *colorActiveButton, 0, 0,
w-1, titleBarHeight-1, true);
draw3DRect(pnt, *colorActiveButton, 0, titleBarHeight,
w-1, 2*titleBarHeight-1, false);
}
pnt.drawPixmap(offX, offY, *qp);
if (qp->height() <= titleBarHeight) {
pnt.drawPixmap(offX, titleBarHeight+offY, *qp);
}
pnt.end();
delete qp;
} else {
p[Active] = qp;
}
str = locate("appdata", TQString("icewm-themes/")
+ themeName + s1 + "I" + s2);
if (str.isEmpty())
str = locate("appdata", TQString("icewm-themes/")
+ themeName + s1 + s2);
qp = new TQPixmap(str);
if (!qp->isNull() && themeLook > 0) {
int w = qp->width();
if (titleBarHeight > w) w = titleBarHeight;
p[InActive] = new TQPixmap(w, 2*titleBarHeight );
p[InActive] -> fill(cInActive);
TQPainter pnt( p[InActive] );
int offX = (w - qp->width())/2;
int offY = (titleBarHeight - qp->height())/2;
if (offY < 0) offY = 0;
if (themeLook == WIN95) {
draw3DRect(pnt, *colorInActiveButton, offX-1, offY-1,
qp->width()+1, qp->height()+1, true);
draw3DRect(pnt, *colorInActiveButton, offX-1, offY-1 + titleBarHeight,
qp->width()+1, qp->height()+1, false);
} else if (themeLook != WARP4) {
draw3DRect(pnt, *colorInActiveButton, 0, 0,
w-1, titleBarHeight-1, true);
draw3DRect(pnt, *colorActiveButton, 0, titleBarHeight,
w-1, 2*titleBarHeight-1, false);
}
pnt.drawPixmap(offX, offY, *qp);
if (qp->height() <= titleBarHeight) {
pnt.drawPixmap(offX, titleBarHeight+offY, *qp);
}
pnt.end();
delete qp;
} else {
p[InActive] = qp;
}
}
// Loads the specified Active/InActive files into the specific pixmaps, and
// can perform horizontal / vertical stretching if required for speed.
// Tries to implement some icewm specific pixmap handling for some dodgy themes
void ThemeHandler::setPixmap( TQPixmap* p[], TQString s1, TQString s2,
bool stretch, bool stretchHoriz )
{
if ( p[Active] )
tqWarning("twin-icewm: setPixmap - should be null (1)\n");
if ( p[InActive] )
tqWarning("twin-icewm: setPixmap - should be null (2)\n");
p[Active] = new TQPixmap( locate("data", TQString("twin/icewm-themes/")
+ themeName + s1 + "A" + s2) );
p[InActive] = new TQPixmap( locate("data", TQString("twin/icewm-themes/")
+ themeName + s1 + "I" + s2) );
// Stretch the pixmap if requested.
if ( stretch )
{
if (p[Active])
p[Active] = stretchPixmap( p[Active], stretchHoriz );
if (p[InActive])
p[InActive] = stretchPixmap( p[InActive], stretchHoriz );
}
if ( p[Active] && p[InActive] )
{
// Make sure active and inactive pixmaps are the same width for proper painting
if (p[Active]->width() > p[InActive]->width())
p[InActive] = stretchPixmap( p[InActive], true, p[Active]->width() );
}
}
// returns true if there were enough pixmaps loaded to
// draw the pixmap frame properly.
bool ThemeHandler::isFrameValid()
{
return
( validPixmaps( frameTL ) &&
validPixmaps( frameT ) &&
validPixmaps( frameTR ) &&
validPixmaps( frameL ) &&
validPixmaps( frameR ) &&
validPixmaps( frameBL ) &&
validPixmaps( frameB ) &&
validPixmaps( frameBR ) );
}
// Resets the theme, and re-clients all twin's wrapped windows.
bool ThemeHandler::reset( unsigned long)
{
initialized = false;
freePixmaps();
readConfig();
initTheme();
validframe = isFrameValid();
initialized = true;
// recreate all clients
return true;
}
bool ThemeHandler::supports( Ability ability )
{
switch( ability )
{
case AbilityAnnounceButtons:
case AbilityButtonMenu:
case AbilityButtonOnAllDesktops:
case AbilityButtonMinimize:
case AbilityButtonMaximize:
case AbilityButtonClose:
return true;
default:
return false;
};
}
////////////////////////////////////////////////////////////////////////////////////////////
// IceWM button class
////////////////////////////////////////////////////////////////////////////////////////////
IceWMButton::IceWMButton(IceWMClient *parent, const char *name, TQPixmap* (*p)[2],
bool isToggle, const TQString& tip, const int realizeBtns )
: TQButton(parent->widget(), name)
{
m_realizeButtons = realizeBtns;
setTipText(tip);
setCursor(ArrowCursor);
// Eliminate any possible background flicker
setBackgroundMode( TQWidget::NoBackground );
client = parent;
usePixmap( p );
setFixedSize( sizeHint() );
setToggleButton( isToggle );
}
void IceWMButton::setTipText(const TQString &tip) {
if(KDecoration::options()->showTooltips()) {
TQToolTip::remove(this );
TQToolTip::add(this, tip );
}
}
TQSize IceWMButton::sizeHint() const
{
// Check for invalid data
if ( validPixmaps( (TQPixmap**) (*pix) ) ) // Cast to avoid dumb warning
{
TQPixmap* p = (*pix)[ client->isActive() ? Active : InActive ];
return( TQSize(p->width(), titleBarHeight) );
} else
return( TQSize(0, 0) );
}
void IceWMButton::usePixmap( TQPixmap* (*p)[2] )
{
if (validPixmaps( *p )) {
pix = p;
setFixedSize( (*pix)[Active]->width(), titleBarHeight );
repaint( false );
} else
pix = NULL;
}
void IceWMButton::drawButton(TQPainter *pnt)
{
if ( pix && validPixmaps(*pix) )
{
TQPixmap* p = (*pix)[ client->isActive() ? Active : InActive ];
if( p && (!p->isNull()) )
{
int width = p->width();
// Only draw the lower pixmap 1/2 for down, and upper 1/2 for up state
if( isDown() || isOn() )
pnt->drawPixmap(0, 0, *p, 0, titleBarHeight, width, titleBarHeight);
else
pnt->drawPixmap(0, 0, *p, 0, 0, width, titleBarHeight);
}
} else
tqWarning("twin-icewm: Can't paint a null pixmap button");
}
void IceWMButton::turnOn( bool isOn )
{
if ( isToggleButton() )
setOn( isOn );
}
void IceWMButton::mousePressEvent( TQMouseEvent* e )
{
last_button = e->button();
TQMouseEvent me ( e->type(), e->pos(), e->globalPos(),
(e->button()&m_realizeButtons)?TQt::LeftButton:TQt::NoButton, e->state() );
TQButton::mousePressEvent( &me );
}
void IceWMButton::mouseReleaseEvent( TQMouseEvent* e )
{
last_button = e->button();
TQMouseEvent me ( e->type(), e->pos(), e->globalPos(),
(e->button()&m_realizeButtons)?TQt::LeftButton:TQt::NoButton, e->state() );
TQButton::mouseReleaseEvent( &me );
}
////////////////////////////////////////////////////////////////////////////////////////////
// IceWMClient class
////////////////////////////////////////////////////////////////////////////////////////////
IceWMClient::IceWMClient( KDecorationBridge* bridge, KDecorationFactory* factory )
: KDecoration (bridge, factory),
m_closing(false)
{
}
IceWMClient::~IceWMClient()
{
// Free the menu pixmaps if previously allocated
if ( menuButtonWithIconPix[Active] )
delete menuButtonWithIconPix[Active];
if ( menuButtonWithIconPix[InActive] )
delete menuButtonWithIconPix[InActive];
}
void IceWMClient::init()
{
createMainWidget( WNoAutoErase | WStaticContents );
widget()->installEventFilter( this );
// Set button pointers to null so we can track things
for(int i= IceWMClient::BtnSysMenu; i < IceWMClient::BtnCount; i++)
button[i] = NULL;
// Make sure we can track the menu pixmaps too.
menuButtonWithIconPix[Active] = NULL;
menuButtonWithIconPix[InActive] = NULL;
// No flicker thanks
widget()->setBackgroundMode( NoBackground );
// Pack the windowWrapper() window within a grid layout
grid = new TQGridLayout(widget(), 0, 0, 0);
grid->setResizeMode(TQLayout::FreeResize);
grid->addRowSpacing(0, borderSizeY); // Top grab bar
// Do something IceWM can't do :)
if (titleBarOnTop) {
if( isPreview())
grid->addWidget( new TQLabel( i18n( "<center><b>IceWM preview</b></center>" ), widget() ), 2, 1);
else
grid->addItem( new TQSpacerItem( 0, 0 ), 2, 1);
// no shade flicker
grid->addItem( new TQSpacerItem( 0, 0, TQSizePolicy::Fixed, TQSizePolicy::Expanding ) );
}
else {
// no shade flicker
grid->addItem( new TQSpacerItem( 0, 0, TQSizePolicy::Fixed, TQSizePolicy::Expanding ) );
if( isPreview())
grid->addWidget( new TQLabel( i18n( "<center><b>IceWM preview</b></center>" ), widget() ), 1, 1);
else
grid->addItem( new TQSpacerItem( 0, 0 ), 1, 1);
}
grid->setRowStretch(1, 10);
grid->setRowStretch(2, 10);
grid->setColStretch(1, 10);
grid->addRowSpacing(3, borderSizeY);
grid->addColSpacing(0, borderSizeX);
grid->addColSpacing(2, borderSizeX);
// Pack the titlebar with spacers and buttons
hb = new TQBoxLayout(0, TQBoxLayout::LeftToRight, 0, 0, 0);
hb->setResizeMode( TQLayout::FreeResize );
titleSpacerJ = addPixmapSpacer( titleJ );
addClientButtons( *titleButtonsLeft );
titleSpacerL = addPixmapSpacer( titleL );
// Centre titlebar if required.
TQSizePolicy::SizeType spTitleBar;
spTitleBar = titleBarCentered ? TQSizePolicy::Expanding : TQSizePolicy::Maximum;
titleSpacerS = addPixmapSpacer( titleS, spTitleBar, 1 );
titleSpacerP = addPixmapSpacer( titleP );
titlebar = new TQSpacerItem( titleTextWidth(caption()), titleBarHeight,
TQSizePolicy::Preferred, TQSizePolicy::Fixed );
hb->addItem(titlebar);
titleSpacerM = addPixmapSpacer( titleM );
titleSpacerB = addPixmapSpacer( titleB, TQSizePolicy::Expanding, 1 );
titleSpacerR = addPixmapSpacer( titleR );
addClientButtons( *titleButtonsRight );
titleSpacerQ = addPixmapSpacer( titleQ );
if (titleBarOnTop)
grid->addLayout ( hb, 1, 1 );
else
grid->addLayout ( hb, 2, 1 );
}
// Adds the buttons to the hbox layout as per the buttons specified
// in the button string 's'
void IceWMClient::addClientButtons( const TQString& s )
{
if (!s.isEmpty())
for(unsigned int i = 0; i < s.length(); i++)
{
switch ( s[i].latin1() )
{
case 's':
// Create the menu icons, and render with the current mini-icon
// if explicitly requested by the theme.
if ( (validPixmaps(menuButtonPix) || showMenuButtonIcon) && !button[BtnSysMenu])
{
if (showMenuButtonIcon) {
renderMenuIcons();
button[BtnSysMenu] = new IceWMButton(this, "menu",
&menuButtonWithIconPix, false, i18n("Menu"), TQt::LeftButton|TQt::RightButton);
}
else
button[BtnSysMenu] = new IceWMButton(this, "menu",
&menuButtonPix, false, i18n("Menu"));
connect( button[BtnSysMenu], TQT_SIGNAL(pressed()),
this, TQT_SLOT(menuButtonPressed()));
connect( button[BtnSysMenu], TQT_SIGNAL(released()),
this, TQT_SLOT(menuButtonReleased()));
hb->addWidget( button[BtnSysMenu] );
}
break;
case 'x':
if ( validPixmaps(closePix) && !button[BtnClose] && isCloseable())
{
button[BtnClose] = new IceWMButton(this, "close",
&closePix, false, i18n("Close"));
hb->addWidget( button[BtnClose] );
connect( button[BtnClose], TQT_SIGNAL(clicked()),
this, TQT_SLOT(closeWindow()));
}
break;
case 'm':
if ( validPixmaps(maximizePix) && !button[BtnMaximize] && isMaximizable() )
{
button[BtnMaximize] = new IceWMButton(this, "maximize",
&maximizePix, false, i18n("Maximize"), TQt::LeftButton|TQt::MidButton|TQt::RightButton);
hb->addWidget( button[BtnMaximize] );
connect( button[BtnMaximize], TQT_SIGNAL(clicked()),
this, TQT_SLOT(slotMaximize()));
}
break;
case 'i':
if ( validPixmaps(minimizePix) && !button[BtnMinimize] &&
isMinimizable() )
{
button[BtnMinimize] = new IceWMButton(this, "minimize",
&minimizePix, false, i18n("Minimize"));
hb->addWidget( button[BtnMinimize] );
connect( button[BtnMinimize], TQT_SIGNAL(clicked()),
this, TQT_SLOT(minimize()));
}
break;
/* Not yet implemented - how's hide useful anyway?
case 'h':
if ( button[BtnHide] && !button[BtnHide] )
hb->addWidget( button[BtnHide] );
break; */
case 'r':
// NOTE: twin doesn't have toggleShade() in clients.h !
if ( validPixmaps(rollupPix) && !button[BtnRollup] )
{
button[BtnRollup] = new IceWMButton(this, "shade",
isSetShade() ? &rolldownPix : &rollupPix,
false, i18n("Rollup"));
hb->addWidget( button[BtnRollup] );
connect( button[BtnRollup], TQT_SIGNAL(clicked()),
this, TQT_SLOT(toggleShade()));
}
break;
case 'd':
// Make depth == on all desktops
if ( validPixmaps(depthPix) && !button[BtnDepth] )
{
button[BtnDepth] = new IceWMButton(this, "on_all_desktops",
&depthPix, true, isOnAllDesktops()?i18n("Not on all desktops"):i18n("On all desktops"));
button[BtnDepth]->turnOn( isOnAllDesktops() );
hb->addWidget( button[BtnDepth] );
connect( button[BtnDepth], TQT_SIGNAL(clicked()),
this, TQT_SLOT(toggleOnAllDesktops()));
}
break;
}
}
}
// Adds a pixmap to the titlebar layout via the use of a nice TQSpacerItem
TQSpacerItem* IceWMClient::addPixmapSpacer( TQPixmap* p[], TQSizePolicy::SizeType s, int hsize )
{
TQSpacerItem* sp;
// Add a null spacer for zero image
if ( p && p[Active] )
{
int w = (hsize == -1) ? p[Active]->width(): hsize;
sp = new TQSpacerItem( w, titleBarHeight, s, TQSizePolicy::Fixed );
}
else
sp = new TQSpacerItem(0, 0, TQSizePolicy::Maximum, TQSizePolicy::Fixed );
hb->addItem( sp );
return sp;
}
void IceWMClient::renderMenuIcons()
{
TQPixmap miniIcon( icon().pixmap( TQIconSet::Small, TQIconSet::Normal) );
if (!miniIcon.isNull())
for(int i = 0; i < 2; i++) {
if ( menuButtonWithIconPix[i] )
delete menuButtonWithIconPix[i];
// Try to be more friendly to dodgy themes - icewm assumes a square menu button
// but some pixmap themes don't provide a square menu button.
int w = titleBarHeight;
if (validPixmaps(menuButtonPix) && menuButtonPix[i]->width() > w)
w = menuButtonPix[i]->width();
menuButtonWithIconPix[i] = new TQPixmap(w, 2*titleBarHeight );
if (themeLook != WIN95)
menuButtonWithIconPix[i] -> fill((i==0) ? *colorInActiveButton : *colorActiveButton);
else
menuButtonWithIconPix[i] -> fill((i==0) ? *colorInActiveTitleBar : *colorActiveTitleBar);
TQPainter pnt( menuButtonWithIconPix[i] );
if (themeLook > 0 && themeLook != WIN95 && themeLook != WARP4) {
draw3DRect(pnt, *colorActiveButton, 0, 0,
w-1, titleBarHeight-1, true);
draw3DRect(pnt, *colorActiveButton, 0, titleBarHeight,
w-1, 2*titleBarHeight-1, false);
}
if (validPixmaps(menuButtonPix)) {
pnt.drawPixmap(0, 0, *menuButtonPix[i]);
}
int offset = (titleBarHeight - miniIcon.width())/2;
if (offset<0) offset = 0;
// Paint the mini icon over the menu pixmap in the centre
pnt.drawPixmap( offset, offset, miniIcon );
pnt.drawPixmap( offset, titleBarHeight+offset, miniIcon );
pnt.end();
}
}
void IceWMClient::slotMaximize()
{
maximize(button[BtnMaximize]->last_button);
}
void IceWMClient::toggleShade()
{
setShade(!isSetShade());
}
int IceWMClient::titleTextWidth( const TQString& s )
{
// Obtains the actual width of the text, using the titlebar font
TQSize size;
TQFontMetrics fm( options()->font(true) );
size = fm.size( 0, s );
return size.width();
}
void IceWMClient::borders(int& left, int& right, int& top, int& bottom) const
{
left = borderSizeX;
right = borderSizeX;
if( titleBarOnTop ) {
top = titleBarHeight + borderSizeY;
bottom = borderSizeY;
} else {
top = borderSizeY;
bottom = titleBarHeight + borderSizeY;
}
}
void IceWMClient::resize( const TQSize& s )
{
widget()->resize( s );
}
TQSize IceWMClient::minimumSize() const
{
return widget()->minimumSize();
}
// Repaint nicely upon resize to minimise flicker.
void IceWMClient::resizeEvent( TQResizeEvent* e )
{
calcHiddenButtons();
if (widget()->isVisibleToTLW())
{
widget()->update(widget()->rect());
int dx = 0;
int dy = 0;
if ( e->oldSize().width() != widget()->width() )
dx = 32 + TQABS( e->oldSize().width() - width() );
if ( e->oldSize().height() != height() )
dy = 8 + TQABS( e->oldSize().height() - height() );
if ( dy )
widget()->update( 0, height() - dy + 1, width(), dy );
if ( dx )
{
widget()->update( width() - dx + 1, 0, dx, height() );
widget()->update( TQRect( TQPoint(4,4), titlebar->geometry().bottomLeft() - TQPoint(1,0) ) );
widget()->update( TQRect( titlebar->geometry().topRight(), TQPoint( width() - 4, titlebar->geometry().bottom() ) ) );
widget()->repaint(titlebar->geometry(), false);
}
}
}
// IceWM Paint magic goes here.
void IceWMClient::paintEvent( TQPaintEvent* )
{
TQColor colorTitleShadow;
TQColor colorTitle;
TQColor c1;
int rx, rw;
TQPainter p( widget() );
int act = isActive() ? Active: InActive;
// Determine titlebar shadow colors
bool useShadow = isActive() ? useActiveShadow : useInActiveShadow;
if ( useShadow )
colorTitleShadow = isActive() ? *colorActiveTitleTextShadow : *colorInActiveTitleTextShadow;
if ( themeTitleTextColors )
colorTitle = isActive()? *colorActiveTitleBarText : *colorInActiveTitleBarText;
else
colorTitle = options()->color(ColorFont, isActive());
// Obtain widget bounds.
TQRect r;
r = widget()->rect();
int fillWidth = r.width() - 2*borderSizeX;
int y = r.y();
int x = r.x();
int w = r.width();
int h = r.height();
// Do we have pixmaps for the frame?
if (validframe)
{
// Top corner
p.drawPixmap(0, 0, *frameTL[ act ], 0, 0, cornerSizeX, borderSizeY);
p.drawPixmap(0, 0, *frameTL[ act ], 0, 0, borderSizeX, cornerSizeY);
// Top right corner
p.drawPixmap(w-cornerSizeX, 0, *frameTR[ act ],
frameTR[act]->width()-cornerSizeX, 0, cornerSizeX, borderSizeY);
p.drawPixmap(w-borderSizeX, 0, *frameTR[ act ],
frameTR[act]->width()-borderSizeX, 0, borderSizeX, cornerSizeY);
// Top bar
p.drawTiledPixmap( cornerSizeX, 0, w-(2*cornerSizeX), borderSizeY, *frameT[ act ] );
// Left bar
p.drawTiledPixmap( 0, cornerSizeY, borderSizeX, h-(2*cornerSizeY), *frameL[ act ] );
// Right bar
p.drawTiledPixmap( w-borderSizeX, cornerSizeY, borderSizeX, h-(2*cornerSizeY),
*frameR[ act ],frameR[act]->width()-borderSizeX );
// Bottom left corner
p.drawPixmap(0, h-borderSizeY, *frameBL[ act ],
0, frameBL[act]->height()-borderSizeY, cornerSizeX, borderSizeY);
p.drawPixmap(0, h-cornerSizeY, *frameBL[ act ],
0, frameBL[act]->height()-cornerSizeY, borderSizeX, cornerSizeY);
// Bottom right corner
p.drawPixmap(w-cornerSizeX, h-borderSizeY, *frameBR[ act ],
frameBR[act]->width()-cornerSizeX, frameBR[act]->height()-borderSizeY,
cornerSizeX, borderSizeY);
p.drawPixmap(w-borderSizeX, h-cornerSizeY, *frameBR[ act ],
frameBR[act]->width()-borderSizeX, frameBR[act]->height()-cornerSizeY,
borderSizeX, cornerSizeY);
// Bottom bar
p.drawTiledPixmap(cornerSizeX, h-borderSizeY, w-(2*cornerSizeX), borderSizeY,
*frameB[ act ], 0, frameB[ act ]->height()-borderSizeY );
// Ensure uncovered areas during shading are painted with something
p.setPen( *colorInActiveBorder );
if (titleBarOnTop)
p.drawLine( x+borderSizeX, y+h-borderSizeY-1,
x+w-borderSizeX-1, y+h-borderSizeY-1);
else
p.drawLine( x+borderSizeX, y+borderSizeY,
x+w-borderSizeX-1, y+borderSizeY);
} else
{
// Draw a stock IceWM frame instead of a pixmap frame
c1 = isActive() ? *colorActiveBorder : *colorInActiveBorder;
if (themeLook == WARP3 || themeLook == MOTIF) {
draw3DRect(p, c1, x, y, w-1, h-1, true);
p.setPen(c1);
p.drawRect(x+1, y+1, w-2, h-2);
} else {
p.setPen( c1.light(135) );
p.drawLine(0, 0, w-2, 0);
p.drawLine(0, 0, 0, h-2);
p.setPen(c1);
p.drawLine(1, 1, w-3, 1);
p.drawLine(1, 1, 1, h-3);
p.setPen( c1.dark(140) );
p.drawLine(1, h-2, w-2, h-2);
p.drawLine(w-2, 1, w-2, h-2);
p.setPen( TQt::black );
p.drawLine(w-1, 0, w-1, h-1);
p.drawLine(0, h-1, w-1, h-1);
}
// Fill frame border if required
if (borderSizeX > 2)
{
// Fill Vertical sizes
p.fillRect( x+2, y+2, borderSizeX-2, h-4, c1);
p.fillRect( w-borderSizeX, y+2, borderSizeX-2, h-4, c1);
}
if (borderSizeY > 2)
{
// Fill horizontal frame parts
p.fillRect( x+borderSizeX, y+2, fillWidth, borderSizeY-2, c1);
p.fillRect( x+borderSizeX, h-borderSizeY, fillWidth, borderSizeY-2, c1);
}
if (themeLook == WARP3 || themeLook == MOTIF) {
draw3DRect(p, c1, x+borderSizeX-1, y+borderSizeY-1,
w+1-2*borderSizeX, h+1-2*borderSizeY, false);
}
if (themeLook == MOTIF && !isShade()) {
int xext = titleBarHeight + borderSizeX - 1;
int yext = titleBarHeight + borderSizeY - 1;
int xext2 = w-xext-2;
int yext2 = h-yext-2;
int bX = w - borderSizeX-1;
int bY = h - borderSizeY-1;
p.setPen( c1.dark(140) );
p.drawLine(xext, 0, xext, borderSizeY);
p.drawLine(xext2, 0, xext2, borderSizeY);
p.drawLine(xext, bY, xext, h-1);
p.drawLine(xext2, bY, xext2, h-1);
p.drawLine(0, yext, borderSizeX, yext);
p.drawLine(0, yext2, borderSizeX, yext2);
p.drawLine(bX, yext, w-1, yext);
p.drawLine(bX, yext2, w-1, yext2);
p.setPen( c1.light(135) );
++xext; ++yext; ++xext2; ++yext2;
p.drawLine(xext, 0, xext, borderSizeY);
p.drawLine(xext2, 0, xext2, borderSizeY);
p.drawLine(xext, bY, xext, h-1);
p.drawLine(xext2, bY, xext2, h-1);
p.drawLine(0, yext, borderSizeX, yext);
p.drawLine(0, yext2, borderSizeX, yext2);
p.drawLine(bX, yext, w-1, yext);
p.drawLine(bX, yext2, w-1, yext2);
}
// Ensure uncovered areas during shading are painted with something
p.setPen( *colorInActiveBorder );
if (titleBarOnTop)
p.drawLine( x+borderSizeX, y+h-borderSizeY-1,
x+w-borderSizeX-1, y+h-borderSizeY-1);
else
p.drawLine( x+borderSizeX, y+borderSizeY,
x+w-borderSizeX-1, y+borderSizeY);
}
// Draw the title elements, if we need to draw a titlebar.
if (titleBarHeight > 0)
{
TQPixmap* titleBuffer = new TQPixmap( width()-(2*borderSizeX), titleBarHeight );
TQPainter p2( titleBuffer, this );
titleBuffer->fill( act ? *colorActiveTitleBar : *colorInActiveTitleBar );
r = titleSpacerJ->geometry();
if (!r.isEmpty() && titleJ[ act ])
p2.drawPixmap( r.x()-borderSizeX, 0, *titleJ[ act ]);
r = titleSpacerL->geometry();
if (!r.isEmpty() && titleL[ act ])
p2.drawPixmap( r.x()-borderSizeX, 0, *titleL[ act ]);
r = titleSpacerS->geometry();
if (!r.isEmpty() && titleS[ act ])
p2.drawTiledPixmap( r.x()-borderSizeX, 0, r.width(), titleBarHeight, *titleS[ act ]);
r = titleSpacerP->geometry();
if (!r.isEmpty() && titleP[ act ])
p2.drawPixmap( r.x()-borderSizeX, 0, *titleP[ act ]);
r = titlebar->geometry();
if (!r.isEmpty() && titleT[ act ] )
p2.drawTiledPixmap( r.x()-borderSizeX, 0, r.width(), titleBarHeight, *titleT[ act ]);
r = titleSpacerM->geometry();
if (!r.isEmpty() && titleM[ act ])
p2.drawPixmap( r.x()-borderSizeX, 0, *titleM[ act ], 0, 0, r.width(), r.height());
r = titleSpacerB->geometry();
if (!r.isEmpty() && titleB[ act ])
p2.drawTiledPixmap( r.x()-borderSizeX, 0, r.width(), titleBarHeight, *titleB[ act ]);
r = titleSpacerR->geometry();
if (!r.isEmpty() && titleR[ act ])
p2.drawPixmap( r.x()-borderSizeX, 0, *titleR[ act ], 0, 0, r.width(), r.height());
r = titleSpacerQ->geometry();
if (!r.isEmpty() && titleQ[ act ])
p2.drawPixmap( r.x()-borderSizeX, 0, *titleQ[ act ], 0, 0, r.width(), r.height());
p2.setFont( options()->font(true) );
// Pre-compute as much as possible
r = titlebar->geometry();
rx = r.x() - borderSizeX;
rw = width()-(2*borderSizeX)-r.x();
// Paint a title text shadow if requested
if ( useShadow )
{
p2.setPen( colorTitleShadow );
p2.drawText(rx+1, 1, rw, titleBarHeight, AlignLeft|AlignVCenter, caption());
}
// Draw the title text
p2.setPen( colorTitle );
p2.drawText(rx, 0, rw, titleBarHeight, AlignLeft|AlignVCenter, caption());
p2.end();
bitBlt( widget(), borderSizeX, hb->geometry().y(), titleBuffer );
delete titleBuffer;
}
}
void IceWMClient::showEvent(TQShowEvent *ev)
{
calcHiddenButtons();
titlebar->changeSize( titleTextWidth(caption()), titleBarHeight,
TQSizePolicy::Preferred, TQSizePolicy::Fixed );
grid->activate();
widget()->show();
IceWMClient::showEvent(ev);
}
void IceWMClient::mouseDoubleClickEvent( TQMouseEvent * e )
{
if( e->button() != TQt::LeftButton )
return;
TQRect r;
if (titleBarOnTop)
r.setRect( borderSizeX, borderSizeY, width()-(2*borderSizeX), titleBarHeight);
else
r.setRect( borderSizeX, height()-borderSizeY-titleBarHeight,
width()-(2*borderSizeX), titleBarHeight);
if (r.contains( e->pos() ) )
titlebarDblClickOperation();
}
void IceWMClient::wheelEvent(TQWheelEvent *e)
{
if (isSetShade() || TQRect( 0, 0, width(), titleBarHeight ).contains( e->pos() ) )
titlebarMouseWheelOperation( e->delta());
}
// Called via Client class when the miniIcon() changes
void IceWMClient::iconChange()
{
if (validPixmaps(menuButtonPix) && showMenuButtonIcon)
{
if (button[BtnSysMenu])
{
renderMenuIcons();
button[BtnSysMenu]->usePixmap( &menuButtonWithIconPix );
if (button[BtnSysMenu]->isVisible())
button[BtnSysMenu]->repaint(false);
}
}
}
void IceWMClient::desktopChange()
{
if (button[BtnDepth])
{
button[BtnDepth]->turnOn( isOnAllDesktops() );
button[BtnDepth]->repaint(false);
button[BtnDepth]->setTipText(isOnAllDesktops() ? i18n("Not on all desktops") : i18n("On all desktops"));
}
}
// Please don't modify the following unless you want layout problems
void IceWMClient::captionChange()
{
TQRect r( 0, borderSizeY, geometry().width(), titleBarHeight);
titlebar->changeSize( titleTextWidth( caption() ), titleBarHeight,
TQSizePolicy::Preferred, TQSizePolicy::Fixed );
titlebar->invalidate();
grid->activate();
widget()->repaint( r, false );
}
void IceWMClient::maximizeChange()
{
// Change the button pixmap to restore if required
if (button[BtnMaximize] && validPixmaps(restorePix))
{
button[BtnMaximize]->usePixmap( (maximizeMode()==MaximizeFull) ? &restorePix : &maximizePix );
button[BtnMaximize]->setTipText( (maximizeMode()==MaximizeFull) ? i18n("Restore") : i18n("Maximize"));
}
}
void IceWMClient::shadeChange()
{
// Change the button pixmap to rolldown if required
if (button[BtnRollup] && validPixmaps(rolldownPix))
{
button[BtnRollup]->usePixmap( isSetShade() ? &rolldownPix : &rollupPix );
button[BtnRollup]->setTipText( isSetShade() ? i18n("Rolldown") : i18n("Rollup"));
}
}
void IceWMClient::activeChange()
{
widget()->repaint(false);
// Reset the button pixmaps.
for(int i= IceWMClient::BtnSysMenu; i < IceWMClient::BtnCount; i++)
if(button[i])
button[i]->repaint( false );
}
// This does the showing / hiding button magic
// for variable positioned buttons.
void IceWMClient::calcHiddenButtons()
{
const int minwidth = 220; // Minimum width where all buttons are shown
const int btn_width = 20; // Average width
// Show/Hide buttons in this order - OnAllDesktops, Maximize, Menu, Rollup, Minimize, Close.
IceWMButton* btnArray[] = { button[BtnDepth], button[BtnMaximize], button[BtnSysMenu],
button[BtnRollup], button[BtnMinimize], button[BtnClose] };
int current_width = width();
int count = 0;
int i;
// Find out how many buttons we have to hide.
while (current_width < minwidth)
{
current_width += btn_width;
count++;
}
// Bound the number of buttons to hide
if (count > 6) count = 6;
// Hide the required buttons...
for(i = 0; i < count; i++)
{
if (btnArray[i] && btnArray[i]->isVisible() )
btnArray[i]->hide();
}
// Show the rest of the buttons...
for(i = count; i < 6; i++)
{
if (btnArray[i] && (!btnArray[i]->isVisible()) )
btnArray[i]->show();
}
}
// Mouse position code modified from that in workspace.cpp
IceWMClient::Position IceWMClient::mousePosition( const TQPoint& p ) const
{
int rangeX = cornerSizeX;
int rangeY = cornerSizeY;
int borderX = borderSizeX;
int borderY = borderSizeY;
Position m = PositionCenter;
if ((p.x() > borderX && p.x() < width() - borderX) &&
( p.y() > borderY && p.y() < height() - borderY))
return PositionCenter;
if ( p.y() <= rangeY && p.x() <= rangeX)
m = PositionTopLeft;
else if ( p.y() >= height()-rangeY && p.x() >= width()-rangeX)
m = PositionBottomRight;
else if ( p.y() >= height()-rangeX && p.x() <= rangeX)
m = PositionBottomLeft;
else if ( p.y() <= rangeY && p.x() >= width()-rangeX)
m = PositionTopRight;
else if ( p.y() <= borderY )
m = PositionTop;
else if ( p.y() >= height()-borderY )
m = PositionBottom;
else if ( p.x() <= borderX )
m = PositionLeft;
else if ( p.x() >= width()-borderX )
m = PositionRight;
else
m = PositionCenter;
return m;
}
void IceWMClient::menuButtonPressed()
{
static TQTime t;
static IceWMClient* lastClient = NULL;
bool dbl = ( lastClient == this && t.elapsed() <= TQApplication::doubleClickInterval());
lastClient = this;
t.start();
if (dbl)
{
m_closing = true;
return;
}
TQPoint menuPoint ( button[BtnSysMenu]->rect().bottomLeft() );
// Move to right if menu on rhs, otherwise on left
// and make this depend on windowWrapper(), not button.
KDecorationFactory* f = factory();
showWindowMenu( button[BtnSysMenu]->mapToGlobal(menuPoint) );
if( !f->exists( this )) // 'this' was deleted
return;
button[BtnSysMenu]->setDown(false);
}
void IceWMClient::menuButtonReleased()
{
if (m_closing)
closeWindow();
}
bool IceWMClient::eventFilter( TQObject* o, TQEvent* e )
{
if( o != widget())
return false;
switch( e->type())
{
case TQEvent::Resize:
resizeEvent(static_cast<TQResizeEvent*>( e ) );
return true;
case TQEvent::Paint:
paintEvent(static_cast<TQPaintEvent*>( e ) );
return true;
case TQEvent::MouseButtonDblClick:
mouseDoubleClickEvent(static_cast<TQMouseEvent*>( e ) );
return true;
case TQEvent::MouseButtonPress:
processMousePressEvent(static_cast<TQMouseEvent*>( e ) );
return true;
case TQEvent::Wheel:
wheelEvent( static_cast<TQWheelEvent*>( e ));
return true;
default:
break;
}
return false;
}
}
extern "C"
{
KDE_EXPORT KDecorationFactory *create_factory()
{
IceWM::clientHandler = new IceWM::ThemeHandler;
return IceWM::clientHandler;
}
}
#include "icewm.moc"