xref: /haiku/src/apps/deskbar/StatusView.cpp (revision 3641948f78e312bc04986d74659cf80fe255f147)
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 
9341281cf3SAxel Dörfler static void
9441281cf3SAxel Dörfler DumpItem(DeskbarItemInfo* item)
9541281cf3SAxel Dörfler {
96e9632898SAlex Smith 	printf("is addon: %i, id: %" B_PRId32 "\n", item->isAddOn, item->id);
97e9632898SAlex Smith 	printf("entry_ref:  %" B_PRIdDEV ", %" B_PRIdINO ", %s\n",
98e9632898SAlex Smith 		item->entryRef.device, item->entryRef.directory, item->entryRef.name);
99e9632898SAlex Smith 	printf("node_ref:  %" B_PRIdDEV ", %" B_PRIdINO "\n", item->nodeRef.device,
100e9632898SAlex Smith 		item->nodeRef.node);
10141281cf3SAxel Dörfler }
10241281cf3SAxel Dörfler 
10341281cf3SAxel Dörfler 
10441281cf3SAxel Dörfler static void
10541281cf3SAxel Dörfler DumpList(BList* itemlist)
10641281cf3SAxel Dörfler {
10741281cf3SAxel Dörfler 	int32 count = itemlist->CountItems() - 1;
10841281cf3SAxel Dörfler 	if (count < 0) {
10941281cf3SAxel Dörfler 		printf("no items in list\n");
11041281cf3SAxel Dörfler 		return;
11141281cf3SAxel Dörfler 	}
11241281cf3SAxel Dörfler 	for (int32 i = count; i >= 0; i--) {
11341281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)itemlist->ItemAt(i);
11441281cf3SAxel Dörfler 		if (!item)
11541281cf3SAxel Dörfler 			continue;
11641281cf3SAxel Dörfler 
11741281cf3SAxel Dörfler 		DumpItem(item);
11841281cf3SAxel Dörfler 	}
11941281cf3SAxel Dörfler }
12041281cf3SAxel Dörfler #endif	/* DB_ADDONS */
12141281cf3SAxel Dörfler 
12241281cf3SAxel Dörfler 
123546208a5SOliver Tappe #undef B_TRANSLATION_CONTEXT
124546208a5SOliver Tappe #define B_TRANSLATION_CONTEXT "Tray"
125ae7a6e3aSAdrien Destugues 
1267da06231SAxel Dörfler // don't change the name of this view to anything other than "Status"!
1277da06231SAxel Dörfler 
12841281cf3SAxel Dörfler TReplicantTray::TReplicantTray(TBarView* parent, bool vertical)
1290d2ac94cSJohn Scipione 	:
1309ac6fee1SJohn Scipione 	BView(BRect(0, 0, 1, 1), "Status", B_FOLLOW_LEFT | B_FOLLOW_TOP,
1310d2ac94cSJohn Scipione 		B_WILL_DRAW | B_FRAME_EVENTS),
132573f748cSJohn Scipione 	fTime(NULL),
13341281cf3SAxel Dörfler 	fBarView(parent),
13441281cf3SAxel Dörfler 	fShelf(new TReplicantShelf(this)),
13541281cf3SAxel Dörfler 	fMultiRowMode(vertical),
13641281cf3SAxel Dörfler 	fAlignmentSupport(false)
13741281cf3SAxel Dörfler {
138fb956f25SAxel Dörfler 	if (vertical)
139fb956f25SAxel Dörfler 		fMinimumTrayWidth = gMinimumWindowWidth - kGutter - kDragRegionWidth;
1401bb2e623SJohn Scipione 	else
1411bb2e623SJohn Scipione 		fMinimumTrayWidth = kMinimumTrayWidth;
142573f748cSJohn Scipione 
143573f748cSJohn Scipione 	// Create the time view
14431c0024dSJohn Scipione 	fTime = new TTimeView(fMinimumTrayWidth, kMaxReplicantHeight - 1.0);
14541281cf3SAxel Dörfler }
14641281cf3SAxel Dörfler 
14741281cf3SAxel Dörfler 
14841281cf3SAxel Dörfler TReplicantTray::~TReplicantTray()
14941281cf3SAxel Dörfler {
15041281cf3SAxel Dörfler 	delete fShelf;
151573f748cSJohn Scipione 	delete fTime;
15241281cf3SAxel Dörfler }
15341281cf3SAxel Dörfler 
15441281cf3SAxel Dörfler 
15541281cf3SAxel Dörfler void
15641281cf3SAxel Dörfler TReplicantTray::AttachedToWindow()
15741281cf3SAxel Dörfler {
15841281cf3SAxel Dörfler 	BView::AttachedToWindow();
15941281cf3SAxel Dörfler 
160cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
161cef0a7b2Slooncraz 		AdoptParentColors();
162cb6afcb1SStephan Aßmus 	} else {
16316c10517Slooncraz 		SetViewUIColor(B_MENU_BACKGROUND_COLOR,	B_DARKEN_1_TINT);
164cb6afcb1SStephan Aßmus 	}
16541281cf3SAxel Dörfler 	SetDrawingMode(B_OP_COPY);
16666eba86fSAxel Dörfler 
16741281cf3SAxel Dörfler 	Window()->SetPulseRate(1000000);
16841281cf3SAxel Dörfler 
1693722e640SJohn Scipione 	clock_settings* clock = ((TBarApp*)be_app)->ClockSettings();
1703722e640SJohn Scipione 	fTime->SetShowSeconds(clock->showSeconds);
1713722e640SJohn Scipione 	fTime->SetShowDayOfWeek(clock->showDayOfWeek);
1723722e640SJohn Scipione 	fTime->SetShowTimeZone(clock->showTimeZone);
17331c0024dSJohn Scipione 
174573f748cSJohn Scipione 	AddChild(fTime);
1751bb2e623SJohn Scipione 
1761bb2e623SJohn Scipione 	float deltaX = Bounds().right - fTime->Bounds().Width() - 1;
1771bb2e623SJohn Scipione 	if (fBarView->Vertical() && !fBarView->Left())
1781bb2e623SJohn Scipione 		deltaX -= kDragWidth;
1791bb2e623SJohn Scipione 
1801bb2e623SJohn Scipione 	fTime->MoveTo(deltaX, 2);
181ae7a6e3aSAdrien Destugues 
1823722e640SJohn Scipione 	if (!((TBarApp*)be_app)->Settings()->showClock)
1833722e640SJohn Scipione 		fTime->Hide();
1843722e640SJohn Scipione 
18541281cf3SAxel Dörfler #ifdef DB_ADDONS
18641281cf3SAxel Dörfler 	// load addons and rehydrate archives
1873a6add49SAxel Dörfler #if !defined(HAIKU_TARGET_PLATFORM_LIBBE_TEST)
18841281cf3SAxel Dörfler 	InitAddOnSupport();
18941281cf3SAxel Dörfler #endif
1909c910f9eSStephan Aßmus #endif
19141281cf3SAxel Dörfler 	ResizeToPreferred();
19241281cf3SAxel Dörfler }
19341281cf3SAxel Dörfler 
19441281cf3SAxel Dörfler 
19541281cf3SAxel Dörfler void
19641281cf3SAxel Dörfler TReplicantTray::DetachedFromWindow()
19741281cf3SAxel Dörfler {
19841281cf3SAxel Dörfler #ifdef DB_ADDONS
19941281cf3SAxel Dörfler 	// clean up add-on support
200465b54cdSAxel Dörfler #if !defined(HAIKU_TARGET_PLATFORM_LIBBE_TEST)
20141281cf3SAxel Dörfler 	DeleteAddOnSupport();
20241281cf3SAxel Dörfler #endif
203465b54cdSAxel Dörfler #endif
20441281cf3SAxel Dörfler 	BView::DetachedFromWindow();
20541281cf3SAxel Dörfler }
20641281cf3SAxel Dörfler 
20741281cf3SAxel Dörfler 
2081ad8c760SFredrik Holmqvist /*! Width is set to a minimum of kMinimumReplicantCount by kMaxReplicantWidth
20966eba86fSAxel Dörfler 	if not in multirowmode and greater than kMinimumReplicantCount
2102ce9bab8SJohn Scipione 	the width should be calculated based on the actual replicant widths
21141281cf3SAxel Dörfler */
21241281cf3SAxel Dörfler void
21341281cf3SAxel Dörfler TReplicantTray::GetPreferredSize(float* preferredWidth, float* preferredHeight)
21441281cf3SAxel Dörfler {
2151bb2e623SJohn Scipione 	float width = 0;
216e4d65fe7SJohn Scipione 	float height = kMinimumTrayHeight;
21741281cf3SAxel Dörfler 
21841281cf3SAxel Dörfler 	if (fMultiRowMode) {
2191bb2e623SJohn Scipione 		width = static_cast<TBarApp*>(be_app)->Settings()->width
2201bb2e623SJohn Scipione 			- kDragWidth - kGutter;
2217da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0)
2227da06231SAxel Dörfler 			height = fRightBottomReplicant.bottom;
2231ad8c760SFredrik Holmqvist 		// the height will be uniform for the number of rows necessary to show
2241ad8c760SFredrik Holmqvist 		// all the reps + any gutters necessary for spacing
22541281cf3SAxel Dörfler 		int32 rowCount = (int32)(height / kMaxReplicantHeight);
22641281cf3SAxel Dörfler 		height = kGutter + (rowCount * kMaxReplicantHeight)
22741281cf3SAxel Dörfler 			+ ((rowCount - 1) * kIconGap) + kGutter;
2281bb2e623SJohn Scipione 		height = std::max(kMinimumTrayHeight, height);
22941281cf3SAxel Dörfler 	} else {
2301ad8c760SFredrik Holmqvist 		// if last replicant overruns clock then resize to accomodate
2317da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0) {
232e4d65fe7SJohn Scipione 			if (!fTime->IsHidden()
2331bb2e623SJohn Scipione 				&& fTime->Frame().left < fRightBottomReplicant.right + 12) {
2341bb2e623SJohn Scipione 				width = fRightBottomReplicant.right + 12
235573f748cSJohn Scipione 					+ fTime->Frame().Width();
23641281cf3SAxel Dörfler 			} else
2377da06231SAxel Dörfler 				width = fRightBottomReplicant.right + 3;
23841281cf3SAxel Dörfler 		}
2391ad8c760SFredrik Holmqvist 
24041281cf3SAxel Dörfler 		// this view has a fixed minimum width
2411bb2e623SJohn Scipione 		width = std::max(fMinimumTrayWidth, width);
2422ce9bab8SJohn Scipione 		height = kGutter + static_cast<TBarApp*>(be_app)->IconSize() + kGutter;
24341281cf3SAxel Dörfler 	}
24441281cf3SAxel Dörfler 
24541281cf3SAxel Dörfler 	*preferredWidth = width;
2461ad8c760SFredrik Holmqvist 	// add 1 for the border
24741281cf3SAxel Dörfler 	*preferredHeight = height + 1;
24841281cf3SAxel Dörfler }
24941281cf3SAxel Dörfler 
25041281cf3SAxel Dörfler 
25141281cf3SAxel Dörfler void
25241281cf3SAxel Dörfler TReplicantTray::AdjustPlacement()
25341281cf3SAxel Dörfler {
25441281cf3SAxel Dörfler 	// called when an add-on has been added or removed
25541281cf3SAxel Dörfler 	// need to resize the parent of this accordingly
256a5210ab1SAxel Dörfler 
25741281cf3SAxel Dörfler 	BRect bounds = Bounds();
25841281cf3SAxel Dörfler 	float width, height;
25941281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
26041281cf3SAxel Dörfler 
26141281cf3SAxel Dörfler 	if (width == bounds.Width() && height == bounds.Height()) {
26241281cf3SAxel Dörfler 		// no need to change anything
26341281cf3SAxel Dörfler 		return;
26441281cf3SAxel Dörfler 	}
26541281cf3SAxel Dörfler 
26641281cf3SAxel Dörfler 	Parent()->ResizeToPreferred();
26741281cf3SAxel Dörfler 	fBarView->UpdatePlacement();
26841281cf3SAxel Dörfler 	Parent()->Invalidate();
26941281cf3SAxel Dörfler 	Invalidate();
27041281cf3SAxel Dörfler }
27141281cf3SAxel Dörfler 
27241281cf3SAxel Dörfler 
27341281cf3SAxel Dörfler void
27441281cf3SAxel Dörfler TReplicantTray::MessageReceived(BMessage* message)
27541281cf3SAxel Dörfler {
27641281cf3SAxel Dörfler 	switch (message->what) {
27730a55802SJohn Scipione 		case B_LOCALE_CHANGED:
278573f748cSJohn Scipione 		{
279573f748cSJohn Scipione 			if (fTime == NULL)
280573f748cSJohn Scipione 				return;
281573f748cSJohn Scipione 
28231c0024dSJohn Scipione 			fTime->Update();
28330a55802SJohn Scipione 
28441281cf3SAxel Dörfler 			// time string reformat -> realign
28541281cf3SAxel Dörfler 			RealignReplicants();
28641281cf3SAxel Dörfler 			AdjustPlacement();
28741281cf3SAxel Dörfler 			break;
288573f748cSJohn Scipione 		}
289573f748cSJohn Scipione 
29030a55802SJohn Scipione 		case kShowHideTime:
29130a55802SJohn Scipione 			// from context menu in clock and in this view
29230a55802SJohn Scipione 			ShowHideTime();
29330a55802SJohn Scipione 			break;
29430a55802SJohn Scipione 
2953cf2d117SJohn Scipione 		case kShowSeconds:
296573f748cSJohn Scipione 			if (fTime == NULL)
297573f748cSJohn Scipione 				return;
298573f748cSJohn Scipione 
2993cf2d117SJohn Scipione 			fTime->SetShowSeconds(!fTime->ShowSeconds());
300573f748cSJohn Scipione 
3013cf2d117SJohn Scipione 			// time string reformat -> realign
3023cf2d117SJohn Scipione 			RealignReplicants();
3033cf2d117SJohn Scipione 			AdjustPlacement();
304573f748cSJohn Scipione 			break;
3053cf2d117SJohn Scipione 
3063cf2d117SJohn Scipione 		case kShowDayOfWeek:
3073cf2d117SJohn Scipione 			if (fTime == NULL)
3083cf2d117SJohn Scipione 				return;
3093cf2d117SJohn Scipione 
3103cf2d117SJohn Scipione 			fTime->SetShowDayOfWeek(!fTime->ShowDayOfWeek());
3113cf2d117SJohn Scipione 
3123cf2d117SJohn Scipione 			// time string reformat -> realign
3133cf2d117SJohn Scipione 			RealignReplicants();
3143cf2d117SJohn Scipione 			AdjustPlacement();
3153cf2d117SJohn Scipione 			break;
3163cf2d117SJohn Scipione 
31731c0024dSJohn Scipione 		case kShowTimeZone:
31831c0024dSJohn Scipione 			if (fTime == NULL)
31931c0024dSJohn Scipione 				return;
32031c0024dSJohn Scipione 
32131c0024dSJohn Scipione 			fTime->SetShowTimeZone(!fTime->ShowTimeZone());
32231c0024dSJohn Scipione 
32331c0024dSJohn Scipione 			// time string reformat -> realign
32431c0024dSJohn Scipione 			RealignReplicants();
32531c0024dSJohn Scipione 			AdjustPlacement();
32631c0024dSJohn Scipione 			break;
32731c0024dSJohn Scipione 
32831c0024dSJohn Scipione 		case kGetClockSettings:
32931c0024dSJohn Scipione 		{
33031c0024dSJohn Scipione 			if (fTime == NULL)
33131c0024dSJohn Scipione 				return;
33231c0024dSJohn Scipione 
33331c0024dSJohn Scipione 			bool showClock = !fTime->IsHidden();
33431c0024dSJohn Scipione 			bool showSeconds = fTime->ShowSeconds();
33531c0024dSJohn Scipione 			bool showDayOfWeek = fTime->ShowDayOfWeek();
33631c0024dSJohn Scipione 			bool showTimeZone = fTime->ShowTimeZone();
33731c0024dSJohn Scipione 
3380fc30d89SDario Casalinuovo 			BMessage reply(kGetClockSettings);
3390fc30d89SDario Casalinuovo 			reply.AddBool("showClock", showClock);
3400fc30d89SDario Casalinuovo 			reply.AddBool("showSeconds", showSeconds);
3410fc30d89SDario Casalinuovo 			reply.AddBool("showDayOfWeek", showDayOfWeek);
3420fc30d89SDario Casalinuovo 			reply.AddBool("showTimeZone", showTimeZone);
3430fc30d89SDario Casalinuovo 			message->SendReply(&reply);
34431c0024dSJohn Scipione 			break;
34531c0024dSJohn Scipione 		}
34631c0024dSJohn Scipione 
34741281cf3SAxel Dörfler #ifdef DB_ADDONS
34841281cf3SAxel Dörfler 		case B_NODE_MONITOR:
34941281cf3SAxel Dörfler 			HandleEntryUpdate(message);
35041281cf3SAxel Dörfler 			break;
35141281cf3SAxel Dörfler #endif
35241281cf3SAxel Dörfler 
35341281cf3SAxel Dörfler 		default:
35441281cf3SAxel Dörfler 			BView::MessageReceived(message);
35541281cf3SAxel Dörfler 			break;
35641281cf3SAxel Dörfler 	}
35741281cf3SAxel Dörfler }
35841281cf3SAxel Dörfler 
35941281cf3SAxel Dörfler 
36041281cf3SAxel Dörfler void
36141281cf3SAxel Dörfler TReplicantTray::MouseDown(BPoint where)
36241281cf3SAxel Dörfler {
36341281cf3SAxel Dörfler #ifdef DB_ADDONS
36441281cf3SAxel Dörfler 	if (modifiers() & B_CONTROL_KEY)
36541281cf3SAxel Dörfler 		DumpList(fItemList);
36641281cf3SAxel Dörfler #endif
36741281cf3SAxel Dörfler 
36841281cf3SAxel Dörfler 	uint32 buttons;
36941281cf3SAxel Dörfler 
37041281cf3SAxel Dörfler 	Window()->CurrentMessage()->FindInt32("buttons", (int32*)&buttons);
37141281cf3SAxel Dörfler 	if (buttons == B_SECONDARY_MOUSE_BUTTON) {
37241281cf3SAxel Dörfler 		ShowReplicantMenu(where);
37341281cf3SAxel Dörfler 	} else {
37441281cf3SAxel Dörfler 		BPoint save = where;
37541281cf3SAxel Dörfler 		bigtime_t doubleClickSpeed;
37641281cf3SAxel Dörfler 		bigtime_t start = system_time();
37741281cf3SAxel Dörfler 		uint32 buttons;
37841281cf3SAxel Dörfler 
37941281cf3SAxel Dörfler 		get_click_speed(&doubleClickSpeed);
38041281cf3SAxel Dörfler 
38141281cf3SAxel Dörfler 		do {
38241281cf3SAxel Dörfler 			if (fabs(where.x - save.x) > 4 || fabs(where.y - save.y) > 4)
38341281cf3SAxel Dörfler 				// user moved out of bounds of click area
38441281cf3SAxel Dörfler 				break;
38541281cf3SAxel Dörfler 
38641281cf3SAxel Dörfler 			if ((system_time() - start) > (2 * doubleClickSpeed)) {
38741281cf3SAxel Dörfler 				ShowReplicantMenu(where);
38841281cf3SAxel Dörfler 				break;
38941281cf3SAxel Dörfler 			}
39041281cf3SAxel Dörfler 
39141281cf3SAxel Dörfler 			snooze(50000);
39241281cf3SAxel Dörfler 			GetMouse(&where, &buttons);
39341281cf3SAxel Dörfler 		} while (buttons);
39441281cf3SAxel Dörfler 	}
39541281cf3SAxel Dörfler 	BView::MouseDown(where);
39641281cf3SAxel Dörfler }
39741281cf3SAxel Dörfler 
3983cf2d117SJohn Scipione 
3993cf2d117SJohn Scipione void
4003cf2d117SJohn Scipione TReplicantTray::ShowReplicantMenu(BPoint point)
4013cf2d117SJohn Scipione {
4023cf2d117SJohn Scipione 	BPopUpMenu* menu = new BPopUpMenu("", false, false);
4033cf2d117SJohn Scipione 	menu->SetFont(be_plain_font);
4043cf2d117SJohn Scipione 
40531c0024dSJohn Scipione 	// If clock is visible show the extended menu, otherwise show "Show clock"
4063cf2d117SJohn Scipione 
4073cf2d117SJohn Scipione 	if (!fTime->IsHidden())
4083cf2d117SJohn Scipione 		fTime->ShowTimeOptions(ConvertToScreen(point));
4093cf2d117SJohn Scipione 	else {
41031c0024dSJohn Scipione 		BMenuItem* item = new BMenuItem(B_TRANSLATE("Show clock"),
4113cf2d117SJohn Scipione 			new BMessage(kShowHideTime));
4123cf2d117SJohn Scipione 		menu->AddItem(item);
4133cf2d117SJohn Scipione 		menu->SetTargetForItems(this);
4143cf2d117SJohn Scipione 		BPoint where = ConvertToScreen(point);
4153cf2d117SJohn Scipione 		menu->Go(where, true, true, BRect(where - BPoint(4, 4),
4163cf2d117SJohn Scipione 			where + BPoint(4, 4)), true);
4173cf2d117SJohn Scipione 	}
4183cf2d117SJohn Scipione }
4193cf2d117SJohn Scipione 
4203cf2d117SJohn Scipione 
4213cf2d117SJohn Scipione void
4223cf2d117SJohn Scipione TReplicantTray::SetMultiRow(bool state)
4233cf2d117SJohn Scipione {
4243cf2d117SJohn Scipione 	fMultiRowMode = state;
4253cf2d117SJohn Scipione }
4263cf2d117SJohn Scipione 
4273cf2d117SJohn Scipione 
4283cf2d117SJohn Scipione void
4293cf2d117SJohn Scipione TReplicantTray::ShowHideTime()
4303cf2d117SJohn Scipione {
4313cf2d117SJohn Scipione 	if (fTime == NULL)
4323cf2d117SJohn Scipione 		return;
4333cf2d117SJohn Scipione 
4342a5c1f12SJohn Scipione 	// Check from the point of view of fTime because we need to ignore
4352a5c1f12SJohn Scipione 	// whether or not the parent window is hidden.
4362a5c1f12SJohn Scipione 	if (fTime->IsHidden(fTime))
4373cf2d117SJohn Scipione 		fTime->Show();
4383cf2d117SJohn Scipione 	else
4393cf2d117SJohn Scipione 		fTime->Hide();
4403cf2d117SJohn Scipione 
4413cf2d117SJohn Scipione 	RealignReplicants();
4423cf2d117SJohn Scipione 	AdjustPlacement();
44331c0024dSJohn Scipione 
4442a5c1f12SJohn Scipione 	// Check from the point of view of fTime ignoring parent's state.
4452a5c1f12SJohn Scipione 	bool showClock = !fTime->IsHidden(fTime);
4463722e640SJohn Scipione 
4473722e640SJohn Scipione 	// Update showClock setting that gets saved to disk on quit
4481bb2e623SJohn Scipione 	static_cast<TBarApp*>(be_app)->Settings()->showClock = showClock;
4493722e640SJohn Scipione 
4503722e640SJohn Scipione 	// Send a message to Time preferences telling it to update
45131c0024dSJohn Scipione 	BMessenger messenger("application/x-vnd.Haiku-Time");
4520fc30d89SDario Casalinuovo 	BMessage message(kShowHideTime);
4530fc30d89SDario Casalinuovo 	message.AddBool("showClock", showClock);
4540fc30d89SDario Casalinuovo 	messenger.SendMessage(&message);
4553cf2d117SJohn Scipione }
4563cf2d117SJohn Scipione 
4573cf2d117SJohn Scipione 
45841281cf3SAxel Dörfler #ifdef DB_ADDONS
45941281cf3SAxel Dörfler 
4603cf2d117SJohn Scipione 
46141281cf3SAxel Dörfler void
46241281cf3SAxel Dörfler TReplicantTray::InitAddOnSupport()
46341281cf3SAxel Dörfler {
46441281cf3SAxel Dörfler 	// list to maintain refs to each rep added/deleted
46541281cf3SAxel Dörfler 	fItemList = new BList();
46641281cf3SAxel Dörfler 	BPath path;
4671ad8c760SFredrik Holmqvist 
46826ea7066SIngo Weinhold 	if (GetDeskbarSettingsDirectory(path, true) == B_OK) {
4693e6ff860SRene Gollent 		path.Append(kReplicantSettingsFile);
47041281cf3SAxel Dörfler 
47141281cf3SAxel Dörfler 		BFile file(path.Path(), B_READ_ONLY);
4723e6ff860SRene Gollent 		if (file.InitCheck() == B_OK) {
4733e6ff860SRene Gollent 			status_t result;
4743e6ff860SRene Gollent 			BEntry entry;
4753e6ff860SRene Gollent 			int32 id;
476c0ad1c91SRene Gollent 			BString path;
4773e6ff860SRene Gollent 			if (fAddOnSettings.Unflatten(&file) == B_OK) {
478c0ad1c91SRene Gollent 				for (int32 i = 0; fAddOnSettings.FindString(kReplicantPathField,
479c0ad1c91SRene Gollent 					i, &path) == B_OK; i++) {
480c0ad1c91SRene Gollent 					if (entry.SetTo(path.String()) == B_OK && entry.Exists()) {
4813e6ff860SRene Gollent 						result = LoadAddOn(&entry, &id, false);
4823e6ff860SRene Gollent 					} else
4833e6ff860SRene Gollent 						result = B_ENTRY_NOT_FOUND;
48441281cf3SAxel Dörfler 
4853e6ff860SRene Gollent 					if (result != B_OK) {
486c0ad1c91SRene Gollent 						fAddOnSettings.RemoveData(kReplicantPathField, i);
4873e6ff860SRene Gollent 						--i;
48841281cf3SAxel Dörfler 					}
48941281cf3SAxel Dörfler 				}
49041281cf3SAxel Dörfler 			}
4913e6ff860SRene Gollent 		}
4923e6ff860SRene Gollent 	}
49341281cf3SAxel Dörfler }
49441281cf3SAxel Dörfler 
49541281cf3SAxel Dörfler 
49641281cf3SAxel Dörfler void
49741281cf3SAxel Dörfler TReplicantTray::DeleteAddOnSupport()
49841281cf3SAxel Dörfler {
4997625ce51SRene Gollent 	_SaveSettings();
5003e6ff860SRene Gollent 
5019f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
50241281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->RemoveItem(i);
50341281cf3SAxel Dörfler 		if (item) {
50441281cf3SAxel Dörfler 			if (item->isAddOn)
50541281cf3SAxel Dörfler 				watch_node(&(item->nodeRef), B_STOP_WATCHING, this, Window());
50641281cf3SAxel Dörfler 
50741281cf3SAxel Dörfler 			delete item;
50841281cf3SAxel Dörfler 		}
50941281cf3SAxel Dörfler 	}
51041281cf3SAxel Dörfler 	delete fItemList;
51141281cf3SAxel Dörfler 
51241281cf3SAxel Dörfler 	// stop the volume mount/unmount watch
51341281cf3SAxel Dörfler 	stop_watching(this, Window());
51441281cf3SAxel Dörfler }
51541281cf3SAxel Dörfler 
51641281cf3SAxel Dörfler 
51741281cf3SAxel Dörfler DeskbarItemInfo*
51841281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(node_ref& nodeRef)
51941281cf3SAxel Dörfler {
5209f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
52166eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
52241281cf3SAxel Dörfler 		if (item == NULL)
52341281cf3SAxel Dörfler 			continue;
52441281cf3SAxel Dörfler 
52541281cf3SAxel Dörfler 		if (item->nodeRef == nodeRef)
52641281cf3SAxel Dörfler 			return item;
52741281cf3SAxel Dörfler 	}
52841281cf3SAxel Dörfler 
52941281cf3SAxel Dörfler 	return NULL;
53041281cf3SAxel Dörfler }
53141281cf3SAxel Dörfler 
53241281cf3SAxel Dörfler 
53341281cf3SAxel Dörfler DeskbarItemInfo*
53441281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(int32 id)
53541281cf3SAxel Dörfler {
5369f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
53766eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
53841281cf3SAxel Dörfler 		if (item == NULL)
53941281cf3SAxel Dörfler 			continue;
54041281cf3SAxel Dörfler 
54141281cf3SAxel Dörfler 		if (item->id == id)
54241281cf3SAxel Dörfler 			return item;
54341281cf3SAxel Dörfler 	}
54441281cf3SAxel Dörfler 
54541281cf3SAxel Dörfler 	return NULL;
54641281cf3SAxel Dörfler }
54741281cf3SAxel Dörfler 
54841281cf3SAxel Dörfler 
54941281cf3SAxel Dörfler bool
55041281cf3SAxel Dörfler TReplicantTray::NodeExists(node_ref& nodeRef)
55141281cf3SAxel Dörfler {
55241281cf3SAxel Dörfler 	return DeskbarItemFor(nodeRef) != NULL;
55341281cf3SAxel Dörfler }
55441281cf3SAxel Dörfler 
55541281cf3SAxel Dörfler 
55666eba86fSAxel Dörfler /*! This handles B_NODE_MONITOR & B_QUERY_UPDATE messages received
55766eba86fSAxel Dörfler 	for the registered add-ons.
55841281cf3SAxel Dörfler */
55941281cf3SAxel Dörfler void
56041281cf3SAxel Dörfler TReplicantTray::HandleEntryUpdate(BMessage* message)
56141281cf3SAxel Dörfler {
56241281cf3SAxel Dörfler 	int32 opcode;
56341281cf3SAxel Dörfler 	if (message->FindInt32("opcode", &opcode) != B_OK)
56441281cf3SAxel Dörfler 		return;
56541281cf3SAxel Dörfler 
56641281cf3SAxel Dörfler 	BPath path;
56741281cf3SAxel Dörfler 	switch (opcode) {
56841281cf3SAxel Dörfler 		case B_ENTRY_MOVED:
56941281cf3SAxel Dörfler 		{
57041281cf3SAxel Dörfler 			entry_ref ref;
57141281cf3SAxel Dörfler 			ino_t todirectory;
57241281cf3SAxel Dörfler 			ino_t node;
57341281cf3SAxel Dörfler 			const char* name;
57441281cf3SAxel Dörfler 			if (message->FindString("name", &name) == B_OK
57571bd3ba5SJonas Sundström 				&& message->FindInt64("from directory", &(ref.directory))
57671bd3ba5SJonas Sundström 					== B_OK
57741281cf3SAxel Dörfler 				&& message->FindInt64("to directory", &todirectory) == B_OK
57841281cf3SAxel Dörfler 				&& message->FindInt32("device", &(ref.device)) == B_OK
57941281cf3SAxel Dörfler 				&& message->FindInt64("node", &node) == B_OK ) {
58041281cf3SAxel Dörfler 
581e4d65fe7SJohn Scipione 				if (name == NULL)
58241281cf3SAxel Dörfler 					break;
58341281cf3SAxel Dörfler 
58441281cf3SAxel Dörfler 				ref.set_name(name);
58541281cf3SAxel Dörfler 				// change the directory reference to
58641281cf3SAxel Dörfler 				// the new directory
58741281cf3SAxel Dörfler 				MoveItem(&ref, todirectory);
58841281cf3SAxel Dörfler 			}
58941281cf3SAxel Dörfler 			break;
59066eba86fSAxel Dörfler 		}
59141281cf3SAxel Dörfler 
59241281cf3SAxel Dörfler 		case B_ENTRY_REMOVED:
59341281cf3SAxel Dörfler 		{
59441281cf3SAxel Dörfler 			// entry was rm'd from the device
59541281cf3SAxel Dörfler 			node_ref nodeRef;
59641281cf3SAxel Dörfler 			if (message->FindInt32("device", &(nodeRef.device)) == B_OK
59741281cf3SAxel Dörfler 				&& message->FindInt64("node", &(nodeRef.node)) == B_OK) {
59841281cf3SAxel Dörfler 				DeskbarItemInfo* item = DeskbarItemFor(nodeRef);
59941281cf3SAxel Dörfler 				if (item == NULL)
60041281cf3SAxel Dörfler 					break;
60141281cf3SAxel Dörfler 
60241281cf3SAxel Dörfler 				// If there is a team running where the add-on comes from,
60341281cf3SAxel Dörfler 				// we don't want to remove the icon yet.
60441281cf3SAxel Dörfler 				if (be_roster->IsRunning(&item->entryRef))
60541281cf3SAxel Dörfler 					break;
60641281cf3SAxel Dörfler 
60741281cf3SAxel Dörfler 				UnloadAddOn(&nodeRef, NULL, true, false);
60841281cf3SAxel Dörfler 			}
60941281cf3SAxel Dörfler 			break;
61066eba86fSAxel Dörfler 		}
61141281cf3SAxel Dörfler 	}
61266eba86fSAxel Dörfler }
61341281cf3SAxel Dörfler 
61441281cf3SAxel Dörfler 
6151ad8c760SFredrik Holmqvist /*! The add-ons must support the exported C function API
61666eba86fSAxel Dörfler 	if they do, they will be loaded and added to deskbar
61766eba86fSAxel Dörfler 	primary function is the Instantiate function
6187da06231SAxel Dörfler */
61941281cf3SAxel Dörfler status_t
6203e6ff860SRene Gollent TReplicantTray::LoadAddOn(BEntry* entry, int32* id, bool addToSettings)
62141281cf3SAxel Dörfler {
622e4d65fe7SJohn Scipione 	if (entry == NULL)
623a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
62441281cf3SAxel Dörfler 
62541281cf3SAxel Dörfler 	node_ref nodeRef;
62641281cf3SAxel Dörfler 	entry->GetNodeRef(&nodeRef);
62741281cf3SAxel Dörfler 	// no duplicates
62841281cf3SAxel Dörfler 	if (NodeExists(nodeRef))
62941281cf3SAxel Dörfler 		return B_ERROR;
63041281cf3SAxel Dörfler 
63141281cf3SAxel Dörfler 	BNode node(entry);
63241281cf3SAxel Dörfler 	BPath path;
63366eba86fSAxel Dörfler 	status_t status = entry->GetPath(&path);
634e4d65fe7SJohn Scipione 	if (status != B_OK)
63566eba86fSAxel Dörfler 		return status;
63641281cf3SAxel Dörfler 
63741281cf3SAxel Dörfler 	// load the add-on
63841281cf3SAxel Dörfler 	image_id image = load_add_on(path.Path());
63966eba86fSAxel Dörfler 	if (image < B_OK)
64066eba86fSAxel Dörfler 		return image;
64141281cf3SAxel Dörfler 
64241281cf3SAxel Dörfler 	// get the view loading function symbol
64341281cf3SAxel Dörfler 	//    we first look for a symbol that takes an image_id
64441281cf3SAxel Dörfler 	//    and entry_ref pointer, if not found, go with normal
64541281cf3SAxel Dörfler 	//    instantiate function
64641281cf3SAxel Dörfler 	BView* (*entryFunction)(image_id, const entry_ref*);
64741281cf3SAxel Dörfler 	BView* (*itemFunction)(void);
64841281cf3SAxel Dörfler 	BView* view = NULL;
64941281cf3SAxel Dörfler 
65041281cf3SAxel Dörfler 	entry_ref ref;
65141281cf3SAxel Dörfler 	entry->GetRef(&ref);
65241281cf3SAxel Dörfler 
65341281cf3SAxel Dörfler 	if (get_image_symbol(image, kInstantiateEntryCFunctionName,
65466eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&entryFunction) >= B_OK) {
65541281cf3SAxel Dörfler 		view = (*entryFunction)(image, &ref);
65641281cf3SAxel Dörfler 	} else if (get_image_symbol(image, kInstantiateItemCFunctionName,
65766eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&itemFunction) >= B_OK) {
65841281cf3SAxel Dörfler 		view = (*itemFunction)();
65941281cf3SAxel Dörfler 	} else {
66041281cf3SAxel Dörfler 		unload_add_on(image);
66141281cf3SAxel Dörfler 		return B_ERROR;
66241281cf3SAxel Dörfler 	}
66341281cf3SAxel Dörfler 
66466eba86fSAxel Dörfler 	if (view == NULL || IconExists(view->Name())) {
66541281cf3SAxel Dörfler 		delete view;
66641281cf3SAxel Dörfler 		unload_add_on(image);
66741281cf3SAxel Dörfler 		return B_ERROR;
66841281cf3SAxel Dörfler 	}
66941281cf3SAxel Dörfler 
67041281cf3SAxel Dörfler 	BMessage* data = new BMessage;
67141281cf3SAxel Dörfler 	view->Archive(data);
67241281cf3SAxel Dörfler 	delete view;
67341281cf3SAxel Dörfler 
67441281cf3SAxel Dörfler 	// add the rep; adds info to list
6750fc30d89SDario Casalinuovo 	if (AddIcon(data, id, &ref) != B_OK)
6760fc30d89SDario Casalinuovo 		delete data;
67741281cf3SAxel Dörfler 
6783e6ff860SRene Gollent 	if (addToSettings) {
679c0ad1c91SRene Gollent 		fAddOnSettings.AddString(kReplicantPathField, path.Path());
6807625ce51SRene Gollent 		_SaveSettings();
6813e6ff860SRene Gollent 	}
68241281cf3SAxel Dörfler 
68341281cf3SAxel Dörfler 	return B_OK;
68441281cf3SAxel Dörfler }
68541281cf3SAxel Dörfler 
68641281cf3SAxel Dörfler 
68741281cf3SAxel Dörfler status_t
68841281cf3SAxel Dörfler TReplicantTray::AddItem(int32 id, node_ref nodeRef, BEntry& entry, bool isAddOn)
68941281cf3SAxel Dörfler {
69041281cf3SAxel Dörfler 	DeskbarItemInfo* item = new DeskbarItemInfo;
69141281cf3SAxel Dörfler 	if (item == NULL)
69241281cf3SAxel Dörfler 		return B_NO_MEMORY;
69341281cf3SAxel Dörfler 
69441281cf3SAxel Dörfler 	item->id = id;
69541281cf3SAxel Dörfler 	item->isAddOn = isAddOn;
69641281cf3SAxel Dörfler 
697e4d65fe7SJohn Scipione 	if (entry.GetRef(&item->entryRef) != B_OK) {
69841281cf3SAxel Dörfler 		item->entryRef.device = -1;
69941281cf3SAxel Dörfler 		item->entryRef.directory = -1;
70041281cf3SAxel Dörfler 		item->entryRef.name = NULL;
70141281cf3SAxel Dörfler 	}
70241281cf3SAxel Dörfler 	item->nodeRef = nodeRef;
70341281cf3SAxel Dörfler 
70441281cf3SAxel Dörfler 	fItemList->AddItem(item);
70541281cf3SAxel Dörfler 
70641281cf3SAxel Dörfler 	if (isAddOn)
70741281cf3SAxel Dörfler 		watch_node(&nodeRef, B_WATCH_NAME | B_WATCH_ATTR, this, Window());
70841281cf3SAxel Dörfler 
70941281cf3SAxel Dörfler 	return B_OK;
71041281cf3SAxel Dörfler }
71141281cf3SAxel Dörfler 
71241281cf3SAxel Dörfler 
71341281cf3SAxel Dörfler /**	from entry_removed message, when attribute removed
71441281cf3SAxel Dörfler  *	or when a device is unmounted (use removeall, by device)
71541281cf3SAxel Dörfler  */
71641281cf3SAxel Dörfler 
71741281cf3SAxel Dörfler void
718e4d65fe7SJohn Scipione TReplicantTray::UnloadAddOn(node_ref* nodeRef, dev_t* device, bool which,
719e4d65fe7SJohn Scipione 	bool removeAll)
72041281cf3SAxel Dörfler {
7219f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
72241281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
723e4d65fe7SJohn Scipione 		if (item == NULL)
72441281cf3SAxel Dörfler 			continue;
72541281cf3SAxel Dörfler 
726e4d65fe7SJohn Scipione 		if ((which && nodeRef != NULL && item->nodeRef == *nodeRef)
727e4d65fe7SJohn Scipione 			|| (device != NULL && item->nodeRef.device == *device)) {
72841281cf3SAxel Dörfler 
729e4d65fe7SJohn Scipione 			if (device != NULL && be_roster->IsRunning(&item->entryRef))
73041281cf3SAxel Dörfler 				continue;
73141281cf3SAxel Dörfler 
73241281cf3SAxel Dörfler 			RemoveIcon(item->id);
73341281cf3SAxel Dörfler 
73441281cf3SAxel Dörfler 			if (!removeAll)
73541281cf3SAxel Dörfler 				break;
73641281cf3SAxel Dörfler 		}
73741281cf3SAxel Dörfler 	}
73841281cf3SAxel Dörfler }
73941281cf3SAxel Dörfler 
74041281cf3SAxel Dörfler 
74141281cf3SAxel Dörfler void
74241281cf3SAxel Dörfler TReplicantTray::RemoveItem(int32 id)
74341281cf3SAxel Dörfler {
74441281cf3SAxel Dörfler 	DeskbarItemInfo* item = DeskbarItemFor(id);
74541281cf3SAxel Dörfler 	if (item == NULL)
74641281cf3SAxel Dörfler 		return;
74741281cf3SAxel Dörfler 
74841281cf3SAxel Dörfler 	// attribute was added via Deskbar API (AddItem(entry_ref*, int32*)
74941281cf3SAxel Dörfler 	if (item->isAddOn) {
7507625ce51SRene Gollent 		BPath path(&item->entryRef);
7517625ce51SRene Gollent 		BString storedPath;
7527625ce51SRene Gollent 		for (int32 i = 0;
7538aab28f1SRene Gollent 			fAddOnSettings.FindString(kReplicantPathField, i, &storedPath)
7547625ce51SRene Gollent 				== B_OK; i++) {
7557625ce51SRene Gollent 			if (storedPath == path.Path()) {
7568aab28f1SRene Gollent 				fAddOnSettings.RemoveData(kReplicantPathField, i);
7577625ce51SRene Gollent 				break;
7587625ce51SRene Gollent 			}
7597625ce51SRene Gollent 		}
7607625ce51SRene Gollent 		_SaveSettings();
7617625ce51SRene Gollent 
76241281cf3SAxel Dörfler 		BNode node(&item->entryRef);
76341281cf3SAxel Dörfler 		watch_node(&item->nodeRef, B_STOP_WATCHING, this, Window());
76441281cf3SAxel Dörfler 	}
76541281cf3SAxel Dörfler 
76641281cf3SAxel Dörfler 	fItemList->RemoveItem(item);
76741281cf3SAxel Dörfler 	delete item;
76841281cf3SAxel Dörfler }
76941281cf3SAxel Dörfler 
77041281cf3SAxel Dörfler 
77141281cf3SAxel Dörfler /**	ENTRY_MOVED message, moving only occurs on a device
77241281cf3SAxel Dörfler  *	copying will occur (ENTRY_CREATED) between devices
77341281cf3SAxel Dörfler  */
77441281cf3SAxel Dörfler 
77541281cf3SAxel Dörfler void
77641281cf3SAxel Dörfler TReplicantTray::MoveItem(entry_ref* ref, ino_t toDirectory)
77741281cf3SAxel Dörfler {
778e4d65fe7SJohn Scipione 	if (ref == NULL)
77941281cf3SAxel Dörfler 		return;
78041281cf3SAxel Dörfler 
78141281cf3SAxel Dörfler 	// scan for a matching entry_ref and update it
78241281cf3SAxel Dörfler 	//
78341281cf3SAxel Dörfler 	// don't need to change node info as it does not change
78441281cf3SAxel Dörfler 
7859f2cce2fSJohn Scipione 	for (int32 i = fItemList->CountItems() - 1; i >= 0; i--) {
78641281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
787e4d65fe7SJohn Scipione 		if (item == NULL)
78841281cf3SAxel Dörfler 			continue;
78941281cf3SAxel Dörfler 
790e4d65fe7SJohn Scipione 		if (strcmp(item->entryRef.name, ref->name) == 0
79141281cf3SAxel Dörfler 			&& item->entryRef.device == ref->device
79241281cf3SAxel Dörfler 			&& item->entryRef.directory == ref->directory) {
79341281cf3SAxel Dörfler 			item->entryRef.directory = toDirectory;
79441281cf3SAxel Dörfler 			break;
79541281cf3SAxel Dörfler 		}
79641281cf3SAxel Dörfler 	}
79741281cf3SAxel Dörfler }
79841281cf3SAxel Dörfler 
79941281cf3SAxel Dörfler #endif // add-on support
80041281cf3SAxel Dörfler 
80141281cf3SAxel Dörfler //	external add-on API routines
80241281cf3SAxel Dörfler //	called using the new BDeskbar class
80341281cf3SAxel Dörfler 
80441281cf3SAxel Dörfler //	existence of icon/replicant by name or ID
80541281cf3SAxel Dörfler //	returns opposite
80641281cf3SAxel Dörfler //	note: name and id are semi-private limiting
80741281cf3SAxel Dörfler //		the ability of non-host apps to remove
80841281cf3SAxel Dörfler //		icons without a little bit of work
80941281cf3SAxel Dörfler 
81041281cf3SAxel Dörfler /**	for a specific id
81141281cf3SAxel Dörfler  *	return the name of the replicant (name of view)
81241281cf3SAxel Dörfler  */
81341281cf3SAxel Dörfler 
81441281cf3SAxel Dörfler status_t
81541281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 id, const char** name)
81641281cf3SAxel Dörfler {
81741281cf3SAxel Dörfler 	if (id < 0)
818a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
81941281cf3SAxel Dörfler 
820e4d65fe7SJohn Scipione 	int32 index;
821e4d65fe7SJohn Scipione 	int32 temp;
82241281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &temp, id, false);
823e4d65fe7SJohn Scipione 	if (view != NULL) {
82441281cf3SAxel Dörfler 		*name = view->Name();
82541281cf3SAxel Dörfler 		return B_OK;
82641281cf3SAxel Dörfler 	}
82741281cf3SAxel Dörfler 
82841281cf3SAxel Dörfler 	return B_ERROR;
82941281cf3SAxel Dörfler }
83041281cf3SAxel Dörfler 
83141281cf3SAxel Dörfler 
8327da06231SAxel Dörfler /**	for a specific name
8337da06231SAxel Dörfler  *	return the id (internal to Deskbar)
8347da06231SAxel Dörfler  */
8357da06231SAxel Dörfler 
83641281cf3SAxel Dörfler status_t
83741281cf3SAxel Dörfler TReplicantTray::ItemInfo(const char* name, int32* id)
83841281cf3SAxel Dörfler {
839e4d65fe7SJohn Scipione 	if (name == NULL || *name == '\0')
840a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
84141281cf3SAxel Dörfler 
84241281cf3SAxel Dörfler 	int32 index;
84341281cf3SAxel Dörfler 	BView* view = ViewAt(&index, id, name);
844e4d65fe7SJohn Scipione 	if (view != NULL)
84541281cf3SAxel Dörfler 		return B_OK;
84641281cf3SAxel Dörfler 
84741281cf3SAxel Dörfler 	return B_ERROR;
84841281cf3SAxel Dörfler }
84941281cf3SAxel Dörfler 
85041281cf3SAxel Dörfler 
8517da06231SAxel Dörfler /**	at a specific index
8527da06231SAxel Dörfler  *	return both the name and the id of the replicant
8537da06231SAxel Dörfler  */
8547da06231SAxel Dörfler 
85541281cf3SAxel Dörfler status_t
85641281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 index, const char** name, int32* id)
85741281cf3SAxel Dörfler {
85841281cf3SAxel Dörfler 	if (index < 0)
859a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
86041281cf3SAxel Dörfler 
86141281cf3SAxel Dörfler 	BView* view;
86241281cf3SAxel Dörfler 	fShelf->ReplicantAt(index, &view, (uint32*)id, NULL);
863e4d65fe7SJohn Scipione 	if (view != NULL) {
86441281cf3SAxel Dörfler 		*name = view->Name();
86541281cf3SAxel Dörfler 		return B_OK;
86641281cf3SAxel Dörfler 	}
86741281cf3SAxel Dörfler 
86841281cf3SAxel Dörfler 	return B_ERROR;
86941281cf3SAxel Dörfler }
87041281cf3SAxel Dörfler 
87141281cf3SAxel Dörfler 
8727da06231SAxel Dörfler /**	replicant exists, by id/index */
8737da06231SAxel Dörfler 
87441281cf3SAxel Dörfler bool
87541281cf3SAxel Dörfler TReplicantTray::IconExists(int32 target, bool byIndex)
87641281cf3SAxel Dörfler {
877e4d65fe7SJohn Scipione 	int32 index;
878e4d65fe7SJohn Scipione 	int32 id;
87941281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
88041281cf3SAxel Dörfler 
88141281cf3SAxel Dörfler 	return view && index >= 0;
88241281cf3SAxel Dörfler }
88341281cf3SAxel Dörfler 
88441281cf3SAxel Dörfler 
8857da06231SAxel Dörfler /**	replicant exists, by name */
8867da06231SAxel Dörfler 
88741281cf3SAxel Dörfler bool
88841281cf3SAxel Dörfler TReplicantTray::IconExists(const char* name)
88941281cf3SAxel Dörfler {
890e4d65fe7SJohn Scipione 	if (name == NULL || *name == '\0')
89141281cf3SAxel Dörfler 		return false;
89241281cf3SAxel Dörfler 
893e4d65fe7SJohn Scipione 	int32 index;
894e4d65fe7SJohn Scipione 	int32 id;
89541281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
89641281cf3SAxel Dörfler 
897e4d65fe7SJohn Scipione 	return view != NULL && index >= 0;
89841281cf3SAxel Dörfler }
89941281cf3SAxel Dörfler 
90041281cf3SAxel Dörfler 
90141281cf3SAxel Dörfler int32
90241281cf3SAxel Dörfler TReplicantTray::IconCount() const
90341281cf3SAxel Dörfler {
90441281cf3SAxel Dörfler 	return fShelf->CountReplicants();
90541281cf3SAxel Dörfler }
90641281cf3SAxel Dörfler 
90741281cf3SAxel Dörfler 
908ced3755cSAxel Dörfler /*! Message must contain an archivable view for later rehydration.
909ced3755cSAxel Dörfler 	This function takes over ownership of the provided message on success
910ced3755cSAxel Dörfler 	only.
911ced3755cSAxel Dörfler 	Returns the current replicant ID.
9127da06231SAxel Dörfler */
91341281cf3SAxel Dörfler status_t
914ced3755cSAxel Dörfler TReplicantTray::AddIcon(BMessage* archive, int32* id, const entry_ref* addOn)
91541281cf3SAxel Dörfler {
916ced3755cSAxel Dörfler 	if (archive == NULL || id == NULL)
917a6be6c4dSJohn Scipione 		return B_BAD_VALUE;
91841281cf3SAxel Dörfler 
9199c50f36eSAxel Dörfler 	// find entry_ref
9209c50f36eSAxel Dörfler 
9219c50f36eSAxel Dörfler 	entry_ref ref;
922e4d65fe7SJohn Scipione 	if (addOn != NULL) {
9239c50f36eSAxel Dörfler 		// Use it if we got it
9249c50f36eSAxel Dörfler 		ref = *addOn;
9259c50f36eSAxel Dörfler 	} else {
9269c50f36eSAxel Dörfler 		const char* signature;
9271ad8c760SFredrik Holmqvist 
928ced3755cSAxel Dörfler 		status_t status = archive->FindString("add_on", &signature);
9299c50f36eSAxel Dörfler 		if (status == B_OK) {
9309c50f36eSAxel Dörfler 			BRoster roster;
9319c50f36eSAxel Dörfler 			status = roster.FindApp(signature, &ref);
9329c50f36eSAxel Dörfler 		}
933e4d65fe7SJohn Scipione 		if (status != B_OK)
9349c50f36eSAxel Dörfler 			return status;
9359c50f36eSAxel Dörfler 	}
9369c50f36eSAxel Dörfler 
9379c50f36eSAxel Dörfler 	BFile file;
9389c50f36eSAxel Dörfler 	status_t status = file.SetTo(&ref, B_READ_ONLY);
939e4d65fe7SJohn Scipione 	if (status != B_OK)
9409c50f36eSAxel Dörfler 		return status;
9419c50f36eSAxel Dörfler 
9429c50f36eSAxel Dörfler 	node_ref nodeRef;
9439c50f36eSAxel Dörfler 	status = file.GetNodeRef(&nodeRef);
944e4d65fe7SJohn Scipione 	if (status != B_OK)
9459c50f36eSAxel Dörfler 		return status;
9469c50f36eSAxel Dörfler 
9479c50f36eSAxel Dörfler 	BEntry entry(&ref, true);
9481ad8c760SFredrik Holmqvist 		// TODO: this resolves an eventual link for the item being added - this
9491ad8c760SFredrik Holmqvist 		// is okay for now, but in multi-user environments, one might want to
9501ad8c760SFredrik Holmqvist 		// have links that carry the be:deskbar_item_status attribute
9519c50f36eSAxel Dörfler 	status = entry.InitCheck();
9529c50f36eSAxel Dörfler 	if (status != B_OK)
9539c50f36eSAxel Dörfler 		return status;
9549c50f36eSAxel Dörfler 
95541281cf3SAxel Dörfler 	*id = 999;
956ced3755cSAxel Dörfler 	if (archive->what == B_ARCHIVED_OBJECT)
957ced3755cSAxel Dörfler 		archive->what = 0;
95841281cf3SAxel Dörfler 
959ced3755cSAxel Dörfler 	BRect originalBounds = archive->FindRect("_frame");
9601ad8c760SFredrik Holmqvist 		// this is a work-around for buggy replicants that change their size in
9611ad8c760SFredrik Holmqvist 		// AttachedToWindow() (such as "SVM")
962a5210ab1SAxel Dörfler 
9631ad8c760SFredrik Holmqvist 	// TODO: check for name collisions?
964ced3755cSAxel Dörfler 	status = fShelf->AddReplicant(archive, BPoint(1, 1));
9659c50f36eSAxel Dörfler 	if (status != B_OK)
9669c50f36eSAxel Dörfler 		return status;
96741281cf3SAxel Dörfler 
968a5210ab1SAxel Dörfler 	int32 count = fShelf->CountReplicants();
969a5210ab1SAxel Dörfler 	BView* view;
970a5210ab1SAxel Dörfler 	fShelf->ReplicantAt(count - 1, &view, (uint32*)id, NULL);
971a5210ab1SAxel Dörfler 
972a6be6c4dSJohn Scipione 	if (view != NULL && originalBounds != view->Bounds()) {
973a5210ab1SAxel Dörfler 		// The replicant changed its size when added to the window, so we need
97471bd3ba5SJonas Sundström 		// to recompute all over again (it's already done once via
97571bd3ba5SJonas Sundström 		// BShelf::AddReplicant() and TReplicantShelf::CanAcceptReplicantView())
976a5210ab1SAxel Dörfler 		RealignReplicants();
977a5210ab1SAxel Dörfler 	}
978a5210ab1SAxel Dörfler 
97941281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
98041281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
98141281cf3SAxel Dörfler 	float width, height;
98241281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
98341281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height)
98441281cf3SAxel Dörfler 		AdjustPlacement();
98541281cf3SAxel Dörfler 
98641281cf3SAxel Dörfler 	// add the item to the add-on list
98741281cf3SAxel Dörfler 
98841281cf3SAxel Dörfler 	AddItem(*id, nodeRef, entry, addOn != NULL);
98941281cf3SAxel Dörfler 	return B_OK;
99041281cf3SAxel Dörfler }
99141281cf3SAxel Dörfler 
99241281cf3SAxel Dörfler 
99341281cf3SAxel Dörfler void
99441281cf3SAxel Dörfler TReplicantTray::RemoveIcon(int32 target, bool byIndex)
99541281cf3SAxel Dörfler {
99641281cf3SAxel Dörfler 	if (target < 0)
99741281cf3SAxel Dörfler 		return;
99841281cf3SAxel Dörfler 
999e4d65fe7SJohn Scipione 	int32 index;
1000e4d65fe7SJohn Scipione 	int32 id;
100141281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
1002e4d65fe7SJohn Scipione 	if (view != NULL && index >= 0) {
10037da06231SAxel Dörfler 		// remove the reference from the item list & the shelf
100441281cf3SAxel Dörfler 		RemoveItem(id);
100541281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
10067da06231SAxel Dörfler 
100741281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
100841281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
100941281cf3SAxel Dörfler 	}
101041281cf3SAxel Dörfler }
101141281cf3SAxel Dörfler 
101241281cf3SAxel Dörfler 
101341281cf3SAxel Dörfler void
101441281cf3SAxel Dörfler TReplicantTray::RemoveIcon(const char* name)
101541281cf3SAxel Dörfler {
1016e4d65fe7SJohn Scipione 	if (name == NULL || *name == '\0')
101741281cf3SAxel Dörfler 		return;
101841281cf3SAxel Dörfler 
101941281cf3SAxel Dörfler 	int32 id, index;
102041281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
1021e4d65fe7SJohn Scipione 	if (view != NULL && index >= 0) {
10227da06231SAxel Dörfler 		// remove the reference from the item list & shelf
102341281cf3SAxel Dörfler 		RemoveItem(id);
102441281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
10257da06231SAxel Dörfler 
102641281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
102741281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
102841281cf3SAxel Dörfler 	}
102941281cf3SAxel Dörfler }
103041281cf3SAxel Dörfler 
103141281cf3SAxel Dörfler 
103241281cf3SAxel Dörfler void
10337da06231SAxel Dörfler TReplicantTray::RealReplicantAdjustment(int32 startIndex)
103441281cf3SAxel Dörfler {
10357da06231SAxel Dörfler 	if (startIndex < 0)
103641281cf3SAxel Dörfler 		return;
10377da06231SAxel Dörfler 
10387da06231SAxel Dörfler 	if (startIndex == fLastReplicant)
10397da06231SAxel Dörfler 		startIndex = 0;
10407da06231SAxel Dörfler 
104141281cf3SAxel Dörfler 	// reset the locations of all replicants after the one deleted
10427da06231SAxel Dörfler 	RealignReplicants(startIndex);
104341281cf3SAxel Dörfler 
104441281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
104541281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
104641281cf3SAxel Dörfler 	float width, height;
104741281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
104841281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height) {
10491ad8c760SFredrik Holmqvist 		// resize view to accomodate the replicants, redraw as necessary
105041281cf3SAxel Dörfler 		AdjustPlacement();
105141281cf3SAxel Dörfler 	}
105241281cf3SAxel Dörfler }
105341281cf3SAxel Dörfler 
105441281cf3SAxel Dörfler 
10557da06231SAxel Dörfler /**	looking for a replicant by id/index
10567da06231SAxel Dörfler  *	return the view and index
10577da06231SAxel Dörfler  */
10587da06231SAxel Dörfler 
105941281cf3SAxel Dörfler BView*
106041281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, int32 target, bool byIndex)
106141281cf3SAxel Dörfler {
106241281cf3SAxel Dörfler 	*index = -1;
106341281cf3SAxel Dörfler 
106441281cf3SAxel Dörfler 	BView* view;
106541281cf3SAxel Dörfler 	if (byIndex) {
106641281cf3SAxel Dörfler 		if (fShelf->ReplicantAt(target, &view, (uint32*)id)) {
1067e4d65fe7SJohn Scipione 			if (view != NULL) {
106841281cf3SAxel Dörfler 				*index = target;
1069e4d65fe7SJohn Scipione 
107041281cf3SAxel Dörfler 				return view;
107141281cf3SAxel Dörfler 			}
107241281cf3SAxel Dörfler 		}
107341281cf3SAxel Dörfler 	} else {
107441281cf3SAxel Dörfler 		int32 count = fShelf->CountReplicants() - 1;
107541281cf3SAxel Dörfler 		int32 localid;
107641281cf3SAxel Dörfler 		for (int32 repIndex = count; repIndex >= 0; repIndex--) {
107741281cf3SAxel Dörfler 			fShelf->ReplicantAt(repIndex, &view, (uint32*)&localid);
1078e4d65fe7SJohn Scipione 			if (localid == target && view != NULL) {
107941281cf3SAxel Dörfler 				*index = repIndex;
108041281cf3SAxel Dörfler 				*id = localid;
1081e4d65fe7SJohn Scipione 
108241281cf3SAxel Dörfler 				return view;
108341281cf3SAxel Dörfler 			}
108441281cf3SAxel Dörfler 		}
108541281cf3SAxel Dörfler 	}
1086e4d65fe7SJohn Scipione 
108741281cf3SAxel Dörfler 	return NULL;
108841281cf3SAxel Dörfler }
108941281cf3SAxel Dörfler 
109041281cf3SAxel Dörfler 
10917da06231SAxel Dörfler /**	looking for a replicant with a view by name
10927da06231SAxel Dörfler  *	return the view, index and the id of the replicant
10937da06231SAxel Dörfler  */
10947da06231SAxel Dörfler 
109541281cf3SAxel Dörfler BView*
109641281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, const char* name)
109741281cf3SAxel Dörfler {
109841281cf3SAxel Dörfler 	*index = -1;
109941281cf3SAxel Dörfler 	*id = -1;
110041281cf3SAxel Dörfler 
110141281cf3SAxel Dörfler 	BView* view;
110241281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants() - 1;
110341281cf3SAxel Dörfler 	for (int32 repIndex = count; repIndex >= 0; repIndex--) {
110441281cf3SAxel Dörfler 		fShelf->ReplicantAt(repIndex, &view, (uint32*)id);
1105*3641948fSJohn Scipione 		if (view != NULL && view->Name() != NULL
1106e4d65fe7SJohn Scipione 			&& strcmp(name, view->Name()) == 0) {
110741281cf3SAxel Dörfler 			*index = repIndex;
1108e4d65fe7SJohn Scipione 
110941281cf3SAxel Dörfler 			return view;
111041281cf3SAxel Dörfler 		}
111141281cf3SAxel Dörfler 	}
1112e4d65fe7SJohn Scipione 
111341281cf3SAxel Dörfler 	return NULL;
111441281cf3SAxel Dörfler }
111541281cf3SAxel Dörfler 
111641281cf3SAxel Dörfler 
11177da06231SAxel Dörfler /**	Shelf will call to determine where and if
11187da06231SAxel Dörfler  *	the replicant is to be added
11197da06231SAxel Dörfler  */
11207da06231SAxel Dörfler 
112141281cf3SAxel Dörfler bool
112241281cf3SAxel Dörfler TReplicantTray::AcceptAddon(BRect replicantFrame, BMessage* message)
112341281cf3SAxel Dörfler {
1124e4d65fe7SJohn Scipione 	if (message == NULL)
112541281cf3SAxel Dörfler 		return false;
112641281cf3SAxel Dörfler 
112741281cf3SAxel Dörfler 	if (replicantFrame.Height() > kMaxReplicantHeight)
112841281cf3SAxel Dörfler 		return false;
112941281cf3SAxel Dörfler 
113041281cf3SAxel Dörfler 	alignment align = B_ALIGN_LEFT;
113141281cf3SAxel Dörfler 	if (fAlignmentSupport && message->HasBool("deskbar:dynamic_align")) {
113241281cf3SAxel Dörfler 		if (!fBarView->Vertical())
113341281cf3SAxel Dörfler 			align = B_ALIGN_RIGHT;
113441281cf3SAxel Dörfler 		else
113541281cf3SAxel Dörfler 			align = fBarView->Left() ? B_ALIGN_LEFT : B_ALIGN_RIGHT;
113641281cf3SAxel Dörfler 	} else if (message->HasInt32("deskbar:align"))
113741281cf3SAxel Dörfler 		message->FindInt32("deskbar:align", (int32*)&align);
113841281cf3SAxel Dörfler 
113941281cf3SAxel Dörfler 	if (message->HasInt32("deskbar:private_align"))
114041281cf3SAxel Dörfler 		message->FindInt32("deskbar:private_align", (int32*)&align);
114141281cf3SAxel Dörfler 	else
114241281cf3SAxel Dörfler 		align = B_ALIGN_LEFT;
114341281cf3SAxel Dörfler 
11447da06231SAxel Dörfler 	BPoint loc = LocationForReplicant(fShelf->CountReplicants(),
11457da06231SAxel Dörfler 		replicantFrame.Width());
114641281cf3SAxel Dörfler 	message->AddPoint("_pjp_loc", loc);
1147e4d65fe7SJohn Scipione 
114841281cf3SAxel Dörfler 	return true;
114941281cf3SAxel Dörfler }
115041281cf3SAxel Dörfler 
115141281cf3SAxel Dörfler 
115241281cf3SAxel Dörfler /**	based on the previous (index - 1) replicant in the list
115341281cf3SAxel Dörfler  *	calculate where the left point should be for this
115441281cf3SAxel Dörfler  *	replicant.  replicant will flow to the right on its own
115541281cf3SAxel Dörfler  */
115641281cf3SAxel Dörfler 
115741281cf3SAxel Dörfler BPoint
11587da06231SAxel Dörfler TReplicantTray::LocationForReplicant(int32 index, float width)
115941281cf3SAxel Dörfler {
116041281cf3SAxel Dörfler 	BPoint loc(kIconGap + 1, kGutter + 1);
11611bb2e623SJohn Scipione 	if (fBarView->Vertical() && !fBarView->Left())
11621bb2e623SJohn Scipione 		loc.x += kDragWidth;
11631bb2e623SJohn Scipione 	else if (!fBarView->Vertical())
11641bb2e623SJohn Scipione 		loc.x += 2;
116541281cf3SAxel Dörfler 
11667da06231SAxel Dörfler 	if (fMultiRowMode) {
11677da06231SAxel Dörfler 		// try to find free space in every row
11687da06231SAxel Dörfler 		for (int32 row = 0; ; loc.y += kMaxReplicantHeight + kIconGap, row++) {
11697da06231SAxel Dörfler 			// determine free space in this row
11701bb2e623SJohn Scipione 			BRect rect(loc.x, loc.y,
11711bb2e623SJohn Scipione 				loc.x + static_cast<TBarApp*>(be_app)->Settings()->width
11721bb2e623SJohn Scipione 					- kDragRegionWidth - kGutter,
11731bb2e623SJohn Scipione 				loc.y + kMaxReplicantHeight);
11741bb2e623SJohn Scipione 			if (row == 0 && !fTime->IsHidden()) {
11751bb2e623SJohn Scipione 				rect.right -= fTime->Frame().Width() + kDragRegionWidth
11761bb2e623SJohn Scipione 					+ kIconGap + kDragWidth;
11771bb2e623SJohn Scipione 			}
11787da06231SAxel Dörfler 
11797da06231SAxel Dörfler 			for (int32 i = 0; i < index; i++) {
11807da06231SAxel Dörfler 				BView* view = NULL;
11817da06231SAxel Dörfler 				fShelf->ReplicantAt(i, &view);
11827da06231SAxel Dörfler 				if (view == NULL || view->Frame().top != rect.top)
11837da06231SAxel Dörfler 					continue;
11847da06231SAxel Dörfler 
11857da06231SAxel Dörfler 				rect.left = view->Frame().right + kIconGap + 1;
11867da06231SAxel Dörfler 			}
11877da06231SAxel Dörfler 
11887da06231SAxel Dörfler 			if (rect.Width() >= width) {
11897da06231SAxel Dörfler 				// the icon fits in this row
11907da06231SAxel Dörfler 				loc = rect.LeftTop();
11917da06231SAxel Dörfler 				break;
11927da06231SAxel Dörfler 			}
11937da06231SAxel Dörfler 		}
11947da06231SAxel Dörfler 	} else {
119541281cf3SAxel Dörfler 		if (index > 0) {
119641281cf3SAxel Dörfler 			// get the last replicant added for placement reference
119741281cf3SAxel Dörfler 			BView* view = NULL;
11987da06231SAxel Dörfler 			fShelf->ReplicantAt(index - 1, &view);
1199e4d65fe7SJohn Scipione 			if (view != NULL) {
120041281cf3SAxel Dörfler 				// push this rep placement past the last one
120141281cf3SAxel Dörfler 				loc.x = view->Frame().right + kIconGap + 1;
120241281cf3SAxel Dörfler 				loc.y = view->Frame().top;
120341281cf3SAxel Dörfler 			}
120441281cf3SAxel Dörfler 		}
12057da06231SAxel Dörfler 	}
120641281cf3SAxel Dörfler 
1207e4d65fe7SJohn Scipione 	if (loc.y > fRightBottomReplicant.top
1208e4d65fe7SJohn Scipione 		|| (loc.y == fRightBottomReplicant.top
1209e4d65fe7SJohn Scipione 			&& loc.x > fRightBottomReplicant.left)) {
1210e4d65fe7SJohn Scipione 		fRightBottomReplicant.Set(loc.x, loc.y, loc.x + width,
1211e4d65fe7SJohn Scipione 			loc.y + kMaxReplicantHeight);
12127da06231SAxel Dörfler 		fLastReplicant = index;
121341281cf3SAxel Dörfler 	}
12147da06231SAxel Dörfler 
121541281cf3SAxel Dörfler 	return loc;
121641281cf3SAxel Dörfler }
121741281cf3SAxel Dörfler 
121841281cf3SAxel Dörfler 
121941281cf3SAxel Dörfler BRect
122041281cf3SAxel Dörfler TReplicantTray::IconFrame(int32 target, bool byIndex)
122141281cf3SAxel Dörfler {
122241281cf3SAxel Dörfler 	int32 index, id;
122341281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
1224e4d65fe7SJohn Scipione 	if (view != NULL)
122541281cf3SAxel Dörfler 		return view->Frame();
122641281cf3SAxel Dörfler 
122741281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
122841281cf3SAxel Dörfler }
122941281cf3SAxel Dörfler 
123041281cf3SAxel Dörfler 
123141281cf3SAxel Dörfler BRect
123241281cf3SAxel Dörfler TReplicantTray::IconFrame(const char* name)
123341281cf3SAxel Dörfler {
1234e4d65fe7SJohn Scipione 	if (name == NULL)
123541281cf3SAxel Dörfler 		return BRect(0, 0, 0, 0);
123641281cf3SAxel Dörfler 
123741281cf3SAxel Dörfler 	int32 id, index;
123841281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
1239e4d65fe7SJohn Scipione 	if (view != NULL)
124041281cf3SAxel Dörfler 		return view->Frame();
124141281cf3SAxel Dörfler 
124241281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
124341281cf3SAxel Dörfler }
124441281cf3SAxel Dörfler 
124541281cf3SAxel Dörfler 
12467da06231SAxel Dörfler /**	Scan from the startIndex and reset the location
12477da06231SAxel Dörfler  *	as defined in LocationForReplicant()
124841281cf3SAxel Dörfler  */
124941281cf3SAxel Dörfler 
125041281cf3SAxel Dörfler void
125141281cf3SAxel Dörfler TReplicantTray::RealignReplicants(int32 startIndex)
125241281cf3SAxel Dörfler {
125341281cf3SAxel Dörfler 	if (startIndex < 0)
125441281cf3SAxel Dörfler 		startIndex = 0;
125541281cf3SAxel Dörfler 
125641281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants();
125741281cf3SAxel Dörfler 	if (count <= 0)
125841281cf3SAxel Dörfler 		return;
125941281cf3SAxel Dörfler 
12607da06231SAxel Dörfler 	if (startIndex == 0)
12617da06231SAxel Dörfler 		fRightBottomReplicant.Set(0, 0, 0, 0);
12627da06231SAxel Dörfler 
126341281cf3SAxel Dörfler 	BView* view = NULL;
126441281cf3SAxel Dörfler 	for (int32 i = startIndex; i < count; i++) {
126541281cf3SAxel Dörfler 		fShelf->ReplicantAt(i, &view);
12660bec7100SStefano Ceccherini 		if (view != NULL) {
12677da06231SAxel Dörfler 			BPoint loc = LocationForReplicant(i, view->Frame().Width());
12680bec7100SStefano Ceccherini 			if (view->Frame().LeftTop() != loc)
126941281cf3SAxel Dörfler 				view->MoveTo(loc);
127041281cf3SAxel Dörfler 		}
127141281cf3SAxel Dörfler 	}
127241281cf3SAxel Dörfler }
127341281cf3SAxel Dörfler 
127441281cf3SAxel Dörfler 
12758aab28f1SRene Gollent status_t
12767625ce51SRene Gollent TReplicantTray::_SaveSettings()
12777625ce51SRene Gollent {
12787625ce51SRene Gollent 	status_t result;
12797625ce51SRene Gollent 	BPath path;
128026ea7066SIngo Weinhold 	if ((result = GetDeskbarSettingsDirectory(path, true)) == B_OK) {
12817625ce51SRene Gollent 		path.Append(kReplicantSettingsFile);
12827625ce51SRene Gollent 
12837625ce51SRene Gollent 		BFile file(path.Path(), B_READ_WRITE | B_CREATE_FILE | B_ERASE_FILE);
12847625ce51SRene Gollent 		if ((result = file.InitCheck()) == B_OK)
12857625ce51SRene Gollent 			result = fAddOnSettings.Flatten(&file);
12867625ce51SRene Gollent 	}
12877625ce51SRene Gollent 
12887625ce51SRene Gollent 	return result;
12897625ce51SRene Gollent }
12907625ce51SRene Gollent 
12917625ce51SRene Gollent 
12923cf2d117SJohn Scipione void
12933cf2d117SJohn Scipione TReplicantTray::SaveTimeSettings()
12943cf2d117SJohn Scipione {
12953cf2d117SJohn Scipione 	if (fTime == NULL)
12963cf2d117SJohn Scipione 		return;
12973cf2d117SJohn Scipione 
129831c0024dSJohn Scipione 	clock_settings* settings = ((TBarApp*)be_app)->ClockSettings();
12993cf2d117SJohn Scipione 	settings->showSeconds = fTime->ShowSeconds();
13003cf2d117SJohn Scipione 	settings->showDayOfWeek = fTime->ShowDayOfWeek();
130131c0024dSJohn Scipione 	settings->showTimeZone = fTime->ShowTimeZone();
13023cf2d117SJohn Scipione }
13033cf2d117SJohn Scipione 
13043cf2d117SJohn Scipione 
13051bb2e623SJohn Scipione //	#pragma mark - TDragRegion
130641281cf3SAxel Dörfler 
130741281cf3SAxel Dörfler 
1308ca9acc20SAxel Dörfler /*! Draggable region that is asynchronous so that dragging does not block
1309ca9acc20SAxel Dörfler 	other activities.
13107da06231SAxel Dörfler */
131141281cf3SAxel Dörfler TDragRegion::TDragRegion(TBarView* parent, BView* child)
1312ca9acc20SAxel Dörfler 	:
1313ca9acc20SAxel Dörfler 	BControl(BRect(0, 0, 0, 0), "", "", NULL, B_FOLLOW_NONE,
131441281cf3SAxel Dörfler 		B_WILL_DRAW | B_FRAME_EVENTS),
131541281cf3SAxel Dörfler 	fBarView(parent),
131641281cf3SAxel Dörfler 	fChild(child),
131741281cf3SAxel Dörfler 	fDragLocation(kAutoPlaceDragRegion)
131841281cf3SAxel Dörfler {
131941281cf3SAxel Dörfler }
132041281cf3SAxel Dörfler 
132141281cf3SAxel Dörfler 
132241281cf3SAxel Dörfler void
132341281cf3SAxel Dörfler TDragRegion::AttachedToWindow()
132441281cf3SAxel Dörfler {
132541281cf3SAxel Dörfler 	BView::AttachedToWindow();
1326e4d65fe7SJohn Scipione 
1327cb6afcb1SStephan Aßmus 	if (be_control_look != NULL)
13281bb2e623SJohn Scipione 		SetViewUIColor(B_MENU_BACKGROUND_COLOR, 1.1);
1329cb6afcb1SStephan Aßmus 	else
133016c10517Slooncraz 		SetViewUIColor(B_MENU_BACKGROUND_COLOR);
1331e4d65fe7SJohn Scipione 
133241281cf3SAxel Dörfler 	ResizeToPreferred();
133341281cf3SAxel Dörfler }
133441281cf3SAxel Dörfler 
133541281cf3SAxel Dörfler 
133641281cf3SAxel Dörfler void
133741281cf3SAxel Dörfler TDragRegion::GetPreferredSize(float* width, float* height)
133841281cf3SAxel Dörfler {
133941281cf3SAxel Dörfler 	fChild->ResizeToPreferred();
134041281cf3SAxel Dörfler 	*width = fChild->Bounds().Width();
134141281cf3SAxel Dörfler 	*height = fChild->Bounds().Height();
134241281cf3SAxel Dörfler 
134341281cf3SAxel Dörfler 	if (fDragLocation != kNoDragRegion)
13441bb2e623SJohn Scipione 		*width += kDragWidth + kGutter;
134541281cf3SAxel Dörfler 	else
134641281cf3SAxel Dörfler 		*width += 6;
134741281cf3SAxel Dörfler 
13481bb2e623SJohn Scipione 	*height += 2;
134941281cf3SAxel Dörfler }
135041281cf3SAxel Dörfler 
135141281cf3SAxel Dörfler 
135241281cf3SAxel Dörfler void
135341281cf3SAxel Dörfler TDragRegion::Draw(BRect)
135441281cf3SAxel Dörfler {
1355eaa9af99SStephan Aßmus 	rgb_color menuColor = ViewColor();
135641281cf3SAxel Dörfler 	rgb_color hilite = tint_color(menuColor, B_DARKEN_1_TINT);
1357cb6afcb1SStephan Aßmus 	rgb_color ldark = tint_color(menuColor, 1.02);
1358cb6afcb1SStephan Aßmus 	rgb_color dark = tint_color(menuColor, B_DARKEN_2_TINT);
135941281cf3SAxel Dörfler 	rgb_color vvdark = tint_color(menuColor, B_DARKEN_4_TINT);
136041281cf3SAxel Dörfler 	rgb_color light = tint_color(menuColor, B_LIGHTEN_2_TINT);
136141281cf3SAxel Dörfler 
136241281cf3SAxel Dörfler 	BRect frame(Bounds());
136341281cf3SAxel Dörfler 	BeginLineArray(4);
136441281cf3SAxel Dörfler 
1365cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1366cb6afcb1SStephan Aßmus 		if (fBarView->Vertical()) {
1367cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(), frame.RightTop(), dark);
1368cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1369cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.top + 1), ldark);
1370cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
1371cb6afcb1SStephan Aßmus 		} else if (fBarView->AcrossTop() || fBarView->AcrossBottom()) {
1372cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(),
1373cb6afcb1SStephan Aßmus 				BPoint(frame.left, frame.bottom), dark);
1374cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.top + 1),
1375cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
1376cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right, frame.top + 2),
1377cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.bottom), hilite);
1378cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.bottom),
1379cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
1380cb6afcb1SStephan Aßmus 		}
1381cb6afcb1SStephan Aßmus 	} else {
138241281cf3SAxel Dörfler 		if (fBarView->Vertical()) {
138341281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.RightTop(), light);
138441281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.LeftBottom(), light);
138541281cf3SAxel Dörfler 			AddLine(frame.RightBottom(), frame.RightTop(), hilite);
138641281cf3SAxel Dörfler 		} else if (fBarView->AcrossTop()) {
1387cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1388cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
138941281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1390cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 2),
1391cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1392cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(),
1393cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
139441281cf3SAxel Dörfler 		} else if (fBarView->AcrossBottom()) {
1395cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1396cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
139741281cf3SAxel Dörfler 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
139841281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1399cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 1),
1400cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1401cb6afcb1SStephan Aßmus 		}
140241281cf3SAxel Dörfler 	}
140341281cf3SAxel Dörfler 
140441281cf3SAxel Dörfler 	EndLineArray();
140541281cf3SAxel Dörfler 
140641281cf3SAxel Dörfler 	if (fDragLocation != kDontDrawDragRegion || fDragLocation != kNoDragRegion)
140741281cf3SAxel Dörfler 		DrawDragRegion();
140841281cf3SAxel Dörfler }
140941281cf3SAxel Dörfler 
141041281cf3SAxel Dörfler 
141141281cf3SAxel Dörfler void
141241281cf3SAxel Dörfler TDragRegion::DrawDragRegion()
141341281cf3SAxel Dörfler {
141441281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
141541281cf3SAxel Dörfler 
1416cb6afcb1SStephan Aßmus 	rgb_color menuColor = ViewColor();
1417cb6afcb1SStephan Aßmus 	rgb_color menuHilite = menuColor;
141841281cf3SAxel Dörfler 	if (IsTracking()) {
14191bb2e623SJohn Scipione 		// draw drag region highlighted if tracking mouse
1420cb6afcb1SStephan Aßmus 		menuHilite = tint_color(menuColor, B_HIGHLIGHT_BACKGROUND_TINT);
142141281cf3SAxel Dörfler 		SetHighColor(menuHilite);
142241281cf3SAxel Dörfler 		FillRect(dragRegion);
14231bb2e623SJohn Scipione 	} else {
14241bb2e623SJohn Scipione 		SetHighColor(menuColor);
14251bb2e623SJohn Scipione 		FillRect(dragRegion);
142641281cf3SAxel Dörfler 	}
14271bb2e623SJohn Scipione 
1428cb6afcb1SStephan Aßmus 	rgb_color vdark = tint_color(menuHilite, B_DARKEN_3_TINT);
1429cb6afcb1SStephan Aßmus 	rgb_color light = tint_color(menuHilite, B_LIGHTEN_2_TINT);
1430cb6afcb1SStephan Aßmus 
1431cb6afcb1SStephan Aßmus 	BeginLineArray(dragRegion.IntegerHeight());
1432e4d65fe7SJohn Scipione 	BPoint where;
1433e4d65fe7SJohn Scipione 	where.x = floorf((dragRegion.left + dragRegion.right) / 2 + 0.5) - 1;
1434e4d65fe7SJohn Scipione 	where.y = dragRegion.top + 2;
1435cb6afcb1SStephan Aßmus 
1436e4d65fe7SJohn Scipione 	while (where.y + 1 <= dragRegion.bottom) {
1437e4d65fe7SJohn Scipione 		AddLine(where, where, vdark);
1438e4d65fe7SJohn Scipione 		AddLine(where + BPoint(1, 1), where + BPoint(1, 1), light);
143941281cf3SAxel Dörfler 
1440e4d65fe7SJohn Scipione 		where.y += 3;
144141281cf3SAxel Dörfler 	}
144241281cf3SAxel Dörfler 	EndLineArray();
144341281cf3SAxel Dörfler }
144441281cf3SAxel Dörfler 
144541281cf3SAxel Dörfler 
144641281cf3SAxel Dörfler BRect
144741281cf3SAxel Dörfler TDragRegion::DragRegion() const
144841281cf3SAxel Dörfler {
1449cb6afcb1SStephan Aßmus 	float kTopBottomInset = 2;
1450cb6afcb1SStephan Aßmus 	float kLeftRightInset = 1;
1451cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1452cb6afcb1SStephan Aßmus 		kTopBottomInset = 1;
1453cb6afcb1SStephan Aßmus 		kLeftRightInset = 0;
1454cb6afcb1SStephan Aßmus 	}
1455cb6afcb1SStephan Aßmus 
145641281cf3SAxel Dörfler 	BRect dragRegion(Bounds());
1457cb6afcb1SStephan Aßmus 	dragRegion.top += kTopBottomInset;
1458cb6afcb1SStephan Aßmus 	dragRegion.bottom -= kTopBottomInset;
145941281cf3SAxel Dörfler 
146041281cf3SAxel Dörfler 	bool placeOnLeft = false;
146141281cf3SAxel Dörfler 	if (fDragLocation == kAutoPlaceDragRegion) {
146241281cf3SAxel Dörfler 		if (fBarView->Vertical() && fBarView->Left())
146341281cf3SAxel Dörfler 			placeOnLeft = true;
146441281cf3SAxel Dörfler 		else
146541281cf3SAxel Dörfler 			placeOnLeft = false;
14661bb2e623SJohn Scipione 	} else
14671bb2e623SJohn Scipione 		placeOnLeft = fDragLocation == kDragRegionLeft;
146841281cf3SAxel Dörfler 
146941281cf3SAxel Dörfler 	if (placeOnLeft) {
1470cb6afcb1SStephan Aßmus 		dragRegion.left += kLeftRightInset;
1471cb6afcb1SStephan Aßmus 		dragRegion.right = dragRegion.left + kDragWidth;
147241281cf3SAxel Dörfler 	} else {
14731bb2e623SJohn Scipione 		dragRegion.right += kLeftRightInset;
1474cb6afcb1SStephan Aßmus 		dragRegion.left = dragRegion.right - kDragWidth;
147541281cf3SAxel Dörfler 	}
147641281cf3SAxel Dörfler 
147741281cf3SAxel Dörfler 	return dragRegion;
147841281cf3SAxel Dörfler }
147941281cf3SAxel Dörfler 
148041281cf3SAxel Dörfler 
148141281cf3SAxel Dörfler void
1482e4d65fe7SJohn Scipione TDragRegion::MouseDown(BPoint where)
148341281cf3SAxel Dörfler {
1484e9632898SAlex Smith 	uint32 buttons;
1485e4d65fe7SJohn Scipione 	BPoint mouseLoc;
148641281cf3SAxel Dörfler 
1487e4d65fe7SJohn Scipione 	BRect dragRegion(DragRegion());
14881bb2e623SJohn Scipione 	dragRegion.InsetBy(-2, -2);
14897da06231SAxel Dörfler 		// DragRegion() is designed for drawing, not clicking
149041281cf3SAxel Dörfler 
1491e4d65fe7SJohn Scipione 	if (!dragRegion.Contains(where))
149241281cf3SAxel Dörfler 		return;
149341281cf3SAxel Dörfler 
149441281cf3SAxel Dörfler 	while (true) {
1495e4d65fe7SJohn Scipione 		GetMouse(&mouseLoc, &buttons);
1496e4d65fe7SJohn Scipione 		if (buttons == 0)
149741281cf3SAxel Dörfler 			break;
149841281cf3SAxel Dörfler 
149941281cf3SAxel Dörfler 		if ((Window()->Flags() & B_ASYNCHRONOUS_CONTROLS) != 0) {
1500e4d65fe7SJohn Scipione 			fPreviousPosition = where;
150141281cf3SAxel Dörfler 			SetTracking(true);
150241281cf3SAxel Dörfler 			SetMouseEventMask(B_POINTER_EVENTS,
150341281cf3SAxel Dörfler 				B_NO_POINTER_HISTORY | B_LOCK_WINDOW_FOCUS);
150441281cf3SAxel Dörfler 			Invalidate(DragRegion());
150541281cf3SAxel Dörfler 			break;
150641281cf3SAxel Dörfler 		}
150741281cf3SAxel Dörfler 
150841281cf3SAxel Dörfler 		snooze(25000);
150941281cf3SAxel Dörfler 	}
151041281cf3SAxel Dörfler }
151141281cf3SAxel Dörfler 
151241281cf3SAxel Dörfler 
151341281cf3SAxel Dörfler void
1514e4d65fe7SJohn Scipione TDragRegion::MouseUp(BPoint where)
151541281cf3SAxel Dörfler {
151641281cf3SAxel Dörfler 	if (IsTracking()) {
151741281cf3SAxel Dörfler 		SetTracking(false);
151841281cf3SAxel Dörfler 		Invalidate(DragRegion());
151941281cf3SAxel Dörfler 	} else
1520e4d65fe7SJohn Scipione 		BControl::MouseUp(where);
152141281cf3SAxel Dörfler }
152241281cf3SAxel Dörfler 
152341281cf3SAxel Dörfler 
152441281cf3SAxel Dörfler bool
1525e4d65fe7SJohn Scipione TDragRegion::SwitchModeForRect(BPoint where, BRect rect,
152641281cf3SAxel Dörfler 	bool newVertical, bool newLeft, bool newTop, int32 newState)
152741281cf3SAxel Dörfler {
1528e4d65fe7SJohn Scipione 	if (!rect.Contains(where)) {
152941281cf3SAxel Dörfler 		// not our rect
153041281cf3SAxel Dörfler 		return false;
15311ad8c760SFredrik Holmqvist 	}
153241281cf3SAxel Dörfler 
15331ad8c760SFredrik Holmqvist 	if (newVertical == fBarView->Vertical() && newLeft == fBarView->Left()
15341ad8c760SFredrik Holmqvist 		&& newTop == fBarView->Top() && newState == fBarView->State()) {
153541281cf3SAxel Dörfler 		// already in the correct mode
153641281cf3SAxel Dörfler 		return true;
15371ad8c760SFredrik Holmqvist 	}
153841281cf3SAxel Dörfler 
1539d7ed9414SRene Gollent 	fBarView->ChangeState(newState, newVertical, newLeft, newTop, true);
1540e4d65fe7SJohn Scipione 
154141281cf3SAxel Dörfler 	return true;
154241281cf3SAxel Dörfler }
154341281cf3SAxel Dörfler 
154441281cf3SAxel Dörfler 
154541281cf3SAxel Dörfler void
154641281cf3SAxel Dörfler TDragRegion::MouseMoved(BPoint where, uint32 code, const BMessage* message)
154741281cf3SAxel Dörfler {
154841281cf3SAxel Dörfler 	if (IsTracking()) {
154941281cf3SAxel Dörfler 		BScreen screen;
155041281cf3SAxel Dörfler 		BRect frame = screen.Frame();
155141281cf3SAxel Dörfler 
155241281cf3SAxel Dörfler 		float hDivider = frame.Width() / 6;
15531bb2e623SJohn Scipione 		hDivider = (hDivider < gMinimumWindowWidth + 10)
15541bb2e623SJohn Scipione 			? gMinimumWindowWidth + 10 : hDivider;
15551bb2e623SJohn Scipione 		float miniDivider = frame.top + kMiniHeight + 10;
155641281cf3SAxel Dörfler 		float vDivider = frame.Height() / 2;
155741281cf3SAxel Dörfler #ifdef FULL_MODE
155841281cf3SAxel Dörfler 		float thirdScreen = frame.Height() / 3;
155941281cf3SAxel Dörfler #endif
156071bd3ba5SJonas Sundström 		BRect topLeft(frame.left, frame.top, frame.left + hDivider,
156171bd3ba5SJonas Sundström 			miniDivider);
156271bd3ba5SJonas Sundström 		BRect topMiddle(frame.left + hDivider, frame.top, frame.right
156371bd3ba5SJonas Sundström 			- hDivider, vDivider);
156471bd3ba5SJonas Sundström 		BRect topRight(frame.right - hDivider, frame.top, frame.right,
156571bd3ba5SJonas Sundström 			miniDivider);
156641281cf3SAxel Dörfler 
156741281cf3SAxel Dörfler #ifdef FULL_MODE
156841281cf3SAxel Dörfler 		vDivider = miniDivider + thirdScreen;
156941281cf3SAxel Dörfler #endif
157071bd3ba5SJonas Sundström 		BRect middleLeft(frame.left, miniDivider, frame.left + hDivider,
157171bd3ba5SJonas Sundström 			vDivider);
157271bd3ba5SJonas Sundström 		BRect middleRight(frame.right - hDivider, miniDivider, frame.right,
157371bd3ba5SJonas Sundström 			vDivider);
157441281cf3SAxel Dörfler 
157541281cf3SAxel Dörfler #ifdef FULL_MODE
157671bd3ba5SJonas Sundström 		BRect leftSide(frame.left, vDivider, frame.left + hDivider,
157771bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
157871bd3ba5SJonas Sundström 		BRect rightSide(frame.right - hDivider, vDivider, frame.right,
157971bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
158041281cf3SAxel Dörfler 
158141281cf3SAxel Dörfler 		vDivider = frame.bottom - thirdScreen;
158241281cf3SAxel Dörfler #endif
158371bd3ba5SJonas Sundström 		BRect bottomLeft(frame.left, vDivider, frame.left + hDivider,
158471bd3ba5SJonas Sundström 			frame.bottom);
158571bd3ba5SJonas Sundström 		BRect bottomMiddle(frame.left + hDivider, vDivider, frame.right
158671bd3ba5SJonas Sundström 			- hDivider, frame.bottom);
158771bd3ba5SJonas Sundström 		BRect bottomRight(frame.right - hDivider, vDivider, frame.right,
158871bd3ba5SJonas Sundström 			frame.bottom);
158941281cf3SAxel Dörfler 
159041281cf3SAxel Dörfler 		if (where != fPreviousPosition) {
159141281cf3SAxel Dörfler 			fPreviousPosition = where;
159241281cf3SAxel Dörfler 			ConvertToScreen(&where);
159341281cf3SAxel Dörfler 
159441281cf3SAxel Dörfler 			// use short circuit evaluation for convenience
159541281cf3SAxel Dörfler 			if (SwitchModeForRect(where, topLeft, true, true, true, kMiniState)
159671bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topMiddle, false, true, true,
159771bd3ba5SJonas Sundström 					kExpandoState)
159871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topRight, true, false, true,
159971bd3ba5SJonas Sundström 					kMiniState)
160071bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleLeft, true, true, true,
160171bd3ba5SJonas Sundström 					kExpandoState)
160271bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleRight, true, false, true,
160371bd3ba5SJonas Sundström 					kExpandoState)
160441281cf3SAxel Dörfler 
160541281cf3SAxel Dörfler #ifdef FULL_MODE
160671bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, leftSide, true, true, true,
160771bd3ba5SJonas Sundström 					kFullState)
160871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, rightSide, true, false, true,
160971bd3ba5SJonas Sundström 					kFullState)
161041281cf3SAxel Dörfler #endif
161171bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomLeft, true, true, false,
161271bd3ba5SJonas Sundström 					kMiniState)
161371bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomMiddle, false, true, false,
161471bd3ba5SJonas Sundström 					kExpandoState)
161571bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomRight, true, false, false,
161671bd3ba5SJonas Sundström 					kMiniState))
161741281cf3SAxel Dörfler 				;
161841281cf3SAxel Dörfler 		}
161941281cf3SAxel Dörfler 	} else
162041281cf3SAxel Dörfler 		BControl::MouseMoved(where, code, message);
162141281cf3SAxel Dörfler }
162241281cf3SAxel Dörfler 
162341281cf3SAxel Dörfler 
162441281cf3SAxel Dörfler int32
162541281cf3SAxel Dörfler TDragRegion::DragRegionLocation() const
162641281cf3SAxel Dörfler {
162741281cf3SAxel Dörfler 	return fDragLocation;
162841281cf3SAxel Dörfler }
162941281cf3SAxel Dörfler 
163041281cf3SAxel Dörfler 
163141281cf3SAxel Dörfler void
163241281cf3SAxel Dörfler TDragRegion::SetDragRegionLocation(int32 location)
163341281cf3SAxel Dörfler {
163441281cf3SAxel Dörfler 	if (location == fDragLocation)
163541281cf3SAxel Dörfler 		return;
163641281cf3SAxel Dörfler 
163741281cf3SAxel Dörfler 	fDragLocation = location;
163841281cf3SAxel Dörfler 	Invalidate();
163941281cf3SAxel Dörfler }
16401bb2e623SJohn Scipione 
16411bb2e623SJohn Scipione 
16421bb2e623SJohn Scipione //	#pragma mark - TResizeControl
16431bb2e623SJohn Scipione 
16441bb2e623SJohn Scipione 
16451bb2e623SJohn Scipione /*! Draggable region that is asynchronous so that resizing does not block.
16461bb2e623SJohn Scipione */
16471bb2e623SJohn Scipione TResizeControl::TResizeControl(TBarView* barView)
16481bb2e623SJohn Scipione 	:
16491bb2e623SJohn Scipione 	BControl(BRect(0, kDragWidth, 0, kMenuBarHeight), "", "", NULL,
16501bb2e623SJohn Scipione 		B_FOLLOW_NONE, B_WILL_DRAW | B_FRAME_EVENTS),
16511bb2e623SJohn Scipione 	fBarView(barView)
16521bb2e623SJohn Scipione {
16531bb2e623SJohn Scipione }
16541bb2e623SJohn Scipione 
16551bb2e623SJohn Scipione 
16561bb2e623SJohn Scipione TResizeControl::~TResizeControl()
16571bb2e623SJohn Scipione {
16581bb2e623SJohn Scipione }
16591bb2e623SJohn Scipione 
16601bb2e623SJohn Scipione 
16611bb2e623SJohn Scipione void
16621bb2e623SJohn Scipione TResizeControl::AttachedToWindow()
16631bb2e623SJohn Scipione {
16641bb2e623SJohn Scipione 	BView::AttachedToWindow();
16651bb2e623SJohn Scipione 
16661bb2e623SJohn Scipione 	if (be_control_look != NULL)
16671bb2e623SJohn Scipione 		SetViewUIColor(B_MENU_BACKGROUND_COLOR, 1.1);
16681bb2e623SJohn Scipione 	else
16691bb2e623SJohn Scipione 		SetViewUIColor(B_MENU_BACKGROUND_COLOR);
16701bb2e623SJohn Scipione }
16711bb2e623SJohn Scipione 
16721bb2e623SJohn Scipione 
16731bb2e623SJohn Scipione void
16741bb2e623SJohn Scipione TResizeControl::Draw(BRect updateRect)
16751bb2e623SJohn Scipione {
16761bb2e623SJohn Scipione 	if (!fBarView->Vertical())
16771bb2e623SJohn Scipione 		return;
16781bb2e623SJohn Scipione 
16791bb2e623SJohn Scipione 	BRect dragRegion(updateRect);
16801bb2e623SJohn Scipione 
16811bb2e623SJohn Scipione 	rgb_color menuColor = ViewColor();
16821bb2e623SJohn Scipione 	rgb_color menuHilite = menuColor;
16831bb2e623SJohn Scipione 	if (IsTracking()) {
16841bb2e623SJohn Scipione 		// draw drag region highlighted if tracking mouse
16851bb2e623SJohn Scipione 		menuHilite = tint_color(menuColor, B_HIGHLIGHT_BACKGROUND_TINT);
16861bb2e623SJohn Scipione 		SetHighColor(menuHilite);
16871bb2e623SJohn Scipione 		FillRect(dragRegion);
16881bb2e623SJohn Scipione 	} else {
16891bb2e623SJohn Scipione 		SetHighColor(menuColor);
16901bb2e623SJohn Scipione 		FillRect(dragRegion);
16911bb2e623SJohn Scipione 	}
16921bb2e623SJohn Scipione 
16931bb2e623SJohn Scipione 	rgb_color vdark = tint_color(menuHilite, B_DARKEN_3_TINT);
16941bb2e623SJohn Scipione 	rgb_color light = tint_color(menuHilite, B_LIGHTEN_2_TINT);
16951bb2e623SJohn Scipione 
16961bb2e623SJohn Scipione 	BeginLineArray(dragRegion.IntegerHeight());
16971bb2e623SJohn Scipione 	BPoint where;
16981bb2e623SJohn Scipione 	where.x = floorf((dragRegion.left + dragRegion.right) / 2 + 0.5) - 1;
16991bb2e623SJohn Scipione 	where.y = dragRegion.top + 2;
17001bb2e623SJohn Scipione 
17011bb2e623SJohn Scipione 	while (where.y + 1 <= dragRegion.bottom) {
17021bb2e623SJohn Scipione 		AddLine(where, where, vdark);
17031bb2e623SJohn Scipione 		AddLine(where + BPoint(1, 1), where + BPoint(1, 1), light);
17041bb2e623SJohn Scipione 
17051bb2e623SJohn Scipione 		where.y += 3;
17061bb2e623SJohn Scipione 	}
17071bb2e623SJohn Scipione 	EndLineArray();
17081bb2e623SJohn Scipione }
17091bb2e623SJohn Scipione 
17101bb2e623SJohn Scipione 
17111bb2e623SJohn Scipione void
17121bb2e623SJohn Scipione TResizeControl::MouseDown(BPoint where)
17131bb2e623SJohn Scipione {
17141bb2e623SJohn Scipione 	uint32 buttons;
17151bb2e623SJohn Scipione 	BPoint mouseLoc;
17161bb2e623SJohn Scipione 
17171bb2e623SJohn Scipione 	while (true) {
17181bb2e623SJohn Scipione 		GetMouse(&mouseLoc, &buttons);
17191bb2e623SJohn Scipione 		if (buttons == 0)
17201bb2e623SJohn Scipione 			break;
17211bb2e623SJohn Scipione 
17221bb2e623SJohn Scipione 		if ((Window()->Flags() & B_ASYNCHRONOUS_CONTROLS) != 0) {
17231bb2e623SJohn Scipione 			SetTracking(true);
17241bb2e623SJohn Scipione 			SetMouseEventMask(B_POINTER_EVENTS,
17251bb2e623SJohn Scipione 				B_NO_POINTER_HISTORY | B_LOCK_WINDOW_FOCUS);
17261bb2e623SJohn Scipione 			Invalidate();
17271bb2e623SJohn Scipione 			break;
17281bb2e623SJohn Scipione 		}
17291bb2e623SJohn Scipione 
17301bb2e623SJohn Scipione 		snooze(25000);
17311bb2e623SJohn Scipione 	}
17321bb2e623SJohn Scipione }
17331bb2e623SJohn Scipione 
17341bb2e623SJohn Scipione 
17351bb2e623SJohn Scipione void
17361bb2e623SJohn Scipione TResizeControl::MouseUp(BPoint where)
17371bb2e623SJohn Scipione {
17381bb2e623SJohn Scipione 	if (IsTracking()) {
17391bb2e623SJohn Scipione 		SetTracking(false);
17401bb2e623SJohn Scipione 		Invalidate();
17411bb2e623SJohn Scipione 	} else
17421bb2e623SJohn Scipione 		BControl::MouseUp(where);
17431bb2e623SJohn Scipione }
17441bb2e623SJohn Scipione 
17451bb2e623SJohn Scipione 
17461bb2e623SJohn Scipione void
17471bb2e623SJohn Scipione TResizeControl::MouseMoved(BPoint where, uint32 code,
17481bb2e623SJohn Scipione 	const BMessage* dragMessage)
17491bb2e623SJohn Scipione {
17501bb2e623SJohn Scipione 	if (fBarView->Vertical()) {
17511bb2e623SJohn Scipione 		if (IsResizing()) {
17521bb2e623SJohn Scipione 			float windowWidth = Window()->Frame().Width();
17531bb2e623SJohn Scipione 			float delta = 0;
17541bb2e623SJohn Scipione 			BPoint whereScreen = ConvertToScreen(where);
17551bb2e623SJohn Scipione 
17561bb2e623SJohn Scipione 			if (fBarView->Left()) {
17571bb2e623SJohn Scipione 				delta = whereScreen.x - Window()->Frame().right;
17581bb2e623SJohn Scipione 				if (delta > 0 && windowWidth >= gMaximumWindowWidth)
17591bb2e623SJohn Scipione 					; // do nothing
17601bb2e623SJohn Scipione 				else if (delta < 0 && windowWidth <= gMinimumWindowWidth)
17611bb2e623SJohn Scipione 					; // do nothing
17621bb2e623SJohn Scipione 				else
17631bb2e623SJohn Scipione 					Window()->ResizeBy(delta, 0);
17641bb2e623SJohn Scipione 			} else {
17651bb2e623SJohn Scipione 				delta = Window()->Frame().left - whereScreen.x;
17661bb2e623SJohn Scipione 				if (delta > 0 && windowWidth >= gMaximumWindowWidth)
17671bb2e623SJohn Scipione 					; // do nothing
17681bb2e623SJohn Scipione 				else if (delta < 0 && windowWidth <= gMinimumWindowWidth)
17691bb2e623SJohn Scipione 					; // do nothing
17701bb2e623SJohn Scipione 				else {
17711bb2e623SJohn Scipione 					Window()->MoveBy(delta, 0);
17721bb2e623SJohn Scipione 					Window()->ResizeBy(delta, 0);
17731bb2e623SJohn Scipione 				}
17741bb2e623SJohn Scipione 			}
17751bb2e623SJohn Scipione 
17761bb2e623SJohn Scipione 			windowWidth = Window()->Frame().Width();
17771bb2e623SJohn Scipione 		}
17781bb2e623SJohn Scipione 	}
17791bb2e623SJohn Scipione 
17801bb2e623SJohn Scipione 	BControl::MouseMoved(where, code, dragMessage);
17811bb2e623SJohn Scipione }
1782