xref: /haiku/src/apps/deskbar/StatusView.cpp (revision c3c3360f40a6d4fc4045d02065de2abef819b0da)
141281cf3SAxel Dörfler /*
241281cf3SAxel Dörfler Open Tracker License
341281cf3SAxel Dörfler 
441281cf3SAxel Dörfler Terms and Conditions
541281cf3SAxel Dörfler 
641281cf3SAxel Dörfler Copyright (c) 1991-2000, Be Incorporated. All rights reserved.
741281cf3SAxel Dörfler 
841281cf3SAxel Dörfler Permission is hereby granted, free of charge, to any person obtaining a copy of
941281cf3SAxel Dörfler this software and associated documentation files (the "Software"), to deal in
1041281cf3SAxel Dörfler the Software without restriction, including without limitation the rights to
1141281cf3SAxel Dörfler use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
1241281cf3SAxel Dörfler of the Software, and to permit persons to whom the Software is furnished to do
1341281cf3SAxel Dörfler so, subject to the following conditions:
1441281cf3SAxel Dörfler 
1541281cf3SAxel Dörfler The above copyright notice and this permission notice applies to all licensees
1641281cf3SAxel Dörfler and shall be included in all copies or substantial portions of the Software.
1741281cf3SAxel Dörfler 
1841281cf3SAxel Dörfler THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1941281cf3SAxel Dörfler IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF TITLE, MERCHANTABILITY,
2041281cf3SAxel Dörfler FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2141281cf3SAxel Dörfler BE INCORPORATED BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
2241281cf3SAxel Dörfler AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF, OR IN CONNECTION
2341281cf3SAxel Dörfler WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
2441281cf3SAxel Dörfler 
2541281cf3SAxel Dörfler Except as contained in this notice, the name of Be Incorporated shall not be
2641281cf3SAxel Dörfler used in advertising or otherwise to promote the sale, use or other dealings in
2741281cf3SAxel Dörfler this Software without prior written authorization from Be Incorporated.
2841281cf3SAxel Dörfler 
291ad8c760SFredrik Holmqvist Tracker(TM), Be(R), BeOS(R), and BeIA(TM) are trademarks or registered
301ad8c760SFredrik Holmqvist trademarks of Be Incorporated in the United States and other countries. Other
311ad8c760SFredrik Holmqvist brand product names are registered trademarks or trademarks of their respective
321ad8c760SFredrik Holmqvist holders.
3341281cf3SAxel Dörfler All rights reserved.
3441281cf3SAxel Dörfler */
3541281cf3SAxel Dörfler 
361cd61330SJohn Scipione 
371cd61330SJohn Scipione #include "StatusView.h"
3841281cf3SAxel Dörfler 
3941281cf3SAxel Dörfler #include <errno.h>
4041281cf3SAxel Dörfler #include <stdio.h>
4141281cf3SAxel Dörfler #include <string.h>
4241281cf3SAxel Dörfler #include <time.h>
4341281cf3SAxel Dörfler #include <unistd.h>
4441281cf3SAxel Dörfler #include <algorithm>
4541281cf3SAxel Dörfler 
4641281cf3SAxel Dörfler #include <fs_index.h>
4741281cf3SAxel Dörfler #include <fs_info.h>
4841281cf3SAxel Dörfler 
4941281cf3SAxel Dörfler #include <Application.h>
5041281cf3SAxel Dörfler #include <Beep.h>
51cb6afcb1SStephan Aßmus #include <Bitmap.h>
52ae7a6e3aSAdrien Destugues #include <Catalog.h>
53cb6afcb1SStephan Aßmus #include <ControlLook.h>
541cd61330SJohn Scipione #include <Debug.h>
5541281cf3SAxel Dörfler #include <Directory.h>
5641281cf3SAxel Dörfler #include <FindDirectory.h>
57ae7a6e3aSAdrien Destugues #include <Locale.h>
5841281cf3SAxel Dörfler #include <MenuItem.h>
5941281cf3SAxel Dörfler #include <NodeInfo.h>
6041281cf3SAxel Dörfler #include <NodeMonitor.h>
6141281cf3SAxel Dörfler #include <Path.h>
6241281cf3SAxel Dörfler #include <PopUpMenu.h>
6341281cf3SAxel Dörfler #include <Roster.h>
6441281cf3SAxel Dörfler #include <Screen.h>
6541281cf3SAxel Dörfler #include <Volume.h>
6641281cf3SAxel Dörfler #include <VolumeRoster.h>
6741281cf3SAxel Dörfler #include <Window.h>
6841281cf3SAxel Dörfler 
69615d572dSJohn Scipione #include "icons.h"
70615d572dSJohn Scipione 
71cb6afcb1SStephan Aßmus #include "BarApp.h"
72323b6546SOliver Tappe #include "DeskbarUtils.h"
73cb6afcb1SStephan Aßmus #include "ResourceSet.h"
7441281cf3SAxel Dörfler #include "StatusViewShelf.h"
7541281cf3SAxel Dörfler #include "TimeView.h"
7641281cf3SAxel Dörfler 
77615d572dSJohn Scipione 
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";
8726ea7066SIngo Weinhold const char* const kReplicantSettingsFile = "replicants";
88c0ad1c91SRene Gollent const char* const kReplicantPathField = "replicant_path";
8941281cf3SAxel Dörfler 
90b49e806dSAxel Dörfler float gMinimumWindowWidth = kGutter + kMinimumTrayWidth + kDragRegionWidth;
911bb2e623SJohn Scipione float gMaximumWindowWidth = gMinimumWindowWidth * 2;
9241281cf3SAxel Dörfler 
93*c3c3360fSJohn Scipione 
9441281cf3SAxel Dörfler static void
9541281cf3SAxel Dörfler DumpItem(DeskbarItemInfo* item)
9641281cf3SAxel Dörfler {
97e9632898SAlex Smith 	printf("is addon: %i, id: %" B_PRId32 "\n", item->isAddOn, item->id);
98e9632898SAlex Smith 	printf("entry_ref:  %" B_PRIdDEV ", %" B_PRIdINO ", %s\n",
99e9632898SAlex Smith 		item->entryRef.device, item->entryRef.directory, item->entryRef.name);
100e9632898SAlex Smith 	printf("node_ref:  %" B_PRIdDEV ", %" B_PRIdINO "\n", item->nodeRef.device,
101e9632898SAlex Smith 		item->nodeRef.node);
10241281cf3SAxel Dörfler }
10341281cf3SAxel Dörfler 
10441281cf3SAxel Dörfler 
10541281cf3SAxel Dörfler static void
10641281cf3SAxel Dörfler DumpList(BList* itemlist)
10741281cf3SAxel Dörfler {
10841281cf3SAxel Dörfler 	int32 count = itemlist->CountItems() - 1;
10941281cf3SAxel Dörfler 	if (count < 0) {
11041281cf3SAxel Dörfler 		printf("no items in list\n");
11141281cf3SAxel Dörfler 		return;
11241281cf3SAxel Dörfler 	}
11341281cf3SAxel Dörfler 	for (int32 i = count; i >= 0; i--) {
11441281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)itemlist->ItemAt(i);
11541281cf3SAxel Dörfler 		if (!item)
11641281cf3SAxel Dörfler 			continue;
11741281cf3SAxel Dörfler 
11841281cf3SAxel Dörfler 		DumpItem(item);
11941281cf3SAxel Dörfler 	}
12041281cf3SAxel Dörfler }
12141281cf3SAxel Dörfler #endif	/* DB_ADDONS */
12241281cf3SAxel Dörfler 
12341281cf3SAxel Dörfler 
124546208a5SOliver Tappe #undef B_TRANSLATION_CONTEXT
125546208a5SOliver Tappe #define B_TRANSLATION_CONTEXT "Tray"
126ae7a6e3aSAdrien Destugues 
1277da06231SAxel Dörfler // don't change the name of this view to anything other than "Status"!
1287da06231SAxel Dörfler 
12941281cf3SAxel Dörfler TReplicantTray::TReplicantTray(TBarView* parent, bool vertical)
1300d2ac94cSJohn Scipione 	:
1319ac6fee1SJohn Scipione 	BView(BRect(0, 0, 1, 1), "Status", B_FOLLOW_LEFT | B_FOLLOW_TOP,
1320d2ac94cSJohn Scipione 		B_WILL_DRAW | B_FRAME_EVENTS),
133573f748cSJohn Scipione 	fTime(NULL),
13441281cf3SAxel Dörfler 	fBarView(parent),
13541281cf3SAxel Dörfler 	fShelf(new TReplicantShelf(this)),
13641281cf3SAxel Dörfler 	fMultiRowMode(vertical),
13741281cf3SAxel Dörfler 	fAlignmentSupport(false)
13841281cf3SAxel Dörfler {
139fb956f25SAxel Dörfler 	if (vertical)
140fb956f25SAxel Dörfler 		fMinimumTrayWidth = gMinimumWindowWidth - kGutter - kDragRegionWidth;
1411bb2e623SJohn Scipione 	else
1421bb2e623SJohn Scipione 		fMinimumTrayWidth = kMinimumTrayWidth;
143573f748cSJohn Scipione 
144573f748cSJohn Scipione 	// Create the time view
14531c0024dSJohn Scipione 	fTime = new TTimeView(fMinimumTrayWidth, kMaxReplicantHeight - 1.0);
14641281cf3SAxel Dörfler }
14741281cf3SAxel Dörfler 
14841281cf3SAxel Dörfler 
14941281cf3SAxel Dörfler TReplicantTray::~TReplicantTray()
15041281cf3SAxel Dörfler {
15141281cf3SAxel Dörfler 	delete fShelf;
152573f748cSJohn Scipione 	delete fTime;
15341281cf3SAxel Dörfler }
15441281cf3SAxel Dörfler 
15541281cf3SAxel Dörfler 
15641281cf3SAxel Dörfler void
15741281cf3SAxel Dörfler TReplicantTray::AttachedToWindow()
15841281cf3SAxel Dörfler {
15941281cf3SAxel Dörfler 	BView::AttachedToWindow();
16041281cf3SAxel Dörfler 
161cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
162cef0a7b2Slooncraz 		AdoptParentColors();
163cb6afcb1SStephan Aßmus 	} else {
16416c10517Slooncraz 		SetViewUIColor(B_MENU_BACKGROUND_COLOR,	B_DARKEN_1_TINT);
165cb6afcb1SStephan Aßmus 	}
16641281cf3SAxel Dörfler 	SetDrawingMode(B_OP_COPY);
16766eba86fSAxel Dörfler 
16841281cf3SAxel Dörfler 	Window()->SetPulseRate(1000000);
16941281cf3SAxel Dörfler 
1703722e640SJohn Scipione 	clock_settings* clock = ((TBarApp*)be_app)->ClockSettings();
1713722e640SJohn Scipione 	fTime->SetShowSeconds(clock->showSeconds);
1723722e640SJohn Scipione 	fTime->SetShowDayOfWeek(clock->showDayOfWeek);
1733722e640SJohn Scipione 	fTime->SetShowTimeZone(clock->showTimeZone);
17431c0024dSJohn Scipione 
175573f748cSJohn Scipione 	AddChild(fTime);
1761bb2e623SJohn Scipione 
17770ad5289SJohn Scipione 	fTime->MoveTo(Bounds().right - fTime->Bounds().Width() - kTrayPadding, 2);
178ae7a6e3aSAdrien Destugues 
1793722e640SJohn Scipione 	if (!((TBarApp*)be_app)->Settings()->showClock)
1803722e640SJohn Scipione 		fTime->Hide();
1813722e640SJohn Scipione 
18241281cf3SAxel Dörfler #ifdef DB_ADDONS
18341281cf3SAxel Dörfler 	// load addons and rehydrate archives
1843a6add49SAxel Dörfler #if !defined(HAIKU_TARGET_PLATFORM_LIBBE_TEST)
18541281cf3SAxel Dörfler 	InitAddOnSupport();
18641281cf3SAxel Dörfler #endif
1879c910f9eSStephan Aßmus #endif
18841281cf3SAxel Dörfler 	ResizeToPreferred();
18941281cf3SAxel Dörfler }
19041281cf3SAxel Dörfler 
19141281cf3SAxel Dörfler 
19241281cf3SAxel Dörfler void
19341281cf3SAxel Dörfler TReplicantTray::DetachedFromWindow()
19441281cf3SAxel Dörfler {
19541281cf3SAxel Dörfler #ifdef DB_ADDONS
19641281cf3SAxel Dörfler 	// clean up add-on support
197465b54cdSAxel Dörfler #if !defined(HAIKU_TARGET_PLATFORM_LIBBE_TEST)
19841281cf3SAxel Dörfler 	DeleteAddOnSupport();
19941281cf3SAxel Dörfler #endif
200465b54cdSAxel Dörfler #endif
20141281cf3SAxel Dörfler 	BView::DetachedFromWindow();
20241281cf3SAxel Dörfler }
20341281cf3SAxel Dörfler 
20441281cf3SAxel Dörfler 
2051ad8c760SFredrik Holmqvist /*! Width is set to a minimum of kMinimumReplicantCount by kMaxReplicantWidth
20666eba86fSAxel Dörfler 	if not in multirowmode and greater than kMinimumReplicantCount
2072ce9bab8SJohn Scipione 	the width should be calculated based on the actual replicant widths
20841281cf3SAxel Dörfler */
20941281cf3SAxel Dörfler void
21041281cf3SAxel Dörfler TReplicantTray::GetPreferredSize(float* preferredWidth, float* preferredHeight)
21141281cf3SAxel Dörfler {
2121bb2e623SJohn Scipione 	float width = 0;
213e4d65fe7SJohn Scipione 	float height = kMinimumTrayHeight;
21441281cf3SAxel Dörfler 
21541281cf3SAxel Dörfler 	if (fMultiRowMode) {
2161bb2e623SJohn Scipione 		width = static_cast<TBarApp*>(be_app)->Settings()->width
2171bb2e623SJohn Scipione 			- kDragWidth - kGutter;
2187da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0)
2197da06231SAxel Dörfler 			height = fRightBottomReplicant.bottom;
2201ad8c760SFredrik Holmqvist 		// the height will be uniform for the number of rows necessary to show
2211ad8c760SFredrik Holmqvist 		// all the reps + any gutters necessary for spacing
22241281cf3SAxel Dörfler 		int32 rowCount = (int32)(height / kMaxReplicantHeight);
22341281cf3SAxel Dörfler 		height = kGutter + (rowCount * kMaxReplicantHeight)
22441281cf3SAxel Dörfler 			+ ((rowCount - 1) * kIconGap) + kGutter;
2251bb2e623SJohn Scipione 		height = std::max(kMinimumTrayHeight, height);
22641281cf3SAxel Dörfler 	} else {
2271ad8c760SFredrik Holmqvist 		// if last replicant overruns clock then resize to accomodate
2287da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0) {
229e4d65fe7SJohn Scipione 			if (!fTime->IsHidden()
2301bb2e623SJohn Scipione 				&& fTime->Frame().left < fRightBottomReplicant.right + 12) {
2311bb2e623SJohn Scipione 				width = fRightBottomReplicant.right + 12
232573f748cSJohn Scipione 					+ fTime->Frame().Width();
23341281cf3SAxel Dörfler 			} else
23470ad5289SJohn Scipione 				width = fRightBottomReplicant.right + kIconGap + kGutter;
23541281cf3SAxel Dörfler 		}
2361ad8c760SFredrik Holmqvist 
23741281cf3SAxel Dörfler 		// this view has a fixed minimum width
2381bb2e623SJohn Scipione 		width = std::max(fMinimumTrayWidth, width);
2392ce9bab8SJohn Scipione 		height = kGutter + static_cast<TBarApp*>(be_app)->IconSize() + kGutter;
24041281cf3SAxel Dörfler 	}
24141281cf3SAxel Dörfler 
24241281cf3SAxel Dörfler 	*preferredWidth = width;
2431ad8c760SFredrik Holmqvist 	// add 1 for the border
24441281cf3SAxel Dörfler 	*preferredHeight = height + 1;
24541281cf3SAxel Dörfler }
24641281cf3SAxel Dörfler 
24741281cf3SAxel Dörfler 
24841281cf3SAxel Dörfler void
24941281cf3SAxel Dörfler TReplicantTray::AdjustPlacement()
25041281cf3SAxel Dörfler {
25141281cf3SAxel Dörfler 	// called when an add-on has been added or removed
25241281cf3SAxel Dörfler 	// need to resize the parent of this accordingly
253a5210ab1SAxel Dörfler 
25441281cf3SAxel Dörfler 	BRect bounds = Bounds();
25541281cf3SAxel Dörfler 	float width, height;
25641281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
25741281cf3SAxel Dörfler 
25841281cf3SAxel Dörfler 	if (width == bounds.Width() && height == bounds.Height()) {
25941281cf3SAxel Dörfler 		// no need to change anything
26041281cf3SAxel Dörfler 		return;
26141281cf3SAxel Dörfler 	}
26241281cf3SAxel Dörfler 
26341281cf3SAxel Dörfler 	Parent()->ResizeToPreferred();
26441281cf3SAxel Dörfler 	fBarView->UpdatePlacement();
26541281cf3SAxel Dörfler 	Parent()->Invalidate();
26641281cf3SAxel Dörfler 	Invalidate();
26741281cf3SAxel Dörfler }
26841281cf3SAxel Dörfler 
26941281cf3SAxel Dörfler 
27041281cf3SAxel Dörfler void
27141281cf3SAxel Dörfler TReplicantTray::MessageReceived(BMessage* message)
27241281cf3SAxel Dörfler {
27341281cf3SAxel Dörfler 	switch (message->what) {
27430a55802SJohn Scipione 		case B_LOCALE_CHANGED:
275573f748cSJohn Scipione 			if (fTime == NULL)
276573f748cSJohn Scipione 				return;
277573f748cSJohn Scipione 
27831c0024dSJohn Scipione 			fTime->Update();
27930a55802SJohn Scipione 
28041281cf3SAxel Dörfler 			// time string reformat -> realign
28141281cf3SAxel Dörfler 			RealignReplicants();
28241281cf3SAxel Dörfler 			AdjustPlacement();
28341281cf3SAxel Dörfler 			break;
284573f748cSJohn Scipione 
28530a55802SJohn Scipione 		case kShowHideTime:
28630a55802SJohn Scipione 			// from context menu in clock and in this view
28730a55802SJohn Scipione 			ShowHideTime();
28830a55802SJohn Scipione 			break;
28930a55802SJohn Scipione 
2903cf2d117SJohn Scipione 		case kShowSeconds:
291573f748cSJohn Scipione 			if (fTime == NULL)
292573f748cSJohn Scipione 				return;
293573f748cSJohn Scipione 
2943cf2d117SJohn Scipione 			fTime->SetShowSeconds(!fTime->ShowSeconds());
295573f748cSJohn Scipione 
2963cf2d117SJohn Scipione 			// time string reformat -> realign
2973cf2d117SJohn Scipione 			RealignReplicants();
2983cf2d117SJohn Scipione 			AdjustPlacement();
299573f748cSJohn Scipione 			break;
3003cf2d117SJohn Scipione 
3013cf2d117SJohn Scipione 		case kShowDayOfWeek:
3023cf2d117SJohn Scipione 			if (fTime == NULL)
3033cf2d117SJohn Scipione 				return;
3043cf2d117SJohn Scipione 
3053cf2d117SJohn Scipione 			fTime->SetShowDayOfWeek(!fTime->ShowDayOfWeek());
3063cf2d117SJohn Scipione 
3073cf2d117SJohn Scipione 			// time string reformat -> realign
3083cf2d117SJohn Scipione 			RealignReplicants();
3093cf2d117SJohn Scipione 			AdjustPlacement();
3103cf2d117SJohn Scipione 			break;
3113cf2d117SJohn Scipione 
31231c0024dSJohn Scipione 		case kShowTimeZone:
31331c0024dSJohn Scipione 			if (fTime == NULL)
31431c0024dSJohn Scipione 				return;
31531c0024dSJohn Scipione 
31631c0024dSJohn Scipione 			fTime->SetShowTimeZone(!fTime->ShowTimeZone());
31731c0024dSJohn Scipione 
31831c0024dSJohn Scipione 			// time string reformat -> realign
31931c0024dSJohn Scipione 			RealignReplicants();
32031c0024dSJohn Scipione 			AdjustPlacement();
32131c0024dSJohn Scipione 			break;
32231c0024dSJohn Scipione 
32331c0024dSJohn Scipione 		case kGetClockSettings:
32431c0024dSJohn Scipione 		{
32531c0024dSJohn Scipione 			if (fTime == NULL)
32631c0024dSJohn Scipione 				return;
32731c0024dSJohn Scipione 
32831c0024dSJohn Scipione 			bool showClock = !fTime->IsHidden();
32931c0024dSJohn Scipione 			bool showSeconds = fTime->ShowSeconds();
33031c0024dSJohn Scipione 			bool showDayOfWeek = fTime->ShowDayOfWeek();
33131c0024dSJohn Scipione 			bool showTimeZone = fTime->ShowTimeZone();
33231c0024dSJohn Scipione 
3330fc30d89SDario Casalinuovo 			BMessage reply(kGetClockSettings);
3340fc30d89SDario Casalinuovo 			reply.AddBool("showClock", showClock);
3350fc30d89SDario Casalinuovo 			reply.AddBool("showSeconds", showSeconds);
3360fc30d89SDario Casalinuovo 			reply.AddBool("showDayOfWeek", showDayOfWeek);
3370fc30d89SDario Casalinuovo 			reply.AddBool("showTimeZone", showTimeZone);
3380fc30d89SDario Casalinuovo 			message->SendReply(&reply);
33931c0024dSJohn Scipione 			break;
34031c0024dSJohn Scipione 		}
34131c0024dSJohn Scipione 
34241281cf3SAxel Dörfler #ifdef DB_ADDONS
34341281cf3SAxel Dörfler 		case B_NODE_MONITOR:
34441281cf3SAxel Dörfler 			HandleEntryUpdate(message);
34541281cf3SAxel Dörfler 			break;
34641281cf3SAxel Dörfler #endif
34741281cf3SAxel Dörfler 
34841281cf3SAxel Dörfler 		default:
34941281cf3SAxel Dörfler 			BView::MessageReceived(message);
35041281cf3SAxel Dörfler 			break;
35141281cf3SAxel Dörfler 	}
35241281cf3SAxel Dörfler }
35341281cf3SAxel Dörfler 
35441281cf3SAxel Dörfler 
35541281cf3SAxel Dörfler void
35641281cf3SAxel Dörfler TReplicantTray::MouseDown(BPoint where)
35741281cf3SAxel Dörfler {
35841281cf3SAxel Dörfler #ifdef DB_ADDONS
35941281cf3SAxel Dörfler 	if (modifiers() & B_CONTROL_KEY)
36041281cf3SAxel Dörfler 		DumpList(fItemList);
36141281cf3SAxel Dörfler #endif
36241281cf3SAxel Dörfler 
36341281cf3SAxel Dörfler 	uint32 buttons;
36441281cf3SAxel Dörfler 
36541281cf3SAxel Dörfler 	Window()->CurrentMessage()->FindInt32("buttons", (int32*)&buttons);
36641281cf3SAxel Dörfler 	if (buttons == B_SECONDARY_MOUSE_BUTTON) {
36741281cf3SAxel Dörfler 		ShowReplicantMenu(where);
36841281cf3SAxel Dörfler 	} else {
36941281cf3SAxel Dörfler 		BPoint save = where;
37041281cf3SAxel Dörfler 		bigtime_t doubleClickSpeed;
37141281cf3SAxel Dörfler 		bigtime_t start = system_time();
37241281cf3SAxel Dörfler 		uint32 buttons;
37341281cf3SAxel Dörfler 
37441281cf3SAxel Dörfler 		get_click_speed(&doubleClickSpeed);
37541281cf3SAxel Dörfler 
37641281cf3SAxel Dörfler 		do {
37741281cf3SAxel Dörfler 			if (fabs(where.x - save.x) > 4 || fabs(where.y - save.y) > 4)
37841281cf3SAxel Dörfler 				// user moved out of bounds of click area
37941281cf3SAxel Dörfler 				break;
38041281cf3SAxel Dörfler 
38141281cf3SAxel Dörfler 			if ((system_time() - start) > (2 * doubleClickSpeed)) {
38241281cf3SAxel Dörfler 				ShowReplicantMenu(where);
38341281cf3SAxel Dörfler 				break;
38441281cf3SAxel Dörfler 			}
38541281cf3SAxel Dörfler 
38641281cf3SAxel Dörfler 			snooze(50000);
38741281cf3SAxel Dörfler 			GetMouse(&where, &buttons);
38841281cf3SAxel Dörfler 		} while (buttons);
38941281cf3SAxel Dörfler 	}
39041281cf3SAxel Dörfler 	BView::MouseDown(where);
39141281cf3SAxel Dörfler }
39241281cf3SAxel Dörfler 
3933cf2d117SJohn Scipione 
3943cf2d117SJohn Scipione void
3953cf2d117SJohn Scipione TReplicantTray::ShowReplicantMenu(BPoint point)
3963cf2d117SJohn Scipione {
3973cf2d117SJohn Scipione 	BPopUpMenu* menu = new BPopUpMenu("", false, false);
3983cf2d117SJohn Scipione 	menu->SetFont(be_plain_font);
3993cf2d117SJohn Scipione 
40031c0024dSJohn Scipione 	// If clock is visible show the extended menu, otherwise show "Show clock"
4013cf2d117SJohn Scipione 
4023cf2d117SJohn Scipione 	if (!fTime->IsHidden())
4033cf2d117SJohn Scipione 		fTime->ShowTimeOptions(ConvertToScreen(point));
4043cf2d117SJohn Scipione 	else {
40531c0024dSJohn Scipione 		BMenuItem* item = new BMenuItem(B_TRANSLATE("Show clock"),
4063cf2d117SJohn Scipione 			new BMessage(kShowHideTime));
4073cf2d117SJohn Scipione 		menu->AddItem(item);
4083cf2d117SJohn Scipione 		menu->SetTargetForItems(this);
4093cf2d117SJohn Scipione 		BPoint where = ConvertToScreen(point);
4103cf2d117SJohn Scipione 		menu->Go(where, true, true, BRect(where - BPoint(4, 4),
4113cf2d117SJohn Scipione 			where + BPoint(4, 4)), true);
4123cf2d117SJohn Scipione 	}
4133cf2d117SJohn Scipione }
4143cf2d117SJohn Scipione 
4153cf2d117SJohn Scipione 
4163cf2d117SJohn Scipione void
4173cf2d117SJohn Scipione TReplicantTray::SetMultiRow(bool state)
4183cf2d117SJohn Scipione {
4193cf2d117SJohn Scipione 	fMultiRowMode = state;
4203cf2d117SJohn Scipione }
4213cf2d117SJohn Scipione 
4223cf2d117SJohn Scipione 
4233cf2d117SJohn Scipione void
4243cf2d117SJohn Scipione TReplicantTray::ShowHideTime()
4253cf2d117SJohn Scipione {
4263cf2d117SJohn Scipione 	if (fTime == NULL)
4273cf2d117SJohn Scipione 		return;
4283cf2d117SJohn Scipione 
4292a5c1f12SJohn Scipione 	// Check from the point of view of fTime because we need to ignore
4302a5c1f12SJohn Scipione 	// whether or not the parent window is hidden.
4312a5c1f12SJohn Scipione 	if (fTime->IsHidden(fTime))
4323cf2d117SJohn Scipione 		fTime->Show();
4333cf2d117SJohn Scipione 	else
4343cf2d117SJohn Scipione 		fTime->Hide();
4353cf2d117SJohn Scipione 
4363cf2d117SJohn Scipione 	RealignReplicants();
4373cf2d117SJohn Scipione 	AdjustPlacement();
43831c0024dSJohn Scipione 
4392a5c1f12SJohn Scipione 	// Check from the point of view of fTime ignoring parent's state.
4402a5c1f12SJohn Scipione 	bool showClock = !fTime->IsHidden(fTime);
4413722e640SJohn Scipione 
4423722e640SJohn Scipione 	// Update showClock setting that gets saved to disk on quit
4431bb2e623SJohn Scipione 	static_cast<TBarApp*>(be_app)->Settings()->showClock = showClock;
4443722e640SJohn Scipione 
4453722e640SJohn Scipione 	// Send a message to Time preferences telling it to update
44631c0024dSJohn Scipione 	BMessenger messenger("application/x-vnd.Haiku-Time");
4470fc30d89SDario Casalinuovo 	BMessage message(kShowHideTime);
4480fc30d89SDario Casalinuovo 	message.AddBool("showClock", showClock);
4490fc30d89SDario Casalinuovo 	messenger.SendMessage(&message);
4503cf2d117SJohn Scipione }
4513cf2d117SJohn Scipione 
4523cf2d117SJohn Scipione 
45341281cf3SAxel Dörfler #ifdef DB_ADDONS
45441281cf3SAxel Dörfler 
4553cf2d117SJohn Scipione 
45641281cf3SAxel Dörfler void
45741281cf3SAxel Dörfler TReplicantTray::InitAddOnSupport()
45841281cf3SAxel Dörfler {
45941281cf3SAxel Dörfler 	// list to maintain refs to each rep added/deleted
46041281cf3SAxel Dörfler 	fItemList = new BList();
46141281cf3SAxel Dörfler 	BPath path;
4621ad8c760SFredrik Holmqvist 
46326ea7066SIngo Weinhold 	if (GetDeskbarSettingsDirectory(path, true) == B_OK) {
4643e6ff860SRene Gollent 		path.Append(kReplicantSettingsFile);
46541281cf3SAxel Dörfler 
46641281cf3SAxel Dörfler 		BFile file(path.Path(), B_READ_ONLY);
4673e6ff860SRene Gollent 		if (file.InitCheck() == B_OK) {
4683e6ff860SRene Gollent 			status_t result;
4693e6ff860SRene Gollent 			BEntry entry;
4703e6ff860SRene Gollent 			int32 id;
471c0ad1c91SRene Gollent 			BString path;
4723e6ff860SRene Gollent 			if (fAddOnSettings.Unflatten(&file) == B_OK) {
473c0ad1c91SRene Gollent 				for (int32 i = 0; fAddOnSettings.FindString(kReplicantPathField,
474c0ad1c91SRene Gollent 					i, &path) == B_OK; i++) {
475c0ad1c91SRene Gollent 					if (entry.SetTo(path.String()) == B_OK && entry.Exists()) {
4763e6ff860SRene Gollent 						result = LoadAddOn(&entry, &id, false);
4773e6ff860SRene Gollent 					} else
4783e6ff860SRene Gollent 						result = B_ENTRY_NOT_FOUND;
47941281cf3SAxel Dörfler 
4803e6ff860SRene Gollent 					if (result != B_OK) {
481c0ad1c91SRene Gollent 						fAddOnSettings.RemoveData(kReplicantPathField, i);
4823e6ff860SRene Gollent 						--i;
48341281cf3SAxel Dörfler 					}
48441281cf3SAxel Dörfler 				}
48541281cf3SAxel Dörfler 			}
4863e6ff860SRene Gollent 		}
4873e6ff860SRene Gollent 	}
48841281cf3SAxel Dörfler }
48941281cf3SAxel Dörfler 
49041281cf3SAxel Dörfler 
49141281cf3SAxel Dörfler void
49241281cf3SAxel Dörfler TReplicantTray::DeleteAddOnSupport()
49341281cf3SAxel Dörfler {
4947625ce51SRene Gollent 	_SaveSettings();
4953e6ff860SRene Gollent 
4969f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
49741281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->RemoveItem(i);
49841281cf3SAxel Dörfler 		if (item) {
49941281cf3SAxel Dörfler 			if (item->isAddOn)
50041281cf3SAxel Dörfler 				watch_node(&(item->nodeRef), B_STOP_WATCHING, this, Window());
50141281cf3SAxel Dörfler 
50241281cf3SAxel Dörfler 			delete item;
50341281cf3SAxel Dörfler 		}
50441281cf3SAxel Dörfler 	}
50541281cf3SAxel Dörfler 	delete fItemList;
50641281cf3SAxel Dörfler 
50741281cf3SAxel Dörfler 	// stop the volume mount/unmount watch
50841281cf3SAxel Dörfler 	stop_watching(this, Window());
50941281cf3SAxel Dörfler }
51041281cf3SAxel Dörfler 
51141281cf3SAxel Dörfler 
51241281cf3SAxel Dörfler DeskbarItemInfo*
51341281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(node_ref& nodeRef)
51441281cf3SAxel Dörfler {
5159f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
51666eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
51741281cf3SAxel Dörfler 		if (item == NULL)
51841281cf3SAxel Dörfler 			continue;
51941281cf3SAxel Dörfler 
52041281cf3SAxel Dörfler 		if (item->nodeRef == nodeRef)
52141281cf3SAxel Dörfler 			return item;
52241281cf3SAxel Dörfler 	}
52341281cf3SAxel Dörfler 
52441281cf3SAxel Dörfler 	return NULL;
52541281cf3SAxel Dörfler }
52641281cf3SAxel Dörfler 
52741281cf3SAxel Dörfler 
52841281cf3SAxel Dörfler DeskbarItemInfo*
52941281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(int32 id)
53041281cf3SAxel Dörfler {
5319f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
53266eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
53341281cf3SAxel Dörfler 		if (item == NULL)
53441281cf3SAxel Dörfler 			continue;
53541281cf3SAxel Dörfler 
53641281cf3SAxel Dörfler 		if (item->id == id)
53741281cf3SAxel Dörfler 			return item;
53841281cf3SAxel Dörfler 	}
53941281cf3SAxel Dörfler 
54041281cf3SAxel Dörfler 	return NULL;
54141281cf3SAxel Dörfler }
54241281cf3SAxel Dörfler 
54341281cf3SAxel Dörfler 
54441281cf3SAxel Dörfler bool
54541281cf3SAxel Dörfler TReplicantTray::NodeExists(node_ref& nodeRef)
54641281cf3SAxel Dörfler {
54741281cf3SAxel Dörfler 	return DeskbarItemFor(nodeRef) != NULL;
54841281cf3SAxel Dörfler }
54941281cf3SAxel Dörfler 
55041281cf3SAxel Dörfler 
55166eba86fSAxel Dörfler /*! This handles B_NODE_MONITOR & B_QUERY_UPDATE messages received
55266eba86fSAxel Dörfler 	for the registered add-ons.
55341281cf3SAxel Dörfler */
55441281cf3SAxel Dörfler void
55541281cf3SAxel Dörfler TReplicantTray::HandleEntryUpdate(BMessage* message)
55641281cf3SAxel Dörfler {
55741281cf3SAxel Dörfler 	int32 opcode;
55841281cf3SAxel Dörfler 	if (message->FindInt32("opcode", &opcode) != B_OK)
55941281cf3SAxel Dörfler 		return;
56041281cf3SAxel Dörfler 
56141281cf3SAxel Dörfler 	BPath path;
56241281cf3SAxel Dörfler 	switch (opcode) {
56341281cf3SAxel Dörfler 		case B_ENTRY_MOVED:
56441281cf3SAxel Dörfler 		{
56541281cf3SAxel Dörfler 			entry_ref ref;
56641281cf3SAxel Dörfler 			ino_t todirectory;
56741281cf3SAxel Dörfler 			ino_t node;
56841281cf3SAxel Dörfler 			const char* name;
56941281cf3SAxel Dörfler 			if (message->FindString("name", &name) == B_OK
57071bd3ba5SJonas Sundström 				&& message->FindInt64("from directory", &(ref.directory))
57171bd3ba5SJonas Sundström 					== B_OK
57241281cf3SAxel Dörfler 				&& message->FindInt64("to directory", &todirectory) == B_OK
57341281cf3SAxel Dörfler 				&& message->FindInt32("device", &(ref.device)) == B_OK
57441281cf3SAxel Dörfler 				&& message->FindInt64("node", &node) == B_OK ) {
57541281cf3SAxel Dörfler 
576e4d65fe7SJohn Scipione 				if (name == NULL)
57741281cf3SAxel Dörfler 					break;
57841281cf3SAxel Dörfler 
57941281cf3SAxel Dörfler 				ref.set_name(name);
58041281cf3SAxel Dörfler 				// change the directory reference to
58141281cf3SAxel Dörfler 				// the new directory
58241281cf3SAxel Dörfler 				MoveItem(&ref, todirectory);
58341281cf3SAxel Dörfler 			}
58441281cf3SAxel Dörfler 			break;
58566eba86fSAxel Dörfler 		}
58641281cf3SAxel Dörfler 
58741281cf3SAxel Dörfler 		case B_ENTRY_REMOVED:
58841281cf3SAxel Dörfler 		{
58941281cf3SAxel Dörfler 			// entry was rm'd from the device
59041281cf3SAxel Dörfler 			node_ref nodeRef;
59141281cf3SAxel Dörfler 			if (message->FindInt32("device", &(nodeRef.device)) == B_OK
59241281cf3SAxel Dörfler 				&& message->FindInt64("node", &(nodeRef.node)) == B_OK) {
59341281cf3SAxel Dörfler 				DeskbarItemInfo* item = DeskbarItemFor(nodeRef);
59441281cf3SAxel Dörfler 				if (item == NULL)
59541281cf3SAxel Dörfler 					break;
59641281cf3SAxel Dörfler 
59741281cf3SAxel Dörfler 				// If there is a team running where the add-on comes from,
59841281cf3SAxel Dörfler 				// we don't want to remove the icon yet.
59941281cf3SAxel Dörfler 				if (be_roster->IsRunning(&item->entryRef))
60041281cf3SAxel Dörfler 					break;
60141281cf3SAxel Dörfler 
60241281cf3SAxel Dörfler 				UnloadAddOn(&nodeRef, NULL, true, false);
60341281cf3SAxel Dörfler 			}
60441281cf3SAxel Dörfler 			break;
60566eba86fSAxel Dörfler 		}
60641281cf3SAxel Dörfler 	}
60766eba86fSAxel Dörfler }
60841281cf3SAxel Dörfler 
60941281cf3SAxel Dörfler 
6101ad8c760SFredrik Holmqvist /*! The add-ons must support the exported C function API
61166eba86fSAxel Dörfler 	if they do, they will be loaded and added to deskbar
61266eba86fSAxel Dörfler 	primary function is the Instantiate function
6137da06231SAxel Dörfler */
61441281cf3SAxel Dörfler status_t
6153e6ff860SRene Gollent TReplicantTray::LoadAddOn(BEntry* entry, int32* id, bool addToSettings)
61641281cf3SAxel Dörfler {
617e4d65fe7SJohn Scipione 	if (entry == NULL)
618a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
61941281cf3SAxel Dörfler 
62041281cf3SAxel Dörfler 	node_ref nodeRef;
62141281cf3SAxel Dörfler 	entry->GetNodeRef(&nodeRef);
62241281cf3SAxel Dörfler 	// no duplicates
62341281cf3SAxel Dörfler 	if (NodeExists(nodeRef))
62441281cf3SAxel Dörfler 		return B_ERROR;
62541281cf3SAxel Dörfler 
62641281cf3SAxel Dörfler 	BNode node(entry);
62741281cf3SAxel Dörfler 	BPath path;
62866eba86fSAxel Dörfler 	status_t status = entry->GetPath(&path);
629e4d65fe7SJohn Scipione 	if (status != B_OK)
63066eba86fSAxel Dörfler 		return status;
63141281cf3SAxel Dörfler 
63241281cf3SAxel Dörfler 	// load the add-on
63341281cf3SAxel Dörfler 	image_id image = load_add_on(path.Path());
63466eba86fSAxel Dörfler 	if (image < B_OK)
63566eba86fSAxel Dörfler 		return image;
63641281cf3SAxel Dörfler 
63741281cf3SAxel Dörfler 	// get the view loading function symbol
63841281cf3SAxel Dörfler 	//    we first look for a symbol that takes an image_id
63941281cf3SAxel Dörfler 	//    and entry_ref pointer, if not found, go with normal
64041281cf3SAxel Dörfler 	//    instantiate function
64141281cf3SAxel Dörfler 	BView* (*entryFunction)(image_id, const entry_ref*);
64241281cf3SAxel Dörfler 	BView* (*itemFunction)(void);
64341281cf3SAxel Dörfler 	BView* view = NULL;
64441281cf3SAxel Dörfler 
64541281cf3SAxel Dörfler 	entry_ref ref;
64641281cf3SAxel Dörfler 	entry->GetRef(&ref);
64741281cf3SAxel Dörfler 
64841281cf3SAxel Dörfler 	if (get_image_symbol(image, kInstantiateEntryCFunctionName,
64966eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&entryFunction) >= B_OK) {
65041281cf3SAxel Dörfler 		view = (*entryFunction)(image, &ref);
65141281cf3SAxel Dörfler 	} else if (get_image_symbol(image, kInstantiateItemCFunctionName,
65266eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&itemFunction) >= B_OK) {
65341281cf3SAxel Dörfler 		view = (*itemFunction)();
65441281cf3SAxel Dörfler 	} else {
65541281cf3SAxel Dörfler 		unload_add_on(image);
65641281cf3SAxel Dörfler 		return B_ERROR;
65741281cf3SAxel Dörfler 	}
65841281cf3SAxel Dörfler 
65966eba86fSAxel Dörfler 	if (view == NULL || IconExists(view->Name())) {
66041281cf3SAxel Dörfler 		delete view;
66141281cf3SAxel Dörfler 		unload_add_on(image);
66241281cf3SAxel Dörfler 		return B_ERROR;
66341281cf3SAxel Dörfler 	}
66441281cf3SAxel Dörfler 
66541281cf3SAxel Dörfler 	BMessage* data = new BMessage;
66641281cf3SAxel Dörfler 	view->Archive(data);
66741281cf3SAxel Dörfler 	delete view;
66841281cf3SAxel Dörfler 
66941281cf3SAxel Dörfler 	// add the rep; adds info to list
6700fc30d89SDario Casalinuovo 	if (AddIcon(data, id, &ref) != B_OK)
6710fc30d89SDario Casalinuovo 		delete data;
67241281cf3SAxel Dörfler 
6733e6ff860SRene Gollent 	if (addToSettings) {
674c0ad1c91SRene Gollent 		fAddOnSettings.AddString(kReplicantPathField, path.Path());
6757625ce51SRene Gollent 		_SaveSettings();
6763e6ff860SRene Gollent 	}
67741281cf3SAxel Dörfler 
67841281cf3SAxel Dörfler 	return B_OK;
67941281cf3SAxel Dörfler }
68041281cf3SAxel Dörfler 
68141281cf3SAxel Dörfler 
68241281cf3SAxel Dörfler status_t
68341281cf3SAxel Dörfler TReplicantTray::AddItem(int32 id, node_ref nodeRef, BEntry& entry, bool isAddOn)
68441281cf3SAxel Dörfler {
68541281cf3SAxel Dörfler 	DeskbarItemInfo* item = new DeskbarItemInfo;
68641281cf3SAxel Dörfler 	if (item == NULL)
68741281cf3SAxel Dörfler 		return B_NO_MEMORY;
68841281cf3SAxel Dörfler 
68941281cf3SAxel Dörfler 	item->id = id;
69041281cf3SAxel Dörfler 	item->isAddOn = isAddOn;
69141281cf3SAxel Dörfler 
692e4d65fe7SJohn Scipione 	if (entry.GetRef(&item->entryRef) != B_OK) {
69341281cf3SAxel Dörfler 		item->entryRef.device = -1;
69441281cf3SAxel Dörfler 		item->entryRef.directory = -1;
69541281cf3SAxel Dörfler 		item->entryRef.name = NULL;
69641281cf3SAxel Dörfler 	}
69741281cf3SAxel Dörfler 	item->nodeRef = nodeRef;
69841281cf3SAxel Dörfler 
69941281cf3SAxel Dörfler 	fItemList->AddItem(item);
70041281cf3SAxel Dörfler 
70141281cf3SAxel Dörfler 	if (isAddOn)
70241281cf3SAxel Dörfler 		watch_node(&nodeRef, B_WATCH_NAME | B_WATCH_ATTR, this, Window());
70341281cf3SAxel Dörfler 
70441281cf3SAxel Dörfler 	return B_OK;
70541281cf3SAxel Dörfler }
70641281cf3SAxel Dörfler 
70741281cf3SAxel Dörfler 
70841281cf3SAxel Dörfler /**	from entry_removed message, when attribute removed
70941281cf3SAxel Dörfler  *	or when a device is unmounted (use removeall, by device)
71041281cf3SAxel Dörfler  */
71141281cf3SAxel Dörfler 
71241281cf3SAxel Dörfler void
713e4d65fe7SJohn Scipione TReplicantTray::UnloadAddOn(node_ref* nodeRef, dev_t* device, bool which,
714e4d65fe7SJohn Scipione 	bool removeAll)
71541281cf3SAxel Dörfler {
7169f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
71741281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
718e4d65fe7SJohn Scipione 		if (item == NULL)
71941281cf3SAxel Dörfler 			continue;
72041281cf3SAxel Dörfler 
721e4d65fe7SJohn Scipione 		if ((which && nodeRef != NULL && item->nodeRef == *nodeRef)
722e4d65fe7SJohn Scipione 			|| (device != NULL && item->nodeRef.device == *device)) {
72341281cf3SAxel Dörfler 
724e4d65fe7SJohn Scipione 			if (device != NULL && be_roster->IsRunning(&item->entryRef))
72541281cf3SAxel Dörfler 				continue;
72641281cf3SAxel Dörfler 
72741281cf3SAxel Dörfler 			RemoveIcon(item->id);
72841281cf3SAxel Dörfler 
72941281cf3SAxel Dörfler 			if (!removeAll)
73041281cf3SAxel Dörfler 				break;
73141281cf3SAxel Dörfler 		}
73241281cf3SAxel Dörfler 	}
73341281cf3SAxel Dörfler }
73441281cf3SAxel Dörfler 
73541281cf3SAxel Dörfler 
73641281cf3SAxel Dörfler void
73741281cf3SAxel Dörfler TReplicantTray::RemoveItem(int32 id)
73841281cf3SAxel Dörfler {
73941281cf3SAxel Dörfler 	DeskbarItemInfo* item = DeskbarItemFor(id);
74041281cf3SAxel Dörfler 	if (item == NULL)
74141281cf3SAxel Dörfler 		return;
74241281cf3SAxel Dörfler 
74341281cf3SAxel Dörfler 	// attribute was added via Deskbar API (AddItem(entry_ref*, int32*)
74441281cf3SAxel Dörfler 	if (item->isAddOn) {
7457625ce51SRene Gollent 		BPath path(&item->entryRef);
7467625ce51SRene Gollent 		BString storedPath;
7477625ce51SRene Gollent 		for (int32 i = 0;
7488aab28f1SRene Gollent 			fAddOnSettings.FindString(kReplicantPathField, i, &storedPath)
7497625ce51SRene Gollent 				== B_OK; i++) {
7507625ce51SRene Gollent 			if (storedPath == path.Path()) {
7518aab28f1SRene Gollent 				fAddOnSettings.RemoveData(kReplicantPathField, i);
7527625ce51SRene Gollent 				break;
7537625ce51SRene Gollent 			}
7547625ce51SRene Gollent 		}
7557625ce51SRene Gollent 		_SaveSettings();
7567625ce51SRene Gollent 
75741281cf3SAxel Dörfler 		BNode node(&item->entryRef);
75841281cf3SAxel Dörfler 		watch_node(&item->nodeRef, B_STOP_WATCHING, this, Window());
75941281cf3SAxel Dörfler 	}
76041281cf3SAxel Dörfler 
76141281cf3SAxel Dörfler 	fItemList->RemoveItem(item);
76241281cf3SAxel Dörfler 	delete item;
76341281cf3SAxel Dörfler }
76441281cf3SAxel Dörfler 
76541281cf3SAxel Dörfler 
76641281cf3SAxel Dörfler /**	ENTRY_MOVED message, moving only occurs on a device
76741281cf3SAxel Dörfler  *	copying will occur (ENTRY_CREATED) between devices
76841281cf3SAxel Dörfler  */
76941281cf3SAxel Dörfler 
77041281cf3SAxel Dörfler void
77141281cf3SAxel Dörfler TReplicantTray::MoveItem(entry_ref* ref, ino_t toDirectory)
77241281cf3SAxel Dörfler {
773e4d65fe7SJohn Scipione 	if (ref == NULL)
77441281cf3SAxel Dörfler 		return;
77541281cf3SAxel Dörfler 
77641281cf3SAxel Dörfler 	// scan for a matching entry_ref and update it
77741281cf3SAxel Dörfler 	//
77841281cf3SAxel Dörfler 	// don't need to change node info as it does not change
77941281cf3SAxel Dörfler 
7809f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
78141281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
782e4d65fe7SJohn Scipione 		if (item == NULL)
78341281cf3SAxel Dörfler 			continue;
78441281cf3SAxel Dörfler 
785e4d65fe7SJohn Scipione 		if (strcmp(item->entryRef.name, ref->name) == 0
78641281cf3SAxel Dörfler 			&& item->entryRef.device == ref->device
78741281cf3SAxel Dörfler 			&& item->entryRef.directory == ref->directory) {
78841281cf3SAxel Dörfler 			item->entryRef.directory = toDirectory;
78941281cf3SAxel Dörfler 			break;
79041281cf3SAxel Dörfler 		}
79141281cf3SAxel Dörfler 	}
79241281cf3SAxel Dörfler }
79341281cf3SAxel Dörfler 
79441281cf3SAxel Dörfler #endif // add-on support
79541281cf3SAxel Dörfler 
79641281cf3SAxel Dörfler //	external add-on API routines
79741281cf3SAxel Dörfler //	called using the new BDeskbar class
79841281cf3SAxel Dörfler 
79941281cf3SAxel Dörfler //	existence of icon/replicant by name or ID
80041281cf3SAxel Dörfler //	returns opposite
80141281cf3SAxel Dörfler //	note: name and id are semi-private limiting
80241281cf3SAxel Dörfler //		the ability of non-host apps to remove
80341281cf3SAxel Dörfler //		icons without a little bit of work
80441281cf3SAxel Dörfler 
80541281cf3SAxel Dörfler /**	for a specific id
80641281cf3SAxel Dörfler  *	return the name of the replicant (name of view)
80741281cf3SAxel Dörfler  */
80841281cf3SAxel Dörfler 
80941281cf3SAxel Dörfler status_t
81041281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 id, const char** name)
81141281cf3SAxel Dörfler {
81241281cf3SAxel Dörfler 	if (id < 0)
813a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
81441281cf3SAxel Dörfler 
815e4d65fe7SJohn Scipione 	int32 index;
816e4d65fe7SJohn Scipione 	int32 temp;
81741281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &temp, id, false);
818e4d65fe7SJohn Scipione 	if (view != NULL) {
81941281cf3SAxel Dörfler 		*name = view->Name();
82041281cf3SAxel Dörfler 		return B_OK;
82141281cf3SAxel Dörfler 	}
82241281cf3SAxel Dörfler 
82341281cf3SAxel Dörfler 	return B_ERROR;
82441281cf3SAxel Dörfler }
82541281cf3SAxel Dörfler 
82641281cf3SAxel Dörfler 
8277da06231SAxel Dörfler /**	for a specific name
8287da06231SAxel Dörfler  *	return the id (internal to Deskbar)
8297da06231SAxel Dörfler  */
8307da06231SAxel Dörfler 
83141281cf3SAxel Dörfler status_t
83241281cf3SAxel Dörfler TReplicantTray::ItemInfo(const char* name, int32* id)
83341281cf3SAxel Dörfler {
834e4d65fe7SJohn Scipione 	if (name == NULL || *name == '\0')
835a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
83641281cf3SAxel Dörfler 
83741281cf3SAxel Dörfler 	int32 index;
83841281cf3SAxel Dörfler 	BView* view = ViewAt(&index, id, name);
83941281cf3SAxel Dörfler 
840378aa08bSJohn Scipione 	return view != NULL ? B_OK : B_ERROR;
84141281cf3SAxel Dörfler }
84241281cf3SAxel Dörfler 
84341281cf3SAxel Dörfler 
8447da06231SAxel Dörfler /**	at a specific index
8457da06231SAxel Dörfler  *	return both the name and the id of the replicant
8467da06231SAxel Dörfler  */
8477da06231SAxel Dörfler 
84841281cf3SAxel Dörfler status_t
84941281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 index, const char** name, int32* id)
85041281cf3SAxel Dörfler {
85141281cf3SAxel Dörfler 	if (index < 0)
852a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
85341281cf3SAxel Dörfler 
85441281cf3SAxel Dörfler 	BView* view;
85541281cf3SAxel Dörfler 	fShelf->ReplicantAt(index, &view, (uint32*)id, NULL);
856e4d65fe7SJohn Scipione 	if (view != NULL) {
85741281cf3SAxel Dörfler 		*name = view->Name();
85841281cf3SAxel Dörfler 		return B_OK;
85941281cf3SAxel Dörfler 	}
86041281cf3SAxel Dörfler 
86141281cf3SAxel Dörfler 	return B_ERROR;
86241281cf3SAxel Dörfler }
86341281cf3SAxel Dörfler 
86441281cf3SAxel Dörfler 
8657da06231SAxel Dörfler /**	replicant exists, by id/index */
8667da06231SAxel Dörfler 
86741281cf3SAxel Dörfler bool
86841281cf3SAxel Dörfler TReplicantTray::IconExists(int32 target, bool byIndex)
86941281cf3SAxel Dörfler {
870e4d65fe7SJohn Scipione 	int32 index;
871e4d65fe7SJohn Scipione 	int32 id;
87241281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
87341281cf3SAxel Dörfler 
87441281cf3SAxel Dörfler 	return view && index >= 0;
87541281cf3SAxel Dörfler }
87641281cf3SAxel Dörfler 
87741281cf3SAxel Dörfler 
8787da06231SAxel Dörfler /**	replicant exists, by name */
8797da06231SAxel Dörfler 
88041281cf3SAxel Dörfler bool
88141281cf3SAxel Dörfler TReplicantTray::IconExists(const char* name)
88241281cf3SAxel Dörfler {
883e4d65fe7SJohn Scipione 	if (name == NULL || *name == '\0')
88441281cf3SAxel Dörfler 		return false;
88541281cf3SAxel Dörfler 
886e4d65fe7SJohn Scipione 	int32 index;
887e4d65fe7SJohn Scipione 	int32 id;
88841281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
88941281cf3SAxel Dörfler 
890e4d65fe7SJohn Scipione 	return view != NULL && index >= 0;
89141281cf3SAxel Dörfler }
89241281cf3SAxel Dörfler 
89341281cf3SAxel Dörfler 
89441281cf3SAxel Dörfler int32
89541281cf3SAxel Dörfler TReplicantTray::IconCount() const
89641281cf3SAxel Dörfler {
89741281cf3SAxel Dörfler 	return fShelf->CountReplicants();
89841281cf3SAxel Dörfler }
89941281cf3SAxel Dörfler 
90041281cf3SAxel Dörfler 
901ced3755cSAxel Dörfler /*! Message must contain an archivable view for later rehydration.
902ced3755cSAxel Dörfler 	This function takes over ownership of the provided message on success
903ced3755cSAxel Dörfler 	only.
904ced3755cSAxel Dörfler 	Returns the current replicant ID.
9057da06231SAxel Dörfler */
90641281cf3SAxel Dörfler status_t
907ced3755cSAxel Dörfler TReplicantTray::AddIcon(BMessage* archive, int32* id, const entry_ref* addOn)
90841281cf3SAxel Dörfler {
909ced3755cSAxel Dörfler 	if (archive == NULL || id == NULL)
910a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
91141281cf3SAxel Dörfler 
9129c50f36eSAxel Dörfler 	// find entry_ref
9139c50f36eSAxel Dörfler 
9149c50f36eSAxel Dörfler 	entry_ref ref;
915e4d65fe7SJohn Scipione 	if (addOn != NULL) {
9169c50f36eSAxel Dörfler 		// Use it if we got it
9179c50f36eSAxel Dörfler 		ref = *addOn;
9189c50f36eSAxel Dörfler 	} else {
9199c50f36eSAxel Dörfler 		const char* signature;
9201ad8c760SFredrik Holmqvist 
921ced3755cSAxel Dörfler 		status_t status = archive->FindString("add_on", &signature);
9229c50f36eSAxel Dörfler 		if (status == B_OK) {
9239c50f36eSAxel Dörfler 			BRoster roster;
9249c50f36eSAxel Dörfler 			status = roster.FindApp(signature, &ref);
9259c50f36eSAxel Dörfler 		}
926e4d65fe7SJohn Scipione 		if (status != B_OK)
9279c50f36eSAxel Dörfler 			return status;
9289c50f36eSAxel Dörfler 	}
9299c50f36eSAxel Dörfler 
9309c50f36eSAxel Dörfler 	BFile file;
9319c50f36eSAxel Dörfler 	status_t status = file.SetTo(&ref, B_READ_ONLY);
932e4d65fe7SJohn Scipione 	if (status != B_OK)
9339c50f36eSAxel Dörfler 		return status;
9349c50f36eSAxel Dörfler 
9359c50f36eSAxel Dörfler 	node_ref nodeRef;
9369c50f36eSAxel Dörfler 	status = file.GetNodeRef(&nodeRef);
937e4d65fe7SJohn Scipione 	if (status != B_OK)
9389c50f36eSAxel Dörfler 		return status;
9399c50f36eSAxel Dörfler 
9409c50f36eSAxel Dörfler 	BEntry entry(&ref, true);
9411ad8c760SFredrik Holmqvist 		// TODO: this resolves an eventual link for the item being added - this
9421ad8c760SFredrik Holmqvist 		// is okay for now, but in multi-user environments, one might want to
9431ad8c760SFredrik Holmqvist 		// have links that carry the be:deskbar_item_status attribute
9449c50f36eSAxel Dörfler 	status = entry.InitCheck();
9459c50f36eSAxel Dörfler 	if (status != B_OK)
9469c50f36eSAxel Dörfler 		return status;
9479c50f36eSAxel Dörfler 
94841281cf3SAxel Dörfler 	*id = 999;
949ced3755cSAxel Dörfler 	if (archive->what == B_ARCHIVED_OBJECT)
950ced3755cSAxel Dörfler 		archive->what = 0;
95141281cf3SAxel Dörfler 
952ced3755cSAxel Dörfler 	BRect originalBounds = archive->FindRect("_frame");
9531ad8c760SFredrik Holmqvist 		// this is a work-around for buggy replicants that change their size in
9541ad8c760SFredrik Holmqvist 		// AttachedToWindow() (such as "SVM")
955a5210ab1SAxel Dörfler 
9561ad8c760SFredrik Holmqvist 	// TODO: check for name collisions?
957ced3755cSAxel Dörfler 	status = fShelf->AddReplicant(archive, BPoint(1, 1));
9589c50f36eSAxel Dörfler 	if (status != B_OK)
9599c50f36eSAxel Dörfler 		return status;
96041281cf3SAxel Dörfler 
961a5210ab1SAxel Dörfler 	int32 count = fShelf->CountReplicants();
962a5210ab1SAxel Dörfler 	BView* view;
963a5210ab1SAxel Dörfler 	fShelf->ReplicantAt(count - 1, &view, (uint32*)id, NULL);
964a5210ab1SAxel Dörfler 
965a6be6c4dSJohn Scipione 	if (view != NULL && originalBounds != view->Bounds()) {
966a5210ab1SAxel Dörfler 		// The replicant changed its size when added to the window, so we need
96771bd3ba5SJonas Sundström 		// to recompute all over again (it's already done once via
96871bd3ba5SJonas Sundström 		// BShelf::AddReplicant() and TReplicantShelf::CanAcceptReplicantView())
969a5210ab1SAxel Dörfler 		RealignReplicants();
970a5210ab1SAxel Dörfler 	}
971a5210ab1SAxel Dörfler 
97241281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
97341281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
97441281cf3SAxel Dörfler 	float width, height;
97541281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
97641281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height)
97741281cf3SAxel Dörfler 		AdjustPlacement();
97841281cf3SAxel Dörfler 
97941281cf3SAxel Dörfler 	// add the item to the add-on list
98041281cf3SAxel Dörfler 
98141281cf3SAxel Dörfler 	AddItem(*id, nodeRef, entry, addOn != NULL);
98241281cf3SAxel Dörfler 	return B_OK;
98341281cf3SAxel Dörfler }
98441281cf3SAxel Dörfler 
98541281cf3SAxel Dörfler 
98641281cf3SAxel Dörfler void
98741281cf3SAxel Dörfler TReplicantTray::RemoveIcon(int32 target, bool byIndex)
98841281cf3SAxel Dörfler {
98941281cf3SAxel Dörfler 	if (target < 0)
99041281cf3SAxel Dörfler 		return;
99141281cf3SAxel Dörfler 
992e4d65fe7SJohn Scipione 	int32 index;
993e4d65fe7SJohn Scipione 	int32 id;
99441281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
995e4d65fe7SJohn Scipione 	if (view != NULL && index >= 0) {
9967da06231SAxel Dörfler 		// remove the reference from the item list & the shelf
99741281cf3SAxel Dörfler 		RemoveItem(id);
99841281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9997da06231SAxel Dörfler 
100041281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
100141281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
100241281cf3SAxel Dörfler 	}
100341281cf3SAxel Dörfler }
100441281cf3SAxel Dörfler 
100541281cf3SAxel Dörfler 
100641281cf3SAxel Dörfler void
100741281cf3SAxel Dörfler TReplicantTray::RemoveIcon(const char* name)
100841281cf3SAxel Dörfler {
1009e4d65fe7SJohn Scipione 	if (name == NULL || *name == '\0')
101041281cf3SAxel Dörfler 		return;
101141281cf3SAxel Dörfler 
1012378aa08bSJohn Scipione 	int32 index;
1013378aa08bSJohn Scipione 	int32 id;
101441281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
1015e4d65fe7SJohn Scipione 	if (view != NULL && index >= 0) {
10167da06231SAxel Dörfler 		// remove the reference from the item list & shelf
101741281cf3SAxel Dörfler 		RemoveItem(id);
101841281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
10197da06231SAxel Dörfler 
102041281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
102141281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
102241281cf3SAxel Dörfler 	}
102341281cf3SAxel Dörfler }
102441281cf3SAxel Dörfler 
102541281cf3SAxel Dörfler 
102641281cf3SAxel Dörfler void
10277da06231SAxel Dörfler TReplicantTray::RealReplicantAdjustment(int32 startIndex)
102841281cf3SAxel Dörfler {
10297da06231SAxel Dörfler 	if (startIndex < 0)
103041281cf3SAxel Dörfler 		return;
10317da06231SAxel Dörfler 
10327da06231SAxel Dörfler 	if (startIndex == fLastReplicant)
10337da06231SAxel Dörfler 		startIndex = 0;
10347da06231SAxel Dörfler 
103541281cf3SAxel Dörfler 	// reset the locations of all replicants after the one deleted
10367da06231SAxel Dörfler 	RealignReplicants(startIndex);
103741281cf3SAxel Dörfler 
103841281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
103941281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
104041281cf3SAxel Dörfler 	float width, height;
104141281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
104241281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height) {
10431ad8c760SFredrik Holmqvist 		// resize view to accomodate the replicants, redraw as necessary
104441281cf3SAxel Dörfler 		AdjustPlacement();
104541281cf3SAxel Dörfler 	}
104641281cf3SAxel Dörfler }
104741281cf3SAxel Dörfler 
104841281cf3SAxel Dörfler 
10497da06231SAxel Dörfler /**	looking for a replicant by id/index
10507da06231SAxel Dörfler  *	return the view and index
10517da06231SAxel Dörfler  */
10527da06231SAxel Dörfler 
105341281cf3SAxel Dörfler BView*
105441281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, int32 target, bool byIndex)
105541281cf3SAxel Dörfler {
105641281cf3SAxel Dörfler 	*index = -1;
105741281cf3SAxel Dörfler 
105841281cf3SAxel Dörfler 	BView* view;
105941281cf3SAxel Dörfler 	if (byIndex) {
106041281cf3SAxel Dörfler 		if (fShelf->ReplicantAt(target, &view, (uint32*)id)) {
1061e4d65fe7SJohn Scipione 			if (view != NULL) {
106241281cf3SAxel Dörfler 				*index = target;
1063e4d65fe7SJohn Scipione 
106441281cf3SAxel Dörfler 				return view;
106541281cf3SAxel Dörfler 			}
106641281cf3SAxel Dörfler 		}
106741281cf3SAxel Dörfler 	} else {
106841281cf3SAxel Dörfler 		int32 count = fShelf->CountReplicants() - 1;
106941281cf3SAxel Dörfler 		int32 localid;
107041281cf3SAxel Dörfler 		for (int32 repIndex = count; repIndex >= 0; repIndex--) {
107141281cf3SAxel Dörfler 			fShelf->ReplicantAt(repIndex, &view, (uint32*)&localid);
1072e4d65fe7SJohn Scipione 			if (localid == target && view != NULL) {
107341281cf3SAxel Dörfler 				*index = repIndex;
107441281cf3SAxel Dörfler 				*id = localid;
1075e4d65fe7SJohn Scipione 
107641281cf3SAxel Dörfler 				return view;
107741281cf3SAxel Dörfler 			}
107841281cf3SAxel Dörfler 		}
107941281cf3SAxel Dörfler 	}
1080e4d65fe7SJohn Scipione 
108141281cf3SAxel Dörfler 	return NULL;
108241281cf3SAxel Dörfler }
108341281cf3SAxel Dörfler 
108441281cf3SAxel Dörfler 
10857da06231SAxel Dörfler /**	looking for a replicant with a view by name
10867da06231SAxel Dörfler  *	return the view, index and the id of the replicant
10877da06231SAxel Dörfler  */
10887da06231SAxel Dörfler 
108941281cf3SAxel Dörfler BView*
109041281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, const char* name)
109141281cf3SAxel Dörfler {
109241281cf3SAxel Dörfler 	*index = -1;
109341281cf3SAxel Dörfler 	*id = -1;
109441281cf3SAxel Dörfler 
109541281cf3SAxel Dörfler 	BView* view;
109641281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants() - 1;
109741281cf3SAxel Dörfler 	for (int32 repIndex = count; repIndex >= 0; repIndex--) {
109841281cf3SAxel Dörfler 		fShelf->ReplicantAt(repIndex, &view, (uint32*)id);
10993641948fSJohn Scipione 		if (view != NULL && view->Name() != NULL
1100e4d65fe7SJohn Scipione 			&& strcmp(name, view->Name()) == 0) {
110141281cf3SAxel Dörfler 			*index = repIndex;
1102e4d65fe7SJohn Scipione 
110341281cf3SAxel Dörfler 			return view;
110441281cf3SAxel Dörfler 		}
110541281cf3SAxel Dörfler 	}
1106e4d65fe7SJohn Scipione 
110741281cf3SAxel Dörfler 	return NULL;
110841281cf3SAxel Dörfler }
110941281cf3SAxel Dörfler 
111041281cf3SAxel Dörfler 
11117da06231SAxel Dörfler /**	Shelf will call to determine where and if
11127da06231SAxel Dörfler  *	the replicant is to be added
11137da06231SAxel Dörfler  */
11147da06231SAxel Dörfler 
111541281cf3SAxel Dörfler bool
111641281cf3SAxel Dörfler TReplicantTray::AcceptAddon(BRect replicantFrame, BMessage* message)
111741281cf3SAxel Dörfler {
1118e4d65fe7SJohn Scipione 	if (message == NULL)
111941281cf3SAxel Dörfler 		return false;
112041281cf3SAxel Dörfler 
112141281cf3SAxel Dörfler 	if (replicantFrame.Height() > kMaxReplicantHeight)
112241281cf3SAxel Dörfler 		return false;
112341281cf3SAxel Dörfler 
112441281cf3SAxel Dörfler 	alignment align = B_ALIGN_LEFT;
112541281cf3SAxel Dörfler 	if (fAlignmentSupport && message->HasBool("deskbar:dynamic_align")) {
112641281cf3SAxel Dörfler 		if (!fBarView->Vertical())
112741281cf3SAxel Dörfler 			align = B_ALIGN_RIGHT;
112841281cf3SAxel Dörfler 		else
112941281cf3SAxel Dörfler 			align = fBarView->Left() ? B_ALIGN_LEFT : B_ALIGN_RIGHT;
113041281cf3SAxel Dörfler 	} else if (message->HasInt32("deskbar:align"))
113141281cf3SAxel Dörfler 		message->FindInt32("deskbar:align", (int32*)&align);
113241281cf3SAxel Dörfler 
113341281cf3SAxel Dörfler 	if (message->HasInt32("deskbar:private_align"))
113441281cf3SAxel Dörfler 		message->FindInt32("deskbar:private_align", (int32*)&align);
113541281cf3SAxel Dörfler 	else
113641281cf3SAxel Dörfler 		align = B_ALIGN_LEFT;
113741281cf3SAxel Dörfler 
11387da06231SAxel Dörfler 	BPoint loc = LocationForReplicant(fShelf->CountReplicants(),
11397da06231SAxel Dörfler 		replicantFrame.Width());
114041281cf3SAxel Dörfler 	message->AddPoint("_pjp_loc", loc);
1141e4d65fe7SJohn Scipione 
114241281cf3SAxel Dörfler 	return true;
114341281cf3SAxel Dörfler }
114441281cf3SAxel Dörfler 
114541281cf3SAxel Dörfler 
114641281cf3SAxel Dörfler /**	based on the previous (index - 1) replicant in the list
114741281cf3SAxel Dörfler  *	calculate where the left point should be for this
114841281cf3SAxel Dörfler  *	replicant.  replicant will flow to the right on its own
114941281cf3SAxel Dörfler  */
115041281cf3SAxel Dörfler 
115141281cf3SAxel Dörfler BPoint
11527da06231SAxel Dörfler TReplicantTray::LocationForReplicant(int32 index, float width)
115341281cf3SAxel Dörfler {
115470ad5289SJohn Scipione 	BPoint loc(kTrayPadding, 2);
115570ad5289SJohn Scipione 	if (fBarView->Vertical()) {
115670ad5289SJohn Scipione 		if (!fBarView->Left())
115770ad5289SJohn Scipione 			loc.x += kDragWidth; // move past dragger
115870ad5289SJohn Scipione 	} else
115970ad5289SJohn Scipione 		loc.x += 1; // keeps everything lined up nicely
116041281cf3SAxel Dörfler 
11617da06231SAxel Dörfler 	if (fMultiRowMode) {
11627da06231SAxel Dörfler 		// try to find free space in every row
11637da06231SAxel Dörfler 		for (int32 row = 0; ; loc.y += kMaxReplicantHeight + kIconGap, row++) {
11647da06231SAxel Dörfler 			// determine free space in this row
11651bb2e623SJohn Scipione 			BRect rect(loc.x, loc.y,
11661bb2e623SJohn Scipione 				loc.x + static_cast<TBarApp*>(be_app)->Settings()->width
116770ad5289SJohn Scipione 					- kDragRegionWidth * 2,
11681bb2e623SJohn Scipione 				loc.y + kMaxReplicantHeight);
11691bb2e623SJohn Scipione 			if (row == 0 && !fTime->IsHidden()) {
117070ad5289SJohn Scipione 				rowRect.right -= kClockMargin + fTime->Frame().Width()
117170ad5289SJohn Scipione 					+ kTrayPadding;
11721bb2e623SJohn Scipione 			}
11737da06231SAxel Dörfler 
11747da06231SAxel Dörfler 			for (int32 i = 0; i < index; i++) {
11757da06231SAxel Dörfler 				BView* view = NULL;
11767da06231SAxel Dörfler 				fShelf->ReplicantAt(i, &view);
11777da06231SAxel Dörfler 				if (view == NULL || view->Frame().top != rect.top)
11787da06231SAxel Dörfler 					continue;
11797da06231SAxel Dörfler 
11807da06231SAxel Dörfler 				rect.left = view->Frame().right + kIconGap + 1;
11817da06231SAxel Dörfler 			}
11827da06231SAxel Dörfler 
11837da06231SAxel Dörfler 			if (rect.Width() >= width) {
11847da06231SAxel Dörfler 				// the icon fits in this row
11857da06231SAxel Dörfler 				loc = rect.LeftTop();
11867da06231SAxel Dörfler 				break;
11877da06231SAxel Dörfler 			}
11887da06231SAxel Dörfler 		}
11897da06231SAxel Dörfler 	} else {
119041281cf3SAxel Dörfler 		if (index > 0) {
119141281cf3SAxel Dörfler 			// get the last replicant added for placement reference
119241281cf3SAxel Dörfler 			BView* view = NULL;
11937da06231SAxel Dörfler 			fShelf->ReplicantAt(index - 1, &view);
1194e4d65fe7SJohn Scipione 			if (view != NULL) {
119541281cf3SAxel Dörfler 				// push this rep placement past the last one
119641281cf3SAxel Dörfler 				loc.x = view->Frame().right + kIconGap + 1;
119741281cf3SAxel Dörfler 				loc.y = view->Frame().top;
119841281cf3SAxel Dörfler 			}
119941281cf3SAxel Dörfler 		}
12007da06231SAxel Dörfler 	}
120141281cf3SAxel Dörfler 
1202e4d65fe7SJohn Scipione 	if (loc.y > fRightBottomReplicant.top
1203e4d65fe7SJohn Scipione 		|| (loc.y == fRightBottomReplicant.top
1204e4d65fe7SJohn Scipione 			&& loc.x > fRightBottomReplicant.left)) {
1205e4d65fe7SJohn Scipione 		fRightBottomReplicant.Set(loc.x, loc.y, loc.x + width,
1206e4d65fe7SJohn Scipione 			loc.y + kMaxReplicantHeight);
12077da06231SAxel Dörfler 		fLastReplicant = index;
120841281cf3SAxel Dörfler 	}
12097da06231SAxel Dörfler 
121041281cf3SAxel Dörfler 	return loc;
121141281cf3SAxel Dörfler }
121241281cf3SAxel Dörfler 
121341281cf3SAxel Dörfler 
121441281cf3SAxel Dörfler BRect
121541281cf3SAxel Dörfler TReplicantTray::IconFrame(int32 target, bool byIndex)
121641281cf3SAxel Dörfler {
1217378aa08bSJohn Scipione 	int32 index;
1218378aa08bSJohn Scipione 	int32 id;
121941281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
122041281cf3SAxel Dörfler 
1221378aa08bSJohn Scipione 	return view != NULL ? view->Frame() : BRect(0, 0, 0, 0);
122241281cf3SAxel Dörfler }
122341281cf3SAxel Dörfler 
122441281cf3SAxel Dörfler 
122541281cf3SAxel Dörfler BRect
122641281cf3SAxel Dörfler TReplicantTray::IconFrame(const char* name)
122741281cf3SAxel Dörfler {
1228e4d65fe7SJohn Scipione 	if (name == NULL)
122941281cf3SAxel Dörfler 		return BRect(0, 0, 0, 0);
123041281cf3SAxel Dörfler 
1231378aa08bSJohn Scipione 	int32 index;
1232378aa08bSJohn Scipione 	int32 id;
123341281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
123441281cf3SAxel Dörfler 
1235378aa08bSJohn Scipione 	return view != NULL ? view->Frame() : BRect(0, 0, 0, 0);
123641281cf3SAxel Dörfler }
123741281cf3SAxel Dörfler 
123841281cf3SAxel Dörfler 
12397da06231SAxel Dörfler /**	Scan from the startIndex and reset the location
12407da06231SAxel Dörfler  *	as defined in LocationForReplicant()
124141281cf3SAxel Dörfler  */
124241281cf3SAxel Dörfler 
124341281cf3SAxel Dörfler void
124441281cf3SAxel Dörfler TReplicantTray::RealignReplicants(int32 startIndex)
124541281cf3SAxel Dörfler {
124641281cf3SAxel Dörfler 	if (startIndex < 0)
124741281cf3SAxel Dörfler 		startIndex = 0;
124841281cf3SAxel Dörfler 
124941281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants();
125041281cf3SAxel Dörfler 	if (count <= 0)
125141281cf3SAxel Dörfler 		return;
125241281cf3SAxel Dörfler 
12537da06231SAxel Dörfler 	if (startIndex == 0)
12547da06231SAxel Dörfler 		fRightBottomReplicant.Set(0, 0, 0, 0);
12557da06231SAxel Dörfler 
125641281cf3SAxel Dörfler 	BView* view = NULL;
125741281cf3SAxel Dörfler 	for (int32 i = startIndex; i < count; i++) {
125841281cf3SAxel Dörfler 		fShelf->ReplicantAt(i, &view);
12590bec7100SStefano Ceccherini 		if (view != NULL) {
12607da06231SAxel Dörfler 			BPoint loc = LocationForReplicant(i, view->Frame().Width());
12610bec7100SStefano Ceccherini 			if (view->Frame().LeftTop() != loc)
126241281cf3SAxel Dörfler 				view->MoveTo(loc);
126341281cf3SAxel Dörfler 		}
126441281cf3SAxel Dörfler 	}
126541281cf3SAxel Dörfler }
126641281cf3SAxel Dörfler 
126741281cf3SAxel Dörfler 
12688aab28f1SRene Gollent status_t
12697625ce51SRene Gollent TReplicantTray::_SaveSettings()
12707625ce51SRene Gollent {
12717625ce51SRene Gollent 	status_t result;
12727625ce51SRene Gollent 	BPath path;
127326ea7066SIngo Weinhold 	if ((result = GetDeskbarSettingsDirectory(path, true)) == B_OK) {
12747625ce51SRene Gollent 		path.Append(kReplicantSettingsFile);
12757625ce51SRene Gollent 
12767625ce51SRene Gollent 		BFile file(path.Path(), B_READ_WRITE | B_CREATE_FILE | B_ERASE_FILE);
12777625ce51SRene Gollent 		if ((result = file.InitCheck()) == B_OK)
12787625ce51SRene Gollent 			result = fAddOnSettings.Flatten(&file);
12797625ce51SRene Gollent 	}
12807625ce51SRene Gollent 
12817625ce51SRene Gollent 	return result;
12827625ce51SRene Gollent }
12837625ce51SRene Gollent 
12847625ce51SRene Gollent 
12853cf2d117SJohn Scipione void
12863cf2d117SJohn Scipione TReplicantTray::SaveTimeSettings()
12873cf2d117SJohn Scipione {
12883cf2d117SJohn Scipione 	if (fTime == NULL)
12893cf2d117SJohn Scipione 		return;
12903cf2d117SJohn Scipione 
129131c0024dSJohn Scipione 	clock_settings* settings = ((TBarApp*)be_app)->ClockSettings();
12923cf2d117SJohn Scipione 	settings->showSeconds = fTime->ShowSeconds();
12933cf2d117SJohn Scipione 	settings->showDayOfWeek = fTime->ShowDayOfWeek();
129431c0024dSJohn Scipione 	settings->showTimeZone = fTime->ShowTimeZone();
12953cf2d117SJohn Scipione }
12963cf2d117SJohn Scipione 
12973cf2d117SJohn Scipione 
12981bb2e623SJohn Scipione //	#pragma mark - TDragRegion
129941281cf3SAxel Dörfler 
130041281cf3SAxel Dörfler 
1301ca9acc20SAxel Dörfler /*! Draggable region that is asynchronous so that dragging does not block
1302ca9acc20SAxel Dörfler 	other activities.
13037da06231SAxel Dörfler */
130441281cf3SAxel Dörfler TDragRegion::TDragRegion(TBarView* parent, BView* child)
1305ca9acc20SAxel Dörfler 	:
1306ca9acc20SAxel Dörfler 	BControl(BRect(0, 0, 0, 0), "", "", NULL, B_FOLLOW_NONE,
130741281cf3SAxel Dörfler 		B_WILL_DRAW | B_FRAME_EVENTS),
130841281cf3SAxel Dörfler 	fBarView(parent),
130941281cf3SAxel Dörfler 	fChild(child),
131041281cf3SAxel Dörfler 	fDragLocation(kAutoPlaceDragRegion)
131141281cf3SAxel Dörfler {
131241281cf3SAxel Dörfler }
131341281cf3SAxel Dörfler 
131441281cf3SAxel Dörfler 
131541281cf3SAxel Dörfler void
131641281cf3SAxel Dörfler TDragRegion::AttachedToWindow()
131741281cf3SAxel Dörfler {
131841281cf3SAxel Dörfler 	BView::AttachedToWindow();
1319e4d65fe7SJohn Scipione 
1320cb6afcb1SStephan Aßmus 	if (be_control_look != NULL)
13211bb2e623SJohn Scipione 		SetViewUIColor(B_MENU_BACKGROUND_COLOR, 1.1);
1322cb6afcb1SStephan Aßmus 	else
132316c10517Slooncraz 		SetViewUIColor(B_MENU_BACKGROUND_COLOR);
1324e4d65fe7SJohn Scipione 
132541281cf3SAxel Dörfler 	ResizeToPreferred();
132641281cf3SAxel Dörfler }
132741281cf3SAxel Dörfler 
132841281cf3SAxel Dörfler 
132941281cf3SAxel Dörfler void
133041281cf3SAxel Dörfler TDragRegion::GetPreferredSize(float* width, float* height)
133141281cf3SAxel Dörfler {
133241281cf3SAxel Dörfler 	fChild->ResizeToPreferred();
133341281cf3SAxel Dörfler 	*width = fChild->Bounds().Width();
133441281cf3SAxel Dörfler 	*height = fChild->Bounds().Height();
133541281cf3SAxel Dörfler 
133641281cf3SAxel Dörfler 	if (fDragLocation != kNoDragRegion)
13371bb2e623SJohn Scipione 		*width += kDragWidth + kGutter;
133841281cf3SAxel Dörfler 	else
133941281cf3SAxel Dörfler 		*width += 6;
134041281cf3SAxel Dörfler 
13411bb2e623SJohn Scipione 	*height += 2;
134241281cf3SAxel Dörfler }
134341281cf3SAxel Dörfler 
134441281cf3SAxel Dörfler 
134541281cf3SAxel Dörfler void
134670ad5289SJohn Scipione TDragRegion::Draw(BRect updateRect)
134741281cf3SAxel Dörfler {
1348eaa9af99SStephan Aßmus 	rgb_color menuColor = ViewColor();
134941281cf3SAxel Dörfler 	rgb_color hilite = tint_color(menuColor, B_DARKEN_1_TINT);
1350cb6afcb1SStephan Aßmus 	rgb_color ldark = tint_color(menuColor, 1.02);
1351cb6afcb1SStephan Aßmus 	rgb_color dark = tint_color(menuColor, B_DARKEN_2_TINT);
135241281cf3SAxel Dörfler 	rgb_color vvdark = tint_color(menuColor, B_DARKEN_4_TINT);
135341281cf3SAxel Dörfler 	rgb_color light = tint_color(menuColor, B_LIGHTEN_2_TINT);
135441281cf3SAxel Dörfler 
135541281cf3SAxel Dörfler 	BRect frame(Bounds());
135641281cf3SAxel Dörfler 	BeginLineArray(4);
135741281cf3SAxel Dörfler 
1358cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1359cb6afcb1SStephan Aßmus 		if (fBarView->Vertical()) {
1360cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(), frame.RightTop(), dark);
1361cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1362cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.top + 1), ldark);
1363cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
1364cb6afcb1SStephan Aßmus 		} else if (fBarView->AcrossTop() || fBarView->AcrossBottom()) {
1365cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(),
1366cb6afcb1SStephan Aßmus 				BPoint(frame.left, frame.bottom), dark);
1367cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.top + 1),
1368cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
1369cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right, frame.top + 2),
1370cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.bottom), hilite);
1371cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.bottom),
1372cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
1373cb6afcb1SStephan Aßmus 		}
1374cb6afcb1SStephan Aßmus 	} else {
137541281cf3SAxel Dörfler 		if (fBarView->Vertical()) {
137641281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.RightTop(), light);
137741281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.LeftBottom(), light);
137841281cf3SAxel Dörfler 			AddLine(frame.RightBottom(), frame.RightTop(), hilite);
137941281cf3SAxel Dörfler 		} else if (fBarView->AcrossTop()) {
1380cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1381cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
138241281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1383cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 2),
1384cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1385cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(),
1386cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
138741281cf3SAxel Dörfler 		} else if (fBarView->AcrossBottom()) {
1388cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1389cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
139041281cf3SAxel Dörfler 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
139141281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1392cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 1),
1393cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1394cb6afcb1SStephan Aßmus 		}
139541281cf3SAxel Dörfler 	}
139641281cf3SAxel Dörfler 
139741281cf3SAxel Dörfler 	EndLineArray();
139841281cf3SAxel Dörfler 
139941281cf3SAxel Dörfler 	if (fDragLocation != kDontDrawDragRegion || fDragLocation != kNoDragRegion)
140041281cf3SAxel Dörfler 		DrawDragRegion();
140141281cf3SAxel Dörfler }
140241281cf3SAxel Dörfler 
140341281cf3SAxel Dörfler 
140441281cf3SAxel Dörfler void
140541281cf3SAxel Dörfler TDragRegion::DrawDragRegion()
140641281cf3SAxel Dörfler {
140741281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
140841281cf3SAxel Dörfler 
1409cb6afcb1SStephan Aßmus 	rgb_color menuColor = ViewColor();
1410cb6afcb1SStephan Aßmus 	rgb_color menuHilite = menuColor;
141141281cf3SAxel Dörfler 	if (IsTracking()) {
14121bb2e623SJohn Scipione 		// draw drag region highlighted if tracking mouse
1413cb6afcb1SStephan Aßmus 		menuHilite = tint_color(menuColor, B_HIGHLIGHT_BACKGROUND_TINT);
141441281cf3SAxel Dörfler 		SetHighColor(menuHilite);
141541281cf3SAxel Dörfler 		FillRect(dragRegion);
14161bb2e623SJohn Scipione 	} else {
14171bb2e623SJohn Scipione 		SetHighColor(menuColor);
14181bb2e623SJohn Scipione 		FillRect(dragRegion);
141941281cf3SAxel Dörfler 	}
14201bb2e623SJohn Scipione 
1421cb6afcb1SStephan Aßmus 	rgb_color vdark = tint_color(menuHilite, B_DARKEN_3_TINT);
1422cb6afcb1SStephan Aßmus 	rgb_color light = tint_color(menuHilite, B_LIGHTEN_2_TINT);
1423cb6afcb1SStephan Aßmus 
1424cb6afcb1SStephan Aßmus 	BeginLineArray(dragRegion.IntegerHeight());
1425e4d65fe7SJohn Scipione 	BPoint where;
1426e4d65fe7SJohn Scipione 	where.x = floorf((dragRegion.left + dragRegion.right) / 2 + 0.5) - 1;
1427e4d65fe7SJohn Scipione 	where.y = dragRegion.top + 2;
1428cb6afcb1SStephan Aßmus 
1429e4d65fe7SJohn Scipione 	while (where.y + 1 <= dragRegion.bottom) {
1430e4d65fe7SJohn Scipione 		AddLine(where, where, vdark);
1431e4d65fe7SJohn Scipione 		AddLine(where + BPoint(1, 1), where + BPoint(1, 1), light);
143241281cf3SAxel Dörfler 
1433e4d65fe7SJohn Scipione 		where.y += 3;
143441281cf3SAxel Dörfler 	}
143541281cf3SAxel Dörfler 	EndLineArray();
143641281cf3SAxel Dörfler }
143741281cf3SAxel Dörfler 
143841281cf3SAxel Dörfler 
143941281cf3SAxel Dörfler BRect
144041281cf3SAxel Dörfler TDragRegion::DragRegion() const
144141281cf3SAxel Dörfler {
1442cb6afcb1SStephan Aßmus 	float kTopBottomInset = 2;
1443cb6afcb1SStephan Aßmus 	float kLeftRightInset = 1;
1444cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1445cb6afcb1SStephan Aßmus 		kTopBottomInset = 1;
1446cb6afcb1SStephan Aßmus 		kLeftRightInset = 0;
1447cb6afcb1SStephan Aßmus 	}
1448cb6afcb1SStephan Aßmus 
144941281cf3SAxel Dörfler 	BRect dragRegion(Bounds());
1450cb6afcb1SStephan Aßmus 	dragRegion.top += kTopBottomInset;
1451cb6afcb1SStephan Aßmus 	dragRegion.bottom -= kTopBottomInset;
145241281cf3SAxel Dörfler 
145341281cf3SAxel Dörfler 	bool placeOnLeft = false;
145441281cf3SAxel Dörfler 	if (fDragLocation == kAutoPlaceDragRegion) {
145541281cf3SAxel Dörfler 		if (fBarView->Vertical() && fBarView->Left())
145641281cf3SAxel Dörfler 			placeOnLeft = true;
145741281cf3SAxel Dörfler 		else
145841281cf3SAxel Dörfler 			placeOnLeft = false;
14591bb2e623SJohn Scipione 	} else
14601bb2e623SJohn Scipione 		placeOnLeft = fDragLocation == kDragRegionLeft;
146141281cf3SAxel Dörfler 
146241281cf3SAxel Dörfler 	if (placeOnLeft) {
1463cb6afcb1SStephan Aßmus 		dragRegion.left += kLeftRightInset;
1464cb6afcb1SStephan Aßmus 		dragRegion.right = dragRegion.left + kDragWidth;
146541281cf3SAxel Dörfler 	} else {
14661bb2e623SJohn Scipione 		dragRegion.right += kLeftRightInset;
1467cb6afcb1SStephan Aßmus 		dragRegion.left = dragRegion.right - kDragWidth;
146841281cf3SAxel Dörfler 	}
146941281cf3SAxel Dörfler 
147041281cf3SAxel Dörfler 	return dragRegion;
147141281cf3SAxel Dörfler }
147241281cf3SAxel Dörfler 
147341281cf3SAxel Dörfler 
147441281cf3SAxel Dörfler void
1475e4d65fe7SJohn Scipione TDragRegion::MouseDown(BPoint where)
147641281cf3SAxel Dörfler {
1477e9632898SAlex Smith 	uint32 buttons;
1478e4d65fe7SJohn Scipione 	BPoint mouseLoc;
147941281cf3SAxel Dörfler 
1480e4d65fe7SJohn Scipione 	BRect dragRegion(DragRegion());
14811bb2e623SJohn Scipione 	dragRegion.InsetBy(-2, -2);
14827da06231SAxel Dörfler 		// DragRegion() is designed for drawing, not clicking
148341281cf3SAxel Dörfler 
1484e4d65fe7SJohn Scipione 	if (!dragRegion.Contains(where))
148541281cf3SAxel Dörfler 		return;
148641281cf3SAxel Dörfler 
148741281cf3SAxel Dörfler 	while (true) {
1488e4d65fe7SJohn Scipione 		GetMouse(&mouseLoc, &buttons);
1489e4d65fe7SJohn Scipione 		if (buttons == 0)
149041281cf3SAxel Dörfler 			break;
149141281cf3SAxel Dörfler 
149241281cf3SAxel Dörfler 		if ((Window()->Flags() & B_ASYNCHRONOUS_CONTROLS) != 0) {
1493e4d65fe7SJohn Scipione 			fPreviousPosition = where;
149441281cf3SAxel Dörfler 			SetTracking(true);
149541281cf3SAxel Dörfler 			SetMouseEventMask(B_POINTER_EVENTS,
149641281cf3SAxel Dörfler 				B_NO_POINTER_HISTORY | B_LOCK_WINDOW_FOCUS);
149741281cf3SAxel Dörfler 			Invalidate(DragRegion());
149841281cf3SAxel Dörfler 			break;
149941281cf3SAxel Dörfler 		}
150041281cf3SAxel Dörfler 
150141281cf3SAxel Dörfler 		snooze(25000);
150241281cf3SAxel Dörfler 	}
150341281cf3SAxel Dörfler }
150441281cf3SAxel Dörfler 
150541281cf3SAxel Dörfler 
150641281cf3SAxel Dörfler void
1507e4d65fe7SJohn Scipione TDragRegion::MouseUp(BPoint where)
150841281cf3SAxel Dörfler {
150941281cf3SAxel Dörfler 	if (IsTracking()) {
151041281cf3SAxel Dörfler 		SetTracking(false);
151141281cf3SAxel Dörfler 		Invalidate(DragRegion());
151241281cf3SAxel Dörfler 	} else
1513e4d65fe7SJohn Scipione 		BControl::MouseUp(where);
151441281cf3SAxel Dörfler }
151541281cf3SAxel Dörfler 
151641281cf3SAxel Dörfler 
151741281cf3SAxel Dörfler bool
1518e4d65fe7SJohn Scipione TDragRegion::SwitchModeForRect(BPoint where, BRect rect,
151941281cf3SAxel Dörfler 	bool newVertical, bool newLeft, bool newTop, int32 newState)
152041281cf3SAxel Dörfler {
1521e4d65fe7SJohn Scipione 	if (!rect.Contains(where)) {
152241281cf3SAxel Dörfler 		// not our rect
152341281cf3SAxel Dörfler 		return false;
15241ad8c760SFredrik Holmqvist 	}
152541281cf3SAxel Dörfler 
15261ad8c760SFredrik Holmqvist 	if (newVertical == fBarView->Vertical() && newLeft == fBarView->Left()
15271ad8c760SFredrik Holmqvist 		&& newTop == fBarView->Top() && newState == fBarView->State()) {
152841281cf3SAxel Dörfler 		// already in the correct mode
152941281cf3SAxel Dörfler 		return true;
15301ad8c760SFredrik Holmqvist 	}
153141281cf3SAxel Dörfler 
1532d7ed9414SRene Gollent 	fBarView->ChangeState(newState, newVertical, newLeft, newTop, true);
1533e4d65fe7SJohn Scipione 
153441281cf3SAxel Dörfler 	return true;
153541281cf3SAxel Dörfler }
153641281cf3SAxel Dörfler 
153741281cf3SAxel Dörfler 
153841281cf3SAxel Dörfler void
153941281cf3SAxel Dörfler TDragRegion::MouseMoved(BPoint where, uint32 code, const BMessage* message)
154041281cf3SAxel Dörfler {
154141281cf3SAxel Dörfler 	if (IsTracking()) {
154241281cf3SAxel Dörfler 		BScreen screen;
154341281cf3SAxel Dörfler 		BRect frame = screen.Frame();
154441281cf3SAxel Dörfler 
154541281cf3SAxel Dörfler 		float hDivider = frame.Width() / 6;
15461bb2e623SJohn Scipione 		hDivider = (hDivider < gMinimumWindowWidth + 10)
15471bb2e623SJohn Scipione 			? gMinimumWindowWidth + 10 : hDivider;
15481bb2e623SJohn Scipione 		float miniDivider = frame.top + kMiniHeight + 10;
154941281cf3SAxel Dörfler 		float vDivider = frame.Height() / 2;
155041281cf3SAxel Dörfler #ifdef FULL_MODE
155141281cf3SAxel Dörfler 		float thirdScreen = frame.Height() / 3;
155241281cf3SAxel Dörfler #endif
155371bd3ba5SJonas Sundström 		BRect topLeft(frame.left, frame.top, frame.left + hDivider,
155471bd3ba5SJonas Sundström 			miniDivider);
155571bd3ba5SJonas Sundström 		BRect topMiddle(frame.left + hDivider, frame.top, frame.right
155671bd3ba5SJonas Sundström 			- hDivider, vDivider);
155771bd3ba5SJonas Sundström 		BRect topRight(frame.right - hDivider, frame.top, frame.right,
155871bd3ba5SJonas Sundström 			miniDivider);
155941281cf3SAxel Dörfler 
156041281cf3SAxel Dörfler #ifdef FULL_MODE
156141281cf3SAxel Dörfler 		vDivider = miniDivider + thirdScreen;
156241281cf3SAxel Dörfler #endif
156371bd3ba5SJonas Sundström 		BRect middleLeft(frame.left, miniDivider, frame.left + hDivider,
156471bd3ba5SJonas Sundström 			vDivider);
156571bd3ba5SJonas Sundström 		BRect middleRight(frame.right - hDivider, miniDivider, frame.right,
156671bd3ba5SJonas Sundström 			vDivider);
156741281cf3SAxel Dörfler 
156841281cf3SAxel Dörfler #ifdef FULL_MODE
156971bd3ba5SJonas Sundström 		BRect leftSide(frame.left, vDivider, frame.left + hDivider,
157071bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
157171bd3ba5SJonas Sundström 		BRect rightSide(frame.right - hDivider, vDivider, frame.right,
157271bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
157341281cf3SAxel Dörfler 
157441281cf3SAxel Dörfler 		vDivider = frame.bottom - thirdScreen;
157541281cf3SAxel Dörfler #endif
157671bd3ba5SJonas Sundström 		BRect bottomLeft(frame.left, vDivider, frame.left + hDivider,
157771bd3ba5SJonas Sundström 			frame.bottom);
157871bd3ba5SJonas Sundström 		BRect bottomMiddle(frame.left + hDivider, vDivider, frame.right
157971bd3ba5SJonas Sundström 			- hDivider, frame.bottom);
158071bd3ba5SJonas Sundström 		BRect bottomRight(frame.right - hDivider, vDivider, frame.right,
158171bd3ba5SJonas Sundström 			frame.bottom);
158241281cf3SAxel Dörfler 
158341281cf3SAxel Dörfler 		if (where != fPreviousPosition) {
158441281cf3SAxel Dörfler 			fPreviousPosition = where;
158541281cf3SAxel Dörfler 			ConvertToScreen(&where);
158641281cf3SAxel Dörfler 
158741281cf3SAxel Dörfler 			// use short circuit evaluation for convenience
158841281cf3SAxel Dörfler 			if (SwitchModeForRect(where, topLeft, true, true, true, kMiniState)
158971bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topMiddle, false, true, true,
159071bd3ba5SJonas Sundström 					kExpandoState)
159171bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topRight, true, false, true,
159271bd3ba5SJonas Sundström 					kMiniState)
159371bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleLeft, true, true, true,
159471bd3ba5SJonas Sundström 					kExpandoState)
159571bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleRight, true, false, true,
159671bd3ba5SJonas Sundström 					kExpandoState)
159741281cf3SAxel Dörfler 
159841281cf3SAxel Dörfler #ifdef FULL_MODE
159971bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, leftSide, true, true, true,
160071bd3ba5SJonas Sundström 					kFullState)
160171bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, rightSide, true, false, true,
160271bd3ba5SJonas Sundström 					kFullState)
160341281cf3SAxel Dörfler #endif
160471bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomLeft, true, true, false,
160571bd3ba5SJonas Sundström 					kMiniState)
160671bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomMiddle, false, true, false,
160771bd3ba5SJonas Sundström 					kExpandoState)
160871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomRight, true, false, false,
160971bd3ba5SJonas Sundström 					kMiniState))
161041281cf3SAxel Dörfler 				;
161141281cf3SAxel Dörfler 		}
161241281cf3SAxel Dörfler 	} else
161341281cf3SAxel Dörfler 		BControl::MouseMoved(where, code, message);
161441281cf3SAxel Dörfler }
161541281cf3SAxel Dörfler 
161641281cf3SAxel Dörfler 
161741281cf3SAxel Dörfler int32
161841281cf3SAxel Dörfler TDragRegion::DragRegionLocation() const
161941281cf3SAxel Dörfler {
162041281cf3SAxel Dörfler 	return fDragLocation;
162141281cf3SAxel Dörfler }
162241281cf3SAxel Dörfler 
162341281cf3SAxel Dörfler 
162441281cf3SAxel Dörfler void
162541281cf3SAxel Dörfler TDragRegion::SetDragRegionLocation(int32 location)
162641281cf3SAxel Dörfler {
162741281cf3SAxel Dörfler 	if (location == fDragLocation)
162841281cf3SAxel Dörfler 		return;
162941281cf3SAxel Dörfler 
163041281cf3SAxel Dörfler 	fDragLocation = location;
163141281cf3SAxel Dörfler 	Invalidate();
163241281cf3SAxel Dörfler }
16331bb2e623SJohn Scipione 
16341bb2e623SJohn Scipione 
16351bb2e623SJohn Scipione //	#pragma mark - TResizeControl
16361bb2e623SJohn Scipione 
16371bb2e623SJohn Scipione 
16381bb2e623SJohn Scipione /*! Draggable region that is asynchronous so that resizing does not block.
16391bb2e623SJohn Scipione */
16401bb2e623SJohn Scipione TResizeControl::TResizeControl(TBarView* barView)
16411bb2e623SJohn Scipione 	:
16421bb2e623SJohn Scipione 	BControl(BRect(0, kDragWidth, 0, kMenuBarHeight), "", "", NULL,
16431bb2e623SJohn Scipione 		B_FOLLOW_NONE, B_WILL_DRAW | B_FRAME_EVENTS),
16441bb2e623SJohn Scipione 	fBarView(barView)
16451bb2e623SJohn Scipione {
16461bb2e623SJohn Scipione }
16471bb2e623SJohn Scipione 
16481bb2e623SJohn Scipione 
16491bb2e623SJohn Scipione TResizeControl::~TResizeControl()
16501bb2e623SJohn Scipione {
16511bb2e623SJohn Scipione }
16521bb2e623SJohn Scipione 
16531bb2e623SJohn Scipione 
16541bb2e623SJohn Scipione void
16551bb2e623SJohn Scipione TResizeControl::AttachedToWindow()
16561bb2e623SJohn Scipione {
16571bb2e623SJohn Scipione 	BView::AttachedToWindow();
16581bb2e623SJohn Scipione 
16591bb2e623SJohn Scipione 	if (be_control_look != NULL)
16601bb2e623SJohn Scipione 		SetViewUIColor(B_MENU_BACKGROUND_COLOR, 1.1);
16611bb2e623SJohn Scipione 	else
16621bb2e623SJohn Scipione 		SetViewUIColor(B_MENU_BACKGROUND_COLOR);
16631bb2e623SJohn Scipione }
16641bb2e623SJohn Scipione 
16651bb2e623SJohn Scipione 
16661bb2e623SJohn Scipione void
16671bb2e623SJohn Scipione TResizeControl::Draw(BRect updateRect)
16681bb2e623SJohn Scipione {
16691bb2e623SJohn Scipione 	if (!fBarView->Vertical())
16701bb2e623SJohn Scipione 		return;
16711bb2e623SJohn Scipione 
1672092e918bSJohn Scipione 	BRect dragRegion(Bounds());
16731bb2e623SJohn Scipione 
167400171c6fSJohn Scipione 	int32 height = dragRegion.IntegerHeight();
167500171c6fSJohn Scipione 	if (height <= 0)
167600171c6fSJohn Scipione 		return;
167700171c6fSJohn Scipione 
16781bb2e623SJohn Scipione 	rgb_color menuColor = ViewColor();
16791bb2e623SJohn Scipione 	rgb_color menuHilite = menuColor;
16801bb2e623SJohn Scipione 	if (IsTracking()) {
16811bb2e623SJohn Scipione 		// draw drag region highlighted if tracking mouse
16821bb2e623SJohn Scipione 		menuHilite = tint_color(menuColor, B_HIGHLIGHT_BACKGROUND_TINT);
16831bb2e623SJohn Scipione 		SetHighColor(menuHilite);
16841bb2e623SJohn Scipione 		FillRect(dragRegion);
16851bb2e623SJohn Scipione 	} else {
16861bb2e623SJohn Scipione 		SetHighColor(menuColor);
16871bb2e623SJohn Scipione 		FillRect(dragRegion);
16881bb2e623SJohn Scipione 	}
16891bb2e623SJohn Scipione 
16901bb2e623SJohn Scipione 	rgb_color vdark = tint_color(menuHilite, B_DARKEN_3_TINT);
16911bb2e623SJohn Scipione 	rgb_color light = tint_color(menuHilite, B_LIGHTEN_2_TINT);
16921bb2e623SJohn Scipione 
169300171c6fSJohn Scipione 	BeginLineArray(height);
16941bb2e623SJohn Scipione 	BPoint where;
16951bb2e623SJohn Scipione 	where.x = floorf((dragRegion.left + dragRegion.right) / 2 + 0.5) - 1;
16961bb2e623SJohn Scipione 	where.y = dragRegion.top + 2;
16971bb2e623SJohn Scipione 
16981bb2e623SJohn Scipione 	while (where.y + 1 <= dragRegion.bottom) {
16991bb2e623SJohn Scipione 		AddLine(where, where, vdark);
17001bb2e623SJohn Scipione 		AddLine(where + BPoint(1, 1), where + BPoint(1, 1), light);
17011bb2e623SJohn Scipione 
17021bb2e623SJohn Scipione 		where.y += 3;
17031bb2e623SJohn Scipione 	}
17041bb2e623SJohn Scipione 	EndLineArray();
17051bb2e623SJohn Scipione }
17061bb2e623SJohn Scipione 
17071bb2e623SJohn Scipione 
17081bb2e623SJohn Scipione void
17091bb2e623SJohn Scipione TResizeControl::MouseDown(BPoint where)
17101bb2e623SJohn Scipione {
17111bb2e623SJohn Scipione 	uint32 buttons;
17121bb2e623SJohn Scipione 	BPoint mouseLoc;
17131bb2e623SJohn Scipione 
17141bb2e623SJohn Scipione 	while (true) {
17151bb2e623SJohn Scipione 		GetMouse(&mouseLoc, &buttons);
17161bb2e623SJohn Scipione 		if (buttons == 0)
17171bb2e623SJohn Scipione 			break;
17181bb2e623SJohn Scipione 
17191bb2e623SJohn Scipione 		if ((Window()->Flags() & B_ASYNCHRONOUS_CONTROLS) != 0) {
17201bb2e623SJohn Scipione 			SetTracking(true);
17211bb2e623SJohn Scipione 			SetMouseEventMask(B_POINTER_EVENTS,
17221bb2e623SJohn Scipione 				B_NO_POINTER_HISTORY | B_LOCK_WINDOW_FOCUS);
17231bb2e623SJohn Scipione 			Invalidate();
17241bb2e623SJohn Scipione 			break;
17251bb2e623SJohn Scipione 		}
17261bb2e623SJohn Scipione 
17271bb2e623SJohn Scipione 		snooze(25000);
17281bb2e623SJohn Scipione 	}
17291bb2e623SJohn Scipione }
17301bb2e623SJohn Scipione 
17311bb2e623SJohn Scipione 
17321bb2e623SJohn Scipione void
17331bb2e623SJohn Scipione TResizeControl::MouseUp(BPoint where)
17341bb2e623SJohn Scipione {
17351bb2e623SJohn Scipione 	if (IsTracking()) {
17361bb2e623SJohn Scipione 		SetTracking(false);
17371bb2e623SJohn Scipione 		Invalidate();
17381bb2e623SJohn Scipione 	} else
17391bb2e623SJohn Scipione 		BControl::MouseUp(where);
17401bb2e623SJohn Scipione }
17411bb2e623SJohn Scipione 
17421bb2e623SJohn Scipione 
17431bb2e623SJohn Scipione void
17441bb2e623SJohn Scipione TResizeControl::MouseMoved(BPoint where, uint32 code,
17451bb2e623SJohn Scipione 	const BMessage* dragMessage)
17461bb2e623SJohn Scipione {
17471bb2e623SJohn Scipione 	if (fBarView->Vertical()) {
17481bb2e623SJohn Scipione 		if (IsResizing()) {
17491bb2e623SJohn Scipione 			float windowWidth = Window()->Frame().Width();
17501bb2e623SJohn Scipione 			float delta = 0;
17511bb2e623SJohn Scipione 			BPoint whereScreen = ConvertToScreen(where);
17521bb2e623SJohn Scipione 
17531bb2e623SJohn Scipione 			if (fBarView->Left()) {
17541bb2e623SJohn Scipione 				delta = whereScreen.x - Window()->Frame().right;
17551bb2e623SJohn Scipione 				if (delta > 0 && windowWidth >= gMaximumWindowWidth)
17561bb2e623SJohn Scipione 					; // do nothing
17571bb2e623SJohn Scipione 				else if (delta < 0 && windowWidth <= gMinimumWindowWidth)
17581bb2e623SJohn Scipione 					; // do nothing
17591bb2e623SJohn Scipione 				else
17601bb2e623SJohn Scipione 					Window()->ResizeBy(delta, 0);
17611bb2e623SJohn Scipione 			} else {
17621bb2e623SJohn Scipione 				delta = Window()->Frame().left - whereScreen.x;
17631bb2e623SJohn Scipione 				if (delta > 0 && windowWidth >= gMaximumWindowWidth)
17641bb2e623SJohn Scipione 					; // do nothing
17651bb2e623SJohn Scipione 				else if (delta < 0 && windowWidth <= gMinimumWindowWidth)
17661bb2e623SJohn Scipione 					; // do nothing
17671bb2e623SJohn Scipione 				else {
17681bb2e623SJohn Scipione 					Window()->MoveBy(delta, 0);
17691bb2e623SJohn Scipione 					Window()->ResizeBy(delta, 0);
17701bb2e623SJohn Scipione 				}
17711bb2e623SJohn Scipione 			}
17721bb2e623SJohn Scipione 
17731bb2e623SJohn Scipione 			windowWidth = Window()->Frame().Width();
17741bb2e623SJohn Scipione 		}
17751bb2e623SJohn Scipione 	}
17761bb2e623SJohn Scipione 
17771bb2e623SJohn Scipione 	BControl::MouseMoved(where, code, dragMessage);
17781bb2e623SJohn Scipione }
1779