clients by adding an unnecessary 'enable-bidi' checkbox which was never agreed upon on the kwin mailing list. The use of --reverse is sufficient without needing this checkbox. Please discuss largish patches like these first, especially during a feature freeze! (Everyone also seems to forget the kwin-styles in kdeartwork) - Make clients look the same with --reverse instead of the broken state they were in previously. If the button ordering is not to your liking in reverse mode, just change the button positions. (Or send patches for those clients that have not got these features yet). - Clean up popupMenu handling somewhat. - Remove a global static object in keramik. svn path=/trunk/kdebase/kwin/; revision=174079
764 lines
15 KiB
C++
764 lines
15 KiB
C++
/*
|
|
RISC OS KWin client
|
|
|
|
Copyright 2000
|
|
Rik Hemsley <rik@kde.org>
|
|
|
|
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., 59 Temple Place - Suite 330,
|
|
Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include <unistd.h> // for usleep
|
|
#include <config.h> // for usleep on non-linux platforms
|
|
#include <math.h> // for sin and cos
|
|
|
|
#include <qpainter.h>
|
|
#include <qimage.h>
|
|
#include <qlayout.h>
|
|
#include <qvariant.h>
|
|
|
|
#include <kapplication.h>
|
|
|
|
#include "../../options.h"
|
|
#include "../../workspace.h"
|
|
|
|
#include <netwm.h>
|
|
|
|
#include "Manager.h"
|
|
#include "Static.h"
|
|
#include "LowerButton.h"
|
|
#include "CloseButton.h"
|
|
#include "IconifyButton.h"
|
|
#include "MaximiseButton.h"
|
|
#include "StickyButton.h"
|
|
#include "HelpButton.h"
|
|
|
|
extern "C"
|
|
{
|
|
KWinInternal::Client *
|
|
allocate(KWinInternal::Workspace * workSpace, WId winId, int tool)
|
|
{
|
|
if (tool)
|
|
return new RiscOS::ToolManager(workSpace, winId, 0, "ToolManager");
|
|
else
|
|
return new RiscOS::Manager(workSpace, winId, 0, "Manager");
|
|
}
|
|
|
|
void
|
|
init()
|
|
{
|
|
(void) RiscOS::Static::instance();
|
|
}
|
|
|
|
void
|
|
reset()
|
|
{
|
|
RiscOS::Static::instance()->reset();
|
|
}
|
|
|
|
void
|
|
deinit()
|
|
{
|
|
delete RiscOS::Static::instance();
|
|
}
|
|
|
|
}
|
|
|
|
using namespace KWinInternal;
|
|
|
|
namespace RiscOS
|
|
{
|
|
|
|
|
|
Manager::Manager
|
|
(
|
|
KWinInternal::Workspace * workSpace,
|
|
WId id,
|
|
QWidget * parent,
|
|
const char * name
|
|
)
|
|
: KWinInternal::Client(workSpace, id, parent, name),
|
|
topLayout_ (0L),
|
|
titleLayout_ (0L),
|
|
titleSpacer_ (0L)
|
|
{
|
|
setBackgroundMode(NoBackground);
|
|
|
|
leftButtonList_.setAutoDelete(true);
|
|
rightButtonList_.setAutoDelete(true);
|
|
|
|
resetLayout();
|
|
|
|
connect(options, SIGNAL(resetClients()), this, SLOT(slotReset()));
|
|
}
|
|
|
|
Manager::~Manager()
|
|
{
|
|
}
|
|
|
|
void
|
|
Manager::paintEvent(QPaintEvent * e)
|
|
{
|
|
QPainter p(this);
|
|
|
|
QRect r(e->rect());
|
|
|
|
bool intersectsLeft =
|
|
r.intersects(QRect(0, 0, 1, height()));
|
|
|
|
bool intersectsRight =
|
|
r.intersects(QRect(width() - 1, 0, width(), height()));
|
|
|
|
if (intersectsLeft || intersectsRight) {
|
|
|
|
p.setPen(Qt::black);
|
|
|
|
if (intersectsLeft)
|
|
p.drawLine(0, r.top(), 0, r.bottom());
|
|
|
|
if (intersectsRight)
|
|
p.drawLine(width() - 1, r.top(), width() - 1, r.bottom());
|
|
}
|
|
|
|
Static * s = Static::instance();
|
|
|
|
bool active = isActive();
|
|
|
|
// Title bar.
|
|
|
|
QRect tr = titleSpacer_->geometry();
|
|
bitBlt(this, tr.topLeft(), &titleBuf_);
|
|
|
|
// Resize bar.
|
|
|
|
if (isResizable())
|
|
{
|
|
int rbt = height() - Static::instance()->resizeHeight(); // Resize bar top.
|
|
|
|
bitBlt(this, 0, rbt, &(s->resize(active)));
|
|
bitBlt(this, 30, rbt, &(s->resizeMidLeft(active)));
|
|
|
|
p.drawTiledPixmap(
|
|
32,
|
|
rbt,
|
|
width() - 34,
|
|
Static::instance()->resizeHeight(),
|
|
s->resizeMidMid(active)
|
|
);
|
|
|
|
bitBlt(this, width() - 32, rbt, &(s->resizeMidRight(active)));
|
|
bitBlt(this, width() - 30, rbt, &(s->resize(active)));
|
|
}
|
|
else
|
|
{
|
|
p.drawLine(1, height() - 1, width() - 2, height() - 1);
|
|
}
|
|
}
|
|
|
|
void
|
|
Manager::resizeEvent(QResizeEvent * e)
|
|
{
|
|
KWinInternal::Client::resizeEvent(e);
|
|
updateButtonVisibility();
|
|
updateTitleBuffer();
|
|
repaint();
|
|
}
|
|
|
|
void
|
|
Manager::updateButtonVisibility()
|
|
{
|
|
#if 0
|
|
int sizeProblem = 0;
|
|
|
|
if (width() < 80) sizeProblem = 3;
|
|
else if (width() < 100) sizeProblem = 2;
|
|
else if (width() < 180) sizeProblem = 1;
|
|
|
|
switch (sizeProblem) {
|
|
|
|
case 1:
|
|
lower_ ->hide();
|
|
sticky_ ->hide();
|
|
help_ ->hide();
|
|
iconify_ ->show();
|
|
maximise_ ->hide();
|
|
close_ ->show();
|
|
break;
|
|
|
|
case 2:
|
|
lower_ ->hide();
|
|
sticky_ ->hide();
|
|
help_ ->hide();
|
|
iconify_ ->hide();
|
|
maximise_ ->hide();
|
|
close_ ->show();
|
|
break;
|
|
|
|
case 3:
|
|
lower_ ->hide();
|
|
sticky_ ->hide();
|
|
help_ ->hide();
|
|
iconify_ ->hide();
|
|
maximise_ ->hide();
|
|
close_ ->hide();
|
|
break;
|
|
|
|
case 0:
|
|
default:
|
|
lower_ ->show();
|
|
sticky_ ->show();
|
|
if (providesContextHelp())
|
|
help_->show();
|
|
iconify_ ->show();
|
|
maximise_ ->show();
|
|
close_ ->show();
|
|
break;
|
|
}
|
|
|
|
layout()->activate();
|
|
#endif
|
|
}
|
|
|
|
void
|
|
Manager::updateTitleBuffer()
|
|
{
|
|
bool active = isActive();
|
|
|
|
Static * s = Static::instance();
|
|
|
|
QRect tr = titleSpacer_->geometry();
|
|
|
|
if (tr.width() == 0 || tr.height() == 0)
|
|
titleBuf_.resize(8, 8);
|
|
else
|
|
titleBuf_.resize(tr.size());
|
|
|
|
QPainter p(&titleBuf_);
|
|
|
|
p.drawPixmap(0, 0, s->titleTextLeft(active));
|
|
|
|
p.drawTiledPixmap(
|
|
3,
|
|
0,
|
|
tr.width() - 6,
|
|
Static::instance()->titleHeight(),
|
|
s->titleTextMid(active)
|
|
);
|
|
|
|
p.setPen(options->color(Options::Font, active));
|
|
|
|
p.setFont(options->font(true)); // XXX false doesn't work right at the moment
|
|
|
|
p.drawText(
|
|
4,
|
|
2,
|
|
tr.width() - 8,
|
|
Static::instance()->titleHeight() - 4,
|
|
AlignCenter, caption()
|
|
);
|
|
|
|
p.drawPixmap(tr.width() - 3, 0, s->titleTextRight(active));
|
|
}
|
|
|
|
KWinInternal::Client::MousePosition
|
|
Manager::mousePosition(const QPoint & p) const
|
|
{
|
|
MousePosition m = Nowhere;
|
|
|
|
// Look out for off-by-one errors here.
|
|
|
|
if (isResizable())
|
|
{
|
|
if (p.y() > (height() - (Static::instance()->resizeHeight() + 1)))
|
|
{
|
|
// Keep order !
|
|
|
|
if (p.x() >= (width() - 30))
|
|
m = BottomRight;
|
|
else if (p.x() <= 30)
|
|
m = BottomLeft;
|
|
else
|
|
m = Bottom;
|
|
}
|
|
else
|
|
{
|
|
m = Center;
|
|
//Client::mousePosition(p);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m = Client::mousePosition(p);
|
|
m = Center;
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
void
|
|
Manager::mouseDoubleClickEvent(QMouseEvent * e)
|
|
{
|
|
if (titleSpacer_->geometry().contains(e->pos()))
|
|
workspace()
|
|
->performWindowOperation(this, options->operationTitlebarDblClick());
|
|
}
|
|
|
|
void
|
|
Manager::slotReset()
|
|
{
|
|
resetLayout();
|
|
}
|
|
|
|
void
|
|
Manager::captionChange(const QString &)
|
|
{
|
|
updateTitleBuffer();
|
|
repaint();
|
|
}
|
|
|
|
void
|
|
Manager::paletteChange(const QPalette &)
|
|
{
|
|
slotReset();
|
|
}
|
|
|
|
void
|
|
Manager::activeChange(bool b)
|
|
{
|
|
emit(activeChanged(b));
|
|
updateTitleBuffer();
|
|
repaint();
|
|
}
|
|
|
|
void
|
|
Manager::maximizeChange(bool b)
|
|
{
|
|
emit(maximiseChanged(b));
|
|
}
|
|
|
|
void
|
|
Manager::stickyChange(bool b)
|
|
{
|
|
emit(stickyChanged(b));
|
|
}
|
|
|
|
void
|
|
Manager::slotLower()
|
|
{
|
|
workspace()->lowerClient(this);
|
|
}
|
|
|
|
void
|
|
Manager::slotRaise()
|
|
{
|
|
workspace()->raiseClient(this);
|
|
}
|
|
|
|
void
|
|
Manager::slotMax()
|
|
{
|
|
maximize(MaximizeFull);
|
|
}
|
|
|
|
void
|
|
Manager::slotVMax()
|
|
{
|
|
maximize(MaximizeVertical);
|
|
}
|
|
|
|
void
|
|
Manager::slotSetSticky(bool b)
|
|
{
|
|
setSticky(b);
|
|
}
|
|
|
|
void
|
|
Manager::slotHelp()
|
|
{
|
|
contextHelp();
|
|
}
|
|
|
|
void
|
|
Manager::animateIconifyOrDeiconify(bool iconify)
|
|
{
|
|
if (!options->animateMinimize)
|
|
return;
|
|
animate(iconify, Static::instance()->animationStyle());
|
|
}
|
|
|
|
void Manager::animate(bool iconify, int style)
|
|
{
|
|
switch (style) {
|
|
|
|
case 1:
|
|
{
|
|
// Double twisting double back, with pike ;)
|
|
|
|
if (!iconify) // No animation for restore.
|
|
return;
|
|
|
|
// Go away quick.
|
|
hide();
|
|
qApp->syncX();
|
|
|
|
NETRect r = netWinInfo()->iconGeometry();
|
|
|
|
if (!QRect(r.pos.x, r.pos.y, r.size.width, r.size.height).isValid())
|
|
return;
|
|
|
|
// Algorithm taken from Window Maker (http://www.windowmaker.org)
|
|
|
|
int sx = x();
|
|
int sy = y();
|
|
int sw = width();
|
|
int sh = height();
|
|
int dx = r.pos.x;
|
|
int dy = r.pos.y;
|
|
int dw = r.size.width;
|
|
int dh = r.size.height;
|
|
|
|
double steps = 12;
|
|
|
|
double xstep = double((dx-sx)/steps);
|
|
double ystep = double((dy-sy)/steps);
|
|
double wstep = double((dw-sw)/steps);
|
|
double hstep = double((dh-sh)/steps);
|
|
|
|
double cx = sx;
|
|
double cy = sy;
|
|
double cw = sw;
|
|
double ch = sh;
|
|
|
|
double finalAngle = 3.14159265358979323846;
|
|
|
|
double delta = finalAngle / steps;
|
|
|
|
QPainter p(workspace()->desktopWidget());
|
|
p.setRasterOp(Qt::NotROP);
|
|
|
|
for (double angle = 0; ; angle += delta) {
|
|
|
|
if (angle > finalAngle)
|
|
angle = finalAngle;
|
|
|
|
double dx = (cw / 10) - ((cw / 5) * sin(angle));
|
|
double dch = (ch / 2) * cos(angle);
|
|
double midy = cy + (ch / 2);
|
|
|
|
QPoint p1(int(cx + dx), int(midy - dch));
|
|
QPoint p2(int(cx + cw - dx), p1.y());
|
|
QPoint p3(int(cx + dw + dx), int(midy + dch));
|
|
QPoint p4(int(cx - dx), p3.y());
|
|
|
|
XGrabServer(qt_xdisplay());
|
|
|
|
p.drawLine(p1, p2);
|
|
p.drawLine(p2, p3);
|
|
p.drawLine(p3, p4);
|
|
p.drawLine(p4, p1);
|
|
|
|
p.flush();
|
|
|
|
usleep(500);
|
|
|
|
p.drawLine(p1, p2);
|
|
p.drawLine(p2, p3);
|
|
p.drawLine(p3, p4);
|
|
p.drawLine(p4, p1);
|
|
|
|
XUngrabServer(qt_xdisplay());
|
|
|
|
kapp->processEvents();
|
|
|
|
cx += xstep;
|
|
cy += ystep;
|
|
cw += wstep;
|
|
ch += hstep;
|
|
|
|
if (angle >= finalAngle)
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 2:
|
|
{
|
|
// KVirc style ? Maybe. For qwertz.
|
|
|
|
if (!iconify) // No animation for restore.
|
|
return;
|
|
|
|
// Go away quick.
|
|
hide();
|
|
qApp->syncX();
|
|
|
|
int stepCount = 12;
|
|
|
|
QRect r(geometry());
|
|
|
|
int dx = r.width() / (stepCount * 2);
|
|
int dy = r.height() / (stepCount * 2);
|
|
|
|
QPainter p(workspace()->desktopWidget());
|
|
p.setRasterOp(Qt::NotROP);
|
|
|
|
for (int step = 0; step < stepCount; step++) {
|
|
|
|
r.moveBy(dx, dy);
|
|
r.setWidth(r.width() - 2 * dx);
|
|
r.setHeight(r.height() - 2 * dy);
|
|
|
|
XGrabServer(qt_xdisplay());
|
|
|
|
p.drawRect(r);
|
|
p.flush();
|
|
usleep(200);
|
|
p.drawRect(r);
|
|
|
|
XUngrabServer(qt_xdisplay());
|
|
|
|
kapp->processEvents();
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
default:
|
|
{
|
|
NETRect r = netWinInfo()->iconGeometry();
|
|
|
|
QRect icongeom(r.pos.x, r.pos.y, r.size.width, r.size.height);
|
|
|
|
if (!icongeom.isValid())
|
|
return;
|
|
|
|
QRect wingeom(x(), y(), width(), height());
|
|
|
|
QPainter p(workspace()->desktopWidget());
|
|
|
|
p.setRasterOp(Qt::NotROP);
|
|
|
|
#if 0
|
|
if (iconify)
|
|
p.setClipRegion(
|
|
QRegion(workspace()->desktopWidget()->rect()) - wingeom
|
|
);
|
|
#endif
|
|
|
|
XGrabServer(qt_xdisplay());
|
|
|
|
p.drawLine(wingeom.bottomRight(), icongeom.bottomRight());
|
|
p.drawLine(wingeom.bottomLeft(), icongeom.bottomLeft());
|
|
p.drawLine(wingeom.topLeft(), icongeom.topLeft());
|
|
p.drawLine(wingeom.topRight(), icongeom.topRight());
|
|
|
|
p.flush();
|
|
|
|
qApp->syncX();
|
|
|
|
usleep(30000);
|
|
|
|
p.drawLine(wingeom.bottomRight(), icongeom.bottomRight());
|
|
p.drawLine(wingeom.bottomLeft(), icongeom.bottomLeft());
|
|
p.drawLine(wingeom.topLeft(), icongeom.topLeft());
|
|
p.drawLine(wingeom.topRight(), icongeom.topRight());
|
|
|
|
XUngrabServer(qt_xdisplay());
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
Manager::createTitle()
|
|
{
|
|
leftButtonList_.clear();
|
|
rightButtonList_.clear();
|
|
|
|
QString buttons;
|
|
|
|
if (options->customButtonPositions())
|
|
buttons = options->titleButtonsLeft() + "|" + options->titleButtonsRight();
|
|
else
|
|
buttons = "XSH|IA";
|
|
|
|
|
|
QPtrList<Button> *buttonList = &leftButtonList_;
|
|
|
|
for (unsigned int i = 0; i < buttons.length(); ++i)
|
|
{
|
|
Button * tb = 0;
|
|
|
|
switch (buttons[i].latin1())
|
|
{
|
|
case 'S': // Sticky
|
|
tb = createButton(Button::Sticky, this);
|
|
break;
|
|
|
|
case 'H': // Help
|
|
tb = createButton(Button::Help, this);
|
|
break;
|
|
|
|
case 'I': // Minimize
|
|
tb = createButton(Button::Iconify, this);
|
|
break;
|
|
|
|
case 'A': // Maximize
|
|
tb = createButton(Button::Maximise, this);
|
|
break;
|
|
|
|
case 'X': // Close
|
|
tb = createButton(Button::Close, this);
|
|
break;
|
|
|
|
case '|':
|
|
buttonList = &rightButtonList_;
|
|
break;
|
|
}
|
|
|
|
if (0 != tb)
|
|
buttonList->append(tb);
|
|
}
|
|
|
|
for (QPtrListIterator<Button> it(leftButtonList_); it.current(); ++it)
|
|
{
|
|
it.current()->setAlignment(Button::Left);
|
|
titleLayout_->addWidget(it.current());
|
|
}
|
|
|
|
titleSpacer_ =
|
|
new QSpacerItem
|
|
(
|
|
0,
|
|
Static::instance()->titleHeight(),
|
|
QSizePolicy::Expanding,
|
|
QSizePolicy::Fixed
|
|
);
|
|
|
|
titleLayout_->addItem(titleSpacer_);
|
|
|
|
for (QPtrListIterator<Button> it(rightButtonList_); it.current(); ++it)
|
|
{
|
|
it.current()->setAlignment(Button::Right);
|
|
titleLayout_->addWidget(it.current());
|
|
}
|
|
}
|
|
|
|
Button *
|
|
Manager::createButton(int type, QWidget * parent)
|
|
{
|
|
Button * b = 0;
|
|
|
|
switch (Button::Type(type))
|
|
{
|
|
case Button::Help:
|
|
|
|
if (providesContextHelp())
|
|
{
|
|
b = new HelpButton(parent);
|
|
}
|
|
|
|
break;
|
|
|
|
case Button::Sticky:
|
|
|
|
b = new StickyButton(parent);
|
|
|
|
emit(stickyChanged(isSticky()));
|
|
|
|
break;
|
|
|
|
case Button::Iconify:
|
|
|
|
if (isMinimizable())
|
|
{
|
|
b = new IconifyButton(parent);
|
|
}
|
|
|
|
break;
|
|
|
|
case Button::Maximise:
|
|
|
|
if (isMaximizable())
|
|
{
|
|
b = new MaximiseButton(parent);
|
|
|
|
emit(maximiseChanged(isMaximized()));
|
|
}
|
|
|
|
break;
|
|
|
|
case Button::Close:
|
|
|
|
b = new CloseButton(parent);
|
|
|
|
break;
|
|
|
|
case Button::Lower:
|
|
|
|
b = new LowerButton(parent);
|
|
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return b;
|
|
}
|
|
|
|
void
|
|
Manager::resetLayout()
|
|
{
|
|
delete topLayout_;
|
|
topLayout_ = new QVBoxLayout(this, 0, 0);
|
|
topLayout_->setResizeMode(QLayout::FreeResize);
|
|
|
|
titleLayout_ = new QBoxLayout(topLayout_, QBoxLayout::LeftToRight, 0, 0);
|
|
titleLayout_->setResizeMode(QLayout::FreeResize);
|
|
|
|
createTitle();
|
|
|
|
QBoxLayout * midLayout = new QBoxLayout(topLayout_, QBoxLayout::LeftToRight, 0, 0);
|
|
midLayout->setResizeMode(QLayout::FreeResize);
|
|
midLayout->addSpacing(1);
|
|
midLayout->addWidget(windowWrapper());
|
|
midLayout->addSpacing(1);
|
|
|
|
if (isResizable())
|
|
topLayout_->addSpacing(Static::instance()->resizeHeight());
|
|
else
|
|
topLayout_->addSpacing(1);
|
|
}
|
|
|
|
ToolManager::ToolManager(
|
|
KWinInternal::Workspace * workSpace,
|
|
WId id,
|
|
QWidget * parent,
|
|
const char * name
|
|
)
|
|
: Manager(workSpace, id, parent, name)
|
|
{
|
|
}
|
|
|
|
ToolManager::~ToolManager()
|
|
{
|
|
}
|
|
|
|
|
|
} // End namespace
|
|
|
|
// vim:ts=2:sw=2:tw=78
|
|
#include "Manager.moc"
|