xref: /haiku/src/apps/deskbar/StatusView.cpp (revision 0d2ac94c4df1061e1f2b835703f018553c929744)
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 
3641281cf3SAxel Dörfler #include <Debug.h>
3741281cf3SAxel Dörfler 
3841281cf3SAxel Dörfler #include <errno.h>
3941281cf3SAxel Dörfler #include <stdio.h>
4041281cf3SAxel Dörfler #include <string.h>
4141281cf3SAxel Dörfler #include <time.h>
4241281cf3SAxel Dörfler #include <unistd.h>
4341281cf3SAxel Dörfler #include <algorithm>
4441281cf3SAxel Dörfler 
4541281cf3SAxel Dörfler #include <fs_index.h>
4641281cf3SAxel Dörfler #include <fs_info.h>
4741281cf3SAxel Dörfler 
4841281cf3SAxel Dörfler #include <Application.h>
4941281cf3SAxel Dörfler #include <Beep.h>
50cb6afcb1SStephan Aßmus #include <Bitmap.h>
51ae7a6e3aSAdrien Destugues #include <Catalog.h>
52cb6afcb1SStephan Aßmus #include <ControlLook.h>
5341281cf3SAxel Dörfler #include <Directory.h>
5441281cf3SAxel Dörfler #include <FindDirectory.h>
55ae7a6e3aSAdrien Destugues #include <Locale.h>
5641281cf3SAxel Dörfler #include <MenuItem.h>
5741281cf3SAxel Dörfler #include <NodeInfo.h>
5841281cf3SAxel Dörfler #include <NodeMonitor.h>
5941281cf3SAxel Dörfler #include <Path.h>
6041281cf3SAxel Dörfler #include <PopUpMenu.h>
6141281cf3SAxel Dörfler #include <Roster.h>
6241281cf3SAxel Dörfler #include <Screen.h>
6341281cf3SAxel Dörfler #include <Volume.h>
6441281cf3SAxel Dörfler #include <VolumeRoster.h>
6541281cf3SAxel Dörfler #include <Window.h>
6641281cf3SAxel Dörfler 
67cb6afcb1SStephan Aßmus #include "icons_logo.h"
68cb6afcb1SStephan Aßmus #include "BarApp.h"
69323b6546SOliver Tappe #include "DeskbarUtils.h"
70cb6afcb1SStephan Aßmus #include "ResourceSet.h"
7141281cf3SAxel Dörfler #include "StatusView.h"
7241281cf3SAxel Dörfler #include "StatusViewShelf.h"
7341281cf3SAxel Dörfler #include "TimeView.h"
7441281cf3SAxel Dörfler 
756d01b66dSAxel Dörfler using std::max;
7641281cf3SAxel Dörfler 
7741281cf3SAxel Dörfler #ifdef DB_ADDONS
7841281cf3SAxel Dörfler // Add-on support
7941281cf3SAxel Dörfler //
8041281cf3SAxel Dörfler // Item - internal item list (node, eref, etc)
8141281cf3SAxel Dörfler // Icon - physical replicant handed to the DeskbarClass class
8241281cf3SAxel Dörfler // AddOn - attribute based add-on
8341281cf3SAxel Dörfler 
8441281cf3SAxel Dörfler const char* const kInstantiateItemCFunctionName = "instantiate_deskbar_item";
8541281cf3SAxel Dörfler const char* const kInstantiateEntryCFunctionName = "instantiate_deskbar_entry";
863e6ff860SRene Gollent const char* const kReplicantSettingsFile = "Deskbar_replicants";
87c0ad1c91SRene Gollent const char* const kReplicantPathField = "replicant_path";
8841281cf3SAxel Dörfler 
89cb6afcb1SStephan Aßmus float sMinimumWindowWidth = kGutter + kMinimumTrayWidth + kDragRegionWidth;
90cb6afcb1SStephan Aßmus 
9141281cf3SAxel Dörfler 
9241281cf3SAxel Dörfler static void
9341281cf3SAxel Dörfler DumpItem(DeskbarItemInfo* item)
9441281cf3SAxel Dörfler {
9541281cf3SAxel Dörfler 	printf("is addon: %i, id: %li\n", item->isAddOn, item->id);
9666eba86fSAxel Dörfler 	printf("entry_ref:  %ld, %Ld, %s\n", item->entryRef.device,
9766eba86fSAxel Dörfler 		item->entryRef.directory, item->entryRef.name);
9841281cf3SAxel Dörfler 	printf("node_ref:  %ld, %Ld\n", item->nodeRef.device, item->nodeRef.node);
9941281cf3SAxel Dörfler }
10041281cf3SAxel Dörfler 
10141281cf3SAxel Dörfler 
10241281cf3SAxel Dörfler static void
10341281cf3SAxel Dörfler DumpList(BList* itemlist)
10441281cf3SAxel Dörfler {
10541281cf3SAxel Dörfler 	int32 count = itemlist->CountItems() - 1;
10641281cf3SAxel Dörfler 	if (count < 0) {
10741281cf3SAxel Dörfler 		printf("no items in list\n");
10841281cf3SAxel Dörfler 		return;
10941281cf3SAxel Dörfler 	}
11041281cf3SAxel Dörfler 	for (int32 i = count; i >= 0; i--) {
11141281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)itemlist->ItemAt(i);
11241281cf3SAxel Dörfler 		if (!item)
11341281cf3SAxel Dörfler 			continue;
11441281cf3SAxel Dörfler 
11541281cf3SAxel Dörfler 		DumpItem(item);
11641281cf3SAxel Dörfler 	}
11741281cf3SAxel Dörfler }
11841281cf3SAxel Dörfler #endif	/* DB_ADDONS */
11941281cf3SAxel Dörfler 
12041281cf3SAxel Dörfler 
121546208a5SOliver Tappe #undef B_TRANSLATION_CONTEXT
122546208a5SOliver Tappe #define B_TRANSLATION_CONTEXT "Tray"
123ae7a6e3aSAdrien Destugues 
1247da06231SAxel Dörfler // don't change the name of this view to anything other than "Status"!
1257da06231SAxel Dörfler 
12641281cf3SAxel Dörfler TReplicantTray::TReplicantTray(TBarView* parent, bool vertical)
127*0d2ac94cSJohn Scipione 	:
128*0d2ac94cSJohn Scipione 	BView(BRect(0, 0, 1, 1), "_deskbar_rt_", B_FOLLOW_LEFT | B_FOLLOW_TOP,
129*0d2ac94cSJohn Scipione 		B_WILL_DRAW | B_FRAME_EVENTS),
130573f748cSJohn Scipione 	fTime(NULL),
13141281cf3SAxel Dörfler 	fBarView(parent),
13241281cf3SAxel Dörfler 	fShelf(new TReplicantShelf(this)),
13341281cf3SAxel Dörfler 	fMultiRowMode(vertical),
134cb6afcb1SStephan Aßmus 	fMinimumTrayWidth(kMinimumTrayWidth),
13541281cf3SAxel Dörfler 	fAlignmentSupport(false)
13641281cf3SAxel Dörfler {
137cb6afcb1SStephan Aßmus 	// init the minimum window width according to the logo.
138cb6afcb1SStephan Aßmus 	const BBitmap* logoBitmap = AppResSet()->FindBitmap(B_MESSAGE_TYPE,
1399e8a970bSJonas Sundström 		R_LeafLogoBitmap);
140cb6afcb1SStephan Aßmus 	if (logoBitmap != NULL) {
141cb6afcb1SStephan Aßmus 		sMinimumWindowWidth = max_c(sMinimumWindowWidth,
142cb6afcb1SStephan Aßmus 			2 * (logoBitmap->Bounds().Width() + 8));
143cb6afcb1SStephan Aßmus 		fMinimumTrayWidth = sMinimumWindowWidth - kGutter - kDragRegionWidth;
144cb6afcb1SStephan Aßmus 	}
145573f748cSJohn Scipione 
1463cf2d117SJohn Scipione 	BFormattingConventions conventions;
1473cf2d117SJohn Scipione 	BLocale::Default()->GetFormattingConventions(&conventions);
1483cf2d117SJohn Scipione 	bool use24HourClock = conventions.Use24HourClock();
1493cf2d117SJohn Scipione 	desk_settings* settings = ((TBarApp*)be_app)->Settings();
1503cf2d117SJohn Scipione 
151573f748cSJohn Scipione 	// Create the time view
152573f748cSJohn Scipione 	fTime = new TTimeView(fMinimumTrayWidth, kMaxReplicantHeight - 1.0,
1533cf2d117SJohn Scipione 		use24HourClock, settings->showSeconds, settings->showDayOfWeek,
1543cf2d117SJohn Scipione 		settings->showTimeZone);
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 
3223cf2d117SJohn Scipione 		case kShowTimeZone:
3233cf2d117SJohn Scipione 			if (fTime == NULL)
3243cf2d117SJohn Scipione 				return;
3253cf2d117SJohn Scipione 
3263cf2d117SJohn Scipione 			fTime->SetShowTimeZone(!fTime->ShowTimeZone());
3273cf2d117SJohn Scipione 
3283cf2d117SJohn Scipione 			// time string reformat -> realign
3293cf2d117SJohn Scipione 			RealignReplicants();
3303cf2d117SJohn Scipione 			AdjustPlacement();
3313cf2d117SJohn Scipione 			break;
33241281cf3SAxel Dörfler 
33341281cf3SAxel Dörfler #ifdef DB_ADDONS
33441281cf3SAxel Dörfler 		case B_NODE_MONITOR:
33541281cf3SAxel Dörfler 			HandleEntryUpdate(message);
33641281cf3SAxel Dörfler 			break;
33741281cf3SAxel Dörfler #endif
33841281cf3SAxel Dörfler 
33941281cf3SAxel Dörfler 		default:
34041281cf3SAxel Dörfler 			BView::MessageReceived(message);
34141281cf3SAxel Dörfler 			break;
34241281cf3SAxel Dörfler 	}
34341281cf3SAxel Dörfler }
34441281cf3SAxel Dörfler 
34541281cf3SAxel Dörfler 
34641281cf3SAxel Dörfler void
34741281cf3SAxel Dörfler TReplicantTray::MouseDown(BPoint where)
34841281cf3SAxel Dörfler {
34941281cf3SAxel Dörfler #ifdef DB_ADDONS
35041281cf3SAxel Dörfler 	if (modifiers() & B_CONTROL_KEY)
35141281cf3SAxel Dörfler 		DumpList(fItemList);
35241281cf3SAxel Dörfler #endif
35341281cf3SAxel Dörfler 
35441281cf3SAxel Dörfler 	uint32 buttons;
35541281cf3SAxel Dörfler 
35641281cf3SAxel Dörfler 	Window()->CurrentMessage()->FindInt32("buttons", (int32*)&buttons);
35741281cf3SAxel Dörfler 	if (buttons == B_SECONDARY_MOUSE_BUTTON) {
35841281cf3SAxel Dörfler 		ShowReplicantMenu(where);
35941281cf3SAxel Dörfler 	} else {
36041281cf3SAxel Dörfler 		BPoint save = where;
36141281cf3SAxel Dörfler 		bigtime_t doubleClickSpeed;
36241281cf3SAxel Dörfler 		bigtime_t start = system_time();
36341281cf3SAxel Dörfler 		uint32 buttons;
36441281cf3SAxel Dörfler 
36541281cf3SAxel Dörfler 		get_click_speed(&doubleClickSpeed);
36641281cf3SAxel Dörfler 
36741281cf3SAxel Dörfler 		do {
36841281cf3SAxel Dörfler 			if (fabs(where.x - save.x) > 4 || fabs(where.y - save.y) > 4)
36941281cf3SAxel Dörfler 				// user moved out of bounds of click area
37041281cf3SAxel Dörfler 				break;
37141281cf3SAxel Dörfler 
37241281cf3SAxel Dörfler 			if ((system_time() - start) > (2 * doubleClickSpeed)) {
37341281cf3SAxel Dörfler 				ShowReplicantMenu(where);
37441281cf3SAxel Dörfler 				break;
37541281cf3SAxel Dörfler 			}
37641281cf3SAxel Dörfler 
37741281cf3SAxel Dörfler 			snooze(50000);
37841281cf3SAxel Dörfler 			GetMouse(&where, &buttons);
37941281cf3SAxel Dörfler 		} while (buttons);
38041281cf3SAxel Dörfler 	}
38141281cf3SAxel Dörfler 	BView::MouseDown(where);
38241281cf3SAxel Dörfler }
38341281cf3SAxel Dörfler 
3843cf2d117SJohn Scipione 
3853cf2d117SJohn Scipione void
3863cf2d117SJohn Scipione TReplicantTray::ShowReplicantMenu(BPoint point)
3873cf2d117SJohn Scipione {
3883cf2d117SJohn Scipione 	BPopUpMenu* menu = new BPopUpMenu("", false, false);
3893cf2d117SJohn Scipione 	menu->SetFont(be_plain_font);
3903cf2d117SJohn Scipione 
3913cf2d117SJohn Scipione 	// If clock is visible show the extended menu, otherwise show "Show time"
3923cf2d117SJohn Scipione 
3933cf2d117SJohn Scipione 	if (!fTime->IsHidden())
3943cf2d117SJohn Scipione 		fTime->ShowTimeOptions(ConvertToScreen(point));
3953cf2d117SJohn Scipione 	else {
3963cf2d117SJohn Scipione 		BMenuItem* item = new BMenuItem(B_TRANSLATE("Show time"),
3973cf2d117SJohn Scipione 			new BMessage(kShowHideTime));
3983cf2d117SJohn Scipione 		menu->AddItem(item);
3993cf2d117SJohn Scipione 		menu->SetTargetForItems(this);
4003cf2d117SJohn Scipione 		BPoint where = ConvertToScreen(point);
4013cf2d117SJohn Scipione 		menu->Go(where, true, true, BRect(where - BPoint(4, 4),
4023cf2d117SJohn Scipione 			where + BPoint(4, 4)), true);
4033cf2d117SJohn Scipione 	}
4043cf2d117SJohn Scipione }
4053cf2d117SJohn Scipione 
4063cf2d117SJohn Scipione 
4073cf2d117SJohn Scipione void
4083cf2d117SJohn Scipione TReplicantTray::SetMultiRow(bool state)
4093cf2d117SJohn Scipione {
4103cf2d117SJohn Scipione 	fMultiRowMode = state;
4113cf2d117SJohn Scipione }
4123cf2d117SJohn Scipione 
4133cf2d117SJohn Scipione 
4143cf2d117SJohn Scipione void
4153cf2d117SJohn Scipione TReplicantTray::ShowHideTime()
4163cf2d117SJohn Scipione {
4173cf2d117SJohn Scipione 	if (fTime == NULL)
4183cf2d117SJohn Scipione 		return;
4193cf2d117SJohn Scipione 
4203cf2d117SJohn Scipione 	if (fTime->IsHidden())
4213cf2d117SJohn Scipione 		fTime->Show();
4223cf2d117SJohn Scipione 	else
4233cf2d117SJohn Scipione 		fTime->Hide();
4243cf2d117SJohn Scipione 
4253cf2d117SJohn Scipione 	RealignReplicants();
4263cf2d117SJohn Scipione 	AdjustPlacement();
4273cf2d117SJohn Scipione }
4283cf2d117SJohn Scipione 
4293cf2d117SJohn Scipione 
43041281cf3SAxel Dörfler #ifdef DB_ADDONS
43141281cf3SAxel Dörfler 
4323cf2d117SJohn Scipione 
43341281cf3SAxel Dörfler void
43441281cf3SAxel Dörfler TReplicantTray::InitAddOnSupport()
43541281cf3SAxel Dörfler {
43641281cf3SAxel Dörfler 	// list to maintain refs to each rep added/deleted
43741281cf3SAxel Dörfler 	fItemList = new BList();
43841281cf3SAxel Dörfler 	BPath path;
4391ad8c760SFredrik Holmqvist 
44041281cf3SAxel Dörfler 	if (find_directory(B_USER_SETTINGS_DIRECTORY, &path, true) == B_OK) {
4413e6ff860SRene Gollent 		path.Append(kReplicantSettingsFile);
44241281cf3SAxel Dörfler 
44341281cf3SAxel Dörfler 		BFile file(path.Path(), B_READ_ONLY);
4443e6ff860SRene Gollent 		if (file.InitCheck() == B_OK) {
4453e6ff860SRene Gollent 			status_t result;
4463e6ff860SRene Gollent 			BEntry entry;
4473e6ff860SRene Gollent 			int32 id;
448c0ad1c91SRene Gollent 			BString path;
4493e6ff860SRene Gollent 			if (fAddOnSettings.Unflatten(&file) == B_OK) {
450c0ad1c91SRene Gollent 				for (int32 i = 0; fAddOnSettings.FindString(kReplicantPathField,
451c0ad1c91SRene Gollent 					i, &path) == B_OK; i++) {
452c0ad1c91SRene Gollent 					if (entry.SetTo(path.String()) == B_OK && entry.Exists()) {
4533e6ff860SRene Gollent 						result = LoadAddOn(&entry, &id, false);
4543e6ff860SRene Gollent 					} else
4553e6ff860SRene Gollent 						result = B_ENTRY_NOT_FOUND;
45641281cf3SAxel Dörfler 
4573e6ff860SRene Gollent 					if (result != B_OK) {
458c0ad1c91SRene Gollent 						fAddOnSettings.RemoveData(kReplicantPathField, i);
4593e6ff860SRene Gollent 						--i;
46041281cf3SAxel Dörfler 					}
46141281cf3SAxel Dörfler 				}
46241281cf3SAxel Dörfler 			}
4633e6ff860SRene Gollent 		}
4643e6ff860SRene Gollent 	}
46541281cf3SAxel Dörfler }
46641281cf3SAxel Dörfler 
46741281cf3SAxel Dörfler 
46841281cf3SAxel Dörfler void
46941281cf3SAxel Dörfler TReplicantTray::DeleteAddOnSupport()
47041281cf3SAxel Dörfler {
4717625ce51SRene Gollent 	_SaveSettings();
4723e6ff860SRene Gollent 
47341281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
47441281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->RemoveItem(i);
47541281cf3SAxel Dörfler 		if (item) {
47641281cf3SAxel Dörfler 			if (item->isAddOn)
47741281cf3SAxel Dörfler 				watch_node(&(item->nodeRef), B_STOP_WATCHING, this, Window());
47841281cf3SAxel Dörfler 
47941281cf3SAxel Dörfler 			delete item;
48041281cf3SAxel Dörfler 		}
48141281cf3SAxel Dörfler 	}
48241281cf3SAxel Dörfler 	delete fItemList;
48341281cf3SAxel Dörfler 
48441281cf3SAxel Dörfler 	// stop the volume mount/unmount watch
48541281cf3SAxel Dörfler 	stop_watching(this, Window());
48641281cf3SAxel Dörfler }
48741281cf3SAxel Dörfler 
48841281cf3SAxel Dörfler 
48941281cf3SAxel Dörfler DeskbarItemInfo*
49041281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(node_ref& nodeRef)
49141281cf3SAxel Dörfler {
49241281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
49366eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
49441281cf3SAxel Dörfler 		if (item == NULL)
49541281cf3SAxel Dörfler 			continue;
49641281cf3SAxel Dörfler 
49741281cf3SAxel Dörfler 		if (item->nodeRef == nodeRef)
49841281cf3SAxel Dörfler 			return item;
49941281cf3SAxel Dörfler 	}
50041281cf3SAxel Dörfler 
50141281cf3SAxel Dörfler 	return NULL;
50241281cf3SAxel Dörfler }
50341281cf3SAxel Dörfler 
50441281cf3SAxel Dörfler 
50541281cf3SAxel Dörfler DeskbarItemInfo*
50641281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(int32 id)
50741281cf3SAxel Dörfler {
50841281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
50966eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
51041281cf3SAxel Dörfler 		if (item == NULL)
51141281cf3SAxel Dörfler 			continue;
51241281cf3SAxel Dörfler 
51341281cf3SAxel Dörfler 		if (item->id == id)
51441281cf3SAxel Dörfler 			return item;
51541281cf3SAxel Dörfler 	}
51641281cf3SAxel Dörfler 
51741281cf3SAxel Dörfler 	return NULL;
51841281cf3SAxel Dörfler }
51941281cf3SAxel Dörfler 
52041281cf3SAxel Dörfler 
52141281cf3SAxel Dörfler bool
52241281cf3SAxel Dörfler TReplicantTray::NodeExists(node_ref& nodeRef)
52341281cf3SAxel Dörfler {
52441281cf3SAxel Dörfler 	return DeskbarItemFor(nodeRef) != NULL;
52541281cf3SAxel Dörfler }
52641281cf3SAxel Dörfler 
52741281cf3SAxel Dörfler 
52866eba86fSAxel Dörfler /*! This handles B_NODE_MONITOR & B_QUERY_UPDATE messages received
52966eba86fSAxel Dörfler 	for the registered add-ons.
53041281cf3SAxel Dörfler */
53141281cf3SAxel Dörfler void
53241281cf3SAxel Dörfler TReplicantTray::HandleEntryUpdate(BMessage* message)
53341281cf3SAxel Dörfler {
53441281cf3SAxel Dörfler 	int32 opcode;
53541281cf3SAxel Dörfler 	if (message->FindInt32("opcode", &opcode) != B_OK)
53641281cf3SAxel Dörfler 		return;
53741281cf3SAxel Dörfler 
53841281cf3SAxel Dörfler 	BPath path;
53941281cf3SAxel Dörfler 	switch (opcode) {
54041281cf3SAxel Dörfler 		case B_ENTRY_MOVED:
54141281cf3SAxel Dörfler 		{
54241281cf3SAxel Dörfler 			entry_ref ref;
54341281cf3SAxel Dörfler 			ino_t todirectory;
54441281cf3SAxel Dörfler 			ino_t node;
54541281cf3SAxel Dörfler 			const char* name;
54641281cf3SAxel Dörfler 			if (message->FindString("name", &name) == B_OK
54771bd3ba5SJonas Sundström 				&& message->FindInt64("from directory", &(ref.directory))
54871bd3ba5SJonas Sundström 				== B_OK
54941281cf3SAxel Dörfler 				&& message->FindInt64("to directory", &todirectory) == B_OK
55041281cf3SAxel Dörfler 				&& message->FindInt32("device", &(ref.device)) == B_OK
55141281cf3SAxel Dörfler 				&& message->FindInt64("node", &node) == B_OK ) {
55241281cf3SAxel Dörfler 
55341281cf3SAxel Dörfler 				if (!name)
55441281cf3SAxel Dörfler 					break;
55541281cf3SAxel Dörfler 
55641281cf3SAxel Dörfler 				ref.set_name(name);
55741281cf3SAxel Dörfler 				// change the directory reference to
55841281cf3SAxel Dörfler 				// the new directory
55941281cf3SAxel Dörfler 				MoveItem(&ref, todirectory);
56041281cf3SAxel Dörfler 			}
56141281cf3SAxel Dörfler 			break;
56266eba86fSAxel Dörfler 		}
56341281cf3SAxel Dörfler 
56441281cf3SAxel Dörfler 		case B_ENTRY_REMOVED:
56541281cf3SAxel Dörfler 		{
56641281cf3SAxel Dörfler 			// entry was rm'd from the device
56741281cf3SAxel Dörfler 			node_ref nodeRef;
56841281cf3SAxel Dörfler 			if (message->FindInt32("device", &(nodeRef.device)) == B_OK
56941281cf3SAxel Dörfler 				&& message->FindInt64("node", &(nodeRef.node)) == B_OK) {
57041281cf3SAxel Dörfler 				DeskbarItemInfo* item = DeskbarItemFor(nodeRef);
57141281cf3SAxel Dörfler 				if (item == NULL)
57241281cf3SAxel Dörfler 					break;
57341281cf3SAxel Dörfler 
57441281cf3SAxel Dörfler 				// If there is a team running where the add-on comes from,
57541281cf3SAxel Dörfler 				// we don't want to remove the icon yet.
57641281cf3SAxel Dörfler 				if (be_roster->IsRunning(&item->entryRef))
57741281cf3SAxel Dörfler 					break;
57841281cf3SAxel Dörfler 
57941281cf3SAxel Dörfler 				UnloadAddOn(&nodeRef, NULL, true, false);
58041281cf3SAxel Dörfler 			}
58141281cf3SAxel Dörfler 			break;
58266eba86fSAxel Dörfler 		}
58341281cf3SAxel Dörfler 	}
58466eba86fSAxel Dörfler }
58541281cf3SAxel Dörfler 
58641281cf3SAxel Dörfler 
5871ad8c760SFredrik Holmqvist /*! The add-ons must support the exported C function API
58866eba86fSAxel Dörfler 	if they do, they will be loaded and added to deskbar
58966eba86fSAxel Dörfler 	primary function is the Instantiate function
5907da06231SAxel Dörfler */
59141281cf3SAxel Dörfler status_t
5923e6ff860SRene Gollent TReplicantTray::LoadAddOn(BEntry* entry, int32* id, bool addToSettings)
59341281cf3SAxel Dörfler {
59441281cf3SAxel Dörfler 	if (!entry)
59541281cf3SAxel Dörfler 		return B_ERROR;
59641281cf3SAxel Dörfler 
59741281cf3SAxel Dörfler 	node_ref nodeRef;
59841281cf3SAxel Dörfler 	entry->GetNodeRef(&nodeRef);
59941281cf3SAxel Dörfler 	// no duplicates
60041281cf3SAxel Dörfler 	if (NodeExists(nodeRef))
60141281cf3SAxel Dörfler 		return B_ERROR;
60241281cf3SAxel Dörfler 
60341281cf3SAxel Dörfler 	BNode node(entry);
60441281cf3SAxel Dörfler 	BPath path;
60566eba86fSAxel Dörfler 	status_t status = entry->GetPath(&path);
60666eba86fSAxel Dörfler 	if (status < B_OK)
60766eba86fSAxel Dörfler 		return status;
60841281cf3SAxel Dörfler 
60941281cf3SAxel Dörfler 	// load the add-on
61041281cf3SAxel Dörfler 	image_id image = load_add_on(path.Path());
61166eba86fSAxel Dörfler 	if (image < B_OK)
61266eba86fSAxel Dörfler 		return image;
61341281cf3SAxel Dörfler 
61441281cf3SAxel Dörfler 	// get the view loading function symbol
61541281cf3SAxel Dörfler 	//    we first look for a symbol that takes an image_id
61641281cf3SAxel Dörfler 	//    and entry_ref pointer, if not found, go with normal
61741281cf3SAxel Dörfler 	//    instantiate function
61841281cf3SAxel Dörfler 	BView* (*entryFunction)(image_id, const entry_ref*);
61941281cf3SAxel Dörfler 	BView* (*itemFunction)(void);
62041281cf3SAxel Dörfler 	BView* view = NULL;
62141281cf3SAxel Dörfler 
62241281cf3SAxel Dörfler 	entry_ref ref;
62341281cf3SAxel Dörfler 	entry->GetRef(&ref);
62441281cf3SAxel Dörfler 
62541281cf3SAxel Dörfler 	if (get_image_symbol(image, kInstantiateEntryCFunctionName,
62666eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&entryFunction) >= B_OK) {
62741281cf3SAxel Dörfler 		view = (*entryFunction)(image, &ref);
62841281cf3SAxel Dörfler 	} else if (get_image_symbol(image, kInstantiateItemCFunctionName,
62966eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&itemFunction) >= B_OK) {
63041281cf3SAxel Dörfler 		view = (*itemFunction)();
63141281cf3SAxel Dörfler 	} else {
63241281cf3SAxel Dörfler 		unload_add_on(image);
63341281cf3SAxel Dörfler 		return B_ERROR;
63441281cf3SAxel Dörfler 	}
63541281cf3SAxel Dörfler 
63666eba86fSAxel Dörfler 	if (view == NULL || IconExists(view->Name())) {
63741281cf3SAxel Dörfler 		delete view;
63841281cf3SAxel Dörfler 		unload_add_on(image);
63941281cf3SAxel Dörfler 		return B_ERROR;
64041281cf3SAxel Dörfler 	}
64141281cf3SAxel Dörfler 
64241281cf3SAxel Dörfler 	BMessage* data = new BMessage;
64341281cf3SAxel Dörfler 	view->Archive(data);
64441281cf3SAxel Dörfler 	delete view;
64541281cf3SAxel Dörfler 
64641281cf3SAxel Dörfler 	AddIcon(data, id, &ref);
64741281cf3SAxel Dörfler 		// add the rep; adds info to list
64841281cf3SAxel Dörfler 
6493e6ff860SRene Gollent 	if (addToSettings) {
650c0ad1c91SRene Gollent 		fAddOnSettings.AddString(kReplicantPathField, path.Path());
6517625ce51SRene Gollent 		_SaveSettings();
6523e6ff860SRene Gollent 	}
65341281cf3SAxel Dörfler 
65441281cf3SAxel Dörfler 	return B_OK;
65541281cf3SAxel Dörfler }
65641281cf3SAxel Dörfler 
65741281cf3SAxel Dörfler 
65841281cf3SAxel Dörfler status_t
65941281cf3SAxel Dörfler TReplicantTray::AddItem(int32 id, node_ref nodeRef, BEntry& entry, bool isAddOn)
66041281cf3SAxel Dörfler {
66141281cf3SAxel Dörfler 	DeskbarItemInfo* item = new DeskbarItemInfo;
66241281cf3SAxel Dörfler 	if (item == NULL)
66341281cf3SAxel Dörfler 		return B_NO_MEMORY;
66441281cf3SAxel Dörfler 
66541281cf3SAxel Dörfler 	item->id = id;
66641281cf3SAxel Dörfler 	item->isAddOn = isAddOn;
66741281cf3SAxel Dörfler 
66841281cf3SAxel Dörfler 	if (entry.GetRef(&item->entryRef) < B_OK) {
66941281cf3SAxel Dörfler 		item->entryRef.device = -1;
67041281cf3SAxel Dörfler 		item->entryRef.directory = -1;
67141281cf3SAxel Dörfler 		item->entryRef.name = NULL;
67241281cf3SAxel Dörfler 	}
67341281cf3SAxel Dörfler 	item->nodeRef = nodeRef;
67441281cf3SAxel Dörfler 
67541281cf3SAxel Dörfler 	fItemList->AddItem(item);
67641281cf3SAxel Dörfler 
67741281cf3SAxel Dörfler 	if (isAddOn)
67841281cf3SAxel Dörfler 		watch_node(&nodeRef, B_WATCH_NAME | B_WATCH_ATTR, this, Window());
67941281cf3SAxel Dörfler 
68041281cf3SAxel Dörfler 	return B_OK;
68141281cf3SAxel Dörfler }
68241281cf3SAxel Dörfler 
68341281cf3SAxel Dörfler 
68441281cf3SAxel Dörfler /**	from entry_removed message, when attribute removed
68541281cf3SAxel Dörfler  *	or when a device is unmounted (use removeall, by device)
68641281cf3SAxel Dörfler  */
68741281cf3SAxel Dörfler 
68841281cf3SAxel Dörfler void
68941281cf3SAxel Dörfler TReplicantTray::UnloadAddOn(node_ref* nodeRef, dev_t* device,
69041281cf3SAxel Dörfler 	bool which, bool removeAll)
69141281cf3SAxel Dörfler {
69241281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
69341281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
69441281cf3SAxel Dörfler 		if (!item)
69541281cf3SAxel Dörfler 			continue;
69641281cf3SAxel Dörfler 
69741281cf3SAxel Dörfler 		if ((which && nodeRef && item->nodeRef == *nodeRef)
69841281cf3SAxel Dörfler 			|| (device && item->nodeRef.device == *device)) {
69941281cf3SAxel Dörfler 
70041281cf3SAxel Dörfler 			if (device && be_roster->IsRunning(&item->entryRef))
70141281cf3SAxel Dörfler 				continue;
70241281cf3SAxel Dörfler 
70341281cf3SAxel Dörfler 			RemoveIcon(item->id);
70441281cf3SAxel Dörfler 
70541281cf3SAxel Dörfler 			if (!removeAll)
70641281cf3SAxel Dörfler 				break;
70741281cf3SAxel Dörfler 		}
70841281cf3SAxel Dörfler 	}
70941281cf3SAxel Dörfler }
71041281cf3SAxel Dörfler 
71141281cf3SAxel Dörfler 
71241281cf3SAxel Dörfler void
71341281cf3SAxel Dörfler TReplicantTray::RemoveItem(int32 id)
71441281cf3SAxel Dörfler {
71541281cf3SAxel Dörfler 	DeskbarItemInfo* item = DeskbarItemFor(id);
71641281cf3SAxel Dörfler 	if (item == NULL)
71741281cf3SAxel Dörfler 		return;
71841281cf3SAxel Dörfler 
71941281cf3SAxel Dörfler 	// attribute was added via Deskbar API (AddItem(entry_ref*, int32*)
72041281cf3SAxel Dörfler 	if (item->isAddOn) {
7217625ce51SRene Gollent 		BPath path(&item->entryRef);
7227625ce51SRene Gollent 		BString storedPath;
7237625ce51SRene Gollent 		for (int32 i = 0;
7248aab28f1SRene Gollent 			fAddOnSettings.FindString(kReplicantPathField, i, &storedPath)
7257625ce51SRene Gollent 				== B_OK; i++) {
7267625ce51SRene Gollent 			if (storedPath == path.Path()) {
7278aab28f1SRene Gollent 				fAddOnSettings.RemoveData(kReplicantPathField, i);
7287625ce51SRene Gollent 				break;
7297625ce51SRene Gollent 			}
7307625ce51SRene Gollent 		}
7317625ce51SRene Gollent 		_SaveSettings();
7327625ce51SRene Gollent 
73341281cf3SAxel Dörfler 		BNode node(&item->entryRef);
73441281cf3SAxel Dörfler 		watch_node(&item->nodeRef, B_STOP_WATCHING, this, Window());
73541281cf3SAxel Dörfler 	}
73641281cf3SAxel Dörfler 
73741281cf3SAxel Dörfler 	fItemList->RemoveItem(item);
73841281cf3SAxel Dörfler 	delete item;
73941281cf3SAxel Dörfler }
74041281cf3SAxel Dörfler 
74141281cf3SAxel Dörfler 
74241281cf3SAxel Dörfler /**	ENTRY_MOVED message, moving only occurs on a device
74341281cf3SAxel Dörfler  *	copying will occur (ENTRY_CREATED) between devices
74441281cf3SAxel Dörfler  */
74541281cf3SAxel Dörfler 
74641281cf3SAxel Dörfler void
74741281cf3SAxel Dörfler TReplicantTray::MoveItem(entry_ref* ref, ino_t toDirectory)
74841281cf3SAxel Dörfler {
74941281cf3SAxel Dörfler 	if (!ref)
75041281cf3SAxel Dörfler 		return;
75141281cf3SAxel Dörfler 
75241281cf3SAxel Dörfler 	// scan for a matching entry_ref and update it
75341281cf3SAxel Dörfler 	//
75441281cf3SAxel Dörfler 	// don't need to change node info as it does not change
75541281cf3SAxel Dörfler 
75641281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
75741281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
75841281cf3SAxel Dörfler 		if (!item)
75941281cf3SAxel Dörfler 			continue;
76041281cf3SAxel Dörfler 
76141281cf3SAxel Dörfler 		if (!strcmp(item->entryRef.name, ref->name)
76241281cf3SAxel Dörfler 			&& item->entryRef.device == ref->device
76341281cf3SAxel Dörfler 			&& item->entryRef.directory == ref->directory) {
76441281cf3SAxel Dörfler 			item->entryRef.directory = toDirectory;
76541281cf3SAxel Dörfler 			break;
76641281cf3SAxel Dörfler 		}
76741281cf3SAxel Dörfler 	}
76841281cf3SAxel Dörfler }
76941281cf3SAxel Dörfler 
77041281cf3SAxel Dörfler #endif // add-on support
77141281cf3SAxel Dörfler 
77241281cf3SAxel Dörfler //	external add-on API routines
77341281cf3SAxel Dörfler //	called using the new BDeskbar class
77441281cf3SAxel Dörfler 
77541281cf3SAxel Dörfler //	existence of icon/replicant by name or ID
77641281cf3SAxel Dörfler //	returns opposite
77741281cf3SAxel Dörfler //	note: name and id are semi-private limiting
77841281cf3SAxel Dörfler //		the ability of non-host apps to remove
77941281cf3SAxel Dörfler //		icons without a little bit of work
78041281cf3SAxel Dörfler 
78141281cf3SAxel Dörfler /**	for a specific id
78241281cf3SAxel Dörfler  *	return the name of the replicant (name of view)
78341281cf3SAxel Dörfler  */
78441281cf3SAxel Dörfler 
78541281cf3SAxel Dörfler status_t
78641281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 id, const char** name)
78741281cf3SAxel Dörfler {
78841281cf3SAxel Dörfler 	if (id < 0)
78941281cf3SAxel Dörfler 		return B_ERROR;
79041281cf3SAxel Dörfler 
79141281cf3SAxel Dörfler 	int32 index, temp;
79241281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &temp, id, false);
79341281cf3SAxel Dörfler 	if (view) {
79441281cf3SAxel Dörfler 		*name = view->Name();
79541281cf3SAxel Dörfler 		return B_OK;
79641281cf3SAxel Dörfler 	}
79741281cf3SAxel Dörfler 
79841281cf3SAxel Dörfler 	return B_ERROR;
79941281cf3SAxel Dörfler }
80041281cf3SAxel Dörfler 
80141281cf3SAxel Dörfler 
8027da06231SAxel Dörfler /**	for a specific name
8037da06231SAxel Dörfler  *	return the id (internal to Deskbar)
8047da06231SAxel Dörfler  */
8057da06231SAxel Dörfler 
80641281cf3SAxel Dörfler status_t
80741281cf3SAxel Dörfler TReplicantTray::ItemInfo(const char* name, int32* id)
80841281cf3SAxel Dörfler {
80941281cf3SAxel Dörfler 	if (!name || strlen(name) <= 0)
81041281cf3SAxel Dörfler 		return B_ERROR;
81141281cf3SAxel Dörfler 
81241281cf3SAxel Dörfler 	int32 index;
81341281cf3SAxel Dörfler 	BView* view = ViewAt(&index, id, name);
81441281cf3SAxel Dörfler 	if (view)
81541281cf3SAxel Dörfler 		return B_OK;
81641281cf3SAxel Dörfler 
81741281cf3SAxel Dörfler 	return B_ERROR;
81841281cf3SAxel Dörfler }
81941281cf3SAxel Dörfler 
82041281cf3SAxel Dörfler 
8217da06231SAxel Dörfler /**	at a specific index
8227da06231SAxel Dörfler  *	return both the name and the id of the replicant
8237da06231SAxel Dörfler  */
8247da06231SAxel Dörfler 
82541281cf3SAxel Dörfler status_t
82641281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 index, const char** name, int32* id)
82741281cf3SAxel Dörfler {
82841281cf3SAxel Dörfler 	if (index < 0)
82941281cf3SAxel Dörfler 		return B_ERROR;
83041281cf3SAxel Dörfler 
83141281cf3SAxel Dörfler 	BView* view;
83241281cf3SAxel Dörfler 	fShelf->ReplicantAt(index, &view, (uint32*)id, NULL);
83341281cf3SAxel Dörfler 	if (view) {
83441281cf3SAxel Dörfler 		*name = view->Name();
83541281cf3SAxel Dörfler 		return B_OK;
83641281cf3SAxel Dörfler 	}
83741281cf3SAxel Dörfler 
83841281cf3SAxel Dörfler 	return B_ERROR;
83941281cf3SAxel Dörfler }
84041281cf3SAxel Dörfler 
84141281cf3SAxel Dörfler 
8427da06231SAxel Dörfler /**	replicant exists, by id/index */
8437da06231SAxel Dörfler 
84441281cf3SAxel Dörfler bool
84541281cf3SAxel Dörfler TReplicantTray::IconExists(int32 target, bool byIndex)
84641281cf3SAxel Dörfler {
84741281cf3SAxel Dörfler 	int32 index, id;
84841281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
84941281cf3SAxel Dörfler 
85041281cf3SAxel Dörfler 	return view && index >= 0;
85141281cf3SAxel Dörfler }
85241281cf3SAxel Dörfler 
85341281cf3SAxel Dörfler 
8547da06231SAxel Dörfler /**	replicant exists, by name */
8557da06231SAxel Dörfler 
85641281cf3SAxel Dörfler bool
85741281cf3SAxel Dörfler TReplicantTray::IconExists(const char* name)
85841281cf3SAxel Dörfler {
85941281cf3SAxel Dörfler 	if (!name || strlen(name) == 0)
86041281cf3SAxel Dörfler 		return false;
86141281cf3SAxel Dörfler 
86241281cf3SAxel Dörfler 	int32 index, id;
86341281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
86441281cf3SAxel Dörfler 
86541281cf3SAxel Dörfler 	return view && index >= 0;
86641281cf3SAxel Dörfler }
86741281cf3SAxel Dörfler 
86841281cf3SAxel Dörfler 
86941281cf3SAxel Dörfler int32
87041281cf3SAxel Dörfler TReplicantTray::IconCount() const
87141281cf3SAxel Dörfler {
87241281cf3SAxel Dörfler 	return fShelf->CountReplicants();
87341281cf3SAxel Dörfler }
87441281cf3SAxel Dörfler 
87541281cf3SAxel Dörfler 
876ced3755cSAxel Dörfler /*! Message must contain an archivable view for later rehydration.
877ced3755cSAxel Dörfler 	This function takes over ownership of the provided message on success
878ced3755cSAxel Dörfler 	only.
879ced3755cSAxel Dörfler 	Returns the current replicant ID.
8807da06231SAxel Dörfler */
88141281cf3SAxel Dörfler status_t
882ced3755cSAxel Dörfler TReplicantTray::AddIcon(BMessage* archive, int32* id, const entry_ref* addOn)
88341281cf3SAxel Dörfler {
884ced3755cSAxel Dörfler 	if (archive == NULL || id == NULL)
88541281cf3SAxel Dörfler 		return B_ERROR;
88641281cf3SAxel Dörfler 
8879c50f36eSAxel Dörfler 	// find entry_ref
8889c50f36eSAxel Dörfler 
8899c50f36eSAxel Dörfler 	entry_ref ref;
8909c50f36eSAxel Dörfler 	if (addOn) {
8919c50f36eSAxel Dörfler 		// Use it if we got it
8929c50f36eSAxel Dörfler 		ref = *addOn;
8939c50f36eSAxel Dörfler 	} else {
8949c50f36eSAxel Dörfler 		const char* signature;
8951ad8c760SFredrik Holmqvist 
896ced3755cSAxel Dörfler 		status_t status = archive->FindString("add_on", &signature);
8979c50f36eSAxel Dörfler 		if (status == B_OK) {
8989c50f36eSAxel Dörfler 			BRoster roster;
8999c50f36eSAxel Dörfler 			status = roster.FindApp(signature, &ref);
9009c50f36eSAxel Dörfler 		}
9019c50f36eSAxel Dörfler 		if (status < B_OK)
9029c50f36eSAxel Dörfler 			return status;
9039c50f36eSAxel Dörfler 	}
9049c50f36eSAxel Dörfler 
9059c50f36eSAxel Dörfler 	BFile file;
9069c50f36eSAxel Dörfler 	status_t status = file.SetTo(&ref, B_READ_ONLY);
9079c50f36eSAxel Dörfler 	if (status < B_OK)
9089c50f36eSAxel Dörfler 		return status;
9099c50f36eSAxel Dörfler 
9109c50f36eSAxel Dörfler 	node_ref nodeRef;
9119c50f36eSAxel Dörfler 	status = file.GetNodeRef(&nodeRef);
9129c50f36eSAxel Dörfler 	if (status < B_OK)
9139c50f36eSAxel Dörfler 		return status;
9149c50f36eSAxel Dörfler 
9159c50f36eSAxel Dörfler 	BEntry entry(&ref, true);
9161ad8c760SFredrik Holmqvist 		// TODO: this resolves an eventual link for the item being added - this
9171ad8c760SFredrik Holmqvist 		// is okay for now, but in multi-user environments, one might want to
9181ad8c760SFredrik Holmqvist 		// have links that carry the be:deskbar_item_status attribute
9199c50f36eSAxel Dörfler 	status = entry.InitCheck();
9209c50f36eSAxel Dörfler 	if (status != B_OK)
9219c50f36eSAxel Dörfler 		return status;
9229c50f36eSAxel Dörfler 
92341281cf3SAxel Dörfler 	*id = 999;
924ced3755cSAxel Dörfler 	if (archive->what == B_ARCHIVED_OBJECT)
925ced3755cSAxel Dörfler 		archive->what = 0;
92641281cf3SAxel Dörfler 
927ced3755cSAxel Dörfler 	BRect originalBounds = archive->FindRect("_frame");
9281ad8c760SFredrik Holmqvist 		// this is a work-around for buggy replicants that change their size in
9291ad8c760SFredrik Holmqvist 		// AttachedToWindow() (such as "SVM")
930a5210ab1SAxel Dörfler 
9311ad8c760SFredrik Holmqvist 	// TODO: check for name collisions?
932ced3755cSAxel Dörfler 	status = fShelf->AddReplicant(archive, BPoint(1, 1));
9339c50f36eSAxel Dörfler 	if (status != B_OK)
9349c50f36eSAxel Dörfler 		return status;
93541281cf3SAxel Dörfler 
936a5210ab1SAxel Dörfler 	int32 count = fShelf->CountReplicants();
937a5210ab1SAxel Dörfler 	BView* view;
938a5210ab1SAxel Dörfler 	fShelf->ReplicantAt(count - 1, &view, (uint32*)id, NULL);
939a5210ab1SAxel Dörfler 
940a5210ab1SAxel Dörfler 	if (originalBounds != view->Bounds()) {
941a5210ab1SAxel Dörfler 		// The replicant changed its size when added to the window, so we need
94271bd3ba5SJonas Sundström 		// to recompute all over again (it's already done once via
94371bd3ba5SJonas Sundström 		// BShelf::AddReplicant() and TReplicantShelf::CanAcceptReplicantView())
944a5210ab1SAxel Dörfler 		RealignReplicants();
945a5210ab1SAxel Dörfler 	}
946a5210ab1SAxel Dörfler 
94741281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
94841281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
94941281cf3SAxel Dörfler 	float width, height;
95041281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
95141281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height)
95241281cf3SAxel Dörfler 		AdjustPlacement();
95341281cf3SAxel Dörfler 
95441281cf3SAxel Dörfler 	// add the item to the add-on list
95541281cf3SAxel Dörfler 
95641281cf3SAxel Dörfler 	AddItem(*id, nodeRef, entry, addOn != NULL);
95741281cf3SAxel Dörfler 	return B_OK;
95841281cf3SAxel Dörfler }
95941281cf3SAxel Dörfler 
96041281cf3SAxel Dörfler 
96141281cf3SAxel Dörfler void
96241281cf3SAxel Dörfler TReplicantTray::RemoveIcon(int32 target, bool byIndex)
96341281cf3SAxel Dörfler {
96441281cf3SAxel Dörfler 	if (target < 0)
96541281cf3SAxel Dörfler 		return;
96641281cf3SAxel Dörfler 
96741281cf3SAxel Dörfler 	int32 index, id;
96841281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
96941281cf3SAxel Dörfler 	if (view && index >= 0) {
9707da06231SAxel Dörfler 		// remove the reference from the item list & the shelf
97141281cf3SAxel Dörfler 		RemoveItem(id);
97241281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9737da06231SAxel Dörfler 
97441281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
97541281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
97641281cf3SAxel Dörfler 	}
97741281cf3SAxel Dörfler }
97841281cf3SAxel Dörfler 
97941281cf3SAxel Dörfler 
98041281cf3SAxel Dörfler void
98141281cf3SAxel Dörfler TReplicantTray::RemoveIcon(const char* name)
98241281cf3SAxel Dörfler {
98341281cf3SAxel Dörfler 	if (!name || strlen(name) <= 0)
98441281cf3SAxel Dörfler 		return;
98541281cf3SAxel Dörfler 
98641281cf3SAxel Dörfler 	int32 id, index;
98741281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
98841281cf3SAxel Dörfler 	if (view && index >= 0) {
9897da06231SAxel Dörfler 		// remove the reference from the item list & shelf
99041281cf3SAxel Dörfler 		RemoveItem(id);
99141281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9927da06231SAxel Dörfler 
99341281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
99441281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
99541281cf3SAxel Dörfler 	}
99641281cf3SAxel Dörfler }
99741281cf3SAxel Dörfler 
99841281cf3SAxel Dörfler 
99941281cf3SAxel Dörfler void
10007da06231SAxel Dörfler TReplicantTray::RealReplicantAdjustment(int32 startIndex)
100141281cf3SAxel Dörfler {
10027da06231SAxel Dörfler 	if (startIndex < 0)
100341281cf3SAxel Dörfler 		return;
10047da06231SAxel Dörfler 
10057da06231SAxel Dörfler 	if (startIndex == fLastReplicant)
10067da06231SAxel Dörfler 		startIndex = 0;
10077da06231SAxel Dörfler 
100841281cf3SAxel Dörfler 	// reset the locations of all replicants after the one deleted
10097da06231SAxel Dörfler 	RealignReplicants(startIndex);
101041281cf3SAxel Dörfler 
101141281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
101241281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
101341281cf3SAxel Dörfler 	float width, height;
101441281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
101541281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height) {
10161ad8c760SFredrik Holmqvist 		// resize view to accomodate the replicants, redraw as necessary
101741281cf3SAxel Dörfler 		AdjustPlacement();
101841281cf3SAxel Dörfler 	}
101941281cf3SAxel Dörfler }
102041281cf3SAxel Dörfler 
102141281cf3SAxel Dörfler 
10227da06231SAxel Dörfler /**	looking for a replicant by id/index
10237da06231SAxel Dörfler  *	return the view and index
10247da06231SAxel Dörfler  */
10257da06231SAxel Dörfler 
102641281cf3SAxel Dörfler BView*
102741281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, int32 target, bool byIndex)
102841281cf3SAxel Dörfler {
102941281cf3SAxel Dörfler 	*index = -1;
103041281cf3SAxel Dörfler 
103141281cf3SAxel Dörfler 	BView* view;
103241281cf3SAxel Dörfler 	if (byIndex) {
103341281cf3SAxel Dörfler 		if (fShelf->ReplicantAt(target, &view, (uint32*)id)) {
103441281cf3SAxel Dörfler 			if (view) {
103541281cf3SAxel Dörfler 				*index = target;
103641281cf3SAxel Dörfler 				return view;
103741281cf3SAxel Dörfler 			}
103841281cf3SAxel Dörfler 		}
103941281cf3SAxel Dörfler 	} else {
104041281cf3SAxel Dörfler 		int32 count = fShelf->CountReplicants() - 1;
104141281cf3SAxel Dörfler 		int32 localid;
104241281cf3SAxel Dörfler 		for (int32 repIndex = count ; repIndex >= 0 ; repIndex--) {
104341281cf3SAxel Dörfler 			fShelf->ReplicantAt(repIndex, &view, (uint32*)&localid);
104441281cf3SAxel Dörfler 			if (localid == target && view) {
104541281cf3SAxel Dörfler 				*index = repIndex;
104641281cf3SAxel Dörfler 				*id = localid;
104741281cf3SAxel Dörfler 				return view;
104841281cf3SAxel Dörfler 			}
104941281cf3SAxel Dörfler 		}
105041281cf3SAxel Dörfler 	}
105141281cf3SAxel Dörfler 	return NULL;
105241281cf3SAxel Dörfler }
105341281cf3SAxel Dörfler 
105441281cf3SAxel Dörfler 
10557da06231SAxel Dörfler /**	looking for a replicant with a view by name
10567da06231SAxel Dörfler  *	return the view, index and the id of the replicant
10577da06231SAxel Dörfler  */
10587da06231SAxel Dörfler 
105941281cf3SAxel Dörfler BView*
106041281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, const char* name)
106141281cf3SAxel Dörfler {
106241281cf3SAxel Dörfler 	*index = -1;
106341281cf3SAxel Dörfler 	*id = -1;
106441281cf3SAxel Dörfler 
106541281cf3SAxel Dörfler 	BView* view;
106641281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants()-1;
106741281cf3SAxel Dörfler 	for (int32 repIndex = count ; repIndex >= 0 ; repIndex--) {
106841281cf3SAxel Dörfler 		fShelf->ReplicantAt(repIndex, &view, (uint32*)id);
106941281cf3SAxel Dörfler 		if (view && view->Name() && strcmp(name, view->Name()) == 0) {
107041281cf3SAxel Dörfler 			*index = repIndex;
107141281cf3SAxel Dörfler 			return view;
107241281cf3SAxel Dörfler 		}
107341281cf3SAxel Dörfler 	}
107441281cf3SAxel Dörfler 	return NULL;
107541281cf3SAxel Dörfler }
107641281cf3SAxel Dörfler 
107741281cf3SAxel Dörfler 
10787da06231SAxel Dörfler /**	Shelf will call to determine where and if
10797da06231SAxel Dörfler  *	the replicant is to be added
10807da06231SAxel Dörfler  */
10817da06231SAxel Dörfler 
108241281cf3SAxel Dörfler bool
108341281cf3SAxel Dörfler TReplicantTray::AcceptAddon(BRect replicantFrame, BMessage* message)
108441281cf3SAxel Dörfler {
108541281cf3SAxel Dörfler 	if (!message)
108641281cf3SAxel Dörfler 		return false;
108741281cf3SAxel Dörfler 
108841281cf3SAxel Dörfler 	if (replicantFrame.Height() > kMaxReplicantHeight)
108941281cf3SAxel Dörfler 		return false;
109041281cf3SAxel Dörfler 
109141281cf3SAxel Dörfler 	alignment align = B_ALIGN_LEFT;
109241281cf3SAxel Dörfler 	if (fAlignmentSupport && message->HasBool("deskbar:dynamic_align")) {
109341281cf3SAxel Dörfler 		if (!fBarView->Vertical())
109441281cf3SAxel Dörfler 			align = B_ALIGN_RIGHT;
109541281cf3SAxel Dörfler 		else
109641281cf3SAxel Dörfler 			align = fBarView->Left() ? B_ALIGN_LEFT : B_ALIGN_RIGHT;
109741281cf3SAxel Dörfler 	} else if (message->HasInt32("deskbar:align"))
109841281cf3SAxel Dörfler 		message->FindInt32("deskbar:align", (int32*)&align);
109941281cf3SAxel Dörfler 
110041281cf3SAxel Dörfler 	if (message->HasInt32("deskbar:private_align"))
110141281cf3SAxel Dörfler 		message->FindInt32("deskbar:private_align", (int32*)&align);
110241281cf3SAxel Dörfler 	else
110341281cf3SAxel Dörfler 		align = B_ALIGN_LEFT;
110441281cf3SAxel Dörfler 
11057da06231SAxel Dörfler 	BPoint loc = LocationForReplicant(fShelf->CountReplicants(),
11067da06231SAxel Dörfler 		replicantFrame.Width());
110741281cf3SAxel Dörfler 
110841281cf3SAxel Dörfler 	message->AddPoint("_pjp_loc", loc);
110941281cf3SAxel Dörfler 	return true;
111041281cf3SAxel Dörfler }
111141281cf3SAxel Dörfler 
111241281cf3SAxel Dörfler 
111341281cf3SAxel Dörfler /**	based on the previous (index - 1) replicant in the list
111441281cf3SAxel Dörfler  *	calculate where the left point should be for this
111541281cf3SAxel Dörfler  *	replicant.  replicant will flow to the right on its own
111641281cf3SAxel Dörfler  */
111741281cf3SAxel Dörfler 
111841281cf3SAxel Dörfler BPoint
11197da06231SAxel Dörfler TReplicantTray::LocationForReplicant(int32 index, float width)
112041281cf3SAxel Dörfler {
112141281cf3SAxel Dörfler 	BPoint loc(kIconGap + 1, kGutter + 1);
112241281cf3SAxel Dörfler 
11237da06231SAxel Dörfler 	if (fMultiRowMode) {
11247da06231SAxel Dörfler 		// try to find free space in every row
11257da06231SAxel Dörfler 		for (int32 row = 0; ; loc.y += kMaxReplicantHeight + kIconGap, row++) {
11267da06231SAxel Dörfler 			// determine free space in this row
11272ce9bab8SJohn Scipione 			BRect rect(loc.x, loc.y, loc.x + fMinimumTrayWidth - kIconGap
11282ce9bab8SJohn Scipione 				- 2.0, loc.y + kMaxReplicantHeight);
1129573f748cSJohn Scipione 			if (row == 0 && !fTime->IsHidden())
1130573f748cSJohn Scipione 				rect.right -= fTime->Frame().Width() + kIconGap;
11317da06231SAxel Dörfler 
11327da06231SAxel Dörfler 			for (int32 i = 0; i < index; i++) {
11337da06231SAxel Dörfler 				BView* view = NULL;
11347da06231SAxel Dörfler 				fShelf->ReplicantAt(i, &view);
11357da06231SAxel Dörfler 				if (view == NULL || view->Frame().top != rect.top)
11367da06231SAxel Dörfler 					continue;
11377da06231SAxel Dörfler 
11387da06231SAxel Dörfler 				rect.left = view->Frame().right + kIconGap + 1;
11397da06231SAxel Dörfler 			}
11407da06231SAxel Dörfler 
11417da06231SAxel Dörfler 			if (rect.Width() >= width) {
11427da06231SAxel Dörfler 				// the icon fits in this row
11437da06231SAxel Dörfler 				loc = rect.LeftTop();
11447da06231SAxel Dörfler 				break;
11457da06231SAxel Dörfler 			}
11467da06231SAxel Dörfler 		}
11477da06231SAxel Dörfler 	} else {
114841281cf3SAxel Dörfler 		if (index > 0) {
114941281cf3SAxel Dörfler 			// get the last replicant added for placement reference
115041281cf3SAxel Dörfler 			BView* view = NULL;
11517da06231SAxel Dörfler 			fShelf->ReplicantAt(index - 1, &view);
115241281cf3SAxel Dörfler 			if (view) {
115341281cf3SAxel Dörfler 				// push this rep placement past the last one
115441281cf3SAxel Dörfler 				loc.x = view->Frame().right + kIconGap + 1;
115541281cf3SAxel Dörfler 				loc.y = view->Frame().top;
115641281cf3SAxel Dörfler 			}
115741281cf3SAxel Dörfler 		}
11587da06231SAxel Dörfler 	}
115941281cf3SAxel Dörfler 
116071bd3ba5SJonas Sundström 	if ((loc.y == fRightBottomReplicant.top && loc.x
116171bd3ba5SJonas Sundström 		> fRightBottomReplicant.left) || loc.y > fRightBottomReplicant.top) {
116271bd3ba5SJonas Sundström 		fRightBottomReplicant.Set(loc.x, loc.y, loc.x + width, loc.y
116371bd3ba5SJonas Sundström 		+ kMaxReplicantHeight);
11647da06231SAxel Dörfler 		fLastReplicant = index;
116541281cf3SAxel Dörfler 	}
11667da06231SAxel Dörfler 
116741281cf3SAxel Dörfler 	return loc;
116841281cf3SAxel Dörfler }
116941281cf3SAxel Dörfler 
117041281cf3SAxel Dörfler 
117141281cf3SAxel Dörfler BRect
117241281cf3SAxel Dörfler TReplicantTray::IconFrame(int32 target, bool byIndex)
117341281cf3SAxel Dörfler {
117441281cf3SAxel Dörfler 	int32 index, id;
117541281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
117641281cf3SAxel Dörfler 	if (view)
117741281cf3SAxel Dörfler 		return view->Frame();
117841281cf3SAxel Dörfler 
117941281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
118041281cf3SAxel Dörfler }
118141281cf3SAxel Dörfler 
118241281cf3SAxel Dörfler 
118341281cf3SAxel Dörfler BRect
118441281cf3SAxel Dörfler TReplicantTray::IconFrame(const char* name)
118541281cf3SAxel Dörfler {
118641281cf3SAxel Dörfler 	if (!name)
118741281cf3SAxel Dörfler 		return BRect(0, 0, 0, 0);
118841281cf3SAxel Dörfler 
118941281cf3SAxel Dörfler 	int32 id, index;
119041281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
119141281cf3SAxel Dörfler 	if (view)
119241281cf3SAxel Dörfler 		return view->Frame();
119341281cf3SAxel Dörfler 
119441281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
119541281cf3SAxel Dörfler }
119641281cf3SAxel Dörfler 
119741281cf3SAxel Dörfler 
11987da06231SAxel Dörfler /**	Scan from the startIndex and reset the location
11997da06231SAxel Dörfler  *	as defined in LocationForReplicant()
120041281cf3SAxel Dörfler  */
120141281cf3SAxel Dörfler 
120241281cf3SAxel Dörfler void
120341281cf3SAxel Dörfler TReplicantTray::RealignReplicants(int32 startIndex)
120441281cf3SAxel Dörfler {
120541281cf3SAxel Dörfler 	if (startIndex < 0)
120641281cf3SAxel Dörfler 		startIndex = 0;
120741281cf3SAxel Dörfler 
120841281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants();
120941281cf3SAxel Dörfler 	if (count <= 0)
121041281cf3SAxel Dörfler 		return;
121141281cf3SAxel Dörfler 
12127da06231SAxel Dörfler 	if (startIndex == 0)
12137da06231SAxel Dörfler 		fRightBottomReplicant.Set(0, 0, 0, 0);
12147da06231SAxel Dörfler 
121541281cf3SAxel Dörfler 	BView* view = NULL;
121641281cf3SAxel Dörfler 	for (int32 i = startIndex ; i < count ; i++) {
121741281cf3SAxel Dörfler 		fShelf->ReplicantAt(i, &view);
12180bec7100SStefano Ceccherini 		if (view != NULL) {
12197da06231SAxel Dörfler 			BPoint loc = LocationForReplicant(i, view->Frame().Width());
12200bec7100SStefano Ceccherini 			if (view->Frame().LeftTop() != loc)
122141281cf3SAxel Dörfler 				view->MoveTo(loc);
122241281cf3SAxel Dörfler 		}
122341281cf3SAxel Dörfler 	}
122441281cf3SAxel Dörfler }
122541281cf3SAxel Dörfler 
122641281cf3SAxel Dörfler 
12278aab28f1SRene Gollent status_t
12287625ce51SRene Gollent TReplicantTray::_SaveSettings()
12297625ce51SRene Gollent {
12307625ce51SRene Gollent 	status_t result;
12317625ce51SRene Gollent 	BPath path;
12327625ce51SRene Gollent 	if ((result = find_directory(B_USER_SETTINGS_DIRECTORY, &path, true))
12337625ce51SRene Gollent 		 == B_OK) {
12347625ce51SRene Gollent 		path.Append(kReplicantSettingsFile);
12357625ce51SRene Gollent 
12367625ce51SRene Gollent 		BFile file(path.Path(), B_READ_WRITE | B_CREATE_FILE | B_ERASE_FILE);
12377625ce51SRene Gollent 		if ((result = file.InitCheck()) == B_OK)
12387625ce51SRene Gollent 			result = fAddOnSettings.Flatten(&file);
12397625ce51SRene Gollent 	}
12407625ce51SRene Gollent 
12417625ce51SRene Gollent 	return result;
12427625ce51SRene Gollent }
12437625ce51SRene Gollent 
12447625ce51SRene Gollent 
12453cf2d117SJohn Scipione void
12463cf2d117SJohn Scipione TReplicantTray::SaveTimeSettings()
12473cf2d117SJohn Scipione {
12483cf2d117SJohn Scipione 	if (fTime == NULL)
12493cf2d117SJohn Scipione 		return;
12503cf2d117SJohn Scipione 
12513cf2d117SJohn Scipione 	desk_settings* settings = ((TBarApp*)be_app)->Settings();
12523cf2d117SJohn Scipione 	settings->showTime = !fTime->IsHidden();
12533cf2d117SJohn Scipione 	settings->showSeconds = fTime->ShowSeconds();
12543cf2d117SJohn Scipione 	settings->showDayOfWeek = fTime->ShowDayOfWeek();
12553cf2d117SJohn Scipione 	settings->showTimeZone = fTime->ShowTimeZone();
12563cf2d117SJohn Scipione }
12573cf2d117SJohn Scipione 
12583cf2d117SJohn Scipione 
125941281cf3SAxel Dörfler //	#pragma mark -
126041281cf3SAxel Dörfler 
126141281cf3SAxel Dörfler 
1262ca9acc20SAxel Dörfler /*! Draggable region that is asynchronous so that dragging does not block
1263ca9acc20SAxel Dörfler 	other activities.
12647da06231SAxel Dörfler */
126541281cf3SAxel Dörfler TDragRegion::TDragRegion(TBarView* parent, BView* child)
1266ca9acc20SAxel Dörfler 	:
1267ca9acc20SAxel Dörfler 	BControl(BRect(0, 0, 0, 0), "", "", NULL, B_FOLLOW_NONE,
126841281cf3SAxel Dörfler 		B_WILL_DRAW | B_FRAME_EVENTS),
126941281cf3SAxel Dörfler 	fBarView(parent),
127041281cf3SAxel Dörfler 	fChild(child),
127141281cf3SAxel Dörfler 	fDragLocation(kAutoPlaceDragRegion)
127241281cf3SAxel Dörfler {
127341281cf3SAxel Dörfler }
127441281cf3SAxel Dörfler 
127541281cf3SAxel Dörfler 
127641281cf3SAxel Dörfler void
127741281cf3SAxel Dörfler TDragRegion::AttachedToWindow()
127841281cf3SAxel Dörfler {
127941281cf3SAxel Dörfler 	BView::AttachedToWindow();
1280cb6afcb1SStephan Aßmus 	if (be_control_look != NULL)
1281cb6afcb1SStephan Aßmus 		SetViewColor(tint_color(ui_color(B_MENU_BACKGROUND_COLOR), 1.1));
1282cb6afcb1SStephan Aßmus 	else
128341281cf3SAxel Dörfler 		SetViewColor(ui_color(B_MENU_BACKGROUND_COLOR));
128441281cf3SAxel Dörfler 	ResizeToPreferred();
128541281cf3SAxel Dörfler }
128641281cf3SAxel Dörfler 
128741281cf3SAxel Dörfler 
128841281cf3SAxel Dörfler void
128941281cf3SAxel Dörfler TDragRegion::GetPreferredSize(float* width, float* height)
129041281cf3SAxel Dörfler {
129141281cf3SAxel Dörfler 	fChild->ResizeToPreferred();
129241281cf3SAxel Dörfler 	*width = fChild->Bounds().Width();
129341281cf3SAxel Dörfler 	*height = fChild->Bounds().Height();
129441281cf3SAxel Dörfler 
129541281cf3SAxel Dörfler 	if (fDragLocation != kNoDragRegion)
129641281cf3SAxel Dörfler 		*width += 7;
129741281cf3SAxel Dörfler 	else
129841281cf3SAxel Dörfler 		*width += 6;
129941281cf3SAxel Dörfler 
130041281cf3SAxel Dörfler 	*height += 3;
130141281cf3SAxel Dörfler }
130241281cf3SAxel Dörfler 
130341281cf3SAxel Dörfler 
130441281cf3SAxel Dörfler void
130541281cf3SAxel Dörfler TDragRegion::FrameMoved(BPoint)
130641281cf3SAxel Dörfler {
1307ca9acc20SAxel Dörfler 	if (fBarView->Left() && fBarView->Vertical()
1308ca9acc20SAxel Dörfler 		&& fDragLocation != kNoDragRegion)
130941281cf3SAxel Dörfler 		fChild->MoveTo(5, 2);
131041281cf3SAxel Dörfler 	else
131141281cf3SAxel Dörfler 		fChild->MoveTo(2, 2);
131241281cf3SAxel Dörfler }
131341281cf3SAxel Dörfler 
131441281cf3SAxel Dörfler 
131541281cf3SAxel Dörfler void
131641281cf3SAxel Dörfler TDragRegion::Draw(BRect)
131741281cf3SAxel Dörfler {
1318eaa9af99SStephan Aßmus 	rgb_color menuColor = ViewColor();
131941281cf3SAxel Dörfler 	rgb_color hilite = tint_color(menuColor, B_DARKEN_1_TINT);
1320cb6afcb1SStephan Aßmus 	rgb_color ldark = tint_color(menuColor, 1.02);
1321cb6afcb1SStephan Aßmus 	rgb_color dark = tint_color(menuColor, B_DARKEN_2_TINT);
132241281cf3SAxel Dörfler 	rgb_color vvdark = tint_color(menuColor, B_DARKEN_4_TINT);
132341281cf3SAxel Dörfler 	rgb_color light = tint_color(menuColor, B_LIGHTEN_2_TINT);
132441281cf3SAxel Dörfler 
132541281cf3SAxel Dörfler 	BRect frame(Bounds());
132641281cf3SAxel Dörfler 	BeginLineArray(4);
132741281cf3SAxel Dörfler 
1328cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1329cb6afcb1SStephan Aßmus 		if (fBarView->Vertical()) {
1330cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(), frame.RightTop(), dark);
1331cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1332cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.top + 1), ldark);
1333cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
1334cb6afcb1SStephan Aßmus 		} else if (fBarView->AcrossTop() || fBarView->AcrossBottom()) {
1335cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(),
1336cb6afcb1SStephan Aßmus 				BPoint(frame.left, frame.bottom), dark);
1337cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.top + 1),
1338cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
1339cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right, frame.top + 2),
1340cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.bottom), hilite);
1341cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.bottom),
1342cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
1343cb6afcb1SStephan Aßmus 		}
1344cb6afcb1SStephan Aßmus 	} else {
134541281cf3SAxel Dörfler 		if (fBarView->Vertical()) {
134641281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.RightTop(), light);
134741281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.LeftBottom(), light);
134841281cf3SAxel Dörfler 			AddLine(frame.RightBottom(), frame.RightTop(), hilite);
134941281cf3SAxel Dörfler 		} else if (fBarView->AcrossTop()) {
1350cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1351cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
135241281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1353cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 2),
1354cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1355cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(),
1356cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
135741281cf3SAxel Dörfler 		} else if (fBarView->AcrossBottom()) {
1358cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1359cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
136041281cf3SAxel Dörfler 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
136141281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1362cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 1),
1363cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1364cb6afcb1SStephan Aßmus 		}
136541281cf3SAxel Dörfler 	}
136641281cf3SAxel Dörfler 
136741281cf3SAxel Dörfler 	EndLineArray();
136841281cf3SAxel Dörfler 
136941281cf3SAxel Dörfler 	if (fDragLocation != kDontDrawDragRegion || fDragLocation != kNoDragRegion)
137041281cf3SAxel Dörfler 		DrawDragRegion();
137141281cf3SAxel Dörfler }
137241281cf3SAxel Dörfler 
137341281cf3SAxel Dörfler 
137441281cf3SAxel Dörfler void
137541281cf3SAxel Dörfler TDragRegion::DrawDragRegion()
137641281cf3SAxel Dörfler {
137741281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
137841281cf3SAxel Dörfler 
1379cb6afcb1SStephan Aßmus 	rgb_color menuColor = ViewColor();
1380cb6afcb1SStephan Aßmus 	rgb_color menuHilite = menuColor;
138141281cf3SAxel Dörfler 	if (IsTracking()) {
1382cb6afcb1SStephan Aßmus 		// Draw drag region highlighted if tracking mouse
1383cb6afcb1SStephan Aßmus 		menuHilite = tint_color(menuColor, B_HIGHLIGHT_BACKGROUND_TINT);
138441281cf3SAxel Dörfler 		SetHighColor(menuHilite);
138541281cf3SAxel Dörfler 		FillRect(dragRegion);
138641281cf3SAxel Dörfler 	}
1387cb6afcb1SStephan Aßmus 	rgb_color vdark = tint_color(menuHilite, B_DARKEN_3_TINT);
1388cb6afcb1SStephan Aßmus 	rgb_color light = tint_color(menuHilite, B_LIGHTEN_2_TINT);
1389cb6afcb1SStephan Aßmus 
1390cb6afcb1SStephan Aßmus 	BeginLineArray(dragRegion.IntegerHeight());
1391cb6afcb1SStephan Aßmus 	BPoint pt;
1392cb6afcb1SStephan Aßmus 	pt.x = floorf((dragRegion.left + dragRegion.right) / 2 + 0.5) - 1;
1393cb6afcb1SStephan Aßmus 	pt.y = dragRegion.top + 2;
1394cb6afcb1SStephan Aßmus 
139541281cf3SAxel Dörfler 	while (pt.y + 1 <= dragRegion.bottom) {
139641281cf3SAxel Dörfler 		AddLine(pt, pt, vdark);
139741281cf3SAxel Dörfler 		AddLine(pt + BPoint(1, 1), pt + BPoint(1, 1), light);
139841281cf3SAxel Dörfler 
139941281cf3SAxel Dörfler 		pt.y += 3;
140041281cf3SAxel Dörfler 	}
140141281cf3SAxel Dörfler 	EndLineArray();
140241281cf3SAxel Dörfler }
140341281cf3SAxel Dörfler 
140441281cf3SAxel Dörfler 
140541281cf3SAxel Dörfler BRect
140641281cf3SAxel Dörfler TDragRegion::DragRegion() const
140741281cf3SAxel Dörfler {
1408cb6afcb1SStephan Aßmus 	float kTopBottomInset = 2;
1409cb6afcb1SStephan Aßmus 	float kLeftRightInset = 1;
1410cb6afcb1SStephan Aßmus 	float kDragWidth = 3;
1411cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1412cb6afcb1SStephan Aßmus 		kTopBottomInset = 1;
1413cb6afcb1SStephan Aßmus 		kLeftRightInset = 0;
1414cb6afcb1SStephan Aßmus 		kDragWidth = 4;
1415cb6afcb1SStephan Aßmus 	}
1416cb6afcb1SStephan Aßmus 
141741281cf3SAxel Dörfler 	BRect dragRegion(Bounds());
1418cb6afcb1SStephan Aßmus 	dragRegion.top += kTopBottomInset;
1419cb6afcb1SStephan Aßmus 	dragRegion.bottom -= kTopBottomInset;
142041281cf3SAxel Dörfler 
142141281cf3SAxel Dörfler 	bool placeOnLeft = false;
142241281cf3SAxel Dörfler 	if (fDragLocation == kAutoPlaceDragRegion) {
142341281cf3SAxel Dörfler 		if (fBarView->Vertical() && fBarView->Left())
142441281cf3SAxel Dörfler 			placeOnLeft = true;
142541281cf3SAxel Dörfler 		else
142641281cf3SAxel Dörfler 			placeOnLeft = false;
142741281cf3SAxel Dörfler 	} else if (fDragLocation == kDragRegionLeft)
142841281cf3SAxel Dörfler 		placeOnLeft = true;
142941281cf3SAxel Dörfler 	else if (fDragLocation == kDragRegionRight)
143041281cf3SAxel Dörfler 		placeOnLeft = false;
143141281cf3SAxel Dörfler 
143241281cf3SAxel Dörfler 	if (placeOnLeft) {
1433cb6afcb1SStephan Aßmus 		dragRegion.left += kLeftRightInset;
1434cb6afcb1SStephan Aßmus 		dragRegion.right = dragRegion.left + kDragWidth;
143541281cf3SAxel Dörfler 	} else {
1436cb6afcb1SStephan Aßmus 		dragRegion.right -= kLeftRightInset;
1437cb6afcb1SStephan Aßmus 		dragRegion.left = dragRegion.right - kDragWidth;
143841281cf3SAxel Dörfler 	}
143941281cf3SAxel Dörfler 
144041281cf3SAxel Dörfler 	return dragRegion;
144141281cf3SAxel Dörfler }
144241281cf3SAxel Dörfler 
144341281cf3SAxel Dörfler 
144441281cf3SAxel Dörfler void
144541281cf3SAxel Dörfler TDragRegion::MouseDown(BPoint thePoint)
144641281cf3SAxel Dörfler {
144741281cf3SAxel Dörfler 	ulong buttons;
144841281cf3SAxel Dörfler 	BPoint where;
144941281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
145041281cf3SAxel Dörfler 
14517da06231SAxel Dörfler 	dragRegion.InsetBy(-2.0f, -2.0f);
14527da06231SAxel Dörfler 		// DragRegion() is designed for drawing, not clicking
145341281cf3SAxel Dörfler 
145441281cf3SAxel Dörfler 	if (!dragRegion.Contains(thePoint))
145541281cf3SAxel Dörfler 		return;
145641281cf3SAxel Dörfler 
145741281cf3SAxel Dörfler 	while (true) {
145841281cf3SAxel Dörfler 		GetMouse(&where, &buttons);
145941281cf3SAxel Dörfler 		if (!buttons)
146041281cf3SAxel Dörfler 			break;
146141281cf3SAxel Dörfler 
146241281cf3SAxel Dörfler 		if ((Window()->Flags() & B_ASYNCHRONOUS_CONTROLS) != 0) {
146341281cf3SAxel Dörfler 			fPreviousPosition = thePoint;
146441281cf3SAxel Dörfler 			SetTracking(true);
146541281cf3SAxel Dörfler 			SetMouseEventMask(B_POINTER_EVENTS,
146641281cf3SAxel Dörfler 				B_NO_POINTER_HISTORY | B_LOCK_WINDOW_FOCUS);
146741281cf3SAxel Dörfler 			Invalidate(DragRegion());
146841281cf3SAxel Dörfler 			break;
146941281cf3SAxel Dörfler 		}
147041281cf3SAxel Dörfler 
147141281cf3SAxel Dörfler 		snooze(25000);
147241281cf3SAxel Dörfler 	}
147341281cf3SAxel Dörfler }
147441281cf3SAxel Dörfler 
147541281cf3SAxel Dörfler 
147641281cf3SAxel Dörfler void
147741281cf3SAxel Dörfler TDragRegion::MouseUp(BPoint pt)
147841281cf3SAxel Dörfler {
147941281cf3SAxel Dörfler 	if (IsTracking()) {
148041281cf3SAxel Dörfler 		SetTracking(false);
148141281cf3SAxel Dörfler 		Invalidate(DragRegion());
148241281cf3SAxel Dörfler 	} else
148341281cf3SAxel Dörfler 		BControl::MouseUp(pt);
148441281cf3SAxel Dörfler }
148541281cf3SAxel Dörfler 
148641281cf3SAxel Dörfler 
148741281cf3SAxel Dörfler bool
148841281cf3SAxel Dörfler TDragRegion::SwitchModeForRect(BPoint mouse, BRect rect,
148941281cf3SAxel Dörfler 	bool newVertical, bool newLeft, bool newTop, int32 newState)
149041281cf3SAxel Dörfler {
14911ad8c760SFredrik Holmqvist 	if (!rect.Contains(mouse)) {
149241281cf3SAxel Dörfler 		// not our rect
149341281cf3SAxel Dörfler 		return false;
14941ad8c760SFredrik Holmqvist 	}
149541281cf3SAxel Dörfler 
14961ad8c760SFredrik Holmqvist 	if (newVertical == fBarView->Vertical() && newLeft == fBarView->Left()
14971ad8c760SFredrik Holmqvist 		&& newTop == fBarView->Top() && newState == fBarView->State()) {
149841281cf3SAxel Dörfler 		// already in the correct mode
149941281cf3SAxel Dörfler 		return true;
15001ad8c760SFredrik Holmqvist 	}
150141281cf3SAxel Dörfler 
150241281cf3SAxel Dörfler 	fBarView->ChangeState(newState, newVertical, newLeft, newTop);
150341281cf3SAxel Dörfler 	return true;
150441281cf3SAxel Dörfler }
150541281cf3SAxel Dörfler 
150641281cf3SAxel Dörfler 
150741281cf3SAxel Dörfler void
150841281cf3SAxel Dörfler TDragRegion::MouseMoved(BPoint where, uint32 code, const BMessage* message)
150941281cf3SAxel Dörfler {
151041281cf3SAxel Dörfler 	if (IsTracking()) {
151141281cf3SAxel Dörfler 		BScreen screen;
151241281cf3SAxel Dörfler 		BRect frame = screen.Frame();
151341281cf3SAxel Dörfler 
151441281cf3SAxel Dörfler 		float hDivider = frame.Width() / 6;
151571bd3ba5SJonas Sundström 		hDivider = (hDivider < sMinimumWindowWidth + 10.0f)
151671bd3ba5SJonas Sundström 			? sMinimumWindowWidth + 10.0f : hDivider;
151741281cf3SAxel Dörfler 		float miniDivider = frame.top + kMiniHeight + 10.0f;
151841281cf3SAxel Dörfler 		float vDivider = frame.Height() / 2;
151941281cf3SAxel Dörfler #ifdef FULL_MODE
152041281cf3SAxel Dörfler 		float thirdScreen = frame.Height() / 3;
152141281cf3SAxel Dörfler #endif
152271bd3ba5SJonas Sundström 		BRect topLeft(frame.left, frame.top, frame.left + hDivider,
152371bd3ba5SJonas Sundström 			miniDivider);
152471bd3ba5SJonas Sundström 		BRect topMiddle(frame.left + hDivider, frame.top, frame.right
152571bd3ba5SJonas Sundström 			- hDivider, vDivider);
152671bd3ba5SJonas Sundström 		BRect topRight(frame.right - hDivider, frame.top, frame.right,
152771bd3ba5SJonas Sundström 			miniDivider);
152841281cf3SAxel Dörfler 
152941281cf3SAxel Dörfler #ifdef FULL_MODE
153041281cf3SAxel Dörfler 		vDivider = miniDivider + thirdScreen;
153141281cf3SAxel Dörfler #endif
153271bd3ba5SJonas Sundström 		BRect middleLeft(frame.left, miniDivider, frame.left + hDivider,
153371bd3ba5SJonas Sundström 			vDivider);
153471bd3ba5SJonas Sundström 		BRect middleRight(frame.right - hDivider, miniDivider, frame.right,
153571bd3ba5SJonas Sundström 			vDivider);
153641281cf3SAxel Dörfler 
153741281cf3SAxel Dörfler #ifdef FULL_MODE
153871bd3ba5SJonas Sundström 		BRect leftSide(frame.left, vDivider, frame.left + hDivider,
153971bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
154071bd3ba5SJonas Sundström 		BRect rightSide(frame.right - hDivider, vDivider, frame.right,
154171bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
154241281cf3SAxel Dörfler 
154341281cf3SAxel Dörfler 		vDivider = frame.bottom - thirdScreen;
154441281cf3SAxel Dörfler #endif
154571bd3ba5SJonas Sundström 		BRect bottomLeft(frame.left, vDivider, frame.left + hDivider,
154671bd3ba5SJonas Sundström 			frame.bottom);
154771bd3ba5SJonas Sundström 		BRect bottomMiddle(frame.left + hDivider, vDivider, frame.right
154871bd3ba5SJonas Sundström 			- hDivider, frame.bottom);
154971bd3ba5SJonas Sundström 		BRect bottomRight(frame.right - hDivider, vDivider, frame.right,
155071bd3ba5SJonas Sundström 			frame.bottom);
155141281cf3SAxel Dörfler 
155241281cf3SAxel Dörfler 		if (where != fPreviousPosition) {
155341281cf3SAxel Dörfler 			fPreviousPosition = where;
155441281cf3SAxel Dörfler 			ConvertToScreen(&where);
155541281cf3SAxel Dörfler 
155641281cf3SAxel Dörfler 			// use short circuit evaluation for convenience
155741281cf3SAxel Dörfler 			if (SwitchModeForRect(where, topLeft, true, true, true, kMiniState)
155871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topMiddle, false, true, true,
155971bd3ba5SJonas Sundström 					kExpandoState)
156071bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topRight, true, false, true,
156171bd3ba5SJonas Sundström 					kMiniState)
156271bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleLeft, true, true, true,
156371bd3ba5SJonas Sundström 					kExpandoState)
156471bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleRight, true, false, true,
156571bd3ba5SJonas Sundström 					kExpandoState)
156641281cf3SAxel Dörfler 
156741281cf3SAxel Dörfler #ifdef FULL_MODE
156871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, leftSide, true, true, true,
156971bd3ba5SJonas Sundström 					kFullState)
157071bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, rightSide, true, false, true,
157171bd3ba5SJonas Sundström 					kFullState)
157241281cf3SAxel Dörfler #endif
157371bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomLeft, true, true, false,
157471bd3ba5SJonas Sundström 					kMiniState)
157571bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomMiddle, false, true, false,
157671bd3ba5SJonas Sundström 					kExpandoState)
157771bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomRight, true, false, false,
157871bd3ba5SJonas Sundström 					kMiniState))
157941281cf3SAxel Dörfler 				;
158041281cf3SAxel Dörfler 		}
158141281cf3SAxel Dörfler 	} else
158241281cf3SAxel Dörfler 		BControl::MouseMoved(where, code, message);
158341281cf3SAxel Dörfler }
158441281cf3SAxel Dörfler 
158541281cf3SAxel Dörfler 
158641281cf3SAxel Dörfler int32
158741281cf3SAxel Dörfler TDragRegion::DragRegionLocation() const
158841281cf3SAxel Dörfler {
158941281cf3SAxel Dörfler 	return fDragLocation;
159041281cf3SAxel Dörfler }
159141281cf3SAxel Dörfler 
159241281cf3SAxel Dörfler 
159341281cf3SAxel Dörfler void
159441281cf3SAxel Dörfler TDragRegion::SetDragRegionLocation(int32 location)
159541281cf3SAxel Dörfler {
159641281cf3SAxel Dörfler 	if (location == fDragLocation)
159741281cf3SAxel Dörfler 		return;
159841281cf3SAxel Dörfler 
159941281cf3SAxel Dörfler 	fDragLocation = location;
160041281cf3SAxel Dörfler 	Invalidate();
160141281cf3SAxel Dörfler }
160241281cf3SAxel Dörfler 
1603