xref: /haiku/src/apps/deskbar/StatusView.cpp (revision 573f748c5f8fb7ed75beb0ceb7eb097d3ab0c038)
141281cf3SAxel Dörfler /*
241281cf3SAxel Dörfler Open Tracker License
341281cf3SAxel Dörfler 
441281cf3SAxel Dörfler Terms and Conditions
541281cf3SAxel Dörfler 
641281cf3SAxel Dörfler Copyright (c) 1991-2000, Be Incorporated. All rights reserved.
741281cf3SAxel Dörfler 
841281cf3SAxel Dörfler Permission is hereby granted, free of charge, to any person obtaining a copy of
941281cf3SAxel Dörfler this software and associated documentation files (the "Software"), to deal in
1041281cf3SAxel Dörfler the Software without restriction, including without limitation the rights to
1141281cf3SAxel Dörfler use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
1241281cf3SAxel Dörfler of the Software, and to permit persons to whom the Software is furnished to do
1341281cf3SAxel Dörfler so, subject to the following conditions:
1441281cf3SAxel Dörfler 
1541281cf3SAxel Dörfler The above copyright notice and this permission notice applies to all licensees
1641281cf3SAxel Dörfler and shall be included in all copies or substantial portions of the Software.
1741281cf3SAxel Dörfler 
1841281cf3SAxel Dörfler THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1941281cf3SAxel Dörfler IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF TITLE, MERCHANTABILITY,
2041281cf3SAxel Dörfler FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2141281cf3SAxel Dörfler BE INCORPORATED BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
2241281cf3SAxel Dörfler AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF, OR IN CONNECTION
2341281cf3SAxel Dörfler WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
2441281cf3SAxel Dörfler 
2541281cf3SAxel Dörfler Except as contained in this notice, the name of Be Incorporated shall not be
2641281cf3SAxel Dörfler used in advertising or otherwise to promote the sale, use or other dealings in
2741281cf3SAxel Dörfler this Software without prior written authorization from Be Incorporated.
2841281cf3SAxel Dörfler 
291ad8c760SFredrik Holmqvist Tracker(TM), Be(R), BeOS(R), and BeIA(TM) are trademarks or registered
301ad8c760SFredrik Holmqvist trademarks of Be Incorporated in the United States and other countries. Other
311ad8c760SFredrik Holmqvist brand product names are registered trademarks or trademarks of their respective
321ad8c760SFredrik Holmqvist holders.
3341281cf3SAxel Dörfler All rights reserved.
3441281cf3SAxel Dörfler */
3541281cf3SAxel Dörfler 
3641281cf3SAxel Dörfler #include <Debug.h>
3741281cf3SAxel Dörfler 
3841281cf3SAxel Dörfler #include <errno.h>
3941281cf3SAxel Dörfler #include <stdio.h>
4041281cf3SAxel Dörfler #include <string.h>
4141281cf3SAxel Dörfler #include <time.h>
4241281cf3SAxel Dörfler #include <unistd.h>
4341281cf3SAxel Dörfler #include <algorithm>
4441281cf3SAxel Dörfler 
4541281cf3SAxel Dörfler #include <fs_index.h>
4641281cf3SAxel Dörfler #include <fs_info.h>
4741281cf3SAxel Dörfler 
4841281cf3SAxel Dörfler #include <Application.h>
4941281cf3SAxel Dörfler #include <Beep.h>
50cb6afcb1SStephan Aßmus #include <Bitmap.h>
51ae7a6e3aSAdrien Destugues #include <Catalog.h>
52cb6afcb1SStephan Aßmus #include <ControlLook.h>
5341281cf3SAxel Dörfler #include <Directory.h>
5441281cf3SAxel Dörfler #include <FindDirectory.h>
55ae7a6e3aSAdrien Destugues #include <Locale.h>
5641281cf3SAxel Dörfler #include <MenuItem.h>
57*573f748cSJohn Scipione #include <MutableLocaleRoster.h>
5841281cf3SAxel Dörfler #include <NodeInfo.h>
5941281cf3SAxel Dörfler #include <NodeMonitor.h>
6041281cf3SAxel Dörfler #include <Path.h>
6141281cf3SAxel Dörfler #include <PopUpMenu.h>
6241281cf3SAxel Dörfler #include <Roster.h>
6341281cf3SAxel Dörfler #include <Screen.h>
6441281cf3SAxel Dörfler #include <Volume.h>
6541281cf3SAxel Dörfler #include <VolumeRoster.h>
6641281cf3SAxel Dörfler #include <Window.h>
6741281cf3SAxel Dörfler 
68cb6afcb1SStephan Aßmus #include "icons_logo.h"
69cb6afcb1SStephan Aßmus #include "BarApp.h"
70323b6546SOliver Tappe #include "DeskbarUtils.h"
71cb6afcb1SStephan Aßmus #include "ResourceSet.h"
7241281cf3SAxel Dörfler #include "StatusView.h"
7341281cf3SAxel Dörfler #include "StatusViewShelf.h"
7441281cf3SAxel Dörfler #include "TimeView.h"
7541281cf3SAxel Dörfler 
766d01b66dSAxel Dörfler using std::max;
7741281cf3SAxel Dörfler 
7841281cf3SAxel Dörfler #ifdef DB_ADDONS
7941281cf3SAxel Dörfler // Add-on support
8041281cf3SAxel Dörfler //
8141281cf3SAxel Dörfler // Item - internal item list (node, eref, etc)
8241281cf3SAxel Dörfler // Icon - physical replicant handed to the DeskbarClass class
8341281cf3SAxel Dörfler // AddOn - attribute based add-on
8441281cf3SAxel Dörfler 
8541281cf3SAxel Dörfler const char* const kInstantiateItemCFunctionName = "instantiate_deskbar_item";
8641281cf3SAxel Dörfler const char* const kInstantiateEntryCFunctionName = "instantiate_deskbar_entry";
873e6ff860SRene Gollent const char* const kReplicantSettingsFile = "Deskbar_replicants";
88c0ad1c91SRene Gollent const char* const kReplicantPathField = "replicant_path";
8941281cf3SAxel Dörfler 
90cb6afcb1SStephan Aßmus float sMinimumWindowWidth = kGutter + kMinimumTrayWidth + kDragRegionWidth;
91cb6afcb1SStephan Aßmus 
9241281cf3SAxel Dörfler 
9341281cf3SAxel Dörfler static void
9441281cf3SAxel Dörfler DumpItem(DeskbarItemInfo* item)
9541281cf3SAxel Dörfler {
9641281cf3SAxel Dörfler 	printf("is addon: %i, id: %li\n", item->isAddOn, item->id);
9766eba86fSAxel Dörfler 	printf("entry_ref:  %ld, %Ld, %s\n", item->entryRef.device,
9866eba86fSAxel Dörfler 		item->entryRef.directory, item->entryRef.name);
9941281cf3SAxel Dörfler 	printf("node_ref:  %ld, %Ld\n", item->nodeRef.device, item->nodeRef.node);
10041281cf3SAxel Dörfler }
10141281cf3SAxel Dörfler 
10241281cf3SAxel Dörfler 
10341281cf3SAxel Dörfler static void
10441281cf3SAxel Dörfler DumpList(BList* itemlist)
10541281cf3SAxel Dörfler {
10641281cf3SAxel Dörfler 	int32 count = itemlist->CountItems() - 1;
10741281cf3SAxel Dörfler 	if (count < 0) {
10841281cf3SAxel Dörfler 		printf("no items in list\n");
10941281cf3SAxel Dörfler 		return;
11041281cf3SAxel Dörfler 	}
11141281cf3SAxel Dörfler 	for (int32 i = count; i >= 0; i--) {
11241281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)itemlist->ItemAt(i);
11341281cf3SAxel Dörfler 		if (!item)
11441281cf3SAxel Dörfler 			continue;
11541281cf3SAxel Dörfler 
11641281cf3SAxel Dörfler 		DumpItem(item);
11741281cf3SAxel Dörfler 	}
11841281cf3SAxel Dörfler }
11941281cf3SAxel Dörfler #endif	/* DB_ADDONS */
12041281cf3SAxel Dörfler 
12141281cf3SAxel Dörfler 
1224eb0cbb0SMatt Madia #undef B_TRANSLATE_CONTEXT
1234eb0cbb0SMatt Madia #define B_TRANSLATE_CONTEXT "Tray"
124ae7a6e3aSAdrien Destugues 
1257da06231SAxel Dörfler // don't change the name of this view to anything other than "Status"!
1267da06231SAxel Dörfler 
12741281cf3SAxel Dörfler TReplicantTray::TReplicantTray(TBarView* parent, bool vertical)
12841281cf3SAxel Dörfler 	: BView(BRect(0, 0, 1, 1), "Status", B_FOLLOW_LEFT | B_FOLLOW_TOP,
12941281cf3SAxel Dörfler 			B_WILL_DRAW | B_FRAME_EVENTS),
130*573f748cSJohn Scipione 	fTime(NULL),
13141281cf3SAxel Dörfler 	fBarView(parent),
13241281cf3SAxel Dörfler 	fShelf(new TReplicantShelf(this)),
13341281cf3SAxel Dörfler 	fMultiRowMode(vertical),
134cb6afcb1SStephan Aßmus 	fMinimumTrayWidth(kMinimumTrayWidth),
13541281cf3SAxel Dörfler 	fAlignmentSupport(false)
13641281cf3SAxel Dörfler {
137cb6afcb1SStephan Aßmus 	// init the minimum window width according to the logo.
138cb6afcb1SStephan Aßmus 	const BBitmap* logoBitmap = AppResSet()->FindBitmap(B_MESSAGE_TYPE,
1399e8a970bSJonas Sundström 		R_LeafLogoBitmap);
140cb6afcb1SStephan Aßmus 	if (logoBitmap != NULL) {
141cb6afcb1SStephan Aßmus 		sMinimumWindowWidth = max_c(sMinimumWindowWidth,
142cb6afcb1SStephan Aßmus 			2 * (logoBitmap->Bounds().Width() + 8));
143cb6afcb1SStephan Aßmus 		fMinimumTrayWidth = sMinimumWindowWidth - kGutter - kDragRegionWidth;
144cb6afcb1SStephan Aßmus 	}
145*573f748cSJohn Scipione 
146*573f748cSJohn Scipione 	// Create the time view
147*573f748cSJohn Scipione 	fTime = new TTimeView(fMinimumTrayWidth, kMaxReplicantHeight - 1.0,
148*573f748cSJohn Scipione 		((TBarApp*)be_app)->Settings()->timeFormat);
14941281cf3SAxel Dörfler }
15041281cf3SAxel Dörfler 
15141281cf3SAxel Dörfler 
15241281cf3SAxel Dörfler TReplicantTray::~TReplicantTray()
15341281cf3SAxel Dörfler {
15441281cf3SAxel Dörfler 	delete fShelf;
155*573f748cSJohn Scipione 	delete fTime;
15641281cf3SAxel Dörfler }
15741281cf3SAxel Dörfler 
15841281cf3SAxel Dörfler 
15941281cf3SAxel Dörfler void
16041281cf3SAxel Dörfler TReplicantTray::AttachedToWindow()
16141281cf3SAxel Dörfler {
16241281cf3SAxel Dörfler 	BView::AttachedToWindow();
16341281cf3SAxel Dörfler 
164cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
165cb6afcb1SStephan Aßmus 		SetViewColor(Parent()->ViewColor());
166cb6afcb1SStephan Aßmus 	} else {
16766eba86fSAxel Dörfler 		SetViewColor(tint_color(ui_color(B_MENU_BACKGROUND_COLOR),
16866eba86fSAxel Dörfler 			B_DARKEN_1_TINT));
169cb6afcb1SStephan Aßmus 	}
17041281cf3SAxel Dörfler 	SetDrawingMode(B_OP_COPY);
17166eba86fSAxel Dörfler 
17241281cf3SAxel Dörfler 	Window()->SetPulseRate(1000000);
17341281cf3SAxel Dörfler 
174*573f748cSJohn Scipione 	AddChild(fTime);
175*573f748cSJohn Scipione 	fTime->MoveTo(Bounds().right - fTime->Bounds().Width() - 1, 2);
176*573f748cSJohn Scipione 	if (!((TBarApp*)be_app)->Settings()->showTime) {
177*573f748cSJohn Scipione 		fTime->Hide();
178*573f748cSJohn Scipione 		RealignReplicants();
179*573f748cSJohn Scipione 		AdjustPlacement();
180*573f748cSJohn Scipione 	}
181ae7a6e3aSAdrien Destugues 
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 
20541281cf3SAxel Dörfler void
206*573f748cSJohn Scipione TReplicantTray::SaveTimeSettings()
20741281cf3SAxel Dörfler {
208*573f748cSJohn Scipione 	if (fTime == NULL)
209*573f748cSJohn Scipione 		return;
210*573f748cSJohn Scipione 
21141281cf3SAxel Dörfler 	desk_settings* settings = ((TBarApp*)be_app)->Settings();
212*573f748cSJohn Scipione 	settings->showTime = !fTime->IsHidden();
213*573f748cSJohn Scipione 	settings->timeFormat = fTime->TimeFormat();
21441281cf3SAxel Dörfler }
21541281cf3SAxel Dörfler 
21641281cf3SAxel Dörfler 
21741281cf3SAxel Dörfler void
218*573f748cSJohn Scipione TReplicantTray::ShowHideTime()
21941281cf3SAxel Dörfler {
220*573f748cSJohn Scipione 	if (fTime == NULL)
221*573f748cSJohn Scipione 		return;
22241281cf3SAxel Dörfler 
223*573f748cSJohn Scipione 	if (fTime->IsHidden())
224*573f748cSJohn Scipione 		fTime->Show();
225*573f748cSJohn Scipione 	else
226*573f748cSJohn Scipione 		fTime->Hide();
22741281cf3SAxel Dörfler 
228*573f748cSJohn Scipione 	RealignReplicants();
229*573f748cSJohn Scipione 	AdjustPlacement();
23041281cf3SAxel Dörfler }
23141281cf3SAxel Dörfler 
232*573f748cSJohn Scipione 
233*573f748cSJohn Scipione void
234*573f748cSJohn Scipione TReplicantTray::UpdateTimeFormat(uint32 timeFormat)
235*573f748cSJohn Scipione {
236*573f748cSJohn Scipione 	if (fTime == NULL)
237*573f748cSJohn Scipione 		return;
238*573f748cSJohn Scipione 
239*573f748cSJohn Scipione 	fTime->SetTimeFormat(timeFormat);
240*573f748cSJohn Scipione 
241*573f748cSJohn Scipione 	RealignReplicants();
242*573f748cSJohn Scipione 	AdjustPlacement();
24341281cf3SAxel Dörfler }
24441281cf3SAxel Dörfler 
24541281cf3SAxel Dörfler 
2461ad8c760SFredrik Holmqvist /*! Width is set to a minimum of kMinimumReplicantCount by kMaxReplicantWidth
24766eba86fSAxel Dörfler 	if not in multirowmode and greater than kMinimumReplicantCount
2482ce9bab8SJohn Scipione 	the width should be calculated based on the actual replicant widths
24941281cf3SAxel Dörfler */
25041281cf3SAxel Dörfler void
25141281cf3SAxel Dörfler TReplicantTray::GetPreferredSize(float* preferredWidth, float* preferredHeight)
25241281cf3SAxel Dörfler {
25341281cf3SAxel Dörfler 	float width = 0, height = kMinimumTrayHeight;
25441281cf3SAxel Dörfler 
25541281cf3SAxel Dörfler 	if (fMultiRowMode) {
2567da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0)
2577da06231SAxel Dörfler 			height = fRightBottomReplicant.bottom;
25841281cf3SAxel Dörfler 
2591ad8c760SFredrik Holmqvist 		// the height will be uniform for the number of rows necessary to show
2601ad8c760SFredrik Holmqvist 		// all the reps + any gutters necessary for spacing
26141281cf3SAxel Dörfler 		int32 rowCount = (int32)(height / kMaxReplicantHeight);
26241281cf3SAxel Dörfler 		height = kGutter + (rowCount * kMaxReplicantHeight)
26341281cf3SAxel Dörfler 			+ ((rowCount - 1) * kIconGap) + kGutter;
26441281cf3SAxel Dörfler 		height = max(kMinimumTrayHeight, height);
265cb6afcb1SStephan Aßmus 		width = fMinimumTrayWidth;
26641281cf3SAxel Dörfler 	} else {
2671ad8c760SFredrik Holmqvist 		// if last replicant overruns clock then resize to accomodate
2687da06231SAxel Dörfler 		if (fShelf->CountReplicants() > 0) {
269*573f748cSJohn Scipione 			if (!fTime->IsHidden() && fTime->Frame().left
270*573f748cSJohn Scipione 				< fRightBottomReplicant.right + 6) {
27166eba86fSAxel Dörfler 				width = fRightBottomReplicant.right + 6
272*573f748cSJohn Scipione 					+ fTime->Frame().Width();
27341281cf3SAxel Dörfler 			} else
2747da06231SAxel Dörfler 				width = fRightBottomReplicant.right + 3;
27541281cf3SAxel Dörfler 		}
2761ad8c760SFredrik Holmqvist 
27741281cf3SAxel Dörfler 		// this view has a fixed minimum width
278cb6afcb1SStephan Aßmus 		width = max(fMinimumTrayWidth, width);
2792ce9bab8SJohn Scipione 		height = kGutter + static_cast<TBarApp*>(be_app)->IconSize() + kGutter;
28041281cf3SAxel Dörfler 	}
28141281cf3SAxel Dörfler 
28241281cf3SAxel Dörfler 	*preferredWidth = width;
2831ad8c760SFredrik Holmqvist 	// add 1 for the border
28441281cf3SAxel Dörfler 	*preferredHeight = height + 1;
28541281cf3SAxel Dörfler }
28641281cf3SAxel Dörfler 
28741281cf3SAxel Dörfler 
28841281cf3SAxel Dörfler void
28941281cf3SAxel Dörfler TReplicantTray::AdjustPlacement()
29041281cf3SAxel Dörfler {
29141281cf3SAxel Dörfler 	// called when an add-on has been added or removed
29241281cf3SAxel Dörfler 	// need to resize the parent of this accordingly
293a5210ab1SAxel Dörfler 
29441281cf3SAxel Dörfler 	BRect bounds = Bounds();
29541281cf3SAxel Dörfler 	float width, height;
29641281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
29741281cf3SAxel Dörfler 
29841281cf3SAxel Dörfler 	if (width == bounds.Width() && height == bounds.Height()) {
29941281cf3SAxel Dörfler 		// no need to change anything
30041281cf3SAxel Dörfler 		return;
30141281cf3SAxel Dörfler 	}
30241281cf3SAxel Dörfler 
30341281cf3SAxel Dörfler 	Parent()->ResizeToPreferred();
30441281cf3SAxel Dörfler 	fBarView->UpdatePlacement();
30541281cf3SAxel Dörfler 	Parent()->Invalidate();
30641281cf3SAxel Dörfler 	Invalidate();
30741281cf3SAxel Dörfler }
30841281cf3SAxel Dörfler 
30941281cf3SAxel Dörfler 
31041281cf3SAxel Dörfler void
31141281cf3SAxel Dörfler TReplicantTray::MessageReceived(BMessage* message)
31241281cf3SAxel Dörfler {
31341281cf3SAxel Dörfler 	switch (message->what) {
314*573f748cSJohn Scipione 		case kShowHideTime:
31541281cf3SAxel Dörfler 			// from context menu in clock and in this view
316*573f748cSJohn Scipione 			ShowHideTime();
31741281cf3SAxel Dörfler 			break;
31841281cf3SAxel Dörfler 
319*573f748cSJohn Scipione 		case kTimeIntervalChanged:
320*573f748cSJohn Scipione 		{
321*573f748cSJohn Scipione 			if (fTime == NULL)
322*573f748cSJohn Scipione 				return;
323*573f748cSJohn Scipione 
324*573f748cSJohn Scipione 			bool use24HourClock;
325*573f748cSJohn Scipione 			if (message->FindBool("use24HourClock", &use24HourClock)
326*573f748cSJohn Scipione 				== B_OK) {
327*573f748cSJohn Scipione 				BFormattingConventions conventions;
328*573f748cSJohn Scipione 				BLocale::Default()->GetFormattingConventions(&conventions);
329*573f748cSJohn Scipione 				conventions.SetExplicitUse24HourClock(use24HourClock);
330*573f748cSJohn Scipione 				BPrivate::MutableLocaleRoster::Default()->
331*573f748cSJohn Scipione 					SetDefaultFormattingConventions(conventions);
332*573f748cSJohn Scipione 				fTime->Update();
33341281cf3SAxel Dörfler 				// time string reformat -> realign
33441281cf3SAxel Dörfler 				RealignReplicants();
33541281cf3SAxel Dörfler 				AdjustPlacement();
336*573f748cSJohn Scipione 			}
33741281cf3SAxel Dörfler 
33841281cf3SAxel Dörfler 			break;
339*573f748cSJohn Scipione 		}
340*573f748cSJohn Scipione 
341*573f748cSJohn Scipione 		case kTimeFormatChanged:
342*573f748cSJohn Scipione 		{
343*573f748cSJohn Scipione 			if (fTime == NULL)
344*573f748cSJohn Scipione 				return;
345*573f748cSJohn Scipione 
346*573f748cSJohn Scipione 			uint32 timeFormat;
347*573f748cSJohn Scipione 			if (message->FindUInt32("time format", &timeFormat) == B_OK)
348*573f748cSJohn Scipione 				UpdateTimeFormat(timeFormat);
349*573f748cSJohn Scipione 
350*573f748cSJohn Scipione 			break;
351*573f748cSJohn Scipione 		}
35241281cf3SAxel Dörfler 
35341281cf3SAxel Dörfler #ifdef DB_ADDONS
35441281cf3SAxel Dörfler 		case B_NODE_MONITOR:
35541281cf3SAxel Dörfler 			HandleEntryUpdate(message);
35641281cf3SAxel Dörfler 			break;
35741281cf3SAxel Dörfler #endif
35841281cf3SAxel Dörfler 
35941281cf3SAxel Dörfler 		default:
36041281cf3SAxel Dörfler 			BView::MessageReceived(message);
36141281cf3SAxel Dörfler 			break;
36241281cf3SAxel Dörfler 	}
36341281cf3SAxel Dörfler }
36441281cf3SAxel Dörfler 
36541281cf3SAxel Dörfler 
36641281cf3SAxel Dörfler void
36741281cf3SAxel Dörfler TReplicantTray::ShowReplicantMenu(BPoint point)
36841281cf3SAxel Dörfler {
36941281cf3SAxel Dörfler 	BPopUpMenu* menu = new BPopUpMenu("", false, false);
37041281cf3SAxel Dörfler 	menu->SetFont(be_plain_font);
37141281cf3SAxel Dörfler 
3721ad8c760SFredrik Holmqvist 	// If clock is visible show the extended menu, otherwise show "Show Time"
37341281cf3SAxel Dörfler 
374*573f748cSJohn Scipione 	if (fTime != NULL)
375*573f748cSJohn Scipione 		fTime->ShowTimeOptions(ConvertToScreen(point));
37641281cf3SAxel Dörfler 	else {
37761863f2cSMatt Madia 		BMenuItem* item = new BMenuItem(B_TRANSLATE("Show Time"),
378*573f748cSJohn Scipione 			new BMessage(kShowHideTime));
37941281cf3SAxel Dörfler 		menu->AddItem(item);
38041281cf3SAxel Dörfler 		menu->SetTargetForItems(this);
38141281cf3SAxel Dörfler 		BPoint where = ConvertToScreen(point);
38241281cf3SAxel Dörfler 		menu->Go(where, true, true, BRect(where - BPoint(4, 4),
38341281cf3SAxel Dörfler 			where + BPoint(4, 4)), true);
38441281cf3SAxel Dörfler 	}
38541281cf3SAxel Dörfler }
38641281cf3SAxel Dörfler 
38741281cf3SAxel Dörfler 
38841281cf3SAxel Dörfler void
38941281cf3SAxel Dörfler TReplicantTray::MouseDown(BPoint where)
39041281cf3SAxel Dörfler {
39141281cf3SAxel Dörfler #ifdef DB_ADDONS
39241281cf3SAxel Dörfler 	if (modifiers() & B_CONTROL_KEY)
39341281cf3SAxel Dörfler 		DumpList(fItemList);
39441281cf3SAxel Dörfler #endif
39541281cf3SAxel Dörfler 
39641281cf3SAxel Dörfler 	uint32 buttons;
39741281cf3SAxel Dörfler 
39841281cf3SAxel Dörfler 	Window()->CurrentMessage()->FindInt32("buttons", (int32*)&buttons);
39941281cf3SAxel Dörfler 	if (buttons == B_SECONDARY_MOUSE_BUTTON) {
40041281cf3SAxel Dörfler 		ShowReplicantMenu(where);
40141281cf3SAxel Dörfler 	} else {
40241281cf3SAxel Dörfler 		BPoint save = where;
40341281cf3SAxel Dörfler 		bigtime_t doubleClickSpeed;
40441281cf3SAxel Dörfler 		bigtime_t start = system_time();
40541281cf3SAxel Dörfler 		uint32 buttons;
40641281cf3SAxel Dörfler 
40741281cf3SAxel Dörfler 		get_click_speed(&doubleClickSpeed);
40841281cf3SAxel Dörfler 
40941281cf3SAxel Dörfler 		do {
41041281cf3SAxel Dörfler 			if (fabs(where.x - save.x) > 4 || fabs(where.y - save.y) > 4)
41141281cf3SAxel Dörfler 				// user moved out of bounds of click area
41241281cf3SAxel Dörfler 				break;
41341281cf3SAxel Dörfler 
41441281cf3SAxel Dörfler 			if ((system_time() - start) > (2 * doubleClickSpeed)) {
41541281cf3SAxel Dörfler 				ShowReplicantMenu(where);
41641281cf3SAxel Dörfler 				break;
41741281cf3SAxel Dörfler 			}
41841281cf3SAxel Dörfler 
41941281cf3SAxel Dörfler 			snooze(50000);
42041281cf3SAxel Dörfler 			GetMouse(&where, &buttons);
42141281cf3SAxel Dörfler 		} while (buttons);
42241281cf3SAxel Dörfler 	}
42341281cf3SAxel Dörfler 	BView::MouseDown(where);
42441281cf3SAxel Dörfler }
42541281cf3SAxel Dörfler 
42641281cf3SAxel Dörfler #ifdef DB_ADDONS
42741281cf3SAxel Dörfler 
42841281cf3SAxel Dörfler void
42941281cf3SAxel Dörfler TReplicantTray::InitAddOnSupport()
43041281cf3SAxel Dörfler {
43141281cf3SAxel Dörfler 	// list to maintain refs to each rep added/deleted
43241281cf3SAxel Dörfler 	fItemList = new BList();
43341281cf3SAxel Dörfler 	BPath path;
4341ad8c760SFredrik Holmqvist 
43541281cf3SAxel Dörfler 	if (find_directory(B_USER_SETTINGS_DIRECTORY, &path, true) == B_OK) {
4363e6ff860SRene Gollent 		path.Append(kReplicantSettingsFile);
43741281cf3SAxel Dörfler 
43841281cf3SAxel Dörfler 		BFile file(path.Path(), B_READ_ONLY);
4393e6ff860SRene Gollent 		if (file.InitCheck() == B_OK) {
4403e6ff860SRene Gollent 			status_t result;
4413e6ff860SRene Gollent 			BEntry entry;
4423e6ff860SRene Gollent 			int32 id;
443c0ad1c91SRene Gollent 			BString path;
4443e6ff860SRene Gollent 			if (fAddOnSettings.Unflatten(&file) == B_OK) {
445c0ad1c91SRene Gollent 				for (int32 i = 0; fAddOnSettings.FindString(kReplicantPathField,
446c0ad1c91SRene Gollent 					i, &path) == B_OK; i++) {
447c0ad1c91SRene Gollent 					if (entry.SetTo(path.String()) == B_OK && entry.Exists()) {
4483e6ff860SRene Gollent 						result = LoadAddOn(&entry, &id, false);
4493e6ff860SRene Gollent 					} else
4503e6ff860SRene Gollent 						result = B_ENTRY_NOT_FOUND;
45141281cf3SAxel Dörfler 
4523e6ff860SRene Gollent 					if (result != B_OK) {
453c0ad1c91SRene Gollent 						fAddOnSettings.RemoveData(kReplicantPathField, i);
4543e6ff860SRene Gollent 						--i;
45541281cf3SAxel Dörfler 					}
45641281cf3SAxel Dörfler 				}
45741281cf3SAxel Dörfler 			}
4583e6ff860SRene Gollent 		}
4593e6ff860SRene Gollent 	}
46041281cf3SAxel Dörfler }
46141281cf3SAxel Dörfler 
46241281cf3SAxel Dörfler 
46341281cf3SAxel Dörfler void
46441281cf3SAxel Dörfler TReplicantTray::DeleteAddOnSupport()
46541281cf3SAxel Dörfler {
4667625ce51SRene Gollent 	_SaveSettings();
4673e6ff860SRene Gollent 
46841281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
46941281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->RemoveItem(i);
47041281cf3SAxel Dörfler 		if (item) {
47141281cf3SAxel Dörfler 			if (item->isAddOn)
47241281cf3SAxel Dörfler 				watch_node(&(item->nodeRef), B_STOP_WATCHING, this, Window());
47341281cf3SAxel Dörfler 
47441281cf3SAxel Dörfler 			delete item;
47541281cf3SAxel Dörfler 		}
47641281cf3SAxel Dörfler 	}
47741281cf3SAxel Dörfler 	delete fItemList;
47841281cf3SAxel Dörfler 
47941281cf3SAxel Dörfler 	// stop the volume mount/unmount watch
48041281cf3SAxel Dörfler 	stop_watching(this, Window());
48141281cf3SAxel Dörfler }
48241281cf3SAxel Dörfler 
48341281cf3SAxel Dörfler 
48441281cf3SAxel Dörfler DeskbarItemInfo*
48541281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(node_ref& nodeRef)
48641281cf3SAxel Dörfler {
48741281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
48866eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
48941281cf3SAxel Dörfler 		if (item == NULL)
49041281cf3SAxel Dörfler 			continue;
49141281cf3SAxel Dörfler 
49241281cf3SAxel Dörfler 		if (item->nodeRef == nodeRef)
49341281cf3SAxel Dörfler 			return item;
49441281cf3SAxel Dörfler 	}
49541281cf3SAxel Dörfler 
49641281cf3SAxel Dörfler 	return NULL;
49741281cf3SAxel Dörfler }
49841281cf3SAxel Dörfler 
49941281cf3SAxel Dörfler 
50041281cf3SAxel Dörfler DeskbarItemInfo*
50141281cf3SAxel Dörfler TReplicantTray::DeskbarItemFor(int32 id)
50241281cf3SAxel Dörfler {
50341281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
50466eba86fSAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
50541281cf3SAxel Dörfler 		if (item == NULL)
50641281cf3SAxel Dörfler 			continue;
50741281cf3SAxel Dörfler 
50841281cf3SAxel Dörfler 		if (item->id == id)
50941281cf3SAxel Dörfler 			return item;
51041281cf3SAxel Dörfler 	}
51141281cf3SAxel Dörfler 
51241281cf3SAxel Dörfler 	return NULL;
51341281cf3SAxel Dörfler }
51441281cf3SAxel Dörfler 
51541281cf3SAxel Dörfler 
51641281cf3SAxel Dörfler bool
51741281cf3SAxel Dörfler TReplicantTray::NodeExists(node_ref& nodeRef)
51841281cf3SAxel Dörfler {
51941281cf3SAxel Dörfler 	return DeskbarItemFor(nodeRef) != NULL;
52041281cf3SAxel Dörfler }
52141281cf3SAxel Dörfler 
52241281cf3SAxel Dörfler 
52366eba86fSAxel Dörfler /*! This handles B_NODE_MONITOR & B_QUERY_UPDATE messages received
52466eba86fSAxel Dörfler 	for the registered add-ons.
52541281cf3SAxel Dörfler */
52641281cf3SAxel Dörfler void
52741281cf3SAxel Dörfler TReplicantTray::HandleEntryUpdate(BMessage* message)
52841281cf3SAxel Dörfler {
52941281cf3SAxel Dörfler 	int32 opcode;
53041281cf3SAxel Dörfler 	if (message->FindInt32("opcode", &opcode) != B_OK)
53141281cf3SAxel Dörfler 		return;
53241281cf3SAxel Dörfler 
53341281cf3SAxel Dörfler 	BPath path;
53441281cf3SAxel Dörfler 	switch (opcode) {
53541281cf3SAxel Dörfler 		case B_ENTRY_MOVED:
53641281cf3SAxel Dörfler 		{
53741281cf3SAxel Dörfler 			entry_ref ref;
53841281cf3SAxel Dörfler 			ino_t todirectory;
53941281cf3SAxel Dörfler 			ino_t node;
54041281cf3SAxel Dörfler 			const char* name;
54141281cf3SAxel Dörfler 			if (message->FindString("name", &name) == B_OK
54271bd3ba5SJonas Sundström 				&& message->FindInt64("from directory", &(ref.directory))
54371bd3ba5SJonas Sundström 				== B_OK
54441281cf3SAxel Dörfler 				&& message->FindInt64("to directory", &todirectory) == B_OK
54541281cf3SAxel Dörfler 				&& message->FindInt32("device", &(ref.device)) == B_OK
54641281cf3SAxel Dörfler 				&& message->FindInt64("node", &node) == B_OK ) {
54741281cf3SAxel Dörfler 
54841281cf3SAxel Dörfler 				if (!name)
54941281cf3SAxel Dörfler 					break;
55041281cf3SAxel Dörfler 
55141281cf3SAxel Dörfler 				ref.set_name(name);
55241281cf3SAxel Dörfler 				// change the directory reference to
55341281cf3SAxel Dörfler 				// the new directory
55441281cf3SAxel Dörfler 				MoveItem(&ref, todirectory);
55541281cf3SAxel Dörfler 			}
55641281cf3SAxel Dörfler 			break;
55766eba86fSAxel Dörfler 		}
55841281cf3SAxel Dörfler 
55941281cf3SAxel Dörfler 		case B_ENTRY_REMOVED:
56041281cf3SAxel Dörfler 		{
56141281cf3SAxel Dörfler 			// entry was rm'd from the device
56241281cf3SAxel Dörfler 			node_ref nodeRef;
56341281cf3SAxel Dörfler 			if (message->FindInt32("device", &(nodeRef.device)) == B_OK
56441281cf3SAxel Dörfler 				&& message->FindInt64("node", &(nodeRef.node)) == B_OK) {
56541281cf3SAxel Dörfler 				DeskbarItemInfo* item = DeskbarItemFor(nodeRef);
56641281cf3SAxel Dörfler 				if (item == NULL)
56741281cf3SAxel Dörfler 					break;
56841281cf3SAxel Dörfler 
56941281cf3SAxel Dörfler 				// If there is a team running where the add-on comes from,
57041281cf3SAxel Dörfler 				// we don't want to remove the icon yet.
57141281cf3SAxel Dörfler 				if (be_roster->IsRunning(&item->entryRef))
57241281cf3SAxel Dörfler 					break;
57341281cf3SAxel Dörfler 
57441281cf3SAxel Dörfler 				UnloadAddOn(&nodeRef, NULL, true, false);
57541281cf3SAxel Dörfler 			}
57641281cf3SAxel Dörfler 			break;
57766eba86fSAxel Dörfler 		}
57841281cf3SAxel Dörfler 	}
57966eba86fSAxel Dörfler }
58041281cf3SAxel Dörfler 
58141281cf3SAxel Dörfler 
5821ad8c760SFredrik Holmqvist /*! The add-ons must support the exported C function API
58366eba86fSAxel Dörfler 	if they do, they will be loaded and added to deskbar
58466eba86fSAxel Dörfler 	primary function is the Instantiate function
5857da06231SAxel Dörfler */
58641281cf3SAxel Dörfler status_t
5873e6ff860SRene Gollent TReplicantTray::LoadAddOn(BEntry* entry, int32* id, bool addToSettings)
58841281cf3SAxel Dörfler {
58941281cf3SAxel Dörfler 	if (!entry)
59041281cf3SAxel Dörfler 		return B_ERROR;
59141281cf3SAxel Dörfler 
59241281cf3SAxel Dörfler 	node_ref nodeRef;
59341281cf3SAxel Dörfler 	entry->GetNodeRef(&nodeRef);
59441281cf3SAxel Dörfler 	// no duplicates
59541281cf3SAxel Dörfler 	if (NodeExists(nodeRef))
59641281cf3SAxel Dörfler 		return B_ERROR;
59741281cf3SAxel Dörfler 
59841281cf3SAxel Dörfler 	BNode node(entry);
59941281cf3SAxel Dörfler 	BPath path;
60066eba86fSAxel Dörfler 	status_t status = entry->GetPath(&path);
60166eba86fSAxel Dörfler 	if (status < B_OK)
60266eba86fSAxel Dörfler 		return status;
60341281cf3SAxel Dörfler 
60441281cf3SAxel Dörfler 	// load the add-on
60541281cf3SAxel Dörfler 	image_id image = load_add_on(path.Path());
60666eba86fSAxel Dörfler 	if (image < B_OK)
60766eba86fSAxel Dörfler 		return image;
60841281cf3SAxel Dörfler 
60941281cf3SAxel Dörfler 	// get the view loading function symbol
61041281cf3SAxel Dörfler 	//    we first look for a symbol that takes an image_id
61141281cf3SAxel Dörfler 	//    and entry_ref pointer, if not found, go with normal
61241281cf3SAxel Dörfler 	//    instantiate function
61341281cf3SAxel Dörfler 	BView* (*entryFunction)(image_id, const entry_ref*);
61441281cf3SAxel Dörfler 	BView* (*itemFunction)(void);
61541281cf3SAxel Dörfler 	BView* view = NULL;
61641281cf3SAxel Dörfler 
61741281cf3SAxel Dörfler 	entry_ref ref;
61841281cf3SAxel Dörfler 	entry->GetRef(&ref);
61941281cf3SAxel Dörfler 
62041281cf3SAxel Dörfler 	if (get_image_symbol(image, kInstantiateEntryCFunctionName,
62166eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&entryFunction) >= B_OK) {
62241281cf3SAxel Dörfler 		view = (*entryFunction)(image, &ref);
62341281cf3SAxel Dörfler 	} else if (get_image_symbol(image, kInstantiateItemCFunctionName,
62466eba86fSAxel Dörfler 			B_SYMBOL_TYPE_TEXT, (void**)&itemFunction) >= B_OK) {
62541281cf3SAxel Dörfler 		view = (*itemFunction)();
62641281cf3SAxel Dörfler 	} else {
62741281cf3SAxel Dörfler 		unload_add_on(image);
62841281cf3SAxel Dörfler 		return B_ERROR;
62941281cf3SAxel Dörfler 	}
63041281cf3SAxel Dörfler 
63166eba86fSAxel Dörfler 	if (view == NULL || IconExists(view->Name())) {
63241281cf3SAxel Dörfler 		delete view;
63341281cf3SAxel Dörfler 		unload_add_on(image);
63441281cf3SAxel Dörfler 		return B_ERROR;
63541281cf3SAxel Dörfler 	}
63641281cf3SAxel Dörfler 
63741281cf3SAxel Dörfler 	BMessage* data = new BMessage;
63841281cf3SAxel Dörfler 	view->Archive(data);
63941281cf3SAxel Dörfler 	delete view;
64041281cf3SAxel Dörfler 
64141281cf3SAxel Dörfler 	AddIcon(data, id, &ref);
64241281cf3SAxel Dörfler 		// add the rep; adds info to list
64341281cf3SAxel Dörfler 
6443e6ff860SRene Gollent 	if (addToSettings) {
645c0ad1c91SRene Gollent 		fAddOnSettings.AddString(kReplicantPathField, path.Path());
6467625ce51SRene Gollent 		_SaveSettings();
6473e6ff860SRene Gollent 	}
64841281cf3SAxel Dörfler 
64941281cf3SAxel Dörfler 	return B_OK;
65041281cf3SAxel Dörfler }
65141281cf3SAxel Dörfler 
65241281cf3SAxel Dörfler 
65341281cf3SAxel Dörfler status_t
65441281cf3SAxel Dörfler TReplicantTray::AddItem(int32 id, node_ref nodeRef, BEntry& entry, bool isAddOn)
65541281cf3SAxel Dörfler {
65641281cf3SAxel Dörfler 	DeskbarItemInfo* item = new DeskbarItemInfo;
65741281cf3SAxel Dörfler 	if (item == NULL)
65841281cf3SAxel Dörfler 		return B_NO_MEMORY;
65941281cf3SAxel Dörfler 
66041281cf3SAxel Dörfler 	item->id = id;
66141281cf3SAxel Dörfler 	item->isAddOn = isAddOn;
66241281cf3SAxel Dörfler 
66341281cf3SAxel Dörfler 	if (entry.GetRef(&item->entryRef) < B_OK) {
66441281cf3SAxel Dörfler 		item->entryRef.device = -1;
66541281cf3SAxel Dörfler 		item->entryRef.directory = -1;
66641281cf3SAxel Dörfler 		item->entryRef.name = NULL;
66741281cf3SAxel Dörfler 	}
66841281cf3SAxel Dörfler 	item->nodeRef = nodeRef;
66941281cf3SAxel Dörfler 
67041281cf3SAxel Dörfler 	fItemList->AddItem(item);
67141281cf3SAxel Dörfler 
67241281cf3SAxel Dörfler 	if (isAddOn)
67341281cf3SAxel Dörfler 		watch_node(&nodeRef, B_WATCH_NAME | B_WATCH_ATTR, this, Window());
67441281cf3SAxel Dörfler 
67541281cf3SAxel Dörfler 	return B_OK;
67641281cf3SAxel Dörfler }
67741281cf3SAxel Dörfler 
67841281cf3SAxel Dörfler 
67941281cf3SAxel Dörfler /**	from entry_removed message, when attribute removed
68041281cf3SAxel Dörfler  *	or when a device is unmounted (use removeall, by device)
68141281cf3SAxel Dörfler  */
68241281cf3SAxel Dörfler 
68341281cf3SAxel Dörfler void
68441281cf3SAxel Dörfler TReplicantTray::UnloadAddOn(node_ref* nodeRef, dev_t* device,
68541281cf3SAxel Dörfler 	bool which, bool removeAll)
68641281cf3SAxel Dörfler {
68741281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
68841281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
68941281cf3SAxel Dörfler 		if (!item)
69041281cf3SAxel Dörfler 			continue;
69141281cf3SAxel Dörfler 
69241281cf3SAxel Dörfler 		if ((which && nodeRef && item->nodeRef == *nodeRef)
69341281cf3SAxel Dörfler 			|| (device && item->nodeRef.device == *device)) {
69441281cf3SAxel Dörfler 
69541281cf3SAxel Dörfler 			if (device && be_roster->IsRunning(&item->entryRef))
69641281cf3SAxel Dörfler 				continue;
69741281cf3SAxel Dörfler 
69841281cf3SAxel Dörfler 			RemoveIcon(item->id);
69941281cf3SAxel Dörfler 
70041281cf3SAxel Dörfler 			if (!removeAll)
70141281cf3SAxel Dörfler 				break;
70241281cf3SAxel Dörfler 		}
70341281cf3SAxel Dörfler 	}
70441281cf3SAxel Dörfler }
70541281cf3SAxel Dörfler 
70641281cf3SAxel Dörfler 
70741281cf3SAxel Dörfler void
70841281cf3SAxel Dörfler TReplicantTray::RemoveItem(int32 id)
70941281cf3SAxel Dörfler {
71041281cf3SAxel Dörfler 	DeskbarItemInfo* item = DeskbarItemFor(id);
71141281cf3SAxel Dörfler 	if (item == NULL)
71241281cf3SAxel Dörfler 		return;
71341281cf3SAxel Dörfler 
71441281cf3SAxel Dörfler 	// attribute was added via Deskbar API (AddItem(entry_ref*, int32*)
71541281cf3SAxel Dörfler 	if (item->isAddOn) {
7167625ce51SRene Gollent 		BPath path(&item->entryRef);
7177625ce51SRene Gollent 		BString storedPath;
7187625ce51SRene Gollent 		for (int32 i = 0;
7198aab28f1SRene Gollent 			fAddOnSettings.FindString(kReplicantPathField, i, &storedPath)
7207625ce51SRene Gollent 				== B_OK; i++) {
7217625ce51SRene Gollent 			if (storedPath == path.Path()) {
7228aab28f1SRene Gollent 				fAddOnSettings.RemoveData(kReplicantPathField, i);
7237625ce51SRene Gollent 				break;
7247625ce51SRene Gollent 			}
7257625ce51SRene Gollent 		}
7267625ce51SRene Gollent 		_SaveSettings();
7277625ce51SRene Gollent 
72841281cf3SAxel Dörfler 		BNode node(&item->entryRef);
72941281cf3SAxel Dörfler 		watch_node(&item->nodeRef, B_STOP_WATCHING, this, Window());
73041281cf3SAxel Dörfler 	}
73141281cf3SAxel Dörfler 
73241281cf3SAxel Dörfler 	fItemList->RemoveItem(item);
73341281cf3SAxel Dörfler 	delete item;
73441281cf3SAxel Dörfler }
73541281cf3SAxel Dörfler 
73641281cf3SAxel Dörfler 
73741281cf3SAxel Dörfler /**	ENTRY_MOVED message, moving only occurs on a device
73841281cf3SAxel Dörfler  *	copying will occur (ENTRY_CREATED) between devices
73941281cf3SAxel Dörfler  */
74041281cf3SAxel Dörfler 
74141281cf3SAxel Dörfler void
74241281cf3SAxel Dörfler TReplicantTray::MoveItem(entry_ref* ref, ino_t toDirectory)
74341281cf3SAxel Dörfler {
74441281cf3SAxel Dörfler 	if (!ref)
74541281cf3SAxel Dörfler 		return;
74641281cf3SAxel Dörfler 
74741281cf3SAxel Dörfler 	// scan for a matching entry_ref and update it
74841281cf3SAxel Dörfler 	//
74941281cf3SAxel Dörfler 	// don't need to change node info as it does not change
75041281cf3SAxel Dörfler 
75141281cf3SAxel Dörfler 	for (int32 i = fItemList->CountItems(); i-- > 0 ;) {
75241281cf3SAxel Dörfler 		DeskbarItemInfo* item = (DeskbarItemInfo*)fItemList->ItemAt(i);
75341281cf3SAxel Dörfler 		if (!item)
75441281cf3SAxel Dörfler 			continue;
75541281cf3SAxel Dörfler 
75641281cf3SAxel Dörfler 		if (!strcmp(item->entryRef.name, ref->name)
75741281cf3SAxel Dörfler 			&& item->entryRef.device == ref->device
75841281cf3SAxel Dörfler 			&& item->entryRef.directory == ref->directory) {
75941281cf3SAxel Dörfler 			item->entryRef.directory = toDirectory;
76041281cf3SAxel Dörfler 			break;
76141281cf3SAxel Dörfler 		}
76241281cf3SAxel Dörfler 	}
76341281cf3SAxel Dörfler }
76441281cf3SAxel Dörfler 
76541281cf3SAxel Dörfler #endif // add-on support
76641281cf3SAxel Dörfler 
76741281cf3SAxel Dörfler //	external add-on API routines
76841281cf3SAxel Dörfler //	called using the new BDeskbar class
76941281cf3SAxel Dörfler 
77041281cf3SAxel Dörfler //	existence of icon/replicant by name or ID
77141281cf3SAxel Dörfler //	returns opposite
77241281cf3SAxel Dörfler //	note: name and id are semi-private limiting
77341281cf3SAxel Dörfler //		the ability of non-host apps to remove
77441281cf3SAxel Dörfler //		icons without a little bit of work
77541281cf3SAxel Dörfler 
77641281cf3SAxel Dörfler /**	for a specific id
77741281cf3SAxel Dörfler  *	return the name of the replicant (name of view)
77841281cf3SAxel Dörfler  */
77941281cf3SAxel Dörfler 
78041281cf3SAxel Dörfler status_t
78141281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 id, const char** name)
78241281cf3SAxel Dörfler {
78341281cf3SAxel Dörfler 	if (id < 0)
78441281cf3SAxel Dörfler 		return B_ERROR;
78541281cf3SAxel Dörfler 
78641281cf3SAxel Dörfler 	int32 index, temp;
78741281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &temp, id, false);
78841281cf3SAxel Dörfler 	if (view) {
78941281cf3SAxel Dörfler 		*name = view->Name();
79041281cf3SAxel Dörfler 		return B_OK;
79141281cf3SAxel Dörfler 	}
79241281cf3SAxel Dörfler 
79341281cf3SAxel Dörfler 	return B_ERROR;
79441281cf3SAxel Dörfler }
79541281cf3SAxel Dörfler 
79641281cf3SAxel Dörfler 
7977da06231SAxel Dörfler /**	for a specific name
7987da06231SAxel Dörfler  *	return the id (internal to Deskbar)
7997da06231SAxel Dörfler  */
8007da06231SAxel Dörfler 
80141281cf3SAxel Dörfler status_t
80241281cf3SAxel Dörfler TReplicantTray::ItemInfo(const char* name, int32* id)
80341281cf3SAxel Dörfler {
80441281cf3SAxel Dörfler 	if (!name || strlen(name) <= 0)
80541281cf3SAxel Dörfler 		return B_ERROR;
80641281cf3SAxel Dörfler 
80741281cf3SAxel Dörfler 	int32 index;
80841281cf3SAxel Dörfler 	BView* view = ViewAt(&index, id, name);
80941281cf3SAxel Dörfler 	if (view)
81041281cf3SAxel Dörfler 		return B_OK;
81141281cf3SAxel Dörfler 
81241281cf3SAxel Dörfler 	return B_ERROR;
81341281cf3SAxel Dörfler }
81441281cf3SAxel Dörfler 
81541281cf3SAxel Dörfler 
8167da06231SAxel Dörfler /**	at a specific index
8177da06231SAxel Dörfler  *	return both the name and the id of the replicant
8187da06231SAxel Dörfler  */
8197da06231SAxel Dörfler 
82041281cf3SAxel Dörfler status_t
82141281cf3SAxel Dörfler TReplicantTray::ItemInfo(int32 index, const char** name, int32* id)
82241281cf3SAxel Dörfler {
82341281cf3SAxel Dörfler 	if (index < 0)
82441281cf3SAxel Dörfler 		return B_ERROR;
82541281cf3SAxel Dörfler 
82641281cf3SAxel Dörfler 	BView* view;
82741281cf3SAxel Dörfler 	fShelf->ReplicantAt(index, &view, (uint32*)id, NULL);
82841281cf3SAxel Dörfler 	if (view) {
82941281cf3SAxel Dörfler 		*name = view->Name();
83041281cf3SAxel Dörfler 		return B_OK;
83141281cf3SAxel Dörfler 	}
83241281cf3SAxel Dörfler 
83341281cf3SAxel Dörfler 	return B_ERROR;
83441281cf3SAxel Dörfler }
83541281cf3SAxel Dörfler 
83641281cf3SAxel Dörfler 
8377da06231SAxel Dörfler /**	replicant exists, by id/index */
8387da06231SAxel Dörfler 
83941281cf3SAxel Dörfler bool
84041281cf3SAxel Dörfler TReplicantTray::IconExists(int32 target, bool byIndex)
84141281cf3SAxel Dörfler {
84241281cf3SAxel Dörfler 	int32 index, id;
84341281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
84441281cf3SAxel Dörfler 
84541281cf3SAxel Dörfler 	return view && index >= 0;
84641281cf3SAxel Dörfler }
84741281cf3SAxel Dörfler 
84841281cf3SAxel Dörfler 
8497da06231SAxel Dörfler /**	replicant exists, by name */
8507da06231SAxel Dörfler 
85141281cf3SAxel Dörfler bool
85241281cf3SAxel Dörfler TReplicantTray::IconExists(const char* name)
85341281cf3SAxel Dörfler {
85441281cf3SAxel Dörfler 	if (!name || strlen(name) == 0)
85541281cf3SAxel Dörfler 		return false;
85641281cf3SAxel Dörfler 
85741281cf3SAxel Dörfler 	int32 index, id;
85841281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
85941281cf3SAxel Dörfler 
86041281cf3SAxel Dörfler 	return view && index >= 0;
86141281cf3SAxel Dörfler }
86241281cf3SAxel Dörfler 
86341281cf3SAxel Dörfler 
86441281cf3SAxel Dörfler int32
86541281cf3SAxel Dörfler TReplicantTray::IconCount() const
86641281cf3SAxel Dörfler {
86741281cf3SAxel Dörfler 	return fShelf->CountReplicants();
86841281cf3SAxel Dörfler }
86941281cf3SAxel Dörfler 
87041281cf3SAxel Dörfler 
871ced3755cSAxel Dörfler /*! Message must contain an archivable view for later rehydration.
872ced3755cSAxel Dörfler 	This function takes over ownership of the provided message on success
873ced3755cSAxel Dörfler 	only.
874ced3755cSAxel Dörfler 	Returns the current replicant ID.
8757da06231SAxel Dörfler */
87641281cf3SAxel Dörfler status_t
877ced3755cSAxel Dörfler TReplicantTray::AddIcon(BMessage* archive, int32* id, const entry_ref* addOn)
87841281cf3SAxel Dörfler {
879ced3755cSAxel Dörfler 	if (archive == NULL || id == NULL)
88041281cf3SAxel Dörfler 		return B_ERROR;
88141281cf3SAxel Dörfler 
8829c50f36eSAxel Dörfler 	// find entry_ref
8839c50f36eSAxel Dörfler 
8849c50f36eSAxel Dörfler 	entry_ref ref;
8859c50f36eSAxel Dörfler 	if (addOn) {
8869c50f36eSAxel Dörfler 		// Use it if we got it
8879c50f36eSAxel Dörfler 		ref = *addOn;
8889c50f36eSAxel Dörfler 	} else {
8899c50f36eSAxel Dörfler 		const char* signature;
8901ad8c760SFredrik Holmqvist 
891ced3755cSAxel Dörfler 		status_t status = archive->FindString("add_on", &signature);
8929c50f36eSAxel Dörfler 		if (status == B_OK) {
8939c50f36eSAxel Dörfler 			BRoster roster;
8949c50f36eSAxel Dörfler 			status = roster.FindApp(signature, &ref);
8959c50f36eSAxel Dörfler 		}
8969c50f36eSAxel Dörfler 		if (status < B_OK)
8979c50f36eSAxel Dörfler 			return status;
8989c50f36eSAxel Dörfler 	}
8999c50f36eSAxel Dörfler 
9009c50f36eSAxel Dörfler 	BFile file;
9019c50f36eSAxel Dörfler 	status_t status = file.SetTo(&ref, B_READ_ONLY);
9029c50f36eSAxel Dörfler 	if (status < B_OK)
9039c50f36eSAxel Dörfler 		return status;
9049c50f36eSAxel Dörfler 
9059c50f36eSAxel Dörfler 	node_ref nodeRef;
9069c50f36eSAxel Dörfler 	status = file.GetNodeRef(&nodeRef);
9079c50f36eSAxel Dörfler 	if (status < B_OK)
9089c50f36eSAxel Dörfler 		return status;
9099c50f36eSAxel Dörfler 
9109c50f36eSAxel Dörfler 	BEntry entry(&ref, true);
9111ad8c760SFredrik Holmqvist 		// TODO: this resolves an eventual link for the item being added - this
9121ad8c760SFredrik Holmqvist 		// is okay for now, but in multi-user environments, one might want to
9131ad8c760SFredrik Holmqvist 		// have links that carry the be:deskbar_item_status attribute
9149c50f36eSAxel Dörfler 	status = entry.InitCheck();
9159c50f36eSAxel Dörfler 	if (status != B_OK)
9169c50f36eSAxel Dörfler 		return status;
9179c50f36eSAxel Dörfler 
91841281cf3SAxel Dörfler 	*id = 999;
919ced3755cSAxel Dörfler 	if (archive->what == B_ARCHIVED_OBJECT)
920ced3755cSAxel Dörfler 		archive->what = 0;
92141281cf3SAxel Dörfler 
922ced3755cSAxel Dörfler 	BRect originalBounds = archive->FindRect("_frame");
9231ad8c760SFredrik Holmqvist 		// this is a work-around for buggy replicants that change their size in
9241ad8c760SFredrik Holmqvist 		// AttachedToWindow() (such as "SVM")
925a5210ab1SAxel Dörfler 
9261ad8c760SFredrik Holmqvist 	// TODO: check for name collisions?
927ced3755cSAxel Dörfler 	status = fShelf->AddReplicant(archive, BPoint(1, 1));
9289c50f36eSAxel Dörfler 	if (status != B_OK)
9299c50f36eSAxel Dörfler 		return status;
93041281cf3SAxel Dörfler 
931a5210ab1SAxel Dörfler 	int32 count = fShelf->CountReplicants();
932a5210ab1SAxel Dörfler 	BView* view;
933a5210ab1SAxel Dörfler 	fShelf->ReplicantAt(count - 1, &view, (uint32*)id, NULL);
934a5210ab1SAxel Dörfler 
935a5210ab1SAxel Dörfler 	if (originalBounds != view->Bounds()) {
936a5210ab1SAxel Dörfler 		// The replicant changed its size when added to the window, so we need
93771bd3ba5SJonas Sundström 		// to recompute all over again (it's already done once via
93871bd3ba5SJonas Sundström 		// BShelf::AddReplicant() and TReplicantShelf::CanAcceptReplicantView())
939a5210ab1SAxel Dörfler 		RealignReplicants();
940a5210ab1SAxel Dörfler 	}
941a5210ab1SAxel Dörfler 
94241281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
94341281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
94441281cf3SAxel Dörfler 	float width, height;
94541281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
94641281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height)
94741281cf3SAxel Dörfler 		AdjustPlacement();
94841281cf3SAxel Dörfler 
94941281cf3SAxel Dörfler 	// add the item to the add-on list
95041281cf3SAxel Dörfler 
95141281cf3SAxel Dörfler 	AddItem(*id, nodeRef, entry, addOn != NULL);
95241281cf3SAxel Dörfler 	return B_OK;
95341281cf3SAxel Dörfler }
95441281cf3SAxel Dörfler 
95541281cf3SAxel Dörfler 
95641281cf3SAxel Dörfler void
95741281cf3SAxel Dörfler TReplicantTray::RemoveIcon(int32 target, bool byIndex)
95841281cf3SAxel Dörfler {
95941281cf3SAxel Dörfler 	if (target < 0)
96041281cf3SAxel Dörfler 		return;
96141281cf3SAxel Dörfler 
96241281cf3SAxel Dörfler 	int32 index, id;
96341281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
96441281cf3SAxel Dörfler 	if (view && index >= 0) {
9657da06231SAxel Dörfler 		// remove the reference from the item list & the shelf
96641281cf3SAxel Dörfler 		RemoveItem(id);
96741281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9687da06231SAxel Dörfler 
96941281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
97041281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
97141281cf3SAxel Dörfler 	}
97241281cf3SAxel Dörfler }
97341281cf3SAxel Dörfler 
97441281cf3SAxel Dörfler 
97541281cf3SAxel Dörfler void
97641281cf3SAxel Dörfler TReplicantTray::RemoveIcon(const char* name)
97741281cf3SAxel Dörfler {
97841281cf3SAxel Dörfler 	if (!name || strlen(name) <= 0)
97941281cf3SAxel Dörfler 		return;
98041281cf3SAxel Dörfler 
98141281cf3SAxel Dörfler 	int32 id, index;
98241281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
98341281cf3SAxel Dörfler 	if (view && index >= 0) {
9847da06231SAxel Dörfler 		// remove the reference from the item list & shelf
98541281cf3SAxel Dörfler 		RemoveItem(id);
98641281cf3SAxel Dörfler 		fShelf->DeleteReplicant(index);
9877da06231SAxel Dörfler 
98841281cf3SAxel Dörfler 		// force a placement update,  !! need to fix BShelf
98941281cf3SAxel Dörfler 		RealReplicantAdjustment(index);
99041281cf3SAxel Dörfler 	}
99141281cf3SAxel Dörfler }
99241281cf3SAxel Dörfler 
99341281cf3SAxel Dörfler 
99441281cf3SAxel Dörfler void
9957da06231SAxel Dörfler TReplicantTray::RealReplicantAdjustment(int32 startIndex)
99641281cf3SAxel Dörfler {
9977da06231SAxel Dörfler 	if (startIndex < 0)
99841281cf3SAxel Dörfler 		return;
9997da06231SAxel Dörfler 
10007da06231SAxel Dörfler 	if (startIndex == fLastReplicant)
10017da06231SAxel Dörfler 		startIndex = 0;
10027da06231SAxel Dörfler 
100341281cf3SAxel Dörfler 	// reset the locations of all replicants after the one deleted
10047da06231SAxel Dörfler 	RealignReplicants(startIndex);
100541281cf3SAxel Dörfler 
100641281cf3SAxel Dörfler 	float oldWidth = Bounds().Width();
100741281cf3SAxel Dörfler 	float oldHeight = Bounds().Height();
100841281cf3SAxel Dörfler 	float width, height;
100941281cf3SAxel Dörfler 	GetPreferredSize(&width, &height);
101041281cf3SAxel Dörfler 	if (oldWidth != width || oldHeight != height) {
10111ad8c760SFredrik Holmqvist 		// resize view to accomodate the replicants, redraw as necessary
101241281cf3SAxel Dörfler 		AdjustPlacement();
101341281cf3SAxel Dörfler 	}
101441281cf3SAxel Dörfler }
101541281cf3SAxel Dörfler 
101641281cf3SAxel Dörfler 
10177da06231SAxel Dörfler /**	looking for a replicant by id/index
10187da06231SAxel Dörfler  *	return the view and index
10197da06231SAxel Dörfler  */
10207da06231SAxel Dörfler 
102141281cf3SAxel Dörfler BView*
102241281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, int32 target, bool byIndex)
102341281cf3SAxel Dörfler {
102441281cf3SAxel Dörfler 	*index = -1;
102541281cf3SAxel Dörfler 
102641281cf3SAxel Dörfler 	BView* view;
102741281cf3SAxel Dörfler 	if (byIndex) {
102841281cf3SAxel Dörfler 		if (fShelf->ReplicantAt(target, &view, (uint32*)id)) {
102941281cf3SAxel Dörfler 			if (view) {
103041281cf3SAxel Dörfler 				*index = target;
103141281cf3SAxel Dörfler 				return view;
103241281cf3SAxel Dörfler 			}
103341281cf3SAxel Dörfler 		}
103441281cf3SAxel Dörfler 	} else {
103541281cf3SAxel Dörfler 		int32 count = fShelf->CountReplicants() - 1;
103641281cf3SAxel Dörfler 		int32 localid;
103741281cf3SAxel Dörfler 		for (int32 repIndex = count ; repIndex >= 0 ; repIndex--) {
103841281cf3SAxel Dörfler 			fShelf->ReplicantAt(repIndex, &view, (uint32*)&localid);
103941281cf3SAxel Dörfler 			if (localid == target && view) {
104041281cf3SAxel Dörfler 				*index = repIndex;
104141281cf3SAxel Dörfler 				*id = localid;
104241281cf3SAxel Dörfler 				return view;
104341281cf3SAxel Dörfler 			}
104441281cf3SAxel Dörfler 		}
104541281cf3SAxel Dörfler 	}
104641281cf3SAxel Dörfler 	return NULL;
104741281cf3SAxel Dörfler }
104841281cf3SAxel Dörfler 
104941281cf3SAxel Dörfler 
10507da06231SAxel Dörfler /**	looking for a replicant with a view by name
10517da06231SAxel Dörfler  *	return the view, index and the id of the replicant
10527da06231SAxel Dörfler  */
10537da06231SAxel Dörfler 
105441281cf3SAxel Dörfler BView*
105541281cf3SAxel Dörfler TReplicantTray::ViewAt(int32* index, int32* id, const char* name)
105641281cf3SAxel Dörfler {
105741281cf3SAxel Dörfler 	*index = -1;
105841281cf3SAxel Dörfler 	*id = -1;
105941281cf3SAxel Dörfler 
106041281cf3SAxel Dörfler 	BView* view;
106141281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants()-1;
106241281cf3SAxel Dörfler 	for (int32 repIndex = count ; repIndex >= 0 ; repIndex--) {
106341281cf3SAxel Dörfler 		fShelf->ReplicantAt(repIndex, &view, (uint32*)id);
106441281cf3SAxel Dörfler 		if (view && view->Name() && strcmp(name, view->Name()) == 0) {
106541281cf3SAxel Dörfler 			*index = repIndex;
106641281cf3SAxel Dörfler 			return view;
106741281cf3SAxel Dörfler 		}
106841281cf3SAxel Dörfler 	}
106941281cf3SAxel Dörfler 	return NULL;
107041281cf3SAxel Dörfler }
107141281cf3SAxel Dörfler 
107241281cf3SAxel Dörfler 
10737da06231SAxel Dörfler /**	Shelf will call to determine where and if
10747da06231SAxel Dörfler  *	the replicant is to be added
10757da06231SAxel Dörfler  */
10767da06231SAxel Dörfler 
107741281cf3SAxel Dörfler bool
107841281cf3SAxel Dörfler TReplicantTray::AcceptAddon(BRect replicantFrame, BMessage* message)
107941281cf3SAxel Dörfler {
108041281cf3SAxel Dörfler 	if (!message)
108141281cf3SAxel Dörfler 		return false;
108241281cf3SAxel Dörfler 
108341281cf3SAxel Dörfler 	if (replicantFrame.Height() > kMaxReplicantHeight)
108441281cf3SAxel Dörfler 		return false;
108541281cf3SAxel Dörfler 
108641281cf3SAxel Dörfler 	alignment align = B_ALIGN_LEFT;
108741281cf3SAxel Dörfler 	if (fAlignmentSupport && message->HasBool("deskbar:dynamic_align")) {
108841281cf3SAxel Dörfler 		if (!fBarView->Vertical())
108941281cf3SAxel Dörfler 			align = B_ALIGN_RIGHT;
109041281cf3SAxel Dörfler 		else
109141281cf3SAxel Dörfler 			align = fBarView->Left() ? B_ALIGN_LEFT : B_ALIGN_RIGHT;
109241281cf3SAxel Dörfler 	} else if (message->HasInt32("deskbar:align"))
109341281cf3SAxel Dörfler 		message->FindInt32("deskbar:align", (int32*)&align);
109441281cf3SAxel Dörfler 
109541281cf3SAxel Dörfler 	if (message->HasInt32("deskbar:private_align"))
109641281cf3SAxel Dörfler 		message->FindInt32("deskbar:private_align", (int32*)&align);
109741281cf3SAxel Dörfler 	else
109841281cf3SAxel Dörfler 		align = B_ALIGN_LEFT;
109941281cf3SAxel Dörfler 
11007da06231SAxel Dörfler 	BPoint loc = LocationForReplicant(fShelf->CountReplicants(),
11017da06231SAxel Dörfler 		replicantFrame.Width());
110241281cf3SAxel Dörfler 
110341281cf3SAxel Dörfler 	message->AddPoint("_pjp_loc", loc);
110441281cf3SAxel Dörfler 	return true;
110541281cf3SAxel Dörfler }
110641281cf3SAxel Dörfler 
110741281cf3SAxel Dörfler 
110841281cf3SAxel Dörfler /**	based on the previous (index - 1) replicant in the list
110941281cf3SAxel Dörfler  *	calculate where the left point should be for this
111041281cf3SAxel Dörfler  *	replicant.  replicant will flow to the right on its own
111141281cf3SAxel Dörfler  */
111241281cf3SAxel Dörfler 
111341281cf3SAxel Dörfler BPoint
11147da06231SAxel Dörfler TReplicantTray::LocationForReplicant(int32 index, float width)
111541281cf3SAxel Dörfler {
111641281cf3SAxel Dörfler 	BPoint loc(kIconGap + 1, kGutter + 1);
111741281cf3SAxel Dörfler 
11187da06231SAxel Dörfler 	if (fMultiRowMode) {
11197da06231SAxel Dörfler 		// try to find free space in every row
11207da06231SAxel Dörfler 		for (int32 row = 0; ; loc.y += kMaxReplicantHeight + kIconGap, row++) {
11217da06231SAxel Dörfler 			// determine free space in this row
11222ce9bab8SJohn Scipione 			BRect rect(loc.x, loc.y, loc.x + fMinimumTrayWidth - kIconGap
11232ce9bab8SJohn Scipione 				- 2.0, loc.y + kMaxReplicantHeight);
1124*573f748cSJohn Scipione 			if (row == 0 && !fTime->IsHidden())
1125*573f748cSJohn Scipione 				rect.right -= fTime->Frame().Width() + kIconGap;
11267da06231SAxel Dörfler 
11277da06231SAxel Dörfler 			for (int32 i = 0; i < index; i++) {
11287da06231SAxel Dörfler 				BView* view = NULL;
11297da06231SAxel Dörfler 				fShelf->ReplicantAt(i, &view);
11307da06231SAxel Dörfler 				if (view == NULL || view->Frame().top != rect.top)
11317da06231SAxel Dörfler 					continue;
11327da06231SAxel Dörfler 
11337da06231SAxel Dörfler 				rect.left = view->Frame().right + kIconGap + 1;
11347da06231SAxel Dörfler 			}
11357da06231SAxel Dörfler 
11367da06231SAxel Dörfler 			if (rect.Width() >= width) {
11377da06231SAxel Dörfler 				// the icon fits in this row
11387da06231SAxel Dörfler 				loc = rect.LeftTop();
11397da06231SAxel Dörfler 				break;
11407da06231SAxel Dörfler 			}
11417da06231SAxel Dörfler 		}
11427da06231SAxel Dörfler 	} else {
114341281cf3SAxel Dörfler 		if (index > 0) {
114441281cf3SAxel Dörfler 			// get the last replicant added for placement reference
114541281cf3SAxel Dörfler 			BView* view = NULL;
11467da06231SAxel Dörfler 			fShelf->ReplicantAt(index - 1, &view);
114741281cf3SAxel Dörfler 			if (view) {
114841281cf3SAxel Dörfler 				// push this rep placement past the last one
114941281cf3SAxel Dörfler 				loc.x = view->Frame().right + kIconGap + 1;
115041281cf3SAxel Dörfler 				loc.y = view->Frame().top;
115141281cf3SAxel Dörfler 			}
115241281cf3SAxel Dörfler 		}
11537da06231SAxel Dörfler 	}
115441281cf3SAxel Dörfler 
115571bd3ba5SJonas Sundström 	if ((loc.y == fRightBottomReplicant.top && loc.x
115671bd3ba5SJonas Sundström 		> fRightBottomReplicant.left) || loc.y > fRightBottomReplicant.top) {
115771bd3ba5SJonas Sundström 		fRightBottomReplicant.Set(loc.x, loc.y, loc.x + width, loc.y
115871bd3ba5SJonas Sundström 		+ kMaxReplicantHeight);
11597da06231SAxel Dörfler 		fLastReplicant = index;
116041281cf3SAxel Dörfler 	}
11617da06231SAxel Dörfler 
116241281cf3SAxel Dörfler 	return loc;
116341281cf3SAxel Dörfler }
116441281cf3SAxel Dörfler 
116541281cf3SAxel Dörfler 
116641281cf3SAxel Dörfler BRect
116741281cf3SAxel Dörfler TReplicantTray::IconFrame(int32 target, bool byIndex)
116841281cf3SAxel Dörfler {
116941281cf3SAxel Dörfler 	int32 index, id;
117041281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, target, byIndex);
117141281cf3SAxel Dörfler 	if (view)
117241281cf3SAxel Dörfler 		return view->Frame();
117341281cf3SAxel Dörfler 
117441281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
117541281cf3SAxel Dörfler }
117641281cf3SAxel Dörfler 
117741281cf3SAxel Dörfler 
117841281cf3SAxel Dörfler BRect
117941281cf3SAxel Dörfler TReplicantTray::IconFrame(const char* name)
118041281cf3SAxel Dörfler {
118141281cf3SAxel Dörfler 	if (!name)
118241281cf3SAxel Dörfler 		return BRect(0, 0, 0, 0);
118341281cf3SAxel Dörfler 
118441281cf3SAxel Dörfler 	int32 id, index;
118541281cf3SAxel Dörfler 	BView* view = ViewAt(&index, &id, name);
118641281cf3SAxel Dörfler 	if (view)
118741281cf3SAxel Dörfler 		return view->Frame();
118841281cf3SAxel Dörfler 
118941281cf3SAxel Dörfler 	return BRect(0, 0, 0, 0);
119041281cf3SAxel Dörfler }
119141281cf3SAxel Dörfler 
119241281cf3SAxel Dörfler 
11937da06231SAxel Dörfler /**	Scan from the startIndex and reset the location
11947da06231SAxel Dörfler  *	as defined in LocationForReplicant()
119541281cf3SAxel Dörfler  */
119641281cf3SAxel Dörfler 
119741281cf3SAxel Dörfler void
119841281cf3SAxel Dörfler TReplicantTray::RealignReplicants(int32 startIndex)
119941281cf3SAxel Dörfler {
120041281cf3SAxel Dörfler 	if (startIndex < 0)
120141281cf3SAxel Dörfler 		startIndex = 0;
120241281cf3SAxel Dörfler 
120341281cf3SAxel Dörfler 	int32 count = fShelf->CountReplicants();
120441281cf3SAxel Dörfler 	if (count <= 0)
120541281cf3SAxel Dörfler 		return;
120641281cf3SAxel Dörfler 
12077da06231SAxel Dörfler 	if (startIndex == 0)
12087da06231SAxel Dörfler 		fRightBottomReplicant.Set(0, 0, 0, 0);
12097da06231SAxel Dörfler 
121041281cf3SAxel Dörfler 	BView* view = NULL;
121141281cf3SAxel Dörfler 	for (int32 i = startIndex ; i < count ; i++) {
121241281cf3SAxel Dörfler 		fShelf->ReplicantAt(i, &view);
12130bec7100SStefano Ceccherini 		if (view != NULL) {
12147da06231SAxel Dörfler 			BPoint loc = LocationForReplicant(i, view->Frame().Width());
12150bec7100SStefano Ceccherini 			if (view->Frame().LeftTop() != loc)
121641281cf3SAxel Dörfler 				view->MoveTo(loc);
121741281cf3SAxel Dörfler 		}
121841281cf3SAxel Dörfler 	}
121941281cf3SAxel Dörfler }
122041281cf3SAxel Dörfler 
122141281cf3SAxel Dörfler 
122241281cf3SAxel Dörfler void
122341281cf3SAxel Dörfler TReplicantTray::SetMultiRow(bool state)
122441281cf3SAxel Dörfler {
122541281cf3SAxel Dörfler 	fMultiRowMode = state;
122641281cf3SAxel Dörfler }
122741281cf3SAxel Dörfler 
122841281cf3SAxel Dörfler 
12298aab28f1SRene Gollent status_t
12307625ce51SRene Gollent TReplicantTray::_SaveSettings()
12317625ce51SRene Gollent {
12327625ce51SRene Gollent 	status_t result;
12337625ce51SRene Gollent 	BPath path;
12347625ce51SRene Gollent 	if ((result = find_directory(B_USER_SETTINGS_DIRECTORY, &path, true))
12357625ce51SRene Gollent 		 == B_OK) {
12367625ce51SRene Gollent 		path.Append(kReplicantSettingsFile);
12377625ce51SRene Gollent 
12387625ce51SRene Gollent 		BFile file(path.Path(), B_READ_WRITE | B_CREATE_FILE | B_ERASE_FILE);
12397625ce51SRene Gollent 		if ((result = file.InitCheck()) == B_OK)
12407625ce51SRene Gollent 			result = fAddOnSettings.Flatten(&file);
12417625ce51SRene Gollent 	}
12427625ce51SRene Gollent 
12437625ce51SRene Gollent 	return result;
12447625ce51SRene Gollent }
12457625ce51SRene Gollent 
12467625ce51SRene Gollent 
124741281cf3SAxel Dörfler //	#pragma mark -
124841281cf3SAxel Dörfler 
124941281cf3SAxel Dörfler 
1250ca9acc20SAxel Dörfler /*! Draggable region that is asynchronous so that dragging does not block
1251ca9acc20SAxel Dörfler 	other activities.
12527da06231SAxel Dörfler */
125341281cf3SAxel Dörfler TDragRegion::TDragRegion(TBarView* parent, BView* child)
1254ca9acc20SAxel Dörfler 	:
1255ca9acc20SAxel Dörfler 	BControl(BRect(0, 0, 0, 0), "", "", NULL, B_FOLLOW_NONE,
125641281cf3SAxel Dörfler 		B_WILL_DRAW | B_FRAME_EVENTS),
125741281cf3SAxel Dörfler 	fBarView(parent),
125841281cf3SAxel Dörfler 	fChild(child),
125941281cf3SAxel Dörfler 	fDragLocation(kAutoPlaceDragRegion)
126041281cf3SAxel Dörfler {
126141281cf3SAxel Dörfler }
126241281cf3SAxel Dörfler 
126341281cf3SAxel Dörfler 
126441281cf3SAxel Dörfler void
126541281cf3SAxel Dörfler TDragRegion::AttachedToWindow()
126641281cf3SAxel Dörfler {
126741281cf3SAxel Dörfler 	BView::AttachedToWindow();
1268cb6afcb1SStephan Aßmus 	if (be_control_look != NULL)
1269cb6afcb1SStephan Aßmus 		SetViewColor(tint_color(ui_color(B_MENU_BACKGROUND_COLOR), 1.1));
1270cb6afcb1SStephan Aßmus 	else
127141281cf3SAxel Dörfler 		SetViewColor(ui_color(B_MENU_BACKGROUND_COLOR));
127241281cf3SAxel Dörfler 	ResizeToPreferred();
127341281cf3SAxel Dörfler }
127441281cf3SAxel Dörfler 
127541281cf3SAxel Dörfler 
127641281cf3SAxel Dörfler void
127741281cf3SAxel Dörfler TDragRegion::GetPreferredSize(float* width, float* height)
127841281cf3SAxel Dörfler {
127941281cf3SAxel Dörfler 	fChild->ResizeToPreferred();
128041281cf3SAxel Dörfler 	*width = fChild->Bounds().Width();
128141281cf3SAxel Dörfler 	*height = fChild->Bounds().Height();
128241281cf3SAxel Dörfler 
128341281cf3SAxel Dörfler 	if (fDragLocation != kNoDragRegion)
128441281cf3SAxel Dörfler 		*width += 7;
128541281cf3SAxel Dörfler 	else
128641281cf3SAxel Dörfler 		*width += 6;
128741281cf3SAxel Dörfler 
128841281cf3SAxel Dörfler 	*height += 3;
128941281cf3SAxel Dörfler }
129041281cf3SAxel Dörfler 
129141281cf3SAxel Dörfler 
129241281cf3SAxel Dörfler void
129341281cf3SAxel Dörfler TDragRegion::FrameMoved(BPoint)
129441281cf3SAxel Dörfler {
1295ca9acc20SAxel Dörfler 	if (fBarView->Left() && fBarView->Vertical()
1296ca9acc20SAxel Dörfler 		&& fDragLocation != kNoDragRegion)
129741281cf3SAxel Dörfler 		fChild->MoveTo(5, 2);
129841281cf3SAxel Dörfler 	else
129941281cf3SAxel Dörfler 		fChild->MoveTo(2, 2);
130041281cf3SAxel Dörfler }
130141281cf3SAxel Dörfler 
130241281cf3SAxel Dörfler 
130341281cf3SAxel Dörfler void
130441281cf3SAxel Dörfler TDragRegion::Draw(BRect)
130541281cf3SAxel Dörfler {
1306eaa9af99SStephan Aßmus 	rgb_color menuColor = ViewColor();
130741281cf3SAxel Dörfler 	rgb_color hilite = tint_color(menuColor, B_DARKEN_1_TINT);
1308cb6afcb1SStephan Aßmus 	rgb_color ldark = tint_color(menuColor, 1.02);
1309cb6afcb1SStephan Aßmus 	rgb_color dark = tint_color(menuColor, B_DARKEN_2_TINT);
131041281cf3SAxel Dörfler 	rgb_color vvdark = tint_color(menuColor, B_DARKEN_4_TINT);
131141281cf3SAxel Dörfler 	rgb_color light = tint_color(menuColor, B_LIGHTEN_2_TINT);
131241281cf3SAxel Dörfler 
131341281cf3SAxel Dörfler 	BRect frame(Bounds());
131441281cf3SAxel Dörfler 	BeginLineArray(4);
131541281cf3SAxel Dörfler 
1316cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1317cb6afcb1SStephan Aßmus 		if (fBarView->Vertical()) {
1318cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(), frame.RightTop(), dark);
1319cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1320cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.top + 1), ldark);
1321cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
1322cb6afcb1SStephan Aßmus 		} else if (fBarView->AcrossTop() || fBarView->AcrossBottom()) {
1323cb6afcb1SStephan Aßmus 			AddLine(frame.LeftTop(),
1324cb6afcb1SStephan Aßmus 				BPoint(frame.left, frame.bottom), dark);
1325cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.top + 1),
1326cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
1327cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right, frame.top + 2),
1328cb6afcb1SStephan Aßmus 				BPoint(frame.right, frame.bottom), hilite);
1329cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left + 1, frame.bottom),
1330cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
1331cb6afcb1SStephan Aßmus 		}
1332cb6afcb1SStephan Aßmus 	} else {
133341281cf3SAxel Dörfler 		if (fBarView->Vertical()) {
133441281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.RightTop(), light);
133541281cf3SAxel Dörfler 			AddLine(frame.LeftTop(), frame.LeftBottom(), light);
133641281cf3SAxel Dörfler 			AddLine(frame.RightBottom(), frame.RightTop(), hilite);
133741281cf3SAxel Dörfler 		} else if (fBarView->AcrossTop()) {
1338cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1339cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
134041281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1341cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 2),
1342cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1343cb6afcb1SStephan Aßmus 			AddLine(frame.LeftBottom(),
1344cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom), hilite);
134541281cf3SAxel Dörfler 		} else if (fBarView->AcrossBottom()) {
1346cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.left, frame.top + 1),
1347cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.top + 1), light);
134841281cf3SAxel Dörfler 			AddLine(frame.LeftBottom(), frame.RightBottom(), hilite);
134941281cf3SAxel Dörfler 			AddLine(frame.RightTop(), frame.RightBottom(), vvdark);
1350cb6afcb1SStephan Aßmus 			AddLine(BPoint(frame.right - 1, frame.top + 1),
1351cb6afcb1SStephan Aßmus 				BPoint(frame.right - 1, frame.bottom - 1), hilite);
1352cb6afcb1SStephan Aßmus 		}
135341281cf3SAxel Dörfler 	}
135441281cf3SAxel Dörfler 
135541281cf3SAxel Dörfler 	EndLineArray();
135641281cf3SAxel Dörfler 
135741281cf3SAxel Dörfler 	if (fDragLocation != kDontDrawDragRegion || fDragLocation != kNoDragRegion)
135841281cf3SAxel Dörfler 		DrawDragRegion();
135941281cf3SAxel Dörfler }
136041281cf3SAxel Dörfler 
136141281cf3SAxel Dörfler 
136241281cf3SAxel Dörfler void
136341281cf3SAxel Dörfler TDragRegion::DrawDragRegion()
136441281cf3SAxel Dörfler {
136541281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
136641281cf3SAxel Dörfler 
1367cb6afcb1SStephan Aßmus 	rgb_color menuColor = ViewColor();
1368cb6afcb1SStephan Aßmus 	rgb_color menuHilite = menuColor;
136941281cf3SAxel Dörfler 	if (IsTracking()) {
1370cb6afcb1SStephan Aßmus 		// Draw drag region highlighted if tracking mouse
1371cb6afcb1SStephan Aßmus 		menuHilite = tint_color(menuColor, B_HIGHLIGHT_BACKGROUND_TINT);
137241281cf3SAxel Dörfler 		SetHighColor(menuHilite);
137341281cf3SAxel Dörfler 		FillRect(dragRegion);
137441281cf3SAxel Dörfler 	}
1375cb6afcb1SStephan Aßmus 	rgb_color vdark = tint_color(menuHilite, B_DARKEN_3_TINT);
1376cb6afcb1SStephan Aßmus 	rgb_color light = tint_color(menuHilite, B_LIGHTEN_2_TINT);
1377cb6afcb1SStephan Aßmus 
1378cb6afcb1SStephan Aßmus 	BeginLineArray(dragRegion.IntegerHeight());
1379cb6afcb1SStephan Aßmus 	BPoint pt;
1380cb6afcb1SStephan Aßmus 	pt.x = floorf((dragRegion.left + dragRegion.right) / 2 + 0.5) - 1;
1381cb6afcb1SStephan Aßmus 	pt.y = dragRegion.top + 2;
1382cb6afcb1SStephan Aßmus 
138341281cf3SAxel Dörfler 	while (pt.y + 1 <= dragRegion.bottom) {
138441281cf3SAxel Dörfler 		AddLine(pt, pt, vdark);
138541281cf3SAxel Dörfler 		AddLine(pt + BPoint(1, 1), pt + BPoint(1, 1), light);
138641281cf3SAxel Dörfler 
138741281cf3SAxel Dörfler 		pt.y += 3;
138841281cf3SAxel Dörfler 	}
138941281cf3SAxel Dörfler 	EndLineArray();
139041281cf3SAxel Dörfler }
139141281cf3SAxel Dörfler 
139241281cf3SAxel Dörfler 
139341281cf3SAxel Dörfler BRect
139441281cf3SAxel Dörfler TDragRegion::DragRegion() const
139541281cf3SAxel Dörfler {
1396cb6afcb1SStephan Aßmus 	float kTopBottomInset = 2;
1397cb6afcb1SStephan Aßmus 	float kLeftRightInset = 1;
1398cb6afcb1SStephan Aßmus 	float kDragWidth = 3;
1399cb6afcb1SStephan Aßmus 	if (be_control_look != NULL) {
1400cb6afcb1SStephan Aßmus 		kTopBottomInset = 1;
1401cb6afcb1SStephan Aßmus 		kLeftRightInset = 0;
1402cb6afcb1SStephan Aßmus 		kDragWidth = 4;
1403cb6afcb1SStephan Aßmus 	}
1404cb6afcb1SStephan Aßmus 
140541281cf3SAxel Dörfler 	BRect dragRegion(Bounds());
1406cb6afcb1SStephan Aßmus 	dragRegion.top += kTopBottomInset;
1407cb6afcb1SStephan Aßmus 	dragRegion.bottom -= kTopBottomInset;
140841281cf3SAxel Dörfler 
140941281cf3SAxel Dörfler 	bool placeOnLeft = false;
141041281cf3SAxel Dörfler 	if (fDragLocation == kAutoPlaceDragRegion) {
141141281cf3SAxel Dörfler 		if (fBarView->Vertical() && fBarView->Left())
141241281cf3SAxel Dörfler 			placeOnLeft = true;
141341281cf3SAxel Dörfler 		else
141441281cf3SAxel Dörfler 			placeOnLeft = false;
141541281cf3SAxel Dörfler 	} else if (fDragLocation == kDragRegionLeft)
141641281cf3SAxel Dörfler 		placeOnLeft = true;
141741281cf3SAxel Dörfler 	else if (fDragLocation == kDragRegionRight)
141841281cf3SAxel Dörfler 		placeOnLeft = false;
141941281cf3SAxel Dörfler 
142041281cf3SAxel Dörfler 	if (placeOnLeft) {
1421cb6afcb1SStephan Aßmus 		dragRegion.left += kLeftRightInset;
1422cb6afcb1SStephan Aßmus 		dragRegion.right = dragRegion.left + kDragWidth;
142341281cf3SAxel Dörfler 	} else {
1424cb6afcb1SStephan Aßmus 		dragRegion.right -= kLeftRightInset;
1425cb6afcb1SStephan Aßmus 		dragRegion.left = dragRegion.right - kDragWidth;
142641281cf3SAxel Dörfler 	}
142741281cf3SAxel Dörfler 
142841281cf3SAxel Dörfler 	return dragRegion;
142941281cf3SAxel Dörfler }
143041281cf3SAxel Dörfler 
143141281cf3SAxel Dörfler 
143241281cf3SAxel Dörfler void
143341281cf3SAxel Dörfler TDragRegion::MouseDown(BPoint thePoint)
143441281cf3SAxel Dörfler {
143541281cf3SAxel Dörfler 	ulong buttons;
143641281cf3SAxel Dörfler 	BPoint where;
143741281cf3SAxel Dörfler 	BRect dragRegion(DragRegion());
143841281cf3SAxel Dörfler 
14397da06231SAxel Dörfler 	dragRegion.InsetBy(-2.0f, -2.0f);
14407da06231SAxel Dörfler 		// DragRegion() is designed for drawing, not clicking
144141281cf3SAxel Dörfler 
144241281cf3SAxel Dörfler 	if (!dragRegion.Contains(thePoint))
144341281cf3SAxel Dörfler 		return;
144441281cf3SAxel Dörfler 
144541281cf3SAxel Dörfler 	while (true) {
144641281cf3SAxel Dörfler 		GetMouse(&where, &buttons);
144741281cf3SAxel Dörfler 		if (!buttons)
144841281cf3SAxel Dörfler 			break;
144941281cf3SAxel Dörfler 
145041281cf3SAxel Dörfler 		if ((Window()->Flags() & B_ASYNCHRONOUS_CONTROLS) != 0) {
145141281cf3SAxel Dörfler 			fPreviousPosition = thePoint;
145241281cf3SAxel Dörfler 			SetTracking(true);
145341281cf3SAxel Dörfler 			SetMouseEventMask(B_POINTER_EVENTS,
145441281cf3SAxel Dörfler 				B_NO_POINTER_HISTORY | B_LOCK_WINDOW_FOCUS);
145541281cf3SAxel Dörfler 			Invalidate(DragRegion());
145641281cf3SAxel Dörfler 			break;
145741281cf3SAxel Dörfler 		}
145841281cf3SAxel Dörfler 
145941281cf3SAxel Dörfler 		snooze(25000);
146041281cf3SAxel Dörfler 	}
146141281cf3SAxel Dörfler }
146241281cf3SAxel Dörfler 
146341281cf3SAxel Dörfler 
146441281cf3SAxel Dörfler void
146541281cf3SAxel Dörfler TDragRegion::MouseUp(BPoint pt)
146641281cf3SAxel Dörfler {
146741281cf3SAxel Dörfler 	if (IsTracking()) {
146841281cf3SAxel Dörfler 		SetTracking(false);
146941281cf3SAxel Dörfler 		Invalidate(DragRegion());
147041281cf3SAxel Dörfler 	} else
147141281cf3SAxel Dörfler 		BControl::MouseUp(pt);
147241281cf3SAxel Dörfler }
147341281cf3SAxel Dörfler 
147441281cf3SAxel Dörfler 
147541281cf3SAxel Dörfler bool
147641281cf3SAxel Dörfler TDragRegion::SwitchModeForRect(BPoint mouse, BRect rect,
147741281cf3SAxel Dörfler 	bool newVertical, bool newLeft, bool newTop, int32 newState)
147841281cf3SAxel Dörfler {
14791ad8c760SFredrik Holmqvist 	if (!rect.Contains(mouse)) {
148041281cf3SAxel Dörfler 		// not our rect
148141281cf3SAxel Dörfler 		return false;
14821ad8c760SFredrik Holmqvist 	}
148341281cf3SAxel Dörfler 
14841ad8c760SFredrik Holmqvist 	if (newVertical == fBarView->Vertical() && newLeft == fBarView->Left()
14851ad8c760SFredrik Holmqvist 		&& newTop == fBarView->Top() && newState == fBarView->State()) {
148641281cf3SAxel Dörfler 		// already in the correct mode
148741281cf3SAxel Dörfler 		return true;
14881ad8c760SFredrik Holmqvist 	}
148941281cf3SAxel Dörfler 
149041281cf3SAxel Dörfler 	fBarView->ChangeState(newState, newVertical, newLeft, newTop);
149141281cf3SAxel Dörfler 	return true;
149241281cf3SAxel Dörfler }
149341281cf3SAxel Dörfler 
149441281cf3SAxel Dörfler 
149541281cf3SAxel Dörfler void
149641281cf3SAxel Dörfler TDragRegion::MouseMoved(BPoint where, uint32 code, const BMessage* message)
149741281cf3SAxel Dörfler {
149841281cf3SAxel Dörfler 	if (IsTracking()) {
149941281cf3SAxel Dörfler 		BScreen screen;
150041281cf3SAxel Dörfler 		BRect frame = screen.Frame();
150141281cf3SAxel Dörfler 
150241281cf3SAxel Dörfler 		float hDivider = frame.Width() / 6;
150371bd3ba5SJonas Sundström 		hDivider = (hDivider < sMinimumWindowWidth + 10.0f)
150471bd3ba5SJonas Sundström 			? sMinimumWindowWidth + 10.0f : hDivider;
150541281cf3SAxel Dörfler 		float miniDivider = frame.top + kMiniHeight + 10.0f;
150641281cf3SAxel Dörfler 		float vDivider = frame.Height() / 2;
150741281cf3SAxel Dörfler #ifdef FULL_MODE
150841281cf3SAxel Dörfler 		float thirdScreen = frame.Height() / 3;
150941281cf3SAxel Dörfler #endif
151071bd3ba5SJonas Sundström 		BRect topLeft(frame.left, frame.top, frame.left + hDivider,
151171bd3ba5SJonas Sundström 			miniDivider);
151271bd3ba5SJonas Sundström 		BRect topMiddle(frame.left + hDivider, frame.top, frame.right
151371bd3ba5SJonas Sundström 			- hDivider, vDivider);
151471bd3ba5SJonas Sundström 		BRect topRight(frame.right - hDivider, frame.top, frame.right,
151571bd3ba5SJonas Sundström 			miniDivider);
151641281cf3SAxel Dörfler 
151741281cf3SAxel Dörfler #ifdef FULL_MODE
151841281cf3SAxel Dörfler 		vDivider = miniDivider + thirdScreen;
151941281cf3SAxel Dörfler #endif
152071bd3ba5SJonas Sundström 		BRect middleLeft(frame.left, miniDivider, frame.left + hDivider,
152171bd3ba5SJonas Sundström 			vDivider);
152271bd3ba5SJonas Sundström 		BRect middleRight(frame.right - hDivider, miniDivider, frame.right,
152371bd3ba5SJonas Sundström 			vDivider);
152441281cf3SAxel Dörfler 
152541281cf3SAxel Dörfler #ifdef FULL_MODE
152671bd3ba5SJonas Sundström 		BRect leftSide(frame.left, vDivider, frame.left + hDivider,
152771bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
152871bd3ba5SJonas Sundström 		BRect rightSide(frame.right - hDivider, vDivider, frame.right,
152971bd3ba5SJonas Sundström 			frame.bottom - thirdScreen);
153041281cf3SAxel Dörfler 
153141281cf3SAxel Dörfler 		vDivider = frame.bottom - thirdScreen;
153241281cf3SAxel Dörfler #endif
153371bd3ba5SJonas Sundström 		BRect bottomLeft(frame.left, vDivider, frame.left + hDivider,
153471bd3ba5SJonas Sundström 			frame.bottom);
153571bd3ba5SJonas Sundström 		BRect bottomMiddle(frame.left + hDivider, vDivider, frame.right
153671bd3ba5SJonas Sundström 			- hDivider, frame.bottom);
153771bd3ba5SJonas Sundström 		BRect bottomRight(frame.right - hDivider, vDivider, frame.right,
153871bd3ba5SJonas Sundström 			frame.bottom);
153941281cf3SAxel Dörfler 
154041281cf3SAxel Dörfler 		if (where != fPreviousPosition) {
154141281cf3SAxel Dörfler 			fPreviousPosition = where;
154241281cf3SAxel Dörfler 			ConvertToScreen(&where);
154341281cf3SAxel Dörfler 
154441281cf3SAxel Dörfler 			// use short circuit evaluation for convenience
154541281cf3SAxel Dörfler 			if (SwitchModeForRect(where, topLeft, true, true, true, kMiniState)
154671bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topMiddle, false, true, true,
154771bd3ba5SJonas Sundström 					kExpandoState)
154871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, topRight, true, false, true,
154971bd3ba5SJonas Sundström 					kMiniState)
155071bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleLeft, true, true, true,
155171bd3ba5SJonas Sundström 					kExpandoState)
155271bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, middleRight, true, false, true,
155371bd3ba5SJonas Sundström 					kExpandoState)
155441281cf3SAxel Dörfler 
155541281cf3SAxel Dörfler #ifdef FULL_MODE
155671bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, leftSide, true, true, true,
155771bd3ba5SJonas Sundström 					kFullState)
155871bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, rightSide, true, false, true,
155971bd3ba5SJonas Sundström 					kFullState)
156041281cf3SAxel Dörfler #endif
156171bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomLeft, true, true, false,
156271bd3ba5SJonas Sundström 					kMiniState)
156371bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomMiddle, false, true, false,
156471bd3ba5SJonas Sundström 					kExpandoState)
156571bd3ba5SJonas Sundström 				|| SwitchModeForRect(where, bottomRight, true, false, false,
156671bd3ba5SJonas Sundström 					kMiniState))
156741281cf3SAxel Dörfler 				;
156841281cf3SAxel Dörfler 		}
156941281cf3SAxel Dörfler 	} else
157041281cf3SAxel Dörfler 		BControl::MouseMoved(where, code, message);
157141281cf3SAxel Dörfler }
157241281cf3SAxel Dörfler 
157341281cf3SAxel Dörfler 
157441281cf3SAxel Dörfler int32
157541281cf3SAxel Dörfler TDragRegion::DragRegionLocation() const
157641281cf3SAxel Dörfler {
157741281cf3SAxel Dörfler 	return fDragLocation;
157841281cf3SAxel Dörfler }
157941281cf3SAxel Dörfler 
158041281cf3SAxel Dörfler 
158141281cf3SAxel Dörfler void
158241281cf3SAxel Dörfler TDragRegion::SetDragRegionLocation(int32 location)
158341281cf3SAxel Dörfler {
158441281cf3SAxel Dörfler 	if (location == fDragLocation)
158541281cf3SAxel Dörfler 		return;
158641281cf3SAxel Dörfler 
158741281cf3SAxel Dörfler 	fDragLocation = location;
158841281cf3SAxel Dörfler 	Invalidate();
158941281cf3SAxel Dörfler }
159041281cf3SAxel Dörfler 
1591