kwin/clients/riscos/Manager.cpp
Rik Hemsley 5cd78ffdd3 Fix mouseposition
svn path=/trunk/kdebase/kwin/; revision=134762
2002-01-31 02:37:02 +00:00

760 lines
14 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)
{
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 (iconify)
p.setClipRegion(
QRegion(workspace()->desktopWidget()->rect()) - wingeom
);
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 QHBoxLayout(topLayout_);
titleLayout_->setResizeMode(QLayout::FreeResize);
createTitle();
QHBoxLayout * midLayout = new QHBoxLayout(topLayout_);
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"