xref: /haiku/src/apps/deskbar/StatusView.cpp (revision 1cd61330ecd8257551d58f204332d00145c855ec)
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 
36*1cd61330SJohn Scipione 
37*1cd61330SJohn 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>
54*1cd61330SJohn 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 
69cb6afcb1SStephan Aßmus #include "BarApp.h"
70323b6546SOliver Tappe #include "DeskbarUtils.h"
71*1cd61330SJohn Scipione #include "icons_logo.h"
72cb6afcb1SStephan Aßmus #include "ResourceSet.h"
7341281cf3SAxel Dörfler #include "StatusViewShelf.h"
7441281cf3SAxel Dörfler #include "TimeView.h"
7541281cf3SAxel Dörfler 
766d01b66dSAxel Dörfler using std::max;
7741281cf3SAxel Dörfler 
7841281cf3SAxel Dörfler #ifdef DB_ADDONS
7941281cf3SAxel Dörfler // Add-on support
8041281cf3SAxel Dörfler //
8141281cf3SAxel Dörfler // Item - internal item list (node, eref, etc)
8241281cf3SAxel Dörfler // Icon - physical replicant handed to the DeskbarClass class
8341281cf3SAxel Dörfler // AddOn - attribute based add-on
8441281cf3SAxel Dörfler 
8541281cf3SAxel Dörfler const char* const kInstantiateItemCFunctionName = "instantiate_deskbar_item";
8641281cf3SAxel Dörfler const char* const kInstantiateEntryCFunctionName = "instantiate_deskbar_entry";
873e6ff860SRene Gollent const char* const kReplicantSettingsFile = "Deskbar_replicants";
88c0ad1c91SRene Gollent const char* const kReplicantPathField = "replicant_path";
8941281cf3SAxel Dörfler 
90cb6afcb1SStephan Aßmus float sMinimumWindowWidth = kGutter + kMinimumTrayWidth + kDragRegionWidth;
91cb6afcb1SStephan Aßmus 
9241281cf3SAxel Dörfler 
9341281cf3SAxel Dörfler static void
9441281cf3SAxel Dörfler DumpItem(DeskbarItemInfo* item)
9541281cf3SAxel Dörfler {
9641281cf3SAxel Dörfler 	printf("is addon: %i, id: %li\n", item->isAddOn, item->id);
9766eba86fSAxel Dörfler 	printf("entry_ref:  %ld, %Ld, %s\n", item->entryRef.device,
9866eba86fSAxel Dörfler 		item->entryRef.directory, item->entryRef.name);
9941281cf3SAxel Dörfler 	printf("node_ref:  %ld, %Ld\n", item->nodeRef.device, item->nodeRef.node);
10041281cf3SAxel Dörfler }
10141281cf3SAxel Dörfler 
10241281cf3SAxel Dörfler 
10341281cf3SAxel Dörfler static void
10441281cf3SAxel Dörfler DumpList(BList* itemlist)
10541281cf3SAxel Dörfler {
10641281cf3SAxel Dörfler 	int32 count = itemlist->CountItems() - 1;
10741281cf3SAxel Dörfler 	if (count < 0) {
10841281cf3SAxel Dörfler 		printf("no items in list\n");
10941281cf3SAxel Dörfler 		return;
11041281cf3SAxel Dörfler 	}
11141281cf3SAxel Dörfler 	for (int32 i = count; i >= 0; i--) {
11241281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)itemlist->ItemAt(i);
11341281cf3SAxel Dörfler 		if (!item)
11441281cf3SAxel Dörfler 			continue;
11541281cf3SAxel Dörfler 
11641281cf3SAxel Dörfler 		DumpItem(item);
11741281cf3SAxel Dörfler 	}
11841281cf3SAxel Dörfler }
11941281cf3SAxel Dörfler #endif	/* DB_ADDONS */
12041281cf3SAxel Dörfler 
12141281cf3SAxel Dörfler 
122546208a5SOliver Tappe #undef B_TRANSLATION_CONTEXT
123546208a5SOliver Tappe #define B_TRANSLATION_CONTEXT "Tray"
124ae7a6e3aSAdrien Destugues 
1257da06231SAxel Dörfler // don't change the name of this view to anything other than "Status"!
1267da06231SAxel Dörfler 
12741281cf3SAxel Dörfler TReplicantTray::TReplicantTray(TBarView* parent, bool vertical)
1280d2ac94cSJohn Scipione 	:
1299ac6fee1SJohn Scipione 	BView(BRect(0, 0, 1, 1), "Status", B_FOLLOW_LEFT | B_FOLLOW_TOP,
1300d2ac94cSJohn Scipione 		B_WILL_DRAW | B_FRAME_EVENTS),
131573f748cSJohn Scipione 	fTime(NULL),
13241281cf3SAxel Dörfler 	fBarView(parent),
13341281cf3SAxel Dörfler 	fShelf(new TReplicantShelf(this)),
13441281cf3SAxel Dörfler 	fMultiRowMode(vertical),
135cb6afcb1SStephan Aßmus 	fMinimumTrayWidth(kMinimumTrayWidth),
13641281cf3SAxel Dörfler 	fAlignmentSupport(false)
13741281cf3SAxel Dörfler {
138cb6afcb1SStephan Aßmus 	// init the minimum window width according to the logo.
139cb6afcb1SStephan Aßmus 	const BBitmap* logoBitmap = AppResSet()->FindBitmap(B_MESSAGE_TYPE,
1409e8a970bSJonas Sundström 		R_LeafLogoBitmap);
141cb6afcb1SStephan Aßmus 	if (logoBitmap != NULL) {
142cb6afcb1SStephan Aßmus 		sMinimumWindowWidth = max_c(sMinimumWindowWidth,
143cb6afcb1SStephan Aßmus 			2 * (logoBitmap->Bounds().Width() + 8));
144cb6afcb1SStephan Aßmus 		fMinimumTrayWidth = sMinimumWindowWidth - kGutter - kDragRegionWidth;
145cb6afcb1SStephan Aßmus 	}
146573f748cSJohn Scipione 
1473cf2d117SJohn Scipione 	BFormattingConventions conventions;
1483cf2d117SJohn Scipione 	BLocale::Default()->GetFormattingConventions(&conventions);
1493cf2d117SJohn Scipione 	bool use24HourClock = conventions.Use24HourClock();
1503cf2d117SJohn Scipione 	desk_settings* settings = ((TBarApp*)be_app)->Settings();
1513cf2d117SJohn Scipione 
152573f748cSJohn Scipione 	// Create the time view
153573f748cSJohn Scipione 	fTime = new TTimeView(fMinimumTrayWidth, kMaxReplicantHeight - 1.0,
154f5be13ecSJohn Scipione 		use24HourClock, settings->showSeconds, settings->showDayOfWeek);
15541281cf3SAxel Dörfler }
15641281cf3SAxel Dörfler 
15741281cf3SAxel Dörfler 
15841281cf3SAxel Dörfler TReplicantTray::~TReplicantTray()
15941281cf3SAxel Dörfler {
16041281cf3SAxel Dörfler 	delete fShelf;
161573f748cSJohn Scipione 	delete fTime;
16241281cf3SAxel Dörfler }
16341281cf3SAxel Dörfler 
16441281cf3SAxel Dörfler 
16541281cf3SAxel Dörfler void
16641281cf3SAxel Dörfler TReplicantTray::AttachedToWindow()
16741281cf3SAxel Dörfler {
16841281cf3SAxel Dörfler 	BView::AttachedToWindow();
16941281cf3SAxel Dörfler 
170cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
171cb6afcb1SStephan Aßmus 		SetViewColor(Parent()->ViewColor());
172cb6afcb1SStephan Aßmus 	} else {
17366eba86fSAxel Dörfler 		SetViewColor(tint_color(ui_color(B_MENU_BACKGROUND_COLOR),
17466eba86fSAxel Dörfler 			B_DARKEN_1_TINT));
175cb6afcb1SStephan Aßmus 	}
17641281cf3SAxel Dörfler 	SetDrawingMode(B_OP_COPY);
17766eba86fSAxel Dörfler 
17841281cf3SAxel Dörfler 	Window()->SetPulseRate(1000000);
17941281cf3SAxel Dörfler 
180573f748cSJohn Scipione 	AddChild(fTime);
181573f748cSJohn Scipione 	fTime->MoveTo(Bounds().right - fTime->Bounds().Width() - 1, 2);
182573f748cSJohn Scipione 	if (!((TBarApp*)be_app)->Settings()->showTime) {
183573f748cSJohn Scipione 		fTime->Hide();
184573f748cSJohn Scipione 		RealignReplicants();
185573f748cSJohn Scipione 		AdjustPlacement();
186573f748cSJohn Scipione 	}
187ae7a6e3aSAdrien Destugues 
18841281cf3SAxel Dörfler #ifdef DB_ADDONS
18941281cf3SAxel Dörfler 	// load addons and rehydrate archives
1903a6add49SAxel Dörfler #if !defined(HAIKU_TARGET_PLATFORM_LIBBE_TEST)
19141281cf3SAxel Dörfler 	InitAddOnSupport();
19241281cf3SAxel Dörfler #endif
1939c910f9eSStephan Aßmus #endif
19441281cf3SAxel Dörfler 	ResizeToPreferred();
19541281cf3SAxel Dörfler }
19641281cf3SAxel Dörfler 
19741281cf3SAxel Dörfler 
19841281cf3SAxel Dörfler void
19941281cf3SAxel Dörfler TReplicantTray::DetachedFromWindow()
20041281cf3SAxel Dörfler {
20141281cf3SAxel Dörfler #ifdef DB_ADDONS
20241281cf3SAxel Dörfler 	// clean up add-on support
203465b54cdSAxel Dörfler #if !defined(HAIKU_TARGET_PLATFORM_LIBBE_TEST)
20441281cf3SAxel Dörfler 	DeleteAddOnSupport();
20541281cf3SAxel Dörfler #endif
206465b54cdSAxel Dörfler #endif
20741281cf3SAxel Dörfler 	BView::DetachedFromWindow();
20841281cf3SAxel Dörfler }
20941281cf3SAxel Dörfler 
21041281cf3SAxel Dörfler 
2111ad8c760SFredrik Holmqvist /*! Width is set to a minimum of kMinimumReplicantCount by kMaxReplicantWidth
21266eba86fSAxel Dörfler 	if not in multirowmode and greater than kMinimumReplicantCount
2132ce9bab8SJohn Scipione 	the width should be calculated based on the actual replicant widths
21441281cf3SAxel Dörfler */
21541281cf3SAxel Dörfler void
21641281cf3SAxel Dörfler TReplicantTray::GetPreferredSize(float* preferredWidth, float* preferredHeight)
21741281cf3SAxel Dörfler {
21841281cf3SAxel Dörfler 	float width = 0, height = kMinimumTrayHeight;
21941281cf3SAxel Dörfler 
22041281cf3SAxel Dörfler 	if (fMultiRowMode) {
2217da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0)
2227da06231SAxel Dörfler 			height = fRightBottomReplicant.bottom;
22341281cf3SAxel Dörfler 
2241ad8c760SFredrik Holmqvist 		// the height will be uniform for the number of rows necessary to show
2251ad8c760SFredrik Holmqvist 		// all the reps + any gutters necessary for spacing
22641281cf3SAxel Dörfler 		int32 rowCount = (int32)(height / kMaxReplicantHeight);
22741281cf3SAxel Dörfler 		height = kGutter + (rowCount * kMaxReplicantHeight)
22841281cf3SAxel Dörfler 			+ ((rowCount - 1) * kIconGap) + kGutter;
22941281cf3SAxel Dörfler 		height = max(kMinimumTrayHeight, height);
230cb6afcb1SStephan Aßmus 		width = fMinimumTrayWidth;
23141281cf3SAxel Dörfler 	} else {
2321ad8c760SFredrik Holmqvist 		// if last replicant overruns clock then resize to accomodate
2337da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0) {
234573f748cSJohn Scipione 			if (!fTime->IsHidden() && fTime->Frame().left
235573f748cSJohn Scipione 				< fRightBottomReplicant.right + 6) {
23666eba86fSAxel Dörfler 				width = fRightBottomReplicant.right + 6
237573f748cSJohn Scipione 					+ fTime->Frame().Width();
23841281cf3SAxel Dörfler 			} else
2397da06231SAxel Dörfler 				width = fRightBottomReplicant.right + 3;
24041281cf3SAxel Dörfler 		}
2411ad8c760SFredrik Holmqvist 
24241281cf3SAxel Dörfler 		// this view has a fixed minimum width
243cb6afcb1SStephan Aßmus 		width = max(fMinimumTrayWidth, width);
2442ce9bab8SJohn Scipione 		height = kGutter + static_cast<TBarApp*>(be_app)->IconSize() + kGutter;
24541281cf3SAxel Dörfler 	}
24641281cf3SAxel Dörfler 
24741281cf3SAxel Dörfler 	*preferredWidth = width;
2481ad8c760SFredrik Holmqvist 	// add 1 for the border
24941281cf3SAxel Dörfler 	*preferredHeight = height + 1;
25041281cf3SAxel Dörfler }
25141281cf3SAxel Dörfler 
25241281cf3SAxel Dörfler 
25341281cf3SAxel Dörfler void
25441281cf3SAxel Dörfler TReplicantTray::AdjustPlacement()
25541281cf3SAxel Dörfler {
25641281cf3SAxel Dörfler 	// called when an add-on has been added or removed
25741281cf3SAxel Dörfler 	// need to resize the parent of this accordingly
258a5210ab1SAxel Dörfler 
25941281cf3SAxel Dörfler 	BRect bounds = Bounds();
26041281cf3SAxel Dörfler 	float width, height;
26141281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
26241281cf3SAxel Dörfler 
26341281cf3SAxel Dörfler 	if (width == bounds.Width() && height == bounds.Height()) {
26441281cf3SAxel Dörfler 		// no need to change anything
26541281cf3SAxel Dörfler 		return;
26641281cf3SAxel Dörfler 	}
26741281cf3SAxel Dörfler 
26841281cf3SAxel Dörfler 	Parent()->ResizeToPreferred();
26941281cf3SAxel Dörfler 	fBarView->UpdatePlacement();
27041281cf3SAxel Dörfler 	Parent()->Invalidate();
27141281cf3SAxel Dörfler 	Invalidate();
27241281cf3SAxel Dörfler }
27341281cf3SAxel Dörfler 
27441281cf3SAxel Dörfler 
27541281cf3SAxel Dörfler void
27641281cf3SAxel Dörfler TReplicantTray::MessageReceived(BMessage* message)
27741281cf3SAxel Dörfler {
27841281cf3SAxel Dörfler 	switch (message->what) {
27930a55802SJohn Scipione 		case B_LOCALE_CHANGED:
280573f748cSJohn Scipione 		{
281573f748cSJohn Scipione 			if (fTime == NULL)
282573f748cSJohn Scipione 				return;
283573f748cSJohn Scipione 
28430a55802SJohn Scipione 			// Locale may have updated 12/24 hour clock
285573f748cSJohn Scipione 			BFormattingConventions conventions;
286573f748cSJohn Scipione 			BLocale::Default()->GetFormattingConventions(&conventions);
28730a55802SJohn Scipione 			fTime->SetUse24HourClock(conventions.Use24HourClock());
28830a55802SJohn Scipione 
28941281cf3SAxel Dörfler 			// time string reformat -> realign
29041281cf3SAxel Dörfler 			RealignReplicants();
29141281cf3SAxel Dörfler 			AdjustPlacement();
29241281cf3SAxel Dörfler 			break;
293573f748cSJohn Scipione 		}
294573f748cSJohn Scipione 
29530a55802SJohn Scipione 		case kShowHideTime:
29630a55802SJohn Scipione 			// from context menu in clock and in this view
29730a55802SJohn Scipione 			ShowHideTime();
29830a55802SJohn Scipione 			break;
29930a55802SJohn Scipione 
3003cf2d117SJohn Scipione 		case kShowSeconds:
301573f748cSJohn Scipione 			if (fTime == NULL)
302573f748cSJohn Scipione 				return;
303573f748cSJohn Scipione 
3043cf2d117SJohn Scipione 			fTime->SetShowSeconds(!fTime->ShowSeconds());
305573f748cSJohn Scipione 
3063cf2d117SJohn Scipione 			// time string reformat -> realign
3073cf2d117SJohn Scipione 			RealignReplicants();
3083cf2d117SJohn Scipione 			AdjustPlacement();
309573f748cSJohn Scipione 			break;
3103cf2d117SJohn Scipione 
3113cf2d117SJohn Scipione 		case kShowDayOfWeek:
3123cf2d117SJohn Scipione 			if (fTime == NULL)
3133cf2d117SJohn Scipione 				return;
3143cf2d117SJohn Scipione 
3153cf2d117SJohn Scipione 			fTime->SetShowDayOfWeek(!fTime->ShowDayOfWeek());
3163cf2d117SJohn Scipione 
3173cf2d117SJohn Scipione 			// time string reformat -> realign
3183cf2d117SJohn Scipione 			RealignReplicants();
3193cf2d117SJohn Scipione 			AdjustPlacement();
3203cf2d117SJohn Scipione 			break;
3213cf2d117SJohn Scipione 
32241281cf3SAxel Dörfler #ifdef DB_ADDONS
32341281cf3SAxel Dörfler 		case B_NODE_MONITOR:
32441281cf3SAxel Dörfler 			HandleEntryUpdate(message);
32541281cf3SAxel Dörfler 			break;
32641281cf3SAxel Dörfler #endif
32741281cf3SAxel Dörfler 
32841281cf3SAxel Dörfler 		default:
32941281cf3SAxel Dörfler 			BView::MessageReceived(message);
33041281cf3SAxel Dörfler 			break;
33141281cf3SAxel Dörfler 	}
33241281cf3SAxel Dörfler }
33341281cf3SAxel Dörfler 
33441281cf3SAxel Dörfler 
33541281cf3SAxel Dörfler void
33641281cf3SAxel Dörfler TReplicantTray::MouseDown(BPoint where)
33741281cf3SAxel Dörfler {
33841281cf3SAxel Dörfler #ifdef DB_ADDONS
33941281cf3SAxel Dörfler 	if (modifiers() & B_CONTROL_KEY)
34041281cf3SAxel Dörfler 		DumpList(fItemList);
34141281cf3SAxel Dörfler #endif
34241281cf3SAxel Dörfler 
34341281cf3SAxel Dörfler 	uint32 buttons;
34441281cf3SAxel Dörfler 
34541281cf3SAxel Dörfler 	Window()->CurrentMessage()->FindInt32("buttons", (int32*)&buttons);
34641281cf3SAxel Dörfler 	if (buttons == B_SECONDARY_MOUSE_BUTTON) {
34741281cf3SAxel Dörfler 		ShowReplicantMenu(where);
34841281cf3SAxel Dörfler 	} else {
34941281cf3SAxel Dörfler 		BPoint save = where;
35041281cf3SAxel Dörfler 		bigtime_t doubleClickSpeed;
35141281cf3SAxel Dörfler 		bigtime_t start = system_time();
35241281cf3SAxel Dörfler 		uint32 buttons;
35341281cf3SAxel Dörfler 
35441281cf3SAxel Dörfler 		get_click_speed(&doubleClickSpeed);
35541281cf3SAxel Dörfler 
35641281cf3SAxel Dörfler 		do {
35741281cf3SAxel Dörfler 			if (fabs(where.x - save.x) > 4 || fabs(where.y - save.y) > 4)
35841281cf3SAxel Dörfler 				// user moved out of bounds of click area
35941281cf3SAxel Dörfler 				break;
36041281cf3SAxel Dörfler 
36141281cf3SAxel Dörfler 			if ((system_time() - start) > (2 * doubleClickSpeed)) {
36241281cf3SAxel Dörfler 				ShowReplicantMenu(where);
36341281cf3SAxel Dörfler 				break;
36441281cf3SAxel Dörfler 			}
36541281cf3SAxel Dörfler 
36641281cf3SAxel Dörfler 			snooze(50000);
36741281cf3SAxel Dörfler 			GetMouse(&where, &buttons);
36841281cf3SAxel Dörfler 		} while (buttons);
36941281cf3SAxel Dörfler 	}
37041281cf3SAxel Dörfler 	BView::MouseDown(where);
37141281cf3SAxel Dörfler }
37241281cf3SAxel Dörfler 
3733cf2d117SJohn Scipione 
3743cf2d117SJohn Scipione void
3753cf2d117SJohn Scipione TReplicantTray::ShowReplicantMenu(BPoint point)
3763cf2d117SJohn Scipione {
3773cf2d117SJohn Scipione 	BPopUpMenu* menu = new BPopUpMenu("", false, false);
3783cf2d117SJohn Scipione 	menu->SetFont(be_plain_font);
3793cf2d117SJohn Scipione 
3803cf2d117SJohn Scipione 	// If clock is visible show the extended menu, otherwise show "Show time"
3813cf2d117SJohn Scipione 
3823cf2d117SJohn Scipione 	if (!fTime->IsHidden())
3833cf2d117SJohn Scipione 		fTime->ShowTimeOptions(ConvertToScreen(point));
3843cf2d117SJohn Scipione 	else {
3853cf2d117SJohn Scipione 		BMenuItem* item = new BMenuItem(B_TRANSLATE("Show time"),
3863cf2d117SJohn Scipione 			new BMessage(kShowHideTime));
3873cf2d117SJohn Scipione 		menu->AddItem(item);
3883cf2d117SJohn Scipione 		menu->SetTargetForItems(this);
3893cf2d117SJohn Scipione 		BPoint where = ConvertToScreen(point);
3903cf2d117SJohn Scipione 		menu->Go(where, true, true, BRect(where - BPoint(4, 4),
3913cf2d117SJohn Scipione 			where + BPoint(4, 4)), true);
3923cf2d117SJohn Scipione 	}
3933cf2d117SJohn Scipione }
3943cf2d117SJohn Scipione 
3953cf2d117SJohn Scipione 
3963cf2d117SJohn Scipione void
3973cf2d117SJohn Scipione TReplicantTray::SetMultiRow(bool state)
3983cf2d117SJohn Scipione {
3993cf2d117SJohn Scipione 	fMultiRowMode = state;
4003cf2d117SJohn Scipione }
4013cf2d117SJohn Scipione 
4023cf2d117SJohn Scipione 
4033cf2d117SJohn Scipione void
4043cf2d117SJohn Scipione TReplicantTray::ShowHideTime()
4053cf2d117SJohn Scipione {
4063cf2d117SJohn Scipione 	if (fTime == NULL)
4073cf2d117SJohn Scipione 		return;
4083cf2d117SJohn Scipione 
4093cf2d117SJohn Scipione 	if (fTime->IsHidden())
4103cf2d117SJohn Scipione 		fTime->Show();
4113cf2d117SJohn Scipione 	else
4123cf2d117SJohn Scipione 		fTime->Hide();
4133cf2d117SJohn Scipione 
4143cf2d117SJohn Scipione 	RealignReplicants();
4153cf2d117SJohn Scipione 	AdjustPlacement();
4163cf2d117SJohn Scipione }
4173cf2d117SJohn Scipione 
4183cf2d117SJohn Scipione 
41941281cf3SAxel Dörfler #ifdef DB_ADDONS
42041281cf3SAxel Dörfler 
4213cf2d117SJohn Scipione 
42241281cf3SAxel Dörfler void
42341281cf3SAxel Dörfler TReplicantTray::InitAddOnSupport()
42441281cf3SAxel Dörfler {
42541281cf3SAxel Dörfler 	// list to maintain refs to each rep added/deleted
42641281cf3SAxel Dörfler 	fItemList = new BList();
42741281cf3SAxel Dörfler 	BPath path;
4281ad8c760SFredrik Holmqvist 
42941281cf3SAxel Dörfler 	if (find_directory(B_USER_SETTINGS_DIRECTORY, &path, true) == B_OK) {
4303e6ff860SRene Gollent 		path.Append(kReplicantSettingsFile);
43141281cf3SAxel Dörfler 
43241281cf3SAxel Dörfler 		BFile file(path.Path(), B_READ_ONLY);
4333e6ff860SRene Gollent 		if (file.InitCheck() == B_OK) {
4343e6ff860SRene Gollent 			status_t result;
4353e6ff860SRene Gollent 			BEntry entry;
4363e6ff860SRene Gollent 			int32 id;
437c0ad1c91SRene Gollent 			BString path;
4383e6ff860SRene Gollent 			if (fAddOnSettings.Unflatten(&file) == B_OK) {
439c0ad1c91SRene Gollent 				for (int32 i = 0; fAddOnSettings.FindString(kReplicantPathField,
440c0ad1c91SRene Gollent 					i, &path) == B_OK; i++) {
441c0ad1c91SRene Gollent 					if (entry.SetTo(path.String()) == B_OK && entry.Exists()) {
4423e6ff860SRene Gollent 						result = LoadAddOn(&entry, &id, false);
4433e6ff860SRene Gollent 					} else
4443e6ff860SRene Gollent 						result = B_ENTRY_NOT_FOUND;
44541281cf3SAxel Dörfler 
4463e6ff860SRene Gollent 					if (result != B_OK) {
447c0ad1c91SRene Gollent 						fAddOnSettings.RemoveData(kReplicantPathField, i);
4483e6ff860SRene Gollent 						--i;
44941281cf3SAxel Dörfler 					}
45041281cf3SAxel Dörfler 				}
45141281cf3SAxel Dörfler 			}
4523e6ff860SRene Gollent 		}
4533e6ff860SRene Gollent 	}
45441281cf3SAxel Dörfler }
45541281cf3SAxel Dörfler 
45641281cf3SAxel Dörfler 
45741281cf3SAxel Dörfler void
45841281cf3SAxel Dörfler TReplicantTray::DeleteAddOnSupport()
45941281cf3SAxel Dörfler {
4607625ce51SRene Gollent 	_SaveSettings();
4613e6ff860SRene Gollent 
46241281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
46341281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->RemoveItem(i);
46441281cf3SAxel Dörfler 		if (item) {
46541281cf3SAxel Dörfler 			if (item->isAddOn)
46641281cf3SAxel Dörfler 				watch_node(&(item->nodeRef), B_STOP_WATCHING, this, Window());
46741281cf3SAxel Dörfler 
46841281cf3SAxel Dörfler 			delete item;
46941281cf3SAxel Dörfler 		}
47041281cf3SAxel Dörfler 	}
47141281cf3SAxel Dörfler 	delete fItemList;
47241281cf3SAxel Dörfler 
47341281cf3SAxel Dörfler 	// stop the volume mount/unmount watch
47441281cf3SAxel Dörfler 	stop_watching(this, Window());
47541281cf3SAxel Dörfler }
47641281cf3SAxel Dörfler 
47741281cf3SAxel Dörfler 
47841281cf3SAxel Dörfler DeskbarItemInfo*
47941281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(node_ref& nodeRef)
48041281cf3SAxel Dörfler {
48141281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
48266eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
48341281cf3SAxel Dörfler 		if (item == NULL)
48441281cf3SAxel Dörfler 			continue;
48541281cf3SAxel Dörfler 
48641281cf3SAxel Dörfler 		if (item->nodeRef == nodeRef)
48741281cf3SAxel Dörfler 			return item;
48841281cf3SAxel Dörfler 	}
48941281cf3SAxel Dörfler 
49041281cf3SAxel Dörfler 	return NULL;
49141281cf3SAxel Dörfler }
49241281cf3SAxel Dörfler 
49341281cf3SAxel Dörfler 
49441281cf3SAxel Dörfler DeskbarItemInfo*
49541281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(int32 id)
49641281cf3SAxel Dörfler {
49741281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
49866eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
49941281cf3SAxel Dörfler 		if (item == NULL)
50041281cf3SAxel Dörfler 			continue;
50141281cf3SAxel Dörfler 
50241281cf3SAxel Dörfler 		if (item->id == id)
50341281cf3SAxel Dörfler 			return item;
50441281cf3SAxel Dörfler 	}
50541281cf3SAxel Dörfler 
50641281cf3SAxel Dörfler 	return NULL;
50741281cf3SAxel Dörfler }
50841281cf3SAxel Dörfler 
50941281cf3SAxel Dörfler 
51041281cf3SAxel Dörfler bool
51141281cf3SAxel Dörfler TReplicantTray::NodeExists(node_ref& nodeRef)
51241281cf3SAxel Dörfler {
51341281cf3SAxel Dörfler 	return DeskbarItemFor(nodeRef) != NULL;
51441281cf3SAxel Dörfler }
51541281cf3SAxel Dörfler 
51641281cf3SAxel Dörfler 
51766eba86fSAxel Dörfler /*! This handles B_NODE_MONITOR & B_QUERY_UPDATE messages received
51866eba86fSAxel Dörfler 	for the registered add-ons.
51941281cf3SAxel Dörfler */
52041281cf3SAxel Dörfler void
52141281cf3SAxel Dörfler TReplicantTray::HandleEntryUpdate(BMessage* message)
52241281cf3SAxel Dörfler {
52341281cf3SAxel Dörfler 	int32 opcode;
52441281cf3SAxel Dörfler 	if (message->FindInt32("opcode", &opcode) != B_OK)
52541281cf3SAxel Dörfler 		return;
52641281cf3SAxel Dörfler 
52741281cf3SAxel Dörfler 	BPath path;
52841281cf3SAxel Dörfler 	switch (opcode) {
52941281cf3SAxel Dörfler 		case B_ENTRY_MOVED:
53041281cf3SAxel Dörfler 		{
53141281cf3SAxel Dörfler 			entry_ref ref;
53241281cf3SAxel Dörfler 			ino_t todirectory;
53341281cf3SAxel Dörfler 			ino_t node;
53441281cf3SAxel Dörfler 			const char* name;
53541281cf3SAxel Dörfler 			if (message->FindString("name", &name) == B_OK
53671bd3ba5SJonas Sundström 				&& message->FindInt64("from directory", &(ref.directory))
53771bd3ba5SJonas Sundström 				== B_OK
53841281cf3SAxel Dörfler 				&& message->FindInt64("to directory", &todirectory) == B_OK
53941281cf3SAxel Dörfler 				&& message->FindInt32("device", &(ref.device)) == B_OK
54041281cf3SAxel Dörfler 				&& message->FindInt64("node", &node) == B_OK ) {
54141281cf3SAxel Dörfler 
54241281cf3SAxel Dörfler 				if (!name)
54341281cf3SAxel Dörfler 					break;
54441281cf3SAxel Dörfler 
54541281cf3SAxel Dörfler 				ref.set_name(name);
54641281cf3SAxel Dörfler 				// change the directory reference to
54741281cf3SAxel Dörfler 				// the new directory
54841281cf3SAxel Dörfler 				MoveItem(&ref, todirectory);
54941281cf3SAxel Dörfler 			}
55041281cf3SAxel Dörfler 			break;
55166eba86fSAxel Dörfler 		}
55241281cf3SAxel Dörfler 
55341281cf3SAxel Dörfler 		case B_ENTRY_REMOVED:
55441281cf3SAxel Dörfler 		{
55541281cf3SAxel Dörfler 			// entry was rm'd from the device
55641281cf3SAxel Dörfler 			node_ref nodeRef;
55741281cf3SAxel Dörfler 			if (message->FindInt32("device", &(nodeRef.device)) == B_OK
55841281cf3SAxel Dörfler 				&& message->FindInt64("node", &(nodeRef.node)) == B_OK) {
55941281cf3SAxel Dörfler 				DeskbarItemInfo* item = DeskbarItemFor(nodeRef);
56041281cf3SAxel Dörfler 				if (item == NULL)
56141281cf3SAxel Dörfler 					break;
56241281cf3SAxel Dörfler 
56341281cf3SAxel Dörfler 				// If there is a team running where the add-on comes from,
56441281cf3SAxel Dörfler 				// we don't want to remove the icon yet.
56541281cf3SAxel Dörfler 				if (be_roster->IsRunning(&item->entryRef))
56641281cf3SAxel Dörfler 					break;
56741281cf3SAxel Dörfler 
56841281cf3SAxel Dörfler 				UnloadAddOn(&nodeRef, NULL, true, false);
56941281cf3SAxel Dörfler 			}
57041281cf3SAxel Dörfler 			break;
57166eba86fSAxel Dörfler 		}
57241281cf3SAxel Dörfler 	}
57366eba86fSAxel Dörfler }
57441281cf3SAxel Dörfler 
57541281cf3SAxel Dörfler 
5761ad8c760SFredrik Holmqvist /*! The add-ons must support the exported C function API
57766eba86fSAxel Dörfler 	if they do, they will be loaded and added to deskbar
57866eba86fSAxel Dörfler 	primary function is the Instantiate function
5797da06231SAxel Dörfler */
58041281cf3SAxel Dörfler status_t
5813e6ff860SRene Gollent TReplicantTray::LoadAddOn(BEntry* entry, int32* id, bool addToSettings)
58241281cf3SAxel Dörfler {
58341281cf3SAxel Dörfler 	if (!entry)
58441281cf3SAxel Dörfler 		return B_ERROR;
58541281cf3SAxel Dörfler 
58641281cf3SAxel Dörfler 	node_ref nodeRef;
58741281cf3SAxel Dörfler 	entry->GetNodeRef(&nodeRef);
58841281cf3SAxel Dörfler 	// no duplicates
58941281cf3SAxel Dörfler 	if (NodeExists(nodeRef))
59041281cf3SAxel Dörfler 		return B_ERROR;
59141281cf3SAxel Dörfler 
59241281cf3SAxel Dörfler 	BNode node(entry);
59341281cf3SAxel Dörfler 	BPath path;
59466eba86fSAxel Dörfler 	status_t status = entry->GetPath(&path);
59566eba86fSAxel Dörfler 	if (status < B_OK)
59666eba86fSAxel Dörfler 		return status;
59741281cf3SAxel Dörfler 
59841281cf3SAxel Dörfler 	// load the add-on
59941281cf3SAxel Dörfler 	image_id image = load_add_on(path.Path());
60066eba86fSAxel Dörfler 	if (image < B_OK)
60166eba86fSAxel Dörfler 		return image;
60241281cf3SAxel Dörfler 
60341281cf3SAxel Dörfler 	// get the view loading function symbol
60441281cf3SAxel Dörfler 	//    we first look for a symbol that takes an image_id
60541281cf3SAxel Dörfler 	//    and entry_ref pointer, if not found, go with normal
60641281cf3SAxel Dörfler 	//    instantiate function
60741281cf3SAxel Dörfler 	BView* (*entryFunction)(image_id, const entry_ref*);
60841281cf3SAxel Dörfler 	BView* (*itemFunction)(void);
60941281cf3SAxel Dörfler 	BView* view = NULL;
61041281cf3SAxel Dörfler 
61141281cf3SAxel Dörfler 	entry_ref ref;
61241281cf3SAxel Dörfler 	entry->GetRef(&ref);
61341281cf3SAxel Dörfler 
61441281cf3SAxel Dörfler 	if (get_image_symbol(image, kInstantiateEntryCFunctionName,
61566eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&entryFunction) >= B_OK) {
61641281cf3SAxel Dörfler 		view = (*entryFunction)(image, &ref);
61741281cf3SAxel Dörfler 	} else if (get_image_symbol(image, kInstantiateItemCFunctionName,
61866eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&itemFunction) >= B_OK) {
61941281cf3SAxel Dörfler 		view = (*itemFunction)();
62041281cf3SAxel Dörfler 	} else {
62141281cf3SAxel Dörfler 		unload_add_on(image);
62241281cf3SAxel Dörfler 		return B_ERROR;
62341281cf3SAxel Dörfler 	}
62441281cf3SAxel Dörfler 
62566eba86fSAxel Dörfler 	if (view == NULL || IconExists(view->Name())) {
62641281cf3SAxel Dörfler 		delete view;
62741281cf3SAxel Dörfler 		unload_add_on(image);
62841281cf3SAxel Dörfler 		return B_ERROR;
62941281cf3SAxel Dörfler 	}
63041281cf3SAxel Dörfler 
63141281cf3SAxel Dörfler 	BMessage* data = new BMessage;
63241281cf3SAxel Dörfler 	view->Archive(data);
63341281cf3SAxel Dörfler 	delete view;
63441281cf3SAxel Dörfler 
63541281cf3SAxel Dörfler 	AddIcon(data, id, &ref);
63641281cf3SAxel Dörfler 		// add the rep; adds info to list
63741281cf3SAxel Dörfler 
6383e6ff860SRene Gollent 	if (addToSettings) {
639c0ad1c91SRene Gollent 		fAddOnSettings.AddString(kReplicantPathField, path.Path());
6407625ce51SRene Gollent 		_SaveSettings();
6413e6ff860SRene Gollent 	}
64241281cf3SAxel Dörfler 
64341281cf3SAxel Dörfler 	return B_OK;
64441281cf3SAxel Dörfler }
64541281cf3SAxel Dörfler 
64641281cf3SAxel Dörfler 
64741281cf3SAxel Dörfler status_t
64841281cf3SAxel Dörfler TReplicantTray::AddItem(int32 id, node_ref nodeRef, BEntry& entry, bool isAddOn)
64941281cf3SAxel Dörfler {
65041281cf3SAxel Dörfler 	DeskbarItemInfo* item = new DeskbarItemInfo;
65141281cf3SAxel Dörfler 	if (item == NULL)
65241281cf3SAxel Dörfler 		return B_NO_MEMORY;
65341281cf3SAxel Dörfler 
65441281cf3SAxel Dörfler 	item->id = id;
65541281cf3SAxel Dörfler 	item->isAddOn = isAddOn;
65641281cf3SAxel Dörfler 
65741281cf3SAxel Dörfler 	if (entry.GetRef(&item->entryRef) < B_OK) {
65841281cf3SAxel Dörfler 		item->entryRef.device = -1;
65941281cf3SAxel Dörfler 		item->entryRef.directory = -1;
66041281cf3SAxel Dörfler 		item->entryRef.name = NULL;
66141281cf3SAxel Dörfler 	}
66241281cf3SAxel Dörfler 	item->nodeRef = nodeRef;
66341281cf3SAxel Dörfler 
66441281cf3SAxel Dörfler 	fItemList->AddItem(item);
66541281cf3SAxel Dörfler 
66641281cf3SAxel Dörfler 	if (isAddOn)
66741281cf3SAxel Dörfler 		watch_node(&nodeRef, B_WATCH_NAME | B_WATCH_ATTR, this, Window());
66841281cf3SAxel Dörfler 
66941281cf3SAxel Dörfler 	return B_OK;
67041281cf3SAxel Dörfler }
67141281cf3SAxel Dörfler 
67241281cf3SAxel Dörfler 
67341281cf3SAxel Dörfler /**	from entry_removed message, when attribute removed
67441281cf3SAxel Dörfler  *	or when a device is unmounted (use removeall, by device)
67541281cf3SAxel Dörfler  */
67641281cf3SAxel Dörfler 
67741281cf3SAxel Dörfler void
67841281cf3SAxel Dörfler TReplicantTray::UnloadAddOn(node_ref* nodeRef, dev_t* device,
67941281cf3SAxel Dörfler 	bool which, bool removeAll)
68041281cf3SAxel Dörfler {
68141281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
68241281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
68341281cf3SAxel Dörfler 		if (!item)
68441281cf3SAxel Dörfler 			continue;
68541281cf3SAxel Dörfler 
68641281cf3SAxel Dörfler 		if ((which && nodeRef && item->nodeRef == *nodeRef)
68741281cf3SAxel Dörfler 			|| (device && item->nodeRef.device == *device)) {
68841281cf3SAxel Dörfler 
68941281cf3SAxel Dörfler 			if (device && be_roster->IsRunning(&item->entryRef))
69041281cf3SAxel Dörfler 				continue;
69141281cf3SAxel Dörfler 
69241281cf3SAxel Dörfler 			RemoveIcon(item->id);
69341281cf3SAxel Dörfler 
69441281cf3SAxel Dörfler 			if (!removeAll)
69541281cf3SAxel Dörfler 				break;
69641281cf3SAxel Dörfler 		}
69741281cf3SAxel Dörfler 	}
69841281cf3SAxel Dörfler }
69941281cf3SAxel Dörfler 
70041281cf3SAxel Dörfler 
70141281cf3SAxel Dörfler void
70241281cf3SAxel Dörfler TReplicantTray::RemoveItem(int32 id)
70341281cf3SAxel Dörfler {
70441281cf3SAxel Dörfler 	DeskbarItemInfo* item = DeskbarItemFor(id);
70541281cf3SAxel Dörfler 	if (item == NULL)
70641281cf3SAxel Dörfler 		return;
70741281cf3SAxel Dörfler 
70841281cf3SAxel Dörfler 	// attribute was added via Deskbar API (AddItem(entry_ref*, int32*)
70941281cf3SAxel Dörfler 	if (item->isAddOn) {
7107625ce51SRene Gollent 		BPath path(&item->entryRef);
7117625ce51SRene Gollent 		BString storedPath;
7127625ce51SRene Gollent 		for (int32 i = 0;
7138aab28f1SRene Gollent 			fAddOnSettings.FindString(kReplicantPathField, i, &storedPath)
7147625ce51SRene Gollent 				== B_OK; i++) {
7157625ce51SRene Gollent 			if (storedPath == path.Path()) {
7168aab28f1SRene Gollent 				fAddOnSettings.RemoveData(kReplicantPathField, i);
7177625ce51SRene Gollent 				break;
7187625ce51SRene Gollent 			}
7197625ce51SRene Gollent 		}
7207625ce51SRene Gollent 		_SaveSettings();
7217625ce51SRene Gollent 
72241281cf3SAxel Dörfler 		BNode node(&item->entryRef);
72341281cf3SAxel Dörfler 		watch_node(&item->nodeRef, B_STOP_WATCHING, this, Window());
72441281cf3SAxel Dörfler 	}
72541281cf3SAxel Dörfler 
72641281cf3SAxel Dörfler 	fItemList->RemoveItem(item);
72741281cf3SAxel Dörfler 	delete item;
72841281cf3SAxel Dörfler }
72941281cf3SAxel Dörfler 
73041281cf3SAxel Dörfler 
73141281cf3SAxel Dörfler /**	ENTRY_MOVED message, moving only occurs on a device
73241281cf3SAxel Dörfler  *	copying will occur (ENTRY_CREATED) between devices
73341281cf3SAxel Dörfler  */
73441281cf3SAxel Dörfler 
73541281cf3SAxel Dörfler void
73641281cf3SAxel Dörfler TReplicantTray::MoveItem(entry_ref* ref, ino_t toDirectory)
73741281cf3SAxel Dörfler {
73841281cf3SAxel Dörfler 	if (!ref)
73941281cf3SAxel Dörfler 		return;
74041281cf3SAxel Dörfler 
74141281cf3SAxel Dörfler 	// scan for a matching entry_ref and update it
74241281cf3SAxel Dörfler 	//
74341281cf3SAxel Dörfler 	// don't need to change node info as it does not change
74441281cf3SAxel Dörfler 
74541281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
74641281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
74741281cf3SAxel Dörfler 		if (!item)
74841281cf3SAxel Dörfler 			continue;
74941281cf3SAxel Dörfler 
75041281cf3SAxel Dörfler 		if (!strcmp(item->entryRef.name, ref->name)
75141281cf3SAxel Dörfler 			&& item->entryRef.device == ref->device
75241281cf3SAxel Dörfler 			&& item->entryRef.directory == ref->directory) {
75341281cf3SAxel Dörfler 			item->entryRef.directory = toDirectory;
75441281cf3SAxel Dörfler 			break;
75541281cf3SAxel Dörfler 		}
75641281cf3SAxel Dörfler 	}
75741281cf3SAxel Dörfler }
75841281cf3SAxel Dörfler 
75941281cf3SAxel Dörfler #endif // add-on support
76041281cf3SAxel Dörfler 
76141281cf3SAxel Dörfler //	external add-on API routines
76241281cf3SAxel Dörfler //	called using the new BDeskbar class
76341281cf3SAxel Dörfler 
76441281cf3SAxel Dörfler //	existence of icon/replicant by name or ID
76541281cf3SAxel Dörfler //	returns opposite
76641281cf3SAxel Dörfler //	note: name and id are semi-private limiting
76741281cf3SAxel Dörfler //		the ability of non-host apps to remove
76841281cf3SAxel Dörfler //		icons without a little bit of work
76941281cf3SAxel Dörfler 
77041281cf3SAxel Dörfler /**	for a specific id
77141281cf3SAxel Dörfler  *	return the name of the replicant (name of view)
77241281cf3SAxel Dörfler  */
77341281cf3SAxel Dörfler 
77441281cf3SAxel Dörfler status_t
77541281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 id, const char** name)
77641281cf3SAxel Dörfler {
77741281cf3SAxel Dörfler 	if (id < 0)
77841281cf3SAxel Dörfler 		return B_ERROR;
77941281cf3SAxel Dörfler 
78041281cf3SAxel Dörfler 	int32 index, temp;
78141281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &temp, id, false);
78241281cf3SAxel Dörfler 	if (view) {
78341281cf3SAxel Dörfler 		*name = view->Name();
78441281cf3SAxel Dörfler 		return B_OK;
78541281cf3SAxel Dörfler 	}
78641281cf3SAxel Dörfler 
78741281cf3SAxel Dörfler 	return B_ERROR;
78841281cf3SAxel Dörfler }
78941281cf3SAxel Dörfler 
79041281cf3SAxel Dörfler 
7917da06231SAxel Dörfler /**	for a specific name
7927da06231SAxel Dörfler  *	return the id (internal to Deskbar)
7937da06231SAxel Dörfler  */
7947da06231SAxel Dörfler 
79541281cf3SAxel Dörfler status_t
79641281cf3SAxel Dörfler TReplicantTray::ItemInfo(const char* name, int32* id)
79741281cf3SAxel Dörfler {
79841281cf3SAxel Dörfler 	if (!name || strlen(name) <= 0)
79941281cf3SAxel Dörfler 		return B_ERROR;
80041281cf3SAxel Dörfler 
80141281cf3SAxel Dörfler 	int32 index;
80241281cf3SAxel Dörfler 	BView* view = ViewAt(&index, id, name);
80341281cf3SAxel Dörfler 	if (view)
80441281cf3SAxel Dörfler 		return B_OK;
80541281cf3SAxel Dörfler 
80641281cf3SAxel Dörfler 	return B_ERROR;
80741281cf3SAxel Dörfler }
80841281cf3SAxel Dörfler 
80941281cf3SAxel Dörfler 
8107da06231SAxel Dörfler /**	at a specific index
8117da06231SAxel Dörfler  *	return both the name and the id of the replicant
8127da06231SAxel Dörfler  */
8137da06231SAxel Dörfler 
81441281cf3SAxel Dörfler status_t
81541281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 index, const char** name, int32* id)
81641281cf3SAxel Dörfler {
81741281cf3SAxel Dörfler 	if (index < 0)
81841281cf3SAxel Dörfler 		return B_ERROR;
81941281cf3SAxel Dörfler 
82041281cf3SAxel Dörfler 	BView* view;
82141281cf3SAxel Dörfler 	fShelf->ReplicantAt(index, &view, (uint32*)id, NULL);
82241281cf3SAxel Dörfler 	if (view) {
82341281cf3SAxel Dörfler 		*name = view->Name();
82441281cf3SAxel Dörfler 		return B_OK;
82541281cf3SAxel Dörfler 	}
82641281cf3SAxel Dörfler 
82741281cf3SAxel Dörfler 	return B_ERROR;
82841281cf3SAxel Dörfler }
82941281cf3SAxel Dörfler 
83041281cf3SAxel Dörfler 
8317da06231SAxel Dörfler /**	replicant exists, by id/index */
8327da06231SAxel Dörfler 
83341281cf3SAxel Dörfler bool
83441281cf3SAxel Dörfler TReplicantTray::IconExists(int32 target, bool byIndex)
83541281cf3SAxel Dörfler {
83641281cf3SAxel Dörfler 	int32 index, id;
83741281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
83841281cf3SAxel Dörfler 
83941281cf3SAxel Dörfler 	return view && index >= 0;
84041281cf3SAxel Dörfler }
84141281cf3SAxel Dörfler 
84241281cf3SAxel Dörfler 
8437da06231SAxel Dörfler /**	replicant exists, by name */
8447da06231SAxel Dörfler 
84541281cf3SAxel Dörfler bool
84641281cf3SAxel Dörfler TReplicantTray::IconExists(const char* name)
84741281cf3SAxel Dörfler {
84841281cf3SAxel Dörfler 	if (!name || strlen(name) == 0)
84941281cf3SAxel Dörfler 		return false;
85041281cf3SAxel Dörfler 
85141281cf3SAxel Dörfler 	int32 index, id;
85241281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
85341281cf3SAxel Dörfler 
85441281cf3SAxel Dörfler 	return view && index >= 0;
85541281cf3SAxel Dörfler }
85641281cf3SAxel Dörfler 
85741281cf3SAxel Dörfler 
85841281cf3SAxel Dörfler int32
85941281cf3SAxel Dörfler TReplicantTray::IconCount() const
86041281cf3SAxel Dörfler {
86141281cf3SAxel Dörfler 	return fShelf->CountReplicants();
86241281cf3SAxel Dörfler }
86341281cf3SAxel Dörfler 
86441281cf3SAxel Dörfler 
865ced3755cSAxel Dörfler /*! Message must contain an archivable view for later rehydration.
866ced3755cSAxel Dörfler 	This function takes over ownership of the provided message on success
867ced3755cSAxel Dörfler 	only.
868ced3755cSAxel Dörfler 	Returns the current replicant ID.
8697da06231SAxel Dörfler */
87041281cf3SAxel Dörfler status_t
871ced3755cSAxel Dörfler TReplicantTray::AddIcon(BMessage* archive, int32* id, const entry_ref* addOn)
87241281cf3SAxel Dörfler {
873ced3755cSAxel Dörfler 	if (archive == NULL || id == NULL)
87441281cf3SAxel Dörfler 		return B_ERROR;
87541281cf3SAxel Dörfler 
8769c50f36eSAxel Dörfler 	// find entry_ref
8779c50f36eSAxel Dörfler 
8789c50f36eSAxel Dörfler 	entry_ref ref;
8799c50f36eSAxel Dörfler 	if (addOn) {
8809c50f36eSAxel Dörfler 		// Use it if we got it
8819c50f36eSAxel Dörfler 		ref = *addOn;
8829c50f36eSAxel Dörfler 	} else {
8839c50f36eSAxel Dörfler 		const char* signature;
8841ad8c760SFredrik Holmqvist 
885ced3755cSAxel Dörfler 		status_t status = archive->FindString("add_on", &signature);
8869c50f36eSAxel Dörfler 		if (status == B_OK) {
8879c50f36eSAxel Dörfler 			BRoster roster;
8889c50f36eSAxel Dörfler 			status = roster.FindApp(signature, &ref);
8899c50f36eSAxel Dörfler 		}
8909c50f36eSAxel Dörfler 		if (status < B_OK)
8919c50f36eSAxel Dörfler 			return status;
8929c50f36eSAxel Dörfler 	}
8939c50f36eSAxel Dörfler 
8949c50f36eSAxel Dörfler 	BFile file;
8959c50f36eSAxel Dörfler 	status_t status = file.SetTo(&ref, B_READ_ONLY);
8969c50f36eSAxel Dörfler 	if (status < B_OK)
8979c50f36eSAxel Dörfler 		return status;
8989c50f36eSAxel Dörfler 
8999c50f36eSAxel Dörfler 	node_ref nodeRef;
9009c50f36eSAxel Dörfler 	status = file.GetNodeRef(&nodeRef);
9019c50f36eSAxel Dörfler 	if (status < B_OK)
9029c50f36eSAxel Dörfler 		return status;
9039c50f36eSAxel Dörfler 
9049c50f36eSAxel Dörfler 	BEntry entry(&ref, true);
9051ad8c760SFredrik Holmqvist 		// TODO: this resolves an eventual link for the item being added - this
9061ad8c760SFredrik Holmqvist 		// is okay for now, but in multi-user environments, one might want to
9071ad8c760SFredrik Holmqvist 		// have links that carry the be:deskbar_item_status attribute
9089c50f36eSAxel Dörfler 	status = entry.InitCheck();
9099c50f36eSAxel Dörfler 	if (status != B_OK)
9109c50f36eSAxel Dörfler 		return status;
9119c50f36eSAxel Dörfler 
91241281cf3SAxel Dörfler 	*id = 999;
913ced3755cSAxel Dörfler 	if (archive->what == B_ARCHIVED_OBJECT)
914ced3755cSAxel Dörfler 		archive->what = 0;
91541281cf3SAxel Dörfler 
916ced3755cSAxel Dörfler 	BRect originalBounds = archive->FindRect("_frame");
9171ad8c760SFredrik Holmqvist 		// this is a work-around for buggy replicants that change their size in
9181ad8c760SFredrik Holmqvist 		// AttachedToWindow() (such as "SVM")
919a5210ab1SAxel Dörfler 
9201ad8c760SFredrik Holmqvist 	// TODO: check for name collisions?
921ced3755cSAxel Dörfler 	status = fShelf->AddReplicant(archive, BPoint(1, 1));
9229c50f36eSAxel Dörfler 	if (status != B_OK)
9239c50f36eSAxel Dörfler 		return status;
92441281cf3SAxel Dörfler 
925a5210ab1SAxel Dörfler 	int32 count = fShelf->CountReplicants();
926a5210ab1SAxel Dörfler 	BView* view;
927a5210ab1SAxel Dörfler 	fShelf->ReplicantAt(count - 1, &view, (uint32*)id, NULL);
928a5210ab1SAxel Dörfler 
929a5210ab1SAxel Dörfler 	if (originalBounds != view->Bounds()) {
930a5210ab1SAxel Dörfler 		// The replicant changed its size when added to the window, so we need
93171bd3ba5SJonas Sundström 		// to recompute all over again (it's already done once via
93271bd3ba5SJonas Sundström 		// BShelf::AddReplicant() and TReplicantShelf::CanAcceptReplicantView())
933a5210ab1SAxel Dörfler 		RealignReplicants();
934a5210ab1SAxel Dörfler 	}
935a5210ab1SAxel Dörfler 
93641281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
93741281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
93841281cf3SAxel Dörfler 	float width, height;
93941281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
94041281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height)
94141281cf3SAxel Dörfler 		AdjustPlacement();
94241281cf3SAxel Dörfler 
94341281cf3SAxel Dörfler 	// add the item to the add-on list
94441281cf3SAxel Dörfler 
94541281cf3SAxel Dörfler 	AddItem(*id, nodeRef, entry, addOn != NULL);
94641281cf3SAxel Dörfler 	return B_OK;
94741281cf3SAxel Dörfler }
94841281cf3SAxel Dörfler 
94941281cf3SAxel Dörfler 
95041281cf3SAxel Dörfler void
95141281cf3SAxel Dörfler TReplicantTray::RemoveIcon(int32 target, bool byIndex)
95241281cf3SAxel Dörfler {
95341281cf3SAxel Dörfler 	if (target < 0)
95441281cf3SAxel Dörfler 		return;
95541281cf3SAxel Dörfler 
95641281cf3SAxel Dörfler 	int32 index, id;
95741281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
95841281cf3SAxel Dörfler 	if (view && index >= 0) {
9597da06231SAxel Dörfler 		// remove the reference from the item list & the shelf
96041281cf3SAxel Dörfler 		RemoveItem(id);
96141281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9627da06231SAxel Dörfler 
96341281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
96441281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
96541281cf3SAxel Dörfler 	}
96641281cf3SAxel Dörfler }
96741281cf3SAxel Dörfler 
96841281cf3SAxel Dörfler 
96941281cf3SAxel Dörfler void
97041281cf3SAxel Dörfler TReplicantTray::RemoveIcon(const char* name)
97141281cf3SAxel Dörfler {
97241281cf3SAxel Dörfler 	if (!name || strlen(name) <= 0)
97341281cf3SAxel Dörfler 		return;
97441281cf3SAxel Dörfler 
97541281cf3SAxel Dörfler 	int32 id, index;
97641281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
97741281cf3SAxel Dörfler 	if (view && index >= 0) {
9787da06231SAxel Dörfler 		// remove the reference from the item list & shelf
97941281cf3SAxel Dörfler 		RemoveItem(id);
98041281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9817da06231SAxel Dörfler 
98241281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
98341281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
98441281cf3SAxel Dörfler 	}
98541281cf3SAxel Dörfler }
98641281cf3SAxel Dörfler 
98741281cf3SAxel Dörfler 
98841281cf3SAxel Dörfler void
9897da06231SAxel Dörfler TReplicantTray::RealReplicantAdjustment(int32 startIndex)
99041281cf3SAxel Dörfler {
9917da06231SAxel Dörfler 	if (startIndex < 0)
99241281cf3SAxel Dörfler 		return;
9937da06231SAxel Dörfler 
9947da06231SAxel Dörfler 	if (startIndex == fLastReplicant)
9957da06231SAxel Dörfler 		startIndex = 0;
9967da06231SAxel Dörfler 
99741281cf3SAxel Dörfler 	// reset the locations of all replicants after the one deleted
9987da06231SAxel Dörfler 	RealignReplicants(startIndex);
99941281cf3SAxel Dörfler 
100041281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
100141281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
100241281cf3SAxel Dörfler 	float width, height;
100341281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
100441281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height) {
10051ad8c760SFredrik Holmqvist 		// resize view to accomodate the replicants, redraw as necessary
100641281cf3SAxel Dörfler 		AdjustPlacement();
100741281cf3SAxel Dörfler 	}
100841281cf3SAxel Dörfler }
100941281cf3SAxel Dörfler 
101041281cf3SAxel Dörfler 
10117da06231SAxel Dörfler /**	looking for a replicant by id/index
10127da06231SAxel Dörfler  *	return the view and index
10137da06231SAxel Dörfler  */
10147da06231SAxel Dörfler 
101541281cf3SAxel Dörfler BView*
101641281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, int32 target, bool byIndex)
101741281cf3SAxel Dörfler {
101841281cf3SAxel Dörfler 	*index = -1;
101941281cf3SAxel Dörfler 
102041281cf3SAxel Dörfler 	BView* view;
102141281cf3SAxel Dörfler 	if (byIndex) {
102241281cf3SAxel Dörfler 		if (fShelf->ReplicantAt(target, &view, (uint32*)id)) {
102341281cf3SAxel Dörfler 			if (view) {
102441281cf3SAxel Dörfler 				*index = target;
102541281cf3SAxel Dörfler 				return view;
102641281cf3SAxel Dörfler 			}
102741281cf3SAxel Dörfler 		}
102841281cf3SAxel Dörfler 	} else {
102941281cf3SAxel Dörfler 		int32 count = fShelf->CountReplicants() - 1;
103041281cf3SAxel Dörfler 		int32 localid;
103141281cf3SAxel Dörfler 		for (int32 repIndex = count ; repIndex >= 0 ; repIndex--) {
103241281cf3SAxel Dörfler 			fShelf->ReplicantAt(repIndex, &view, (uint32*)&localid);
103341281cf3SAxel Dörfler 			if (localid == target && view) {
103441281cf3SAxel Dörfler 				*index = repIndex;
103541281cf3SAxel Dörfler 				*id = localid;
103641281cf3SAxel Dörfler 				return view;
103741281cf3SAxel Dörfler 			}
103841281cf3SAxel Dörfler 		}
103941281cf3SAxel Dörfler 	}
104041281cf3SAxel Dörfler 	return NULL;
104141281cf3SAxel Dörfler }
104241281cf3SAxel Dörfler 
104341281cf3SAxel Dörfler 
10447da06231SAxel Dörfler /**	looking for a replicant with a view by name
10457da06231SAxel Dörfler  *	return the view, index and the id of the replicant
10467da06231SAxel Dörfler  */
10477da06231SAxel Dörfler 
104841281cf3SAxel Dörfler BView*
104941281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, const char* name)
105041281cf3SAxel Dörfler {
105141281cf3SAxel Dörfler 	*index = -1;
105241281cf3SAxel Dörfler 	*id = -1;
105341281cf3SAxel Dörfler 
105441281cf3SAxel Dörfler 	BView* view;
105541281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants()-1;
105641281cf3SAxel Dörfler 	for (int32 repIndex = count ; repIndex >= 0 ; repIndex--) {
105741281cf3SAxel Dörfler 		fShelf->ReplicantAt(repIndex, &view, (uint32*)id);
105841281cf3SAxel Dörfler 		if (view && view->Name() && strcmp(name, view->Name()) == 0) {
105941281cf3SAxel Dörfler 			*index = repIndex;
106041281cf3SAxel Dörfler 			return view;
106141281cf3SAxel Dörfler 		}
106241281cf3SAxel Dörfler 	}
106341281cf3SAxel Dörfler 	return NULL;
106441281cf3SAxel Dörfler }
106541281cf3SAxel Dörfler 
106641281cf3SAxel Dörfler 
10677da06231SAxel Dörfler /**	Shelf will call to determine where and if
10687da06231SAxel Dörfler  *	the replicant is to be added
10697da06231SAxel Dörfler  */
10707da06231SAxel Dörfler 
107141281cf3SAxel Dörfler bool
107241281cf3SAxel Dörfler TReplicantTray::AcceptAddon(BRect replicantFrame, BMessage* message)
107341281cf3SAxel Dörfler {
107441281cf3SAxel Dörfler 	if (!message)
107541281cf3SAxel Dörfler 		return false;
107641281cf3SAxel Dörfler 
107741281cf3SAxel Dörfler 	if (replicantFrame.Height() > kMaxReplicantHeight)
107841281cf3SAxel Dörfler 		return false;
107941281cf3SAxel Dörfler 
108041281cf3SAxel Dörfler 	alignment align = B_ALIGN_LEFT;
108141281cf3SAxel Dörfler 	if (fAlignmentSupport && message->HasBool("deskbar:dynamic_align")) {
108241281cf3SAxel Dörfler 		if (!fBarView->Vertical())
108341281cf3SAxel Dörfler 			align = B_ALIGN_RIGHT;
108441281cf3SAxel Dörfler 		else
108541281cf3SAxel Dörfler 			align = fBarView->Left() ? B_ALIGN_LEFT : B_ALIGN_RIGHT;
108641281cf3SAxel Dörfler 	} else if (message->HasInt32("deskbar:align"))
108741281cf3SAxel Dörfler 		message->FindInt32("deskbar:align", (int32*)&align);
108841281cf3SAxel Dörfler 
108941281cf3SAxel Dörfler 	if (message->HasInt32("deskbar:private_align"))
109041281cf3SAxel Dörfler 		message->FindInt32("deskbar:private_align", (int32*)&align);
109141281cf3SAxel Dörfler 	else
109241281cf3SAxel Dörfler 		align = B_ALIGN_LEFT;
109341281cf3SAxel Dörfler 
10947da06231SAxel Dörfler 	BPoint loc = LocationForReplicant(fShelf->CountReplicants(),
10957da06231SAxel Dörfler 		replicantFrame.Width());
109641281cf3SAxel Dörfler 
109741281cf3SAxel Dörfler 	message->AddPoint("_pjp_loc", loc);
109841281cf3SAxel Dörfler 	return true;
109941281cf3SAxel Dörfler }
110041281cf3SAxel Dörfler 
110141281cf3SAxel Dörfler 
110241281cf3SAxel Dörfler /**	based on the previous (index - 1) replicant in the list
110341281cf3SAxel Dörfler  *	calculate where the left point should be for this
110441281cf3SAxel Dörfler  *	replicant.  replicant will flow to the right on its own
110541281cf3SAxel Dörfler  */
110641281cf3SAxel Dörfler 
110741281cf3SAxel Dörfler BPoint
11087da06231SAxel Dörfler TReplicantTray::LocationForReplicant(int32 index, float width)
110941281cf3SAxel Dörfler {
111041281cf3SAxel Dörfler 	BPoint loc(kIconGap + 1, kGutter + 1);
111141281cf3SAxel Dörfler 
11127da06231SAxel Dörfler 	if (fMultiRowMode) {
11137da06231SAxel Dörfler 		// try to find free space in every row
11147da06231SAxel Dörfler 		for (int32 row = 0; ; loc.y += kMaxReplicantHeight + kIconGap, row++) {
11157da06231SAxel Dörfler 			// determine free space in this row
11162ce9bab8SJohn Scipione 			BRect rect(loc.x, loc.y, loc.x + fMinimumTrayWidth - kIconGap
11172ce9bab8SJohn Scipione 				- 2.0, loc.y + kMaxReplicantHeight);
1118573f748cSJohn Scipione 			if (row == 0 && !fTime->IsHidden())
1119573f748cSJohn Scipione 				rect.right -= fTime->Frame().Width() + kIconGap;
11207da06231SAxel Dörfler 
11217da06231SAxel Dörfler 			for (int32 i = 0; i < index; i++) {
11227da06231SAxel Dörfler 				BView* view = NULL;
11237da06231SAxel Dörfler 				fShelf->ReplicantAt(i, &view);
11247da06231SAxel Dörfler 				if (view == NULL || view->Frame().top != rect.top)
11257da06231SAxel Dörfler 					continue;
11267da06231SAxel Dörfler 
11277da06231SAxel Dörfler 				rect.left = view->Frame().right + kIconGap + 1;
11287da06231SAxel Dörfler 			}
11297da06231SAxel Dörfler 
11307da06231SAxel Dörfler 			if (rect.Width() >= width) {
11317da06231SAxel Dörfler 				// the icon fits in this row
11327da06231SAxel Dörfler 				loc = rect.LeftTop();
11337da06231SAxel Dörfler 				break;
11347da06231SAxel Dörfler 			}
11357da06231SAxel Dörfler 		}
11367da06231SAxel Dörfler 	} else {
113741281cf3SAxel Dörfler 		if (index > 0) {
113841281cf3SAxel Dörfler 			// get the last replicant added for placement reference
113941281cf3SAxel Dörfler 			BView* view = NULL;
11407da06231SAxel Dörfler 			fShelf->ReplicantAt(index - 1, &view);
114141281cf3SAxel Dörfler 			if (view) {
114241281cf3SAxel Dörfler 				// push this rep placement past the last one
114341281cf3SAxel Dörfler 				loc.x = view->Frame().right + kIconGap + 1;
114441281cf3SAxel Dörfler 				loc.y = view->Frame().top;
114541281cf3SAxel Dörfler 			}
114641281cf3SAxel Dörfler 		}
11477da06231SAxel Dörfler 	}
114841281cf3SAxel Dörfler 
114971bd3ba5SJonas Sundström 	if ((loc.y == fRightBottomReplicant.top && loc.x
115071bd3ba5SJonas Sundström 		> fRightBottomReplicant.left) || loc.y > fRightBottomReplicant.top) {
115171bd3ba5SJonas Sundström 		fRightBottomReplicant.Set(loc.x, loc.y, loc.x + width, loc.y
115271bd3ba5SJonas Sundström 		+ kMaxReplicantHeight);
11537da06231SAxel Dörfler 		fLastReplicant = index;
115441281cf3SAxel Dörfler 	}
11557da06231SAxel Dörfler 
115641281cf3SAxel Dörfler 	return loc;
115741281cf3SAxel Dörfler }
115841281cf3SAxel Dörfler 
115941281cf3SAxel Dörfler 
116041281cf3SAxel Dörfler BRect
116141281cf3SAxel Dörfler TReplicantTray::IconFrame(int32 target, bool byIndex)
116241281cf3SAxel Dörfler {
116341281cf3SAxel Dörfler 	int32 index, id;
116441281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
116541281cf3SAxel Dörfler 	if (view)
116641281cf3SAxel Dörfler 		return view->Frame();
116741281cf3SAxel Dörfler 
116841281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
116941281cf3SAxel Dörfler }
117041281cf3SAxel Dörfler 
117141281cf3SAxel Dörfler 
117241281cf3SAxel Dörfler BRect
117341281cf3SAxel Dörfler TReplicantTray::IconFrame(const char* name)
117441281cf3SAxel Dörfler {
117541281cf3SAxel Dörfler 	if (!name)
117641281cf3SAxel Dörfler 		return BRect(0, 0, 0, 0);
117741281cf3SAxel Dörfler 
117841281cf3SAxel Dörfler 	int32 id, index;
117941281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
118041281cf3SAxel Dörfler 	if (view)
118141281cf3SAxel Dörfler 		return view->Frame();
118241281cf3SAxel Dörfler 
118341281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
118441281cf3SAxel Dörfler }
118541281cf3SAxel Dörfler 
118641281cf3SAxel Dörfler 
11877da06231SAxel Dörfler /**	Scan from the startIndex and reset the location
11887da06231SAxel Dörfler  *	as defined in LocationForReplicant()
118941281cf3SAxel Dörfler  */
119041281cf3SAxel Dörfler 
119141281cf3SAxel Dörfler void
119241281cf3SAxel Dörfler TReplicantTray::RealignReplicants(int32 startIndex)
119341281cf3SAxel Dörfler {
119441281cf3SAxel Dörfler 	if (startIndex < 0)
119541281cf3SAxel Dörfler 		startIndex = 0;
119641281cf3SAxel Dörfler 
119741281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants();
119841281cf3SAxel Dörfler 	if (count <= 0)
119941281cf3SAxel Dörfler 		return;
120041281cf3SAxel Dörfler 
12017da06231SAxel Dörfler 	if (startIndex == 0)
12027da06231SAxel Dörfler 		fRightBottomReplicant.Set(0, 0, 0, 0);
12037da06231SAxel Dörfler 
120441281cf3SAxel Dörfler 	BView* view = NULL;
120541281cf3SAxel Dörfler 	for (int32 i = startIndex ; i < count ; i++) {
120641281cf3SAxel Dörfler 		fShelf->ReplicantAt(i, &view);
12070bec7100SStefano Ceccherini 		if (view != NULL) {
12087da06231SAxel Dörfler 			BPoint loc = LocationForReplicant(i, view->Frame().Width());
12090bec7100SStefano Ceccherini 			if (view->Frame().LeftTop() != loc)
121041281cf3SAxel Dörfler 				view->MoveTo(loc);
121141281cf3SAxel Dörfler 		}
121241281cf3SAxel Dörfler 	}
121341281cf3SAxel Dörfler }
121441281cf3SAxel Dörfler 
121541281cf3SAxel Dörfler 
12168aab28f1SRene Gollent status_t
12177625ce51SRene Gollent TReplicantTray::_SaveSettings()
12187625ce51SRene Gollent {
12197625ce51SRene Gollent 	status_t result;
12207625ce51SRene Gollent 	BPath path;
12217625ce51SRene Gollent 	if ((result = find_directory(B_USER_SETTINGS_DIRECTORY, &path, true))
12227625ce51SRene Gollent 		 == B_OK) {
12237625ce51SRene Gollent 		path.Append(kReplicantSettingsFile);
12247625ce51SRene Gollent 
12257625ce51SRene Gollent 		BFile file(path.Path(), B_READ_WRITE | B_CREATE_FILE | B_ERASE_FILE);
12267625ce51SRene Gollent 		if ((result = file.InitCheck()) == B_OK)
12277625ce51SRene Gollent 			result = fAddOnSettings.Flatten(&file);
12287625ce51SRene Gollent 	}
12297625ce51SRene Gollent 
12307625ce51SRene Gollent 	return result;
12317625ce51SRene Gollent }
12327625ce51SRene Gollent 
12337625ce51SRene Gollent 
12343cf2d117SJohn Scipione void
12353cf2d117SJohn Scipione TReplicantTray::SaveTimeSettings()
12363cf2d117SJohn Scipione {
12373cf2d117SJohn Scipione 	if (fTime == NULL)
12383cf2d117SJohn Scipione 		return;
12393cf2d117SJohn Scipione 
12403cf2d117SJohn Scipione 	desk_settings* settings = ((TBarApp*)be_app)->Settings();
12413cf2d117SJohn Scipione 	settings->showTime = !fTime->IsHidden();
12423cf2d117SJohn Scipione 	settings->showSeconds = fTime->ShowSeconds();
12433cf2d117SJohn Scipione 	settings->showDayOfWeek = fTime->ShowDayOfWeek();
12443cf2d117SJohn Scipione }
12453cf2d117SJohn Scipione 
12463cf2d117SJohn Scipione 
124741281cf3SAxel Dörfler //	#pragma mark -
124841281cf3SAxel Dörfler 
124941281cf3SAxel Dörfler 
1250ca9acc20SAxel Dörfler /*! Draggable region that is asynchronous so that dragging does not block
1251ca9acc20SAxel Dörfler 	other activities.
12527da06231SAxel Dörfler */
125341281cf3SAxel Dörfler TDragRegion::TDragRegion(TBarView* parent, BView* child)
1254ca9acc20SAxel Dörfler 	:
1255ca9acc20SAxel Dörfler 	BControl(BRect(0, 0, 0, 0), "", "", NULL, B_FOLLOW_NONE,
125641281cf3SAxel Dörfler 		B_WILL_DRAW | B_FRAME_EVENTS),
125741281cf3SAxel Dörfler 	fBarView(parent),
125841281cf3SAxel Dörfler 	fChild(child),
125941281cf3SAxel Dörfler 	fDragLocation(kAutoPlaceDragRegion)
126041281cf3SAxel Dörfler {
126141281cf3SAxel Dörfler }
126241281cf3SAxel Dörfler 
126341281cf3SAxel Dörfler 
126441281cf3SAxel Dörfler void
126541281cf3SAxel Dörfler TDragRegion::AttachedToWindow()
126641281cf3SAxel Dörfler {
126741281cf3SAxel Dörfler 	BView::AttachedToWindow();
1268cb6afcb1SStephan Aßmus 	if (be_control_look != NULL)
1269cb6afcb1SStephan Aßmus 		SetViewColor(tint_color(ui_color(B_MENU_BACKGROUND_COLOR), 1.1));
1270cb6afcb1SStephan Aßmus 	else
127141281cf3SAxel Dörfler 		SetViewColor(ui_color(B_MENU_BACKGROUND_COLOR));
127241281cf3SAxel Dörfler 	ResizeToPreferred();
127341281cf3SAxel Dörfler }
127441281cf3SAxel Dörfler 
127541281cf3SAxel Dörfler 
127641281cf3SAxel Dörfler void
127741281cf3SAxel Dörfler TDragRegion::GetPreferredSize(float* width, float* height)
127841281cf3SAxel Dörfler {
127941281cf3SAxel Dörfler 	fChild->ResizeToPreferred();
128041281cf3SAxel Dörfler 	*width = fChild->Bounds().Width();
128141281cf3SAxel Dörfler 	*height = fChild->Bounds().Height();
128241281cf3SAxel Dörfler 
128341281cf3SAxel Dörfler 	if (fDragLocation != kNoDragRegion)
128441281cf3SAxel Dörfler 		*width += 7;
128541281cf3SAxel Dörfler 	else
128641281cf3SAxel Dörfler 		*width += 6;
128741281cf3SAxel Dörfler 
128841281cf3SAxel Dörfler 	*height += 3;
128941281cf3SAxel Dörfler }
129041281cf3SAxel Dörfler 
129141281cf3SAxel Dörfler 
129241281cf3SAxel Dörfler void
129341281cf3SAxel Dörfler TDragRegion::FrameMoved(BPoint)
129441281cf3SAxel Dörfler {
1295ca9acc20SAxel Dörfler 	if (fBarView->Left() && fBarView->Vertical()
1296ca9acc20SAxel Dörfler 		&& fDragLocation != kNoDragRegion)
129741281cf3SAxel Dörfler 		fChild->MoveTo(5, 2);
129841281cf3SAxel Dörfler 	else
129941281cf3SAxel Dörfler 		fChild->MoveTo(2, 2);
130041281cf3SAxel Dörfler }
130141281cf3SAxel Dörfler 
130241281cf3SAxel Dörfler 
130341281cf3SAxel Dörfler void
130441281cf3SAxel Dörfler TDragRegion::Draw(BRect)
130541281cf3SAxel Dörfler {
1306eaa9af99SStephan Aßmus 	rgb_color menuColor = ViewColor();
130741281cf3SAxel Dörfler 	rgb_color hilite = tint_color(menuColor, B_DARKEN_1_TINT);
1308cb6afcb1SStephan Aßmus 	rgb_color ldark = tint_color(menuColor, 1.02);
1309cb6afcb1SStephan Aßmus 	rgb_color dark = tint_color(menuColor, B_DARKEN_2_TINT);
131041281cf3SAxel Dörfler 	rgb_color vvdark = tint_color(menuColor, B_DARKEN_4_TINT);
131141281cf3SAxel Dörfler 	rgb_color light = tint_color(menuColor, B_LIGHTEN_2_TINT);
131241281cf3SAxel Dörfler 
131341281cf3SAxel Dörfler 	BRect frame(Bounds());
131441281cf3SAxel Dörfler 	BeginLineArray(4);
131541281cf3SAxel Dörfler 
1316cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1317cb6afcb1SStephan Aßmus 		if (fBarView->Vertical()) {
1318cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(), frame.RightTop(), dark);
1319cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1320cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.top + 1), ldark);
1321cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
1322cb6afcb1SStephan Aßmus 		} else if (fBarView->AcrossTop() || fBarView->AcrossBottom()) {
1323cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(),
1324cb6afcb1SStephan Aßmus 				BPoint(frame.left, frame.bottom), dark);
1325cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.top + 1),
1326cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
1327cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right, frame.top + 2),
1328cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.bottom), hilite);
1329cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.bottom),
1330cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
1331cb6afcb1SStephan Aßmus 		}
1332cb6afcb1SStephan Aßmus 	} else {
133341281cf3SAxel Dörfler 		if (fBarView->Vertical()) {
133441281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.RightTop(), light);
133541281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.LeftBottom(), light);
133641281cf3SAxel Dörfler 			AddLine(frame.RightBottom(), frame.RightTop(), hilite);
133741281cf3SAxel Dörfler 		} else if (fBarView->AcrossTop()) {
1338cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1339cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
134041281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1341cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 2),
1342cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1343cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(),
1344cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
134541281cf3SAxel Dörfler 		} else if (fBarView->AcrossBottom()) {
1346cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1347cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
134841281cf3SAxel Dörfler 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
134941281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1350cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 1),
1351cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1352cb6afcb1SStephan Aßmus 		}
135341281cf3SAxel Dörfler 	}
135441281cf3SAxel Dörfler 
135541281cf3SAxel Dörfler 	EndLineArray();
135641281cf3SAxel Dörfler 
135741281cf3SAxel Dörfler 	if (fDragLocation != kDontDrawDragRegion || fDragLocation != kNoDragRegion)
135841281cf3SAxel Dörfler 		DrawDragRegion();
135941281cf3SAxel Dörfler }
136041281cf3SAxel Dörfler 
136141281cf3SAxel Dörfler 
136241281cf3SAxel Dörfler void
136341281cf3SAxel Dörfler TDragRegion::DrawDragRegion()
136441281cf3SAxel Dörfler {
136541281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
136641281cf3SAxel Dörfler 
1367cb6afcb1SStephan Aßmus 	rgb_color menuColor = ViewColor();
1368cb6afcb1SStephan Aßmus 	rgb_color menuHilite = menuColor;
136941281cf3SAxel Dörfler 	if (IsTracking()) {
1370cb6afcb1SStephan Aßmus 		// Draw drag region highlighted if tracking mouse
1371cb6afcb1SStephan Aßmus 		menuHilite = tint_color(menuColor, B_HIGHLIGHT_BACKGROUND_TINT);
137241281cf3SAxel Dörfler 		SetHighColor(menuHilite);
137341281cf3SAxel Dörfler 		FillRect(dragRegion);
137441281cf3SAxel Dörfler 	}
1375cb6afcb1SStephan Aßmus 	rgb_color vdark = tint_color(menuHilite, B_DARKEN_3_TINT);
1376cb6afcb1SStephan Aßmus 	rgb_color light = tint_color(menuHilite, B_LIGHTEN_2_TINT);
1377cb6afcb1SStephan Aßmus 
1378cb6afcb1SStephan Aßmus 	BeginLineArray(dragRegion.IntegerHeight());
1379cb6afcb1SStephan Aßmus 	BPoint pt;
1380cb6afcb1SStephan Aßmus 	pt.x = floorf((dragRegion.left + dragRegion.right) / 2 + 0.5) - 1;
1381cb6afcb1SStephan Aßmus 	pt.y = dragRegion.top + 2;
1382cb6afcb1SStephan Aßmus 
138341281cf3SAxel Dörfler 	while (pt.y + 1 <= dragRegion.bottom) {
138441281cf3SAxel Dörfler 		AddLine(pt, pt, vdark);
138541281cf3SAxel Dörfler 		AddLine(pt + BPoint(1, 1), pt + BPoint(1, 1), light);
138641281cf3SAxel Dörfler 
138741281cf3SAxel Dörfler 		pt.y += 3;
138841281cf3SAxel Dörfler 	}
138941281cf3SAxel Dörfler 	EndLineArray();
139041281cf3SAxel Dörfler }
139141281cf3SAxel Dörfler 
139241281cf3SAxel Dörfler 
139341281cf3SAxel Dörfler BRect
139441281cf3SAxel Dörfler TDragRegion::DragRegion() const
139541281cf3SAxel Dörfler {
1396cb6afcb1SStephan Aßmus 	float kTopBottomInset = 2;
1397cb6afcb1SStephan Aßmus 	float kLeftRightInset = 1;
1398cb6afcb1SStephan Aßmus 	float kDragWidth = 3;
1399cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1400cb6afcb1SStephan Aßmus 		kTopBottomInset = 1;
1401cb6afcb1SStephan Aßmus 		kLeftRightInset = 0;
1402cb6afcb1SStephan Aßmus 		kDragWidth = 4;
1403cb6afcb1SStephan Aßmus 	}
1404cb6afcb1SStephan Aßmus 
140541281cf3SAxel Dörfler 	BRect dragRegion(Bounds());
1406cb6afcb1SStephan Aßmus 	dragRegion.top += kTopBottomInset;
1407cb6afcb1SStephan Aßmus 	dragRegion.bottom -= kTopBottomInset;
140841281cf3SAxel Dörfler 
140941281cf3SAxel Dörfler 	bool placeOnLeft = false;
141041281cf3SAxel Dörfler 	if (fDragLocation == kAutoPlaceDragRegion) {
141141281cf3SAxel Dörfler 		if (fBarView->Vertical() && fBarView->Left())
141241281cf3SAxel Dörfler 			placeOnLeft = true;
141341281cf3SAxel Dörfler 		else
141441281cf3SAxel Dörfler 			placeOnLeft = false;
141541281cf3SAxel Dörfler 	} else if (fDragLocation == kDragRegionLeft)
141641281cf3SAxel Dörfler 		placeOnLeft = true;
141741281cf3SAxel Dörfler 	else if (fDragLocation == kDragRegionRight)
141841281cf3SAxel Dörfler 		placeOnLeft = false;
141941281cf3SAxel Dörfler 
142041281cf3SAxel Dörfler 	if (placeOnLeft) {
1421cb6afcb1SStephan Aßmus 		dragRegion.left += kLeftRightInset;
1422cb6afcb1SStephan Aßmus 		dragRegion.right = dragRegion.left + kDragWidth;
142341281cf3SAxel Dörfler 	} else {
1424cb6afcb1SStephan Aßmus 		dragRegion.right -= kLeftRightInset;
1425cb6afcb1SStephan Aßmus 		dragRegion.left = dragRegion.right - kDragWidth;
142641281cf3SAxel Dörfler 	}
142741281cf3SAxel Dörfler 
142841281cf3SAxel Dörfler 	return dragRegion;
142941281cf3SAxel Dörfler }
143041281cf3SAxel Dörfler 
143141281cf3SAxel Dörfler 
143241281cf3SAxel Dörfler void
143341281cf3SAxel Dörfler TDragRegion::MouseDown(BPoint thePoint)
143441281cf3SAxel Dörfler {
143541281cf3SAxel Dörfler 	ulong buttons;
143641281cf3SAxel Dörfler 	BPoint where;
143741281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
143841281cf3SAxel Dörfler 
14397da06231SAxel Dörfler 	dragRegion.InsetBy(-2.0f, -2.0f);
14407da06231SAxel Dörfler 		// DragRegion() is designed for drawing, not clicking
144141281cf3SAxel Dörfler 
144241281cf3SAxel Dörfler 	if (!dragRegion.Contains(thePoint))
144341281cf3SAxel Dörfler 		return;
144441281cf3SAxel Dörfler 
144541281cf3SAxel Dörfler 	while (true) {
144641281cf3SAxel Dörfler 		GetMouse(&where, &buttons);
144741281cf3SAxel Dörfler 		if (!buttons)
144841281cf3SAxel Dörfler 			break;
144941281cf3SAxel Dörfler 
145041281cf3SAxel Dörfler 		if ((Window()->Flags() & B_ASYNCHRONOUS_CONTROLS) != 0) {
145141281cf3SAxel Dörfler 			fPreviousPosition = thePoint;
145241281cf3SAxel Dörfler 			SetTracking(true);
145341281cf3SAxel Dörfler 			SetMouseEventMask(B_POINTER_EVENTS,
145441281cf3SAxel Dörfler 				B_NO_POINTER_HISTORY | B_LOCK_WINDOW_FOCUS);
145541281cf3SAxel Dörfler 			Invalidate(DragRegion());
145641281cf3SAxel Dörfler 			break;
145741281cf3SAxel Dörfler 		}
145841281cf3SAxel Dörfler 
145941281cf3SAxel Dörfler 		snooze(25000);
146041281cf3SAxel Dörfler 	}
146141281cf3SAxel Dörfler }
146241281cf3SAxel Dörfler 
146341281cf3SAxel Dörfler 
146441281cf3SAxel Dörfler void
146541281cf3SAxel Dörfler TDragRegion::MouseUp(BPoint pt)
146641281cf3SAxel Dörfler {
146741281cf3SAxel Dörfler 	if (IsTracking()) {
146841281cf3SAxel Dörfler 		SetTracking(false);
146941281cf3SAxel Dörfler 		Invalidate(DragRegion());
147041281cf3SAxel Dörfler 	} else
147141281cf3SAxel Dörfler 		BControl::MouseUp(pt);
147241281cf3SAxel Dörfler }
147341281cf3SAxel Dörfler 
147441281cf3SAxel Dörfler 
147541281cf3SAxel Dörfler bool
147641281cf3SAxel Dörfler TDragRegion::SwitchModeForRect(BPoint mouse, BRect rect,
147741281cf3SAxel Dörfler 	bool newVertical, bool newLeft, bool newTop, int32 newState)
147841281cf3SAxel Dörfler {
14791ad8c760SFredrik Holmqvist 	if (!rect.Contains(mouse)) {
148041281cf3SAxel Dörfler 		// not our rect
148141281cf3SAxel Dörfler 		return false;
14821ad8c760SFredrik Holmqvist 	}
148341281cf3SAxel Dörfler 
14841ad8c760SFredrik Holmqvist 	if (newVertical == fBarView->Vertical() && newLeft == fBarView->Left()
14851ad8c760SFredrik Holmqvist 		&& newTop == fBarView->Top() && newState == fBarView->State()) {
148641281cf3SAxel Dörfler 		// already in the correct mode
148741281cf3SAxel Dörfler 		return true;
14881ad8c760SFredrik Holmqvist 	}
148941281cf3SAxel Dörfler 
149041281cf3SAxel Dörfler 	fBarView->ChangeState(newState, newVertical, newLeft, newTop);
149141281cf3SAxel Dörfler 	return true;
149241281cf3SAxel Dörfler }
149341281cf3SAxel Dörfler 
149441281cf3SAxel Dörfler 
149541281cf3SAxel Dörfler void
149641281cf3SAxel Dörfler TDragRegion::MouseMoved(BPoint where, uint32 code, const BMessage* message)
149741281cf3SAxel Dörfler {
149841281cf3SAxel Dörfler 	if (IsTracking()) {
149941281cf3SAxel Dörfler 		BScreen screen;
150041281cf3SAxel Dörfler 		BRect frame = screen.Frame();
150141281cf3SAxel Dörfler 
150241281cf3SAxel Dörfler 		float hDivider = frame.Width() / 6;
150371bd3ba5SJonas Sundström 		hDivider = (hDivider < sMinimumWindowWidth + 10.0f)
150471bd3ba5SJonas Sundström 			? sMinimumWindowWidth + 10.0f : hDivider;
150541281cf3SAxel Dörfler 		float miniDivider = frame.top + kMiniHeight + 10.0f;
150641281cf3SAxel Dörfler 		float vDivider = frame.Height() / 2;
150741281cf3SAxel Dörfler #ifdef FULL_MODE
150841281cf3SAxel Dörfler 		float thirdScreen = frame.Height() / 3;
150941281cf3SAxel Dörfler #endif
151071bd3ba5SJonas Sundström 		BRect topLeft(frame.left, frame.top, frame.left + hDivider,
151171bd3ba5SJonas Sundström 			miniDivider);
151271bd3ba5SJonas Sundström 		BRect topMiddle(frame.left + hDivider, frame.top, frame.right
151371bd3ba5SJonas Sundström 			- hDivider, vDivider);
151471bd3ba5SJonas Sundström 		BRect topRight(frame.right - hDivider, frame.top, frame.right,
151571bd3ba5SJonas Sundström 			miniDivider);
151641281cf3SAxel Dörfler 
151741281cf3SAxel Dörfler #ifdef FULL_MODE
151841281cf3SAxel Dörfler 		vDivider = miniDivider + thirdScreen;
151941281cf3SAxel Dörfler #endif
152071bd3ba5SJonas Sundström 		BRect middleLeft(frame.left, miniDivider, frame.left + hDivider,
152171bd3ba5SJonas Sundström 			vDivider);
152271bd3ba5SJonas Sundström 		BRect middleRight(frame.right - hDivider, miniDivider, frame.right,
152371bd3ba5SJonas Sundström 			vDivider);
152441281cf3SAxel Dörfler 
152541281cf3SAxel Dörfler #ifdef FULL_MODE
152671bd3ba5SJonas Sundström 		BRect leftSide(frame.left, vDivider, frame.left + hDivider,
152771bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
152871bd3ba5SJonas Sundström 		BRect rightSide(frame.right - hDivider, vDivider, frame.right,
152971bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
153041281cf3SAxel Dörfler 
153141281cf3SAxel Dörfler 		vDivider = frame.bottom - thirdScreen;
153241281cf3SAxel Dörfler #endif
153371bd3ba5SJonas Sundström 		BRect bottomLeft(frame.left, vDivider, frame.left + hDivider,
153471bd3ba5SJonas Sundström 			frame.bottom);
153571bd3ba5SJonas Sundström 		BRect bottomMiddle(frame.left + hDivider, vDivider, frame.right
153671bd3ba5SJonas Sundström 			- hDivider, frame.bottom);
153771bd3ba5SJonas Sundström 		BRect bottomRight(frame.right - hDivider, vDivider, frame.right,
153871bd3ba5SJonas Sundström 			frame.bottom);
153941281cf3SAxel Dörfler 
154041281cf3SAxel Dörfler 		if (where != fPreviousPosition) {
154141281cf3SAxel Dörfler 			fPreviousPosition = where;
154241281cf3SAxel Dörfler 			ConvertToScreen(&where);
154341281cf3SAxel Dörfler 
154441281cf3SAxel Dörfler 			// use short circuit evaluation for convenience
154541281cf3SAxel Dörfler 			if (SwitchModeForRect(where, topLeft, true, true, true, kMiniState)
154671bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topMiddle, false, true, true,
154771bd3ba5SJonas Sundström 					kExpandoState)
154871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topRight, true, false, true,
154971bd3ba5SJonas Sundström 					kMiniState)
155071bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleLeft, true, true, true,
155171bd3ba5SJonas Sundström 					kExpandoState)
155271bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleRight, true, false, true,
155371bd3ba5SJonas Sundström 					kExpandoState)
155441281cf3SAxel Dörfler 
155541281cf3SAxel Dörfler #ifdef FULL_MODE
155671bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, leftSide, true, true, true,
155771bd3ba5SJonas Sundström 					kFullState)
155871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, rightSide, true, false, true,
155971bd3ba5SJonas Sundström 					kFullState)
156041281cf3SAxel Dörfler #endif
156171bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomLeft, true, true, false,
156271bd3ba5SJonas Sundström 					kMiniState)
156371bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomMiddle, false, true, false,
156471bd3ba5SJonas Sundström 					kExpandoState)
156571bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomRight, true, false, false,
156671bd3ba5SJonas Sundström 					kMiniState))
156741281cf3SAxel Dörfler 				;
156841281cf3SAxel Dörfler 		}
156941281cf3SAxel Dörfler 	} else
157041281cf3SAxel Dörfler 		BControl::MouseMoved(where, code, message);
157141281cf3SAxel Dörfler }
157241281cf3SAxel Dörfler 
157341281cf3SAxel Dörfler 
157441281cf3SAxel Dörfler int32
157541281cf3SAxel Dörfler TDragRegion::DragRegionLocation() const
157641281cf3SAxel Dörfler {
157741281cf3SAxel Dörfler 	return fDragLocation;
157841281cf3SAxel Dörfler }
157941281cf3SAxel Dörfler 
158041281cf3SAxel Dörfler 
158141281cf3SAxel Dörfler void
158241281cf3SAxel Dörfler TDragRegion::SetDragRegionLocation(int32 location)
158341281cf3SAxel Dörfler {
158441281cf3SAxel Dörfler 	if (location == fDragLocation)
158541281cf3SAxel Dörfler 		return;
158641281cf3SAxel Dörfler 
158741281cf3SAxel Dörfler 	fDragLocation = location;
158841281cf3SAxel Dörfler 	Invalidate();
158941281cf3SAxel Dörfler }
1590