2005-07-05 22:14:24 +04:00
|
|
|
/*
|
2006-02-05 18:58:29 +03:00
|
|
|
* Copyright 2001-2006, Haiku.
|
2005-07-05 22:14:24 +04:00
|
|
|
* Distributed under the terms of the MIT License.
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Adrian Oanca <adioanca@cotty.iren.ro>
|
|
|
|
* Stephan Aßmus <superstippi@gmx.de>
|
|
|
|
* Axel Dörfler, axeld@pinc-software.de
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** Class used to encapsulate desktop management */
|
|
|
|
|
|
|
|
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
#include "Desktop.h"
|
2005-07-05 22:14:24 +04:00
|
|
|
|
2004-01-22 03:32:07 +03:00
|
|
|
#include "AppServer.h"
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
#include "DesktopSettingsPrivate.h"
|
2005-11-04 18:54:16 +03:00
|
|
|
#include "DrawingEngine.h"
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
#include "HWInterface.h"
|
|
|
|
#include "InputManager.h"
|
|
|
|
#include "ServerApp.h"
|
2004-01-22 03:32:07 +03:00
|
|
|
#include "ServerConfig.h"
|
2006-02-05 22:43:44 +03:00
|
|
|
#include "ServerCursor.h"
|
2004-01-22 03:32:07 +03:00
|
|
|
#include "ServerScreen.h"
|
|
|
|
#include "ServerWindow.h"
|
2006-01-15 22:36:16 +03:00
|
|
|
#include "WindowPrivate.h"
|
2005-11-24 20:45:26 +03:00
|
|
|
#include "WindowLayer.h"
|
2004-01-12 01:12:55 +03:00
|
|
|
#include "Workspace.h"
|
2005-12-09 19:20:01 +03:00
|
|
|
#include "WorkspacesLayer.h"
|
2005-07-17 20:25:48 +04:00
|
|
|
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
#include <WindowInfo.h>
|
|
|
|
#include <ServerProtocol.h>
|
2003-02-12 14:24:26 +03:00
|
|
|
|
2006-02-09 18:07:56 +03:00
|
|
|
#include <DirectWindow.h>
|
2005-11-18 16:51:32 +03:00
|
|
|
#include <Entry.h>
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
#include <Message.h>
|
|
|
|
#include <MessageFilter.h>
|
|
|
|
#include <Region.h>
|
2005-06-24 03:46:17 +04:00
|
|
|
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
#include <stdio.h>
|
2005-06-24 03:46:17 +04:00
|
|
|
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
#if TEST_MODE
|
|
|
|
# include "EventStream.h"
|
|
|
|
#endif
|
2005-06-24 03:46:17 +04:00
|
|
|
|
2004-01-22 03:32:07 +03:00
|
|
|
//#define DEBUG_DESKTOP
|
|
|
|
#ifdef DEBUG_DESKTOP
|
2005-07-05 22:38:12 +04:00
|
|
|
# define STRACE(a) printf(a)
|
2004-01-22 03:32:07 +03:00
|
|
|
#else
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
# define STRACE(a) ;
|
2004-01-22 03:32:07 +03:00
|
|
|
#endif
|
2003-02-14 14:04:01 +03:00
|
|
|
|
2006-04-07 23:14:25 +04:00
|
|
|
#if !USE_MULTI_LOCKER
|
|
|
|
# define AutoWriteLocker BAutolock
|
|
|
|
#endif
|
2005-04-25 18:14:09 +04:00
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
class KeyboardFilter : public EventFilter {
|
2005-11-18 16:51:32 +03:00
|
|
|
public:
|
|
|
|
KeyboardFilter(Desktop* desktop);
|
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
virtual filter_result Filter(BMessage* message, EventTarget** _target,
|
2006-01-11 15:25:06 +03:00
|
|
|
int32* _viewToken, BMessage* latestMouseMoved);
|
2006-04-16 18:16:31 +04:00
|
|
|
virtual void RemoveTarget(EventTarget* target);
|
2005-11-18 16:51:32 +03:00
|
|
|
|
|
|
|
private:
|
2006-02-15 21:38:39 +03:00
|
|
|
void _UpdateFocus(int32 key, EventTarget** _target);
|
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
Desktop* fDesktop;
|
|
|
|
EventTarget* fLastFocus;
|
|
|
|
bigtime_t fTimestamp;
|
2005-11-18 16:51:32 +03:00
|
|
|
};
|
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
class MouseFilter : public EventFilter {
|
|
|
|
public:
|
|
|
|
MouseFilter(Desktop* desktop);
|
|
|
|
|
|
|
|
virtual filter_result Filter(BMessage* message, EventTarget** _target,
|
2006-01-11 15:25:06 +03:00
|
|
|
int32* _viewToken, BMessage* latestMouseMoved);
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
|
|
|
private:
|
|
|
|
Desktop* fDesktop;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
2005-11-18 16:51:32 +03:00
|
|
|
|
|
|
|
KeyboardFilter::KeyboardFilter(Desktop* desktop)
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
:
|
|
|
|
fDesktop(desktop),
|
|
|
|
fLastFocus(NULL),
|
|
|
|
fTimestamp(0)
|
2005-11-18 16:51:32 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-15 21:38:39 +03:00
|
|
|
void
|
|
|
|
KeyboardFilter::_UpdateFocus(int32 key, EventTarget** _target)
|
2005-11-18 16:51:32 +03:00
|
|
|
{
|
2006-02-15 21:38:39 +03:00
|
|
|
if (!fDesktop->LockSingleWindow())
|
|
|
|
return;
|
2005-11-18 16:51:32 +03:00
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
bigtime_t now = system_time();
|
|
|
|
|
|
|
|
EventTarget* focus = NULL;
|
2005-12-08 15:41:19 +03:00
|
|
|
if (fDesktop->FocusWindow() != NULL)
|
|
|
|
focus = &fDesktop->FocusWindow()->EventTarget();
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
|
|
|
// TODO: this is a try to not steal focus from the current window
|
|
|
|
// in case you enter some text and a window pops up you haven't
|
|
|
|
// triggered yourself (like a pop-up window in your browser while
|
|
|
|
// you're typing a password in another window) - maybe this should
|
|
|
|
// be done differently, though (using something like B_LOCK_WINDOW_FOCUS)
|
2005-11-29 20:00:08 +03:00
|
|
|
// (at least B_WINDOW_ACTIVATED must be postponed)
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
2006-04-16 18:16:31 +04:00
|
|
|
if (fLastFocus == NULL || (focus != fLastFocus && now - fTimestamp > 100000)) {
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
// if the time span between the key presses is very short
|
|
|
|
// we keep our previous focus alive - this is save even
|
|
|
|
// if the target doesn't exist anymore, as we don't reset
|
|
|
|
// it, and the event focus passed in is always valid (or NULL)
|
|
|
|
*_target = focus;
|
|
|
|
fLastFocus = focus;
|
2006-02-15 21:38:39 +03:00
|
|
|
}
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
fDesktop->UnlockSingleWindow();
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
|
|
|
// we always allow to switch focus after the enter key has pressed
|
|
|
|
if (key == B_ENTER)
|
|
|
|
fTimestamp = 0;
|
|
|
|
else
|
|
|
|
fTimestamp = now;
|
2006-02-15 21:38:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
filter_result
|
|
|
|
KeyboardFilter::Filter(BMessage* message, EventTarget** _target,
|
|
|
|
int32* /*_viewToken*/, BMessage* /*latestMouseMoved*/)
|
|
|
|
{
|
|
|
|
int32 key = 0;
|
|
|
|
int32 modifiers;
|
|
|
|
|
|
|
|
if (message->what == B_KEY_DOWN
|
|
|
|
&& message->FindInt32("key", &key) == B_OK
|
|
|
|
&& message->FindInt32("modifiers", &modifiers) == B_OK) {
|
|
|
|
// TODO: for some reason, one of the above is failing when pressing
|
|
|
|
// a modifier key at least with the old BMessage implementation
|
|
|
|
// (a message dump shows all entries, though)
|
|
|
|
// Try again with BMessage4!
|
|
|
|
|
|
|
|
// Check for safe video mode (F12 + l-cmd + l-ctrl + l-shift)
|
|
|
|
if (key == 0x0d
|
|
|
|
&& (modifiers & (B_LEFT_COMMAND_KEY
|
|
|
|
| B_LEFT_CONTROL_KEY | B_LEFT_SHIFT_KEY)) != 0) {
|
|
|
|
// TODO: Set to Safe Mode in KeyboardEventHandler:B_KEY_DOWN.
|
|
|
|
STRACE(("Safe Video Mode invoked - code unimplemented\n"));
|
|
|
|
return B_SKIP_MESSAGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (key > 0x01 && key < 0x0e) {
|
|
|
|
// workspace change, F1-F12
|
|
|
|
|
|
|
|
#if !TEST_MODE
|
|
|
|
if (modifiers & B_COMMAND_KEY)
|
|
|
|
#else
|
|
|
|
if (modifiers & B_CONTROL_KEY)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
STRACE(("Set Workspace %ld\n", key - 1));
|
|
|
|
|
|
|
|
fDesktop->SetWorkspace(key - 2);
|
|
|
|
return B_SKIP_MESSAGE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: this should be moved client side!
|
|
|
|
// (that's how it is done in BeOS, clients could need this key for
|
|
|
|
// different purposes - also, it's preferrable to let the client
|
|
|
|
// write the dump within his own environment)
|
|
|
|
if (key == 0xe) {
|
|
|
|
// screen dump, PrintScreen
|
|
|
|
char filename[128];
|
|
|
|
BEntry entry;
|
|
|
|
|
|
|
|
int32 index = 1;
|
|
|
|
do {
|
|
|
|
sprintf(filename, "/boot/home/screen%ld.png", index++);
|
|
|
|
entry.SetTo(filename);
|
|
|
|
} while(entry.Exists());
|
|
|
|
|
|
|
|
fDesktop->GetDrawingEngine()->DumpToFile(filename);
|
|
|
|
return B_SKIP_MESSAGE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message->what == B_KEY_DOWN
|
|
|
|
|| message->what == B_MODIFIERS_CHANGED
|
|
|
|
|| message->what == B_UNMAPPED_KEY_DOWN)
|
|
|
|
_UpdateFocus(key, _target);
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
|
|
|
return B_DISPATCH_MESSAGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-16 18:16:31 +04:00
|
|
|
void
|
|
|
|
KeyboardFilter::RemoveTarget(EventTarget* target)
|
|
|
|
{
|
|
|
|
if (target == fLastFocus)
|
|
|
|
fLastFocus = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
// #pragma mark -
|
|
|
|
|
|
|
|
|
|
|
|
MouseFilter::MouseFilter(Desktop* desktop)
|
|
|
|
:
|
|
|
|
fDesktop(desktop)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
filter_result
|
2006-01-11 15:25:06 +03:00
|
|
|
MouseFilter::Filter(BMessage* message, EventTarget** _target, int32* _viewToken,
|
|
|
|
BMessage* latestMouseMoved)
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
{
|
|
|
|
BPoint where;
|
|
|
|
if (message->FindPoint("where", &where) != B_OK)
|
|
|
|
return B_DISPATCH_MESSAGE;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!fDesktop->LockAllWindows())
|
2005-12-10 16:47:13 +03:00
|
|
|
return B_DISPATCH_MESSAGE;
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
2006-04-01 19:43:25 +04:00
|
|
|
int32 viewToken = B_NULL_TOKEN;
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
WindowLayer* window = fDesktop->MouseEventWindow();
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
if (window == NULL)
|
2005-12-08 15:41:19 +03:00
|
|
|
window = fDesktop->WindowAt(where);
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
|
|
|
if (window != NULL) {
|
|
|
|
// dispatch event in the window layers
|
|
|
|
switch (message->what) {
|
|
|
|
case B_MOUSE_DOWN:
|
2006-04-01 19:43:25 +04:00
|
|
|
window->MouseDown(message, where, &viewToken);
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
break;
|
2005-12-08 15:41:19 +03:00
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
case B_MOUSE_UP:
|
2006-04-01 19:43:25 +04:00
|
|
|
window->MouseUp(message, where, &viewToken);
|
2005-12-08 15:41:19 +03:00
|
|
|
fDesktop->SetMouseEventWindow(NULL);
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
break;
|
2005-12-08 15:41:19 +03:00
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
case B_MOUSE_MOVED:
|
2006-04-01 19:43:25 +04:00
|
|
|
window->MouseMoved(message, where, &viewToken,
|
2006-01-11 15:25:06 +03:00
|
|
|
latestMouseMoved == NULL || latestMouseMoved == message);
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2006-04-01 19:43:25 +04:00
|
|
|
if (viewToken != B_NULL_TOKEN) {
|
|
|
|
fDesktop->SetViewUnderMouse(window, viewToken);
|
|
|
|
|
|
|
|
*_viewToken = viewToken;
|
2005-12-08 15:41:19 +03:00
|
|
|
*_target = &window->EventTarget();
|
2006-03-10 16:03:41 +03:00
|
|
|
}
|
2006-04-01 19:43:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (window == NULL || viewToken == B_NULL_TOKEN) {
|
|
|
|
// mouse is not over a window or over a decorator
|
|
|
|
fDesktop->SetViewUnderMouse(window, B_NULL_TOKEN);
|
2006-03-10 01:54:10 +03:00
|
|
|
fDesktop->SetCursor(NULL);
|
2006-04-01 19:43:25 +04:00
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
*_target = NULL;
|
2006-03-10 01:54:10 +03:00
|
|
|
}
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
fDesktop->UnlockAllWindows();
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
2005-11-18 16:51:32 +03:00
|
|
|
return B_DISPATCH_MESSAGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
|
|
|
2005-11-30 18:47:01 +03:00
|
|
|
static inline uint32
|
|
|
|
workspace_to_workspaces(int32 index)
|
|
|
|
{
|
|
|
|
return 1UL << index;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static inline bool
|
2005-12-09 16:28:28 +03:00
|
|
|
workspace_in_workspaces(int32 index, uint32 workspaces)
|
2005-11-30 18:47:01 +03:00
|
|
|
{
|
|
|
|
return (workspaces & (1UL << index)) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
|
|
|
2005-10-31 22:35:46 +03:00
|
|
|
Desktop::Desktop(uid_t userID)
|
2005-07-24 21:14:17 +04:00
|
|
|
: MessageLooper("desktop"),
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-10-31 22:35:46 +03:00
|
|
|
fUserID(userID),
|
2006-04-26 00:12:06 +04:00
|
|
|
fSettings(NULL),
|
|
|
|
fSharedReadOnlyArea(-1),
|
2005-12-08 15:41:19 +03:00
|
|
|
fApplicationsLock("application list"),
|
2005-07-26 01:08:34 +04:00
|
|
|
fShutdownSemaphore(-1),
|
2006-02-05 18:58:29 +03:00
|
|
|
fCurrentWorkspace(0),
|
2005-12-08 15:41:19 +03:00
|
|
|
fAllWindows(kAllWindowList),
|
2005-12-09 16:17:43 +03:00
|
|
|
fSubsetWindows(kSubsetList),
|
2005-12-09 19:20:01 +03:00
|
|
|
fWorkspacesLayer(NULL),
|
2005-08-21 14:28:40 +04:00
|
|
|
fActiveScreen(NULL),
|
2006-02-05 21:03:09 +03:00
|
|
|
fWindowLock("window lock"),
|
|
|
|
fMouseEventWindow(NULL),
|
|
|
|
fFocus(NULL),
|
|
|
|
fFront(NULL),
|
|
|
|
fBack(NULL)
|
2004-01-22 03:32:07 +03:00
|
|
|
{
|
2005-07-24 21:14:17 +04:00
|
|
|
char name[B_OS_NAME_LENGTH];
|
2005-11-17 17:56:12 +03:00
|
|
|
Desktop::_GetLooperName(name, sizeof(name));
|
2005-07-24 21:14:17 +04:00
|
|
|
|
|
|
|
fMessagePort = create_port(DEFAULT_MONITOR_PORT_SIZE, name);
|
|
|
|
if (fMessagePort < B_OK)
|
|
|
|
return;
|
2005-07-26 01:08:34 +04:00
|
|
|
|
|
|
|
fLink.SetReceiverPort(fMessagePort);
|
2003-01-24 18:19:27 +03:00
|
|
|
}
|
2004-01-22 03:32:07 +03:00
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
Desktop::~Desktop()
|
2004-01-22 03:32:07 +03:00
|
|
|
{
|
2005-07-17 20:25:48 +04:00
|
|
|
delete fSettings;
|
2005-07-24 21:14:17 +04:00
|
|
|
|
2006-04-26 00:12:06 +04:00
|
|
|
delete_area(fSharedReadOnlyArea);
|
2005-07-24 21:14:17 +04:00
|
|
|
delete_port(fMessagePort);
|
2005-11-03 20:03:36 +03:00
|
|
|
gFontManager->DetachUser(fUserID);
|
2003-01-24 18:19:27 +03:00
|
|
|
}
|
2004-01-13 03:56:36 +03:00
|
|
|
|
2005-03-29 19:39:08 +04:00
|
|
|
|
2006-04-26 00:12:06 +04:00
|
|
|
status_t
|
2005-06-24 03:46:17 +04:00
|
|
|
Desktop::Init()
|
2004-01-22 03:32:07 +03:00
|
|
|
{
|
2006-04-26 00:12:06 +04:00
|
|
|
if (fMessagePort < B_OK)
|
|
|
|
return fMessagePort;
|
|
|
|
|
|
|
|
char name[B_OS_NAME_LENGTH];
|
|
|
|
snprintf(name, sizeof(name), "d:%d:shared read only", /*id*/0);
|
|
|
|
fSharedReadOnlyArea = create_area(name, (void **)&fServerReadOnlyMemory,
|
|
|
|
B_ANY_ADDRESS, B_PAGE_SIZE, B_NO_LOCK, B_READ_AREA | B_WRITE_AREA);
|
|
|
|
if (fSharedReadOnlyArea < B_OK)
|
|
|
|
return fSharedReadOnlyArea;
|
|
|
|
|
|
|
|
fSettings = new DesktopSettings::Private(fServerReadOnlyMemory);
|
|
|
|
|
|
|
|
for (int32 i = 0; i < kMaxWorkspaces; i++) {
|
|
|
|
_Windows(i).SetIndex(i);
|
|
|
|
fWorkspaces[i].RestoreConfiguration(*fSettings->WorkspacesMessage(i));
|
|
|
|
}
|
|
|
|
|
2005-07-17 20:25:48 +04:00
|
|
|
fVirtualScreen.RestoreConfiguration(*this, fSettings->WorkspacesMessage(0));
|
2005-04-25 18:14:09 +04:00
|
|
|
|
2005-07-15 16:45:23 +04:00
|
|
|
// TODO: temporary workaround, fActiveScreen will be removed
|
|
|
|
fActiveScreen = fVirtualScreen.ScreenAt(0);
|
2005-04-25 18:14:09 +04:00
|
|
|
|
2005-11-15 22:59:53 +03:00
|
|
|
#if TEST_MODE
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
gInputManager->AddStream(new InputServerStream);
|
2005-11-15 22:59:53 +03:00
|
|
|
#endif
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
fEventDispatcher.SetTo(gInputManager->GetStream());
|
|
|
|
fEventDispatcher.SetHWInterface(fVirtualScreen.HWInterface());
|
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
fEventDispatcher.SetMouseFilter(new MouseFilter(this));
|
2005-11-18 16:51:32 +03:00
|
|
|
fEventDispatcher.SetKeyboardFilter(new KeyboardFilter(this));
|
2005-07-18 04:22:08 +04:00
|
|
|
|
2006-02-05 22:43:44 +03:00
|
|
|
SetCursor(NULL);
|
|
|
|
// this will set the default cursor
|
2005-11-18 16:21:07 +03:00
|
|
|
|
|
|
|
fVirtualScreen.HWInterface()->MoveCursorTo(fVirtualScreen.Frame().Width() / 2,
|
|
|
|
fVirtualScreen.Frame().Height() / 2);
|
* the new input event dispatcher is now actually used, although it doesn't
distribute any messages to the clients yet.
* removed the working thread from RootLayer - for now, its event handlers are
still called using input filters in the new event dispatcher, though (to
get things started).
* ServerApp is now using a BMessenger to identify its client, and no longer
stores the port/token separately.
* the input_server handshake is a bit simpler now, as it can now just reply
to the app_server message, removed unused code from ServerProtocol.h
* calmed down the MultiLocker (it always printed thread statistics on startup,
because it's compiled in debug mode).
* removed the cursor thread stuff from AppServer.cpp
* the new event dispatcher now uses a cursor thread when supported (only in
native mode, not in the test environment), although it improves cursor
movement under Qemu, the effect is not as good as expected - this might
need some more investigations (might just be a thread priority problem).
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15012 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-18 14:30:06 +03:00
|
|
|
fVirtualScreen.HWInterface()->SetCursorVisible(true);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
// draw the background
|
|
|
|
|
|
|
|
fScreenRegion = fVirtualScreen.Frame();
|
|
|
|
|
|
|
|
BRegion stillAvailableOnScreen;
|
|
|
|
_RebuildClippingForAllWindows(stillAvailableOnScreen);
|
|
|
|
_SetBackground(stillAvailableOnScreen);
|
2006-04-26 00:12:06 +04:00
|
|
|
|
|
|
|
gFontManager->AttachUser(fUserID);
|
|
|
|
|
|
|
|
return B_OK;
|
2003-01-24 18:19:27 +03:00
|
|
|
}
|
2004-01-22 03:32:07 +03:00
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
|
2005-07-24 21:14:17 +04:00
|
|
|
void
|
|
|
|
Desktop::_GetLooperName(char* name, size_t length)
|
|
|
|
{
|
|
|
|
snprintf(name, length, "d:%d:%s", /*id*/0, /*name*/"baron");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-07-26 01:08:34 +04:00
|
|
|
void
|
|
|
|
Desktop::_PrepareQuit()
|
|
|
|
{
|
|
|
|
// let's kill all remaining applications
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
fApplicationsLock.Lock();
|
2005-07-26 01:08:34 +04:00
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
int32 count = fApplications.CountItems();
|
2005-07-26 01:08:34 +04:00
|
|
|
for (int32 i = 0; i < count; i++) {
|
2005-12-08 15:41:19 +03:00
|
|
|
ServerApp *app = fApplications.ItemAt(i);
|
2005-07-26 01:08:34 +04:00
|
|
|
team_id clientTeam = app->ClientTeam();
|
|
|
|
|
|
|
|
app->Quit();
|
|
|
|
kill_team(clientTeam);
|
|
|
|
}
|
|
|
|
|
|
|
|
// wait for the last app to die
|
|
|
|
if (count > 0)
|
|
|
|
acquire_sem_etc(fShutdownSemaphore, fShutdownCount, B_RELATIVE_TIMEOUT, 250000);
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
fApplicationsLock.Unlock();
|
2005-07-26 01:08:34 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::_DispatchMessage(int32 code, BPrivate::LinkReceiver &link)
|
|
|
|
{
|
|
|
|
switch (code) {
|
|
|
|
case AS_CREATE_APP:
|
|
|
|
{
|
|
|
|
// Create the ServerApp to node monitor a new BApplication
|
|
|
|
|
|
|
|
// Attached data:
|
|
|
|
// 1) port_id - receiver port of a regular app
|
|
|
|
// 2) port_id - client looper port - for sending messages to the client
|
|
|
|
// 2) team_id - app's team ID
|
|
|
|
// 3) int32 - handler token of the regular app
|
|
|
|
// 4) char * - signature of the regular app
|
|
|
|
|
|
|
|
// Find the necessary data
|
|
|
|
team_id clientTeamID = -1;
|
|
|
|
port_id clientLooperPort = -1;
|
|
|
|
port_id clientReplyPort = -1;
|
|
|
|
int32 htoken = B_NULL_TOKEN;
|
|
|
|
char *appSignature = NULL;
|
|
|
|
|
|
|
|
link.Read<port_id>(&clientReplyPort);
|
|
|
|
link.Read<port_id>(&clientLooperPort);
|
|
|
|
link.Read<team_id>(&clientTeamID);
|
|
|
|
link.Read<int32>(&htoken);
|
|
|
|
if (link.ReadString(&appSignature) != B_OK)
|
|
|
|
break;
|
|
|
|
|
|
|
|
ServerApp *app = new ServerApp(this, clientReplyPort,
|
|
|
|
clientLooperPort, clientTeamID, htoken, appSignature);
|
|
|
|
if (app->InitCheck() == B_OK
|
|
|
|
&& app->Run()) {
|
|
|
|
// add the new ServerApp to the known list of ServerApps
|
2005-12-08 15:41:19 +03:00
|
|
|
fApplicationsLock.Lock();
|
|
|
|
fApplications.AddItem(app);
|
|
|
|
fApplicationsLock.Unlock();
|
2005-07-26 01:08:34 +04:00
|
|
|
} else {
|
|
|
|
delete app;
|
|
|
|
|
|
|
|
// if everything went well, ServerApp::Run() will notify
|
|
|
|
// the client - but since it didn't, we do it here
|
|
|
|
BPrivate::LinkSender reply(clientReplyPort);
|
2006-04-26 13:33:08 +04:00
|
|
|
reply.StartMessage(B_ERROR);
|
2005-07-26 01:08:34 +04:00
|
|
|
reply.Flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is necessary because BPortLink::ReadString allocates memory
|
|
|
|
free(appSignature);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case AS_DELETE_APP:
|
|
|
|
{
|
|
|
|
// Delete a ServerApp. Received only from the respective ServerApp when a
|
|
|
|
// BApplication asks it to quit.
|
|
|
|
|
|
|
|
// Attached Data:
|
|
|
|
// 1) thread_id - thread ID of the ServerApp to be deleted
|
|
|
|
|
|
|
|
thread_id thread = -1;
|
|
|
|
if (link.Read<thread_id>(&thread) < B_OK)
|
|
|
|
break;
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
fApplicationsLock.Lock();
|
2005-07-26 01:08:34 +04:00
|
|
|
|
|
|
|
// Run through the list of apps and nuke the proper one
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
int32 count = fApplications.CountItems();
|
2005-07-26 01:08:34 +04:00
|
|
|
ServerApp *removeApp = NULL;
|
|
|
|
|
|
|
|
for (int32 i = 0; i < count; i++) {
|
2005-12-08 15:41:19 +03:00
|
|
|
ServerApp *app = fApplications.ItemAt(i);
|
2005-07-26 01:08:34 +04:00
|
|
|
|
|
|
|
if (app != NULL && app->Thread() == thread) {
|
2005-12-08 15:41:19 +03:00
|
|
|
fApplications.RemoveItemAt(i);
|
2005-07-26 01:08:34 +04:00
|
|
|
removeApp = app;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
fApplicationsLock.Unlock();
|
2005-07-26 01:08:34 +04:00
|
|
|
|
|
|
|
if (removeApp != NULL)
|
|
|
|
removeApp->Quit(fShutdownSemaphore);
|
|
|
|
|
|
|
|
if (fQuitting && count <= 1) {
|
|
|
|
// wait for the last app to die
|
|
|
|
acquire_sem_etc(fShutdownSemaphore, fShutdownCount, B_RELATIVE_TIMEOUT, 500000);
|
|
|
|
PostMessage(kMsgQuitLooper);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-10-31 22:35:46 +03:00
|
|
|
case AS_ACTIVATE_APP:
|
|
|
|
{
|
|
|
|
// Someone is requesting to activation of a certain app.
|
|
|
|
|
|
|
|
// Attached data:
|
|
|
|
// 1) port_id reply port
|
|
|
|
// 2) team_id team
|
|
|
|
|
|
|
|
status_t status;
|
|
|
|
|
|
|
|
// get the parameters
|
|
|
|
port_id replyPort;
|
|
|
|
team_id team;
|
|
|
|
if (link.Read(&replyPort) == B_OK
|
|
|
|
&& link.Read(&team) == B_OK)
|
|
|
|
status = _ActivateApp(team);
|
|
|
|
else
|
|
|
|
status = B_ERROR;
|
|
|
|
|
|
|
|
// send the reply
|
|
|
|
BPrivate::PortLink replyLink(replyPort);
|
|
|
|
replyLink.StartMessage(status);
|
|
|
|
replyLink.Flush();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2006-06-10 01:46:40 +04:00
|
|
|
case AS_APP_CRASHED:
|
|
|
|
{
|
|
|
|
BAutolock locker(fApplicationsLock);
|
|
|
|
|
|
|
|
team_id team;
|
|
|
|
if (link.Read(&team) != B_OK)
|
|
|
|
break;
|
|
|
|
|
|
|
|
for (int32 i = 0; i < fApplications.CountItems(); i++) {
|
|
|
|
ServerApp* app = fApplications.ItemAt(i);
|
|
|
|
|
|
|
|
if (app->ClientTeam() == team)
|
|
|
|
app->PostMessage(AS_APP_CRASHED);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-07-26 01:08:34 +04:00
|
|
|
case B_QUIT_REQUESTED:
|
|
|
|
// We've been asked to quit, so (for now) broadcast to all
|
|
|
|
// test apps to quit. This situation will occur only when the server
|
|
|
|
// is compiled as a regular Be application.
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
fApplicationsLock.Lock();
|
2005-07-26 01:08:34 +04:00
|
|
|
fShutdownSemaphore = create_sem(0, "desktop shutdown");
|
2005-12-08 15:41:19 +03:00
|
|
|
fShutdownCount = fApplications.CountItems();
|
|
|
|
fApplicationsLock.Unlock();
|
2005-07-26 01:08:34 +04:00
|
|
|
|
|
|
|
fQuitting = true;
|
|
|
|
BroadcastToAllApps(AS_QUIT_APP);
|
|
|
|
|
|
|
|
// We now need to process the remaining AS_DELETE_APP messages and
|
|
|
|
// wait for the kMsgShutdownServer message.
|
|
|
|
// If an application does not quit as asked, the picasso thread
|
|
|
|
// will send us this message in 2-3 seconds.
|
|
|
|
|
|
|
|
// if there are no apps to quit, shutdown directly
|
|
|
|
if (fShutdownCount == 0)
|
|
|
|
PostMessage(kMsgQuitLooper);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("Desktop %d:%s received unexpected code %ld\n", 0, "baron", code);
|
|
|
|
|
|
|
|
if (link.NeedsReply()) {
|
|
|
|
// the client is now blocking and waiting for a reply!
|
|
|
|
fLink.StartMessage(B_ERROR);
|
|
|
|
fLink.Flush();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-10-31 22:35:46 +03:00
|
|
|
/*!
|
|
|
|
\brief activate one of the app's windows.
|
|
|
|
*/
|
|
|
|
status_t
|
|
|
|
Desktop::_ActivateApp(team_id team)
|
|
|
|
{
|
|
|
|
status_t status = B_BAD_TEAM_ID;
|
|
|
|
|
|
|
|
// search for an unhidden window to give focus to
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
for (WindowLayer* window = fAllWindows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(kAllWindowList)) {
|
|
|
|
// if window is a normal window of the team, and not hidden,
|
|
|
|
// we've found our target
|
|
|
|
if (!window->IsHidden() && window->IsNormal()
|
|
|
|
&& window->ServerWindow()->ClientTeam() == team) {
|
|
|
|
ActivateWindow(window);
|
2005-11-29 02:36:59 +03:00
|
|
|
return B_OK;
|
2005-10-31 22:35:46 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-07-26 01:08:34 +04:00
|
|
|
/*!
|
|
|
|
\brief Send a quick (no attachments) message to all applications
|
|
|
|
|
|
|
|
Quite useful for notification for things like server shutdown, system
|
|
|
|
color changes, etc.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
Desktop::BroadcastToAllApps(int32 code)
|
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
BAutolock locker(fApplicationsLock);
|
2005-07-26 01:08:34 +04:00
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
for (int32 i = 0; i < fApplications.CountItems(); i++) {
|
|
|
|
fApplications.ItemAt(i)->PostMessage(code);
|
2005-07-26 01:08:34 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-05 22:43:44 +03:00
|
|
|
ServerCursor*
|
|
|
|
Desktop::Cursor() const
|
|
|
|
{
|
|
|
|
return HWInterface()->Cursor();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::SetCursor(ServerCursor* newCursor)
|
|
|
|
{
|
|
|
|
if (newCursor == NULL)
|
|
|
|
newCursor = fCursorManager.GetCursor(B_CURSOR_DEFAULT);
|
|
|
|
|
|
|
|
ServerCursor* oldCursor = Cursor();
|
|
|
|
if (newCursor == oldCursor)
|
|
|
|
return;
|
|
|
|
|
|
|
|
HWInterface()->SetCursor(newCursor);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-13 16:12:10 +03:00
|
|
|
/*!
|
|
|
|
\brief Redraws the background (ie. the desktop window, if any).
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
Desktop::RedrawBackground()
|
|
|
|
{
|
|
|
|
LockAllWindows();
|
|
|
|
|
|
|
|
BRegion redraw;
|
|
|
|
|
|
|
|
WindowLayer* window = _CurrentWindows().FirstWindow();
|
|
|
|
if (window->Feel() == kDesktopWindowFeel) {
|
|
|
|
redraw = window->VisibleContentRegion();
|
2006-02-13 16:43:30 +03:00
|
|
|
|
|
|
|
// look for desktop background view, and update its background color
|
|
|
|
// TODO: is there a better way to do this?
|
|
|
|
ViewLayer* view = window->TopLayer();
|
|
|
|
if (view != NULL)
|
|
|
|
view = view->FirstChild();
|
|
|
|
|
|
|
|
while (view) {
|
|
|
|
if (view->IsDesktopBackground()) {
|
|
|
|
view->SetViewColor(fWorkspaces[fCurrentWorkspace].Color());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
view = view->NextSibling();
|
|
|
|
}
|
|
|
|
|
2006-02-13 16:12:10 +03:00
|
|
|
window->ProcessDirtyRegion(redraw);
|
|
|
|
} else {
|
|
|
|
redraw = BackgroundRegion();
|
|
|
|
fBackgroundRegion.MakeEmpty();
|
|
|
|
_SetBackground(redraw);
|
|
|
|
}
|
|
|
|
|
2006-04-03 15:41:16 +04:00
|
|
|
_WindowChanged(NULL);
|
|
|
|
// update workspaces view as well
|
|
|
|
|
2006-02-13 16:12:10 +03:00
|
|
|
UnlockAllWindows();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-12 20:52:41 +04:00
|
|
|
/*!
|
|
|
|
\brief Store the workspace configuration
|
|
|
|
*/
|
|
|
|
void
|
2006-04-13 00:55:36 +04:00
|
|
|
Desktop::StoreWorkspaceConfiguration(int32 index)
|
2006-04-12 20:52:41 +04:00
|
|
|
{
|
2006-04-14 14:50:47 +04:00
|
|
|
const BMessage *oldSettings = fSettings->WorkspacesMessage(index);
|
2006-04-12 20:52:41 +04:00
|
|
|
// store settings
|
2006-04-13 00:55:36 +04:00
|
|
|
BMessage settings;
|
2006-04-14 14:50:47 +04:00
|
|
|
if (oldSettings)
|
|
|
|
settings = *oldSettings;
|
2006-04-13 00:55:36 +04:00
|
|
|
fWorkspaces[index].StoreConfiguration(settings);
|
|
|
|
fSettings->SetWorkspacesMessage(index, settings);
|
|
|
|
fSettings->Save(kWorkspacesSettings);
|
2006-04-12 20:52:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
void
|
|
|
|
Desktop::UpdateWorkspaces()
|
|
|
|
{
|
|
|
|
// TODO: maybe this should be replaced by a SetWorkspacesCount() method
|
|
|
|
|
2005-12-09 19:20:01 +03:00
|
|
|
_WindowChanged(NULL);
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-11-29 02:36:59 +03:00
|
|
|
void
|
|
|
|
Desktop::SetWorkspace(int32 index)
|
|
|
|
{
|
2005-12-12 13:41:31 +03:00
|
|
|
LockAllWindows();
|
2005-11-29 02:36:59 +03:00
|
|
|
DesktopSettings settings(this);
|
|
|
|
|
2005-12-10 16:47:13 +03:00
|
|
|
if (index < 0 || index >= settings.WorkspacesCount() || index == fCurrentWorkspace) {
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-11-29 02:36:59 +03:00
|
|
|
return;
|
2005-12-10 16:47:13 +03:00
|
|
|
}
|
2005-12-08 17:45:42 +03:00
|
|
|
|
2005-11-30 18:47:01 +03:00
|
|
|
int32 previousIndex = fCurrentWorkspace;
|
2006-04-03 15:41:16 +04:00
|
|
|
RGBColor previousColor = fWorkspaces[fCurrentWorkspace].Color();
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-08 17:45:42 +03:00
|
|
|
if (fMouseEventWindow != NULL) {
|
2005-12-09 16:17:43 +03:00
|
|
|
if (!fMouseEventWindow->InWorkspace(index)) {
|
|
|
|
// the window currently being dragged will follow us to this workspace
|
|
|
|
// if it's not already on it
|
|
|
|
if (fMouseEventWindow->IsNormal()) {
|
|
|
|
// but only normal windows are following
|
2005-12-15 19:42:12 +03:00
|
|
|
uint32 oldWorkspaces = fMouseEventWindow->Workspaces();
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
_Windows(index).AddWindow(fMouseEventWindow);
|
|
|
|
_Windows(previousIndex).RemoveWindow(fMouseEventWindow);
|
2005-12-15 19:42:12 +03:00
|
|
|
|
|
|
|
// send B_WORKSPACES_CHANGED message
|
|
|
|
fMouseEventWindow->WorkspacesChanged(oldWorkspaces,
|
|
|
|
fMouseEventWindow->Workspaces());
|
2005-12-09 16:17:43 +03:00
|
|
|
}
|
2005-12-08 17:45:42 +03:00
|
|
|
} else {
|
|
|
|
// make sure it's frontmost
|
2005-12-09 16:17:43 +03:00
|
|
|
_Windows(index).RemoveWindow(fMouseEventWindow);
|
|
|
|
_Windows(index).AddWindow(fMouseEventWindow,
|
|
|
|
fMouseEventWindow->Frontmost(_Windows(index).FirstWindow(), index));
|
2005-12-08 17:45:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
fMouseEventWindow->Anchor(index).position = fMouseEventWindow->Frame().LeftTop();
|
|
|
|
}
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
// build region of windows that are no longer visible in the new workspace
|
|
|
|
|
|
|
|
BRegion dirty;
|
2005-12-09 16:17:43 +03:00
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
for (WindowLayer* window = _CurrentWindows().FirstWindow();
|
|
|
|
window != NULL; window = window->NextWindow(previousIndex)) {
|
2005-12-09 16:17:43 +03:00
|
|
|
// store current position in Workspace anchor
|
2005-12-08 15:41:19 +03:00
|
|
|
window->Anchor(previousIndex).position = window->Frame().LeftTop();
|
|
|
|
|
2005-12-15 19:42:12 +03:00
|
|
|
window->WorkspaceActivated(previousIndex, false);
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (window->InWorkspace(index))
|
2005-12-08 15:41:19 +03:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!window->IsHidden()) {
|
|
|
|
// this window will no longer be visible
|
|
|
|
dirty.Include(&window->VisibleRegion());
|
|
|
|
}
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
window->SetCurrentWorkspace(-1);
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
|
|
|
|
2005-11-30 18:47:01 +03:00
|
|
|
fCurrentWorkspace = index;
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
// show windows, and include them in the changed region - but only
|
|
|
|
// those that were not visible before (or whose position changed)
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
for (WindowLayer* window = _Windows(index).FirstWindow();
|
2005-12-08 15:41:19 +03:00
|
|
|
window != NULL; window = window->NextWindow(index)) {
|
|
|
|
BPoint position = window->Anchor(index).position;
|
|
|
|
|
|
|
|
window->SetCurrentWorkspace(index);
|
|
|
|
|
|
|
|
if (window->IsHidden())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (position == kInvalidWindowPosition) {
|
|
|
|
// if you enter a workspace for the first time, the position
|
|
|
|
// of the window in the previous workspace is adopted
|
|
|
|
position = window->Frame().LeftTop();
|
|
|
|
// TODO: make sure the window is still on-screen if it
|
|
|
|
// was before!
|
|
|
|
}
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (!window->InWorkspace(previousIndex)) {
|
2006-03-14 02:47:32 +03:00
|
|
|
// This window was not visible before, make sure its frame
|
|
|
|
// is up-to-date
|
|
|
|
if (window->Frame().LeftTop() != position) {
|
|
|
|
BPoint offset = position - window->Frame().LeftTop();
|
|
|
|
window->MoveBy(offset.x, offset.y);
|
|
|
|
}
|
2005-12-08 15:41:19 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->Frame().LeftTop() != position) {
|
|
|
|
// the window was visible before, but its on-screen location changed
|
|
|
|
BPoint offset = position - window->Frame().LeftTop();
|
|
|
|
MoveWindowBy(window, offset.x, offset.y);
|
|
|
|
// TODO: be a bit smarter than this...
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BRegion stillAvailableOnScreen;
|
|
|
|
_RebuildClippingForAllWindows(stillAvailableOnScreen);
|
|
|
|
_SetBackground(stillAvailableOnScreen);
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
for (WindowLayer* window = _Windows(index).FirstWindow(); window != NULL;
|
2005-12-08 15:41:19 +03:00
|
|
|
window = window->NextWindow(index)) {
|
2005-12-15 19:42:12 +03:00
|
|
|
// send B_WORKSPACE_ACTIVATED message
|
|
|
|
window->WorkspaceActivated(index, true);
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (window->InWorkspace(previousIndex) || window == fMouseEventWindow) {
|
2005-12-08 15:41:19 +03:00
|
|
|
// this window was visible before, and is already handled in the above loop
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
dirty.Include(&window->VisibleRegion());
|
|
|
|
}
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
_UpdateFronts(false);
|
|
|
|
_UpdateFloating(previousIndex, index);
|
|
|
|
|
2005-12-09 17:23:51 +03:00
|
|
|
// Set new focus to the front window, but keep focus to a floating
|
|
|
|
// window if still visible
|
|
|
|
if (!_Windows(index).HasWindow(FocusWindow()) || !FocusWindow()->IsFloating())
|
|
|
|
SetFocusWindow(FrontWindow());
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 19:20:01 +03:00
|
|
|
_WindowChanged(NULL);
|
2005-12-08 15:41:19 +03:00
|
|
|
MarkDirty(dirty);
|
2005-12-08 17:45:42 +03:00
|
|
|
|
2006-04-03 15:41:16 +04:00
|
|
|
if (previousColor != fWorkspaces[fCurrentWorkspace].Color())
|
|
|
|
RedrawBackground();
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2006-04-01 19:43:25 +04:00
|
|
|
|
|
|
|
_SendFakeMouseMoved();
|
2005-11-29 02:36:59 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-11-26 19:56:52 +03:00
|
|
|
void
|
2006-02-06 16:36:46 +03:00
|
|
|
Desktop::ScreenChanged(Screen* screen, bool makeDefault)
|
2005-11-26 19:56:52 +03:00
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
// TODO: confirm that everywhere this is used,
|
|
|
|
// the Window WriteLock is held
|
|
|
|
|
|
|
|
// the entire screen is dirty, because we're actually
|
|
|
|
// operating on an all new buffer in memory
|
|
|
|
BRegion dirty(screen->Frame());
|
|
|
|
// update our cached screen region
|
|
|
|
fScreenRegion.Set(screen->Frame());
|
|
|
|
|
|
|
|
BRegion background;
|
|
|
|
_RebuildClippingForAllWindows(background);
|
|
|
|
|
|
|
|
fBackgroundRegion.MakeEmpty();
|
|
|
|
// makes sure that the complete background is redrawn
|
|
|
|
_SetBackground(background);
|
|
|
|
|
|
|
|
// figure out dirty region
|
|
|
|
dirty.Exclude(&background);
|
|
|
|
_TriggerWindowRedrawing(dirty);
|
|
|
|
|
|
|
|
// send B_SCREEN_CHANGED to windows on that screen
|
2005-11-26 19:56:52 +03:00
|
|
|
BMessage update(B_SCREEN_CHANGED);
|
|
|
|
update.AddInt64("when", real_time_clock_usecs());
|
|
|
|
update.AddRect("frame", screen->Frame());
|
|
|
|
update.AddInt32("mode", screen->ColorSpace());
|
|
|
|
|
|
|
|
// TODO: currently ignores the screen argument!
|
2005-12-08 15:41:19 +03:00
|
|
|
for (WindowLayer* window = fAllWindows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(kAllWindowList)) {
|
|
|
|
window->ServerWindow()->SendMessageToClient(&update);
|
|
|
|
}
|
2006-02-06 16:36:46 +03:00
|
|
|
|
|
|
|
if (makeDefault) {
|
|
|
|
// store settings
|
|
|
|
BMessage settings;
|
|
|
|
fVirtualScreen.StoreConfiguration(settings);
|
|
|
|
fWorkspaces[fCurrentWorkspace].StoreConfiguration(settings);
|
|
|
|
|
|
|
|
fSettings->SetWorkspacesMessage(fCurrentWorkspace, settings);
|
|
|
|
fSettings->Save(kWorkspacesSettings);
|
|
|
|
}
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// #pragma mark - Methods for WindowLayer manipulation
|
|
|
|
|
|
|
|
|
|
|
|
WindowList&
|
|
|
|
Desktop::_CurrentWindows()
|
|
|
|
{
|
|
|
|
return fWorkspaces[fCurrentWorkspace].Windows();
|
|
|
|
}
|
2005-11-26 19:56:52 +03:00
|
|
|
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
WindowList&
|
|
|
|
Desktop::_Windows(int32 index)
|
|
|
|
{
|
|
|
|
return fWorkspaces[index].Windows();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::_UpdateFloating(int32 previousWorkspace, int32 nextWorkspace)
|
|
|
|
{
|
|
|
|
if (previousWorkspace == -1)
|
|
|
|
previousWorkspace = fCurrentWorkspace;
|
|
|
|
if (nextWorkspace == -1)
|
|
|
|
nextWorkspace = previousWorkspace;
|
|
|
|
|
|
|
|
for (WindowLayer* floating = fSubsetWindows.FirstWindow(); floating != NULL;
|
|
|
|
floating = floating->NextWindow(kSubsetList)) {
|
|
|
|
// we only care about app/subset floating windows
|
|
|
|
if (floating->Feel() != B_FLOATING_SUBSET_WINDOW_FEEL
|
|
|
|
&& floating->Feel() != B_FLOATING_APP_WINDOW_FEEL)
|
|
|
|
continue;
|
|
|
|
|
2006-03-08 15:56:56 +03:00
|
|
|
if (fFront != NULL && fFront->IsNormal() && floating->HasInSubset(fFront)) {
|
2005-12-09 16:17:43 +03:00
|
|
|
// is now visible
|
|
|
|
if (_Windows(previousWorkspace).HasWindow(floating)
|
|
|
|
&& previousWorkspace != nextWorkspace) {
|
|
|
|
// but no longer on the previous workspace
|
|
|
|
_Windows(previousWorkspace).RemoveWindow(floating);
|
|
|
|
floating->SetCurrentWorkspace(-1);
|
|
|
|
}
|
|
|
|
if (!_Windows(nextWorkspace).HasWindow(floating)) {
|
|
|
|
// but wasn't before
|
|
|
|
_Windows(nextWorkspace).AddWindow(floating,
|
|
|
|
floating->Frontmost(_Windows(nextWorkspace).FirstWindow(), nextWorkspace));
|
|
|
|
floating->SetCurrentWorkspace(nextWorkspace);
|
|
|
|
_ShowWindow(floating);
|
|
|
|
|
|
|
|
// TODO:
|
|
|
|
// put the floating last in the floating window list to preserve
|
|
|
|
// the on screen window order
|
|
|
|
}
|
|
|
|
} else if (_Windows(previousWorkspace).HasWindow(floating)) {
|
|
|
|
// was visible, but is no longer
|
|
|
|
_Windows(previousWorkspace).RemoveWindow(floating);
|
|
|
|
floating->SetCurrentWorkspace(-1);
|
|
|
|
_HideWindow(floating);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Search the visible windows for a valid back window
|
2006-03-16 00:33:12 +03:00
|
|
|
(only desktop windows can't be back windows)
|
2005-12-08 15:41:19 +03:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
Desktop::_UpdateBack()
|
|
|
|
{
|
|
|
|
fBack = NULL;
|
|
|
|
|
|
|
|
for (WindowLayer* window = _CurrentWindows().FirstWindow();
|
|
|
|
window != NULL; window = window->NextWindow(fCurrentWorkspace)) {
|
2006-03-16 00:33:12 +03:00
|
|
|
if (window->IsHidden() || window->Feel() == kDesktopWindowFeel)
|
2005-12-08 15:41:19 +03:00
|
|
|
continue;
|
|
|
|
|
|
|
|
fBack = window;
|
|
|
|
break;
|
2005-11-26 19:56:52 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
/*!
|
|
|
|
Search the visible windows for a valid front window
|
|
|
|
(only normal and modal windows can be front windows)
|
|
|
|
|
|
|
|
The only place where you don't want to update floating windows is
|
|
|
|
during a workspace change - because then you'll call _UpdateFloating()
|
|
|
|
yourself.
|
2005-12-08 15:41:19 +03:00
|
|
|
*/
|
|
|
|
void
|
2005-12-09 16:17:43 +03:00
|
|
|
Desktop::_UpdateFront(bool updateFloating)
|
2005-12-08 15:41:19 +03:00
|
|
|
{
|
|
|
|
fFront = NULL;
|
|
|
|
|
|
|
|
for (WindowLayer* window = _CurrentWindows().LastWindow();
|
|
|
|
window != NULL; window = window->PreviousWindow(fCurrentWorkspace)) {
|
2005-12-09 16:17:43 +03:00
|
|
|
if (window->IsHidden() || window->IsFloating() || !window->SupportsFront())
|
2005-12-08 15:41:19 +03:00
|
|
|
continue;
|
|
|
|
|
|
|
|
fFront = window;
|
|
|
|
break;
|
|
|
|
}
|
2005-12-09 16:17:43 +03:00
|
|
|
|
|
|
|
if (updateFloating)
|
|
|
|
_UpdateFloating();
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
2004-01-22 03:32:07 +03:00
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
|
|
|
|
void
|
2005-12-09 16:17:43 +03:00
|
|
|
Desktop::_UpdateFronts(bool updateFloating)
|
2005-02-28 23:23:51 +03:00
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
_UpdateBack();
|
2005-12-09 16:17:43 +03:00
|
|
|
_UpdateFront(updateFloating);
|
2005-02-28 23:23:51 +03:00
|
|
|
}
|
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
bool
|
|
|
|
Desktop::_WindowHasModal(WindowLayer* window)
|
|
|
|
{
|
2006-04-13 15:49:42 +04:00
|
|
|
if (window == NULL)
|
2005-12-09 16:17:43 +03:00
|
|
|
return false;
|
|
|
|
|
|
|
|
for (WindowLayer* modal = fSubsetWindows.FirstWindow(); modal != NULL;
|
|
|
|
modal = modal->NextWindow(kSubsetList)) {
|
|
|
|
// only visible modal windows count
|
|
|
|
if (!modal->IsModal() || modal->IsHidden())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (modal->HasInSubset(window))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 19:20:01 +03:00
|
|
|
void
|
|
|
|
Desktop::_WindowChanged(WindowLayer* window)
|
|
|
|
{
|
|
|
|
if (fWorkspacesLayer == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fWorkspacesLayer->WindowChanged(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-01 19:43:25 +04:00
|
|
|
/*!
|
|
|
|
\brief Sends a fake B_MOUSE_MOVED event to the window under the mouse,
|
|
|
|
and also updates the current view under the mouse.
|
|
|
|
|
|
|
|
This has only to be done in case the view changed without user interaction,
|
|
|
|
ie. because of a workspace change or a closing window.
|
|
|
|
|
|
|
|
Windows must not be locked when calling this method.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
Desktop::_SendFakeMouseMoved(WindowLayer* window)
|
|
|
|
{
|
|
|
|
BPoint where;
|
|
|
|
int32 buttons;
|
|
|
|
EventDispatcher().GetMouse(where, buttons);
|
|
|
|
|
|
|
|
int32 viewToken = B_NULL_TOKEN;
|
|
|
|
BMessenger target;
|
|
|
|
|
|
|
|
LockAllWindows();
|
|
|
|
|
|
|
|
if (window == NULL)
|
|
|
|
window = MouseEventWindow();
|
|
|
|
if (window == NULL)
|
|
|
|
window = WindowAt(where);
|
|
|
|
|
|
|
|
if (window != NULL) {
|
|
|
|
BMessage message;
|
|
|
|
window->MouseMoved(&message, where, &viewToken, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (viewToken != B_NULL_TOKEN)
|
|
|
|
SetViewUnderMouse(window, viewToken);
|
|
|
|
else {
|
|
|
|
SetViewUnderMouse(NULL, B_NULL_TOKEN);
|
|
|
|
SetCursor(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
UnlockAllWindows();
|
|
|
|
|
|
|
|
if (viewToken != B_NULL_TOKEN)
|
|
|
|
EventDispatcher().SendFakeMouseMoved(target, viewToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
void
|
2005-12-08 15:41:19 +03:00
|
|
|
Desktop::SetFocusWindow(WindowLayer* focus)
|
2004-01-22 03:32:07 +03:00
|
|
|
{
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!LockAllWindows())
|
2005-12-08 15:41:19 +03:00
|
|
|
return;
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
bool hasModal = _WindowHasModal(focus);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
// TODO: test for FFM and B_LOCK_WINDOW_FOCUS
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2006-03-16 00:33:12 +03:00
|
|
|
if (focus == fFocus && focus != NULL && !focus->IsHidden()
|
|
|
|
&& (focus->Flags() & B_AVOID_FOCUS) == 0 && !hasModal) {
|
2005-12-09 16:17:43 +03:00
|
|
|
// the window that is supposed to get focus already has focus
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-09 16:17:43 +03:00
|
|
|
return;
|
|
|
|
}
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 17:23:51 +03:00
|
|
|
if (focus == NULL || hasModal) {
|
2005-12-09 16:17:43 +03:00
|
|
|
focus = FrontWindow();
|
2005-12-09 17:23:51 +03:00
|
|
|
if (focus == NULL) {
|
|
|
|
// there might be no front window in case of only a single
|
|
|
|
// window with B_FLOATING_ALL_WINDOW_FEEL
|
|
|
|
focus = _CurrentWindows().LastWindow();
|
|
|
|
}
|
|
|
|
}
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
// make sure no window is chosen that doesn't want focus or cannot have it
|
|
|
|
while (focus != NULL
|
2005-12-09 17:23:51 +03:00
|
|
|
&& ((focus->Flags() & B_AVOID_FOCUS) != 0
|
|
|
|
|| _WindowHasModal(focus)
|
|
|
|
|| focus->IsHidden())) {
|
2005-12-09 16:17:43 +03:00
|
|
|
focus = focus->PreviousWindow(fCurrentWorkspace);
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
2005-12-09 16:17:43 +03:00
|
|
|
|
2006-02-09 13:44:02 +03:00
|
|
|
if (fFocus == focus) {
|
|
|
|
// turns out the window that is supposed to get focus now already has it
|
|
|
|
UnlockAllWindows();
|
2006-02-09 14:27:06 +03:00
|
|
|
return;
|
2006-02-09 13:44:02 +03:00
|
|
|
}
|
|
|
|
|
2006-01-12 01:30:12 +03:00
|
|
|
ServerApp* oldActiveApp = NULL;
|
|
|
|
ServerApp* newActiveApp = NULL;
|
|
|
|
|
|
|
|
if (fFocus != NULL) {
|
2005-12-09 16:17:43 +03:00
|
|
|
fFocus->SetFocus(false);
|
2006-01-12 01:30:12 +03:00
|
|
|
oldActiveApp = fFocus->ServerWindow()->App();
|
|
|
|
}
|
2005-12-09 16:17:43 +03:00
|
|
|
|
|
|
|
fFocus = focus;
|
|
|
|
|
2006-01-12 01:30:12 +03:00
|
|
|
if (fFocus != NULL) {
|
|
|
|
fFocus->SetFocus(true);
|
|
|
|
newActiveApp = fFocus->ServerWindow()->App();
|
|
|
|
}
|
2005-12-09 16:17:43 +03:00
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2006-01-12 01:30:12 +03:00
|
|
|
|
|
|
|
// change the "active" app if appropriate
|
|
|
|
if (oldActiveApp != newActiveApp) {
|
|
|
|
if (oldActiveApp) {
|
|
|
|
oldActiveApp->Activate(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newActiveApp) {
|
|
|
|
newActiveApp->Activate(true);
|
|
|
|
} else {
|
|
|
|
// make sure the cursor is visible
|
|
|
|
HWInterface()->SetCursorVisible(true);
|
|
|
|
}
|
|
|
|
}
|
2005-11-29 02:36:59 +03:00
|
|
|
}
|
2005-02-28 23:23:51 +03:00
|
|
|
|
|
|
|
|
2005-11-30 13:45:01 +03:00
|
|
|
void
|
2005-12-08 15:41:19 +03:00
|
|
|
Desktop::_BringWindowsToFront(WindowList& windows, int32 list,
|
|
|
|
bool wereVisible)
|
|
|
|
{
|
|
|
|
// we don't need to redraw what is currently
|
|
|
|
// visible of the window
|
|
|
|
BRegion clean;
|
|
|
|
|
|
|
|
for (WindowLayer* window = windows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(list)) {
|
|
|
|
if (wereVisible)
|
|
|
|
clean.Include(&window->VisibleRegion());
|
|
|
|
|
|
|
|
_CurrentWindows().AddWindow(window,
|
|
|
|
window->Frontmost(_CurrentWindows().FirstWindow(),
|
|
|
|
fCurrentWorkspace));
|
2005-12-09 19:20:01 +03:00
|
|
|
|
|
|
|
_WindowChanged(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
BRegion dummy;
|
|
|
|
_RebuildClippingForAllWindows(dummy);
|
|
|
|
|
|
|
|
// redraw what became visible of the window(s)
|
|
|
|
|
|
|
|
BRegion dirty;
|
|
|
|
for (WindowLayer* window = windows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(list)) {
|
|
|
|
dirty.Include(&window->VisibleRegion());
|
|
|
|
}
|
|
|
|
|
|
|
|
dirty.Exclude(&clean);
|
|
|
|
MarkDirty(dirty);
|
|
|
|
|
|
|
|
_UpdateFront();
|
|
|
|
|
|
|
|
if (windows.FirstWindow() == fBack || fBack == NULL)
|
|
|
|
_UpdateBack();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
\brief Tries to move the specified window to the front of the screen,
|
|
|
|
and make it the focus window.
|
|
|
|
|
|
|
|
If there are any modal windows on this screen, it might not actually
|
|
|
|
become the frontmost window, though, as modal windows stay in front
|
|
|
|
of their subset.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
Desktop::ActivateWindow(WindowLayer* window)
|
2005-11-30 13:45:01 +03:00
|
|
|
{
|
2006-03-08 15:56:56 +03:00
|
|
|
STRACE(("ActivateWindow(%p, %s)\n", window, window ? window->Title() : "<none>"));
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
if (window == NULL) {
|
|
|
|
fBack = NULL;
|
|
|
|
fFront = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-05-01 14:14:51 +04:00
|
|
|
bool windowOnOtherWorkspace = !window->InWorkspace(fCurrentWorkspace);
|
|
|
|
|
|
|
|
if (windowOnOtherWorkspace
|
|
|
|
&& (window->Flags() & B_NO_WORKSPACE_ACTIVATION)
|
|
|
|
&& !(window->Flags() & B_NOT_ANCHORED_ON_ACTIVATE))
|
|
|
|
return;
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
// TODO: take care about floating windows
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!LockAllWindows())
|
2005-11-30 13:45:01 +03:00
|
|
|
return;
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2006-05-01 14:14:51 +04:00
|
|
|
if (windowOnOtherWorkspace) {
|
|
|
|
// if the window wants to come to the current workspace,
|
|
|
|
// do so here - else activate the workspace on which this
|
|
|
|
// window is
|
|
|
|
if (window->Flags() & B_NOT_ANCHORED_ON_ACTIVATE) {
|
|
|
|
// bring the window to the current workspace
|
|
|
|
// TODO: what if this window is on multiple workspaces?!?
|
|
|
|
uint32 workspaces = workspace_to_workspaces(fCurrentWorkspace);
|
|
|
|
SetWindowWorkspaces(window, workspaces);
|
|
|
|
} else {
|
|
|
|
// switch to the workspace on which this window is
|
|
|
|
// (we'll take the first one that the window is on)
|
|
|
|
uint32 workspaces = window->Workspaces();
|
|
|
|
for (int32 i = 0; i < 32; i++) {
|
|
|
|
uint32 workspace = workspace_to_workspaces(i);
|
|
|
|
if (workspaces & workspace) {
|
|
|
|
SetWorkspace(i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (window == FrontWindow()) {
|
2006-02-27 17:05:04 +03:00
|
|
|
// see if there is a normal B_AVOID_FRONT window still in front of us
|
|
|
|
WindowLayer* avoidsFront = window->NextWindow(fCurrentWorkspace);
|
|
|
|
while (avoidsFront && avoidsFront->IsNormal()
|
|
|
|
&& (avoidsFront->Flags() & B_AVOID_FRONT) == 0) {
|
|
|
|
avoidsFront = avoidsFront->NextWindow(fCurrentWorkspace);
|
|
|
|
}
|
2005-12-09 16:17:43 +03:00
|
|
|
|
2006-02-27 17:05:04 +03:00
|
|
|
if (avoidsFront == NULL) {
|
|
|
|
// we're already the frontmost window, we might just not have focus yet
|
2006-04-17 17:23:45 +04:00
|
|
|
if ((window->Flags() & B_AVOID_FOCUS) == 0)
|
|
|
|
SetFocusWindow(window);
|
2006-02-27 17:05:04 +03:00
|
|
|
|
|
|
|
UnlockAllWindows();
|
|
|
|
return;
|
|
|
|
}
|
2005-12-09 16:17:43 +03:00
|
|
|
}
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
// we don't need to redraw what is currently
|
|
|
|
// visible of the window
|
|
|
|
BRegion clean(window->VisibleRegion());
|
|
|
|
WindowList windows(kWorkingList);
|
|
|
|
|
|
|
|
WindowLayer* frontmost = window->Frontmost();
|
|
|
|
|
|
|
|
_CurrentWindows().RemoveWindow(window);
|
|
|
|
windows.AddWindow(window);
|
|
|
|
|
|
|
|
if (frontmost != NULL && frontmost->IsModal()) {
|
|
|
|
// all modal windows follow their subsets to the front
|
|
|
|
// (ie. they are staying in front of them, but they are
|
|
|
|
// not supposed to change their order because of that)
|
|
|
|
|
|
|
|
WindowLayer* nextModal;
|
|
|
|
for (WindowLayer* modal = frontmost; modal != NULL; modal = nextModal) {
|
|
|
|
// get the next modal window
|
|
|
|
nextModal = modal->NextWindow(fCurrentWorkspace);
|
|
|
|
while (nextModal != NULL && !nextModal->IsModal()) {
|
|
|
|
nextModal = nextModal->NextWindow(fCurrentWorkspace);
|
|
|
|
}
|
|
|
|
if (nextModal != NULL && !nextModal->HasInSubset(window))
|
|
|
|
nextModal = NULL;
|
|
|
|
|
|
|
|
_CurrentWindows().RemoveWindow(modal);
|
|
|
|
windows.AddWindow(modal);
|
|
|
|
}
|
2005-11-30 18:47:01 +03:00
|
|
|
}
|
2005-11-30 13:45:01 +03:00
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
_BringWindowsToFront(windows, kWorkingList, true);
|
2006-04-17 17:23:45 +04:00
|
|
|
if ((window->Flags() & B_AVOID_FOCUS) == 0)
|
|
|
|
SetFocusWindow(window);
|
2005-12-09 16:17:43 +03:00
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::SendWindowBehind(WindowLayer* window, WindowLayer* behindOf)
|
|
|
|
{
|
2006-04-16 17:49:41 +04:00
|
|
|
// TODO: should the "not in current workspace" be handled anyway?
|
|
|
|
// (the code below would have to be changed then, though)
|
|
|
|
if (window == BackWindow()
|
|
|
|
|| !window->InWorkspace(fCurrentWorkspace)
|
|
|
|
|| (behindOf != NULL && !behindOf->InWorkspace(fCurrentWorkspace))
|
|
|
|
|| !LockAllWindows())
|
2005-12-08 15:41:19 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
// Is this a valid behindOf window?
|
|
|
|
if (behindOf != NULL && window->HasInSubset(behindOf))
|
|
|
|
behindOf = NULL;
|
|
|
|
|
|
|
|
// what is currently visible of the window
|
|
|
|
// might be dirty after the window is send to back
|
|
|
|
BRegion dirty(window->VisibleRegion());
|
|
|
|
|
|
|
|
// detach window and re-attach at desired position
|
|
|
|
WindowLayer* backmost = window->Backmost(behindOf);
|
|
|
|
|
|
|
|
_CurrentWindows().RemoveWindow(window);
|
|
|
|
_CurrentWindows().AddWindow(window, backmost
|
|
|
|
? backmost->NextWindow(fCurrentWorkspace) : BackWindow());
|
|
|
|
|
|
|
|
BRegion dummy;
|
|
|
|
_RebuildClippingForAllWindows(dummy);
|
|
|
|
|
|
|
|
// mark everything dirty that is no longer visible
|
|
|
|
BRegion clean(window->VisibleRegion());
|
|
|
|
dirty.Exclude(&clean);
|
|
|
|
MarkDirty(dirty);
|
|
|
|
|
|
|
|
_UpdateFronts();
|
2006-03-16 00:33:12 +03:00
|
|
|
SetFocusWindow(_CurrentWindows().LastWindow());
|
2006-03-14 01:05:49 +03:00
|
|
|
_WindowChanged(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::ShowWindow(WindowLayer* window)
|
|
|
|
{
|
|
|
|
if (!window->IsHidden())
|
|
|
|
return;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
LockAllWindows();
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
window->SetHidden(false);
|
2005-12-08 16:32:54 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (window->InWorkspace(fCurrentWorkspace)) {
|
2005-12-08 16:32:54 +03:00
|
|
|
_ShowWindow(window, true);
|
2005-12-09 16:17:43 +03:00
|
|
|
_UpdateSubsetWorkspaces(window);
|
2005-12-08 16:32:54 +03:00
|
|
|
ActivateWindow(window);
|
|
|
|
} else {
|
|
|
|
// then we don't need to send the fake mouse event either
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-08 16:32:54 +03:00
|
|
|
return;
|
|
|
|
}
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 19:20:01 +03:00
|
|
|
if (WorkspacesLayer* layer = dynamic_cast<WorkspacesLayer*>(window->TopLayer()))
|
|
|
|
fWorkspacesLayer = layer;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-11-30 13:45:01 +03:00
|
|
|
|
|
|
|
// If the mouse cursor is directly over the newly visible window,
|
|
|
|
// we'll send a fake mouse moved message to the window, so that
|
|
|
|
// it knows the mouse is over it.
|
|
|
|
|
2006-04-01 19:43:25 +04:00
|
|
|
_SendFakeMouseMoved(window);
|
2005-11-30 13:45:01 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::HideWindow(WindowLayer* window)
|
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
if (window->IsHidden())
|
2005-11-30 18:47:01 +03:00
|
|
|
return;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!LockAllWindows())
|
2005-12-08 15:41:19 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
window->SetHidden(true);
|
2006-03-13 00:49:35 +03:00
|
|
|
if (fMouseEventWindow == window)
|
|
|
|
fMouseEventWindow = NULL;
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (window->InWorkspace(fCurrentWorkspace)) {
|
|
|
|
_UpdateSubsetWorkspaces(window);
|
2005-12-08 16:32:54 +03:00
|
|
|
_HideWindow(window);
|
|
|
|
_UpdateFronts();
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-08 16:32:54 +03:00
|
|
|
if (FocusWindow() == window)
|
2006-03-16 00:33:12 +03:00
|
|
|
SetFocusWindow(_CurrentWindows().LastWindow());
|
2005-12-08 16:32:54 +03:00
|
|
|
}
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2006-03-14 01:05:49 +03:00
|
|
|
if (fWorkspacesLayer != NULL)
|
|
|
|
fWorkspacesLayer->WindowRemoved(window);
|
|
|
|
|
2005-12-09 19:20:01 +03:00
|
|
|
if (dynamic_cast<WorkspacesLayer*>(window->TopLayer()) != NULL)
|
|
|
|
fWorkspacesLayer = NULL;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2006-04-01 19:43:25 +04:00
|
|
|
|
|
|
|
if (window == fWindowUnderMouse)
|
|
|
|
_SendFakeMouseMoved();
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Shows the window on the screen - it does this independently of the
|
|
|
|
WindowLayer::IsHidden() state.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
Desktop::_ShowWindow(WindowLayer* window, bool affectsOtherWindows)
|
|
|
|
{
|
|
|
|
BRegion background;
|
|
|
|
_RebuildClippingForAllWindows(background);
|
|
|
|
_SetBackground(background);
|
2005-12-09 19:20:01 +03:00
|
|
|
_WindowChanged(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
BRegion dirty(window->VisibleRegion());
|
|
|
|
|
|
|
|
if (!affectsOtherWindows) {
|
|
|
|
// everything that is now visible in the
|
|
|
|
// window needs a redraw, but other windows
|
|
|
|
// are not affected, we can call ProcessDirtyRegion()
|
|
|
|
// of the window, and don't have to use MarkDirty()
|
|
|
|
window->ProcessDirtyRegion(dirty);
|
|
|
|
} else
|
|
|
|
MarkDirty(dirty);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Hides the window from the screen - it does this independently of the
|
|
|
|
WindowLayer::IsHidden() state.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
Desktop::_HideWindow(WindowLayer* window)
|
|
|
|
{
|
|
|
|
// after rebuilding the clipping,
|
|
|
|
// this window will not have a visible
|
|
|
|
// region anymore, so we need to remember
|
|
|
|
// it now
|
|
|
|
// (actually that's not true, since
|
|
|
|
// hidden windows are excluded from the
|
|
|
|
// clipping calculation, but anyways)
|
|
|
|
BRegion dirty(window->VisibleRegion());
|
|
|
|
|
|
|
|
BRegion background;
|
|
|
|
_RebuildClippingForAllWindows(background);
|
|
|
|
_SetBackground(background);
|
2005-12-09 19:20:01 +03:00
|
|
|
_WindowChanged(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
MarkDirty(dirty);
|
2005-11-30 13:45:01 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-01 13:31:30 +03:00
|
|
|
void
|
2006-03-14 01:05:49 +03:00
|
|
|
Desktop::MoveWindowBy(WindowLayer* window, float x, float y, int32 workspace)
|
2005-12-01 13:31:30 +03:00
|
|
|
{
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!LockAllWindows())
|
2005-12-08 15:41:19 +03:00
|
|
|
return;
|
|
|
|
|
2006-03-14 01:05:49 +03:00
|
|
|
if (workspace == -1)
|
|
|
|
workspace = fCurrentWorkspace;
|
|
|
|
|
|
|
|
if (!window->IsVisible() || workspace != fCurrentWorkspace) {
|
|
|
|
if (workspace != fCurrentWorkspace) {
|
|
|
|
// move the window on another workspace - this doesn't change it's
|
|
|
|
// current position
|
|
|
|
if (window->Anchor(workspace).position == kInvalidWindowPosition)
|
|
|
|
window->Anchor(workspace).position = window->Frame().LeftTop();
|
|
|
|
|
|
|
|
window->Anchor(workspace).position += BPoint(x, y);
|
|
|
|
_WindowChanged(window);
|
|
|
|
} else
|
|
|
|
window->MoveBy(x, y);
|
|
|
|
|
2006-02-09 15:38:46 +03:00
|
|
|
UnlockAllWindows();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-02-09 18:07:56 +03:00
|
|
|
// the dirty region starts with the visible area of the window being moved
|
|
|
|
BRegion newDirtyRegion(window->VisibleRegion());
|
|
|
|
|
|
|
|
// no more drawing for DirectWindows
|
|
|
|
window->ServerWindow()->HandleDirectConnection(B_DIRECT_STOP);
|
|
|
|
|
|
|
|
window->MoveBy(x, y);
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
BRegion background;
|
|
|
|
_RebuildClippingForAllWindows(background);
|
|
|
|
|
|
|
|
// construct the region that is possible to be blitted
|
|
|
|
// to move the contents of the window
|
|
|
|
BRegion copyRegion(window->VisibleRegion());
|
|
|
|
copyRegion.OffsetBy(-x, -y);
|
|
|
|
copyRegion.IntersectWith(&newDirtyRegion);
|
2005-12-21 01:23:48 +03:00
|
|
|
// newDirtyRegion == the windows old visible region
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
// include the the new visible region of the window being
|
|
|
|
// moved into the dirty region (for now)
|
|
|
|
newDirtyRegion.Include(&window->VisibleRegion());
|
|
|
|
|
2005-12-20 00:18:23 +03:00
|
|
|
GetDrawingEngine()->CopyRegion(©Region, x, y);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2006-02-09 18:07:56 +03:00
|
|
|
// allow DirectWindows to draw again after the visual
|
|
|
|
// content is at the new location
|
|
|
|
window->ServerWindow()->HandleDirectConnection(B_DIRECT_START | B_BUFFER_MOVED);
|
|
|
|
|
2005-12-20 00:18:23 +03:00
|
|
|
// in the dirty region, exclude the parts that we
|
|
|
|
// could move by blitting
|
|
|
|
copyRegion.OffsetBy(x, y);
|
|
|
|
newDirtyRegion.Exclude(©Region);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
MarkDirty(newDirtyRegion);
|
|
|
|
_SetBackground(background);
|
2005-12-09 19:20:01 +03:00
|
|
|
_WindowChanged(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-01 13:31:30 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::ResizeWindowBy(WindowLayer* window, float x, float y)
|
|
|
|
{
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!LockAllWindows())
|
2005-12-08 15:41:19 +03:00
|
|
|
return;
|
|
|
|
|
2006-02-09 15:38:46 +03:00
|
|
|
if (!window->IsVisible()) {
|
|
|
|
window->ResizeBy(x, y, NULL);
|
|
|
|
UnlockAllWindows();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2005-12-21 01:23:48 +03:00
|
|
|
// the dirty region for the inside of the window is
|
|
|
|
// constructed by the window itself in ResizeBy()
|
2005-12-08 15:41:19 +03:00
|
|
|
BRegion newDirtyRegion;
|
2005-12-21 01:23:48 +03:00
|
|
|
// track the dirty region outside the window in case
|
|
|
|
// it is shrunk in "previouslyOccupiedRegion"
|
2005-12-08 15:41:19 +03:00
|
|
|
BRegion previouslyOccupiedRegion(window->VisibleRegion());
|
|
|
|
|
|
|
|
window->ResizeBy(x, y, &newDirtyRegion);
|
|
|
|
|
|
|
|
BRegion background;
|
|
|
|
_RebuildClippingForAllWindows(background);
|
|
|
|
|
2005-12-21 01:23:48 +03:00
|
|
|
// we just care for the region outside the window
|
2005-12-08 15:41:19 +03:00
|
|
|
previouslyOccupiedRegion.Exclude(&window->VisibleRegion());
|
|
|
|
|
2005-12-21 01:23:48 +03:00
|
|
|
// make sure the window cannot mark stuff dirty outside
|
|
|
|
// its visible region...
|
2005-12-08 15:41:19 +03:00
|
|
|
newDirtyRegion.IntersectWith(&window->VisibleRegion());
|
2005-12-21 01:23:48 +03:00
|
|
|
// ...because we do this outself
|
2005-12-08 15:41:19 +03:00
|
|
|
newDirtyRegion.Include(&previouslyOccupiedRegion);
|
|
|
|
|
|
|
|
MarkDirty(newDirtyRegion);
|
|
|
|
_SetBackground(background);
|
2005-12-09 19:20:01 +03:00
|
|
|
_WindowChanged(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-01 13:31:30 +03:00
|
|
|
}
|
|
|
|
|
2006-05-25 16:01:28 +04:00
|
|
|
|
2006-05-24 15:05:25 +04:00
|
|
|
bool
|
2006-05-24 04:42:34 +04:00
|
|
|
Desktop::SetWindowTabLocation(WindowLayer* window, float location)
|
|
|
|
{
|
|
|
|
if (!LockAllWindows())
|
2006-05-24 15:05:25 +04:00
|
|
|
return false;
|
2006-05-24 04:42:34 +04:00
|
|
|
|
|
|
|
BRegion dirty;
|
2006-05-24 15:05:25 +04:00
|
|
|
bool changed = window->SetTabLocation(location, dirty);
|
2006-05-24 04:42:34 +04:00
|
|
|
|
2006-05-24 15:05:25 +04:00
|
|
|
if (changed && window->IsVisible() && dirty.CountRects() > 0) {
|
2006-05-24 04:42:34 +04:00
|
|
|
BRegion stillAvailableOnScreen;
|
|
|
|
_RebuildClippingForAllWindows(stillAvailableOnScreen);
|
|
|
|
_SetBackground(stillAvailableOnScreen);
|
2006-05-29 22:41:16 +04:00
|
|
|
|
|
|
|
_WindowChanged(window);
|
2006-05-24 04:42:34 +04:00
|
|
|
_TriggerWindowRedrawing(dirty);
|
|
|
|
}
|
|
|
|
|
|
|
|
UnlockAllWindows();
|
2006-05-24 15:05:25 +04:00
|
|
|
|
|
|
|
return changed;
|
2006-05-24 04:42:34 +04:00
|
|
|
}
|
|
|
|
|
2005-12-01 13:31:30 +03:00
|
|
|
|
2006-05-25 16:01:28 +04:00
|
|
|
bool
|
|
|
|
Desktop::SetWindowDecoratorSettings(WindowLayer* window,
|
|
|
|
const BMessage& settings)
|
|
|
|
{
|
|
|
|
// TODO: almost exact code duplication to above function...
|
|
|
|
|
|
|
|
if (!LockAllWindows())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
BRegion dirty;
|
|
|
|
bool changed = window->SetDecoratorSettings(settings, dirty);
|
|
|
|
|
|
|
|
if (changed && window->IsVisible() && dirty.CountRects() > 0) {
|
|
|
|
BRegion stillAvailableOnScreen;
|
|
|
|
_RebuildClippingForAllWindows(stillAvailableOnScreen);
|
|
|
|
_SetBackground(stillAvailableOnScreen);
|
|
|
|
|
|
|
|
_TriggerWindowRedrawing(dirty);
|
|
|
|
}
|
|
|
|
|
|
|
|
UnlockAllWindows();
|
|
|
|
|
|
|
|
return changed;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
/*!
|
|
|
|
Updates the workspaces of all subset windows with regard to the
|
|
|
|
specifed window.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
Desktop::_UpdateSubsetWorkspaces(WindowLayer* window)
|
|
|
|
{
|
2006-03-08 15:56:56 +03:00
|
|
|
STRACE(("_UpdateSubsetWorkspaces(window %p, %s)\n", window, window->Title()));
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
// if the window is hidden, the subset windows are up-to-date already
|
|
|
|
if (!window->IsNormal() || window->IsHidden())
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (WindowLayer* subset = fSubsetWindows.FirstWindow(); subset != NULL;
|
|
|
|
subset = subset->NextWindow(kSubsetList)) {
|
|
|
|
if (subset->Feel() == B_MODAL_ALL_WINDOW_FEEL
|
|
|
|
|| subset->Feel() == B_FLOATING_ALL_WINDOW_FEEL) {
|
|
|
|
// These windows are always visible on all workspaces,
|
|
|
|
// no need to update them.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (subset->IsFloating()) {
|
|
|
|
// Floating windows are inserted and removed to the current
|
|
|
|
// workspace as the need arises - they are not handled here
|
|
|
|
// but in _UpdateFront()
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (subset->HasInSubset(window)) {
|
|
|
|
// adopt the workspace change
|
|
|
|
SetWindowWorkspaces(subset, subset->SubsetWorkspaces());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-11-30 18:47:01 +03:00
|
|
|
/*!
|
|
|
|
\brief Adds or removes the window to or from the workspaces it's on.
|
|
|
|
*/
|
2005-11-29 02:36:59 +03:00
|
|
|
void
|
2005-11-30 18:47:01 +03:00
|
|
|
Desktop::_ChangeWindowWorkspaces(WindowLayer* window, uint32 oldWorkspaces,
|
|
|
|
uint32 newWorkspaces)
|
2005-11-29 02:36:59 +03:00
|
|
|
{
|
2005-12-08 17:45:42 +03:00
|
|
|
// apply changes to the workspaces' window lists
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
LockAllWindows();
|
2005-11-30 18:47:01 +03:00
|
|
|
|
|
|
|
for (int32 i = 0; i < kMaxWorkspaces; i++) {
|
2005-12-09 16:28:28 +03:00
|
|
|
if (workspace_in_workspaces(i, oldWorkspaces)) {
|
2005-11-30 18:47:01 +03:00
|
|
|
// window is on this workspace, is it anymore?
|
2005-12-09 16:28:28 +03:00
|
|
|
if (!workspace_in_workspaces(i, newWorkspaces)) {
|
2005-12-09 16:17:43 +03:00
|
|
|
_Windows(i).RemoveWindow(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (i == CurrentWorkspace()) {
|
|
|
|
// remove its appearance from the current workspace
|
|
|
|
window->SetCurrentWorkspace(-1);
|
|
|
|
|
|
|
|
if (!window->IsHidden())
|
|
|
|
_HideWindow(window);
|
|
|
|
}
|
2005-11-30 18:47:01 +03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// window was not on this workspace, is it now?
|
2005-12-09 16:28:28 +03:00
|
|
|
if (workspace_in_workspaces(i, newWorkspaces)) {
|
2005-12-09 16:17:43 +03:00
|
|
|
_Windows(i).AddWindow(window,
|
|
|
|
window->Frontmost(_Windows(i).FirstWindow(), i));
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
if (i == CurrentWorkspace()) {
|
2005-12-09 16:17:43 +03:00
|
|
|
// make the window visible in current workspace
|
|
|
|
window->SetCurrentWorkspace(fCurrentWorkspace);
|
|
|
|
|
|
|
|
if (!window->IsHidden()) {
|
|
|
|
// this only affects other windows if this windows has floating or
|
|
|
|
// modal windows that need to be shown as well
|
|
|
|
// TODO: take care of this
|
|
|
|
_ShowWindow(window, FrontWindow() == window);
|
|
|
|
}
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
2005-11-30 18:47:01 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-12-08 17:45:42 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
// take care about modals and floating windows
|
|
|
|
_UpdateSubsetWorkspaces(window);
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-11-30 18:47:01 +03:00
|
|
|
}
|
2004-06-19 17:04:50 +04:00
|
|
|
|
2005-02-28 23:23:51 +03:00
|
|
|
|
2005-11-30 18:47:01 +03:00
|
|
|
void
|
|
|
|
Desktop::SetWindowWorkspaces(WindowLayer* window, uint32 workspaces)
|
|
|
|
{
|
2005-12-12 13:41:31 +03:00
|
|
|
LockAllWindows();
|
2005-12-08 17:45:42 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (window->IsNormal() && workspaces == B_CURRENT_WORKSPACE)
|
2005-11-30 18:47:01 +03:00
|
|
|
workspaces = workspace_to_workspaces(CurrentWorkspace());
|
2005-02-28 23:23:51 +03:00
|
|
|
|
2005-12-21 16:50:45 +03:00
|
|
|
uint32 oldWorkspaces = window->Workspaces();
|
|
|
|
|
|
|
|
window->WorkspacesChanged(oldWorkspaces, workspaces);
|
|
|
|
_ChangeWindowWorkspaces(window, oldWorkspaces, workspaces);
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-02-28 23:23:51 +03:00
|
|
|
}
|
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
|
2006-01-03 14:36:45 +03:00
|
|
|
/*! \brief Adds the window to the desktop.
|
|
|
|
At this point, the window is still hidden and must be shown explicetly
|
|
|
|
via ShowWindow().
|
|
|
|
*/
|
2005-04-25 18:14:09 +04:00
|
|
|
void
|
2005-11-30 18:47:01 +03:00
|
|
|
Desktop::AddWindow(WindowLayer *window)
|
2005-02-28 23:23:51 +03:00
|
|
|
{
|
2005-12-12 13:41:31 +03:00
|
|
|
LockAllWindows();
|
2005-02-28 23:23:51 +03:00
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
fAllWindows.AddWindow(window);
|
2005-12-09 16:17:43 +03:00
|
|
|
if (!window->IsNormal())
|
|
|
|
fSubsetWindows.AddWindow(window);
|
2005-11-30 18:47:01 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
if (window->IsNormal()) {
|
|
|
|
if (window->Workspaces() == B_CURRENT_WORKSPACE)
|
|
|
|
window->SetWorkspaces(workspace_to_workspaces(CurrentWorkspace()));
|
|
|
|
} else {
|
|
|
|
// subset windows are visible on all workspaces their subset is on
|
2006-01-03 19:12:52 +03:00
|
|
|
window->SetWorkspaces(window->SubsetWorkspaces());
|
2005-12-09 16:17:43 +03:00
|
|
|
}
|
2005-11-29 02:36:59 +03:00
|
|
|
|
2005-11-30 18:47:01 +03:00
|
|
|
_ChangeWindowWorkspaces(window, 0, window->Workspaces());
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2004-01-12 01:12:55 +03:00
|
|
|
}
|
2004-01-22 03:32:07 +03:00
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
|
|
|
|
void
|
2005-11-30 18:47:01 +03:00
|
|
|
Desktop::RemoveWindow(WindowLayer *window)
|
2004-01-22 03:32:07 +03:00
|
|
|
{
|
2005-12-12 13:41:31 +03:00
|
|
|
LockAllWindows();
|
2005-11-30 18:47:01 +03:00
|
|
|
|
2005-12-09 17:23:51 +03:00
|
|
|
if (!window->IsHidden())
|
|
|
|
HideWindow(window);
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
fAllWindows.RemoveWindow(window);
|
2005-12-09 16:17:43 +03:00
|
|
|
if (!window->IsNormal())
|
|
|
|
fSubsetWindows.RemoveWindow(window);
|
|
|
|
|
2005-11-30 18:47:01 +03:00
|
|
|
_ChangeWindowWorkspaces(window, window->Workspaces(), 0);
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-11-30 18:47:01 +03:00
|
|
|
|
|
|
|
// make sure this window won't get any events anymore
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
EventDispatcher().RemoveTarget(window->EventTarget());
|
2005-11-30 18:47:01 +03:00
|
|
|
}
|
2005-11-29 02:36:59 +03:00
|
|
|
|
Have I said input event handling is done?
* didn't realize that mouse events always go to the view under the mouse, not
only if its the focus window (FFM can really do harm, after all :-)).
* removed a TODO from the list: EventDispatcher::Target is now a public
class EventTarget, and every ServerWindow has one.
* as a result, EventDispatcher no longer manages targets itself, it
just maintains a list of them. You no longer set messengers, you
only set targets.
* customization of the message filters, they no longer inherit from
BMessageFilter (but EventFilter).
* a message target is no longer set explicetly anywhere, it's only
changed in the message filters if needed.
* therefore, no more locking mess in the EventDispatcher needed.
* this also made the EventDispatcher::fLastFocus stuff superfluous.
* moved the RootLayer::MouseEventHandler() into the message filter.
* Replaced RootLayer::_ChildAt() with WindowAt().
* WindowLayer now has an idea if it has focus or not, it no longer needs
to query the RootLayer for this - maybe we should rename "focus" to
"active", though (as far as layers are concerned).
* the "_view_token" data is now added from the EventDispatcher, not
the (Window)Layer class anymore.
* removed Layer::MouseWheelChanged() as we currently don't need it
(if the need arises, we can add it back later again)
* there is still no mouse moved message sent when opening a window
under the cursor, though...
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@15228 a95241bf-73f2-0310-859d-f6bbb57e9c96
2005-11-29 19:01:41 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
bool
|
|
|
|
Desktop::AddWindowToSubset(WindowLayer* subset, WindowLayer* window)
|
|
|
|
{
|
|
|
|
if (!subset->AddToSubset(window))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
_ChangeWindowWorkspaces(subset, subset->Workspaces(), subset->SubsetWorkspaces());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::RemoveWindowFromSubset(WindowLayer* subset, WindowLayer* window)
|
|
|
|
{
|
|
|
|
subset->RemoveFromSubset(window);
|
|
|
|
_ChangeWindowWorkspaces(subset, subset->Workspaces(), subset->SubsetWorkspaces());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-11-30 18:47:01 +03:00
|
|
|
void
|
2005-12-08 15:41:19 +03:00
|
|
|
Desktop::SetWindowLook(WindowLayer *window, window_look newLook)
|
2005-11-30 18:47:01 +03:00
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
if (window->Look() == newLook)
|
|
|
|
return;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!LockAllWindows())
|
2005-12-08 15:41:19 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
BRegion dirty;
|
|
|
|
window->SetLook(newLook, &dirty);
|
|
|
|
// TODO: test what happens when the window
|
|
|
|
// finds out it needs to resize itself...
|
|
|
|
|
|
|
|
BRegion stillAvailableOnScreen;
|
|
|
|
_RebuildClippingForAllWindows(stillAvailableOnScreen);
|
|
|
|
_SetBackground(stillAvailableOnScreen);
|
2005-12-09 19:20:01 +03:00
|
|
|
_WindowChanged(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
_TriggerWindowRedrawing(dirty);
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2003-08-31 21:38:34 +04:00
|
|
|
}
|
2004-01-22 03:32:07 +03:00
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
|
|
|
|
void
|
2005-12-08 15:41:19 +03:00
|
|
|
Desktop::SetWindowFeel(WindowLayer *window, window_feel newFeel)
|
2005-04-21 22:57:34 +04:00
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
if (window->Feel() == newFeel)
|
|
|
|
return;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
LockAllWindows();
|
2005-12-08 15:41:19 +03:00
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
bool wasNormal = window->IsNormal();
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
window->SetFeel(newFeel);
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
// move the window out of or into the subset window list as needed
|
|
|
|
if (window->IsNormal() && !wasNormal)
|
|
|
|
fSubsetWindows.RemoveWindow(window);
|
|
|
|
else if (!window->IsNormal() && wasNormal)
|
|
|
|
fSubsetWindows.AddWindow(window);
|
|
|
|
|
|
|
|
// A normal window that was once a floating or modal window will
|
|
|
|
// adopt the window's current workspaces
|
|
|
|
|
|
|
|
if (!window->IsNormal())
|
|
|
|
_ChangeWindowWorkspaces(window, window->Workspaces(), window->SubsetWorkspaces());
|
|
|
|
|
2005-12-09 16:28:28 +03:00
|
|
|
// make sure the window has the correct position in the window lists
|
2005-12-09 16:17:43 +03:00
|
|
|
// (ie. all floating windows have to be on the top, ...)
|
|
|
|
|
2005-12-09 16:28:28 +03:00
|
|
|
for (int32 i = 0; i < kMaxWorkspaces; i++) {
|
|
|
|
if (!workspace_in_workspaces(i, window->Workspaces()))
|
|
|
|
continue;
|
|
|
|
|
2006-04-13 18:44:00 +04:00
|
|
|
bool changed = false;
|
|
|
|
BRegion visibleBefore;
|
|
|
|
if (i == fCurrentWorkspace && window->IsVisible())
|
|
|
|
visibleBefore = window->VisibleRegion();
|
|
|
|
|
|
|
|
WindowLayer* backmost = window->Backmost(_Windows(i).LastWindow(), i);
|
|
|
|
if (backmost != NULL) {
|
|
|
|
// check if the backmost window is really behind it
|
|
|
|
WindowLayer* previous = window->PreviousWindow(i);
|
|
|
|
while (previous != NULL) {
|
|
|
|
if (previous == backmost)
|
|
|
|
break;
|
2005-12-09 17:03:00 +03:00
|
|
|
|
2006-04-13 18:44:00 +04:00
|
|
|
previous = previous->PreviousWindow(i);
|
|
|
|
}
|
2005-12-09 17:03:00 +03:00
|
|
|
|
2006-04-13 18:44:00 +04:00
|
|
|
if (previous == NULL) {
|
|
|
|
// need to reinsert window before its backmost window
|
|
|
|
_Windows(i).RemoveWindow(window);
|
|
|
|
_Windows(i).AddWindow(window, backmost->NextWindow(i));
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
}
|
2005-12-09 16:28:28 +03:00
|
|
|
|
2006-04-13 18:44:00 +04:00
|
|
|
WindowLayer* frontmost = window->Frontmost(_Windows(i).FirstWindow(), i);
|
|
|
|
if (frontmost != NULL) {
|
|
|
|
// check if the frontmost window is really in front of it
|
|
|
|
WindowLayer* next = window->NextWindow(i);
|
|
|
|
while (next != NULL) {
|
|
|
|
if (next == frontmost)
|
|
|
|
break;
|
|
|
|
|
|
|
|
next = next->NextWindow(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (next == NULL) {
|
|
|
|
// need to reinsert window behind its frontmost window
|
|
|
|
_Windows(i).RemoveWindow(window);
|
|
|
|
_Windows(i).AddWindow(window, frontmost);
|
|
|
|
changed = true;
|
|
|
|
}
|
2005-12-09 16:28:28 +03:00
|
|
|
}
|
|
|
|
|
2006-04-13 18:44:00 +04:00
|
|
|
if (i == fCurrentWorkspace && changed) {
|
|
|
|
BRegion dummy;
|
|
|
|
_RebuildClippingForAllWindows(dummy);
|
|
|
|
|
|
|
|
// mark everything dirty that is no longer visible, or
|
|
|
|
// is now visible and wasn't before
|
|
|
|
BRegion visibleAfter(window->VisibleRegion());
|
|
|
|
BRegion dirty(visibleAfter);
|
|
|
|
dirty.Exclude(&visibleBefore);
|
|
|
|
visibleBefore.Exclude(&visibleAfter);
|
|
|
|
dirty.Include(&visibleBefore);
|
|
|
|
|
|
|
|
MarkDirty(dirty);
|
2005-12-09 16:28:28 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-12-09 16:17:43 +03:00
|
|
|
_UpdateFronts();
|
2005-12-09 16:30:47 +03:00
|
|
|
|
|
|
|
if (window == FocusWindow() && !window->IsVisible())
|
2006-03-16 00:33:12 +03:00
|
|
|
SetFocusWindow(_CurrentWindows().LastWindow());
|
2005-12-10 16:47:13 +03:00
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-01 16:59:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2005-12-08 15:41:19 +03:00
|
|
|
Desktop::SetWindowFlags(WindowLayer *window, uint32 newFlags)
|
2005-12-01 16:59:04 +03:00
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
if (window->Flags() == newFlags)
|
|
|
|
return;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!LockAllWindows())
|
2005-12-08 15:41:19 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
BRegion dirty;
|
|
|
|
window->SetFlags(newFlags, &dirty);
|
|
|
|
// TODO: test what happens when the window
|
|
|
|
// finds out it needs to resize itself...
|
|
|
|
|
|
|
|
BRegion stillAvailableOnScreen;
|
|
|
|
_RebuildClippingForAllWindows(stillAvailableOnScreen);
|
|
|
|
_SetBackground(stillAvailableOnScreen);
|
2005-12-09 19:20:01 +03:00
|
|
|
_WindowChanged(window);
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
_TriggerWindowRedrawing(dirty);
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-04-21 22:57:34 +04:00
|
|
|
}
|
|
|
|
|
2005-04-25 18:14:09 +04:00
|
|
|
|
2005-12-09 01:15:12 +03:00
|
|
|
void
|
|
|
|
Desktop::SetWindowTitle(WindowLayer *window, const char* title)
|
|
|
|
{
|
2005-12-12 13:41:31 +03:00
|
|
|
if (!LockAllWindows())
|
2005-12-09 01:15:12 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
BRegion dirty;
|
|
|
|
window->SetTitle(title, dirty);
|
|
|
|
|
|
|
|
if (window->IsVisible() && dirty.CountRects() > 0) {
|
|
|
|
BRegion stillAvailableOnScreen;
|
|
|
|
_RebuildClippingForAllWindows(stillAvailableOnScreen);
|
|
|
|
_SetBackground(stillAvailableOnScreen);
|
|
|
|
|
|
|
|
_TriggerWindowRedrawing(dirty);
|
|
|
|
}
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-09 01:15:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-10 16:47:13 +03:00
|
|
|
/*!
|
|
|
|
Returns the window under the mouse cursor.
|
2005-12-29 17:15:59 +03:00
|
|
|
You need to have acquired the All Windows lock when calling this method.
|
2005-12-10 16:47:13 +03:00
|
|
|
*/
|
2005-12-08 15:41:19 +03:00
|
|
|
WindowLayer*
|
|
|
|
Desktop::WindowAt(BPoint where)
|
2005-02-28 23:23:51 +03:00
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
for (WindowLayer* window = _CurrentWindows().LastWindow(); window;
|
|
|
|
window = window->PreviousWindow(fCurrentWorkspace)) {
|
2005-12-29 17:15:59 +03:00
|
|
|
if (window->IsVisible() && window->VisibleRegion().Contains(where))
|
2005-12-08 15:41:19 +03:00
|
|
|
return window;
|
2005-02-28 23:23:51 +03:00
|
|
|
}
|
2005-04-25 18:14:09 +04:00
|
|
|
|
2005-07-05 22:14:24 +04:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
void
|
|
|
|
Desktop::SetMouseEventWindow(WindowLayer* window)
|
2005-10-31 22:35:46 +03:00
|
|
|
{
|
2005-12-08 15:41:19 +03:00
|
|
|
fMouseEventWindow = window;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-10 16:03:41 +03:00
|
|
|
void
|
|
|
|
Desktop::SetViewUnderMouse(const WindowLayer* window, int32 viewToken)
|
|
|
|
{
|
|
|
|
fWindowUnderMouse = window;
|
|
|
|
fViewUnderMouse = viewToken;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int32
|
|
|
|
Desktop::ViewUnderMouse(const WindowLayer* window)
|
|
|
|
{
|
|
|
|
if (fWindowUnderMouse == window)
|
|
|
|
return fViewUnderMouse;
|
|
|
|
|
|
|
|
return B_NULL_TOKEN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
WindowLayer *
|
|
|
|
Desktop::FindWindowLayerByClientToken(int32 token, team_id teamID)
|
|
|
|
{
|
|
|
|
for (WindowLayer *window = fAllWindows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(kAllWindowList)) {
|
|
|
|
if (window->ServerWindow()->ClientToken() == token
|
2005-12-10 16:47:13 +03:00
|
|
|
&& window->ServerWindow()->ClientTeam() == teamID) {
|
2005-12-08 15:41:19 +03:00
|
|
|
return window;
|
2005-12-10 16:47:13 +03:00
|
|
|
}
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
2005-10-31 22:35:46 +03:00
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
return NULL;
|
2005-10-31 22:35:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-09 21:53:36 +03:00
|
|
|
void
|
|
|
|
Desktop::MinimizeApplication(team_id team)
|
|
|
|
{
|
2006-04-07 23:14:25 +04:00
|
|
|
AutoWriteLocker locker(fWindowLock);
|
2006-02-09 21:53:36 +03:00
|
|
|
|
|
|
|
// Just minimize all windows of that application
|
|
|
|
|
|
|
|
for (WindowLayer *window = fAllWindows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(kAllWindowList)) {
|
|
|
|
if (window->ServerWindow()->ClientTeam() != team)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
window->ServerWindow()->NotifyMinimize(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::BringApplicationToFront(team_id team)
|
|
|
|
{
|
2006-04-07 23:14:25 +04:00
|
|
|
AutoWriteLocker locker(fWindowLock);
|
2006-02-09 21:53:36 +03:00
|
|
|
|
|
|
|
// TODO: for now, just maximize all windows of that application
|
|
|
|
|
|
|
|
for (WindowLayer *window = fAllWindows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(kAllWindowList)) {
|
|
|
|
if (window->ServerWindow()->ClientTeam() != team)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
window->ServerWindow()->NotifyMinimize(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-28 21:31:16 +03:00
|
|
|
void
|
|
|
|
Desktop::WindowAction(int32 windowToken, int32 action)
|
|
|
|
{
|
|
|
|
if (action != B_MINIMIZE_WINDOW && action != B_BRING_TO_FRONT)
|
|
|
|
return;
|
|
|
|
|
|
|
|
LockAllWindows();
|
|
|
|
|
|
|
|
::ServerWindow* serverWindow;
|
|
|
|
if (BPrivate::gDefaultTokens.GetToken(windowToken,
|
2006-03-10 14:42:58 +03:00
|
|
|
B_SERVER_TOKEN, (void**)&serverWindow) != B_OK) {
|
|
|
|
UnlockAllWindows();
|
2006-02-28 21:31:16 +03:00
|
|
|
return;
|
2006-03-10 14:42:58 +03:00
|
|
|
}
|
2006-02-28 21:31:16 +03:00
|
|
|
|
|
|
|
if (action == B_BRING_TO_FRONT
|
|
|
|
&& !serverWindow->Window()->IsMinimized()) {
|
|
|
|
// the window is visible, we just need to make it the front window
|
|
|
|
ActivateWindow(serverWindow->Window());
|
|
|
|
} else
|
|
|
|
serverWindow->NotifyMinimize(action == B_MINIMIZE_WINDOW);
|
|
|
|
|
|
|
|
UnlockAllWindows();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-07-05 22:14:24 +04:00
|
|
|
void
|
|
|
|
Desktop::WriteWindowList(team_id team, BPrivate::LinkSender& sender)
|
|
|
|
{
|
2006-04-07 23:14:25 +04:00
|
|
|
AutoWriteLocker locker(fWindowLock);
|
2005-07-05 22:14:24 +04:00
|
|
|
|
|
|
|
// compute the number of windows
|
|
|
|
|
|
|
|
int32 count = 0;
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
for (WindowLayer *window = fAllWindows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(kAllWindowList)) {
|
|
|
|
if (team < B_OK || window->ServerWindow()->ClientTeam() == team)
|
|
|
|
count++;
|
|
|
|
}
|
2005-07-05 22:14:24 +04:00
|
|
|
|
|
|
|
// write list
|
|
|
|
|
2005-11-30 22:56:44 +03:00
|
|
|
sender.StartMessage(B_OK);
|
2005-07-05 22:14:24 +04:00
|
|
|
sender.Attach<int32>(count);
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
for (WindowLayer *window = fAllWindows.FirstWindow(); window != NULL;
|
|
|
|
window = window->NextWindow(kAllWindowList)) {
|
|
|
|
if (team >= B_OK && window->ServerWindow()->ClientTeam() != team)
|
2005-07-05 22:14:24 +04:00
|
|
|
continue;
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
sender.Attach<int32>(window->ServerWindow()->ServerToken());
|
2005-07-05 22:14:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sender.Flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::WriteWindowInfo(int32 serverToken, BPrivate::LinkSender& sender)
|
|
|
|
{
|
2006-04-07 23:14:25 +04:00
|
|
|
AutoWriteLocker locker(fWindowLock);
|
2005-07-05 22:14:24 +04:00
|
|
|
BAutolock tokenLocker(BPrivate::gDefaultTokens);
|
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
::ServerWindow* window;
|
2005-07-05 22:14:24 +04:00
|
|
|
if (BPrivate::gDefaultTokens.GetToken(serverToken,
|
|
|
|
B_SERVER_TOKEN, (void**)&window) != B_OK) {
|
|
|
|
sender.StartMessage(B_ENTRY_NOT_FOUND);
|
|
|
|
sender.Flush();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
window_info info;
|
|
|
|
window->GetInfo(info);
|
|
|
|
|
|
|
|
int32 length = window->Title() ? strlen(window->Title()) : 0;
|
|
|
|
|
|
|
|
sender.StartMessage(B_OK);
|
|
|
|
sender.Attach<int32>(sizeof(window_info) + length + 1);
|
|
|
|
sender.Attach(&info, sizeof(window_info));
|
|
|
|
if (length > 0)
|
|
|
|
sender.Attach(window->Title(), length + 1);
|
|
|
|
else
|
|
|
|
sender.Attach<char>('\0');
|
|
|
|
sender.Flush();
|
2003-01-24 18:19:27 +03:00
|
|
|
}
|
2004-01-22 03:32:07 +03:00
|
|
|
|
2005-12-08 15:41:19 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::MarkDirty(BRegion& region)
|
|
|
|
{
|
|
|
|
if (region.CountRects() == 0)
|
|
|
|
return;
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
if (LockAllWindows()) {
|
2005-12-08 15:41:19 +03:00
|
|
|
// send redraw messages to all windows intersecting the dirty region
|
|
|
|
_TriggerWindowRedrawing(region);
|
|
|
|
|
2005-12-12 13:41:31 +03:00
|
|
|
UnlockAllWindows();
|
2005-12-08 15:41:19 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::_RebuildClippingForAllWindows(BRegion& stillAvailableOnScreen)
|
|
|
|
{
|
|
|
|
// the available region on screen starts with the entire screen area
|
|
|
|
// each window on the screen will take a portion from that area
|
|
|
|
|
|
|
|
// figure out what the entire screen area is
|
|
|
|
stillAvailableOnScreen = fScreenRegion;
|
|
|
|
|
|
|
|
// set clipping of each window
|
|
|
|
for (WindowLayer* window = _CurrentWindows().LastWindow(); window != NULL;
|
|
|
|
window = window->PreviousWindow(fCurrentWorkspace)) {
|
|
|
|
if (!window->IsHidden()) {
|
|
|
|
window->SetClipping(&stillAvailableOnScreen);
|
|
|
|
// that windows region is not available on screen anymore
|
|
|
|
stillAvailableOnScreen.Exclude(&window->VisibleRegion());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::_TriggerWindowRedrawing(BRegion& newDirtyRegion)
|
|
|
|
{
|
|
|
|
// send redraw messages to all windows intersecting the dirty region
|
|
|
|
for (WindowLayer* window = _CurrentWindows().LastWindow(); window != NULL;
|
|
|
|
window = window->PreviousWindow(fCurrentWorkspace)) {
|
|
|
|
if (!window->IsHidden()
|
|
|
|
&& newDirtyRegion.Intersects(window->VisibleRegion().Frame()))
|
|
|
|
window->ProcessDirtyRegion(newDirtyRegion);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Desktop::_SetBackground(BRegion& background)
|
|
|
|
{
|
|
|
|
// NOTE: the drawing operation is caried out
|
|
|
|
// in the clipping region rebuild, but it is
|
|
|
|
// ok actually, because it also avoids trails on
|
|
|
|
// moving windows
|
|
|
|
|
|
|
|
// remember the region not covered by any windows
|
|
|
|
// and redraw the dirty background
|
|
|
|
BRegion dirtyBackground(background);
|
|
|
|
dirtyBackground.Exclude(&fBackgroundRegion);
|
|
|
|
dirtyBackground.IntersectWith(&background);
|
|
|
|
fBackgroundRegion = background;
|
|
|
|
if (dirtyBackground.Frame().IsValid()) {
|
|
|
|
if (GetDrawingEngine()->Lock()) {
|
|
|
|
GetDrawingEngine()->FillRegion(dirtyBackground,
|
|
|
|
fWorkspaces[fCurrentWorkspace].Color());
|
|
|
|
|
|
|
|
GetDrawingEngine()->Unlock();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|