xref: /haiku/src/kits/tracker/ContainerWindow.cpp (revision eea5774f46bba925156498abf9cb1a1165647bf7)
1 /*
2 Open Tracker License
3 
4 Terms and Conditions
5 
6 Copyright (c) 1991-2000, Be Incorporated. All rights reserved.
7 
8 Permission is hereby granted, free of charge, to any person obtaining a copy of
9 this software and associated documentation files (the "Software"), to deal in
10 the Software without restriction, including without limitation the rights to
11 use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
12 of the Software, and to permit persons to whom the Software is furnished to do
13 so, subject to the following conditions:
14 
15 The above copyright notice and this permission notice applies to all licensees
16 and shall be included in all copies or substantial portions of the Software.
17 
18 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF TITLE, MERCHANTABILITY,
20 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
21 BE INCORPORATED BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
22 AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF, OR IN CONNECTION
23 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 
25 Except as contained in this notice, the name of Be Incorporated shall not be
26 used in advertising or otherwise to promote the sale, use or other dealings in
27 this Software without prior written authorization from Be Incorporated.
28 
29 Tracker(TM), Be(R), BeOS(R), and BeIA(TM) are trademarks or registered trademarks
30 of Be Incorporated in the United States and other countries. Other brand product
31 names are registered trademarks or trademarks of their respective holders.
32 All rights reserved.
33 */
34 
35 
36 #include "ContainerWindow.h"
37 
38 #include <Alert.h>
39 #include <Application.h>
40 #include <AppFileInfo.h>
41 #include <Catalog.h>
42 #include <ControlLook.h>
43 #include <Debug.h>
44 #include <Directory.h>
45 #include <Entry.h>
46 #include <FindDirectory.h>
47 #include <GridView.h>
48 #include <GroupLayout.h>
49 #include <Keymap.h>
50 #include <Locale.h>
51 #include <MenuItem.h>
52 #include <MenuBar.h>
53 #include <NodeMonitor.h>
54 #include <Path.h>
55 #include <PopUpMenu.h>
56 #include <Roster.h>
57 #include <Screen.h>
58 #include <UnicodeChar.h>
59 #include <Volume.h>
60 #include <VolumeRoster.h>
61 #include <WindowPrivate.h>
62 
63 #include <fs_attr.h>
64 #include <image.h>
65 #include <strings.h>
66 #include <stdlib.h>
67 
68 #include "Attributes.h"
69 #include "AttributeStream.h"
70 #include "AutoDeleter.h"
71 #include "AutoLock.h"
72 #include "BackgroundImage.h"
73 #include "Commands.h"
74 #include "CountView.h"
75 #include "DeskWindow.h"
76 #include "DraggableContainerIcon.h"
77 #include "FavoritesMenu.h"
78 #include "FindPanel.h"
79 #include "FSClipboard.h"
80 #include "FSUndoRedo.h"
81 #include "FSUtils.h"
82 #include "IconMenuItem.h"
83 #include "OpenWithWindow.h"
84 #include "MimeTypes.h"
85 #include "Model.h"
86 #include "MountMenu.h"
87 #include "Navigator.h"
88 #include "NavMenu.h"
89 #include "PoseView.h"
90 #include "QueryContainerWindow.h"
91 #include "SelectionWindow.h"
92 #include "TitleView.h"
93 #include "Tracker.h"
94 #include "TrackerSettings.h"
95 #include "Thread.h"
96 #include "TemplatesMenu.h"
97 
98 
99 #undef B_TRANSLATION_CONTEXT
100 #define B_TRANSLATION_CONTEXT "ContainerWindow"
101 
102 
103 #ifdef _IMPEXP_BE
104 _IMPEXP_BE
105 #endif
106 void do_minimize_team(BRect zoomRect, team_id team, bool zoom);
107 
108 
109 struct AddOneAddOnParams {
110 	BObjectList<BMenuItem>* primaryList;
111 	BObjectList<BMenuItem>* secondaryList;
112 };
113 
114 struct StaggerOneParams {
115 	bool rectFromParent;
116 };
117 
118 
119 BRect BContainerWindow::sNewWindRect;
120 static int32 sWindowStaggerBy;
121 
122 LockingList<AddOnShortcut>* BContainerWindow::fAddOnsList
123 	= new LockingList<struct AddOnShortcut>(10, true);
124 
125 
126 namespace BPrivate {
127 
128 filter_result
129 ActivateWindowFilter(BMessage*, BHandler** target, BMessageFilter*)
130 {
131 	BView* view = dynamic_cast<BView*>(*target);
132 
133 	// activate the window if no PoseView or DraggableContainerIcon had been
134 	// pressed (those will activate the window themselves, if necessary)
135 	if (view != NULL
136 		&& dynamic_cast<BPoseView*>(view) == NULL
137 		&& dynamic_cast<DraggableContainerIcon*>(view) == NULL
138 		&& view->Window() != NULL) {
139 		view->Window()->Activate(true);
140 	}
141 
142 	return B_DISPATCH_MESSAGE;
143 }
144 
145 }	// namespace BPrivate
146 
147 
148 static int32
149 AddOnMenuGenerate(const entry_ref* addOnRef, BMenu* menu,
150 	BContainerWindow* window)
151 {
152 	BEntry entry(addOnRef);
153 	BPath path;
154 	status_t result = entry.InitCheck();
155 	if (result != B_OK)
156 		return result;
157 
158 	result = entry.GetPath(&path);
159 	if (result != B_OK)
160 		return result;
161 
162 	image_id addOnImage = load_add_on(path.Path());
163 	if (addOnImage < 0)
164 		return addOnImage;
165 
166 	void (*populateMenu)(BMessage*, BMenu*, BHandler*);
167 	result = get_image_symbol(addOnImage, "populate_menu", 2,
168 		(void**)&populateMenu);
169 	if (result < 0) {
170 		PRINT(("Couldn't find populate_menu\n"));
171 		unload_add_on(addOnImage);
172 		return result;
173 	}
174 
175 	BMessage* message = window->AddOnMessage(B_TRACKER_ADDON_MESSAGE);
176 	message->AddRef("addon_ref", addOnRef);
177 
178 	// call add-on code
179 	(*populateMenu)(message, menu, window->PoseView());
180 
181 	unload_add_on(addOnImage);
182 	return B_OK;
183 }
184 
185 
186 static status_t
187 RunAddOnMessageThread(BMessage *message, void *)
188 {
189 	entry_ref addOnRef;
190 	BEntry entry;
191 	BPath path;
192 	status_t result = message->FindRef("addon_ref", &addOnRef);
193 	image_id addOnImage;
194 
195 	if (result != B_OK)
196 		goto end;
197 
198 	entry = BEntry(&addOnRef);
199 	result = entry.InitCheck();
200 	if (result != B_OK)
201 		goto end;
202 
203 	result = entry.GetPath(&path);
204 	if (result != B_OK)
205 		goto end;
206 
207 	addOnImage = load_add_on(path.Path());
208 	if (addOnImage < 0) {
209 		result = addOnImage;
210 		goto end;
211 	}
212 	void (*messageReceived)(BMessage*);
213 	result = get_image_symbol(addOnImage, "message_received", 2,
214 		(void**)&messageReceived);
215 
216 	if (result < 0) {
217 		PRINT(("Couldn't find message_received\n"));
218 		unload_add_on(addOnImage);
219 		goto end;
220 	}
221 	// call add-on code
222 	(*messageReceived)(message);
223 	unload_add_on(addOnImage);
224 	return B_OK;
225 
226 end:
227 	BString buffer(B_TRANSLATE("Error %error loading add-On %name."));
228 	buffer.ReplaceFirst("%error", strerror(result));
229 	buffer.ReplaceFirst("%name", addOnRef.name);
230 
231 	BAlert* alert = new BAlert("", buffer.String(), B_TRANSLATE("Cancel"),
232 		0, 0, B_WIDTH_AS_USUAL, B_WARNING_ALERT);
233 	alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE);
234 	alert->Go();
235 
236 	return result;
237 }
238 
239 
240 static void
241 AddOneAddOn(const Model* model, const char* name, uint32 shortcut,
242 	uint32 modifiers, bool primary, void* context,
243 	BContainerWindow* window, BMenu* menu)
244 {
245 	AddOneAddOnParams* params = (AddOneAddOnParams*)context;
246 
247 	BMessage* message = new BMessage(kLoadAddOn);
248 	message->AddRef("refs", model->EntryRef());
249 
250 	ModelMenuItem* item;
251 	try {
252 		item = new ModelMenuItem(model, name, message,
253 			(char)shortcut, modifiers);
254 	} catch (...) {
255 		delete message;
256 		return;
257 	}
258 
259 	const entry_ref* addOnRef = model->EntryRef();
260 	AddOnMenuGenerate(addOnRef, menu, window);
261 
262 	if (primary)
263 		params->primaryList->AddItem(item);
264 	else
265 		params->secondaryList->AddItem(item);
266 }
267 
268 
269 static int32
270 AddOnThread(BMessage* refsMessage, entry_ref addOnRef, entry_ref directoryRef)
271 {
272 	ObjectDeleter<BMessage> _(refsMessage);
273 
274 	BEntry entry(&addOnRef);
275 	BPath path;
276 	status_t result = entry.InitCheck();
277 	if (result == B_OK)
278 		result = entry.GetPath(&path);
279 
280 	if (result == B_OK) {
281 		image_id addOnImage = load_add_on(path.Path());
282 		if (addOnImage >= 0) {
283 			void (*processRefs)(entry_ref, BMessage*, void*);
284 			result = get_image_symbol(addOnImage, "process_refs", 2,
285 				(void**)&processRefs);
286 
287 			if (result >= 0) {
288 				// call add-on code
289 				(*processRefs)(directoryRef, refsMessage, NULL);
290 
291 				unload_add_on(addOnImage);
292 				return B_OK;
293 			} else
294 				PRINT(("couldn't find process_refs\n"));
295 
296 			unload_add_on(addOnImage);
297 		} else
298 			result = addOnImage;
299 	}
300 
301 	BString buffer(B_TRANSLATE("Error %error loading Add-On %name."));
302 	buffer.ReplaceFirst("%error", strerror(result));
303 	buffer.ReplaceFirst("%name", addOnRef.name);
304 
305 	BAlert* alert = new BAlert("", buffer.String(), B_TRANSLATE("Cancel"),
306 		0, 0, B_WIDTH_AS_USUAL, B_WARNING_ALERT);
307 	alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE);
308 	alert->Go();
309 
310 	return result;
311 }
312 
313 
314 static bool
315 NodeHasSavedState(const BNode* node)
316 {
317 	attr_info info;
318 	return node->GetAttrInfo(kAttrWindowFrame, &info) == B_OK;
319 }
320 
321 
322 static bool
323 OffsetFrameOne(const char* DEBUG_ONLY(name), uint32, off_t, void* castToRect,
324 	void* castToParams)
325 {
326 	ASSERT(strcmp(name, kAttrWindowFrame) == 0);
327 	StaggerOneParams* params = (StaggerOneParams*)castToParams;
328 
329 	if (!params->rectFromParent)
330 		return false;
331 
332 	if (!castToRect)
333 		return false;
334 
335 	((BRect*)castToRect)->OffsetBy(sWindowStaggerBy, sWindowStaggerBy);
336 
337 	return true;
338 }
339 
340 
341 static void
342 AddMimeTypeString(BStringList& list, Model* model)
343 {
344 	if (model == NULL)
345 		return;
346 
347 	const char* modelMimeType = model->MimeType();
348 	if (modelMimeType == NULL || *modelMimeType == '\0')
349 		return;
350 
351 	BString type = BString(modelMimeType);
352 	if (list.HasString(type, true))
353 		return;
354 
355 	list.Add(type);
356 }
357 
358 
359 //	#pragma mark - BContainerWindow
360 
361 
362 BContainerWindow::BContainerWindow(LockingList<BWindow>* list,
363 	uint32 openFlags, window_look look, window_feel feel, uint32 windowFlags,
364 	uint32 workspace, bool useLayout, bool isDeskWindow)
365 	:
366 	BWindow(InitialWindowRect(feel), "TrackerWindow", look, feel, windowFlags,
367 		workspace),
368 	fWindowList(list),
369 	fOpenFlags(openFlags),
370 	fUsesLayout(useLayout),
371 	fMenuContainer(NULL),
372 	fPoseContainer(NULL),
373 	fBorderedView(NULL),
374 	fVScrollBarContainer(NULL),
375 	fCountContainer(NULL),
376 	fContextMenu(NULL),
377 	fFileContextMenu(NULL),
378 	fWindowContextMenu(NULL),
379 	fDropContextMenu(NULL),
380 	fVolumeContextMenu(NULL),
381 	fTrashContextMenu(NULL),
382 	fDragContextMenu(NULL),
383 	fMoveToItem(NULL),
384 	fCopyToItem(NULL),
385 	fCreateLinkItem(NULL),
386 	fOpenWithItem(NULL),
387 	fNavigationItem(NULL),
388 	fMenuBar(NULL),
389 	fDraggableIcon(NULL),
390 	fNavigator(NULL),
391 	fPoseView(NULL),
392 	fAttrMenu(NULL),
393 	fWindowMenu(NULL),
394 	fFileMenu(NULL),
395 	fArrangeByMenu(NULL),
396 	fSelectionWindow(NULL),
397 	fTaskLoop(NULL),
398 	fStateNeedsSaving(false),
399 	fIsTrash(false),
400 	fInTrash(false),
401 	fIsPrinters(false),
402 	fIsDesktop(isDeskWindow),
403 	fBackgroundImage(NULL),
404 	fSavedZoomRect(0, 0, -1, -1),
405 	fDragMessage(NULL),
406 	fCachedTypesList(NULL),
407 	fSaveStateIsEnabled(true),
408 	fIsWatchingPath(false)
409 {
410 	InitIconPreloader();
411 
412 	if (list != NULL) {
413 		ASSERT(list->IsLocked());
414 		list->AddItem(this);
415 	}
416 
417 	if (fUsesLayout) {
418 		SetFlags(Flags() | B_AUTO_UPDATE_SIZE_LIMITS);
419 
420 		fRootLayout = new BGroupLayout(B_VERTICAL, 0);
421 		fRootLayout->SetInsets(0);
422 		SetLayout(fRootLayout);
423 		fRootLayout->Owner()->AdoptSystemColors();
424 
425 		fMenuContainer = new BGroupView(B_HORIZONTAL, 0);
426 		fRootLayout->AddView(fMenuContainer);
427 
428 		fPoseContainer = new BGridView(0.0, 0.0);
429 		fRootLayout->AddView(fPoseContainer);
430 
431 		fBorderedView = new BorderedView;
432 		fPoseContainer->GridLayout()->AddView(fBorderedView, 0, 1);
433 
434 		fCountContainer = new BGroupView(B_HORIZONTAL, 0);
435 		fPoseContainer->GridLayout()->AddView(fCountContainer, 0, 2);
436 	}
437 
438 	AddCommonFilter(new BMessageFilter(B_MOUSE_DOWN, ActivateWindowFilter));
439 
440 	Run();
441 
442 	// watch out for settings changes
443 	TTracker* tracker = dynamic_cast<TTracker*>(be_app);
444 	if (tracker != NULL && tracker->Lock()) {
445 		tracker->StartWatching(this, kWindowsShowFullPathChanged);
446 		tracker->StartWatching(this, kSingleWindowBrowseChanged);
447 		tracker->StartWatching(this, kShowNavigatorChanged);
448 		tracker->Unlock();
449 	}
450 
451 	// ToDo: remove me once we have undo/redo menu items
452 	// (that is, move them to AddShortcuts())
453 	AddShortcut('Z', B_COMMAND_KEY, new BMessage(B_UNDO), this);
454 	AddShortcut('Z', B_COMMAND_KEY | B_SHIFT_KEY, new BMessage(B_REDO), this);
455 }
456 
457 
458 BContainerWindow::~BContainerWindow()
459 {
460 	ASSERT(IsLocked());
461 
462 	// stop the watchers
463 	TTracker* tracker = dynamic_cast<TTracker*>(be_app);
464 	if (tracker != NULL && tracker->Lock()) {
465 		tracker->StopWatching(this, kWindowsShowFullPathChanged);
466 		tracker->StopWatching(this, kSingleWindowBrowseChanged);
467 		tracker->StopWatching(this, kShowNavigatorChanged);
468 		tracker->Unlock();
469 	}
470 
471 	delete fTaskLoop;
472 	delete fBackgroundImage;
473 	delete fDragMessage;
474 	delete fCachedTypesList;
475 
476 	if (fSelectionWindow != NULL && fSelectionWindow->Lock())
477 		fSelectionWindow->Quit();
478 }
479 
480 
481 BRect
482 BContainerWindow::InitialWindowRect(window_feel feel)
483 {
484 	if (!sNewWindRect.IsValid()) {
485 		const float labelSpacing = be_control_look->DefaultLabelSpacing();
486 		// approximately (85, 50, 548, 280) with default spacing
487 		sNewWindRect = BRect(labelSpacing * 14, labelSpacing * 8,
488 			labelSpacing * 91, labelSpacing * 46);
489 		sWindowStaggerBy = (int32)(labelSpacing * 3.0f);
490 	}
491 
492 	if (feel != kDesktopWindowFeel)
493 		return sNewWindRect;
494 
495 	// do not offset desktop window
496 	BRect result = sNewWindRect;
497 	result.OffsetTo(0, 0);
498 	return result;
499 }
500 
501 
502 void
503 BContainerWindow::Minimize(bool minimize)
504 {
505 	if (minimize && (modifiers() & B_OPTION_KEY) != 0)
506 		do_minimize_team(BRect(0, 0, 0, 0), be_app->Team(), true);
507 	else
508 		_inherited::Minimize(minimize);
509 }
510 
511 
512 bool
513 BContainerWindow::QuitRequested()
514 {
515 	// this is a response to the DeskBar sending us a B_QUIT, when it really
516 	// means to say close all your windows. It might be better to have it
517 	// send a kCloseAllWindows message and have windowless apps stay running,
518 	// which is what we will do for the Tracker
519 	if (CurrentMessage() != NULL
520 		&& ((CurrentMessage()->FindInt32("modifiers") & B_CONTROL_KEY)) != 0) {
521 		be_app->PostMessage(kCloseAllWindows);
522 	}
523 
524 	Hide();
525 		// this will close the window instantly, even if
526 		// the file system is very busy right now
527 	return true;
528 }
529 
530 
531 void
532 BContainerWindow::Quit()
533 {
534 	// get rid of context menus
535 	if (fNavigationItem) {
536 		BMenu* menu = fNavigationItem->Menu();
537 		if (menu != NULL)
538 			menu->RemoveItem(fNavigationItem);
539 
540 		delete fNavigationItem;
541 		fNavigationItem = NULL;
542 	}
543 
544 	if (fOpenWithItem != NULL && fOpenWithItem->Menu() == NULL) {
545 		delete fOpenWithItem;
546 		fOpenWithItem = NULL;
547 	}
548 
549 	if (fMoveToItem != NULL && fMoveToItem->Menu() == NULL) {
550 		delete fMoveToItem;
551 		fMoveToItem = NULL;
552 	}
553 
554 	if (fCopyToItem != NULL && fCopyToItem->Menu() == NULL) {
555 		delete fCopyToItem;
556 		fCopyToItem = NULL;
557 	}
558 
559 	if (fCreateLinkItem != NULL && fCreateLinkItem->Menu() == NULL) {
560 		delete fCreateLinkItem;
561 		fCreateLinkItem = NULL;
562 	}
563 
564 	if (fAttrMenu != NULL && fAttrMenu->Supermenu() == NULL) {
565 		delete fAttrMenu;
566 		fAttrMenu = NULL;
567 	}
568 
569 	delete fFileContextMenu;
570 	fFileContextMenu = NULL;
571 
572 	delete fWindowContextMenu;
573 	fWindowContextMenu = NULL;
574 
575 	delete fDropContextMenu;
576 	fDropContextMenu = NULL;
577 
578 	delete fVolumeContextMenu;
579 	fVolumeContextMenu = NULL;
580 
581 	delete fDragContextMenu;
582 	fDragContextMenu = NULL;
583 
584 	delete fTrashContextMenu;
585 	fTrashContextMenu = NULL;
586 
587 	int32 windowCount = 0;
588 
589 	// This is a deadlock code sequence - need to change this
590 	// to acquire the window list while this container window is unlocked
591 	if (fWindowList != NULL) {
592 		AutoLock<LockingList<BWindow> > lock(fWindowList);
593 		if (lock.IsLocked()) {
594 			fWindowList->RemoveItem(this, false);
595 			windowCount = fWindowList->CountItems();
596 		}
597 	}
598 
599 	if (StateNeedsSaving())
600 		SaveState();
601 
602 	if (fWindowList != NULL && windowCount == 0)
603 		be_app->PostMessage(B_QUIT_REQUESTED);
604 
605 	_inherited::Quit();
606 }
607 
608 
609 BPoseView*
610 BContainerWindow::NewPoseView(Model* model, uint32 viewMode)
611 {
612 	return new BPoseView(model, viewMode);
613 }
614 
615 
616 void
617 BContainerWindow::UpdateIfTrash(Model* model)
618 {
619 	BEntry entry(model->EntryRef());
620 
621 	if (entry.InitCheck() == B_OK) {
622 		fIsTrash = model->IsTrash();
623 		fInTrash = FSInTrashDir(model->EntryRef());
624 		fIsPrinters = FSIsPrintersDir(&entry);
625 	}
626 }
627 
628 
629 void
630 BContainerWindow::CreatePoseView(Model* model)
631 {
632 	UpdateIfTrash(model);
633 
634 	fPoseView = NewPoseView(model, kListMode);
635 	fBorderedView->GroupLayout()->AddView(fPoseView);
636 	fBorderedView->GroupLayout()->SetInsets(1, 0, 1, 1);
637 	fBorderedView->EnableBorderHighlight(false);
638 
639 	TrackerSettings settings;
640 	if (settings.SingleWindowBrowse() && model->IsDirectory()
641 		&& !fPoseView->IsFilePanel()) {
642 		fNavigator = new BNavigator(model);
643 		fPoseContainer->GridLayout()->AddView(fNavigator, 0, 0, 2);
644 		if (!settings.ShowNavigator())
645 			fNavigator->Hide();
646 	}
647 
648 	SetPathWatchingEnabled(settings.ShowNavigator()
649 		|| settings.ShowFullPathInTitleBar());
650 }
651 
652 
653 void
654 BContainerWindow::AddContextMenus()
655 {
656 	// create context sensitive menus
657 	fFileContextMenu = new BPopUpMenu("FileContext", false, false);
658 	AddFileContextMenus(fFileContextMenu);
659 
660 	fVolumeContextMenu = new BPopUpMenu("VolumeContext", false, false);
661 	AddVolumeContextMenus(fVolumeContextMenu);
662 
663 	fWindowContextMenu = new BPopUpMenu("WindowContext", false, false);
664 	AddWindowContextMenus(fWindowContextMenu);
665 
666 	fDropContextMenu = new BPopUpMenu("DropContext", false, false);
667 	AddDropContextMenus(fDropContextMenu);
668 
669 	fDragContextMenu = new BPopUpNavMenu("DragContext");
670 		// will get added and built dynamically in ShowContextMenu
671 
672 	fTrashContextMenu = new BPopUpMenu("TrashContext", false, false);
673 	AddTrashContextMenus(fTrashContextMenu);
674 }
675 
676 
677 void
678 BContainerWindow::RepopulateMenus()
679 {
680 	// Avoid these menus to be destroyed:
681 	if (fMoveToItem != NULL && fMoveToItem->Menu() != NULL)
682 		fMoveToItem->Menu()->RemoveItem(fMoveToItem);
683 
684 	if (fCopyToItem != NULL && fCopyToItem->Menu() != NULL)
685 		fCopyToItem->Menu()->RemoveItem(fCopyToItem);
686 
687 	if (fCreateLinkItem != NULL && fCreateLinkItem->Menu() != NULL)
688 		fCreateLinkItem->Menu()->RemoveItem(fCreateLinkItem);
689 
690 	if (fOpenWithItem != NULL && fOpenWithItem->Menu() != NULL) {
691 		fOpenWithItem->Menu()->RemoveItem(fOpenWithItem);
692 		delete fOpenWithItem;
693 		fOpenWithItem = NULL;
694 	}
695 
696 	if (fNavigationItem != NULL) {
697 		BMenu* menu = fNavigationItem->Menu();
698 		if (menu != NULL) {
699 			menu->RemoveItem(fNavigationItem);
700 			BMenuItem* item = menu->RemoveItem((int32)0);
701 			ASSERT(item != fNavigationItem);
702 			delete item;
703 		}
704 	}
705 
706 	delete fFileContextMenu;
707 	fFileContextMenu = new BPopUpMenu("FileContext", false, false);
708 	fFileContextMenu->SetFont(be_plain_font);
709 	AddFileContextMenus(fFileContextMenu);
710 
711 	delete fWindowContextMenu;
712 	fWindowContextMenu = new BPopUpMenu("WindowContext", false, false);
713 	fWindowContextMenu->SetFont(be_plain_font);
714 	AddWindowContextMenus(fWindowContextMenu);
715 
716 	if (fMenuBar != NULL) {
717 		fMenuBar->RemoveItem(fFileMenu);
718 		delete fFileMenu;
719 		fFileMenu = new BMenu(B_TRANSLATE("File"));
720 		AddFileMenu(fFileMenu);
721 		fMenuBar->AddItem(fFileMenu);
722 
723 		fMenuBar->RemoveItem(fWindowMenu);
724 		delete fWindowMenu;
725 		fWindowMenu = new BMenu(B_TRANSLATE("Window"));
726 		fMenuBar->AddItem(fWindowMenu);
727 		AddWindowMenu(fWindowMenu);
728 
729 		// just create the attribute, decide to add it later
730 		fMenuBar->RemoveItem(fAttrMenu);
731 		delete fAttrMenu;
732 		fAttrMenu = new BMenu(B_TRANSLATE("Attributes"));
733 		NewAttributesMenu(fAttrMenu);
734 		if (PoseView()->ViewMode() == kListMode)
735 			ShowAttributesMenu();
736 
737 		PopulateArrangeByMenu(fArrangeByMenu);
738 
739 		int32 selectCount = PoseView()->CountSelected();
740 
741 		SetupOpenWithMenu(fFileMenu);
742 		SetupMoveCopyMenus(selectCount ? PoseView()->SelectionList()
743 				->FirstItem()->TargetModel()->EntryRef() : NULL,
744 			fFileMenu);
745 	}
746 }
747 
748 
749 void
750 BContainerWindow::Init(const BMessage* message)
751 {
752 	// pose view is expected to be setup at this point
753 	if (PoseView() == NULL)
754 		return;
755 
756 	// deal with new unconfigured folders
757 	if (NeedsDefaultStateSetup())
758 		SetupDefaultState();
759 
760 	if (ShouldAddScrollBars())
761 		PoseView()->AddScrollBars();
762 
763 	fMoveToItem = new BMenuItem(new BNavMenu(B_TRANSLATE("Move to"),
764 		kMoveSelectionTo, this));
765 	fCopyToItem = new BMenuItem(new BNavMenu(B_TRANSLATE("Copy to"),
766 		kCopySelectionTo, this));
767 	fCreateLinkItem = new BMenuItem(new BNavMenu(B_TRANSLATE("Create link"),
768 		kCreateLink, this), new BMessage(kCreateLink));
769 
770 	TrackerSettings settings;
771 
772 	if (ShouldAddMenus()) {
773 		fMenuBar = new BMenuBar("MenuBar");
774 		fMenuContainer->GroupLayout()->AddView(fMenuBar);
775 		AddMenus();
776 
777 		if (!TargetModel()->IsRoot() && !IsTrash())
778 			_AddFolderIcon();
779 	} else {
780 		// add equivalents of the menu shortcuts to the menuless
781 		// desktop window
782 		AddShortcuts();
783 	}
784 
785 	AddContextMenus();
786 	AddShortcut('T', B_COMMAND_KEY | B_SHIFT_KEY, new BMessage(kDelete),
787 		PoseView());
788 	AddShortcut('K', B_COMMAND_KEY | B_SHIFT_KEY, new BMessage(kCleanupAll),
789 		PoseView());
790 	AddShortcut('Q', B_COMMAND_KEY | B_OPTION_KEY | B_SHIFT_KEY
791 		| B_CONTROL_KEY, new BMessage(kQuitTracker));
792 
793 	AddShortcut(B_DOWN_ARROW, B_COMMAND_KEY, new BMessage(kOpenSelection),
794 		PoseView());
795 
796 	SetSingleWindowBrowseShortcuts(settings.SingleWindowBrowse());
797 
798 #if DEBUG
799 	// add some debugging shortcuts
800 	AddShortcut('D', B_COMMAND_KEY | B_CONTROL_KEY, new BMessage('dbug'),
801 		PoseView());
802 	AddShortcut('C', B_COMMAND_KEY | B_CONTROL_KEY, new BMessage('dpcc'),
803 		PoseView());
804 	AddShortcut('F', B_COMMAND_KEY | B_CONTROL_KEY, new BMessage('dpfl'),
805 		PoseView());
806 	AddShortcut('F', B_COMMAND_KEY | B_CONTROL_KEY | B_OPTION_KEY,
807 		new BMessage('dpfL'), PoseView());
808 #endif
809 
810 	BKeymap keymap;
811 	if (keymap.SetToCurrent() == B_OK) {
812 		BObjectList<const char> unmodified(3, true);
813 		if (keymap.GetModifiedCharacters("+", B_SHIFT_KEY, 0, &unmodified)
814 				== B_OK) {
815 			int32 count = unmodified.CountItems();
816 			for (int32 i = 0; i < count; i++) {
817 				uint32 key = BUnicodeChar::FromUTF8(unmodified.ItemAt(i));
818 				if (!HasShortcut(key, 0)) {
819 					// Add semantic zoom in shortcut, bug #6692
820 					BMessage* increaseSize = new BMessage(kIconMode);
821 					increaseSize->AddInt32("scale", 1);
822 					AddShortcut(key, B_COMMAND_KEY, increaseSize, PoseView());
823 				}
824 			}
825 		}
826 		unmodified.MakeEmpty();
827 	}
828 
829 	if (message != NULL)
830 		RestoreState(*message);
831 	else
832 		RestoreState();
833 
834 	if (ShouldAddMenus() && PoseView()->ViewMode() == kListMode) {
835 		// for now only show attributes in list view
836 		// eventually enable attribute menu to allow users to select
837 		// using different attributes as titles in icon view modes
838 		ShowAttributesMenu();
839 	}
840 	MarkAttributesMenu(fAttrMenu);
841 	CheckScreenIntersect();
842 
843 	if (fBackgroundImage != NULL && !fIsDesktop
844 		&& PoseView()->ViewMode() != kListMode) {
845 		fBackgroundImage->Show(PoseView(), current_workspace());
846 	}
847 
848 	Show();
849 
850 	// done showing, turn the B_NO_WORKSPACE_ACTIVATION flag off;
851 	// it was on to prevent workspace jerking during boot
852 	SetFlags(Flags() & ~B_NO_WORKSPACE_ACTIVATION);
853 }
854 
855 
856 void
857 BContainerWindow::InitLayout()
858 {
859 	fBorderedView->GroupLayout()->AddView(0, PoseView()->TitleView());
860 
861 	fCountContainer->GroupLayout()->AddView(PoseView()->CountView(), 0.25f);
862 
863 	bool forFilePanel = PoseView()->IsFilePanel();
864 	if (!forFilePanel) {
865 		// Eliminate the extra borders
866 		fPoseContainer->GridLayout()->SetInsets(-1, 0, -1, -1);
867 		fCountContainer->GroupLayout()->SetInsets(0, -1, 0, 0);
868 	}
869 
870 	if (PoseView()->VScrollBar() != NULL) {
871 		fVScrollBarContainer = new BGroupView(B_VERTICAL, 0);
872 		fVScrollBarContainer->GroupLayout()->AddView(PoseView()->VScrollBar());
873 		fVScrollBarContainer->GroupLayout()->SetInsets(-1, forFilePanel ? 0 : -1,
874 			0, 0);
875 		fPoseContainer->GridLayout()->AddView(fVScrollBarContainer, 1, 1);
876 	}
877 	if (PoseView()->HScrollBar() != NULL) {
878 		BGroupView* hScrollBarContainer = new BGroupView(B_VERTICAL, 0);
879 		hScrollBarContainer->GroupLayout()->AddView(PoseView()->HScrollBar());
880 		hScrollBarContainer->GroupLayout()->SetInsets(0, -1, 0,
881 			forFilePanel ? 0 : -1);
882 		fCountContainer->GroupLayout()->AddView(hScrollBarContainer);
883 
884 		BSize size = PoseView()->HScrollBar()->MinSize();
885 		if (forFilePanel) {
886 			// Count view height is 1px smaller than scroll bar because it has
887 			// no upper border.
888 			size.height -= 1;
889 		}
890 		PoseView()->CountView()->SetExplicitMinSize(size);
891 	}
892 }
893 
894 
895 void
896 BContainerWindow::RestoreState()
897 {
898 	UpdateTitle();
899 
900 	WindowStateNodeOpener opener(this, false);
901 	RestoreWindowState(opener.StreamNode());
902 	PoseView()->Init(opener.StreamNode());
903 
904 	RestoreStateCommon();
905 }
906 
907 
908 void
909 BContainerWindow::RestoreState(const BMessage &message)
910 {
911 	UpdateTitle();
912 
913 	RestoreWindowState(message);
914 	PoseView()->Init(message);
915 
916 	RestoreStateCommon();
917 }
918 
919 
920 void
921 BContainerWindow::RestoreStateCommon()
922 {
923 	if (fUsesLayout)
924 		InitLayout();
925 
926 	if (BootedInSafeMode())
927 		// don't pick up backgrounds in safe mode
928 		return;
929 
930 	WindowStateNodeOpener opener(this, false);
931 
932 	if (!TargetModel()->IsRoot() && opener.Node() != NULL) {
933 		// don't pick up background image for root disks
934 		// to do this, would have to have a unique attribute for the
935 		// disks window that doesn't collide with the desktop
936 		// for R4 this was not done to make things simpler
937 		// the default image will still work though
938 		fBackgroundImage = BackgroundImage::GetBackgroundImage(
939 			opener.Node(), fIsDesktop);
940 			// look for background image info in the window's node
941 	}
942 
943 	BNode defaultingNode;
944 	if (fBackgroundImage == NULL && !fIsDesktop
945 		&& DefaultStateSourceNode(kDefaultFolderTemplate, &defaultingNode)) {
946 		// look for background image info in the source for defaults
947 		fBackgroundImage = BackgroundImage::GetBackgroundImage(&defaultingNode,
948 			fIsDesktop);
949 	}
950 }
951 
952 
953 void
954 BContainerWindow::UpdateTitle()
955 {
956 	// set title to full path, if necessary
957 	if (TrackerSettings().ShowFullPathInTitleBar()) {
958 		// use the Entry's full path
959 		BPath path;
960 		TargetModel()->GetPath(&path);
961 		SetTitle(path.Path());
962 	} else {
963 		// use the default look
964 		SetTitle(TargetModel()->Name());
965 	}
966 
967 	if (Navigator() != NULL)
968 		Navigator()->UpdateLocation(TargetModel(), kActionUpdatePath);
969 }
970 
971 
972 void
973 BContainerWindow::UpdateBackgroundImage()
974 {
975 	if (BootedInSafeMode())
976 		return;
977 
978 	WindowStateNodeOpener opener(this, false);
979 
980 	if (!TargetModel()->IsRoot() && opener.Node() != NULL) {
981 		fBackgroundImage = BackgroundImage::Refresh(fBackgroundImage,
982 			opener.Node(), fIsDesktop, PoseView());
983 	}
984 
985 		// look for background image info in the window's node
986 	BNode defaultingNode;
987 	if (!fBackgroundImage && !fIsDesktop
988 		&& DefaultStateSourceNode(kDefaultFolderTemplate, &defaultingNode)) {
989 		// look for background image info in the source for defaults
990 		fBackgroundImage = BackgroundImage::Refresh(fBackgroundImage,
991 			&defaultingNode, fIsDesktop, PoseView());
992 	}
993 }
994 
995 
996 void
997 BContainerWindow::FrameResized(float, float)
998 {
999 	if (PoseView() != NULL && !fIsDesktop) {
1000 		BRect extent = PoseView()->Extent();
1001 		float offsetX = extent.left - PoseView()->Bounds().left;
1002 		float offsetY = extent.top - PoseView()->Bounds().top;
1003 
1004 		// scroll when the size augmented, there is a negative offset
1005 		// and we have resized over the bottom right corner of the extent
1006 		BPoint scroll(B_ORIGIN);
1007 		if (offsetX < 0 && PoseView()->Bounds().right > extent.right
1008 			&& Bounds().Width() > fPreviousBounds.Width()) {
1009 			scroll.x = std::max(fPreviousBounds.Width() - Bounds().Width(),
1010 				offsetX);
1011 		}
1012 
1013 		if (offsetY < 0 && PoseView()->Bounds().bottom > extent.bottom
1014 			&& Bounds().Height() > fPreviousBounds.Height()) {
1015 			scroll.y = std::max(fPreviousBounds.Height() - Bounds().Height(),
1016 				offsetY);
1017 		}
1018 
1019 		if (scroll != B_ORIGIN)
1020 			PoseView()->ScrollBy(scroll.x, scroll.y);
1021 
1022 		PoseView()->UpdateScrollRange();
1023 		PoseView()->ResetPosePlacementHint();
1024 	}
1025 
1026 	fPreviousBounds = Bounds();
1027 	if (IsActive())
1028 		fStateNeedsSaving = true;
1029 }
1030 
1031 
1032 void
1033 BContainerWindow::FrameMoved(BPoint)
1034 {
1035 	if (IsActive())
1036 		fStateNeedsSaving = true;
1037 }
1038 
1039 
1040 void
1041 BContainerWindow::WorkspacesChanged(uint32, uint32)
1042 {
1043 	if (IsActive())
1044 		fStateNeedsSaving = true;
1045 }
1046 
1047 
1048 void
1049 BContainerWindow::ViewModeChanged(uint32 oldMode, uint32 newMode)
1050 {
1051 	if (fBackgroundImage == NULL)
1052 		return;
1053 
1054 	if (newMode == kListMode)
1055 		fBackgroundImage->Remove();
1056 	else if (oldMode == kListMode)
1057 		fBackgroundImage->Show(PoseView(), current_workspace());
1058 }
1059 
1060 
1061 void
1062 BContainerWindow::CheckScreenIntersect()
1063 {
1064 	BScreen screen(this);
1065 	BRect screenFrame(screen.Frame());
1066 	BRect frame(Frame());
1067 
1068 	if (sNewWindRect.bottom > screenFrame.bottom)
1069 		sNewWindRect.OffsetTo(85, 50);
1070 
1071 	if (sNewWindRect.right > screenFrame.right)
1072 		sNewWindRect.OffsetTo(85, 50);
1073 
1074 	if (!frame.Intersects(screenFrame))
1075 		MoveTo(sNewWindRect.LeftTop());
1076 }
1077 
1078 
1079 void
1080 BContainerWindow::SaveState(bool hide)
1081 {
1082 	if (SaveStateIsEnabled()) {
1083 		WindowStateNodeOpener opener(this, true);
1084 		if (opener.StreamNode() != NULL)
1085 			SaveWindowState(opener.StreamNode());
1086 
1087 		if (hide)
1088 			Hide();
1089 
1090 		if (opener.StreamNode())
1091 			PoseView()->SaveState(opener.StreamNode());
1092 
1093 		fStateNeedsSaving = false;
1094 	}
1095 }
1096 
1097 
1098 void
1099 BContainerWindow::SaveState(BMessage& message) const
1100 {
1101 	if (SaveStateIsEnabled()) {
1102 		SaveWindowState(message);
1103 		PoseView()->SaveState(message);
1104 	}
1105 }
1106 
1107 
1108 bool
1109 BContainerWindow::StateNeedsSaving() const
1110 {
1111 	return PoseView() != NULL && (fStateNeedsSaving || PoseView()->StateNeedsSaving());
1112 }
1113 
1114 
1115 status_t
1116 BContainerWindow::GetLayoutState(BNode* node, BMessage* message)
1117 {
1118 	if (node == NULL || message == NULL)
1119 		return B_BAD_VALUE;
1120 
1121 	status_t result = node->InitCheck();
1122 	if (result != B_OK)
1123 		return result;
1124 
1125 	// ToDo: get rid of this, use AttrStream instead
1126 	node->RewindAttrs();
1127 	char attrName[256];
1128 	while (node->GetNextAttrName(attrName) == B_OK) {
1129 		attr_info info;
1130 		if (node->GetAttrInfo(attrName, &info) != B_OK)
1131 			continue;
1132 
1133 		// filter out attributes that are not related to window position
1134 		// and column resizing
1135 		// more can be added as needed
1136 		if (strcmp(attrName, kAttrWindowFrame) != 0
1137 			&& strcmp(attrName, kAttrColumns) != 0
1138 			&& strcmp(attrName, kAttrViewState) != 0
1139 			&& strcmp(attrName, kAttrColumnsForeign) != 0
1140 			&& strcmp(attrName, kAttrViewStateForeign) != 0) {
1141 			continue;
1142 		}
1143 
1144 		char* buffer = new char[info.size];
1145 		if (node->ReadAttr(attrName, info.type, 0, buffer,
1146 				(size_t)info.size) == info.size) {
1147 			message->AddData(attrName, info.type, buffer, (ssize_t)info.size);
1148 		}
1149 		delete[] buffer;
1150 	}
1151 
1152 	return B_OK;
1153 }
1154 
1155 
1156 status_t
1157 BContainerWindow::SetLayoutState(BNode* node, const BMessage* message)
1158 {
1159 	status_t result = node->InitCheck();
1160 	if (result != B_OK)
1161 		return result;
1162 
1163 	for (int32 globalIndex = 0; ;) {
1164 #if B_BEOS_VERSION_DANO
1165 		const char* name;
1166 #else
1167 		char* name;
1168 #endif
1169 		type_code type;
1170 		int32 count;
1171 		status_t result = message->GetInfo(B_ANY_TYPE, globalIndex, &name,
1172 			&type, &count);
1173 		if (result != B_OK)
1174 			break;
1175 
1176 		for (int32 index = 0; index < count; index++) {
1177 			const void* buffer;
1178 			ssize_t size;
1179 			result = message->FindData(name, type, index, &buffer, &size);
1180 			if (result != B_OK) {
1181 				PRINT(("error reading %s \n", name));
1182 				return result;
1183 			}
1184 
1185 			if (node->WriteAttr(name, type, 0, buffer,
1186 					(size_t)size) != size) {
1187 				PRINT(("error writing %s \n", name));
1188 				return result;
1189 			}
1190 			globalIndex++;
1191 		}
1192 	}
1193 
1194 	return B_OK;
1195 }
1196 
1197 
1198 bool
1199 BContainerWindow::ShouldAddMenus() const
1200 {
1201 	return true;
1202 }
1203 
1204 
1205 bool
1206 BContainerWindow::ShouldAddScrollBars() const
1207 {
1208 	return true;
1209 }
1210 
1211 
1212 Model*
1213 BContainerWindow::TargetModel() const
1214 {
1215 	return PoseView()->TargetModel();
1216 }
1217 
1218 
1219 void
1220 BContainerWindow::SelectionChanged()
1221 {
1222 }
1223 
1224 
1225 void
1226 BContainerWindow::Zoom(BPoint, float, float)
1227 {
1228 	BRect oldZoomRect(fSavedZoomRect);
1229 	fSavedZoomRect = Frame();
1230 	ResizeToFit();
1231 
1232 	if (fSavedZoomRect == Frame() && oldZoomRect.IsValid())
1233 		ResizeTo(oldZoomRect.Width(), oldZoomRect.Height());
1234 }
1235 
1236 
1237 void
1238 BContainerWindow::ResizeToFit()
1239 {
1240 	BScreen screen(this);
1241 	BRect screenFrame(screen.Frame());
1242 
1243 	screenFrame.InsetBy(5, 5);
1244 	BMessage decoratorSettings;
1245 	GetDecoratorSettings(&decoratorSettings);
1246 
1247 	float tabHeight = 15;
1248 	BRect tabRect;
1249 	if (decoratorSettings.FindRect("tab frame", &tabRect) == B_OK)
1250 		tabHeight = tabRect.Height();
1251 	screenFrame.top += tabHeight;
1252 
1253 	BRect frame(Frame());
1254 
1255 	float widthDiff = frame.Width() - PoseView()->Frame().Width();
1256 	float heightDiff = frame.Height() - PoseView()->Frame().Height();
1257 
1258 	// move frame left top on screen
1259 	BPoint leftTop(frame.LeftTop());
1260 	leftTop.ConstrainTo(screenFrame);
1261 	frame.OffsetTo(leftTop);
1262 
1263 	// resize to extent size
1264 	BRect extent(PoseView()->Extent());
1265 	frame.right = frame.left + extent.Width() + widthDiff;
1266 	frame.bottom = frame.top + extent.Height() + heightDiff;
1267 
1268 	// make sure entire window fits on screen
1269 	frame = frame & screenFrame;
1270 
1271 	ResizeTo(frame.Width(), frame.Height());
1272 	MoveTo(frame.LeftTop());
1273 	PoseView()->DisableScrollBars();
1274 
1275 	// scroll if there is an offset
1276 	PoseView()->ScrollBy(
1277 		extent.left - PoseView()->Bounds().left,
1278 		extent.top - PoseView()->Bounds().top);
1279 
1280 	PoseView()->UpdateScrollRange();
1281 	PoseView()->EnableScrollBars();
1282 }
1283 
1284 
1285 void
1286 BContainerWindow::MessageReceived(BMessage* message)
1287 {
1288 	switch (message->what) {
1289 		case B_CUT:
1290 		case B_COPY:
1291 		case B_PASTE:
1292 		case B_SELECT_ALL:
1293 		{
1294 			BView* view = CurrentFocus();
1295 			if (dynamic_cast<BTextView*>(view) == NULL) {
1296 				// The selected item is not a BTextView, so forward the
1297 				// message to the PoseView.
1298 				if (PoseView() != NULL)
1299 					PostMessage(message, PoseView());
1300 			} else {
1301 				// Since we catch the generic clipboard shortcuts in a way that
1302 				// means the BTextView will never get them, we must
1303 				// manually forward them ourselves.
1304 				//
1305 				// However, we have to take care to not forward the custom
1306 				// clipboard messages, else we would wind up in infinite
1307 				// recursion.
1308 				PostMessage(message, view);
1309 			}
1310 			break;
1311 		}
1312 
1313 		case kCutMoreSelectionToClipboard:
1314 		case kCopyMoreSelectionToClipboard:
1315 		case kPasteLinksFromClipboard:
1316 			if (PoseView() != NULL)
1317 				PostMessage(message, PoseView());
1318 			break;
1319 
1320 		case B_UNDO: {
1321 			BView* view = CurrentFocus();
1322 			if (dynamic_cast<BTextView*>(view) == NULL) {
1323 				FSUndo();
1324 			} else {
1325 				view->MessageReceived(message);
1326 			}
1327 			break;
1328 		}
1329 
1330 		case B_REDO: {
1331 			BView* view = CurrentFocus();
1332 			if (dynamic_cast<BTextView*>(view) == NULL) {
1333 				FSRedo();
1334 			} else {
1335 				view->MessageReceived(message);
1336 			}
1337 			break;
1338 		}
1339 
1340 		case kNewFolder:
1341 			PostMessage(message, PoseView());
1342 			break;
1343 
1344 		case kRestoreState:
1345 			if (message->HasMessage("state")) {
1346 				BMessage state;
1347 				message->FindMessage("state", &state);
1348 				Init(&state);
1349 			} else
1350 				Init();
1351 			break;
1352 
1353 		case kResizeToFit:
1354 			ResizeToFit();
1355 			break;
1356 
1357 		case kLoadAddOn:
1358 			LoadAddOn(message);
1359 			break;
1360 
1361 		case kCopySelectionTo:
1362 		{
1363 			entry_ref ref;
1364 			if (message->FindRef("refs", &ref) != B_OK)
1365 				break;
1366 
1367 			BRoster().AddToRecentFolders(&ref);
1368 
1369 			Model model(&ref);
1370 			if (model.InitCheck() != B_OK)
1371 				break;
1372 
1373 			if (*model.NodeRef() == *TargetModel()->NodeRef())
1374 				PoseView()->DuplicateSelection();
1375 			else
1376 				PoseView()->MoveSelectionInto(&model, this, true);
1377 			break;
1378 		}
1379 
1380 		case kMoveSelectionTo:
1381 		{
1382 			entry_ref ref;
1383 			if (message->FindRef("refs", &ref) != B_OK)
1384 				break;
1385 
1386 			BRoster().AddToRecentFolders(&ref);
1387 
1388 			Model model(&ref);
1389 			if (model.InitCheck() != B_OK)
1390 				break;
1391 
1392 			PoseView()->MoveSelectionInto(&model, this, false, true);
1393 			break;
1394 		}
1395 
1396 		case kCreateLink:
1397 		case kCreateRelativeLink:
1398 		{
1399 			entry_ref ref;
1400 			if (message->FindRef("refs", &ref) == B_OK) {
1401 				BRoster().AddToRecentFolders(&ref);
1402 
1403 				Model model(&ref);
1404 				if (model.InitCheck() != B_OK)
1405 					break;
1406 
1407 				PoseView()->MoveSelectionInto(&model, this, false, false,
1408 					message->what == kCreateLink,
1409 					message->what == kCreateRelativeLink);
1410 			} else if (!TargetModel()->IsQuery()
1411 				&& !TargetModel()->IsVirtualDirectory()) {
1412 				// no destination specified, create link in same dir as item
1413 				PoseView()->MoveSelectionInto(TargetModel(), this, false, false,
1414 					message->what == kCreateLink,
1415 					message->what == kCreateRelativeLink);
1416 			}
1417 			break;
1418 		}
1419 
1420 		case kShowSelectionWindow:
1421 			ShowSelectionWindow();
1422 			break;
1423 
1424 		case kSelectMatchingEntries:
1425 			PoseView()->SelectMatchingEntries(message);
1426 			break;
1427 
1428 		case kFindButton:
1429 			(new FindWindow())->Show();
1430 			break;
1431 
1432 		case kQuitTracker:
1433 			be_app->PostMessage(B_QUIT_REQUESTED);
1434 			break;
1435 
1436 		case kRestoreBackgroundImage:
1437 			UpdateBackgroundImage();
1438 			break;
1439 
1440 		case kSwitchDirectory:
1441 		{
1442 			entry_ref ref;
1443 			if (message->FindRef("refs", &ref) != B_OK)
1444 				break;
1445 
1446 			BEntry entry;
1447 			if (entry.SetTo(&ref) != B_OK)
1448 				break;
1449 
1450 			if (StateNeedsSaving())
1451 				SaveState(false);
1452 
1453 			bool wasInTrash = IsTrash() || InTrash();
1454 			bool isRoot = TargetModel()->IsRoot();
1455 			bool isVolume = TargetModel()->IsVolume();
1456 
1457 			// Switch dir and apply new state
1458 			WindowStateNodeOpener opener(this, false);
1459 			opener.SetTo(&entry, false);
1460 
1461 			// Update PoseView
1462 			PoseView()->SwitchDir(&ref, opener.StreamNode());
1463 
1464 			fIsTrash = FSIsTrashDir(&entry);
1465 			fInTrash = FSInTrashDir(&ref);
1466 
1467 			if (wasInTrash ^ (IsTrash() || InTrash()) || isRoot != TargetModel()->IsRoot()
1468 				|| isVolume != TargetModel()->IsVolume()) {
1469 				RepopulateMenus();
1470 			}
1471 
1472 			if (Navigator() != NULL) {
1473 				// update Navigation bar
1474 				int32 action = kActionSet;
1475 				if (message->FindInt32("action", &action) != B_OK) {
1476 					// Design problem? Why does FindInt32 touch
1477 					// 'action' at all if he can't find it??
1478 					action = kActionSet;
1479 				}
1480 				Navigator()->UpdateLocation(TargetModel(), action);
1481 			}
1482 
1483 			TrackerSettings settings;
1484 			if (settings.ShowNavigator() || settings.ShowFullPathInTitleBar())
1485 				SetPathWatchingEnabled(true);
1486 
1487 			SetSingleWindowBrowseShortcuts(settings.SingleWindowBrowse());
1488 
1489 			// Update draggable folder icon
1490 			if (fMenuBar != NULL) {
1491 				if (!TargetModel()->IsRoot() && !IsTrash()) {
1492 					// Folder icon should be visible, but in single
1493 					// window navigation, it might not be.
1494 					if (fDraggableIcon != NULL) {
1495 						IconCache::sIconCache->IconChanged(TargetModel());
1496 						if (fDraggableIcon->IsHidden())
1497 							fDraggableIcon->Show();
1498 						fDraggableIcon->Invalidate();
1499 					} else
1500 						_AddFolderIcon();
1501 				} else if (fDraggableIcon != NULL)
1502 					fDraggableIcon->Hide();
1503 			}
1504 
1505 			// Update window title
1506 			UpdateTitle();
1507 			break;
1508 		}
1509 
1510 		case B_REFS_RECEIVED:
1511 			if (Dragging()) {
1512 				// ref in this message is the target,
1513 				// the end point of the drag
1514 
1515 				entry_ref ref;
1516 				if (message->FindRef("refs", &ref) == B_OK) {
1517 					fWaitingForRefs = false;
1518 					BEntry entry(&ref, true);
1519 					// don't copy to printers dir
1520 					if (!FSIsPrintersDir(&entry)) {
1521 						if (entry.InitCheck() == B_OK
1522 							&& entry.IsDirectory()) {
1523 							Model targetModel(&entry, true, false);
1524 							BPoint dropPoint;
1525 							uint32 buttons;
1526 							PoseView()->GetMouse(&dropPoint, &buttons, true);
1527 							PoseView()->HandleDropCommon(fDragMessage,
1528 								&targetModel, NULL, PoseView(), dropPoint);
1529 						}
1530 					}
1531 				}
1532 				DragStop();
1533 			}
1534 			break;
1535 
1536 		case B_TRACKER_ADDON_MESSAGE:
1537 		{
1538 			_PassMessageToAddOn(message);
1539 			break;
1540 		}
1541 
1542 		case B_OBSERVER_NOTICE_CHANGE:
1543 		{
1544 			int32 observerWhat;
1545 			if (message->FindInt32("be:observe_change_what", &observerWhat)
1546 					== B_OK) {
1547 				TrackerSettings settings;
1548 				switch (observerWhat) {
1549 					case kWindowsShowFullPathChanged:
1550 						UpdateTitle();
1551 						if (!IsPathWatchingEnabled()
1552 							&& settings.ShowFullPathInTitleBar()) {
1553 							SetPathWatchingEnabled(true);
1554 						}
1555 						if (IsPathWatchingEnabled()
1556 							&& !(settings.ShowNavigator()
1557 								|| settings.ShowFullPathInTitleBar())) {
1558 							SetPathWatchingEnabled(false);
1559 						}
1560 						break;
1561 
1562 					case kSingleWindowBrowseChanged:
1563 						if (settings.SingleWindowBrowse()
1564 							&& !Navigator()
1565 							&& TargetModel()->IsDirectory()
1566 							&& !PoseView()->IsFilePanel()
1567 							&& !PoseView()->IsDesktopWindow()) {
1568 							fNavigator = new BNavigator(TargetModel());
1569 							fPoseContainer->GridLayout()->AddView(fNavigator,
1570 								0, 0, 2);
1571 							fNavigator->Hide();
1572 							SetPathWatchingEnabled(settings.ShowNavigator()
1573 								|| settings.ShowFullPathInTitleBar());
1574 						}
1575 
1576 						if (!settings.SingleWindowBrowse()
1577 							&& !fIsDesktop && TargetModel()->IsDesktop()) {
1578 							// Close the "Desktop" window, but not the Desktop
1579 							this->Quit();
1580 						}
1581 
1582 						SetSingleWindowBrowseShortcuts(
1583 							settings.SingleWindowBrowse());
1584 						break;
1585 
1586 					case kShowNavigatorChanged:
1587 						ShowNavigator(settings.ShowNavigator());
1588 						if (!IsPathWatchingEnabled()
1589 							&& settings.ShowNavigator()) {
1590 							SetPathWatchingEnabled(true);
1591 						}
1592 						if (IsPathWatchingEnabled()
1593 							&& !(settings.ShowNavigator()
1594 								|| settings.ShowFullPathInTitleBar())) {
1595 							SetPathWatchingEnabled(false);
1596 						}
1597 						SetSingleWindowBrowseShortcuts(
1598 							settings.SingleWindowBrowse());
1599 						break;
1600 
1601 					default:
1602 						_inherited::MessageReceived(message);
1603 						break;
1604 				}
1605 			}
1606 			break;
1607 		}
1608 
1609 		case B_NODE_MONITOR:
1610 			UpdateTitle();
1611 			break;
1612 
1613 		default:
1614 			_inherited::MessageReceived(message);
1615 			break;
1616 	}
1617 }
1618 
1619 
1620 void
1621 BContainerWindow::SetCutItem(BMenu* menu)
1622 {
1623 	BMenuItem* item;
1624 	if ((item = menu->FindItem(B_CUT)) == NULL
1625 		&& (item = menu->FindItem(kCutMoreSelectionToClipboard)) == NULL) {
1626 		return;
1627 	}
1628 
1629 	if (PoseView() != CurrentFocus())
1630 		item->SetEnabled(dynamic_cast<BTextView*>(CurrentFocus()) != NULL);
1631 	else {
1632 		if (TargetModel()->IsRoot() || TargetModel()->IsTrash()
1633 			|| TargetModel()->IsVirtualDirectory()) {
1634 			// cannot cut files in root, trash or in a virtual directory
1635 			item->SetEnabled(false);
1636 		} else {
1637 			item->SetEnabled(PoseView()->CountSelected() > 0
1638 				&& !PoseView()->SelectedVolumeIsReadOnly());
1639 		}
1640 	}
1641 
1642 	if ((modifiers() & B_SHIFT_KEY) != 0) {
1643 		item->SetLabel(B_TRANSLATE("Cut more"));
1644 		item->SetShortcut('X', B_COMMAND_KEY | B_SHIFT_KEY);
1645 		item->SetMessage(new BMessage(kCutMoreSelectionToClipboard));
1646 	} else {
1647 		item->SetLabel(B_TRANSLATE("Cut"));
1648 		item->SetShortcut('X', B_COMMAND_KEY);
1649 		item->SetMessage(new BMessage(B_CUT));
1650 	}
1651 }
1652 
1653 
1654 void
1655 BContainerWindow::SetCopyItem(BMenu* menu)
1656 {
1657 	BMenuItem* item;
1658 	if ((item = menu->FindItem(B_COPY)) == NULL
1659 		&& (item = menu->FindItem(kCopyMoreSelectionToClipboard)) == NULL) {
1660 		return;
1661 	}
1662 
1663 	if (PoseView() != CurrentFocus())
1664 		item->SetEnabled(dynamic_cast<BTextView*>(CurrentFocus()) != NULL);
1665 	else
1666 		item->SetEnabled(PoseView()->CountSelected() > 0);
1667 
1668 	if ((modifiers() & B_SHIFT_KEY) != 0) {
1669 		item->SetLabel(B_TRANSLATE("Copy more"));
1670 		item->SetShortcut('C', B_COMMAND_KEY | B_SHIFT_KEY);
1671 		item->SetMessage(new BMessage(kCopyMoreSelectionToClipboard));
1672 	} else {
1673 		item->SetLabel(B_TRANSLATE("Copy"));
1674 		item->SetShortcut('C', B_COMMAND_KEY);
1675 		item->SetMessage(new BMessage(B_COPY));
1676 	}
1677 }
1678 
1679 
1680 void
1681 BContainerWindow::SetPasteItem(BMenu* menu)
1682 {
1683 	BMenuItem* item;
1684 	if ((item = menu->FindItem(B_PASTE)) == NULL
1685 		&& (item = menu->FindItem(kPasteLinksFromClipboard)) == NULL) {
1686 		return;
1687 	}
1688 
1689 	if (PoseView() != CurrentFocus())
1690 		item->SetEnabled(dynamic_cast<BTextView*>(CurrentFocus()) != NULL);
1691 	else {
1692 		item->SetEnabled(FSClipboardHasRefs()
1693 			&& !PoseView()->TargetVolumeIsReadOnly());
1694 	}
1695 
1696 	if ((modifiers() & B_SHIFT_KEY) != 0) {
1697 		item->SetLabel(B_TRANSLATE("Paste links"));
1698 		item->SetShortcut('V', B_COMMAND_KEY | B_SHIFT_KEY);
1699 		item->SetMessage(new BMessage(kPasteLinksFromClipboard));
1700 	} else {
1701 		item->SetLabel(B_TRANSLATE("Paste"));
1702 		item->SetShortcut('V', B_COMMAND_KEY);
1703 		item->SetMessage(new BMessage(B_PASTE));
1704 	}
1705 }
1706 
1707 
1708 void
1709 BContainerWindow::SetArrangeMenu(BMenu* menu)
1710 {
1711 	BMenuItem* item;
1712 	if ((item = menu->FindItem(kCleanup)) == NULL
1713 		&& (item = menu->FindItem(kCleanupAll)) == NULL) {
1714 		return;
1715 	}
1716 
1717 	item->Menu()->SetEnabled(PoseView()->CountItems() > 0
1718 		&& (PoseView()->ViewMode() != kListMode));
1719 
1720 	BMenu* arrangeMenu;
1721 
1722 	if ((modifiers() & B_SHIFT_KEY) != 0) {
1723 		item->SetLabel(B_TRANSLATE("Clean up all"));
1724 		item->SetShortcut('K', B_COMMAND_KEY | B_SHIFT_KEY);
1725 		item->SetMessage(new BMessage(kCleanupAll));
1726 		arrangeMenu = item->Menu();
1727 	} else {
1728 		item->SetLabel(B_TRANSLATE("Clean up"));
1729 		item->SetShortcut('K', B_COMMAND_KEY);
1730 		item->SetMessage(new BMessage(kCleanup));
1731 		arrangeMenu = item->Menu();
1732 	}
1733 
1734 	MarkArrangeByMenu(arrangeMenu);
1735 }
1736 
1737 
1738 void
1739 BContainerWindow::SetCloseItem(BMenu* menu)
1740 {
1741 	BMenuItem* item;
1742 	if ((item = menu->FindItem(B_QUIT_REQUESTED)) == NULL
1743 		&& (item = menu->FindItem(kCloseAllWindows)) == NULL) {
1744 		return;
1745 	}
1746 
1747 	if ((modifiers() & B_SHIFT_KEY) != 0) {
1748 		item->SetLabel(B_TRANSLATE("Close all"));
1749 		item->SetShortcut('W', B_COMMAND_KEY | B_SHIFT_KEY);
1750 		item->SetTarget(be_app);
1751 		item->SetMessage(new BMessage(kCloseAllWindows));
1752 	} else {
1753 		item->SetLabel(B_TRANSLATE("Close"));
1754 		item->SetShortcut('W', B_COMMAND_KEY);
1755 		item->SetTarget(this);
1756 		item->SetMessage(new BMessage(B_QUIT_REQUESTED));
1757 	}
1758 }
1759 
1760 
1761 bool
1762 BContainerWindow::IsShowing(const node_ref* node) const
1763 {
1764 	return PoseView()->Represents(node);
1765 }
1766 
1767 
1768 bool
1769 BContainerWindow::IsShowing(const entry_ref* entry) const
1770 {
1771 	return PoseView()->Represents(entry);
1772 }
1773 
1774 
1775 void
1776 BContainerWindow::AddMenus()
1777 {
1778 	fFileMenu = new BMenu(B_TRANSLATE("File"));
1779 	AddFileMenu(fFileMenu);
1780 	fMenuBar->AddItem(fFileMenu);
1781 	fWindowMenu = new BMenu(B_TRANSLATE("Window"));
1782 	fMenuBar->AddItem(fWindowMenu);
1783 	AddWindowMenu(fWindowMenu);
1784 	// just create the attribute, decide to add it later
1785 	fAttrMenu = new BMenu(B_TRANSLATE("Attributes"));
1786 	NewAttributesMenu(fAttrMenu);
1787 	PopulateArrangeByMenu(fArrangeByMenu);
1788 }
1789 
1790 
1791 void
1792 BContainerWindow::AddFileMenu(BMenu* menu)
1793 {
1794 	BMenuItem* item;
1795 
1796 	if (!PoseView()->IsFilePanel()) {
1797 		menu->AddItem(new BMenuItem(B_TRANSLATE("Find" B_UTF8_ELLIPSIS),
1798 			new BMessage(kFindButton), 'F'));
1799 	}
1800 
1801 	if (!TargetModel()->IsQuery() && !TargetModel()->IsVirtualDirectory()
1802 		&& !IsTrash() && !IsPrintersDir() && !TargetModel()->IsRoot()) {
1803 		if (!PoseView()->IsFilePanel()) {
1804 			TemplatesMenu* templatesMenu = new TemplatesMenu(PoseView(),
1805 				B_TRANSLATE("New"));
1806 			menu->AddItem(templatesMenu);
1807 			templatesMenu->SetTargetForItems(PoseView());
1808 		} else {
1809 			item = new BMenuItem(B_TRANSLATE("New folder"),
1810 				new BMessage(kNewFolder), 'N');
1811 			menu->AddItem(item);
1812 		}
1813 	}
1814 	menu->AddSeparatorItem();
1815 
1816 	menu->AddItem(new BMenuItem(B_TRANSLATE("Open"),
1817 		new BMessage(kOpenSelection), 'O'));
1818 	menu->AddItem(new BMenuItem(B_TRANSLATE("Get info"),
1819 		new BMessage(kGetInfo), 'I'));
1820 	menu->AddItem(new BMenuItem(B_TRANSLATE("Edit name"),
1821 		new BMessage(kEditItem), 'E'));
1822 
1823 	if (IsTrash() || InTrash()) {
1824 		menu->AddItem(new BMenuItem(B_TRANSLATE("Restore"),
1825 			new BMessage(kRestoreFromTrash)));
1826 		if (IsTrash()) {
1827 			// add as first item in menu
1828 			menu->AddItem(new BMenuItem(B_TRANSLATE("Empty Trash"),
1829 				new BMessage(kEmptyTrash)), 0);
1830 			menu->AddItem(new BSeparatorItem(), 1);
1831 		}
1832 	} else if (IsPrintersDir()) {
1833 		menu->AddItem(new BMenuItem(B_TRANSLATE("Add printer" B_UTF8_ELLIPSIS),
1834 			new BMessage(kAddPrinter), 'N'), 0);
1835 		menu->AddItem(new BSeparatorItem(), 1);
1836 		menu->AddItem(new BMenuItem(B_TRANSLATE("Make active printer"),
1837 			new BMessage(kMakeActivePrinter)));
1838 	} else if (TargetModel()->IsRoot()) {
1839 		item = new BMenuItem(B_TRANSLATE("Unmount"),
1840 			new BMessage(kUnmountVolume), 'U');
1841 		item->SetEnabled(false);
1842 		menu->AddItem(item);
1843 		menu->AddItem(new BMenuItem(
1844 			B_TRANSLATE("Mount settings" B_UTF8_ELLIPSIS),
1845 			new BMessage(kRunAutomounterSettings)));
1846 	} else {
1847 		item = new BMenuItem(B_TRANSLATE("Duplicate"),
1848 			new BMessage(kDuplicateSelection), 'D');
1849 		item->SetEnabled(PoseView()->CanMoveToTrashOrDuplicate());
1850 		menu->AddItem(item);
1851 
1852 		item = new BMenuItem(B_TRANSLATE("Move to Trash"),
1853 			new BMessage(kMoveToTrash), 'T');
1854 		item->SetEnabled(PoseView()->CanMoveToTrashOrDuplicate());
1855 		menu->AddItem(item);
1856 
1857 		menu->AddSeparatorItem();
1858 
1859 		// The "Move To", "Copy To", "Create Link" menus are inserted
1860 		// at this place, have a look at:
1861 		// BContainerWindow::SetupMoveCopyMenus()
1862 	}
1863 
1864 	BMenuItem* cutItem = NULL;
1865 	BMenuItem* copyItem = NULL;
1866 	BMenuItem* pasteItem = NULL;
1867 	if (!IsPrintersDir()) {
1868 		menu->AddSeparatorItem();
1869 
1870 		if (!TargetModel()->IsRoot()) {
1871 			cutItem = new(std::nothrow) BMenuItem(B_TRANSLATE("Cut"),
1872 				new BMessage(B_CUT), 'X');
1873 			menu->AddItem(cutItem);
1874 			copyItem = new(std::nothrow) BMenuItem(B_TRANSLATE("Copy"),
1875 				new BMessage(B_COPY), 'C');
1876 			menu->AddItem(copyItem);
1877 			pasteItem = new(std::nothrow) BMenuItem(B_TRANSLATE("Paste"),
1878 				new BMessage(B_PASTE), 'V');
1879 			menu->AddItem(pasteItem);
1880 			menu->AddSeparatorItem();
1881 
1882 			menu->AddItem(new BMenuItem(B_TRANSLATE("Identify"),
1883 				new BMessage(kIdentifyEntry)));
1884 		}
1885 		BMenu* addOnMenuItem = new BMenu(B_TRANSLATE("Add-ons"));
1886 		addOnMenuItem->SetFont(be_plain_font);
1887 		menu->AddItem(addOnMenuItem);
1888 	}
1889 
1890 	menu->SetTargetForItems(PoseView());
1891 	if (cutItem != NULL)
1892 		cutItem->SetTarget(this);
1893 
1894 	if (copyItem != NULL)
1895 		copyItem->SetTarget(this);
1896 
1897 	if (pasteItem != NULL)
1898 		pasteItem->SetTarget(this);
1899 }
1900 
1901 
1902 void
1903 BContainerWindow::AddWindowMenu(BMenu* menu)
1904 {
1905 	BMenuItem* item;
1906 
1907 	BMenu* iconSizeMenu = new BMenu(B_TRANSLATE("Icon view"));
1908 
1909 	static const uint32 kIconSizes[] = { 32, 40, 48, 64, 96, 128 };
1910 	BMessage* message;
1911 
1912 	for (uint32 i = 0; i < sizeof(kIconSizes) / sizeof(uint32); ++i) {
1913 		uint32 iconSize = kIconSizes[i];
1914 		message = new BMessage(kIconMode);
1915 		message->AddInt32("size", iconSize);
1916 		BString label;
1917 		label.SetToFormat(B_TRANSLATE_COMMENT("%" B_PRId32" × %" B_PRId32,
1918 			"The '×' is the Unicode multiplication sign U+00D7"),
1919 			iconSize, iconSize);
1920 		item = new BMenuItem(label, message);
1921 		item->SetTarget(PoseView());
1922 		iconSizeMenu->AddItem(item);
1923 	}
1924 
1925 	iconSizeMenu->AddSeparatorItem();
1926 
1927 	message = new BMessage(kIconMode);
1928 	message->AddInt32("scale", 0);
1929 	item = new BMenuItem(B_TRANSLATE("Decrease size"), message, '-');
1930 	item->SetTarget(PoseView());
1931 	iconSizeMenu->AddItem(item);
1932 
1933 	message = new BMessage(kIconMode);
1934 	message->AddInt32("scale", 1);
1935 	item = new BMenuItem(B_TRANSLATE("Increase size"), message, '+');
1936 	item->SetTarget(PoseView());
1937 	iconSizeMenu->AddItem(item);
1938 
1939 	// A sub menu where the super item can be invoked.
1940 	menu->AddItem(iconSizeMenu);
1941 	iconSizeMenu->Superitem()->SetShortcut('1', B_COMMAND_KEY);
1942 	iconSizeMenu->Superitem()->SetMessage(new BMessage(kIconMode));
1943 	iconSizeMenu->Superitem()->SetTarget(PoseView());
1944 
1945 	item = new BMenuItem(B_TRANSLATE("Mini icon view"),
1946 		new BMessage(kMiniIconMode), '2');
1947 	item->SetTarget(PoseView());
1948 	menu->AddItem(item);
1949 
1950 	item = new BMenuItem(B_TRANSLATE("List view"),
1951 		new BMessage(kListMode), '3');
1952 	item->SetTarget(PoseView());
1953 	menu->AddItem(item);
1954 
1955 	menu->AddSeparatorItem();
1956 
1957 	item = new BMenuItem(B_TRANSLATE("Resize to fit"),
1958 		new BMessage(kResizeToFit), 'Y');
1959 	item->SetTarget(this);
1960 	menu->AddItem(item);
1961 
1962 	fArrangeByMenu = new BMenu(B_TRANSLATE("Arrange by"));
1963 	menu->AddItem(fArrangeByMenu);
1964 
1965 	item = new BMenuItem(B_TRANSLATE("Select" B_UTF8_ELLIPSIS),
1966 		new BMessage(kShowSelectionWindow), 'A', B_SHIFT_KEY);
1967 	item->SetTarget(PoseView());
1968 	menu->AddItem(item);
1969 
1970 	item = new BMenuItem(B_TRANSLATE("Select all"),
1971 		new BMessage(B_SELECT_ALL), 'A');
1972 	item->SetTarget(this);
1973 	menu->AddItem(item);
1974 
1975 	item = new BMenuItem(B_TRANSLATE("Invert selection"),
1976 		new BMessage(kInvertSelection), 'S');
1977 	item->SetTarget(PoseView());
1978 	menu->AddItem(item);
1979 
1980 	if (!IsTrash()) {
1981 		item = new BMenuItem(B_TRANSLATE("Open parent"),
1982 			new BMessage(kOpenParentDir), B_UP_ARROW);
1983 		item->SetTarget(PoseView());
1984 		menu->AddItem(item);
1985 	}
1986 
1987 	item = new BMenuItem(B_TRANSLATE("Close"),
1988 		new BMessage(B_QUIT_REQUESTED), 'W');
1989 	item->SetTarget(this);
1990 	menu->AddItem(item);
1991 
1992 	item = new BMenuItem(B_TRANSLATE("Close all in workspace"),
1993 		new BMessage(kCloseAllInWorkspace), 'Q');
1994 	item->SetTarget(be_app);
1995 	menu->AddItem(item);
1996 
1997 	menu->AddSeparatorItem();
1998 
1999 	item = new BMenuItem(B_TRANSLATE("Preferences" B_UTF8_ELLIPSIS),
2000 		new BMessage(kShowSettingsWindow), ',');
2001 	item->SetTarget(be_app);
2002 	menu->AddItem(item);
2003 }
2004 
2005 
2006 void
2007 BContainerWindow::AddShortcuts()
2008 {
2009 	// add equivalents of the menu shortcuts to the menuless desktop window
2010 	ASSERT(!IsTrash());
2011 	ASSERT(!PoseView()->IsFilePanel());
2012 	ASSERT(!TargetModel()->IsQuery());
2013 	ASSERT(!TargetModel()->IsVirtualDirectory());
2014 
2015 	AddShortcut('X', B_COMMAND_KEY | B_SHIFT_KEY,
2016 		new BMessage(kCutMoreSelectionToClipboard), this);
2017 	AddShortcut('C', B_COMMAND_KEY | B_SHIFT_KEY,
2018 		new BMessage(kCopyMoreSelectionToClipboard), this);
2019 	AddShortcut('F', B_COMMAND_KEY,
2020 		new BMessage(kFindButton), PoseView());
2021 	AddShortcut('N', B_COMMAND_KEY,
2022 		new BMessage(kNewFolder), PoseView());
2023 	AddShortcut('O', B_COMMAND_KEY,
2024 		new BMessage(kOpenSelection), PoseView());
2025 	AddShortcut('I', B_COMMAND_KEY,
2026 		new BMessage(kGetInfo), PoseView());
2027 	AddShortcut('E', B_COMMAND_KEY,
2028 		new BMessage(kEditItem), PoseView());
2029 	AddShortcut('D', B_COMMAND_KEY,
2030 		new BMessage(kDuplicateSelection), PoseView());
2031 	AddShortcut('T', B_COMMAND_KEY,
2032 		new BMessage(kMoveToTrash), PoseView());
2033 	AddShortcut('K', B_COMMAND_KEY,
2034 		new BMessage(kCleanup), PoseView());
2035 	AddShortcut('A', B_COMMAND_KEY,
2036 		new BMessage(B_SELECT_ALL), PoseView());
2037 	AddShortcut('S', B_COMMAND_KEY,
2038 		new BMessage(kInvertSelection), PoseView());
2039 	AddShortcut('A', B_COMMAND_KEY | B_SHIFT_KEY,
2040 		new BMessage(kShowSelectionWindow), PoseView());
2041 	AddShortcut('G', B_COMMAND_KEY,
2042 		new BMessage(kEditQuery), PoseView());
2043 		// it is ok to add a global Edit query shortcut here, PoseView will
2044 		// filter out cases where selected pose is not a query
2045 	AddShortcut('U', B_COMMAND_KEY,
2046 		new BMessage(kUnmountVolume), PoseView());
2047 	AddShortcut(B_UP_ARROW, B_COMMAND_KEY,
2048 		new BMessage(kOpenParentDir), PoseView());
2049 	AddShortcut('O', B_COMMAND_KEY | B_CONTROL_KEY,
2050 		new BMessage(kOpenSelectionWith), PoseView());
2051 
2052 	BMessage* decreaseSize = new BMessage(kIconMode);
2053 	decreaseSize->AddInt32("scale", 0);
2054 	AddShortcut('-', B_COMMAND_KEY, decreaseSize, PoseView());
2055 
2056 	BMessage* increaseSize = new BMessage(kIconMode);
2057 	increaseSize->AddInt32("scale", 1);
2058 	AddShortcut('+', B_COMMAND_KEY, increaseSize, PoseView());
2059 }
2060 
2061 
2062 void
2063 BContainerWindow::MenusBeginning()
2064 {
2065 	if (fMenuBar == NULL)
2066 		return;
2067 
2068 	if (CurrentMessage() != NULL && CurrentMessage()->what == B_MOUSE_DOWN) {
2069 		// don't commit active pose if only a keyboard shortcut is
2070 		// invoked - this would prevent Cut/Copy/Paste from working
2071 		PoseView()->CommitActivePose();
2072 	}
2073 
2074 	// File menu
2075 	int32 selectCount = PoseView()->SelectionList()->CountItems();
2076 
2077 	SetupOpenWithMenu(fFileMenu);
2078 	SetupMoveCopyMenus(selectCount
2079 		? PoseView()->SelectionList()->FirstItem()->TargetModel()->EntryRef()
2080 		: NULL, fFileMenu);
2081 
2082 	if (TargetModel()->IsRoot()) {
2083 		BVolume boot;
2084 		BVolumeRoster().GetBootVolume(&boot);
2085 
2086 		bool ejectableVolumeSelected = false;
2087 		for (int32 index = 0; index < selectCount; index++) {
2088 			Model* model
2089 				= PoseView()->SelectionList()->ItemAt(index)->TargetModel();
2090 			if (model->IsVolume()) {
2091 				BVolume volume;
2092 				volume.SetTo(model->NodeRef()->device);
2093 				if (volume != boot) {
2094 					ejectableVolumeSelected = true;
2095 					break;
2096 				}
2097 			}
2098 		}
2099 		BMenuItem* item = fMenuBar->FindItem(kUnmountVolume);
2100 		if (item != NULL)
2101 			item->SetEnabled(ejectableVolumeSelected);
2102 	}
2103 
2104 	UpdateMenu(fMenuBar, kMenuBarContext);
2105 
2106 	AddMimeTypesToMenu(fAttrMenu);
2107 
2108 	if (IsPrintersDir())
2109 		EnableNamedMenuItem(fFileMenu, kMakeActivePrinter, selectCount == 1);
2110 }
2111 
2112 
2113 void
2114 BContainerWindow::MenusEnded()
2115 {
2116 	// when we're done we want to clear nav menus for next time
2117 	DeleteSubmenu(fNavigationItem);
2118 	DeleteSubmenu(fMoveToItem);
2119 	DeleteSubmenu(fCopyToItem);
2120 	DeleteSubmenu(fCreateLinkItem);
2121 	DeleteSubmenu(fOpenWithItem);
2122 }
2123 
2124 
2125 void
2126 BContainerWindow::SetupNavigationMenu(const entry_ref* ref, BMenu* parent)
2127 {
2128 	// start by removing nav item (and separator) from old menu
2129 	if (fNavigationItem != NULL) {
2130 		BMenu* menu = fNavigationItem->Menu();
2131 		if (menu != NULL) {
2132 			menu->RemoveItem(fNavigationItem);
2133 			BMenuItem* item = menu->RemoveItem((int32)0);
2134 			ASSERT(item != fNavigationItem);
2135 			delete item;
2136 		}
2137 	}
2138 
2139 	// if we weren't passed a ref then we're navigating this window
2140 	if (ref == NULL)
2141 		ref = TargetModel()->EntryRef();
2142 
2143 	BEntry entry;
2144 	if (entry.SetTo(ref) != B_OK)
2145 		return;
2146 
2147 	// only navigate directories and queries (check for symlink here)
2148 	Model model(&entry);
2149 	entry_ref resolvedRef;
2150 
2151 	if (model.InitCheck() != B_OK
2152 		|| (!model.IsContainer() && !model.IsSymLink())) {
2153 		return;
2154 	}
2155 
2156 	if (model.IsSymLink()) {
2157 		if (entry.SetTo(model.EntryRef(), true) != B_OK)
2158 			return;
2159 
2160 		Model resolvedModel(&entry);
2161 		if (resolvedModel.InitCheck() != B_OK || !resolvedModel.IsContainer())
2162 			return;
2163 
2164 		entry.GetRef(&resolvedRef);
2165 		ref = &resolvedRef;
2166 	}
2167 
2168 	if (fNavigationItem == NULL) {
2169 		fNavigationItem = new ModelMenuItem(&model,
2170 			new BNavMenu(model.Name(), B_REFS_RECEIVED, be_app, this));
2171 	}
2172 
2173 	// setup a navigation menu item which will dynamically load items
2174 	// as menu items are traversed
2175 	BNavMenu* navMenu = dynamic_cast<BNavMenu*>(fNavigationItem->Submenu());
2176 	navMenu->SetNavDir(ref);
2177 	fNavigationItem->SetLabel(model.Name());
2178 	fNavigationItem->SetEntry(&entry);
2179 
2180 	parent->AddItem(fNavigationItem, 0);
2181 	parent->AddItem(new BSeparatorItem(), 1);
2182 
2183 	BMessage* message = new BMessage(B_REFS_RECEIVED);
2184 	message->AddRef("refs", ref);
2185 	fNavigationItem->SetMessage(message);
2186 	fNavigationItem->SetTarget(be_app);
2187 
2188 	if (!Dragging())
2189 		parent->SetTrackingHook(NULL, NULL);
2190 }
2191 
2192 
2193 void
2194 BContainerWindow::SetupEditQueryItem(BMenu* menu)
2195 {
2196 	ASSERT(menu);
2197 	// File menu
2198 	int32 selectCount = PoseView()->CountSelected();
2199 
2200 	// add Edit query if appropriate
2201 	bool queryInSelection = false;
2202 	if (selectCount && selectCount < 100) {
2203 		// only do this for a limited number of selected poses
2204 
2205 		// if any queries selected, add an edit query menu item
2206 		for (int32 index = 0; index < selectCount; index++) {
2207 			BPose* pose = PoseView()->SelectionList()->ItemAt(index);
2208 			Model model(pose->TargetModel()->EntryRef(), true);
2209 			if (model.InitCheck() != B_OK)
2210 				continue;
2211 
2212 			if (model.IsQuery() || model.IsQueryTemplate()) {
2213 				queryInSelection = true;
2214 				break;
2215 			}
2216 		}
2217 	}
2218 
2219 	bool poseViewIsQuery = TargetModel()->IsQuery();
2220 		// if the view is a query pose view, add edit query menu item
2221 
2222 	BMenuItem* item = menu->FindItem(kEditQuery);
2223 	if (!poseViewIsQuery && !queryInSelection && item != NULL)
2224 		item->Menu()->RemoveItem(item);
2225 	else if ((poseViewIsQuery || queryInSelection) && item == NULL) {
2226 		// add edit query item after Open
2227 		item = menu->FindItem(kOpenSelection);
2228 		if (item) {
2229 			int32 itemIndex = item->Menu()->IndexOf(item);
2230 			BMenuItem* query = new BMenuItem(B_TRANSLATE("Edit query"),
2231 				new BMessage(kEditQuery), 'G');
2232 			item->Menu()->AddItem(query, itemIndex + 1);
2233 			query->SetTarget(PoseView());
2234 		}
2235 	}
2236 }
2237 
2238 
2239 void
2240 BContainerWindow::SetupOpenWithMenu(BMenu* parent)
2241 {
2242 	// start by removing nav item (and separator) from old menu
2243 	if (fOpenWithItem) {
2244 		BMenu* menu = fOpenWithItem->Menu();
2245 		if (menu != NULL)
2246 			menu->RemoveItem(fOpenWithItem);
2247 
2248 		delete fOpenWithItem;
2249 		fOpenWithItem = 0;
2250 	}
2251 
2252 	int32 selectCount = PoseView()->CountSelected();
2253 	if (selectCount <= 0) {
2254 		// no selection, nothing to open
2255 		return;
2256 	}
2257 
2258 	if (TargetModel()->IsRoot()) {
2259 		// don't add ourselves if we are root
2260 		return;
2261 	}
2262 
2263 	// ToDo:
2264 	// check if only item in selection list is the root
2265 	// and do not add if true
2266 
2267 	// add after "Open"
2268 	BMenuItem* item = parent->FindItem(kOpenSelection);
2269 
2270 	// build a list of all refs to open
2271 	BMessage message(B_REFS_RECEIVED);
2272 	for (int32 index = 0; index < selectCount; index++) {
2273 		BPose* pose = PoseView()->SelectionList()->ItemAt(index);
2274 		message.AddRef("refs", pose->TargetModel()->EntryRef());
2275 	}
2276 
2277 	// add Tracker token so that refs received recipients can script us
2278 	message.AddMessenger("TrackerViewToken", BMessenger(PoseView()));
2279 
2280 	int32 index = item->Menu()->IndexOf(item);
2281 	fOpenWithItem = new BMenuItem(
2282 		new OpenWithMenu(B_TRANSLATE("Open with" B_UTF8_ELLIPSIS),
2283 			&message, this, be_app), new BMessage(kOpenSelectionWith));
2284 	fOpenWithItem->SetTarget(PoseView());
2285 	fOpenWithItem->SetShortcut('O', B_COMMAND_KEY | B_CONTROL_KEY);
2286 
2287 	item->Menu()->AddItem(fOpenWithItem, index + 1);
2288 }
2289 
2290 
2291 void
2292 BContainerWindow::PopulateMoveCopyNavMenu(BNavMenu* navMenu, uint32 what,
2293 	const entry_ref* ref, bool addLocalOnly)
2294 {
2295 	BVolume volume;
2296 	BVolumeRoster volumeRoster;
2297 	BDirectory directory;
2298 	BEntry entry;
2299 	BPath path;
2300 	Model model;
2301 	dev_t device = ref->device;
2302 
2303 	int32 volumeCount = 0;
2304 
2305 	navMenu->RemoveItems(0, navMenu->CountItems(), true);
2306 
2307 	// count persistent writable volumes
2308 	volumeRoster.Rewind();
2309 	while (volumeRoster.GetNextVolume(&volume) == B_OK) {
2310 		if (!volume.IsReadOnly() && volume.IsPersistent())
2311 			volumeCount++;
2312 	}
2313 
2314 	// add the current folder
2315 	if (entry.SetTo(ref) == B_OK
2316 		&& entry.GetParent(&entry) == B_OK
2317 		&& model.SetTo(&entry) == B_OK) {
2318 		BNavMenu* menu = new BNavMenu(B_TRANSLATE("Current folder"), what,
2319 			this);
2320 		menu->SetNavDir(model.EntryRef());
2321 		menu->SetShowParent(true);
2322 
2323 		BMenuItem* item = new SpecialModelMenuItem(&model, menu);
2324 		item->SetMessage(new BMessage((uint32)what));
2325 
2326 		navMenu->AddItem(item);
2327 	}
2328 
2329 	// add the recent folder menu
2330 	// the "Tracker" settings directory is only used to get its icon
2331 	if (find_directory(B_USER_SETTINGS_DIRECTORY, &path) == B_OK) {
2332 		path.Append("Tracker");
2333 		if (entry.SetTo(path.Path()) == B_OK
2334 			&& model.SetTo(&entry) == B_OK) {
2335 			BMenu* menu = new RecentsMenu(B_TRANSLATE("Recent folders"),
2336 				kRecentFolders, what, this);
2337 
2338 			BMenuItem* item = new SpecialModelMenuItem(&model, menu);
2339 			item->SetMessage(new BMessage((uint32)what));
2340 
2341 			navMenu->AddItem(item);
2342 		}
2343 	}
2344 
2345 	// add Desktop
2346 	FSGetBootDeskDir(&directory);
2347 	if (directory.InitCheck() == B_OK && directory.GetEntry(&entry) == B_OK
2348 		&& model.SetTo(&entry) == B_OK) {
2349 		navMenu->AddNavDir(&model, what, this, true);
2350 			// ask NavMenu to populate submenu for us
2351 	}
2352 
2353 	// add the home dir
2354 	if (find_directory(B_USER_DIRECTORY, &path) == B_OK
2355 		&& entry.SetTo(path.Path()) == B_OK && model.SetTo(&entry) == B_OK) {
2356 		navMenu->AddNavDir(&model, what, this, true);
2357 	}
2358 
2359 	navMenu->AddSeparatorItem();
2360 
2361 	// either add all mounted volumes (for copy), or all the top-level
2362 	// directories from the same device (for move)
2363 	// ToDo: can be changed if cross-device moves are implemented
2364 
2365 	if (addLocalOnly || volumeCount < 2) {
2366 		// add volume this item lives on
2367 		if (volume.SetTo(device) == B_OK
2368 			&& volume.GetRootDirectory(&directory) == B_OK
2369 			&& directory.GetEntry(&entry) == B_OK
2370 			&& model.SetTo(&entry) == B_OK) {
2371 			navMenu->AddNavDir(&model, what, this, false);
2372 				// do not have submenu populated
2373 
2374 			navMenu->SetNavDir(model.EntryRef());
2375 		}
2376 	} else {
2377 		// add all persistent writable volumes
2378 		volumeRoster.Rewind();
2379 		while (volumeRoster.GetNextVolume(&volume) == B_OK) {
2380 			if (volume.IsReadOnly() || !volume.IsPersistent())
2381 				continue;
2382 
2383 			// add root dir
2384 			if (volume.GetRootDirectory(&directory) == B_OK
2385 				&& directory.GetEntry(&entry) == B_OK
2386 				&& model.SetTo(&entry) == B_OK) {
2387 				navMenu->AddNavDir(&model, what, this, true);
2388 					// ask NavMenu to populate submenu for us
2389 			}
2390 		}
2391 	}
2392 }
2393 
2394 
2395 void
2396 BContainerWindow::SetupMoveCopyMenus(const entry_ref* item_ref, BMenu* parent)
2397 {
2398 	if (IsTrash() || InTrash() || IsPrintersDir() || fMoveToItem == NULL
2399 		|| fCopyToItem == NULL || fCreateLinkItem == NULL
2400 		|| TargetModel()->IsRoot()) {
2401 		return;
2402 	}
2403 
2404 	// re-parent items to this menu since they're shared
2405 	BMenuItem* trash = parent->FindItem(kMoveToTrash);
2406 	int32 index = trash != NULL ? parent->IndexOf(trash) + 2 : 0;
2407 
2408 	if (fMoveToItem->Menu() != parent) {
2409 		if (fMoveToItem->Menu() != NULL)
2410 			fMoveToItem->Menu()->RemoveItem(fMoveToItem);
2411 
2412 		parent->AddItem(fMoveToItem, index++);
2413 	}
2414 
2415 	if (fCopyToItem->Menu() != parent) {
2416 		if (fCopyToItem->Menu() != NULL)
2417 			fCopyToItem->Menu()->RemoveItem(fCopyToItem);
2418 
2419 		parent->AddItem(fCopyToItem, index++);
2420 	}
2421 
2422 	if (fCreateLinkItem->Menu() != parent) {
2423 		if (fCreateLinkItem->Menu() != NULL)
2424 			fCreateLinkItem->Menu()->RemoveItem(fCreateLinkItem);
2425 
2426 		parent->AddItem(fCreateLinkItem, index);
2427 	}
2428 
2429 	// Set the "Create Link" item label here so it
2430 	// appears correctly when menus are disabled, too.
2431 	if ((modifiers() & B_SHIFT_KEY) != 0)
2432 		fCreateLinkItem->SetLabel(B_TRANSLATE("Create relative link"));
2433 	else
2434 		fCreateLinkItem->SetLabel(B_TRANSLATE("Create link"));
2435 
2436 	// only enable once the menus are built
2437 	fMoveToItem->SetEnabled(false);
2438 	fCopyToItem->SetEnabled(false);
2439 	fCreateLinkItem->SetEnabled(false);
2440 
2441 	// get ref for item which is selected
2442 	BEntry entry;
2443 	if (entry.SetTo(item_ref) != B_OK)
2444 		return;
2445 
2446 	Model tempModel(&entry);
2447 	if (tempModel.InitCheck() != B_OK)
2448 		return;
2449 
2450 	if (tempModel.IsRoot() || tempModel.IsVolume())
2451 		return;
2452 
2453 	// configure "Move to" menu item
2454 	PopulateMoveCopyNavMenu(dynamic_cast<BNavMenu*>(fMoveToItem->Submenu()),
2455 		kMoveSelectionTo, item_ref, true);
2456 
2457 	// configure "Copy to" menu item
2458 	// add all mounted volumes (except the one this item lives on)
2459 	PopulateMoveCopyNavMenu(dynamic_cast<BNavMenu*>(fCopyToItem->Submenu()),
2460 		kCopySelectionTo, item_ref, false);
2461 
2462 	// Set "Create Link" menu item message and
2463 	// add all mounted volumes (except the one this item lives on)
2464 	if ((modifiers() & B_SHIFT_KEY) != 0) {
2465 		fCreateLinkItem->SetMessage(new BMessage(kCreateRelativeLink));
2466 		PopulateMoveCopyNavMenu(dynamic_cast<BNavMenu*>
2467 				(fCreateLinkItem->Submenu()),
2468 			kCreateRelativeLink, item_ref, false);
2469 	} else {
2470 		fCreateLinkItem->SetMessage(new BMessage(kCreateLink));
2471 		PopulateMoveCopyNavMenu(dynamic_cast<BNavMenu*>
2472 				(fCreateLinkItem->Submenu()),
2473 			kCreateLink, item_ref, false);
2474 	}
2475 
2476 	fMoveToItem->SetEnabled(PoseView()->CountSelected() > 0
2477 		&& !PoseView()->SelectedVolumeIsReadOnly());
2478 	fCopyToItem->SetEnabled(PoseView()->CountSelected() > 0);
2479 	fCreateLinkItem->SetEnabled(PoseView()->CountSelected() > 0);
2480 
2481 	// Set the "Identify" item label
2482 	BMenuItem* identifyItem = parent->FindItem(kIdentifyEntry);
2483 	if (identifyItem != NULL) {
2484 		if ((modifiers() & B_SHIFT_KEY) != 0) {
2485 			identifyItem->SetLabel(B_TRANSLATE("Force identify"));
2486 			identifyItem->Message()->ReplaceBool("force", true);
2487 		} else {
2488 			identifyItem->SetLabel(B_TRANSLATE("Identify"));
2489 			identifyItem->Message()->ReplaceBool("force", false);
2490 		}
2491 	}
2492 }
2493 
2494 
2495 uint32
2496 BContainerWindow::ShowDropContextMenu(BPoint where, BPoseView* source)
2497 {
2498 	BPoint global(where);
2499 
2500 	PoseView()->ConvertToScreen(&global);
2501 	PoseView()->CommitActivePose();
2502 
2503 	// Change the "Create Link" item - allow user to
2504 	// create relative links with the Shift key down.
2505 	BMenuItem* item = fDropContextMenu->FindItem(kCreateLink);
2506 	if (item == NULL)
2507 		item = fDropContextMenu->FindItem(kCreateRelativeLink);
2508 
2509 	if (item != NULL && (modifiers() & B_SHIFT_KEY) != 0) {
2510 		item->SetLabel(B_TRANSLATE("Create relative link here"));
2511 		item->SetMessage(new BMessage(kCreateRelativeLink));
2512 	} else if (item != NULL) {
2513 		item->SetLabel(B_TRANSLATE("Create link here"));
2514 		item->SetMessage(new BMessage(kCreateLink));
2515 	}
2516 
2517 	int32 itemCount = fDropContextMenu->CountItems();
2518 	for(int32 i = 0; i < itemCount - 2; i++) {
2519 		// separator item and Cancel item are skipped
2520 		item = fDropContextMenu->ItemAt(i);
2521 		if (item == NULL)
2522 			break;
2523 
2524 		if (item->Command() == kMoveSelectionTo && source != NULL) {
2525 			item->SetEnabled(!source->SelectedVolumeIsReadOnly()
2526 				&& !PoseView()->TargetVolumeIsReadOnly());
2527 		} else
2528 			item->SetEnabled(!PoseView()->TargetVolumeIsReadOnly());
2529 	}
2530 
2531 	item = fDropContextMenu->Go(global, true, true);
2532 	if (item != NULL)
2533 		return item->Command();
2534 
2535 	return 0;
2536 }
2537 
2538 
2539 void
2540 BContainerWindow::ShowContextMenu(BPoint where, const entry_ref* ref)
2541 {
2542 	ASSERT(IsLocked());
2543 	BPoint global(where);
2544 	PoseView()->ConvertToScreen(&global);
2545 	PoseView()->CommitActivePose();
2546 
2547 	if (ref != NULL) {
2548 		// clicked on a pose, show file or volume context menu
2549 		Model model(ref);
2550 
2551 		if (model.IsTrash()) {
2552 			if (fTrashContextMenu->Window() || Dragging())
2553 				return;
2554 
2555 			DeleteSubmenu(fNavigationItem);
2556 
2557 			// selected item was trash, show the trash context menu instead
2558 
2559 			EnableNamedMenuItem(fTrashContextMenu, kEmptyTrash,
2560 				static_cast<TTracker*>(be_app)->TrashFull());
2561 
2562 			SetupNavigationMenu(ref, fTrashContextMenu);
2563 
2564 			fContextMenu = fTrashContextMenu;
2565 		} else {
2566 			bool showAsVolume = false;
2567 			bool isFilePanel = PoseView()->IsFilePanel();
2568 
2569 			if (Dragging()) {
2570 				fContextMenu = NULL;
2571 
2572 				BEntry entry;
2573 				model.GetEntry(&entry);
2574 
2575 				// only show for directories (directory, volume, root)
2576 				//
2577 				// don't show a popup for the trash or printers
2578 				// trash is handled in DeskWindow
2579 				//
2580 				// since this menu is opened asynchronously
2581 				// we need to make sure we don't open it more
2582 				// than once, the IsShowing flag is set in
2583 				// SlowContextPopup::AttachedToWindow and
2584 				// reset in DetachedFromWindow
2585 				// see the notes in SlowContextPopup::AttachedToWindow
2586 
2587 				if (!FSIsPrintersDir(&entry)
2588 					&& !fDragContextMenu->IsShowing()) {
2589 					//printf("ShowContextMenu - target is %s %i\n",
2590 					//	ref->name, IsShowing(ref));
2591 					fDragContextMenu->ClearMenu();
2592 
2593 					// in case the ref is a symlink, resolve it
2594 					// only pop open for directories
2595 					BEntry resolvedEntry(ref, true);
2596 					if (!resolvedEntry.IsDirectory())
2597 						return;
2598 
2599 					entry_ref resolvedRef;
2600 					resolvedEntry.GetRef(&resolvedRef);
2601 
2602 					// use the resolved ref for the menu
2603 					fDragContextMenu->SetNavDir(&resolvedRef);
2604 					fDragContextMenu->SetTypesList(fCachedTypesList);
2605 					fDragContextMenu->SetTarget(BMessenger(this));
2606 					BPoseView* poseView = PoseView();
2607 					if (poseView != NULL) {
2608 						BMessenger target(poseView);
2609 						fDragContextMenu->InitTrackingHook(
2610 							&BPoseView::MenuTrackingHook, &target,
2611 							fDragMessage);
2612 					}
2613 
2614 					// this is now asynchronous so that we don't
2615 					// deadlock in Window::Quit,
2616 					fDragContextMenu->Go(global);
2617 				}
2618 
2619 				return;
2620 			} else if (TargetModel()->IsRoot() || model.IsVolume()) {
2621 				fContextMenu = fVolumeContextMenu;
2622 				showAsVolume = true;
2623 			} else
2624 				fContextMenu = fFileContextMenu;
2625 
2626 			if (fContextMenu == NULL)
2627 				return;
2628 
2629 			// clean up items from last context menu
2630 			MenusEnded();
2631 
2632 			if (fContextMenu == fFileContextMenu) {
2633 				// Add all mounted volumes (except the one this item lives on.)
2634 				BNavMenu* navMenu = dynamic_cast<BNavMenu*>(
2635 					fCreateLinkItem->Submenu());
2636 				PopulateMoveCopyNavMenu(navMenu,
2637 				fCreateLinkItem->Message()->what, ref, false);
2638 			} else if (showAsVolume) {
2639 				// non-volume enable/disable copy, move, identify
2640 				EnableNamedMenuItem(fContextMenu, kDuplicateSelection, false);
2641 				EnableNamedMenuItem(fContextMenu, kMoveToTrash, false);
2642 				EnableNamedMenuItem(fContextMenu, kIdentifyEntry, false);
2643 
2644 				// volume model, enable/disable the Unmount item
2645 				bool ejectableVolumeSelected = false;
2646 
2647 				BVolume boot;
2648 				BVolumeRoster().GetBootVolume(&boot);
2649 				BVolume volume;
2650 				volume.SetTo(model.NodeRef()->device);
2651 				if (volume != boot)
2652 					ejectableVolumeSelected = true;
2653 
2654 				EnableNamedMenuItem(fContextMenu,
2655 					B_TRANSLATE("Unmount"), ejectableVolumeSelected);
2656 			}
2657 
2658 			SetupNavigationMenu(ref, fContextMenu);
2659 			if (!showAsVolume && !isFilePanel) {
2660 				SetupMoveCopyMenus(ref, fContextMenu);
2661 				SetupOpenWithMenu(fContextMenu);
2662 			}
2663 
2664 			UpdateMenu(fContextMenu, kPosePopUpContext);
2665 		}
2666 	} else if (fWindowContextMenu != NULL) {
2667 		// Repopulate desktop menu if IsDesktop
2668 		if (fIsDesktop)
2669 			RepopulateMenus();
2670 
2671 		MenusEnded();
2672 
2673 		// clicked on a window, show window context menu
2674 
2675 		SetupNavigationMenu(ref, fWindowContextMenu);
2676 		UpdateMenu(fWindowContextMenu, kWindowPopUpContext);
2677 
2678 		fContextMenu = fWindowContextMenu;
2679 	}
2680 
2681 	// context menu invalid or popup window is already open
2682 	if (fContextMenu == NULL || fContextMenu->Window() != NULL)
2683 		return;
2684 
2685 	fContextMenu->Go(global, true, true, true);
2686 	fContextMenu = NULL;
2687 }
2688 
2689 
2690 void
2691 BContainerWindow::AddFileContextMenus(BMenu* menu)
2692 {
2693 	menu->AddItem(new BMenuItem(B_TRANSLATE("Open"),
2694 		new BMessage(kOpenSelection), 'O'));
2695 	menu->AddItem(new BMenuItem(B_TRANSLATE("Get info"),
2696 		new BMessage(kGetInfo), 'I'));
2697 	menu->AddItem(new BMenuItem(B_TRANSLATE("Edit name"),
2698 		new BMessage(kEditItem), 'E'));
2699 
2700 	if (!IsTrash() && !InTrash() && !IsPrintersDir()) {
2701 		menu->AddItem(new BMenuItem(B_TRANSLATE("Duplicate"),
2702 			new BMessage(kDuplicateSelection), 'D'));
2703 	}
2704 
2705 	if (!IsTrash() && !InTrash()) {
2706 		menu->AddItem(new BMenuItem(B_TRANSLATE("Move to Trash"),
2707 			new BMessage(kMoveToTrash), 'T'));
2708 		if (!IsPrintersDir()) {
2709 			// add separator for copy to/move to items (navigation items)
2710 			menu->AddSeparatorItem();
2711 		}
2712 	} else {
2713 		menu->AddItem(new BMenuItem(B_TRANSLATE("Delete"),
2714 			new BMessage(kDelete), 0));
2715 		menu->AddItem(new BMenuItem(B_TRANSLATE("Restore"),
2716 			new BMessage(kRestoreFromTrash), 0));
2717 	}
2718 
2719 #ifdef CUT_COPY_PASTE_IN_CONTEXT_MENU
2720 	menu->AddSeparatorItem();
2721 	BMenuItem* cutItem = new BMenuItem(B_TRANSLATE("Cut"),
2722 		new BMessage(B_CUT), 'X');
2723 	menu->AddItem(cutItem);
2724 	BMenuItem* copyItem = new BMenuItem(B_TRANSLATE("Copy"),
2725 		new BMessage(B_COPY), 'C');
2726 	menu->AddItem(copyItem);
2727 	BMenuItem* pasteItem = new BMenuItem(B_TRANSLATE("Paste"),
2728 		new BMessage(B_PASTE), 'V');
2729 	menu->AddItem(pasteItem);
2730 #endif
2731 	menu->AddSeparatorItem();
2732 
2733 	BMessage* message = new BMessage(kIdentifyEntry);
2734 	message->AddBool("force", false);
2735 	menu->AddItem(new BMenuItem(B_TRANSLATE("Identify"), message));
2736 
2737 	BMenu* addOnMenuItem = new BMenu(B_TRANSLATE("Add-ons"));
2738 	addOnMenuItem->SetFont(be_plain_font);
2739 	menu->AddItem(addOnMenuItem);
2740 
2741 	// set targets as needed
2742 	menu->SetTargetForItems(PoseView());
2743 #ifdef CUT_COPY_PASTE_IN_CONTEXT_MENU
2744 	cutItem->SetTarget(this);
2745 	copyItem->SetTarget(this);
2746 	pasteItem->SetTarget(this);
2747 #endif
2748 }
2749 
2750 
2751 void
2752 BContainerWindow::AddVolumeContextMenus(BMenu* menu)
2753 {
2754 	menu->AddItem(new BMenuItem(B_TRANSLATE("Open"),
2755 		new BMessage(kOpenSelection), 'O'));
2756 	menu->AddItem(new BMenuItem(B_TRANSLATE("Get info"),
2757 		new BMessage(kGetInfo), 'I'));
2758 	menu->AddItem(new BMenuItem(B_TRANSLATE("Edit name"),
2759 		new BMessage(kEditItem), 'E'));
2760 
2761 	menu->AddSeparatorItem();
2762 	menu->AddItem(new MountMenu(B_TRANSLATE("Mount")));
2763 
2764 	BMenuItem* item = new BMenuItem(B_TRANSLATE("Unmount"),
2765 		new BMessage(kUnmountVolume), 'U');
2766 	item->SetEnabled(false);
2767 	menu->AddItem(item);
2768 	menu->AddSeparatorItem();
2769 
2770 #ifdef CUT_COPY_PASTE_IN_CONTEXT_MENU
2771 	menu->AddItem(new BMenuItem(B_TRANSLATE("Paste"),
2772 		new BMessage(B_PASTE), 'V'));
2773 	menu->AddSeparatorItem();
2774 #endif
2775 
2776 	menu->AddItem(new BMenu(B_TRANSLATE("Add-ons")));
2777 
2778 	menu->SetTargetForItems(PoseView());
2779 }
2780 
2781 
2782 void
2783 BContainerWindow::AddWindowContextMenus(BMenu* menu)
2784 {
2785 	// create context sensitive menu for empty area of window
2786 	// since we check view mode before display, this should be a radio
2787 	// mode menu
2788 
2789 	Model* targetModel = TargetModel();
2790 	ASSERT(targetModel != NULL);
2791 
2792 	bool needSeparator = true;
2793 	if (IsTrash()) {
2794 		menu->AddItem(new BMenuItem(B_TRANSLATE("Empty Trash"),
2795 			new BMessage(kEmptyTrash)));
2796 	} else if (IsPrintersDir()) {
2797 		menu->AddItem(new BMenuItem(B_TRANSLATE("Add printer" B_UTF8_ELLIPSIS),
2798 			new BMessage(kAddPrinter), 'N'));
2799 	} else if (InTrash() || targetModel->IsRoot()) {
2800 		needSeparator = false;
2801 	} else {
2802 		if (!PoseView()->IsFilePanel()) {
2803 			TemplatesMenu* templatesMenu = new TemplatesMenu(PoseView(),
2804 				B_TRANSLATE("New"));
2805 			menu->AddItem(templatesMenu);
2806 			templatesMenu->SetTargetForItems(PoseView());
2807 			templatesMenu->SetFont(be_plain_font);
2808 		} else {
2809 			BMenuItem* item = new BMenuItem(B_TRANSLATE("New folder"),
2810 				new BMessage(kNewFolder), 'N');
2811 			menu->AddItem(item);
2812 		}
2813 	}
2814 
2815 	if (needSeparator)
2816 		menu->AddSeparatorItem();
2817 
2818 #ifdef CUT_COPY_PASTE_IN_CONTEXT_MENU
2819 	BMenuItem* pasteItem = new BMenuItem(B_TRANSLATE("Paste"),
2820 		new BMessage(B_PASTE), 'V');
2821 	pasteItem->SetEnabled(FSClipboardHasRefs()
2822 		&& !PoseView()->TargetVolumeIsReadOnly());
2823 	menu->AddItem(pasteItem);
2824 	menu->AddSeparatorItem();
2825 #endif
2826 
2827 	BMenu* arrangeBy = new BMenu(B_TRANSLATE("Arrange by"));
2828 	PopulateArrangeByMenu(arrangeBy);
2829 	menu->AddItem(arrangeBy);
2830 
2831 	menu->AddItem(new BMenuItem(B_TRANSLATE("Select" B_UTF8_ELLIPSIS),
2832 		new BMessage(kShowSelectionWindow), 'A', B_SHIFT_KEY));
2833 	menu->AddItem(new BMenuItem(B_TRANSLATE("Select all"),
2834 		new BMessage(B_SELECT_ALL), 'A'));
2835 	if (!IsTrash()) {
2836 		menu->AddItem(new BMenuItem(B_TRANSLATE("Open parent"),
2837 			new BMessage(kOpenParentDir), B_UP_ARROW));
2838 	}
2839 
2840 	if (targetModel->IsRoot()) {
2841 		menu->AddSeparatorItem();
2842 		menu->AddItem(new MountMenu(B_TRANSLATE("Mount")));
2843 	}
2844 
2845 	menu->AddSeparatorItem();
2846 	BMenu* addOnMenuItem = new BMenu(B_TRANSLATE("Add-ons"));
2847 	addOnMenuItem->SetFont(be_plain_font);
2848 	menu->AddItem(addOnMenuItem);
2849 
2850 #if DEBUG
2851 	menu->AddSeparatorItem();
2852 	BMenuItem* testing = new BMenuItem("Test icon cache",
2853 		new BMessage(kTestIconCache));
2854 	menu->AddItem(testing);
2855 #endif
2856 
2857 	// target items as needed
2858 	menu->SetTargetForItems(PoseView());
2859 #ifdef CUT_COPY_PASTE_IN_CONTEXT_MENU
2860 	pasteItem->SetTarget(this);
2861 #endif
2862 }
2863 
2864 
2865 void
2866 BContainerWindow::AddDropContextMenus(BMenu* menu)
2867 {
2868 	menu->AddItem(new BMenuItem(B_TRANSLATE("Move here"),
2869 		new BMessage(kMoveSelectionTo)));
2870 	menu->AddItem(new BMenuItem(B_TRANSLATE("Copy here"),
2871 		new BMessage(kCopySelectionTo)));
2872 	menu->AddItem(new BMenuItem(B_TRANSLATE("Create link here"),
2873 		new BMessage(kCreateLink)));
2874 	menu->AddSeparatorItem();
2875 	menu->AddItem(new BMenuItem(B_TRANSLATE("Cancel"),
2876 		new BMessage(kCancelButton)));
2877 }
2878 
2879 
2880 void
2881 BContainerWindow::AddTrashContextMenus(BMenu* menu)
2882 {
2883 	// setup special trash context menu
2884 	menu->AddItem(new BMenuItem(B_TRANSLATE("Empty Trash"),
2885 		new BMessage(kEmptyTrash)));
2886 	menu->AddItem(new BMenuItem(B_TRANSLATE("Open"),
2887 		new BMessage(kOpenSelection), 'O'));
2888 	menu->AddItem(new BMenuItem(B_TRANSLATE("Get info"),
2889 		new BMessage(kGetInfo), 'I'));
2890 	menu->SetTargetForItems(PoseView());
2891 }
2892 
2893 
2894 void
2895 BContainerWindow::EachAddOn(void (*eachAddOn)(const Model*, const char*,
2896 		uint32 shortcut, uint32 modifiers, bool primary, void* context,
2897 		BContainerWindow* window, BMenu* menu),
2898 	void* passThru, BStringList& mimeTypes, BMenu* menu)
2899 {
2900 	AutoLock<LockingList<AddOnShortcut> > lock(fAddOnsList);
2901 	if (!lock.IsLocked())
2902 		return;
2903 
2904 	for (int i = fAddOnsList->CountItems() - 1; i >= 0; i--) {
2905 		struct AddOnShortcut* item = fAddOnsList->ItemAt(i);
2906 		bool primary = false;
2907 
2908 		if (mimeTypes.CountStrings() > 0) {
2909 			BFile file(item->model->EntryRef(), B_READ_ONLY);
2910 			if (file.InitCheck() == B_OK) {
2911 				BAppFileInfo info(&file);
2912 				if (info.InitCheck() == B_OK) {
2913 					bool secondary = true;
2914 
2915 					// does this add-on has types set at all?
2916 					BMessage message;
2917 					if (info.GetSupportedTypes(&message) == B_OK) {
2918 						type_code typeCode;
2919 						int32 count;
2920 						if (message.GetInfo("types", &typeCode,
2921 								&count) == B_OK) {
2922 							secondary = false;
2923 						}
2924 					}
2925 
2926 					// check all supported types if it has some set
2927 					if (!secondary) {
2928 						for (int32 i = mimeTypes.CountStrings();
2929 								!primary && i-- > 0;) {
2930 							BString type = mimeTypes.StringAt(i);
2931 							if (info.IsSupportedType(type.String())) {
2932 								BMimeType mimeType(type.String());
2933 								if (info.Supports(&mimeType))
2934 									primary = true;
2935 								else
2936 									secondary = true;
2937 							}
2938 						}
2939 					}
2940 
2941 					if (!secondary && !primary)
2942 						continue;
2943 				}
2944 			}
2945 		}
2946 		((eachAddOn)(item->model, item->model->Name(), item->key,
2947 			item->modifiers, primary, passThru, this, menu));
2948 	}
2949 }
2950 
2951 
2952 void
2953 BContainerWindow::BuildMimeTypeList(BStringList& mimeTypes)
2954 {
2955 	int32 selectCount = PoseView()->CountSelected();
2956 	if (selectCount <= 0) {
2957 		// just add the type of the current directory
2958 		AddMimeTypeString(mimeTypes, TargetModel());
2959 	} else {
2960 		_UpdateSelectionMIMEInfo();
2961 		for (int32 index = 0; index < selectCount; index++) {
2962 			BPose* pose = PoseView()->SelectionList()->ItemAt(index);
2963 			AddMimeTypeString(mimeTypes, pose->TargetModel());
2964 			// If it's a symlink, resolves it and add the Target's MimeType
2965 			if (pose->TargetModel()->IsSymLink()) {
2966 				Model* resolved = new Model(
2967 					pose->TargetModel()->EntryRef(), true, true);
2968 				if (resolved->InitCheck() == B_OK)
2969 					AddMimeTypeString(mimeTypes, resolved);
2970 
2971 				delete resolved;
2972 			}
2973 		}
2974 	}
2975 }
2976 
2977 
2978 void
2979 BContainerWindow::BuildAddOnsMenu(BMenu* parentMenu)
2980 {
2981 	BMenuItem* item = parentMenu->FindItem(B_TRANSLATE("Add-ons"));
2982 	if (parentMenu->IndexOf(item) == 0) {
2983 		// the folder of the context menu seems to be named "Add-Ons"
2984 		// so we just take the last menu item, which is correct if not
2985 		// build with debug option
2986 		item = parentMenu->ItemAt(parentMenu->CountItems() - 1);
2987 	}
2988 	if (item == NULL)
2989 		return;
2990 
2991 	BFont font; {
2992 		AutoLock<BLooper> _(parentMenu->Looper());
2993 		parentMenu->GetFont(&font);
2994 	}
2995 
2996 	BMenu* menu = item->Submenu();
2997 	if (menu == NULL)
2998 		return;
2999 
3000 	menu->SetFont(&font);
3001 
3002 	// found add-ons menu, empty it first
3003 	for (;;) {
3004 		item = menu->RemoveItem((int32)0);
3005 		if (!item)
3006 			break;
3007 		delete item;
3008 	}
3009 
3010 	BObjectList<BMenuItem> primaryList;
3011 	BObjectList<BMenuItem> secondaryList;
3012 	BStringList mimeTypes(10);
3013 	BuildMimeTypeList(mimeTypes);
3014 
3015 	AddOneAddOnParams params;
3016 	params.primaryList = &primaryList;
3017 	params.secondaryList = &secondaryList;
3018 
3019 	// build a list of the MIME types of the selected items
3020 
3021 	EachAddOn(AddOneAddOn, &params, mimeTypes, parentMenu);
3022 
3023 	primaryList.SortItems(CompareLabels);
3024 	secondaryList.SortItems(CompareLabels);
3025 
3026 	int32 count = primaryList.CountItems();
3027 	for (int32 index = 0; index < count; index++)
3028 		menu->AddItem(primaryList.ItemAt(index));
3029 
3030 	if (count > 0)
3031 		menu->AddSeparatorItem();
3032 
3033 	count = secondaryList.CountItems();
3034 	for (int32 index = 0; index < count; index++)
3035 		menu->AddItem(secondaryList.ItemAt(index));
3036 
3037 	menu->SetTargetForItems(this);
3038 }
3039 
3040 
3041 void
3042 BContainerWindow::UpdateMenu(BMenu* menu, UpdateMenuContext context)
3043 {
3044 	const int32 selectCount = PoseView()->CountSelected();
3045 	const int32 poseCount = PoseView()->CountItems();
3046 
3047 	if (context == kMenuBarContext) {
3048 		EnableNamedMenuItem(menu, kOpenSelection, selectCount > 0);
3049 		EnableNamedMenuItem(menu, kIdentifyEntry, selectCount > 0);
3050 		EnableNamedMenuItem(menu, kRestoreFromTrash, selectCount > 0);
3051 		EnableNamedMenuItem(menu, kDelete,
3052 			PoseView()->CanMoveToTrashOrDuplicate());
3053 	}
3054 
3055 	if (context == kMenuBarContext || context == kPosePopUpContext) {
3056 		if (!PoseView()->IsFilePanel())
3057 			EnableNamedMenuItem(menu, B_TRANSLATE("New"), !PoseView()->TargetVolumeIsReadOnly());
3058 		else
3059 			EnableNamedMenuItem(menu, kNewFolder, !PoseView()->TargetVolumeIsReadOnly());
3060 
3061 		SetupEditQueryItem(menu);
3062 
3063 		EnableNamedMenuItem(menu, kEditItem, PoseView()->CanEditName());
3064 		EnableNamedMenuItem(menu, kMoveToTrash,
3065 			PoseView()->CanMoveToTrashOrDuplicate());
3066 		EnableNamedMenuItem(menu, kDuplicateSelection,
3067 			PoseView()->CanMoveToTrashOrDuplicate());
3068 
3069 		SetCutItem(menu);
3070 		SetCopyItem(menu);
3071 		SetPasteItem(menu);
3072 	}
3073 
3074 	if (context == kMenuBarContext || context == kWindowPopUpContext) {
3075 		uint32 viewMode = PoseView()->ViewMode();
3076 
3077 		BMenu* iconSizeMenu = NULL;
3078 		if (BMenuItem* item = menu->FindItem(kIconMode))
3079 			iconSizeMenu = item->Submenu();
3080 
3081 		if (iconSizeMenu != NULL) {
3082 			if (viewMode == kIconMode) {
3083 				int32 iconSize = PoseView()->UnscaledIconSizeInt();
3084 				BMenuItem* item = iconSizeMenu->ItemAt(0);
3085 				for (int32 i = 0; (item = iconSizeMenu->ItemAt(i)) != NULL;
3086 						i++) {
3087 					BMessage* message = item->Message();
3088 					if (message == NULL) {
3089 						item->SetMarked(false);
3090 						continue;
3091 					}
3092 					int32 size;
3093 					if (message->FindInt32("size", &size) != B_OK)
3094 						size = -1;
3095 					item->SetMarked(iconSize == size);
3096 				}
3097 			} else {
3098 				BMenuItem* item;
3099 				for (int32 i = 0; (item = iconSizeMenu->ItemAt(i)) != NULL; i++)
3100 					item->SetMarked(false);
3101 			}
3102 		}
3103 
3104 		MarkNamedMenuItem(menu, kIconMode, viewMode == kIconMode);
3105 		MarkNamedMenuItem(menu, kListMode, viewMode == kListMode);
3106 		MarkNamedMenuItem(menu, kMiniIconMode, viewMode == kMiniIconMode);
3107 
3108 		SetCloseItem(menu);
3109 		SetArrangeMenu(menu);
3110 		SetPasteItem(menu);
3111 
3112 		BEntry entry(TargetModel()->EntryRef());
3113 		BDirectory parent;
3114 		bool parentIsRoot = (entry.GetParent(&parent) == B_OK
3115 			&& parent.GetEntry(&entry) == B_OK
3116 			&& FSIsRootDir(&entry));
3117 
3118 		EnableNamedMenuItem(menu, kOpenParentDir, !TargetModel()->IsDesktop()
3119 			&& !TargetModel()->IsRoot()
3120 			&& (!parentIsRoot
3121 				|| TrackerSettings().SingleWindowBrowse()
3122 				|| TrackerSettings().ShowDisksIcon()
3123 				|| (modifiers() & B_CONTROL_KEY) != 0));
3124 
3125 		EnableNamedMenuItem(menu, kEmptyTrash, poseCount > 0);
3126 		EnableNamedMenuItem(menu, B_SELECT_ALL, poseCount > 0);
3127 
3128 		BMenuItem* item = menu->FindItem(B_TRANSLATE("New"));
3129 		if (item != NULL) {
3130 			TemplatesMenu* templatesMenu = dynamic_cast<TemplatesMenu*>(
3131 				item->Submenu());
3132 			if (templatesMenu != NULL)
3133 				templatesMenu->UpdateMenuState();
3134 		}
3135 	}
3136 
3137 	BuildAddOnsMenu(menu);
3138 }
3139 
3140 
3141 BMessage*
3142 BContainerWindow::AddOnMessage(int32 what)
3143 {
3144 	BMessage* message = new BMessage(what);
3145 
3146 	// add selected refs to message
3147 	BObjectList<BPose>* selectionList = PoseView()->SelectionList();
3148 
3149 	int32 index = 0;
3150 	BPose* pose;
3151 	while ((pose = selectionList->ItemAt(index++)) != NULL)
3152 		message->AddRef("refs", pose->TargetModel()->EntryRef());
3153 
3154 	message->AddRef("dir_ref", TargetModel()->EntryRef());
3155 	message->AddMessenger("TrackerViewToken", BMessenger(PoseView()));
3156 
3157 	return message;
3158 }
3159 
3160 
3161 void
3162 BContainerWindow::LoadAddOn(BMessage* message)
3163 {
3164 	UpdateIfNeeded();
3165 
3166 	entry_ref addOnRef;
3167 	status_t result = message->FindRef("refs", &addOnRef);
3168 	if (result != B_OK) {
3169 		BString buffer(B_TRANSLATE("Error %error loading add-On %name."));
3170 		buffer.ReplaceFirst("%error", strerror(result));
3171 		buffer.ReplaceFirst("%name", addOnRef.name);
3172 
3173 		BAlert* alert = new BAlert("", buffer.String(), B_TRANSLATE("Cancel"),
3174 			0, 0, B_WIDTH_AS_USUAL, B_WARNING_ALERT);
3175 		alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE);
3176 		alert->Go();
3177 		return;
3178 	}
3179 
3180 	// add selected refs to message
3181 	BMessage* refs = AddOnMessage(B_REFS_RECEIVED);
3182 
3183 	LaunchInNewThread("Add-on", B_NORMAL_PRIORITY, &AddOnThread, refs,
3184 		addOnRef, *TargetModel()->EntryRef());
3185 }
3186 
3187 
3188 //	#pragma mark - BContainerWindow private methods
3189 
3190 
3191 void
3192 BContainerWindow::_UpdateSelectionMIMEInfo()
3193 {
3194 	BPose* pose;
3195 	int32 index = 0;
3196 	while ((pose = PoseView()->SelectionList()->ItemAt(index++)) != NULL) {
3197 		BString mimeType(pose->TargetModel()->MimeType());
3198 		if (!mimeType.Length() || mimeType.ICompare(B_FILE_MIMETYPE) == 0) {
3199 			pose->TargetModel()->Mimeset(true);
3200 			if (pose->TargetModel()->IsSymLink()) {
3201 				Model* resolved = new Model(pose->TargetModel()->EntryRef(),
3202 					true, true);
3203 				if (resolved->InitCheck() == B_OK) {
3204 					mimeType.SetTo(resolved->MimeType());
3205 					if (!mimeType.Length()
3206 						|| mimeType.ICompare(B_FILE_MIMETYPE) == 0) {
3207 						resolved->Mimeset(true);
3208 					}
3209 				}
3210 				delete resolved;
3211 			}
3212 		}
3213 	}
3214 }
3215 
3216 
3217 void
3218 BContainerWindow::_AddFolderIcon()
3219 {
3220 	if (fMenuBar == NULL) {
3221 		// We don't want to add the icon if there's no menubar
3222 		return;
3223 	}
3224 
3225 	float baseIconSize = be_control_look->ComposeIconSize(16).Height() + 1,
3226 		iconSize = fMenuBar->Bounds().Height() - 2;
3227 	if (iconSize < baseIconSize)
3228 		iconSize = baseIconSize;
3229 
3230 	fDraggableIcon = new(std::nothrow)
3231 		DraggableContainerIcon(BSize(iconSize - 1, iconSize - 1));
3232 	if (fDraggableIcon != NULL) {
3233 		fMenuContainer->GroupLayout()->AddView(fDraggableIcon);
3234 		fMenuBar->SetBorders(
3235 			BControlLook::B_ALL_BORDERS & ~BControlLook::B_RIGHT_BORDER);
3236 	}
3237 }
3238 
3239 
3240 void
3241 BContainerWindow::_PassMessageToAddOn(BMessage* message)
3242 {
3243 	LaunchInNewThread("Add-on-Pass-Message", B_NORMAL_PRIORITY,
3244 		&RunAddOnMessageThread, new BMessage(*message), (void*)NULL);
3245 }
3246 
3247 
3248 BMenuItem*
3249 BContainerWindow::NewAttributeMenuItem(const char* label, const char* name,
3250 	int32 type, float width, int32 align, bool editable, bool statField)
3251 {
3252 	return NewAttributeMenuItem(label, name, type, NULL, width, align,
3253 		editable, statField);
3254 }
3255 
3256 
3257 BMenuItem*
3258 BContainerWindow::NewAttributeMenuItem(const char* label, const char* name,
3259 	int32 type, const char* displayAs, float width, int32 align,
3260 	bool editable, bool statField)
3261 {
3262 	BMessage* message = new BMessage(kAttributeItem);
3263 	message->AddString("attr_name", name);
3264 	message->AddInt32("attr_type", type);
3265 	message->AddInt32("attr_hash", (int32)AttrHashString(name, (uint32)type));
3266 	message->AddFloat("attr_width", width);
3267 	message->AddInt32("attr_align", align);
3268 	if (displayAs != NULL)
3269 		message->AddString("attr_display_as", displayAs);
3270 	message->AddBool("attr_editable", editable);
3271 	message->AddBool("attr_statfield", statField);
3272 
3273 	BMenuItem* menuItem = new BMenuItem(label, message);
3274 	menuItem->SetTarget(PoseView());
3275 
3276 	return menuItem;
3277 }
3278 
3279 
3280 void
3281 BContainerWindow::NewAttributesMenu(BMenu* menu)
3282 {
3283 	ASSERT(PoseView());
3284 
3285 	BMenuItem* item;
3286 	menu->AddItem(item = new BMenuItem(B_TRANSLATE("Copy layout"),
3287 		new BMessage(kCopyAttributes)));
3288 	item->SetTarget(PoseView());
3289 	menu->AddItem(item = new BMenuItem(B_TRANSLATE("Paste layout"),
3290 		new BMessage(kPasteAttributes)));
3291 	item->SetTarget(PoseView());
3292 	menu->AddSeparatorItem();
3293 
3294 	menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Name"),
3295 		kAttrStatName, B_STRING_TYPE, 145, B_ALIGN_LEFT, true, true));
3296 
3297 	if (gLocalizedNamePreferred) {
3298 		menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Real name"),
3299 			kAttrRealName, B_STRING_TYPE, 145, B_ALIGN_LEFT, true, true));
3300 	}
3301 
3302 	menu->AddItem(NewAttributeMenuItem (B_TRANSLATE("Size"), kAttrStatSize,
3303 		B_OFF_T_TYPE, 80, B_ALIGN_RIGHT, false, true));
3304 
3305 	menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Modified"),
3306 		kAttrStatModified, B_TIME_TYPE, 150, B_ALIGN_LEFT, false, true));
3307 
3308 	menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Created"),
3309 		kAttrStatCreated, B_TIME_TYPE, 150, B_ALIGN_LEFT, false, true));
3310 
3311 	menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Kind"),
3312 		kAttrMIMEType, B_MIME_STRING_TYPE, 145, B_ALIGN_LEFT, false, false));
3313 
3314 	if (IsTrash() || InTrash()) {
3315 		menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Original name"),
3316 			kAttrOriginalPath, B_STRING_TYPE, 225, B_ALIGN_LEFT, false,
3317 			false));
3318 	} else {
3319 		menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Location"), kAttrPath,
3320 			B_STRING_TYPE, 225, B_ALIGN_LEFT, false, false));
3321 	}
3322 
3323 #ifdef OWNER_GROUP_ATTRIBUTES
3324 	menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Owner"), kAttrStatOwner,
3325 		B_STRING_TYPE, 60, B_ALIGN_LEFT, false, true));
3326 
3327 	menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Group"), kAttrStatGroup,
3328 		B_STRING_TYPE, 60, B_ALIGN_LEFT, false, true));
3329 #endif
3330 
3331 	menu->AddItem(NewAttributeMenuItem(B_TRANSLATE("Permissions"),
3332 		kAttrStatMode, B_STRING_TYPE, 80, B_ALIGN_LEFT, false, true));
3333 }
3334 
3335 
3336 void
3337 BContainerWindow::ShowAttributesMenu()
3338 {
3339 	ASSERT(fAttrMenu);
3340 	fMenuBar->AddItem(fAttrMenu);
3341 }
3342 
3343 
3344 void
3345 BContainerWindow::HideAttributesMenu()
3346 {
3347 	ASSERT(fAttrMenu);
3348 	fMenuBar->RemoveItem(fAttrMenu);
3349 }
3350 
3351 
3352 void
3353 BContainerWindow::MarkAttributesMenu()
3354 {
3355 	MarkAttributesMenu(fAttrMenu);
3356 }
3357 
3358 
3359 void
3360 BContainerWindow::MarkAttributesMenu(BMenu* menu)
3361 {
3362 	if (menu == NULL)
3363 		return;
3364 
3365 	int32 count = menu->CountItems();
3366 	for (int32 index = 0; index < count; index++) {
3367 		BMenuItem* item = menu->ItemAt(index);
3368 		int32 attrHash;
3369 		if (item->Message() != NULL) {
3370 			if (item->Message()->FindInt32("attr_hash", &attrHash) == B_OK)
3371 				item->SetMarked(PoseView()->ColumnFor((uint32)attrHash) != 0);
3372 			else
3373 				item->SetMarked(false);
3374 		}
3375 
3376 		BMenu* submenu = item->Submenu();
3377 		if (submenu != NULL) {
3378 			int32 count2 = submenu->CountItems();
3379 			for (int32 subindex = 0; subindex < count2; subindex++) {
3380 				item = submenu->ItemAt(subindex);
3381 				if (item->Message() != NULL) {
3382 					if (item->Message()->FindInt32("attr_hash", &attrHash)
3383 						== B_OK) {
3384 						item->SetMarked(PoseView()->ColumnFor((uint32)attrHash)
3385 							!= 0);
3386 					} else
3387 						item->SetMarked(false);
3388 				}
3389 			}
3390 		}
3391 	}
3392 }
3393 
3394 
3395 void
3396 BContainerWindow::MarkArrangeByMenu(BMenu* menu)
3397 {
3398 	if (menu == NULL)
3399 		return;
3400 
3401 	int32 count = menu->CountItems();
3402 	for (int32 index = 0; index < count; index++) {
3403 		BMenuItem* item = menu->ItemAt(index);
3404 		if (item->Message() != NULL) {
3405 			uint32 attrHash;
3406 			if (item->Message()->FindInt32("attr_hash",
3407 					(int32*)&attrHash) == B_OK) {
3408 				item->SetMarked(PoseView()->PrimarySort() == attrHash);
3409 			} else if (item->Command() == kArrangeReverseOrder)
3410 				item->SetMarked(PoseView()->ReverseSort());
3411 		}
3412 	}
3413 }
3414 
3415 
3416 void
3417 BContainerWindow::AddMimeTypesToMenu()
3418 {
3419 	AddMimeTypesToMenu(fAttrMenu);
3420 }
3421 
3422 
3423 // Adds a menu for a specific MIME type if it doesn't exist already.
3424 // Returns the menu, if it existed or not.
3425 BMenu*
3426 BContainerWindow::AddMimeMenu(const BMimeType& mimeType, bool isSuperType,
3427 	BMenu* menu, int32 start)
3428 {
3429 	AutoLock<BLooper> _(menu->Looper());
3430 
3431 	if (!mimeType.IsValid())
3432 		return NULL;
3433 
3434 	// Check if we already have an entry for this MIME type in the menu.
3435 	for (int32 i = start; BMenuItem* item = menu->ItemAt(i); i++) {
3436 		BMessage* message = item->Message();
3437 		if (message == NULL)
3438 			continue;
3439 
3440 		const char* type;
3441 		if (message->FindString("mimetype", &type) == B_OK
3442 			&& !strcmp(mimeType.Type(), type)) {
3443 			return item->Submenu();
3444 		}
3445 	}
3446 
3447 	BMessage attrInfo;
3448 	char description[B_MIME_TYPE_LENGTH];
3449 	const char* label = mimeType.Type();
3450 
3451 	if (!mimeType.IsInstalled())
3452 		return NULL;
3453 
3454 	// only add things to menu which have "user-visible" data
3455 	if (mimeType.GetAttrInfo(&attrInfo) != B_OK)
3456 		return NULL;
3457 
3458 	if (mimeType.GetShortDescription(description) == B_OK && description[0])
3459 		label = description;
3460 
3461 	// go through each field in meta mime and add it to a menu
3462 	BMenu* mimeMenu = NULL;
3463 	if (isSuperType) {
3464 		// If it is a supertype, we create the menu anyway as it may have
3465 		// submenus later on.
3466 		mimeMenu = new BMenu(label);
3467 		BFont font;
3468 		menu->GetFont(&font);
3469 		mimeMenu->SetFont(&font);
3470 	}
3471 
3472 	int32 index = -1;
3473 	const char* publicName;
3474 	while (attrInfo.FindString("attr:public_name", ++index, &publicName)
3475 			== B_OK) {
3476 		if (!attrInfo.FindBool("attr:viewable", index)) {
3477 			// don't add if attribute not viewable
3478 			continue;
3479 		}
3480 
3481 		int32 type;
3482 		int32 align;
3483 		int32 width;
3484 		bool editable;
3485 		const char* attrName;
3486 		if (attrInfo.FindString("attr:name", index, &attrName) != B_OK
3487 			|| attrInfo.FindInt32("attr:type", index, &type) != B_OK
3488 			|| attrInfo.FindBool("attr:editable", index, &editable) != B_OK
3489 			|| attrInfo.FindInt32("attr:width", index, &width) != B_OK
3490 			|| attrInfo.FindInt32("attr:alignment", index, &align) != B_OK)
3491 			continue;
3492 
3493 		BString displayAs;
3494 		attrInfo.FindString("attr:display_as", index, &displayAs);
3495 
3496 		if (mimeMenu == NULL) {
3497 			// do a lazy allocation of the menu
3498 			mimeMenu = new BMenu(label);
3499 			BFont font;
3500 			menu->GetFont(&font);
3501 			mimeMenu->SetFont(&font);
3502 		}
3503 		mimeMenu->AddItem(NewAttributeMenuItem(publicName, attrName, type,
3504 			displayAs.String(), width, align, editable, false));
3505 	}
3506 
3507 	if (mimeMenu == NULL)
3508 		return NULL;
3509 
3510 	BMessage* message = new BMessage(kMIMETypeItem);
3511 	message->AddString("mimetype", mimeType.Type());
3512 	menu->AddItem(new IconMenuItem(mimeMenu, message, mimeType.Type()));
3513 
3514 	return mimeMenu;
3515 }
3516 
3517 
3518 void
3519 BContainerWindow::AddMimeTypesToMenu(BMenu* menu)
3520 {
3521 	if (menu == NULL)
3522 		return;
3523 
3524 	// Remove old mime type menus
3525 	int32 start = menu->CountItems();
3526 	while (start > 0 && menu->ItemAt(start - 1)->Submenu() != NULL) {
3527 		delete menu->RemoveItem(start - 1);
3528 		start--;
3529 	}
3530 
3531 	// Add a separator item if there is none yet
3532 	if (start > 0
3533 		&& dynamic_cast<BSeparatorItem*>(menu->ItemAt(start - 1)) == NULL)
3534 		menu->AddSeparatorItem();
3535 
3536 	// Add MIME type in case we're a default query type window
3537 	BPath path;
3538 	if (TargetModel() != NULL) {
3539 		TargetModel()->GetPath(&path);
3540 		if (path.InitCheck() == B_OK
3541 			&& strstr(path.Path(), "/" kQueryTemplates "/") != NULL) {
3542 			// demangle MIME type name
3543 			BString name(TargetModel()->Name());
3544 			name.ReplaceFirst('_', '/');
3545 
3546 			PoseView()->AddMimeType(name.String());
3547 		}
3548 	}
3549 
3550 	// Add MIME type menus
3551 
3552 	int32 typeCount = PoseView()->CountMimeTypes();
3553 
3554 	for (int32 index = 0; index < typeCount; index++) {
3555 		BMimeType mimeType(PoseView()->MimeTypeAt(index));
3556 		if (mimeType.InitCheck() == B_OK) {
3557 			BMimeType superType;
3558 			mimeType.GetSupertype(&superType);
3559 			if (superType.InitCheck() == B_OK) {
3560 				BMenu* superMenu = AddMimeMenu(superType, true, menu, start);
3561 				if (superMenu != NULL) {
3562 					// We have a supertype menu.
3563 					AddMimeMenu(mimeType, false, superMenu, 0);
3564 				}
3565 			}
3566 		}
3567 	}
3568 
3569 	// remove empty super menus, promote sub-types if needed
3570 
3571 	for (int32 index = 0; index < typeCount; index++) {
3572 		BMimeType mimeType(PoseView()->MimeTypeAt(index));
3573 		BMimeType superType;
3574 		mimeType.GetSupertype(&superType);
3575 
3576 		BMenu* superMenu = AddMimeMenu(superType, true, menu, start);
3577 		if (superMenu == NULL)
3578 			continue;
3579 
3580 		int32 itemsFound = 0;
3581 		int32 menusFound = 0;
3582 		for (int32 i = 0; BMenuItem* item = superMenu->ItemAt(i); i++) {
3583 			if (item->Submenu() != NULL)
3584 				menusFound++;
3585 			else
3586 				itemsFound++;
3587 		}
3588 
3589 		if (itemsFound == 0) {
3590 			if (menusFound != 0) {
3591 				// promote types to the top level
3592 				while (BMenuItem* item = superMenu->RemoveItem((int32)0)) {
3593 					menu->AddItem(item);
3594 				}
3595 			}
3596 
3597 			menu->RemoveItem(superMenu->Superitem());
3598 			delete superMenu->Superitem();
3599 		}
3600 	}
3601 
3602 	// remove separator if it's the only item in menu
3603 	BMenuItem* item = menu->ItemAt(menu->CountItems() - 1);
3604 	if (dynamic_cast<BSeparatorItem*>(item) != NULL) {
3605 		menu->RemoveItem(item);
3606 		delete item;
3607 	}
3608 
3609 	MarkAttributesMenu(menu);
3610 }
3611 
3612 
3613 BHandler*
3614 BContainerWindow::ResolveSpecifier(BMessage* message, int32 index,
3615 	BMessage* specifier, int32 form, const char* property)
3616 {
3617 	if (strcmp(property, "Poses") == 0) {
3618 //		PRINT(("BContainerWindow::ResolveSpecifier %s\n", property));
3619 		message->PopSpecifier();
3620 		return PoseView();
3621 	}
3622 
3623 	return _inherited::ResolveSpecifier(message, index, specifier,
3624 		form, property);
3625 }
3626 
3627 
3628 PiggybackTaskLoop*
3629 BContainerWindow::DelayedTaskLoop()
3630 {
3631 	if (!fTaskLoop)
3632 		fTaskLoop = new PiggybackTaskLoop;
3633 
3634 	return fTaskLoop;
3635 }
3636 
3637 
3638 bool
3639 BContainerWindow::NeedsDefaultStateSetup()
3640 {
3641 	if (TargetModel() == NULL)
3642 		return false;
3643 
3644 	if (TargetModel()->IsRoot()) {
3645 		// don't try to set up anything if we are root
3646 		return false;
3647 	}
3648 
3649 	WindowStateNodeOpener opener(this, false);
3650 	if (opener.StreamNode() == NULL) {
3651 		// can't read state, give up
3652 		return false;
3653 	}
3654 
3655 	return !NodeHasSavedState(opener.Node());
3656 }
3657 
3658 
3659 bool
3660 BContainerWindow::DefaultStateSourceNode(const char* name, BNode* result,
3661 	bool createNew, bool createFolder)
3662 {
3663 	//PRINT(("looking for default state in tracker settings dir\n"));
3664 	BPath settingsPath;
3665 	if (FSFindTrackerSettingsDir(&settingsPath) != B_OK)
3666 		return false;
3667 
3668 	BDirectory dir(settingsPath.Path());
3669 
3670 	BPath path(settingsPath);
3671 	path.Append(name);
3672 	if (!BEntry(path.Path()).Exists()) {
3673 		if (!createNew)
3674 			return false;
3675 
3676 		BPath tmpPath(settingsPath);
3677 		for (;;) {
3678 			// deal with several levels of folders
3679 			const char* nextSlash = strchr(name, '/');
3680 			if (!nextSlash)
3681 				break;
3682 
3683 			BString tmp;
3684 			tmp.SetTo(name, nextSlash - name);
3685 			tmpPath.Append(tmp.String());
3686 
3687 			mkdir(tmpPath.Path(), 0777);
3688 
3689 			name = nextSlash + 1;
3690 			if (!name[0]) {
3691 				// can't deal with a slash at end
3692 				return false;
3693 			}
3694 		}
3695 
3696 		if (createFolder) {
3697 			if (mkdir(path.Path(), 0777) < 0)
3698 				return false;
3699 		} else {
3700 			BFile file;
3701 			if (dir.CreateFile(name, &file) != B_OK)
3702 				return false;
3703 		}
3704 	}
3705 
3706 	//PRINT(("using default state from %s\n", path.Path()));
3707 	result->SetTo(path.Path());
3708 	return result->InitCheck() == B_OK;
3709 }
3710 
3711 
3712 void
3713 BContainerWindow::SetupDefaultState()
3714 {
3715 	BNode defaultingNode;
3716 		// this is where we'll ulitimately get the state from
3717 	bool gotDefaultingNode = 0;
3718 	bool shouldStagger = false;
3719 
3720 	ASSERT(TargetModel() != NULL);
3721 
3722 	PRINT(("folder %s does not have any saved state\n", TargetModel()->Name()));
3723 
3724 	WindowStateNodeOpener opener(this, true);
3725 		// this is our destination node, whatever it is for this window
3726 	if (opener.StreamNode() == NULL)
3727 		return;
3728 
3729 	if (!TargetModel()->IsRoot()) {
3730 		BDirectory deskDir;
3731 		FSGetDeskDir(&deskDir);
3732 
3733 		// try copying state from our parent directory, unless it is the
3734 		// desktop folder
3735 		BEntry entry(TargetModel()->EntryRef());
3736 		BNode parent;
3737 		if (FSGetParentVirtualDirectoryAware(entry, parent) == B_OK
3738 			&& parent != deskDir) {
3739 			PRINT(("looking at parent for state\n"));
3740 			if (NodeHasSavedState(&parent)) {
3741 				PRINT(("got state from parent\n"));
3742 				defaultingNode = parent;
3743 				gotDefaultingNode = true;
3744 				// when getting state from parent, stagger the window
3745 				shouldStagger = true;
3746 			}
3747 		}
3748 	}
3749 
3750 	if (!gotDefaultingNode
3751 		// parent didn't have any state, use the template directory from
3752 		// tracker settings folder for what our state should be
3753 		// For simplicity we are not picking up the most recent
3754 		// changes that didn't get committed if home is still open in
3755 		// a window, that's probably not a problem; would be OK if state
3756 		// got committed after every change
3757 		&& !DefaultStateSourceNode(kDefaultFolderTemplate, &defaultingNode,
3758 			true)) {
3759 		return;
3760 	}
3761 
3762 	if (fIsDesktop) {
3763 		// don't copy over the attributes if we are the Desktop
3764 		return;
3765 	}
3766 
3767 	// copy over the attributes
3768 
3769 	// set up a filter of the attributes we want copied
3770 	const char* allowAttrs[] = {
3771 		kAttrWindowFrame,
3772 		kAttrWindowWorkspace,
3773 		kAttrViewState,
3774 		kAttrViewStateForeign,
3775 		kAttrColumns,
3776 		kAttrColumnsForeign,
3777 		0
3778 	};
3779 
3780 	// copy over attributes that apply; transform them properly, stripping
3781 	// parts that do not apply, adding a window stagger, etc.
3782 
3783 	StaggerOneParams params;
3784 	params.rectFromParent = shouldStagger;
3785 	SelectiveAttributeTransformer frameOffsetter(kAttrWindowFrame,
3786 		OffsetFrameOne, &params);
3787 	SelectiveAttributeTransformer scrollOriginCleaner(kAttrViewState,
3788 		ClearViewOriginOne, &params);
3789 
3790 	// do it
3791 	AttributeStreamMemoryNode memoryNode;
3792 	NamesToAcceptAttrFilter filter(allowAttrs);
3793 	AttributeStreamFileNode fileNode(&defaultingNode);
3794 
3795 	*opener.StreamNode() << scrollOriginCleaner << frameOffsetter
3796 		<< memoryNode << filter << fileNode;
3797 }
3798 
3799 
3800 void
3801 BContainerWindow::RestoreWindowState(AttributeStreamNode* node)
3802 {
3803 	if (node == NULL || fIsDesktop) {
3804 		// don't restore any window state if we are the Desktop
3805 		return;
3806 	}
3807 
3808 	const char* rectAttributeName;
3809 	const char* workspaceAttributeName;
3810 	if (TargetModel()->IsRoot()) {
3811 		rectAttributeName = kAttrDisksFrame;
3812 		workspaceAttributeName = kAttrDisksWorkspace;
3813 	} else {
3814 		rectAttributeName = kAttrWindowFrame;
3815 		workspaceAttributeName = kAttrWindowWorkspace;
3816 	}
3817 
3818 	BRect frame(Frame());
3819 	if (node->Read(rectAttributeName, 0, B_RECT_TYPE, sizeof(BRect), &frame)
3820 			== sizeof(BRect)) {
3821 		const float scalingFactor = be_plain_font->Size() / 12.0f;
3822 		frame.left *= scalingFactor;
3823 		frame.top *= scalingFactor;
3824 		frame.right *= scalingFactor;
3825 		frame.bottom *= scalingFactor;
3826 
3827 		MoveTo(frame.LeftTop());
3828 		ResizeTo(frame.Width(), frame.Height());
3829 	} else
3830 		sNewWindRect.OffsetBy(sWindowStaggerBy, sWindowStaggerBy);
3831 
3832 	fPreviousBounds = Bounds();
3833 
3834 	uint32 workspace;
3835 	if (((fOpenFlags & kRestoreWorkspace) != 0)
3836 		&& node->Read(workspaceAttributeName, 0, B_INT32_TYPE, sizeof(uint32),
3837 			&workspace) == sizeof(uint32))
3838 		SetWorkspaces(workspace);
3839 
3840 	if ((fOpenFlags & kIsHidden) != 0)
3841 		Minimize(true);
3842 
3843 	// restore window decor settings
3844 	int32 size = node->Contains(kAttrWindowDecor, B_RAW_TYPE);
3845 	if (size > 0) {
3846 		char buffer[size];
3847 		if (((fOpenFlags & kRestoreDecor) != 0)
3848 			&& node->Read(kAttrWindowDecor, 0, B_RAW_TYPE, size, buffer)
3849 				== size) {
3850 			BMessage decorSettings;
3851 			if (decorSettings.Unflatten(buffer) == B_OK)
3852 				SetDecoratorSettings(decorSettings);
3853 		}
3854 	}
3855 }
3856 
3857 
3858 void
3859 BContainerWindow::RestoreWindowState(const BMessage& message)
3860 {
3861 	if (fIsDesktop) {
3862 		// don't restore any window state if we are the Desktop
3863 		return;
3864 	}
3865 
3866 	const char* rectAttributeName;
3867 	const char* workspaceAttributeName;
3868 	if (TargetModel()->IsRoot()) {
3869 		rectAttributeName = kAttrDisksFrame;
3870 		workspaceAttributeName = kAttrDisksWorkspace;
3871 	} else {
3872 		rectAttributeName = kAttrWindowFrame;
3873 		workspaceAttributeName = kAttrWindowWorkspace;
3874 	}
3875 
3876 	BRect frame(Frame());
3877 	if (message.FindRect(rectAttributeName, &frame) == B_OK) {
3878 		const float scalingFactor = be_plain_font->Size() / 12.0f;
3879 		frame.left *= scalingFactor;
3880 		frame.top *= scalingFactor;
3881 		frame.right *= scalingFactor;
3882 		frame.bottom *= scalingFactor;
3883 
3884 		MoveTo(frame.LeftTop());
3885 		ResizeTo(frame.Width(), frame.Height());
3886 	} else
3887 		sNewWindRect.OffsetBy(sWindowStaggerBy, sWindowStaggerBy);
3888 
3889 	uint32 workspace;
3890 	if (((fOpenFlags & kRestoreWorkspace) != 0)
3891 		&& message.FindInt32(workspaceAttributeName,
3892 			(int32*)&workspace) == B_OK) {
3893 		SetWorkspaces(workspace);
3894 	}
3895 
3896 	if ((fOpenFlags & kIsHidden) != 0)
3897 		Minimize(true);
3898 
3899 	// restore window decor settings
3900 	BMessage decorSettings;
3901 	if (((fOpenFlags & kRestoreDecor) != 0)
3902 		&& message.FindMessage(kAttrWindowDecor, &decorSettings) == B_OK) {
3903 		SetDecoratorSettings(decorSettings);
3904 	}
3905 
3906 	fStateNeedsSaving = false;
3907 		// Undo the effect of the above MoveTo and ResizeTo calls
3908 }
3909 
3910 
3911 void
3912 BContainerWindow::SaveWindowState(AttributeStreamNode* node)
3913 {
3914 	if (fIsDesktop) {
3915 		// don't save window state if we are the Desktop
3916 		return;
3917 	}
3918 
3919 	ASSERT(node != NULL);
3920 
3921 	const char* rectAttributeName;
3922 	const char* workspaceAttributeName;
3923 	if (TargetModel() != NULL && TargetModel()->IsRoot()) {
3924 		rectAttributeName = kAttrDisksFrame;
3925 		workspaceAttributeName = kAttrDisksWorkspace;
3926 	} else {
3927 		rectAttributeName = kAttrWindowFrame;
3928 		workspaceAttributeName = kAttrWindowWorkspace;
3929 	}
3930 
3931 	// node is null if it already got deleted
3932 	BRect frame(Frame());
3933 	const float scalingFactor = be_plain_font->Size() / 12.0f;
3934 	frame.left /= scalingFactor;
3935 	frame.top /= scalingFactor;
3936 	frame.right /= scalingFactor;
3937 	frame.bottom /= scalingFactor;
3938 	node->Write(rectAttributeName, 0, B_RECT_TYPE, sizeof(BRect), &frame);
3939 
3940 	uint32 workspaces = Workspaces();
3941 	node->Write(workspaceAttributeName, 0, B_INT32_TYPE, sizeof(uint32),
3942 		&workspaces);
3943 
3944 	BMessage decorSettings;
3945 	if (GetDecoratorSettings(&decorSettings) == B_OK) {
3946 		int32 size = decorSettings.FlattenedSize();
3947 		char buffer[size];
3948 		if (decorSettings.Flatten(buffer, size) == B_OK) {
3949 			node->Write(kAttrWindowDecor, 0, B_RAW_TYPE, size, buffer);
3950 		}
3951 	}
3952 }
3953 
3954 
3955 void
3956 BContainerWindow::SaveWindowState(BMessage& message) const
3957 {
3958 	const char* rectAttributeName;
3959 	const char* workspaceAttributeName;
3960 
3961 	if (TargetModel() != NULL && TargetModel()->IsRoot()) {
3962 		rectAttributeName = kAttrDisksFrame;
3963 		workspaceAttributeName = kAttrDisksWorkspace;
3964 	} else {
3965 		rectAttributeName = kAttrWindowFrame;
3966 		workspaceAttributeName = kAttrWindowWorkspace;
3967 	}
3968 
3969 	// node is null if it already got deleted
3970 	BRect frame(Frame());
3971 	const float scalingFactor = be_plain_font->Size() / 12.0f;
3972 	frame.left /= scalingFactor;
3973 	frame.top /= scalingFactor;
3974 	frame.right /= scalingFactor;
3975 	frame.bottom /= scalingFactor;
3976 	message.AddRect(rectAttributeName, frame);
3977 	message.AddInt32(workspaceAttributeName, (int32)Workspaces());
3978 
3979 	BMessage decorSettings;
3980 	if (GetDecoratorSettings(&decorSettings) == B_OK) {
3981 		message.AddMessage(kAttrWindowDecor, &decorSettings);
3982 	}
3983 }
3984 
3985 
3986 status_t
3987 BContainerWindow::DragStart(const BMessage* dragMessage)
3988 {
3989 	if (dragMessage == NULL)
3990 		return B_ERROR;
3991 
3992 	// if already dragging, or
3993 	// if all the refs match
3994 	if (Dragging()
3995 		&& SpringLoadedFolderCompareMessages(dragMessage, fDragMessage)) {
3996 		return B_OK;
3997 	}
3998 
3999 	// cache the current drag message
4000 	// build a list of the mimetypes in the message
4001 	SpringLoadedFolderCacheDragData(dragMessage, &fDragMessage,
4002 		&fCachedTypesList);
4003 
4004 	fWaitingForRefs = true;
4005 
4006 	return B_OK;
4007 }
4008 
4009 
4010 void
4011 BContainerWindow::DragStop()
4012 {
4013 	delete fDragMessage;
4014 	fDragMessage = NULL;
4015 
4016 	delete fCachedTypesList;
4017 	fCachedTypesList = NULL;
4018 
4019 	fWaitingForRefs = false;
4020 }
4021 
4022 
4023 void
4024 BContainerWindow::ShowSelectionWindow()
4025 {
4026 	if (fSelectionWindow == NULL) {
4027 		fSelectionWindow = new SelectionWindow(this);
4028 		fSelectionWindow->Show();
4029 	} else if (fSelectionWindow->Lock()) {
4030 		// The window is already there, just bring it close
4031 		fSelectionWindow->MoveCloseToMouse();
4032 		if (fSelectionWindow->IsHidden())
4033 			fSelectionWindow->Show();
4034 
4035 		fSelectionWindow->Unlock();
4036 	}
4037 }
4038 
4039 
4040 void
4041 BContainerWindow::ShowNavigator(bool show)
4042 {
4043 	if (PoseView()->IsDesktopWindow() || !TargetModel()->IsDirectory()
4044 		|| PoseView()->IsFilePanel()) {
4045 		return;
4046 	}
4047 
4048 	if (show) {
4049 		if (Navigator() && !Navigator()->IsHidden())
4050 			return;
4051 
4052 		if (Navigator() == NULL) {
4053 			fNavigator = new BNavigator(TargetModel());
4054 			fPoseContainer->GridLayout()->AddView(fNavigator, 0, 0, 2);
4055 		}
4056 
4057 		if (Navigator()->IsHidden())
4058 			Navigator()->Show();
4059 
4060 		if (PoseView()->VScrollBar())
4061 			PoseView()->UpdateScrollRange();
4062 	} else {
4063 		if (!Navigator() || Navigator()->IsHidden())
4064 			return;
4065 
4066 		if (PoseView()->VScrollBar())
4067 			PoseView()->UpdateScrollRange();
4068 
4069 		fNavigator->Hide();
4070 	}
4071 }
4072 
4073 
4074 void
4075 BContainerWindow::SetSingleWindowBrowseShortcuts(bool enabled)
4076 {
4077 	if (PoseView()->IsDesktopWindow())
4078 		return;
4079 
4080 	if (enabled) {
4081 		if (!Navigator())
4082 			return;
4083 
4084 		RemoveShortcut(B_DOWN_ARROW, B_COMMAND_KEY | B_OPTION_KEY);
4085 		RemoveShortcut(B_UP_ARROW, B_COMMAND_KEY);
4086 		RemoveShortcut(B_UP_ARROW, B_COMMAND_KEY | B_OPTION_KEY);
4087 
4088 		AddShortcut(B_LEFT_ARROW, B_COMMAND_KEY,
4089 			new BMessage(kNavigatorCommandBackward), Navigator());
4090 		AddShortcut(B_RIGHT_ARROW, B_COMMAND_KEY,
4091 			new BMessage(kNavigatorCommandForward), Navigator());
4092 		AddShortcut(B_UP_ARROW, B_COMMAND_KEY,
4093 			new BMessage(kNavigatorCommandUp), Navigator());
4094 
4095 		AddShortcut(B_LEFT_ARROW, B_OPTION_KEY | B_COMMAND_KEY,
4096 			new BMessage(kNavigatorCommandBackward), Navigator());
4097 		AddShortcut(B_RIGHT_ARROW, B_OPTION_KEY | B_COMMAND_KEY,
4098 			new BMessage(kNavigatorCommandForward), Navigator());
4099 		AddShortcut(B_UP_ARROW, B_OPTION_KEY | B_COMMAND_KEY,
4100 			new BMessage(kNavigatorCommandUp), Navigator());
4101 		AddShortcut(B_DOWN_ARROW, B_OPTION_KEY | B_COMMAND_KEY,
4102 			new BMessage(kOpenSelection), PoseView());
4103 		AddShortcut('L', B_COMMAND_KEY,
4104 			new BMessage(kNavigatorCommandSetFocus), Navigator());
4105 
4106 	} else {
4107 		RemoveShortcut(B_LEFT_ARROW, B_COMMAND_KEY);
4108 		RemoveShortcut(B_RIGHT_ARROW, B_COMMAND_KEY);
4109 		RemoveShortcut(B_UP_ARROW, B_COMMAND_KEY);
4110 			// This is added again, below, with a new meaning.
4111 
4112 		RemoveShortcut(B_LEFT_ARROW, B_OPTION_KEY | B_COMMAND_KEY);
4113 		RemoveShortcut(B_RIGHT_ARROW, B_OPTION_KEY | B_COMMAND_KEY);
4114 		RemoveShortcut(B_UP_ARROW, B_OPTION_KEY | B_COMMAND_KEY);
4115 		RemoveShortcut(B_DOWN_ARROW, B_COMMAND_KEY | B_OPTION_KEY);
4116 			// This also changes meaning, added again below.
4117 
4118 		AddShortcut(B_DOWN_ARROW, B_COMMAND_KEY | B_OPTION_KEY,
4119 			new BMessage(kOpenSelection), PoseView());
4120 		AddShortcut(B_UP_ARROW, B_COMMAND_KEY,
4121 			new BMessage(kOpenParentDir), PoseView());
4122 			// We change the meaning from kNavigatorCommandUp
4123 			// to kOpenParentDir.
4124 		AddShortcut(B_UP_ARROW, B_COMMAND_KEY | B_OPTION_KEY,
4125 			new BMessage(kOpenParentDir), PoseView());
4126 			// command + option results in closing the parent window
4127 		RemoveShortcut('L', B_COMMAND_KEY);
4128 	}
4129 }
4130 
4131 
4132 void
4133 BContainerWindow::SetPathWatchingEnabled(bool enable)
4134 {
4135 	if (IsPathWatchingEnabled()) {
4136 		stop_watching(this);
4137 		fIsWatchingPath = false;
4138 	}
4139 
4140 	if (enable) {
4141 		if (TargetModel() != NULL) {
4142 			BEntry entry;
4143 
4144 			TargetModel()->GetEntry(&entry);
4145 			status_t err;
4146 			do {
4147 				err = entry.GetParent(&entry);
4148 				if (err != B_OK)
4149 					break;
4150 
4151 				char name[B_FILE_NAME_LENGTH];
4152 				entry.GetName(name);
4153 				if (strcmp(name, "/") == 0)
4154 					break;
4155 
4156 				node_ref ref;
4157 				entry.GetNodeRef(&ref);
4158 				watch_node(&ref, B_WATCH_NAME, this);
4159 			} while (err == B_OK);
4160 
4161 			fIsWatchingPath = err == B_OK;
4162 		} else
4163 			fIsWatchingPath = false;
4164 	}
4165 }
4166 
4167 
4168 void
4169 BContainerWindow::PulseTaskLoop()
4170 {
4171 	if (fTaskLoop)
4172 		fTaskLoop->PulseMe();
4173 }
4174 
4175 
4176 void
4177 BContainerWindow::PopulateArrangeByMenu(BMenu* menu)
4178 {
4179 	if (!fAttrMenu || !menu)
4180 		return;
4181 	// empty fArrangeByMenu...
4182 	BMenuItem* item;
4183 	while ((item = menu->RemoveItem((int32)0)) != NULL)
4184 		delete item;
4185 
4186 	int32 itemCount = fAttrMenu->CountItems();
4187 	for (int32 i = 0; i < itemCount; i++) {
4188 		item = fAttrMenu->ItemAt(i);
4189 		if (item->Command() == kAttributeItem) {
4190 			BMessage* message = new BMessage(*(item->Message()));
4191 			message->what = kArrangeBy;
4192 			BMenuItem* newItem = new BMenuItem(item->Label(), message);
4193 			newItem->SetTarget(PoseView());
4194 			menu->AddItem(newItem);
4195 		}
4196 	}
4197 
4198 	menu->AddSeparatorItem();
4199 
4200 	item = new BMenuItem(B_TRANSLATE("Reverse order"),
4201 		new BMessage(kArrangeReverseOrder));
4202 
4203 	item->SetTarget(PoseView());
4204 	menu->AddItem(item);
4205 
4206 	menu->AddSeparatorItem();
4207 
4208 	item = new BMenuItem(B_TRANSLATE("Clean up"), new BMessage(kCleanup), 'K');
4209 	item->SetTarget(PoseView());
4210 	menu->AddItem(item);
4211 }
4212 
4213 
4214 //	#pragma mark - WindowStateNodeOpener
4215 
4216 
4217 WindowStateNodeOpener::WindowStateNodeOpener(BContainerWindow* window,
4218 	bool forWriting)
4219 	:
4220 	fModelOpener(NULL),
4221 	fNode(NULL),
4222 	fStreamNode(NULL)
4223 {
4224 	if (window->TargetModel() && window->TargetModel()->IsRoot()) {
4225 		BDirectory dir;
4226 		if (FSGetDeskDir(&dir) == B_OK) {
4227 			fNode = new BDirectory(dir);
4228 			fStreamNode = new AttributeStreamFileNode(fNode);
4229 		}
4230 	} else if (window->TargetModel()){
4231 		fModelOpener = new ModelNodeLazyOpener(window->TargetModel(),
4232 			forWriting, false);
4233 		if (fModelOpener->IsOpen(forWriting)) {
4234 			fStreamNode = new AttributeStreamFileNode(
4235 				fModelOpener->TargetModel()->Node());
4236 		}
4237 	}
4238 }
4239 
4240 WindowStateNodeOpener::~WindowStateNodeOpener()
4241 {
4242 	delete fModelOpener;
4243 	delete fNode;
4244 	delete fStreamNode;
4245 }
4246 
4247 
4248 void
4249 WindowStateNodeOpener::SetTo(const BDirectory* node)
4250 {
4251 	delete fModelOpener;
4252 	delete fNode;
4253 	delete fStreamNode;
4254 
4255 	fModelOpener = NULL;
4256 	fNode = new BDirectory(*node);
4257 	fStreamNode = new AttributeStreamFileNode(fNode);
4258 }
4259 
4260 
4261 void
4262 WindowStateNodeOpener::SetTo(const BEntry* entry, bool forWriting)
4263 {
4264 	delete fModelOpener;
4265 	delete fNode;
4266 	delete fStreamNode;
4267 
4268 	fModelOpener = NULL;
4269 	fNode = new BFile(entry, (uint32)(forWriting ? O_RDWR : O_RDONLY));
4270 	fStreamNode = new AttributeStreamFileNode(fNode);
4271 }
4272 
4273 
4274 void
4275 WindowStateNodeOpener::SetTo(Model* model, bool forWriting)
4276 {
4277 	delete fModelOpener;
4278 	delete fNode;
4279 	delete fStreamNode;
4280 
4281 	fNode = NULL;
4282 	fStreamNode = NULL;
4283 	fModelOpener = new ModelNodeLazyOpener(model, forWriting, false);
4284 	if (fModelOpener->IsOpen(forWriting)) {
4285 		fStreamNode = new AttributeStreamFileNode(
4286 			fModelOpener->TargetModel()->Node());
4287 	}
4288 }
4289 
4290 
4291 AttributeStreamNode*
4292 WindowStateNodeOpener::StreamNode() const
4293 {
4294 	return fStreamNode;
4295 }
4296 
4297 
4298 BNode*
4299 WindowStateNodeOpener::Node() const
4300 {
4301 	if (!fStreamNode)
4302 		return NULL;
4303 
4304 	if (fNode)
4305 		return fNode;
4306 
4307 	return fModelOpener->TargetModel()->Node();
4308 }
4309 
4310 
4311 //	#pragma mark - BorderedView
4312 
4313 
4314 BorderedView::BorderedView()
4315 	:
4316 	BGroupView(B_VERTICAL, 0),
4317 	fEnableBorderHighlight(true)
4318 {
4319 	GroupLayout()->SetInsets(1);
4320 }
4321 
4322 
4323 void
4324 BorderedView::WindowActivated(bool active)
4325 {
4326 	BContainerWindow* window = dynamic_cast<BContainerWindow*>(Window());
4327 	if (window == NULL)
4328 		return;
4329 
4330 	if (window->PoseView()->IsFocus())
4331 		PoseViewFocused(active); // Update border color
4332 }
4333 
4334 
4335 void BorderedView::EnableBorderHighlight(bool enable)
4336 {
4337 	fEnableBorderHighlight = enable;
4338 	PoseViewFocused(false);
4339 }
4340 
4341 
4342 void
4343 BorderedView::PoseViewFocused(bool focused)
4344 {
4345 	BContainerWindow* window = dynamic_cast<BContainerWindow*>(Window());
4346 	if (window == NULL)
4347 		return;
4348 
4349 	color_which base = B_DOCUMENT_BACKGROUND_COLOR;
4350 	float tint = B_DARKEN_2_TINT;
4351 	if (focused && window->IsActive() && fEnableBorderHighlight) {
4352 		base = B_KEYBOARD_NAVIGATION_COLOR;
4353 		tint = B_NO_TINT;
4354 	}
4355 
4356 	BScrollBar* hScrollBar = window->PoseView()->HScrollBar();
4357 	if (hScrollBar != NULL)
4358 		hScrollBar->SetBorderHighlighted(focused);
4359 
4360 	BScrollBar* vScrollBar = window->PoseView()->VScrollBar();
4361 	if (vScrollBar != NULL)
4362 		vScrollBar->SetBorderHighlighted(focused);
4363 
4364 	SetViewUIColor(base, tint);
4365 	Invalidate();
4366 }
4367 
4368 
4369 void
4370 BorderedView::Pulse()
4371 {
4372 	BContainerWindow* window = dynamic_cast<BContainerWindow*>(Window());
4373 	if (window != NULL)
4374 		window->PulseTaskLoop();
4375 }
4376