xref: /haiku/src/apps/deskbar/StatusView.cpp (revision 615d572db00c03d9de04a6d3ed0f3c50929e0635)
141281cf3SAxel Dörfler /*
241281cf3SAxel Dörfler Open Tracker License
341281cf3SAxel Dörfler 
441281cf3SAxel Dörfler Terms and Conditions
541281cf3SAxel Dörfler 
641281cf3SAxel Dörfler Copyright (c) 1991-2000, Be Incorporated. All rights reserved.
741281cf3SAxel Dörfler 
841281cf3SAxel Dörfler Permission is hereby granted, free of charge, to any person obtaining a copy of
941281cf3SAxel Dörfler this software and associated documentation files (the "Software"), to deal in
1041281cf3SAxel Dörfler the Software without restriction, including without limitation the rights to
1141281cf3SAxel Dörfler use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
1241281cf3SAxel Dörfler of the Software, and to permit persons to whom the Software is furnished to do
1341281cf3SAxel Dörfler so, subject to the following conditions:
1441281cf3SAxel Dörfler 
1541281cf3SAxel Dörfler The above copyright notice and this permission notice applies to all licensees
1641281cf3SAxel Dörfler and shall be included in all copies or substantial portions of the Software.
1741281cf3SAxel Dörfler 
1841281cf3SAxel Dörfler THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1941281cf3SAxel Dörfler IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF TITLE, MERCHANTABILITY,
2041281cf3SAxel Dörfler FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2141281cf3SAxel Dörfler BE INCORPORATED BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
2241281cf3SAxel Dörfler AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF, OR IN CONNECTION
2341281cf3SAxel Dörfler WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
2441281cf3SAxel Dörfler 
2541281cf3SAxel Dörfler Except as contained in this notice, the name of Be Incorporated shall not be
2641281cf3SAxel Dörfler used in advertising or otherwise to promote the sale, use or other dealings in
2741281cf3SAxel Dörfler this Software without prior written authorization from Be Incorporated.
2841281cf3SAxel Dörfler 
291ad8c760SFredrik Holmqvist Tracker(TM), Be(R), BeOS(R), and BeIA(TM) are trademarks or registered
301ad8c760SFredrik Holmqvist trademarks of Be Incorporated in the United States and other countries. Other
311ad8c760SFredrik Holmqvist brand product names are registered trademarks or trademarks of their respective
321ad8c760SFredrik Holmqvist holders.
3341281cf3SAxel Dörfler All rights reserved.
3441281cf3SAxel Dörfler */
3541281cf3SAxel Dörfler 
361cd61330SJohn Scipione 
371cd61330SJohn Scipione #include "StatusView.h"
3841281cf3SAxel Dörfler 
3941281cf3SAxel Dörfler #include <errno.h>
4041281cf3SAxel Dörfler #include <stdio.h>
4141281cf3SAxel Dörfler #include <string.h>
4241281cf3SAxel Dörfler #include <time.h>
4341281cf3SAxel Dörfler #include <unistd.h>
4441281cf3SAxel Dörfler #include <algorithm>
4541281cf3SAxel Dörfler 
4641281cf3SAxel Dörfler #include <fs_index.h>
4741281cf3SAxel Dörfler #include <fs_info.h>
4841281cf3SAxel Dörfler 
4941281cf3SAxel Dörfler #include <Application.h>
5041281cf3SAxel Dörfler #include <Beep.h>
51cb6afcb1SStephan Aßmus #include <Bitmap.h>
52ae7a6e3aSAdrien Destugues #include <Catalog.h>
53cb6afcb1SStephan Aßmus #include <ControlLook.h>
541cd61330SJohn Scipione #include <Debug.h>
5541281cf3SAxel Dörfler #include <Directory.h>
5641281cf3SAxel Dörfler #include <FindDirectory.h>
57ae7a6e3aSAdrien Destugues #include <Locale.h>
5841281cf3SAxel Dörfler #include <MenuItem.h>
5941281cf3SAxel Dörfler #include <NodeInfo.h>
6041281cf3SAxel Dörfler #include <NodeMonitor.h>
6141281cf3SAxel Dörfler #include <Path.h>
6241281cf3SAxel Dörfler #include <PopUpMenu.h>
6341281cf3SAxel Dörfler #include <Roster.h>
6441281cf3SAxel Dörfler #include <Screen.h>
6541281cf3SAxel Dörfler #include <Volume.h>
6641281cf3SAxel Dörfler #include <VolumeRoster.h>
6741281cf3SAxel Dörfler #include <Window.h>
6841281cf3SAxel Dörfler 
69*615d572dSJohn Scipione #include "icons.h"
70*615d572dSJohn Scipione 
71cb6afcb1SStephan Aßmus #include "BarApp.h"
72323b6546SOliver Tappe #include "DeskbarUtils.h"
73cb6afcb1SStephan Aßmus #include "ResourceSet.h"
7441281cf3SAxel Dörfler #include "StatusViewShelf.h"
7541281cf3SAxel Dörfler #include "TimeView.h"
7641281cf3SAxel Dörfler 
77*615d572dSJohn Scipione 
786d01b66dSAxel Dörfler using std::max;
7941281cf3SAxel Dörfler 
8041281cf3SAxel Dörfler #ifdef DB_ADDONS
8141281cf3SAxel Dörfler // Add-on support
8241281cf3SAxel Dörfler //
8341281cf3SAxel Dörfler // Item - internal item list (node, eref, etc)
8441281cf3SAxel Dörfler // Icon - physical replicant handed to the DeskbarClass class
8541281cf3SAxel Dörfler // AddOn - attribute based add-on
8641281cf3SAxel Dörfler 
8741281cf3SAxel Dörfler const char* const kInstantiateItemCFunctionName = "instantiate_deskbar_item";
8841281cf3SAxel Dörfler const char* const kInstantiateEntryCFunctionName = "instantiate_deskbar_entry";
893e6ff860SRene Gollent const char* const kReplicantSettingsFile = "Deskbar_replicants";
90c0ad1c91SRene Gollent const char* const kReplicantPathField = "replicant_path";
9141281cf3SAxel Dörfler 
92cb6afcb1SStephan Aßmus float sMinimumWindowWidth = kGutter + kMinimumTrayWidth + kDragRegionWidth;
93cb6afcb1SStephan Aßmus 
9441281cf3SAxel Dörfler 
9541281cf3SAxel Dörfler static void
9641281cf3SAxel Dörfler DumpItem(DeskbarItemInfo* item)
9741281cf3SAxel Dörfler {
9841281cf3SAxel Dörfler 	printf("is addon: %i, id: %li\n", item->isAddOn, item->id);
9966eba86fSAxel Dörfler 	printf("entry_ref:  %ld, %Ld, %s\n", item->entryRef.device,
10066eba86fSAxel Dörfler 		item->entryRef.directory, item->entryRef.name);
10141281cf3SAxel Dörfler 	printf("node_ref:  %ld, %Ld\n", item->nodeRef.device, item->nodeRef.node);
10241281cf3SAxel Dörfler }
10341281cf3SAxel Dörfler 
10441281cf3SAxel Dörfler 
10541281cf3SAxel Dörfler static void
10641281cf3SAxel Dörfler DumpList(BList* itemlist)
10741281cf3SAxel Dörfler {
10841281cf3SAxel Dörfler 	int32 count = itemlist->CountItems() - 1;
10941281cf3SAxel Dörfler 	if (count < 0) {
11041281cf3SAxel Dörfler 		printf("no items in list\n");
11141281cf3SAxel Dörfler 		return;
11241281cf3SAxel Dörfler 	}
11341281cf3SAxel Dörfler 	for (int32 i = count; i >= 0; i--) {
11441281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)itemlist->ItemAt(i);
11541281cf3SAxel Dörfler 		if (!item)
11641281cf3SAxel Dörfler 			continue;
11741281cf3SAxel Dörfler 
11841281cf3SAxel Dörfler 		DumpItem(item);
11941281cf3SAxel Dörfler 	}
12041281cf3SAxel Dörfler }
12141281cf3SAxel Dörfler #endif	/* DB_ADDONS */
12241281cf3SAxel Dörfler 
12341281cf3SAxel Dörfler 
124546208a5SOliver Tappe #undef B_TRANSLATION_CONTEXT
125546208a5SOliver Tappe #define B_TRANSLATION_CONTEXT "Tray"
126ae7a6e3aSAdrien Destugues 
1277da06231SAxel Dörfler // don't change the name of this view to anything other than "Status"!
1287da06231SAxel Dörfler 
12941281cf3SAxel Dörfler TReplicantTray::TReplicantTray(TBarView* parent, bool vertical)
1300d2ac94cSJohn Scipione 	:
1319ac6fee1SJohn Scipione 	BView(BRect(0, 0, 1, 1), "Status", B_FOLLOW_LEFT | B_FOLLOW_TOP,
1320d2ac94cSJohn Scipione 		B_WILL_DRAW | B_FRAME_EVENTS),
133573f748cSJohn Scipione 	fTime(NULL),
13441281cf3SAxel Dörfler 	fBarView(parent),
13541281cf3SAxel Dörfler 	fShelf(new TReplicantShelf(this)),
13641281cf3SAxel Dörfler 	fMultiRowMode(vertical),
137cb6afcb1SStephan Aßmus 	fMinimumTrayWidth(kMinimumTrayWidth),
13841281cf3SAxel Dörfler 	fAlignmentSupport(false)
13941281cf3SAxel Dörfler {
140cb6afcb1SStephan Aßmus 	// init the minimum window width according to the logo.
141cb6afcb1SStephan Aßmus 	const BBitmap* logoBitmap = AppResSet()->FindBitmap(B_MESSAGE_TYPE,
1429e8a970bSJonas Sundström 		R_LeafLogoBitmap);
143cb6afcb1SStephan Aßmus 	if (logoBitmap != NULL) {
144cb6afcb1SStephan Aßmus 		sMinimumWindowWidth = max_c(sMinimumWindowWidth,
145cb6afcb1SStephan Aßmus 			2 * (logoBitmap->Bounds().Width() + 8));
146cb6afcb1SStephan Aßmus 		fMinimumTrayWidth = sMinimumWindowWidth - kGutter - kDragRegionWidth;
147cb6afcb1SStephan Aßmus 	}
148573f748cSJohn Scipione 
1493cf2d117SJohn Scipione 	BFormattingConventions conventions;
1503cf2d117SJohn Scipione 	BLocale::Default()->GetFormattingConventions(&conventions);
1513cf2d117SJohn Scipione 	bool use24HourClock = conventions.Use24HourClock();
1523cf2d117SJohn Scipione 	desk_settings* settings = ((TBarApp*)be_app)->Settings();
1533cf2d117SJohn Scipione 
154573f748cSJohn Scipione 	// Create the time view
155573f748cSJohn Scipione 	fTime = new TTimeView(fMinimumTrayWidth, kMaxReplicantHeight - 1.0,
156f5be13ecSJohn Scipione 		use24HourClock, settings->showSeconds, settings->showDayOfWeek);
15741281cf3SAxel Dörfler }
15841281cf3SAxel Dörfler 
15941281cf3SAxel Dörfler 
16041281cf3SAxel Dörfler TReplicantTray::~TReplicantTray()
16141281cf3SAxel Dörfler {
16241281cf3SAxel Dörfler 	delete fShelf;
163573f748cSJohn Scipione 	delete fTime;
16441281cf3SAxel Dörfler }
16541281cf3SAxel Dörfler 
16641281cf3SAxel Dörfler 
16741281cf3SAxel Dörfler void
16841281cf3SAxel Dörfler TReplicantTray::AttachedToWindow()
16941281cf3SAxel Dörfler {
17041281cf3SAxel Dörfler 	BView::AttachedToWindow();
17141281cf3SAxel Dörfler 
172cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
173cb6afcb1SStephan Aßmus 		SetViewColor(Parent()->ViewColor());
174cb6afcb1SStephan Aßmus 	} else {
17566eba86fSAxel Dörfler 		SetViewColor(tint_color(ui_color(B_MENU_BACKGROUND_COLOR),
17666eba86fSAxel Dörfler 			B_DARKEN_1_TINT));
177cb6afcb1SStephan Aßmus 	}
17841281cf3SAxel Dörfler 	SetDrawingMode(B_OP_COPY);
17966eba86fSAxel Dörfler 
18041281cf3SAxel Dörfler 	Window()->SetPulseRate(1000000);
18141281cf3SAxel Dörfler 
182573f748cSJohn Scipione 	AddChild(fTime);
183573f748cSJohn Scipione 	fTime->MoveTo(Bounds().right - fTime->Bounds().Width() - 1, 2);
184573f748cSJohn Scipione 	if (!((TBarApp*)be_app)->Settings()->showTime) {
185573f748cSJohn Scipione 		fTime->Hide();
186573f748cSJohn Scipione 		RealignReplicants();
187573f748cSJohn Scipione 		AdjustPlacement();
188573f748cSJohn Scipione 	}
189ae7a6e3aSAdrien Destugues 
19041281cf3SAxel Dörfler #ifdef DB_ADDONS
19141281cf3SAxel Dörfler 	// load addons and rehydrate archives
1923a6add49SAxel Dörfler #if !defined(HAIKU_TARGET_PLATFORM_LIBBE_TEST)
19341281cf3SAxel Dörfler 	InitAddOnSupport();
19441281cf3SAxel Dörfler #endif
1959c910f9eSStephan Aßmus #endif
19641281cf3SAxel Dörfler 	ResizeToPreferred();
19741281cf3SAxel Dörfler }
19841281cf3SAxel Dörfler 
19941281cf3SAxel Dörfler 
20041281cf3SAxel Dörfler void
20141281cf3SAxel Dörfler TReplicantTray::DetachedFromWindow()
20241281cf3SAxel Dörfler {
20341281cf3SAxel Dörfler #ifdef DB_ADDONS
20441281cf3SAxel Dörfler 	// clean up add-on support
205465b54cdSAxel Dörfler #if !defined(HAIKU_TARGET_PLATFORM_LIBBE_TEST)
20641281cf3SAxel Dörfler 	DeleteAddOnSupport();
20741281cf3SAxel Dörfler #endif
208465b54cdSAxel Dörfler #endif
20941281cf3SAxel Dörfler 	BView::DetachedFromWindow();
21041281cf3SAxel Dörfler }
21141281cf3SAxel Dörfler 
21241281cf3SAxel Dörfler 
2131ad8c760SFredrik Holmqvist /*! Width is set to a minimum of kMinimumReplicantCount by kMaxReplicantWidth
21466eba86fSAxel Dörfler 	if not in multirowmode and greater than kMinimumReplicantCount
2152ce9bab8SJohn Scipione 	the width should be calculated based on the actual replicant widths
21641281cf3SAxel Dörfler */
21741281cf3SAxel Dörfler void
21841281cf3SAxel Dörfler TReplicantTray::GetPreferredSize(float* preferredWidth, float* preferredHeight)
21941281cf3SAxel Dörfler {
22041281cf3SAxel Dörfler 	float width = 0, height = kMinimumTrayHeight;
22141281cf3SAxel Dörfler 
22241281cf3SAxel Dörfler 	if (fMultiRowMode) {
2237da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0)
2247da06231SAxel Dörfler 			height = fRightBottomReplicant.bottom;
22541281cf3SAxel Dörfler 
2261ad8c760SFredrik Holmqvist 		// the height will be uniform for the number of rows necessary to show
2271ad8c760SFredrik Holmqvist 		// all the reps + any gutters necessary for spacing
22841281cf3SAxel Dörfler 		int32 rowCount = (int32)(height / kMaxReplicantHeight);
22941281cf3SAxel Dörfler 		height = kGutter + (rowCount * kMaxReplicantHeight)
23041281cf3SAxel Dörfler 			+ ((rowCount - 1) * kIconGap) + kGutter;
23141281cf3SAxel Dörfler 		height = max(kMinimumTrayHeight, height);
232cb6afcb1SStephan Aßmus 		width = fMinimumTrayWidth;
23341281cf3SAxel Dörfler 	} else {
2341ad8c760SFredrik Holmqvist 		// if last replicant overruns clock then resize to accomodate
2357da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0) {
236573f748cSJohn Scipione 			if (!fTime->IsHidden() && fTime->Frame().left
237573f748cSJohn Scipione 				< fRightBottomReplicant.right + 6) {
23866eba86fSAxel Dörfler 				width = fRightBottomReplicant.right + 6
239573f748cSJohn Scipione 					+ fTime->Frame().Width();
24041281cf3SAxel Dörfler 			} else
2417da06231SAxel Dörfler 				width = fRightBottomReplicant.right + 3;
24241281cf3SAxel Dörfler 		}
2431ad8c760SFredrik Holmqvist 
24441281cf3SAxel Dörfler 		// this view has a fixed minimum width
245cb6afcb1SStephan Aßmus 		width = max(fMinimumTrayWidth, width);
2462ce9bab8SJohn Scipione 		height = kGutter + static_cast<TBarApp*>(be_app)->IconSize() + kGutter;
24741281cf3SAxel Dörfler 	}
24841281cf3SAxel Dörfler 
24941281cf3SAxel Dörfler 	*preferredWidth = width;
2501ad8c760SFredrik Holmqvist 	// add 1 for the border
25141281cf3SAxel Dörfler 	*preferredHeight = height + 1;
25241281cf3SAxel Dörfler }
25341281cf3SAxel Dörfler 
25441281cf3SAxel Dörfler 
25541281cf3SAxel Dörfler void
25641281cf3SAxel Dörfler TReplicantTray::AdjustPlacement()
25741281cf3SAxel Dörfler {
25841281cf3SAxel Dörfler 	// called when an add-on has been added or removed
25941281cf3SAxel Dörfler 	// need to resize the parent of this accordingly
260a5210ab1SAxel Dörfler 
26141281cf3SAxel Dörfler 	BRect bounds = Bounds();
26241281cf3SAxel Dörfler 	float width, height;
26341281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
26441281cf3SAxel Dörfler 
26541281cf3SAxel Dörfler 	if (width == bounds.Width() && height == bounds.Height()) {
26641281cf3SAxel Dörfler 		// no need to change anything
26741281cf3SAxel Dörfler 		return;
26841281cf3SAxel Dörfler 	}
26941281cf3SAxel Dörfler 
27041281cf3SAxel Dörfler 	Parent()->ResizeToPreferred();
27141281cf3SAxel Dörfler 	fBarView->UpdatePlacement();
27241281cf3SAxel Dörfler 	Parent()->Invalidate();
27341281cf3SAxel Dörfler 	Invalidate();
27441281cf3SAxel Dörfler }
27541281cf3SAxel Dörfler 
27641281cf3SAxel Dörfler 
27741281cf3SAxel Dörfler void
27841281cf3SAxel Dörfler TReplicantTray::MessageReceived(BMessage* message)
27941281cf3SAxel Dörfler {
28041281cf3SAxel Dörfler 	switch (message->what) {
28130a55802SJohn Scipione 		case B_LOCALE_CHANGED:
282573f748cSJohn Scipione 		{
283573f748cSJohn Scipione 			if (fTime == NULL)
284573f748cSJohn Scipione 				return;
285573f748cSJohn Scipione 
28630a55802SJohn Scipione 			// Locale may have updated 12/24 hour clock
287573f748cSJohn Scipione 			BFormattingConventions conventions;
288573f748cSJohn Scipione 			BLocale::Default()->GetFormattingConventions(&conventions);
28930a55802SJohn Scipione 			fTime->SetUse24HourClock(conventions.Use24HourClock());
29030a55802SJohn Scipione 
29141281cf3SAxel Dörfler 			// time string reformat -> realign
29241281cf3SAxel Dörfler 			RealignReplicants();
29341281cf3SAxel Dörfler 			AdjustPlacement();
29441281cf3SAxel Dörfler 			break;
295573f748cSJohn Scipione 		}
296573f748cSJohn Scipione 
29730a55802SJohn Scipione 		case kShowHideTime:
29830a55802SJohn Scipione 			// from context menu in clock and in this view
29930a55802SJohn Scipione 			ShowHideTime();
30030a55802SJohn Scipione 			break;
30130a55802SJohn Scipione 
3023cf2d117SJohn Scipione 		case kShowSeconds:
303573f748cSJohn Scipione 			if (fTime == NULL)
304573f748cSJohn Scipione 				return;
305573f748cSJohn Scipione 
3063cf2d117SJohn Scipione 			fTime->SetShowSeconds(!fTime->ShowSeconds());
307573f748cSJohn Scipione 
3083cf2d117SJohn Scipione 			// time string reformat -> realign
3093cf2d117SJohn Scipione 			RealignReplicants();
3103cf2d117SJohn Scipione 			AdjustPlacement();
311573f748cSJohn Scipione 			break;
3123cf2d117SJohn Scipione 
3133cf2d117SJohn Scipione 		case kShowDayOfWeek:
3143cf2d117SJohn Scipione 			if (fTime == NULL)
3153cf2d117SJohn Scipione 				return;
3163cf2d117SJohn Scipione 
3173cf2d117SJohn Scipione 			fTime->SetShowDayOfWeek(!fTime->ShowDayOfWeek());
3183cf2d117SJohn Scipione 
3193cf2d117SJohn Scipione 			// time string reformat -> realign
3203cf2d117SJohn Scipione 			RealignReplicants();
3213cf2d117SJohn Scipione 			AdjustPlacement();
3223cf2d117SJohn Scipione 			break;
3233cf2d117SJohn Scipione 
32441281cf3SAxel Dörfler #ifdef DB_ADDONS
32541281cf3SAxel Dörfler 		case B_NODE_MONITOR:
32641281cf3SAxel Dörfler 			HandleEntryUpdate(message);
32741281cf3SAxel Dörfler 			break;
32841281cf3SAxel Dörfler #endif
32941281cf3SAxel Dörfler 
33041281cf3SAxel Dörfler 		default:
33141281cf3SAxel Dörfler 			BView::MessageReceived(message);
33241281cf3SAxel Dörfler 			break;
33341281cf3SAxel Dörfler 	}
33441281cf3SAxel Dörfler }
33541281cf3SAxel Dörfler 
33641281cf3SAxel Dörfler 
33741281cf3SAxel Dörfler void
33841281cf3SAxel Dörfler TReplicantTray::MouseDown(BPoint where)
33941281cf3SAxel Dörfler {
34041281cf3SAxel Dörfler #ifdef DB_ADDONS
34141281cf3SAxel Dörfler 	if (modifiers() & B_CONTROL_KEY)
34241281cf3SAxel Dörfler 		DumpList(fItemList);
34341281cf3SAxel Dörfler #endif
34441281cf3SAxel Dörfler 
34541281cf3SAxel Dörfler 	uint32 buttons;
34641281cf3SAxel Dörfler 
34741281cf3SAxel Dörfler 	Window()->CurrentMessage()->FindInt32("buttons", (int32*)&buttons);
34841281cf3SAxel Dörfler 	if (buttons == B_SECONDARY_MOUSE_BUTTON) {
34941281cf3SAxel Dörfler 		ShowReplicantMenu(where);
35041281cf3SAxel Dörfler 	} else {
35141281cf3SAxel Dörfler 		BPoint save = where;
35241281cf3SAxel Dörfler 		bigtime_t doubleClickSpeed;
35341281cf3SAxel Dörfler 		bigtime_t start = system_time();
35441281cf3SAxel Dörfler 		uint32 buttons;
35541281cf3SAxel Dörfler 
35641281cf3SAxel Dörfler 		get_click_speed(&doubleClickSpeed);
35741281cf3SAxel Dörfler 
35841281cf3SAxel Dörfler 		do {
35941281cf3SAxel Dörfler 			if (fabs(where.x - save.x) > 4 || fabs(where.y - save.y) > 4)
36041281cf3SAxel Dörfler 				// user moved out of bounds of click area
36141281cf3SAxel Dörfler 				break;
36241281cf3SAxel Dörfler 
36341281cf3SAxel Dörfler 			if ((system_time() - start) > (2 * doubleClickSpeed)) {
36441281cf3SAxel Dörfler 				ShowReplicantMenu(where);
36541281cf3SAxel Dörfler 				break;
36641281cf3SAxel Dörfler 			}
36741281cf3SAxel Dörfler 
36841281cf3SAxel Dörfler 			snooze(50000);
36941281cf3SAxel Dörfler 			GetMouse(&where, &buttons);
37041281cf3SAxel Dörfler 		} while (buttons);
37141281cf3SAxel Dörfler 	}
37241281cf3SAxel Dörfler 	BView::MouseDown(where);
37341281cf3SAxel Dörfler }
37441281cf3SAxel Dörfler 
3753cf2d117SJohn Scipione 
3763cf2d117SJohn Scipione void
3773cf2d117SJohn Scipione TReplicantTray::ShowReplicantMenu(BPoint point)
3783cf2d117SJohn Scipione {
3793cf2d117SJohn Scipione 	BPopUpMenu* menu = new BPopUpMenu("", false, false);
3803cf2d117SJohn Scipione 	menu->SetFont(be_plain_font);
3813cf2d117SJohn Scipione 
3823cf2d117SJohn Scipione 	// If clock is visible show the extended menu, otherwise show "Show time"
3833cf2d117SJohn Scipione 
3843cf2d117SJohn Scipione 	if (!fTime->IsHidden())
3853cf2d117SJohn Scipione 		fTime->ShowTimeOptions(ConvertToScreen(point));
3863cf2d117SJohn Scipione 	else {
3873cf2d117SJohn Scipione 		BMenuItem* item = new BMenuItem(B_TRANSLATE("Show time"),
3883cf2d117SJohn Scipione 			new BMessage(kShowHideTime));
3893cf2d117SJohn Scipione 		menu->AddItem(item);
3903cf2d117SJohn Scipione 		menu->SetTargetForItems(this);
3913cf2d117SJohn Scipione 		BPoint where = ConvertToScreen(point);
3923cf2d117SJohn Scipione 		menu->Go(where, true, true, BRect(where - BPoint(4, 4),
3933cf2d117SJohn Scipione 			where + BPoint(4, 4)), true);
3943cf2d117SJohn Scipione 	}
3953cf2d117SJohn Scipione }
3963cf2d117SJohn Scipione 
3973cf2d117SJohn Scipione 
3983cf2d117SJohn Scipione void
3993cf2d117SJohn Scipione TReplicantTray::SetMultiRow(bool state)
4003cf2d117SJohn Scipione {
4013cf2d117SJohn Scipione 	fMultiRowMode = state;
4023cf2d117SJohn Scipione }
4033cf2d117SJohn Scipione 
4043cf2d117SJohn Scipione 
4053cf2d117SJohn Scipione void
4063cf2d117SJohn Scipione TReplicantTray::ShowHideTime()
4073cf2d117SJohn Scipione {
4083cf2d117SJohn Scipione 	if (fTime == NULL)
4093cf2d117SJohn Scipione 		return;
4103cf2d117SJohn Scipione 
4113cf2d117SJohn Scipione 	if (fTime->IsHidden())
4123cf2d117SJohn Scipione 		fTime->Show();
4133cf2d117SJohn Scipione 	else
4143cf2d117SJohn Scipione 		fTime->Hide();
4153cf2d117SJohn Scipione 
4163cf2d117SJohn Scipione 	RealignReplicants();
4173cf2d117SJohn Scipione 	AdjustPlacement();
4183cf2d117SJohn Scipione }
4193cf2d117SJohn Scipione 
4203cf2d117SJohn Scipione 
42141281cf3SAxel Dörfler #ifdef DB_ADDONS
42241281cf3SAxel Dörfler 
4233cf2d117SJohn Scipione 
42441281cf3SAxel Dörfler void
42541281cf3SAxel Dörfler TReplicantTray::InitAddOnSupport()
42641281cf3SAxel Dörfler {
42741281cf3SAxel Dörfler 	// list to maintain refs to each rep added/deleted
42841281cf3SAxel Dörfler 	fItemList = new BList();
42941281cf3SAxel Dörfler 	BPath path;
4301ad8c760SFredrik Holmqvist 
43141281cf3SAxel Dörfler 	if (find_directory(B_USER_SETTINGS_DIRECTORY, &path, true) == B_OK) {
4323e6ff860SRene Gollent 		path.Append(kReplicantSettingsFile);
43341281cf3SAxel Dörfler 
43441281cf3SAxel Dörfler 		BFile file(path.Path(), B_READ_ONLY);
4353e6ff860SRene Gollent 		if (file.InitCheck() == B_OK) {
4363e6ff860SRene Gollent 			status_t result;
4373e6ff860SRene Gollent 			BEntry entry;
4383e6ff860SRene Gollent 			int32 id;
439c0ad1c91SRene Gollent 			BString path;
4403e6ff860SRene Gollent 			if (fAddOnSettings.Unflatten(&file) == B_OK) {
441c0ad1c91SRene Gollent 				for (int32 i = 0; fAddOnSettings.FindString(kReplicantPathField,
442c0ad1c91SRene Gollent 					i, &path) == B_OK; i++) {
443c0ad1c91SRene Gollent 					if (entry.SetTo(path.String()) == B_OK && entry.Exists()) {
4443e6ff860SRene Gollent 						result = LoadAddOn(&entry, &id, false);
4453e6ff860SRene Gollent 					} else
4463e6ff860SRene Gollent 						result = B_ENTRY_NOT_FOUND;
44741281cf3SAxel Dörfler 
4483e6ff860SRene Gollent 					if (result != B_OK) {
449c0ad1c91SRene Gollent 						fAddOnSettings.RemoveData(kReplicantPathField, i);
4503e6ff860SRene Gollent 						--i;
45141281cf3SAxel Dörfler 					}
45241281cf3SAxel Dörfler 				}
45341281cf3SAxel Dörfler 			}
4543e6ff860SRene Gollent 		}
4553e6ff860SRene Gollent 	}
45641281cf3SAxel Dörfler }
45741281cf3SAxel Dörfler 
45841281cf3SAxel Dörfler 
45941281cf3SAxel Dörfler void
46041281cf3SAxel Dörfler TReplicantTray::DeleteAddOnSupport()
46141281cf3SAxel Dörfler {
4627625ce51SRene Gollent 	_SaveSettings();
4633e6ff860SRene Gollent 
46441281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
46541281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->RemoveItem(i);
46641281cf3SAxel Dörfler 		if (item) {
46741281cf3SAxel Dörfler 			if (item->isAddOn)
46841281cf3SAxel Dörfler 				watch_node(&(item->nodeRef), B_STOP_WATCHING, this, Window());
46941281cf3SAxel Dörfler 
47041281cf3SAxel Dörfler 			delete item;
47141281cf3SAxel Dörfler 		}
47241281cf3SAxel Dörfler 	}
47341281cf3SAxel Dörfler 	delete fItemList;
47441281cf3SAxel Dörfler 
47541281cf3SAxel Dörfler 	// stop the volume mount/unmount watch
47641281cf3SAxel Dörfler 	stop_watching(this, Window());
47741281cf3SAxel Dörfler }
47841281cf3SAxel Dörfler 
47941281cf3SAxel Dörfler 
48041281cf3SAxel Dörfler DeskbarItemInfo*
48141281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(node_ref& nodeRef)
48241281cf3SAxel Dörfler {
48341281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
48466eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
48541281cf3SAxel Dörfler 		if (item == NULL)
48641281cf3SAxel Dörfler 			continue;
48741281cf3SAxel Dörfler 
48841281cf3SAxel Dörfler 		if (item->nodeRef == nodeRef)
48941281cf3SAxel Dörfler 			return item;
49041281cf3SAxel Dörfler 	}
49141281cf3SAxel Dörfler 
49241281cf3SAxel Dörfler 	return NULL;
49341281cf3SAxel Dörfler }
49441281cf3SAxel Dörfler 
49541281cf3SAxel Dörfler 
49641281cf3SAxel Dörfler DeskbarItemInfo*
49741281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(int32 id)
49841281cf3SAxel Dörfler {
49941281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
50066eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
50141281cf3SAxel Dörfler 		if (item == NULL)
50241281cf3SAxel Dörfler 			continue;
50341281cf3SAxel Dörfler 
50441281cf3SAxel Dörfler 		if (item->id == id)
50541281cf3SAxel Dörfler 			return item;
50641281cf3SAxel Dörfler 	}
50741281cf3SAxel Dörfler 
50841281cf3SAxel Dörfler 	return NULL;
50941281cf3SAxel Dörfler }
51041281cf3SAxel Dörfler 
51141281cf3SAxel Dörfler 
51241281cf3SAxel Dörfler bool
51341281cf3SAxel Dörfler TReplicantTray::NodeExists(node_ref& nodeRef)
51441281cf3SAxel Dörfler {
51541281cf3SAxel Dörfler 	return DeskbarItemFor(nodeRef) != NULL;
51641281cf3SAxel Dörfler }
51741281cf3SAxel Dörfler 
51841281cf3SAxel Dörfler 
51966eba86fSAxel Dörfler /*! This handles B_NODE_MONITOR & B_QUERY_UPDATE messages received
52066eba86fSAxel Dörfler 	for the registered add-ons.
52141281cf3SAxel Dörfler */
52241281cf3SAxel Dörfler void
52341281cf3SAxel Dörfler TReplicantTray::HandleEntryUpdate(BMessage* message)
52441281cf3SAxel Dörfler {
52541281cf3SAxel Dörfler 	int32 opcode;
52641281cf3SAxel Dörfler 	if (message->FindInt32("opcode", &opcode) != B_OK)
52741281cf3SAxel Dörfler 		return;
52841281cf3SAxel Dörfler 
52941281cf3SAxel Dörfler 	BPath path;
53041281cf3SAxel Dörfler 	switch (opcode) {
53141281cf3SAxel Dörfler 		case B_ENTRY_MOVED:
53241281cf3SAxel Dörfler 		{
53341281cf3SAxel Dörfler 			entry_ref ref;
53441281cf3SAxel Dörfler 			ino_t todirectory;
53541281cf3SAxel Dörfler 			ino_t node;
53641281cf3SAxel Dörfler 			const char* name;
53741281cf3SAxel Dörfler 			if (message->FindString("name", &name) == B_OK
53871bd3ba5SJonas Sundström 				&& message->FindInt64("from directory", &(ref.directory))
53971bd3ba5SJonas Sundström 				== B_OK
54041281cf3SAxel Dörfler 				&& message->FindInt64("to directory", &todirectory) == B_OK
54141281cf3SAxel Dörfler 				&& message->FindInt32("device", &(ref.device)) == B_OK
54241281cf3SAxel Dörfler 				&& message->FindInt64("node", &node) == B_OK ) {
54341281cf3SAxel Dörfler 
54441281cf3SAxel Dörfler 				if (!name)
54541281cf3SAxel Dörfler 					break;
54641281cf3SAxel Dörfler 
54741281cf3SAxel Dörfler 				ref.set_name(name);
54841281cf3SAxel Dörfler 				// change the directory reference to
54941281cf3SAxel Dörfler 				// the new directory
55041281cf3SAxel Dörfler 				MoveItem(&ref, todirectory);
55141281cf3SAxel Dörfler 			}
55241281cf3SAxel Dörfler 			break;
55366eba86fSAxel Dörfler 		}
55441281cf3SAxel Dörfler 
55541281cf3SAxel Dörfler 		case B_ENTRY_REMOVED:
55641281cf3SAxel Dörfler 		{
55741281cf3SAxel Dörfler 			// entry was rm'd from the device
55841281cf3SAxel Dörfler 			node_ref nodeRef;
55941281cf3SAxel Dörfler 			if (message->FindInt32("device", &(nodeRef.device)) == B_OK
56041281cf3SAxel Dörfler 				&& message->FindInt64("node", &(nodeRef.node)) == B_OK) {
56141281cf3SAxel Dörfler 				DeskbarItemInfo* item = DeskbarItemFor(nodeRef);
56241281cf3SAxel Dörfler 				if (item == NULL)
56341281cf3SAxel Dörfler 					break;
56441281cf3SAxel Dörfler 
56541281cf3SAxel Dörfler 				// If there is a team running where the add-on comes from,
56641281cf3SAxel Dörfler 				// we don't want to remove the icon yet.
56741281cf3SAxel Dörfler 				if (be_roster->IsRunning(&item->entryRef))
56841281cf3SAxel Dörfler 					break;
56941281cf3SAxel Dörfler 
57041281cf3SAxel Dörfler 				UnloadAddOn(&nodeRef, NULL, true, false);
57141281cf3SAxel Dörfler 			}
57241281cf3SAxel Dörfler 			break;
57366eba86fSAxel Dörfler 		}
57441281cf3SAxel Dörfler 	}
57566eba86fSAxel Dörfler }
57641281cf3SAxel Dörfler 
57741281cf3SAxel Dörfler 
5781ad8c760SFredrik Holmqvist /*! The add-ons must support the exported C function API
57966eba86fSAxel Dörfler 	if they do, they will be loaded and added to deskbar
58066eba86fSAxel Dörfler 	primary function is the Instantiate function
5817da06231SAxel Dörfler */
58241281cf3SAxel Dörfler status_t
5833e6ff860SRene Gollent TReplicantTray::LoadAddOn(BEntry* entry, int32* id, bool addToSettings)
58441281cf3SAxel Dörfler {
58541281cf3SAxel Dörfler 	if (!entry)
58641281cf3SAxel Dörfler 		return B_ERROR;
58741281cf3SAxel Dörfler 
58841281cf3SAxel Dörfler 	node_ref nodeRef;
58941281cf3SAxel Dörfler 	entry->GetNodeRef(&nodeRef);
59041281cf3SAxel Dörfler 	// no duplicates
59141281cf3SAxel Dörfler 	if (NodeExists(nodeRef))
59241281cf3SAxel Dörfler 		return B_ERROR;
59341281cf3SAxel Dörfler 
59441281cf3SAxel Dörfler 	BNode node(entry);
59541281cf3SAxel Dörfler 	BPath path;
59666eba86fSAxel Dörfler 	status_t status = entry->GetPath(&path);
59766eba86fSAxel Dörfler 	if (status < B_OK)
59866eba86fSAxel Dörfler 		return status;
59941281cf3SAxel Dörfler 
60041281cf3SAxel Dörfler 	// load the add-on
60141281cf3SAxel Dörfler 	image_id image = load_add_on(path.Path());
60266eba86fSAxel Dörfler 	if (image < B_OK)
60366eba86fSAxel Dörfler 		return image;
60441281cf3SAxel Dörfler 
60541281cf3SAxel Dörfler 	// get the view loading function symbol
60641281cf3SAxel Dörfler 	//    we first look for a symbol that takes an image_id
60741281cf3SAxel Dörfler 	//    and entry_ref pointer, if not found, go with normal
60841281cf3SAxel Dörfler 	//    instantiate function
60941281cf3SAxel Dörfler 	BView* (*entryFunction)(image_id, const entry_ref*);
61041281cf3SAxel Dörfler 	BView* (*itemFunction)(void);
61141281cf3SAxel Dörfler 	BView* view = NULL;
61241281cf3SAxel Dörfler 
61341281cf3SAxel Dörfler 	entry_ref ref;
61441281cf3SAxel Dörfler 	entry->GetRef(&ref);
61541281cf3SAxel Dörfler 
61641281cf3SAxel Dörfler 	if (get_image_symbol(image, kInstantiateEntryCFunctionName,
61766eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&entryFunction) >= B_OK) {
61841281cf3SAxel Dörfler 		view = (*entryFunction)(image, &ref);
61941281cf3SAxel Dörfler 	} else if (get_image_symbol(image, kInstantiateItemCFunctionName,
62066eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&itemFunction) >= B_OK) {
62141281cf3SAxel Dörfler 		view = (*itemFunction)();
62241281cf3SAxel Dörfler 	} else {
62341281cf3SAxel Dörfler 		unload_add_on(image);
62441281cf3SAxel Dörfler 		return B_ERROR;
62541281cf3SAxel Dörfler 	}
62641281cf3SAxel Dörfler 
62766eba86fSAxel Dörfler 	if (view == NULL || IconExists(view->Name())) {
62841281cf3SAxel Dörfler 		delete view;
62941281cf3SAxel Dörfler 		unload_add_on(image);
63041281cf3SAxel Dörfler 		return B_ERROR;
63141281cf3SAxel Dörfler 	}
63241281cf3SAxel Dörfler 
63341281cf3SAxel Dörfler 	BMessage* data = new BMessage;
63441281cf3SAxel Dörfler 	view->Archive(data);
63541281cf3SAxel Dörfler 	delete view;
63641281cf3SAxel Dörfler 
63741281cf3SAxel Dörfler 	AddIcon(data, id, &ref);
63841281cf3SAxel Dörfler 		// add the rep; adds info to list
63941281cf3SAxel Dörfler 
6403e6ff860SRene Gollent 	if (addToSettings) {
641c0ad1c91SRene Gollent 		fAddOnSettings.AddString(kReplicantPathField, path.Path());
6427625ce51SRene Gollent 		_SaveSettings();
6433e6ff860SRene Gollent 	}
64441281cf3SAxel Dörfler 
64541281cf3SAxel Dörfler 	return B_OK;
64641281cf3SAxel Dörfler }
64741281cf3SAxel Dörfler 
64841281cf3SAxel Dörfler 
64941281cf3SAxel Dörfler status_t
65041281cf3SAxel Dörfler TReplicantTray::AddItem(int32 id, node_ref nodeRef, BEntry& entry, bool isAddOn)
65141281cf3SAxel Dörfler {
65241281cf3SAxel Dörfler 	DeskbarItemInfo* item = new DeskbarItemInfo;
65341281cf3SAxel Dörfler 	if (item == NULL)
65441281cf3SAxel Dörfler 		return B_NO_MEMORY;
65541281cf3SAxel Dörfler 
65641281cf3SAxel Dörfler 	item->id = id;
65741281cf3SAxel Dörfler 	item->isAddOn = isAddOn;
65841281cf3SAxel Dörfler 
65941281cf3SAxel Dörfler 	if (entry.GetRef(&item->entryRef) < B_OK) {
66041281cf3SAxel Dörfler 		item->entryRef.device = -1;
66141281cf3SAxel Dörfler 		item->entryRef.directory = -1;
66241281cf3SAxel Dörfler 		item->entryRef.name = NULL;
66341281cf3SAxel Dörfler 	}
66441281cf3SAxel Dörfler 	item->nodeRef = nodeRef;
66541281cf3SAxel Dörfler 
66641281cf3SAxel Dörfler 	fItemList->AddItem(item);
66741281cf3SAxel Dörfler 
66841281cf3SAxel Dörfler 	if (isAddOn)
66941281cf3SAxel Dörfler 		watch_node(&nodeRef, B_WATCH_NAME | B_WATCH_ATTR, this, Window());
67041281cf3SAxel Dörfler 
67141281cf3SAxel Dörfler 	return B_OK;
67241281cf3SAxel Dörfler }
67341281cf3SAxel Dörfler 
67441281cf3SAxel Dörfler 
67541281cf3SAxel Dörfler /**	from entry_removed message, when attribute removed
67641281cf3SAxel Dörfler  *	or when a device is unmounted (use removeall, by device)
67741281cf3SAxel Dörfler  */
67841281cf3SAxel Dörfler 
67941281cf3SAxel Dörfler void
68041281cf3SAxel Dörfler TReplicantTray::UnloadAddOn(node_ref* nodeRef, dev_t* device,
68141281cf3SAxel Dörfler 	bool which, bool removeAll)
68241281cf3SAxel Dörfler {
68341281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
68441281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
68541281cf3SAxel Dörfler 		if (!item)
68641281cf3SAxel Dörfler 			continue;
68741281cf3SAxel Dörfler 
68841281cf3SAxel Dörfler 		if ((which && nodeRef && item->nodeRef == *nodeRef)
68941281cf3SAxel Dörfler 			|| (device && item->nodeRef.device == *device)) {
69041281cf3SAxel Dörfler 
69141281cf3SAxel Dörfler 			if (device && be_roster->IsRunning(&item->entryRef))
69241281cf3SAxel Dörfler 				continue;
69341281cf3SAxel Dörfler 
69441281cf3SAxel Dörfler 			RemoveIcon(item->id);
69541281cf3SAxel Dörfler 
69641281cf3SAxel Dörfler 			if (!removeAll)
69741281cf3SAxel Dörfler 				break;
69841281cf3SAxel Dörfler 		}
69941281cf3SAxel Dörfler 	}
70041281cf3SAxel Dörfler }
70141281cf3SAxel Dörfler 
70241281cf3SAxel Dörfler 
70341281cf3SAxel Dörfler void
70441281cf3SAxel Dörfler TReplicantTray::RemoveItem(int32 id)
70541281cf3SAxel Dörfler {
70641281cf3SAxel Dörfler 	DeskbarItemInfo* item = DeskbarItemFor(id);
70741281cf3SAxel Dörfler 	if (item == NULL)
70841281cf3SAxel Dörfler 		return;
70941281cf3SAxel Dörfler 
71041281cf3SAxel Dörfler 	// attribute was added via Deskbar API (AddItem(entry_ref*, int32*)
71141281cf3SAxel Dörfler 	if (item->isAddOn) {
7127625ce51SRene Gollent 		BPath path(&item->entryRef);
7137625ce51SRene Gollent 		BString storedPath;
7147625ce51SRene Gollent 		for (int32 i = 0;
7158aab28f1SRene Gollent 			fAddOnSettings.FindString(kReplicantPathField, i, &storedPath)
7167625ce51SRene Gollent 				== B_OK; i++) {
7177625ce51SRene Gollent 			if (storedPath == path.Path()) {
7188aab28f1SRene Gollent 				fAddOnSettings.RemoveData(kReplicantPathField, i);
7197625ce51SRene Gollent 				break;
7207625ce51SRene Gollent 			}
7217625ce51SRene Gollent 		}
7227625ce51SRene Gollent 		_SaveSettings();
7237625ce51SRene Gollent 
72441281cf3SAxel Dörfler 		BNode node(&item->entryRef);
72541281cf3SAxel Dörfler 		watch_node(&item->nodeRef, B_STOP_WATCHING, this, Window());
72641281cf3SAxel Dörfler 	}
72741281cf3SAxel Dörfler 
72841281cf3SAxel Dörfler 	fItemList->RemoveItem(item);
72941281cf3SAxel Dörfler 	delete item;
73041281cf3SAxel Dörfler }
73141281cf3SAxel Dörfler 
73241281cf3SAxel Dörfler 
73341281cf3SAxel Dörfler /**	ENTRY_MOVED message, moving only occurs on a device
73441281cf3SAxel Dörfler  *	copying will occur (ENTRY_CREATED) between devices
73541281cf3SAxel Dörfler  */
73641281cf3SAxel Dörfler 
73741281cf3SAxel Dörfler void
73841281cf3SAxel Dörfler TReplicantTray::MoveItem(entry_ref* ref, ino_t toDirectory)
73941281cf3SAxel Dörfler {
74041281cf3SAxel Dörfler 	if (!ref)
74141281cf3SAxel Dörfler 		return;
74241281cf3SAxel Dörfler 
74341281cf3SAxel Dörfler 	// scan for a matching entry_ref and update it
74441281cf3SAxel Dörfler 	//
74541281cf3SAxel Dörfler 	// don't need to change node info as it does not change
74641281cf3SAxel Dörfler 
74741281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
74841281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
74941281cf3SAxel Dörfler 		if (!item)
75041281cf3SAxel Dörfler 			continue;
75141281cf3SAxel Dörfler 
75241281cf3SAxel Dörfler 		if (!strcmp(item->entryRef.name, ref->name)
75341281cf3SAxel Dörfler 			&& item->entryRef.device == ref->device
75441281cf3SAxel Dörfler 			&& item->entryRef.directory == ref->directory) {
75541281cf3SAxel Dörfler 			item->entryRef.directory = toDirectory;
75641281cf3SAxel Dörfler 			break;
75741281cf3SAxel Dörfler 		}
75841281cf3SAxel Dörfler 	}
75941281cf3SAxel Dörfler }
76041281cf3SAxel Dörfler 
76141281cf3SAxel Dörfler #endif // add-on support
76241281cf3SAxel Dörfler 
76341281cf3SAxel Dörfler //	external add-on API routines
76441281cf3SAxel Dörfler //	called using the new BDeskbar class
76541281cf3SAxel Dörfler 
76641281cf3SAxel Dörfler //	existence of icon/replicant by name or ID
76741281cf3SAxel Dörfler //	returns opposite
76841281cf3SAxel Dörfler //	note: name and id are semi-private limiting
76941281cf3SAxel Dörfler //		the ability of non-host apps to remove
77041281cf3SAxel Dörfler //		icons without a little bit of work
77141281cf3SAxel Dörfler 
77241281cf3SAxel Dörfler /**	for a specific id
77341281cf3SAxel Dörfler  *	return the name of the replicant (name of view)
77441281cf3SAxel Dörfler  */
77541281cf3SAxel Dörfler 
77641281cf3SAxel Dörfler status_t
77741281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 id, const char** name)
77841281cf3SAxel Dörfler {
77941281cf3SAxel Dörfler 	if (id < 0)
78041281cf3SAxel Dörfler 		return B_ERROR;
78141281cf3SAxel Dörfler 
78241281cf3SAxel Dörfler 	int32 index, temp;
78341281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &temp, id, false);
78441281cf3SAxel Dörfler 	if (view) {
78541281cf3SAxel Dörfler 		*name = view->Name();
78641281cf3SAxel Dörfler 		return B_OK;
78741281cf3SAxel Dörfler 	}
78841281cf3SAxel Dörfler 
78941281cf3SAxel Dörfler 	return B_ERROR;
79041281cf3SAxel Dörfler }
79141281cf3SAxel Dörfler 
79241281cf3SAxel Dörfler 
7937da06231SAxel Dörfler /**	for a specific name
7947da06231SAxel Dörfler  *	return the id (internal to Deskbar)
7957da06231SAxel Dörfler  */
7967da06231SAxel Dörfler 
79741281cf3SAxel Dörfler status_t
79841281cf3SAxel Dörfler TReplicantTray::ItemInfo(const char* name, int32* id)
79941281cf3SAxel Dörfler {
80041281cf3SAxel Dörfler 	if (!name || strlen(name) <= 0)
80141281cf3SAxel Dörfler 		return B_ERROR;
80241281cf3SAxel Dörfler 
80341281cf3SAxel Dörfler 	int32 index;
80441281cf3SAxel Dörfler 	BView* view = ViewAt(&index, id, name);
80541281cf3SAxel Dörfler 	if (view)
80641281cf3SAxel Dörfler 		return B_OK;
80741281cf3SAxel Dörfler 
80841281cf3SAxel Dörfler 	return B_ERROR;
80941281cf3SAxel Dörfler }
81041281cf3SAxel Dörfler 
81141281cf3SAxel Dörfler 
8127da06231SAxel Dörfler /**	at a specific index
8137da06231SAxel Dörfler  *	return both the name and the id of the replicant
8147da06231SAxel Dörfler  */
8157da06231SAxel Dörfler 
81641281cf3SAxel Dörfler status_t
81741281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 index, const char** name, int32* id)
81841281cf3SAxel Dörfler {
81941281cf3SAxel Dörfler 	if (index < 0)
82041281cf3SAxel Dörfler 		return B_ERROR;
82141281cf3SAxel Dörfler 
82241281cf3SAxel Dörfler 	BView* view;
82341281cf3SAxel Dörfler 	fShelf->ReplicantAt(index, &view, (uint32*)id, NULL);
82441281cf3SAxel Dörfler 	if (view) {
82541281cf3SAxel Dörfler 		*name = view->Name();
82641281cf3SAxel Dörfler 		return B_OK;
82741281cf3SAxel Dörfler 	}
82841281cf3SAxel Dörfler 
82941281cf3SAxel Dörfler 	return B_ERROR;
83041281cf3SAxel Dörfler }
83141281cf3SAxel Dörfler 
83241281cf3SAxel Dörfler 
8337da06231SAxel Dörfler /**	replicant exists, by id/index */
8347da06231SAxel Dörfler 
83541281cf3SAxel Dörfler bool
83641281cf3SAxel Dörfler TReplicantTray::IconExists(int32 target, bool byIndex)
83741281cf3SAxel Dörfler {
83841281cf3SAxel Dörfler 	int32 index, id;
83941281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
84041281cf3SAxel Dörfler 
84141281cf3SAxel Dörfler 	return view && index >= 0;
84241281cf3SAxel Dörfler }
84341281cf3SAxel Dörfler 
84441281cf3SAxel Dörfler 
8457da06231SAxel Dörfler /**	replicant exists, by name */
8467da06231SAxel Dörfler 
84741281cf3SAxel Dörfler bool
84841281cf3SAxel Dörfler TReplicantTray::IconExists(const char* name)
84941281cf3SAxel Dörfler {
85041281cf3SAxel Dörfler 	if (!name || strlen(name) == 0)
85141281cf3SAxel Dörfler 		return false;
85241281cf3SAxel Dörfler 
85341281cf3SAxel Dörfler 	int32 index, id;
85441281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
85541281cf3SAxel Dörfler 
85641281cf3SAxel Dörfler 	return view && index >= 0;
85741281cf3SAxel Dörfler }
85841281cf3SAxel Dörfler 
85941281cf3SAxel Dörfler 
86041281cf3SAxel Dörfler int32
86141281cf3SAxel Dörfler TReplicantTray::IconCount() const
86241281cf3SAxel Dörfler {
86341281cf3SAxel Dörfler 	return fShelf->CountReplicants();
86441281cf3SAxel Dörfler }
86541281cf3SAxel Dörfler 
86641281cf3SAxel Dörfler 
867ced3755cSAxel Dörfler /*! Message must contain an archivable view for later rehydration.
868ced3755cSAxel Dörfler 	This function takes over ownership of the provided message on success
869ced3755cSAxel Dörfler 	only.
870ced3755cSAxel Dörfler 	Returns the current replicant ID.
8717da06231SAxel Dörfler */
87241281cf3SAxel Dörfler status_t
873ced3755cSAxel Dörfler TReplicantTray::AddIcon(BMessage* archive, int32* id, const entry_ref* addOn)
87441281cf3SAxel Dörfler {
875ced3755cSAxel Dörfler 	if (archive == NULL || id == NULL)
87641281cf3SAxel Dörfler 		return B_ERROR;
87741281cf3SAxel Dörfler 
8789c50f36eSAxel Dörfler 	// find entry_ref
8799c50f36eSAxel Dörfler 
8809c50f36eSAxel Dörfler 	entry_ref ref;
8819c50f36eSAxel Dörfler 	if (addOn) {
8829c50f36eSAxel Dörfler 		// Use it if we got it
8839c50f36eSAxel Dörfler 		ref = *addOn;
8849c50f36eSAxel Dörfler 	} else {
8859c50f36eSAxel Dörfler 		const char* signature;
8861ad8c760SFredrik Holmqvist 
887ced3755cSAxel Dörfler 		status_t status = archive->FindString("add_on", &signature);
8889c50f36eSAxel Dörfler 		if (status == B_OK) {
8899c50f36eSAxel Dörfler 			BRoster roster;
8909c50f36eSAxel Dörfler 			status = roster.FindApp(signature, &ref);
8919c50f36eSAxel Dörfler 		}
8929c50f36eSAxel Dörfler 		if (status < B_OK)
8939c50f36eSAxel Dörfler 			return status;
8949c50f36eSAxel Dörfler 	}
8959c50f36eSAxel Dörfler 
8969c50f36eSAxel Dörfler 	BFile file;
8979c50f36eSAxel Dörfler 	status_t status = file.SetTo(&ref, B_READ_ONLY);
8989c50f36eSAxel Dörfler 	if (status < B_OK)
8999c50f36eSAxel Dörfler 		return status;
9009c50f36eSAxel Dörfler 
9019c50f36eSAxel Dörfler 	node_ref nodeRef;
9029c50f36eSAxel Dörfler 	status = file.GetNodeRef(&nodeRef);
9039c50f36eSAxel Dörfler 	if (status < B_OK)
9049c50f36eSAxel Dörfler 		return status;
9059c50f36eSAxel Dörfler 
9069c50f36eSAxel Dörfler 	BEntry entry(&ref, true);
9071ad8c760SFredrik Holmqvist 		// TODO: this resolves an eventual link for the item being added - this
9081ad8c760SFredrik Holmqvist 		// is okay for now, but in multi-user environments, one might want to
9091ad8c760SFredrik Holmqvist 		// have links that carry the be:deskbar_item_status attribute
9109c50f36eSAxel Dörfler 	status = entry.InitCheck();
9119c50f36eSAxel Dörfler 	if (status != B_OK)
9129c50f36eSAxel Dörfler 		return status;
9139c50f36eSAxel Dörfler 
91441281cf3SAxel Dörfler 	*id = 999;
915ced3755cSAxel Dörfler 	if (archive->what == B_ARCHIVED_OBJECT)
916ced3755cSAxel Dörfler 		archive->what = 0;
91741281cf3SAxel Dörfler 
918ced3755cSAxel Dörfler 	BRect originalBounds = archive->FindRect("_frame");
9191ad8c760SFredrik Holmqvist 		// this is a work-around for buggy replicants that change their size in
9201ad8c760SFredrik Holmqvist 		// AttachedToWindow() (such as "SVM")
921a5210ab1SAxel Dörfler 
9221ad8c760SFredrik Holmqvist 	// TODO: check for name collisions?
923ced3755cSAxel Dörfler 	status = fShelf->AddReplicant(archive, BPoint(1, 1));
9249c50f36eSAxel Dörfler 	if (status != B_OK)
9259c50f36eSAxel Dörfler 		return status;
92641281cf3SAxel Dörfler 
927a5210ab1SAxel Dörfler 	int32 count = fShelf->CountReplicants();
928a5210ab1SAxel Dörfler 	BView* view;
929a5210ab1SAxel Dörfler 	fShelf->ReplicantAt(count - 1, &view, (uint32*)id, NULL);
930a5210ab1SAxel Dörfler 
931a5210ab1SAxel Dörfler 	if (originalBounds != view->Bounds()) {
932a5210ab1SAxel Dörfler 		// The replicant changed its size when added to the window, so we need
93371bd3ba5SJonas Sundström 		// to recompute all over again (it's already done once via
93471bd3ba5SJonas Sundström 		// BShelf::AddReplicant() and TReplicantShelf::CanAcceptReplicantView())
935a5210ab1SAxel Dörfler 		RealignReplicants();
936a5210ab1SAxel Dörfler 	}
937a5210ab1SAxel Dörfler 
93841281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
93941281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
94041281cf3SAxel Dörfler 	float width, height;
94141281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
94241281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height)
94341281cf3SAxel Dörfler 		AdjustPlacement();
94441281cf3SAxel Dörfler 
94541281cf3SAxel Dörfler 	// add the item to the add-on list
94641281cf3SAxel Dörfler 
94741281cf3SAxel Dörfler 	AddItem(*id, nodeRef, entry, addOn != NULL);
94841281cf3SAxel Dörfler 	return B_OK;
94941281cf3SAxel Dörfler }
95041281cf3SAxel Dörfler 
95141281cf3SAxel Dörfler 
95241281cf3SAxel Dörfler void
95341281cf3SAxel Dörfler TReplicantTray::RemoveIcon(int32 target, bool byIndex)
95441281cf3SAxel Dörfler {
95541281cf3SAxel Dörfler 	if (target < 0)
95641281cf3SAxel Dörfler 		return;
95741281cf3SAxel Dörfler 
95841281cf3SAxel Dörfler 	int32 index, id;
95941281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
96041281cf3SAxel Dörfler 	if (view && index >= 0) {
9617da06231SAxel Dörfler 		// remove the reference from the item list & the shelf
96241281cf3SAxel Dörfler 		RemoveItem(id);
96341281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9647da06231SAxel Dörfler 
96541281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
96641281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
96741281cf3SAxel Dörfler 	}
96841281cf3SAxel Dörfler }
96941281cf3SAxel Dörfler 
97041281cf3SAxel Dörfler 
97141281cf3SAxel Dörfler void
97241281cf3SAxel Dörfler TReplicantTray::RemoveIcon(const char* name)
97341281cf3SAxel Dörfler {
97441281cf3SAxel Dörfler 	if (!name || strlen(name) <= 0)
97541281cf3SAxel Dörfler 		return;
97641281cf3SAxel Dörfler 
97741281cf3SAxel Dörfler 	int32 id, index;
97841281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
97941281cf3SAxel Dörfler 	if (view && index >= 0) {
9807da06231SAxel Dörfler 		// remove the reference from the item list & shelf
98141281cf3SAxel Dörfler 		RemoveItem(id);
98241281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9837da06231SAxel Dörfler 
98441281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
98541281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
98641281cf3SAxel Dörfler 	}
98741281cf3SAxel Dörfler }
98841281cf3SAxel Dörfler 
98941281cf3SAxel Dörfler 
99041281cf3SAxel Dörfler void
9917da06231SAxel Dörfler TReplicantTray::RealReplicantAdjustment(int32 startIndex)
99241281cf3SAxel Dörfler {
9937da06231SAxel Dörfler 	if (startIndex < 0)
99441281cf3SAxel Dörfler 		return;
9957da06231SAxel Dörfler 
9967da06231SAxel Dörfler 	if (startIndex == fLastReplicant)
9977da06231SAxel Dörfler 		startIndex = 0;
9987da06231SAxel Dörfler 
99941281cf3SAxel Dörfler 	// reset the locations of all replicants after the one deleted
10007da06231SAxel Dörfler 	RealignReplicants(startIndex);
100141281cf3SAxel Dörfler 
100241281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
100341281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
100441281cf3SAxel Dörfler 	float width, height;
100541281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
100641281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height) {
10071ad8c760SFredrik Holmqvist 		// resize view to accomodate the replicants, redraw as necessary
100841281cf3SAxel Dörfler 		AdjustPlacement();
100941281cf3SAxel Dörfler 	}
101041281cf3SAxel Dörfler }
101141281cf3SAxel Dörfler 
101241281cf3SAxel Dörfler 
10137da06231SAxel Dörfler /**	looking for a replicant by id/index
10147da06231SAxel Dörfler  *	return the view and index
10157da06231SAxel Dörfler  */
10167da06231SAxel Dörfler 
101741281cf3SAxel Dörfler BView*
101841281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, int32 target, bool byIndex)
101941281cf3SAxel Dörfler {
102041281cf3SAxel Dörfler 	*index = -1;
102141281cf3SAxel Dörfler 
102241281cf3SAxel Dörfler 	BView* view;
102341281cf3SAxel Dörfler 	if (byIndex) {
102441281cf3SAxel Dörfler 		if (fShelf->ReplicantAt(target, &view, (uint32*)id)) {
102541281cf3SAxel Dörfler 			if (view) {
102641281cf3SAxel Dörfler 				*index = target;
102741281cf3SAxel Dörfler 				return view;
102841281cf3SAxel Dörfler 			}
102941281cf3SAxel Dörfler 		}
103041281cf3SAxel Dörfler 	} else {
103141281cf3SAxel Dörfler 		int32 count = fShelf->CountReplicants() - 1;
103241281cf3SAxel Dörfler 		int32 localid;
103341281cf3SAxel Dörfler 		for (int32 repIndex = count ; repIndex >= 0 ; repIndex--) {
103441281cf3SAxel Dörfler 			fShelf->ReplicantAt(repIndex, &view, (uint32*)&localid);
103541281cf3SAxel Dörfler 			if (localid == target && view) {
103641281cf3SAxel Dörfler 				*index = repIndex;
103741281cf3SAxel Dörfler 				*id = localid;
103841281cf3SAxel Dörfler 				return view;
103941281cf3SAxel Dörfler 			}
104041281cf3SAxel Dörfler 		}
104141281cf3SAxel Dörfler 	}
104241281cf3SAxel Dörfler 	return NULL;
104341281cf3SAxel Dörfler }
104441281cf3SAxel Dörfler 
104541281cf3SAxel Dörfler 
10467da06231SAxel Dörfler /**	looking for a replicant with a view by name
10477da06231SAxel Dörfler  *	return the view, index and the id of the replicant
10487da06231SAxel Dörfler  */
10497da06231SAxel Dörfler 
105041281cf3SAxel Dörfler BView*
105141281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, const char* name)
105241281cf3SAxel Dörfler {
105341281cf3SAxel Dörfler 	*index = -1;
105441281cf3SAxel Dörfler 	*id = -1;
105541281cf3SAxel Dörfler 
105641281cf3SAxel Dörfler 	BView* view;
105741281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants()-1;
105841281cf3SAxel Dörfler 	for (int32 repIndex = count ; repIndex >= 0 ; repIndex--) {
105941281cf3SAxel Dörfler 		fShelf->ReplicantAt(repIndex, &view, (uint32*)id);
106041281cf3SAxel Dörfler 		if (view && view->Name() && strcmp(name, view->Name()) == 0) {
106141281cf3SAxel Dörfler 			*index = repIndex;
106241281cf3SAxel Dörfler 			return view;
106341281cf3SAxel Dörfler 		}
106441281cf3SAxel Dörfler 	}
106541281cf3SAxel Dörfler 	return NULL;
106641281cf3SAxel Dörfler }
106741281cf3SAxel Dörfler 
106841281cf3SAxel Dörfler 
10697da06231SAxel Dörfler /**	Shelf will call to determine where and if
10707da06231SAxel Dörfler  *	the replicant is to be added
10717da06231SAxel Dörfler  */
10727da06231SAxel Dörfler 
107341281cf3SAxel Dörfler bool
107441281cf3SAxel Dörfler TReplicantTray::AcceptAddon(BRect replicantFrame, BMessage* message)
107541281cf3SAxel Dörfler {
107641281cf3SAxel Dörfler 	if (!message)
107741281cf3SAxel Dörfler 		return false;
107841281cf3SAxel Dörfler 
107941281cf3SAxel Dörfler 	if (replicantFrame.Height() > kMaxReplicantHeight)
108041281cf3SAxel Dörfler 		return false;
108141281cf3SAxel Dörfler 
108241281cf3SAxel Dörfler 	alignment align = B_ALIGN_LEFT;
108341281cf3SAxel Dörfler 	if (fAlignmentSupport && message->HasBool("deskbar:dynamic_align")) {
108441281cf3SAxel Dörfler 		if (!fBarView->Vertical())
108541281cf3SAxel Dörfler 			align = B_ALIGN_RIGHT;
108641281cf3SAxel Dörfler 		else
108741281cf3SAxel Dörfler 			align = fBarView->Left() ? B_ALIGN_LEFT : B_ALIGN_RIGHT;
108841281cf3SAxel Dörfler 	} else if (message->HasInt32("deskbar:align"))
108941281cf3SAxel Dörfler 		message->FindInt32("deskbar:align", (int32*)&align);
109041281cf3SAxel Dörfler 
109141281cf3SAxel Dörfler 	if (message->HasInt32("deskbar:private_align"))
109241281cf3SAxel Dörfler 		message->FindInt32("deskbar:private_align", (int32*)&align);
109341281cf3SAxel Dörfler 	else
109441281cf3SAxel Dörfler 		align = B_ALIGN_LEFT;
109541281cf3SAxel Dörfler 
10967da06231SAxel Dörfler 	BPoint loc = LocationForReplicant(fShelf->CountReplicants(),
10977da06231SAxel Dörfler 		replicantFrame.Width());
109841281cf3SAxel Dörfler 
109941281cf3SAxel Dörfler 	message->AddPoint("_pjp_loc", loc);
110041281cf3SAxel Dörfler 	return true;
110141281cf3SAxel Dörfler }
110241281cf3SAxel Dörfler 
110341281cf3SAxel Dörfler 
110441281cf3SAxel Dörfler /**	based on the previous (index - 1) replicant in the list
110541281cf3SAxel Dörfler  *	calculate where the left point should be for this
110641281cf3SAxel Dörfler  *	replicant.  replicant will flow to the right on its own
110741281cf3SAxel Dörfler  */
110841281cf3SAxel Dörfler 
110941281cf3SAxel Dörfler BPoint
11107da06231SAxel Dörfler TReplicantTray::LocationForReplicant(int32 index, float width)
111141281cf3SAxel Dörfler {
111241281cf3SAxel Dörfler 	BPoint loc(kIconGap + 1, kGutter + 1);
111341281cf3SAxel Dörfler 
11147da06231SAxel Dörfler 	if (fMultiRowMode) {
11157da06231SAxel Dörfler 		// try to find free space in every row
11167da06231SAxel Dörfler 		for (int32 row = 0; ; loc.y += kMaxReplicantHeight + kIconGap, row++) {
11177da06231SAxel Dörfler 			// determine free space in this row
11182ce9bab8SJohn Scipione 			BRect rect(loc.x, loc.y, loc.x + fMinimumTrayWidth - kIconGap
11192ce9bab8SJohn Scipione 				- 2.0, loc.y + kMaxReplicantHeight);
1120573f748cSJohn Scipione 			if (row == 0 && !fTime->IsHidden())
1121573f748cSJohn Scipione 				rect.right -= fTime->Frame().Width() + kIconGap;
11227da06231SAxel Dörfler 
11237da06231SAxel Dörfler 			for (int32 i = 0; i < index; i++) {
11247da06231SAxel Dörfler 				BView* view = NULL;
11257da06231SAxel Dörfler 				fShelf->ReplicantAt(i, &view);
11267da06231SAxel Dörfler 				if (view == NULL || view->Frame().top != rect.top)
11277da06231SAxel Dörfler 					continue;
11287da06231SAxel Dörfler 
11297da06231SAxel Dörfler 				rect.left = view->Frame().right + kIconGap + 1;
11307da06231SAxel Dörfler 			}
11317da06231SAxel Dörfler 
11327da06231SAxel Dörfler 			if (rect.Width() >= width) {
11337da06231SAxel Dörfler 				// the icon fits in this row
11347da06231SAxel Dörfler 				loc = rect.LeftTop();
11357da06231SAxel Dörfler 				break;
11367da06231SAxel Dörfler 			}
11377da06231SAxel Dörfler 		}
11387da06231SAxel Dörfler 	} else {
113941281cf3SAxel Dörfler 		if (index > 0) {
114041281cf3SAxel Dörfler 			// get the last replicant added for placement reference
114141281cf3SAxel Dörfler 			BView* view = NULL;
11427da06231SAxel Dörfler 			fShelf->ReplicantAt(index - 1, &view);
114341281cf3SAxel Dörfler 			if (view) {
114441281cf3SAxel Dörfler 				// push this rep placement past the last one
114541281cf3SAxel Dörfler 				loc.x = view->Frame().right + kIconGap + 1;
114641281cf3SAxel Dörfler 				loc.y = view->Frame().top;
114741281cf3SAxel Dörfler 			}
114841281cf3SAxel Dörfler 		}
11497da06231SAxel Dörfler 	}
115041281cf3SAxel Dörfler 
115171bd3ba5SJonas Sundström 	if ((loc.y == fRightBottomReplicant.top && loc.x
115271bd3ba5SJonas Sundström 		> fRightBottomReplicant.left) || loc.y > fRightBottomReplicant.top) {
115371bd3ba5SJonas Sundström 		fRightBottomReplicant.Set(loc.x, loc.y, loc.x + width, loc.y
115471bd3ba5SJonas Sundström 		+ kMaxReplicantHeight);
11557da06231SAxel Dörfler 		fLastReplicant = index;
115641281cf3SAxel Dörfler 	}
11577da06231SAxel Dörfler 
115841281cf3SAxel Dörfler 	return loc;
115941281cf3SAxel Dörfler }
116041281cf3SAxel Dörfler 
116141281cf3SAxel Dörfler 
116241281cf3SAxel Dörfler BRect
116341281cf3SAxel Dörfler TReplicantTray::IconFrame(int32 target, bool byIndex)
116441281cf3SAxel Dörfler {
116541281cf3SAxel Dörfler 	int32 index, id;
116641281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
116741281cf3SAxel Dörfler 	if (view)
116841281cf3SAxel Dörfler 		return view->Frame();
116941281cf3SAxel Dörfler 
117041281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
117141281cf3SAxel Dörfler }
117241281cf3SAxel Dörfler 
117341281cf3SAxel Dörfler 
117441281cf3SAxel Dörfler BRect
117541281cf3SAxel Dörfler TReplicantTray::IconFrame(const char* name)
117641281cf3SAxel Dörfler {
117741281cf3SAxel Dörfler 	if (!name)
117841281cf3SAxel Dörfler 		return BRect(0, 0, 0, 0);
117941281cf3SAxel Dörfler 
118041281cf3SAxel Dörfler 	int32 id, index;
118141281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
118241281cf3SAxel Dörfler 	if (view)
118341281cf3SAxel Dörfler 		return view->Frame();
118441281cf3SAxel Dörfler 
118541281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
118641281cf3SAxel Dörfler }
118741281cf3SAxel Dörfler 
118841281cf3SAxel Dörfler 
11897da06231SAxel Dörfler /**	Scan from the startIndex and reset the location
11907da06231SAxel Dörfler  *	as defined in LocationForReplicant()
119141281cf3SAxel Dörfler  */
119241281cf3SAxel Dörfler 
119341281cf3SAxel Dörfler void
119441281cf3SAxel Dörfler TReplicantTray::RealignReplicants(int32 startIndex)
119541281cf3SAxel Dörfler {
119641281cf3SAxel Dörfler 	if (startIndex < 0)
119741281cf3SAxel Dörfler 		startIndex = 0;
119841281cf3SAxel Dörfler 
119941281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants();
120041281cf3SAxel Dörfler 	if (count <= 0)
120141281cf3SAxel Dörfler 		return;
120241281cf3SAxel Dörfler 
12037da06231SAxel Dörfler 	if (startIndex == 0)
12047da06231SAxel Dörfler 		fRightBottomReplicant.Set(0, 0, 0, 0);
12057da06231SAxel Dörfler 
120641281cf3SAxel Dörfler 	BView* view = NULL;
120741281cf3SAxel Dörfler 	for (int32 i = startIndex ; i < count ; i++) {
120841281cf3SAxel Dörfler 		fShelf->ReplicantAt(i, &view);
12090bec7100SStefano Ceccherini 		if (view != NULL) {
12107da06231SAxel Dörfler 			BPoint loc = LocationForReplicant(i, view->Frame().Width());
12110bec7100SStefano Ceccherini 			if (view->Frame().LeftTop() != loc)
121241281cf3SAxel Dörfler 				view->MoveTo(loc);
121341281cf3SAxel Dörfler 		}
121441281cf3SAxel Dörfler 	}
121541281cf3SAxel Dörfler }
121641281cf3SAxel Dörfler 
121741281cf3SAxel Dörfler 
12188aab28f1SRene Gollent status_t
12197625ce51SRene Gollent TReplicantTray::_SaveSettings()
12207625ce51SRene Gollent {
12217625ce51SRene Gollent 	status_t result;
12227625ce51SRene Gollent 	BPath path;
12237625ce51SRene Gollent 	if ((result = find_directory(B_USER_SETTINGS_DIRECTORY, &path, true))
12247625ce51SRene Gollent 		 == B_OK) {
12257625ce51SRene Gollent 		path.Append(kReplicantSettingsFile);
12267625ce51SRene Gollent 
12277625ce51SRene Gollent 		BFile file(path.Path(), B_READ_WRITE | B_CREATE_FILE | B_ERASE_FILE);
12287625ce51SRene Gollent 		if ((result = file.InitCheck()) == B_OK)
12297625ce51SRene Gollent 			result = fAddOnSettings.Flatten(&file);
12307625ce51SRene Gollent 	}
12317625ce51SRene Gollent 
12327625ce51SRene Gollent 	return result;
12337625ce51SRene Gollent }
12347625ce51SRene Gollent 
12357625ce51SRene Gollent 
12363cf2d117SJohn Scipione void
12373cf2d117SJohn Scipione TReplicantTray::SaveTimeSettings()
12383cf2d117SJohn Scipione {
12393cf2d117SJohn Scipione 	if (fTime == NULL)
12403cf2d117SJohn Scipione 		return;
12413cf2d117SJohn Scipione 
12423cf2d117SJohn Scipione 	desk_settings* settings = ((TBarApp*)be_app)->Settings();
12433cf2d117SJohn Scipione 	settings->showTime = !fTime->IsHidden();
12443cf2d117SJohn Scipione 	settings->showSeconds = fTime->ShowSeconds();
12453cf2d117SJohn Scipione 	settings->showDayOfWeek = fTime->ShowDayOfWeek();
12463cf2d117SJohn Scipione }
12473cf2d117SJohn Scipione 
12483cf2d117SJohn Scipione 
124941281cf3SAxel Dörfler //	#pragma mark -
125041281cf3SAxel Dörfler 
125141281cf3SAxel Dörfler 
1252ca9acc20SAxel Dörfler /*! Draggable region that is asynchronous so that dragging does not block
1253ca9acc20SAxel Dörfler 	other activities.
12547da06231SAxel Dörfler */
125541281cf3SAxel Dörfler TDragRegion::TDragRegion(TBarView* parent, BView* child)
1256ca9acc20SAxel Dörfler 	:
1257ca9acc20SAxel Dörfler 	BControl(BRect(0, 0, 0, 0), "", "", NULL, B_FOLLOW_NONE,
125841281cf3SAxel Dörfler 		B_WILL_DRAW | B_FRAME_EVENTS),
125941281cf3SAxel Dörfler 	fBarView(parent),
126041281cf3SAxel Dörfler 	fChild(child),
126141281cf3SAxel Dörfler 	fDragLocation(kAutoPlaceDragRegion)
126241281cf3SAxel Dörfler {
126341281cf3SAxel Dörfler }
126441281cf3SAxel Dörfler 
126541281cf3SAxel Dörfler 
126641281cf3SAxel Dörfler void
126741281cf3SAxel Dörfler TDragRegion::AttachedToWindow()
126841281cf3SAxel Dörfler {
126941281cf3SAxel Dörfler 	BView::AttachedToWindow();
1270cb6afcb1SStephan Aßmus 	if (be_control_look != NULL)
1271cb6afcb1SStephan Aßmus 		SetViewColor(tint_color(ui_color(B_MENU_BACKGROUND_COLOR), 1.1));
1272cb6afcb1SStephan Aßmus 	else
127341281cf3SAxel Dörfler 		SetViewColor(ui_color(B_MENU_BACKGROUND_COLOR));
127441281cf3SAxel Dörfler 	ResizeToPreferred();
127541281cf3SAxel Dörfler }
127641281cf3SAxel Dörfler 
127741281cf3SAxel Dörfler 
127841281cf3SAxel Dörfler void
127941281cf3SAxel Dörfler TDragRegion::GetPreferredSize(float* width, float* height)
128041281cf3SAxel Dörfler {
128141281cf3SAxel Dörfler 	fChild->ResizeToPreferred();
128241281cf3SAxel Dörfler 	*width = fChild->Bounds().Width();
128341281cf3SAxel Dörfler 	*height = fChild->Bounds().Height();
128441281cf3SAxel Dörfler 
128541281cf3SAxel Dörfler 	if (fDragLocation != kNoDragRegion)
128641281cf3SAxel Dörfler 		*width += 7;
128741281cf3SAxel Dörfler 	else
128841281cf3SAxel Dörfler 		*width += 6;
128941281cf3SAxel Dörfler 
129041281cf3SAxel Dörfler 	*height += 3;
129141281cf3SAxel Dörfler }
129241281cf3SAxel Dörfler 
129341281cf3SAxel Dörfler 
129441281cf3SAxel Dörfler void
129541281cf3SAxel Dörfler TDragRegion::FrameMoved(BPoint)
129641281cf3SAxel Dörfler {
1297ca9acc20SAxel Dörfler 	if (fBarView->Left() && fBarView->Vertical()
1298ca9acc20SAxel Dörfler 		&& fDragLocation != kNoDragRegion)
129941281cf3SAxel Dörfler 		fChild->MoveTo(5, 2);
130041281cf3SAxel Dörfler 	else
130141281cf3SAxel Dörfler 		fChild->MoveTo(2, 2);
130241281cf3SAxel Dörfler }
130341281cf3SAxel Dörfler 
130441281cf3SAxel Dörfler 
130541281cf3SAxel Dörfler void
130641281cf3SAxel Dörfler TDragRegion::Draw(BRect)
130741281cf3SAxel Dörfler {
1308eaa9af99SStephan Aßmus 	rgb_color menuColor = ViewColor();
130941281cf3SAxel Dörfler 	rgb_color hilite = tint_color(menuColor, B_DARKEN_1_TINT);
1310cb6afcb1SStephan Aßmus 	rgb_color ldark = tint_color(menuColor, 1.02);
1311cb6afcb1SStephan Aßmus 	rgb_color dark = tint_color(menuColor, B_DARKEN_2_TINT);
131241281cf3SAxel Dörfler 	rgb_color vvdark = tint_color(menuColor, B_DARKEN_4_TINT);
131341281cf3SAxel Dörfler 	rgb_color light = tint_color(menuColor, B_LIGHTEN_2_TINT);
131441281cf3SAxel Dörfler 
131541281cf3SAxel Dörfler 	BRect frame(Bounds());
131641281cf3SAxel Dörfler 	BeginLineArray(4);
131741281cf3SAxel Dörfler 
1318cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1319cb6afcb1SStephan Aßmus 		if (fBarView->Vertical()) {
1320cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(), frame.RightTop(), dark);
1321cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1322cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.top + 1), ldark);
1323cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
1324cb6afcb1SStephan Aßmus 		} else if (fBarView->AcrossTop() || fBarView->AcrossBottom()) {
1325cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(),
1326cb6afcb1SStephan Aßmus 				BPoint(frame.left, frame.bottom), dark);
1327cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.top + 1),
1328cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
1329cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right, frame.top + 2),
1330cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.bottom), hilite);
1331cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.bottom),
1332cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
1333cb6afcb1SStephan Aßmus 		}
1334cb6afcb1SStephan Aßmus 	} else {
133541281cf3SAxel Dörfler 		if (fBarView->Vertical()) {
133641281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.RightTop(), light);
133741281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.LeftBottom(), light);
133841281cf3SAxel Dörfler 			AddLine(frame.RightBottom(), frame.RightTop(), hilite);
133941281cf3SAxel Dörfler 		} else if (fBarView->AcrossTop()) {
1340cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1341cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
134241281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1343cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 2),
1344cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1345cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(),
1346cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
134741281cf3SAxel Dörfler 		} else if (fBarView->AcrossBottom()) {
1348cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1349cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
135041281cf3SAxel Dörfler 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
135141281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1352cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 1),
1353cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1354cb6afcb1SStephan Aßmus 		}
135541281cf3SAxel Dörfler 	}
135641281cf3SAxel Dörfler 
135741281cf3SAxel Dörfler 	EndLineArray();
135841281cf3SAxel Dörfler 
135941281cf3SAxel Dörfler 	if (fDragLocation != kDontDrawDragRegion || fDragLocation != kNoDragRegion)
136041281cf3SAxel Dörfler 		DrawDragRegion();
136141281cf3SAxel Dörfler }
136241281cf3SAxel Dörfler 
136341281cf3SAxel Dörfler 
136441281cf3SAxel Dörfler void
136541281cf3SAxel Dörfler TDragRegion::DrawDragRegion()
136641281cf3SAxel Dörfler {
136741281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
136841281cf3SAxel Dörfler 
1369cb6afcb1SStephan Aßmus 	rgb_color menuColor = ViewColor();
1370cb6afcb1SStephan Aßmus 	rgb_color menuHilite = menuColor;
137141281cf3SAxel Dörfler 	if (IsTracking()) {
1372cb6afcb1SStephan Aßmus 		// Draw drag region highlighted if tracking mouse
1373cb6afcb1SStephan Aßmus 		menuHilite = tint_color(menuColor, B_HIGHLIGHT_BACKGROUND_TINT);
137441281cf3SAxel Dörfler 		SetHighColor(menuHilite);
137541281cf3SAxel Dörfler 		FillRect(dragRegion);
137641281cf3SAxel Dörfler 	}
1377cb6afcb1SStephan Aßmus 	rgb_color vdark = tint_color(menuHilite, B_DARKEN_3_TINT);
1378cb6afcb1SStephan Aßmus 	rgb_color light = tint_color(menuHilite, B_LIGHTEN_2_TINT);
1379cb6afcb1SStephan Aßmus 
1380cb6afcb1SStephan Aßmus 	BeginLineArray(dragRegion.IntegerHeight());
1381cb6afcb1SStephan Aßmus 	BPoint pt;
1382cb6afcb1SStephan Aßmus 	pt.x = floorf((dragRegion.left + dragRegion.right) / 2 + 0.5) - 1;
1383cb6afcb1SStephan Aßmus 	pt.y = dragRegion.top + 2;
1384cb6afcb1SStephan Aßmus 
138541281cf3SAxel Dörfler 	while (pt.y + 1 <= dragRegion.bottom) {
138641281cf3SAxel Dörfler 		AddLine(pt, pt, vdark);
138741281cf3SAxel Dörfler 		AddLine(pt + BPoint(1, 1), pt + BPoint(1, 1), light);
138841281cf3SAxel Dörfler 
138941281cf3SAxel Dörfler 		pt.y += 3;
139041281cf3SAxel Dörfler 	}
139141281cf3SAxel Dörfler 	EndLineArray();
139241281cf3SAxel Dörfler }
139341281cf3SAxel Dörfler 
139441281cf3SAxel Dörfler 
139541281cf3SAxel Dörfler BRect
139641281cf3SAxel Dörfler TDragRegion::DragRegion() const
139741281cf3SAxel Dörfler {
1398cb6afcb1SStephan Aßmus 	float kTopBottomInset = 2;
1399cb6afcb1SStephan Aßmus 	float kLeftRightInset = 1;
1400cb6afcb1SStephan Aßmus 	float kDragWidth = 3;
1401cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1402cb6afcb1SStephan Aßmus 		kTopBottomInset = 1;
1403cb6afcb1SStephan Aßmus 		kLeftRightInset = 0;
1404cb6afcb1SStephan Aßmus 		kDragWidth = 4;
1405cb6afcb1SStephan Aßmus 	}
1406cb6afcb1SStephan Aßmus 
140741281cf3SAxel Dörfler 	BRect dragRegion(Bounds());
1408cb6afcb1SStephan Aßmus 	dragRegion.top += kTopBottomInset;
1409cb6afcb1SStephan Aßmus 	dragRegion.bottom -= kTopBottomInset;
141041281cf3SAxel Dörfler 
141141281cf3SAxel Dörfler 	bool placeOnLeft = false;
141241281cf3SAxel Dörfler 	if (fDragLocation == kAutoPlaceDragRegion) {
141341281cf3SAxel Dörfler 		if (fBarView->Vertical() && fBarView->Left())
141441281cf3SAxel Dörfler 			placeOnLeft = true;
141541281cf3SAxel Dörfler 		else
141641281cf3SAxel Dörfler 			placeOnLeft = false;
141741281cf3SAxel Dörfler 	} else if (fDragLocation == kDragRegionLeft)
141841281cf3SAxel Dörfler 		placeOnLeft = true;
141941281cf3SAxel Dörfler 	else if (fDragLocation == kDragRegionRight)
142041281cf3SAxel Dörfler 		placeOnLeft = false;
142141281cf3SAxel Dörfler 
142241281cf3SAxel Dörfler 	if (placeOnLeft) {
1423cb6afcb1SStephan Aßmus 		dragRegion.left += kLeftRightInset;
1424cb6afcb1SStephan Aßmus 		dragRegion.right = dragRegion.left + kDragWidth;
142541281cf3SAxel Dörfler 	} else {
1426cb6afcb1SStephan Aßmus 		dragRegion.right -= kLeftRightInset;
1427cb6afcb1SStephan Aßmus 		dragRegion.left = dragRegion.right - kDragWidth;
142841281cf3SAxel Dörfler 	}
142941281cf3SAxel Dörfler 
143041281cf3SAxel Dörfler 	return dragRegion;
143141281cf3SAxel Dörfler }
143241281cf3SAxel Dörfler 
143341281cf3SAxel Dörfler 
143441281cf3SAxel Dörfler void
143541281cf3SAxel Dörfler TDragRegion::MouseDown(BPoint thePoint)
143641281cf3SAxel Dörfler {
143741281cf3SAxel Dörfler 	ulong buttons;
143841281cf3SAxel Dörfler 	BPoint where;
143941281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
144041281cf3SAxel Dörfler 
14417da06231SAxel Dörfler 	dragRegion.InsetBy(-2.0f, -2.0f);
14427da06231SAxel Dörfler 		// DragRegion() is designed for drawing, not clicking
144341281cf3SAxel Dörfler 
144441281cf3SAxel Dörfler 	if (!dragRegion.Contains(thePoint))
144541281cf3SAxel Dörfler 		return;
144641281cf3SAxel Dörfler 
144741281cf3SAxel Dörfler 	while (true) {
144841281cf3SAxel Dörfler 		GetMouse(&where, &buttons);
144941281cf3SAxel Dörfler 		if (!buttons)
145041281cf3SAxel Dörfler 			break;
145141281cf3SAxel Dörfler 
145241281cf3SAxel Dörfler 		if ((Window()->Flags() & B_ASYNCHRONOUS_CONTROLS) != 0) {
145341281cf3SAxel Dörfler 			fPreviousPosition = thePoint;
145441281cf3SAxel Dörfler 			SetTracking(true);
145541281cf3SAxel Dörfler 			SetMouseEventMask(B_POINTER_EVENTS,
145641281cf3SAxel Dörfler 				B_NO_POINTER_HISTORY | B_LOCK_WINDOW_FOCUS);
145741281cf3SAxel Dörfler 			Invalidate(DragRegion());
145841281cf3SAxel Dörfler 			break;
145941281cf3SAxel Dörfler 		}
146041281cf3SAxel Dörfler 
146141281cf3SAxel Dörfler 		snooze(25000);
146241281cf3SAxel Dörfler 	}
146341281cf3SAxel Dörfler }
146441281cf3SAxel Dörfler 
146541281cf3SAxel Dörfler 
146641281cf3SAxel Dörfler void
146741281cf3SAxel Dörfler TDragRegion::MouseUp(BPoint pt)
146841281cf3SAxel Dörfler {
146941281cf3SAxel Dörfler 	if (IsTracking()) {
147041281cf3SAxel Dörfler 		SetTracking(false);
147141281cf3SAxel Dörfler 		Invalidate(DragRegion());
147241281cf3SAxel Dörfler 	} else
147341281cf3SAxel Dörfler 		BControl::MouseUp(pt);
147441281cf3SAxel Dörfler }
147541281cf3SAxel Dörfler 
147641281cf3SAxel Dörfler 
147741281cf3SAxel Dörfler bool
147841281cf3SAxel Dörfler TDragRegion::SwitchModeForRect(BPoint mouse, BRect rect,
147941281cf3SAxel Dörfler 	bool newVertical, bool newLeft, bool newTop, int32 newState)
148041281cf3SAxel Dörfler {
14811ad8c760SFredrik Holmqvist 	if (!rect.Contains(mouse)) {
148241281cf3SAxel Dörfler 		// not our rect
148341281cf3SAxel Dörfler 		return false;
14841ad8c760SFredrik Holmqvist 	}
148541281cf3SAxel Dörfler 
14861ad8c760SFredrik Holmqvist 	if (newVertical == fBarView->Vertical() && newLeft == fBarView->Left()
14871ad8c760SFredrik Holmqvist 		&& newTop == fBarView->Top() && newState == fBarView->State()) {
148841281cf3SAxel Dörfler 		// already in the correct mode
148941281cf3SAxel Dörfler 		return true;
14901ad8c760SFredrik Holmqvist 	}
149141281cf3SAxel Dörfler 
149241281cf3SAxel Dörfler 	fBarView->ChangeState(newState, newVertical, newLeft, newTop);
149341281cf3SAxel Dörfler 	return true;
149441281cf3SAxel Dörfler }
149541281cf3SAxel Dörfler 
149641281cf3SAxel Dörfler 
149741281cf3SAxel Dörfler void
149841281cf3SAxel Dörfler TDragRegion::MouseMoved(BPoint where, uint32 code, const BMessage* message)
149941281cf3SAxel Dörfler {
150041281cf3SAxel Dörfler 	if (IsTracking()) {
150141281cf3SAxel Dörfler 		BScreen screen;
150241281cf3SAxel Dörfler 		BRect frame = screen.Frame();
150341281cf3SAxel Dörfler 
150441281cf3SAxel Dörfler 		float hDivider = frame.Width() / 6;
150571bd3ba5SJonas Sundström 		hDivider = (hDivider < sMinimumWindowWidth + 10.0f)
150671bd3ba5SJonas Sundström 			? sMinimumWindowWidth + 10.0f : hDivider;
150741281cf3SAxel Dörfler 		float miniDivider = frame.top + kMiniHeight + 10.0f;
150841281cf3SAxel Dörfler 		float vDivider = frame.Height() / 2;
150941281cf3SAxel Dörfler #ifdef FULL_MODE
151041281cf3SAxel Dörfler 		float thirdScreen = frame.Height() / 3;
151141281cf3SAxel Dörfler #endif
151271bd3ba5SJonas Sundström 		BRect topLeft(frame.left, frame.top, frame.left + hDivider,
151371bd3ba5SJonas Sundström 			miniDivider);
151471bd3ba5SJonas Sundström 		BRect topMiddle(frame.left + hDivider, frame.top, frame.right
151571bd3ba5SJonas Sundström 			- hDivider, vDivider);
151671bd3ba5SJonas Sundström 		BRect topRight(frame.right - hDivider, frame.top, frame.right,
151771bd3ba5SJonas Sundström 			miniDivider);
151841281cf3SAxel Dörfler 
151941281cf3SAxel Dörfler #ifdef FULL_MODE
152041281cf3SAxel Dörfler 		vDivider = miniDivider + thirdScreen;
152141281cf3SAxel Dörfler #endif
152271bd3ba5SJonas Sundström 		BRect middleLeft(frame.left, miniDivider, frame.left + hDivider,
152371bd3ba5SJonas Sundström 			vDivider);
152471bd3ba5SJonas Sundström 		BRect middleRight(frame.right - hDivider, miniDivider, frame.right,
152571bd3ba5SJonas Sundström 			vDivider);
152641281cf3SAxel Dörfler 
152741281cf3SAxel Dörfler #ifdef FULL_MODE
152871bd3ba5SJonas Sundström 		BRect leftSide(frame.left, vDivider, frame.left + hDivider,
152971bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
153071bd3ba5SJonas Sundström 		BRect rightSide(frame.right - hDivider, vDivider, frame.right,
153171bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
153241281cf3SAxel Dörfler 
153341281cf3SAxel Dörfler 		vDivider = frame.bottom - thirdScreen;
153441281cf3SAxel Dörfler #endif
153571bd3ba5SJonas Sundström 		BRect bottomLeft(frame.left, vDivider, frame.left + hDivider,
153671bd3ba5SJonas Sundström 			frame.bottom);
153771bd3ba5SJonas Sundström 		BRect bottomMiddle(frame.left + hDivider, vDivider, frame.right
153871bd3ba5SJonas Sundström 			- hDivider, frame.bottom);
153971bd3ba5SJonas Sundström 		BRect bottomRight(frame.right - hDivider, vDivider, frame.right,
154071bd3ba5SJonas Sundström 			frame.bottom);
154141281cf3SAxel Dörfler 
154241281cf3SAxel Dörfler 		if (where != fPreviousPosition) {
154341281cf3SAxel Dörfler 			fPreviousPosition = where;
154441281cf3SAxel Dörfler 			ConvertToScreen(&where);
154541281cf3SAxel Dörfler 
154641281cf3SAxel Dörfler 			// use short circuit evaluation for convenience
154741281cf3SAxel Dörfler 			if (SwitchModeForRect(where, topLeft, true, true, true, kMiniState)
154871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topMiddle, false, true, true,
154971bd3ba5SJonas Sundström 					kExpandoState)
155071bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topRight, true, false, true,
155171bd3ba5SJonas Sundström 					kMiniState)
155271bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleLeft, true, true, true,
155371bd3ba5SJonas Sundström 					kExpandoState)
155471bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleRight, true, false, true,
155571bd3ba5SJonas Sundström 					kExpandoState)
155641281cf3SAxel Dörfler 
155741281cf3SAxel Dörfler #ifdef FULL_MODE
155871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, leftSide, true, true, true,
155971bd3ba5SJonas Sundström 					kFullState)
156071bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, rightSide, true, false, true,
156171bd3ba5SJonas Sundström 					kFullState)
156241281cf3SAxel Dörfler #endif
156371bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomLeft, true, true, false,
156471bd3ba5SJonas Sundström 					kMiniState)
156571bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomMiddle, false, true, false,
156671bd3ba5SJonas Sundström 					kExpandoState)
156771bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomRight, true, false, false,
156871bd3ba5SJonas Sundström 					kMiniState))
156941281cf3SAxel Dörfler 				;
157041281cf3SAxel Dörfler 		}
157141281cf3SAxel Dörfler 	} else
157241281cf3SAxel Dörfler 		BControl::MouseMoved(where, code, message);
157341281cf3SAxel Dörfler }
157441281cf3SAxel Dörfler 
157541281cf3SAxel Dörfler 
157641281cf3SAxel Dörfler int32
157741281cf3SAxel Dörfler TDragRegion::DragRegionLocation() const
157841281cf3SAxel Dörfler {
157941281cf3SAxel Dörfler 	return fDragLocation;
158041281cf3SAxel Dörfler }
158141281cf3SAxel Dörfler 
158241281cf3SAxel Dörfler 
158341281cf3SAxel Dörfler void
158441281cf3SAxel Dörfler TDragRegion::SetDragRegionLocation(int32 location)
158541281cf3SAxel Dörfler {
158641281cf3SAxel Dörfler 	if (location == fDragLocation)
158741281cf3SAxel Dörfler 		return;
158841281cf3SAxel Dörfler 
158941281cf3SAxel Dörfler 	fDragLocation = location;
159041281cf3SAxel Dörfler 	Invalidate();
159141281cf3SAxel Dörfler }
1592