kwin/clients/riscos/Manager.cpp
Diego Iastrubni 27998b71ac dded a new checkbox to kcontrol-> wm -> decorations ->general. By defaut this checkbox is on.
if it's on, kwin clients will reverse thire display when the desktop is reversed (arabic, hebrew etc).

Clinets updated to use this options, and options.h has a few more lines :)
see clients for the changes.

svn path=/trunk/kdebase/kwin/; revision=173581
2002-08-24 12:02:26 +00:00

779 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_;
unsigned int str_len = buttons.length();
for (unsigned int i = 0; i < str_len; ++i)
{
Button * tb = 0;
switch ((QApplication::reverseLayout() && (!options->reverseBIDIWindows()))?
buttons[str_len-i-1].latin1():
buttons[i].latin1()
)
//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(QApplication::reverseLayout()?rightButtonList_:leftButtonList_);
it(leftButtonList_);
it.current(); ++it)
{
// it.current()->setAlignment(Button::Left);
it.current()->setAlignment(
QApplication::reverseLayout() && (!options->reverseBIDIWindows())?
Button::Right:Button::Left);
titleLayout_->addWidget(it.current());
}
titleSpacer_ =
new QSpacerItem
(
0,
Static::instance()->titleHeight(),
QSizePolicy::Expanding,
QSizePolicy::Fixed
);
titleLayout_->addItem(titleSpacer_);
for (QPtrListIterator<Button>
// it(QApplication::reverseLayout()?leftButtonList_:rightButtonList_);
it(rightButtonList_);
it.current(); ++it)
{
it.current()->setAlignment(
QApplication::reverseLayout() && (!options->reverseBIDIWindows())?
Button::Left: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"