xref: /haiku/src/kits/tracker/Utilities.cpp (revision e705c841d784f0035a0ef3e9e96f6e017df16681)
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 "Utilities.h"
37 
38 #include <ctype.h>
39 #include <fs_attr.h>
40 #include <fs_info.h>
41 #include <stdarg.h>
42 #include <string.h>
43 #include <stdlib.h>
44 #include <time.h>
45 
46 #include <BitmapStream.h>
47 #include <Catalog.h>
48 #include <Debug.h>
49 #include <Font.h>
50 #include <IconUtils.h>
51 #include <MenuItem.h>
52 #include <OS.h>
53 #include <PopUpMenu.h>
54 #include <Region.h>
55 #include <StorageDefs.h>
56 #include <TextView.h>
57 #include <Volume.h>
58 #include <VolumeRoster.h>
59 #include <Window.h>
60 
61 #include "Attributes.h"
62 #include "ContainerWindow.h"
63 #include "MimeTypes.h"
64 #include "Model.h"
65 #include "PoseView.h"
66 
67 
68 #ifndef _IMPEXP_BE
69 #	define _IMPEXP_BE
70 #endif
71 extern _IMPEXP_BE const uint32	LARGE_ICON_TYPE;
72 extern _IMPEXP_BE const uint32	MINI_ICON_TYPE;
73 
74 
75 FILE* logFile = NULL;
76 
77 static const float kMinSeparatorStubX = 10;
78 static const float kStubToStringSlotX = 5;
79 
80 
81 namespace BPrivate {
82 
83 const float kExactMatchScore = INFINITY;
84 
85 
86 bool gLocalizedNamePreferred;
87 
88 
89 bool
90 SecondaryMouseButtonDown(int32 modifiers, int32 buttons)
91 {
92 	return (buttons & B_SECONDARY_MOUSE_BUTTON) != 0
93 		|| ((buttons & B_PRIMARY_MOUSE_BUTTON) != 0
94 			&& (modifiers & B_CONTROL_KEY) != 0);
95 }
96 
97 
98 uint32
99 HashString(const char* string, uint32 seed)
100 {
101 	char ch;
102 	uint32 hash = seed;
103 
104 	while((ch = *string++) != 0) {
105 		hash = (hash << 7) ^ (hash >> 24);
106 		hash ^= ch;
107 	}
108 	hash ^= hash << 12;
109 
110 	return hash;
111 }
112 
113 
114 uint32
115 AttrHashString(const char* string, uint32 type)
116 {
117 	char c;
118 	uint32 hash = 0;
119 
120 	while((c = *string++) != 0) {
121 		hash = (hash << 7) ^ (hash >> 24);
122 		hash ^= c;
123 	}
124 	hash ^= hash << 12;
125 
126 	hash &= ~0xff;
127 	hash |= type;
128 
129 	return hash;
130 }
131 
132 
133 bool
134 ValidateStream(BMallocIO* stream, uint32 key, int32 version)
135 {
136 	uint32 testKey;
137 	int32 testVersion;
138 
139 	if (stream->Read(&testKey, sizeof(uint32)) <= 0
140 		|| stream->Read(&testVersion, sizeof(int32)) <= 0) {
141 		return false;
142 	}
143 
144 	return testKey == key && testVersion == version;
145 }
146 
147 
148 void
149 DisallowFilenameKeys(BTextView* textView)
150 {
151 	textView->DisallowChar('/');
152 }
153 
154 
155 void
156 DisallowMetaKeys(BTextView* textView)
157 {
158 	textView->DisallowChar(B_TAB);
159 	textView->DisallowChar(B_ESCAPE);
160 	textView->DisallowChar(B_INSERT);
161 	textView->DisallowChar(B_DELETE);
162 	textView->DisallowChar(B_HOME);
163 	textView->DisallowChar(B_END);
164 	textView->DisallowChar(B_PAGE_UP);
165 	textView->DisallowChar(B_PAGE_DOWN);
166 	textView->DisallowChar(B_FUNCTION_KEY);
167 }
168 
169 
170 PeriodicUpdatePoses::PeriodicUpdatePoses()
171 	:
172 	fPoseList(20, true)
173 {
174 	fLock = new Benaphore("PeriodicUpdatePoses");
175 }
176 
177 
178 PeriodicUpdatePoses::~PeriodicUpdatePoses()
179 {
180 	fLock->Lock();
181 	fPoseList.MakeEmpty();
182 	delete fLock;
183 }
184 
185 
186 void
187 PeriodicUpdatePoses::AddPose(BPose* pose, BPoseView* poseView,
188 	PeriodicUpdateCallback callback, void* cookie)
189 {
190 	periodic_pose* periodic = new periodic_pose;
191 	periodic->pose = pose;
192 	periodic->pose_view = poseView;
193 	periodic->callback = callback;
194 	periodic->cookie = cookie;
195 	fPoseList.AddItem(periodic);
196 }
197 
198 
199 bool
200 PeriodicUpdatePoses::RemovePose(BPose* pose, void** cookie)
201 {
202 	int32 count = fPoseList.CountItems();
203 	for (int32 index = 0; index < count; index++) {
204 		if (fPoseList.ItemAt(index)->pose == pose) {
205 			if (!fLock->Lock())
206 				return false;
207 
208 			periodic_pose* periodic = fPoseList.RemoveItemAt(index);
209 			if (cookie)
210 				*cookie = periodic->cookie;
211 			delete periodic;
212 			fLock->Unlock();
213 			return true;
214 		}
215 	}
216 
217 	return false;
218 }
219 
220 
221 void
222 PeriodicUpdatePoses::DoPeriodicUpdate(bool forceRedraw)
223 {
224 	if (!fLock->Lock())
225 		return;
226 
227 	int32 count = fPoseList.CountItems();
228 	for (int32 index = 0; index < count; index++) {
229 		periodic_pose* periodic = fPoseList.ItemAt(index);
230 		if ((periodic->callback(periodic->pose, periodic->cookie)
231 			|| forceRedraw) && periodic->pose_view->LockLooper()) {
232 			periodic->pose_view->UpdateIcon(periodic->pose);
233 			periodic->pose_view->UnlockLooper();
234 		}
235 	}
236 
237 	fLock->Unlock();
238 }
239 
240 
241 PeriodicUpdatePoses gPeriodicUpdatePoses;
242 
243 }	// namespace BPrivate
244 
245 
246 void
247 PoseInfo::EndianSwap(void* castToThis)
248 {
249 	PoseInfo* self = (PoseInfo*)castToThis;
250 
251 	PRINT(("swapping PoseInfo\n"));
252 
253 	STATIC_ASSERT(sizeof(ino_t) == sizeof(int64));
254 	self->fInitedDirectory = SwapInt64(self->fInitedDirectory);
255 	swap_data(B_POINT_TYPE, &self->fLocation, sizeof(BPoint), B_SWAP_ALWAYS);
256 
257 	// do a sanity check on the icon position
258 	if (self->fLocation.x < -20000 || self->fLocation.x > 20000
259 		|| self->fLocation.y < -20000 || self->fLocation.y > 20000) {
260 		// position out of range, force autoplcemement
261 		PRINT((" rejecting icon position out of range\n"));
262 		self->fInitedDirectory = -1LL;
263 		self->fLocation = BPoint(0, 0);
264 	}
265 }
266 
267 
268 void
269 PoseInfo::PrintToStream()
270 {
271 	PRINT(("%s, inode:%" B_PRIx64 ", location %f %f\n",
272 		fInvisible ? "hidden" : "visible",
273 		fInitedDirectory, fLocation.x, fLocation.y));
274 }
275 
276 
277 // #pragma mark - ExtendedPoseInfo
278 
279 
280 size_t
281 ExtendedPoseInfo::Size() const
282 {
283 	return sizeof(ExtendedPoseInfo) + fNumFrames * sizeof(FrameLocation);
284 }
285 
286 
287 size_t
288 ExtendedPoseInfo::Size(int32 count)
289 {
290 	return sizeof(ExtendedPoseInfo) + count * sizeof(FrameLocation);
291 }
292 
293 
294 size_t
295 ExtendedPoseInfo::SizeWithHeadroom() const
296 {
297 	return sizeof(ExtendedPoseInfo) + (fNumFrames + 1) * sizeof(FrameLocation);
298 }
299 
300 
301 size_t
302 ExtendedPoseInfo::SizeWithHeadroom(size_t oldSize)
303 {
304 	int32 count = (ssize_t)oldSize - (ssize_t)sizeof(ExtendedPoseInfo);
305 	if (count > 0)
306 		count /= sizeof(FrameLocation);
307 	else
308 		count = 0;
309 
310 	return Size(count + 1);
311 }
312 
313 
314 bool
315 ExtendedPoseInfo::HasLocationForFrame(BRect frame) const
316 {
317 	for (int32 index = 0; index < fNumFrames; index++) {
318 		if (fLocations[index].fFrame == frame)
319 			return true;
320 	}
321 
322 	return false;
323 }
324 
325 
326 BPoint
327 ExtendedPoseInfo::LocationForFrame(BRect frame) const
328 {
329 	for (int32 index = 0; index < fNumFrames; index++) {
330 		if (fLocations[index].fFrame == frame)
331 			return fLocations[index].fLocation;
332 	}
333 
334 	TRESPASS();
335 	return BPoint(0, 0);
336 }
337 
338 
339 bool
340 ExtendedPoseInfo::SetLocationForFrame(BPoint newLocation, BRect frame)
341 {
342 	for (int32 index = 0; index < fNumFrames; index++) {
343 		if (fLocations[index].fFrame == frame) {
344 			if (fLocations[index].fLocation == newLocation)
345 				return false;
346 
347 			fLocations[index].fLocation = newLocation;
348 			return true;
349 		}
350 	}
351 
352 	fLocations[fNumFrames].fFrame = frame;
353 	fLocations[fNumFrames].fLocation = newLocation;
354 	fLocations[fNumFrames].fWorkspaces = 0xffffffff;
355 	fNumFrames++;
356 
357 	return true;
358 }
359 
360 
361 void
362 ExtendedPoseInfo::EndianSwap(void* castToThis)
363 {
364 	ExtendedPoseInfo* self = (ExtendedPoseInfo *)castToThis;
365 
366 	PRINT(("swapping ExtendedPoseInfo\n"));
367 
368 	self->fWorkspaces = SwapUInt32(self->fWorkspaces);
369 	self->fNumFrames = SwapInt32(self->fNumFrames);
370 
371 	for (int32 index = 0; index < self->fNumFrames; index++) {
372 		swap_data(B_POINT_TYPE, &self->fLocations[index].fLocation,
373 			sizeof(BPoint), B_SWAP_ALWAYS);
374 
375 		if (self->fLocations[index].fLocation.x < -20000
376 			|| self->fLocations[index].fLocation.x > 20000
377 			|| self->fLocations[index].fLocation.y < -20000
378 			|| self->fLocations[index].fLocation.y > 20000) {
379 			// position out of range, force autoplcemement
380 			PRINT((" rejecting icon position out of range\n"));
381 			self->fLocations[index].fLocation = BPoint(0, 0);
382 		}
383 		swap_data(B_RECT_TYPE, &self->fLocations[index].fFrame,
384 			sizeof(BRect), B_SWAP_ALWAYS);
385 	}
386 }
387 
388 
389 void
390 ExtendedPoseInfo::PrintToStream()
391 {
392 }
393 
394 
395 // #pragma mark - OffscreenBitmap
396 
397 
398 OffscreenBitmap::OffscreenBitmap(BRect frame)
399 	:
400 	fBitmap(NULL)
401 {
402 	NewBitmap(frame);
403 }
404 
405 
406 OffscreenBitmap::OffscreenBitmap()
407 	:
408 	fBitmap(NULL)
409 {
410 }
411 
412 
413 OffscreenBitmap::~OffscreenBitmap()
414 {
415 	delete fBitmap;
416 }
417 
418 
419 void
420 OffscreenBitmap::NewBitmap(BRect bounds)
421 {
422 	delete fBitmap;
423 	fBitmap = new(std::nothrow) BBitmap(bounds, B_RGB32, true);
424 	if (fBitmap && fBitmap->Lock()) {
425 		BView* view = new BView(fBitmap->Bounds(), "", B_FOLLOW_NONE, 0);
426 		fBitmap->AddChild(view);
427 
428 		BRect clipRect = view->Bounds();
429 		BRegion newClip;
430 		newClip.Set(clipRect);
431 		view->ConstrainClippingRegion(&newClip);
432 
433 		fBitmap->Unlock();
434 	} else {
435 		delete fBitmap;
436 		fBitmap = NULL;
437 	}
438 }
439 
440 
441 BView*
442 OffscreenBitmap::BeginUsing(BRect frame)
443 {
444 	if (!fBitmap || fBitmap->Bounds() != frame)
445 		NewBitmap(frame);
446 
447 	fBitmap->Lock();
448 	return View();
449 }
450 
451 
452 void
453 OffscreenBitmap::DoneUsing()
454 {
455 	fBitmap->Unlock();
456 }
457 
458 
459 BBitmap*
460 OffscreenBitmap::Bitmap() const
461 {
462 	ASSERT(fBitmap);
463 	ASSERT(fBitmap->IsLocked());
464 	return fBitmap;
465 }
466 
467 
468 BView*
469 OffscreenBitmap::View() const
470 {
471 	ASSERT(fBitmap);
472 	return fBitmap->ChildAt(0);
473 }
474 
475 
476 // #pragma mark - BPrivate functions
477 
478 
479 namespace BPrivate {
480 
481 // Changes the alpha value of the given bitmap to create a nice
482 // horizontal fade out in the specified region.
483 // "from" is always transparent, "to" opaque.
484 void
485 FadeRGBA32Horizontal(uint32* bits, int32 width, int32 height, int32 from,
486 	int32 to)
487 {
488 	// check parameters
489 	if (width < 0 || height < 0 || from < 0 || to < 0)
490 		return;
491 
492 	float change = 1.f / (to - from);
493 	if (from > to) {
494 		int32 temp = from;
495 		from = to;
496 		to = temp;
497 	}
498 
499 	for (int32 y = 0; y < height; y++) {
500 		float alpha = change > 0 ? 0.0f : 1.0f;
501 
502 		for (int32 x = from; x <= to; x++) {
503 			if (bits[x] & 0xff000000) {
504 				uint32 a = uint32((bits[x] >> 24) * alpha);
505 				bits[x] = (bits[x] & 0x00ffffff) | (a << 24);
506 			}
507 			alpha += change;
508 		}
509 		bits += width;
510 	}
511 }
512 
513 
514 /*!	Changes the alpha value of the given bitmap to create a nice
515 	vertical fade out in the specified region.
516 	"from" is always transparent, "to" opaque.
517 */
518 void
519 FadeRGBA32Vertical(uint32* bits, int32 width, int32 height, int32 from,
520 	int32 to)
521 {
522 	// check parameters
523 	if (width < 0 || height < 0 || from < 0 || to < 0)
524 		return;
525 
526 	if (from > to)
527 		bits += width * (height - (from - to));
528 
529 	float change = 1.f / (to - from);
530 	if (from > to) {
531 		int32 temp = from;
532 		from = to;
533 		to = temp;
534 	}
535 
536 	float alpha = change > 0 ? 0.0f : 1.0f;
537 
538 	for (int32 y = from; y <= to; y++) {
539 		for (int32 x = 0; x < width; x++) {
540 			if (bits[x] & 0xff000000) {
541 				uint32 a = uint32((bits[x] >> 24) * alpha);
542 				bits[x] = (bits[x] & 0x00ffffff) | (a << 24);
543 			}
544 		}
545 		alpha += change;
546 		bits += width;
547 	}
548 }
549 
550 
551 }	// namespace BPrivate
552 
553 
554 // #pragma mark - DraggableIcon
555 
556 
557 DraggableIcon::DraggableIcon(BRect rect, const char* name,
558 	const char* mimeType, icon_size which, const BMessage* message,
559 	BMessenger target, uint32 resizingMode, uint32 flags)
560 	:
561 	BView(rect, name, resizingMode, flags),
562 	fMessage(*message),
563 	fTarget(target)
564 {
565 	fBitmap = new BBitmap(Bounds(), kDefaultIconDepth);
566 	BMimeType mime(mimeType);
567 	status_t result = mime.GetIcon(fBitmap, which);
568 	ASSERT(mime.IsValid());
569 	if (result != B_OK) {
570 		PRINT(("failed to get icon for %s, %s\n", mimeType, strerror(result)));
571 		BMimeType mime(B_FILE_MIMETYPE);
572 		ASSERT(mime.IsInstalled());
573 		mime.GetIcon(fBitmap, which);
574 	}
575 }
576 
577 
578 DraggableIcon::~DraggableIcon()
579 {
580 	delete fBitmap;
581 }
582 
583 
584 void
585 DraggableIcon::SetTarget(BMessenger target)
586 {
587 	fTarget = target;
588 }
589 
590 
591 BRect
592 DraggableIcon::PreferredRect(BPoint offset, icon_size which)
593 {
594 	BRect rect(0, 0, which - 1, which - 1);
595 	rect.OffsetTo(offset);
596 	return rect;
597 }
598 
599 
600 void
601 DraggableIcon::AttachedToWindow()
602 {
603 	AdoptParentColors();
604 }
605 
606 
607 void
608 DraggableIcon::MouseDown(BPoint point)
609 {
610 	if (!DragStarted(&fMessage))
611 		return;
612 
613 	BRect rect(Bounds());
614 	BBitmap* dragBitmap = new BBitmap(rect, B_RGBA32, true);
615 	dragBitmap->Lock();
616 	BView* view = new BView(dragBitmap->Bounds(), "", B_FOLLOW_NONE, 0);
617 	dragBitmap->AddChild(view);
618 	view->SetOrigin(0, 0);
619 	BRect clipRect(view->Bounds());
620 	BRegion newClip;
621 	newClip.Set(clipRect);
622 	view->ConstrainClippingRegion(&newClip);
623 
624 	// Transparent draw magic
625 	view->SetHighColor(0, 0, 0, 0);
626 	view->FillRect(view->Bounds());
627 	view->SetDrawingMode(B_OP_ALPHA);
628 	view->SetHighColor(0, 0, 0, 128);
629 		// set the level of transparency by value
630 	view->SetBlendingMode(B_CONSTANT_ALPHA, B_ALPHA_COMPOSITE);
631 	view->DrawBitmap(fBitmap);
632 	view->Sync();
633 	dragBitmap->Unlock();
634 	DragMessage(&fMessage, dragBitmap, B_OP_ALPHA, point, fTarget.Target(0));
635 }
636 
637 
638 bool
639 DraggableIcon::DragStarted(BMessage*)
640 {
641 	return true;
642 }
643 
644 
645 void
646 DraggableIcon::Draw(BRect)
647 {
648 	SetDrawingMode(B_OP_ALPHA);
649 	SetBlendingMode(B_PIXEL_ALPHA, B_ALPHA_OVERLAY);
650 	DrawBitmap(fBitmap);
651 }
652 
653 
654 // #pragma mark - FlickerFreeStringView
655 
656 
657 FlickerFreeStringView::FlickerFreeStringView(BRect bounds, const char* name,
658 	const char* text, uint32 resizingMode, uint32 flags)
659 	:
660 	BStringView(bounds, name, text, resizingMode, flags),
661 	fBitmap(NULL),
662 	fViewColor(ViewColor()),
663 	fLowColor(LowColor()),
664 	fOriginalBitmap(NULL)
665 {
666 }
667 
668 
669 FlickerFreeStringView::FlickerFreeStringView(BRect bounds, const char* name,
670 	const char* text, BBitmap* inBitmap, uint32 resizingMode, uint32 flags)
671 	:
672 	BStringView(bounds, name, text, resizingMode, flags),
673 	fBitmap(NULL),
674 	fViewColor(ViewColor()),
675 	fLowColor(LowColor()),
676 	fOriginalBitmap(inBitmap)
677 {
678 }
679 
680 
681 FlickerFreeStringView::~FlickerFreeStringView()
682 {
683 	delete fBitmap;
684 }
685 
686 
687 void
688 FlickerFreeStringView::Draw(BRect)
689 {
690 	BRect bounds(Bounds());
691 	if (fBitmap == NULL)
692 		fBitmap = new OffscreenBitmap(Bounds());
693 
694 	BView* offscreen = fBitmap->BeginUsing(bounds);
695 
696 	if (Parent() != NULL) {
697 		fViewColor = Parent()->ViewColor();
698 		fLowColor = Parent()->ViewColor();
699 	}
700 
701 	offscreen->SetViewColor(fViewColor);
702 	offscreen->SetHighColor(HighColor());
703 	offscreen->SetLowColor(fLowColor);
704 
705 	BFont font;
706 	GetFont(&font);
707 	offscreen->SetFont(&font);
708 
709 	offscreen->Sync();
710 	if (fOriginalBitmap != NULL)
711 		offscreen->DrawBitmap(fOriginalBitmap, Frame(), bounds);
712 	else
713 		offscreen->FillRect(bounds, B_SOLID_LOW);
714 
715 	if (Text() != NULL) {
716 		BPoint loc;
717 
718 		font_height	height;
719 		GetFontHeight(&height);
720 
721 		edge_info eInfo;
722 		switch (Alignment()) {
723 			case B_ALIGN_LEFT:
724 			case B_ALIGN_HORIZONTAL_UNSET:
725 			case B_ALIGN_USE_FULL_WIDTH:
726 			{
727 				// If the first char has a negative left edge give it
728 				// some more room by shifting that much more to the right.
729 				font.GetEdges(Text(), 1, &eInfo);
730 				loc.x = bounds.left + (2 - eInfo.left);
731 				break;
732 			}
733 
734 			case B_ALIGN_CENTER:
735 			{
736 				float width = StringWidth(Text());
737 				float center = (bounds.right - bounds.left) / 2;
738 				loc.x = center - (width/2);
739 				break;
740 			}
741 
742 			case B_ALIGN_RIGHT:
743 			{
744 				float width = StringWidth(Text());
745 				loc.x = bounds.right - width - 2;
746 				break;
747 			}
748 		}
749 		loc.y = bounds.bottom - (1 + height.descent);
750 		offscreen->DrawString(Text(), loc);
751 	}
752 	offscreen->Sync();
753 	SetDrawingMode(B_OP_COPY);
754 	DrawBitmap(fBitmap->Bitmap());
755 	fBitmap->DoneUsing();
756 }
757 
758 
759 void
760 FlickerFreeStringView::AttachedToWindow()
761 {
762 	_inherited::AttachedToWindow();
763 	if (Parent() != NULL) {
764 		fViewColor = Parent()->ViewColor();
765 		fLowColor = Parent()->ViewColor();
766 	}
767 	SetViewColor(B_TRANSPARENT_32_BIT);
768 	SetLowColor(B_TRANSPARENT_32_BIT);
769 }
770 
771 
772 void
773 FlickerFreeStringView::SetViewColor(rgb_color color)
774 {
775 	if (fViewColor != color) {
776 		fViewColor = color;
777 		Invalidate();
778 	}
779 	_inherited::SetViewColor(B_TRANSPARENT_32_BIT);
780 }
781 
782 
783 void
784 FlickerFreeStringView::SetLowColor(rgb_color color)
785 {
786 	if (fLowColor != color) {
787 		fLowColor = color;
788 		Invalidate();
789 	}
790 	_inherited::SetLowColor(B_TRANSPARENT_32_BIT);
791 }
792 
793 
794 // #pragma mark - TitledSeparatorItem
795 
796 
797 TitledSeparatorItem::TitledSeparatorItem(const char* label)
798 	:
799 	BMenuItem(label, 0)
800 {
801 	_inherited::SetEnabled(false);
802 }
803 
804 
805 TitledSeparatorItem::~TitledSeparatorItem()
806 {
807 }
808 
809 
810 void
811 TitledSeparatorItem::SetEnabled(bool)
812 {
813 	// leave disabled
814 }
815 
816 
817 void
818 TitledSeparatorItem::GetContentSize(float* width, float* height)
819 {
820 	_inherited::GetContentSize(width, height);
821 }
822 
823 
824 inline rgb_color
825 ShiftMenuBackgroundColor(float by)
826 {
827 	return tint_color(ui_color(B_MENU_BACKGROUND_COLOR), by);
828 }
829 
830 
831 void
832 TitledSeparatorItem::Draw()
833 {
834 	BRect frame(Frame());
835 
836 	BMenu* parent = Menu();
837 	ASSERT(parent != NULL);
838 
839 	menu_info minfo;
840 	get_menu_info(&minfo);
841 
842 	if (minfo.separator > 0) {
843 		frame.left += 10;
844 		frame.right -= 10;
845 	} else {
846 		frame.left += 1;
847 		frame.right -= 1;
848 	}
849 
850 	float startX = frame.left;
851 	float endX = frame.right;
852 
853 	float maxStringWidth = endX - startX - (2 * kMinSeparatorStubX
854 		+ 2 * kStubToStringSlotX);
855 
856 	// ToDo:
857 	// handle case where maxStringWidth turns out negative here
858 
859 	BString truncatedLabel(Label());
860 	parent->TruncateString(&truncatedLabel, B_TRUNCATE_END, maxStringWidth);
861 
862 	maxStringWidth = parent->StringWidth(truncatedLabel.String());
863 
864 	// first calculate the length of the stub part of the
865 	// divider line, so we can use it for secondStartX
866 	float firstEndX = ((endX - startX) - maxStringWidth) / 2
867 		- kStubToStringSlotX;
868 	if (firstEndX < 0)
869 		firstEndX = 0;
870 
871 	float secondStartX = endX - firstEndX;
872 
873 	// now finish calculating firstEndX
874 	firstEndX += startX;
875 
876 	parent->PushState();
877 
878 	int32 y = (int32) (frame.top + (frame.bottom - frame.top) / 2);
879 
880 	parent->BeginLineArray(minfo.separator == 2 ? 6 : 4);
881 	parent->AddLine(BPoint(frame.left, y), BPoint(firstEndX, y),
882 		ShiftMenuBackgroundColor(B_DARKEN_1_TINT));
883 	parent->AddLine(BPoint(secondStartX, y), BPoint(frame.right, y),
884 		ShiftMenuBackgroundColor(B_DARKEN_1_TINT));
885 
886 	if (minfo.separator == 2) {
887 		y++;
888 		frame.left++;
889 		frame.right--;
890 		parent->AddLine(BPoint(frame.left,y), BPoint(firstEndX, y),
891 			ShiftMenuBackgroundColor(B_DARKEN_1_TINT));
892 		parent->AddLine(BPoint(secondStartX,y), BPoint(frame.right, y),
893 			ShiftMenuBackgroundColor(B_DARKEN_1_TINT));
894 	}
895 	y++;
896 	if (minfo.separator == 2) {
897 		frame.left++;
898 		frame.right--;
899 	}
900 	parent->AddLine(BPoint(frame.left, y), BPoint(firstEndX, y),
901 		ShiftMenuBackgroundColor(B_DARKEN_1_TINT));
902 	parent->AddLine(BPoint(secondStartX, y), BPoint(frame.right, y),
903 		ShiftMenuBackgroundColor(B_DARKEN_1_TINT));
904 
905 	parent->EndLineArray();
906 
907 	font_height finfo;
908 	parent->GetFontHeight(&finfo);
909 
910 	parent->SetLowColor(parent->ViewColor());
911 	BPoint loc(firstEndX + kStubToStringSlotX,
912 		ContentLocation().y + finfo.ascent);
913 
914 	parent->MovePenTo(loc + BPoint(1, 1));
915 	parent->SetHighColor(ShiftMenuBackgroundColor(B_DARKEN_1_TINT));
916 	parent->DrawString(truncatedLabel.String());
917 
918 	parent->MovePenTo(loc);
919 	parent->SetHighColor(ShiftMenuBackgroundColor(B_DISABLED_LABEL_TINT));
920 	parent->DrawString(truncatedLabel.String());
921 
922 	parent->PopState();
923 }
924 
925 
926 // #pragma mark - ShortcutFilter
927 
928 
929 ShortcutFilter::ShortcutFilter(uint32 shortcutKey, uint32 shortcutModifier,
930 	uint32 shortcutWhat, BHandler* target)
931 	:
932 	BMessageFilter(B_KEY_DOWN),
933 	fShortcutKey(shortcutKey),
934 	fShortcutModifier(shortcutModifier),
935 	fShortcutWhat(shortcutWhat),
936 	fTarget(target)
937 {
938 }
939 
940 
941 filter_result
942 ShortcutFilter::Filter(BMessage* message, BHandler**)
943 {
944 	if (message->what == B_KEY_DOWN) {
945 		uint32 modifiers;
946 		uint32 rawKeyChar = 0;
947 		uint8 byte = 0;
948 		int32 key = 0;
949 
950 		if (message->FindInt32("modifiers", (int32*)&modifiers) != B_OK
951 			|| message->FindInt32("raw_char", (int32*)&rawKeyChar) != B_OK
952 			|| message->FindInt8("byte", (int8*)&byte) != B_OK
953 			|| message->FindInt32("key", &key) != B_OK) {
954 			return B_DISPATCH_MESSAGE;
955 		}
956 
957 		modifiers &= B_SHIFT_KEY | B_COMMAND_KEY | B_CONTROL_KEY
958 			| B_OPTION_KEY | B_MENU_KEY;
959 			// strip caps lock, etc.
960 
961 		if (modifiers == fShortcutModifier && rawKeyChar == fShortcutKey) {
962 			fTarget->Looper()->PostMessage(fShortcutWhat, fTarget);
963 			return B_SKIP_MESSAGE;
964 		}
965 	}
966 
967 	// let others deal with this
968 	return B_DISPATCH_MESSAGE;
969 }
970 
971 
972 // #pragma mark - BPrivate functions
973 
974 
975 namespace BPrivate {
976 
977 void
978 EmbedUniqueVolumeInfo(BMessage* message, const BVolume* volume)
979 {
980 	BDirectory rootDirectory;
981 	time_t created;
982 	fs_info info;
983 
984 	if (volume->GetRootDirectory(&rootDirectory) == B_OK
985 		&& rootDirectory.GetCreationTime(&created) == B_OK
986 		&& fs_stat_dev(volume->Device(), &info) == 0) {
987 		message->AddInt64("creationDate", created);
988 		message->AddInt64("capacity", volume->Capacity());
989 		message->AddString("deviceName", info.device_name);
990 		message->AddString("volumeName", info.volume_name);
991 		message->AddString("fshName", info.fsh_name);
992 	}
993 }
994 
995 
996 status_t
997 MatchArchivedVolume(BVolume* volume, const BMessage* message, int32 index)
998 {
999 	int64 created64;
1000 	off_t capacity;
1001 
1002 	if (message->FindInt64("creationDate", index, &created64) != B_OK) {
1003 		int32 created32;
1004 		if (message->FindInt32("creationDate", index, &created32) != B_OK)
1005 			return B_ERROR;
1006 		created64 = created32;
1007 	}
1008 
1009 	time_t created = created64;
1010 
1011 	if (message->FindInt64("capacity", index, &capacity) != B_OK)
1012 		return B_ERROR;
1013 
1014 	BVolumeRoster roster;
1015 	BVolume tempVolume;
1016 	BString deviceName;
1017 	BString volumeName;
1018 	BString fshName;
1019 
1020 	if (message->FindString("deviceName", &deviceName) == B_OK
1021 		&& message->FindString("volumeName", &volumeName) == B_OK
1022 		&& message->FindString("fshName", &fshName) == B_OK) {
1023 		// New style volume identifiers: We have a couple of characteristics,
1024 		// and compute a score from them. The volume with the greatest score
1025 		// (if over a certain threshold) is the one we're looking for. We
1026 		// pick the first volume, in case there is more than one with the
1027 		// same score.
1028 		dev_t foundDevice = -1;
1029 		int foundScore = -1;
1030 		roster.Rewind();
1031 		while (roster.GetNextVolume(&tempVolume) == B_OK) {
1032 			if (tempVolume.IsPersistent() && tempVolume.KnowsQuery()) {
1033 				// get creation time and fs_info
1034 				BDirectory root;
1035 				tempVolume.GetRootDirectory(&root);
1036 				time_t cmpCreated;
1037 				fs_info info;
1038 				if (root.GetCreationTime(&cmpCreated) == B_OK
1039 					&& fs_stat_dev(tempVolume.Device(), &info) == 0) {
1040 					// compute the score
1041 					int score = 0;
1042 
1043 					// creation time
1044 					if (created == cmpCreated)
1045 						score += 5;
1046 
1047 					// capacity
1048 					if (capacity == tempVolume.Capacity())
1049 						score += 4;
1050 
1051 					// device name
1052 					if (deviceName == info.device_name)
1053 						score += 3;
1054 
1055 					// volume name
1056 					if (volumeName == info.volume_name)
1057 						score += 2;
1058 
1059 					// fsh name
1060 					if (fshName == info.fsh_name)
1061 						score += 1;
1062 
1063 					// check score
1064 					if (score >= 9 && score > foundScore) {
1065 						foundDevice = tempVolume.Device();
1066 						foundScore = score;
1067 					}
1068 				}
1069 			}
1070 		}
1071 		if (foundDevice >= 0)
1072 			return volume->SetTo(foundDevice);
1073 	} else {
1074 		// Old style volume identifiers: We have only creation time and
1075 		// capacity. Both must match.
1076 		roster.Rewind();
1077 		while (roster.GetNextVolume(&tempVolume) == B_OK) {
1078 			if (tempVolume.IsPersistent() && tempVolume.KnowsQuery()) {
1079 				BDirectory root;
1080 				tempVolume.GetRootDirectory(&root);
1081 				time_t cmpCreated;
1082 				root.GetCreationTime(&cmpCreated);
1083 				if (created == cmpCreated && capacity == tempVolume.Capacity()) {
1084 					*volume = tempVolume;
1085 					return B_OK;
1086 				}
1087 			}
1088 		}
1089 	}
1090 
1091 	return B_DEV_BAD_DRIVE_NUM;
1092 }
1093 
1094 
1095 void
1096 StringFromStream(BString* string, BMallocIO* stream, bool endianSwap)
1097 {
1098 	int32 length;
1099 	stream->Read(&length, sizeof(length));
1100 	if (endianSwap)
1101 		length = SwapInt32(length);
1102 
1103 	if (length < 0 || length > 10000) {
1104 		// TODO: should fail here
1105 		PRINT(("problems instatiating a string, length probably wrong %"
1106 			B_PRId32 "\n", length));
1107 		return;
1108 	}
1109 
1110 	char* buffer = string->LockBuffer(length + 1);
1111 	stream->Read(buffer, (size_t)length + 1);
1112 	string->UnlockBuffer(length);
1113 }
1114 
1115 
1116 void
1117 StringToStream(const BString* string, BMallocIO* stream)
1118 {
1119 	int32 length = string->Length();
1120 	stream->Write(&length, sizeof(int32));
1121 	stream->Write(string->String(), (size_t)string->Length() + 1);
1122 }
1123 
1124 
1125 int32
1126 ArchiveSize(const BString* string)
1127 {
1128 	return string->Length() + 1 + (ssize_t)sizeof(int32);
1129 }
1130 
1131 
1132 int32
1133 CountRefs(const BMessage* message)
1134 {
1135 	uint32 type;
1136 	int32 count;
1137 	message->GetInfo("refs", &type, &count);
1138 
1139 	return count;
1140 }
1141 
1142 
1143 static entry_ref*
1144 EachEntryRefCommon(BMessage* message, entry_ref *(*func)(entry_ref*, void*),
1145 	void* passThru, int32 maxCount)
1146 {
1147 	uint32 type;
1148 	int32 count;
1149 	message->GetInfo("refs", &type, &count);
1150 
1151 	if (maxCount >= 0 && count > maxCount)
1152 		count = maxCount;
1153 
1154 	for (int32 index = 0; index < count; index++) {
1155 		entry_ref ref;
1156 		message->FindRef("refs", index, &ref);
1157 		entry_ref* newRef = (func)(&ref, passThru);
1158 		if (newRef != NULL)
1159 			return newRef;
1160 	}
1161 
1162 	return NULL;
1163 }
1164 
1165 
1166 bool
1167 ContainsEntryRef(const BMessage* message, const entry_ref* ref)
1168 {
1169 	entry_ref match;
1170 	for (int32 index = 0; (message->FindRef("refs", index, &match) == B_OK);
1171 			index++) {
1172 		if (*ref == match)
1173 			return true;
1174 	}
1175 
1176 	return false;
1177 }
1178 
1179 
1180 entry_ref*
1181 EachEntryRef(BMessage* message, entry_ref* (*func)(entry_ref*, void*),
1182 	void* passThru)
1183 {
1184 	return EachEntryRefCommon(message, func, passThru, -1);
1185 }
1186 
1187 typedef entry_ref *(*EachEntryIteratee)(entry_ref *, void *);
1188 
1189 
1190 const entry_ref*
1191 EachEntryRef(const BMessage* message,
1192 	const entry_ref* (*func)(const entry_ref*, void*), void* passThru)
1193 {
1194 	return EachEntryRefCommon(const_cast<BMessage*>(message),
1195 		(EachEntryIteratee)func, passThru, -1);
1196 }
1197 
1198 
1199 entry_ref*
1200 EachEntryRef(BMessage* message, entry_ref* (*func)(entry_ref*, void*),
1201 	void* passThru, int32 maxCount)
1202 {
1203 	return EachEntryRefCommon(message, func, passThru, maxCount);
1204 }
1205 
1206 
1207 const entry_ref *
1208 EachEntryRef(const BMessage* message,
1209 	const entry_ref *(*func)(const entry_ref *, void *), void* passThru,
1210 	int32 maxCount)
1211 {
1212 	return EachEntryRefCommon(const_cast<BMessage *>(message),
1213 		(EachEntryIteratee)func, passThru, maxCount);
1214 }
1215 
1216 
1217 void
1218 TruncateLeaf(BString* string)
1219 {
1220 	for (int32 index = string->Length(); index >= 0; index--) {
1221 		if ((*string)[index] == '/') {
1222 			string->Truncate(index + 1);
1223 			return;
1224 		}
1225 	}
1226 }
1227 
1228 
1229 int64
1230 StringToScalar(const char* text)
1231 {
1232 	char* end;
1233 	int64 val;
1234 
1235 	char* buffer = new char [strlen(text) + 1];
1236 	strcpy(buffer, text);
1237 
1238 	if (strstr(buffer, "k") || strstr(buffer, "K")) {
1239 		val = strtoll(buffer, &end, 10);
1240 		val *= kKBSize;
1241 	} else if (strstr(buffer, "mb") || strstr(buffer, "MB")) {
1242 		val = strtoll(buffer, &end, 10);
1243 		val *= kMBSize;
1244 	} else if (strstr(buffer, "gb") || strstr(buffer, "GB")) {
1245 		val = strtoll(buffer, &end, 10);
1246 		val *= kGBSize;
1247 	} else if (strstr(buffer, "byte") || strstr(buffer, "BYTE")) {
1248 		val = strtoll(buffer, &end, 10);
1249 		val *= kGBSize;
1250 	} else {
1251 		// no suffix, try plain byte conversion
1252 		val = strtoll(buffer, &end, 10);
1253 	}
1254 	delete[] buffer;
1255 
1256 	return val;
1257 }
1258 
1259 
1260 int32
1261 ListIconSize()
1262 {
1263 	static int32 sIconSize = std::max((int32)B_MINI_ICON,
1264 		(int32)ceilf(B_MINI_ICON * be_plain_font->Size() / 12));
1265 	return sIconSize;
1266 }
1267 
1268 
1269 static BRect
1270 LineBounds(BPoint where, float length, bool vertical)
1271 {
1272 	BRect rect;
1273 	rect.SetLeftTop(where);
1274 	rect.SetRightBottom(where + BPoint(2, 2));
1275 	if (vertical)
1276 		rect.bottom = rect.top + length;
1277 	else
1278 		rect.right = rect.left + length;
1279 
1280 	return rect;
1281 }
1282 
1283 
1284 SeparatorLine::SeparatorLine(BPoint where, float length, bool vertical,
1285 	const char* name)
1286 	:
1287 	BView(LineBounds(where, length, vertical), name,
1288 		B_FOLLOW_LEFT | B_FOLLOW_TOP, B_WILL_DRAW)
1289 {
1290 	SetViewUIColor(B_PANEL_BACKGROUND_COLOR);
1291 	SetLowUIColor(B_PANEL_BACKGROUND_COLOR);
1292 }
1293 
1294 
1295 void
1296 SeparatorLine::Draw(BRect)
1297 {
1298 	BRect bounds(Bounds());
1299 	rgb_color hiliteColor = tint_color(ViewColor(), 1.5f);
1300 
1301 	bool vertical = (bounds.left > bounds.right - 3);
1302 	BeginLineArray(2);
1303 	if (vertical) {
1304 		AddLine(bounds.LeftTop(), bounds.LeftBottom(), hiliteColor);
1305 		AddLine(bounds.LeftTop() + BPoint(1, 0),
1306 			bounds.LeftBottom() + BPoint(1, 0), kWhite);
1307 	} else {
1308 		AddLine(bounds.LeftTop(), bounds.RightTop(), hiliteColor);
1309 		AddLine(bounds.LeftTop() + BPoint(0, 1),
1310 			bounds.RightTop() + BPoint(0, 1), kWhite);
1311 	}
1312 	EndLineArray();
1313 }
1314 
1315 
1316 void
1317 HexDump(const void* buf, int32 length)
1318 {
1319 	const int32 kBytesPerLine = 16;
1320 	int32 offset;
1321 	unsigned char* buffer = (unsigned char*)buf;
1322 
1323 	for (offset = 0; ; offset += kBytesPerLine, buffer += kBytesPerLine) {
1324 		int32 remain = length;
1325 		int32 index;
1326 
1327 		printf( "0x%06x: ", (int)offset);
1328 
1329 		for (index = 0; index < kBytesPerLine; index++) {
1330 			if (remain-- > 0)
1331 				printf("%02x%c", buffer[index], remain > 0 ? ',' : ' ');
1332 			else
1333 				printf("   ");
1334 		}
1335 
1336 		remain = length;
1337 		printf(" \'");
1338 		for (index = 0; index < kBytesPerLine; index++) {
1339 			if (remain-- > 0)
1340 				printf("%c", buffer[index] > ' ' ? buffer[index] : '.');
1341 			else
1342 				printf(" ");
1343 		}
1344 		printf("\'\n");
1345 
1346 		length -= kBytesPerLine;
1347 		if (length <= 0)
1348 			break;
1349 	}
1350 	fflush(stdout);
1351 }
1352 
1353 
1354 void
1355 EnableNamedMenuItem(BMenu* menu, const char* itemName, bool on)
1356 {
1357 	BMenuItem* item = menu->FindItem(itemName);
1358 	if (item != NULL)
1359 		item->SetEnabled(on);
1360 }
1361 
1362 
1363 void
1364 MarkNamedMenuItem(BMenu* menu, const char* itemName, bool on)
1365 {
1366 	BMenuItem* item = menu->FindItem(itemName);
1367 	if (item != NULL)
1368 		item->SetMarked(on);
1369 }
1370 
1371 
1372 void
1373 EnableNamedMenuItem(BMenu* menu, uint32 commandName, bool on)
1374 {
1375 	BMenuItem* item = menu->FindItem(commandName);
1376 	if (item != NULL)
1377 		item->SetEnabled(on);
1378 }
1379 
1380 
1381 void
1382 MarkNamedMenuItem(BMenu* menu, uint32 commandName, bool on)
1383 {
1384 	BMenuItem* item = menu->FindItem(commandName);
1385 	if (item != NULL)
1386 		item->SetMarked(on);
1387 }
1388 
1389 
1390 void
1391 DeleteSubmenu(BMenuItem* submenuItem)
1392 {
1393 	if (submenuItem == NULL)
1394 		return;
1395 
1396 	BMenu* menu = submenuItem->Submenu();
1397 	if (menu == NULL)
1398 		return;
1399 
1400 	for (;;) {
1401 		BMenuItem* item = menu->RemoveItem((int32)0);
1402 		if (item == NULL)
1403 			return;
1404 
1405 		delete item;
1406 	}
1407 }
1408 
1409 
1410 status_t
1411 GetAppSignatureFromAttr(BFile* file, char* attr)
1412 {
1413 	// This call is a performance improvement that
1414 	// avoids using the BAppFileInfo API when retrieving the
1415 	// app signature -- the call is expensive because by default
1416 	// the resource fork is scanned to read the attribute
1417 
1418 #ifdef B_APP_FILE_INFO_IS_FAST
1419 	BAppFileInfo appFileInfo(file);
1420 	return appFileInfo.GetSignature(attr);
1421 #else
1422 	ssize_t readResult = file->ReadAttr(kAttrAppSignature, B_MIME_STRING_TYPE,
1423 		0, attr, B_MIME_TYPE_LENGTH);
1424 
1425 	if (readResult <= 0)
1426 		return (status_t)readResult;
1427 
1428 	return B_OK;
1429 #endif // B_APP_FILE_INFO_IS_FAST
1430 }
1431 
1432 
1433 status_t
1434 GetAppIconFromAttr(BFile* file, BBitmap* icon, icon_size which)
1435 {
1436 	// This call is a performance improvement that
1437 	// avoids using the BAppFileInfo API when retrieving the
1438 	// app icons -- the call is expensive because by default
1439 	// the resource fork is scanned to read the icons
1440 
1441 //#ifdef B_APP_FILE_INFO_IS_FAST
1442 	BAppFileInfo appFileInfo(file);
1443 	return appFileInfo.GetIcon(icon, which);
1444 //#else
1445 //
1446 //	const char* attrName = kAttrIcon;
1447 //	uint32 type = B_VECTOR_ICON_TYPE;
1448 //
1449 //	// try vector icon
1450 //	attr_info ainfo;
1451 //	status_t result = file->GetAttrInfo(attrName, &ainfo);
1452 //
1453 //	if (result == B_OK) {
1454 //		uint8 buffer[ainfo.size];
1455 //		ssize_t readResult = file->ReadAttr(attrName, type, 0, buffer,
1456 //			ainfo.size);
1457 //		if (readResult == ainfo.size) {
1458 //			if (BIconUtils::GetVectorIcon(buffer, ainfo.size, icon) == B_OK)
1459 //				return B_OK;
1460 //		}
1461 //	}
1462 //
1463 //	// try again with R5 icons
1464 //	attrName = which == B_LARGE_ICON ? kAttrLargeIcon : kAttrMiniIcon;
1465 //	type = which == B_LARGE_ICON ? LARGE_ICON_TYPE : MINI_ICON_TYPE;
1466 //
1467 //	result = file->GetAttrInfo(attrName, &ainfo);
1468 //	if (result < B_OK)
1469 //		return result;
1470 //
1471 //	uint8 buffer[ainfo.size];
1472 //
1473 //	ssize_t readResult = file->ReadAttr(attrName, type, 0, buffer, ainfo.size);
1474 //	if (readResult <= 0)
1475 //		return (status_t)readResult;
1476 //
1477 //	if (icon->ColorSpace() != B_CMAP8)
1478 //		result = BIconUtils::ConvertFromCMAP8(buffer, which, which, which, icon);
1479 //	else
1480 //		icon->SetBits(buffer, icon->BitsLength(), 0, B_CMAP8);
1481 //
1482 //	return result;
1483 //#endif	// B_APP_FILE_INFO_IS_FAST
1484 }
1485 
1486 
1487 status_t
1488 GetFileIconFromAttr(BNode* node, BBitmap* icon, icon_size which)
1489 {
1490 	BNodeInfo fileInfo(node);
1491 	return fileInfo.GetIcon(icon, which);
1492 }
1493 
1494 
1495 void
1496 PrintToStream(rgb_color color)
1497 {
1498 	printf("r:%x, g:%x, b:%x, a:%x\n",
1499 		color.red, color.green, color.blue, color.alpha);
1500 }
1501 
1502 
1503 extern BMenuItem*
1504 EachMenuItem(BMenu* menu, bool recursive, BMenuItem* (*func)(BMenuItem *))
1505 {
1506 	int32 count = menu->CountItems();
1507 	for (int32 index = 0; index < count; index++) {
1508 		BMenuItem* item = menu->ItemAt(index);
1509 		BMenuItem* newItem = (func)(item);
1510 		if (newItem != NULL)
1511 			return newItem;
1512 
1513 		if (recursive) {
1514 			BMenu* submenu = menu->SubmenuAt(index);
1515 			if (submenu != NULL)
1516 				return EachMenuItem(submenu, true, func);
1517 		}
1518 	}
1519 
1520 	return NULL;
1521 }
1522 
1523 
1524 extern const BMenuItem*
1525 EachMenuItem(const BMenu* menu, bool recursive,
1526 	BMenuItem* (*func)(const BMenuItem *))
1527 {
1528 	int32 count = menu->CountItems();
1529 	for (int32 index = 0; index < count; index++) {
1530 		BMenuItem* item = menu->ItemAt(index);
1531 		BMenuItem* newItem = (func)(item);
1532 		if (newItem != NULL)
1533 			return newItem;
1534 
1535 		if (recursive) {
1536 			BMenu* submenu = menu->SubmenuAt(index);
1537 			if (submenu != NULL)
1538 				return EachMenuItem(submenu, true, func);
1539 		}
1540 	}
1541 
1542 	return NULL;
1543 }
1544 
1545 
1546 //	#pragma mark - PositionPassingMenuItem
1547 
1548 
1549 PositionPassingMenuItem::PositionPassingMenuItem(const char* title,
1550 	BMessage* message, char shortcut, uint32 modifiers)
1551 	:
1552 	BMenuItem(title, message, shortcut, modifiers)
1553 {
1554 }
1555 
1556 
1557 PositionPassingMenuItem::PositionPassingMenuItem(BMenu* menu, BMessage* message)
1558 	:
1559 	BMenuItem(menu, message)
1560 {
1561 }
1562 
1563 
1564 PositionPassingMenuItem::PositionPassingMenuItem(BMessage* data)
1565 	:
1566 	BMenuItem(data)
1567 {
1568 }
1569 
1570 
1571 BArchivable*
1572 PositionPassingMenuItem::Instantiate(BMessage* data)
1573 {
1574 	if (validate_instantiation(data, "PositionPassingMenuItem"))
1575 		return new PositionPassingMenuItem(data);
1576 
1577 	return NULL;
1578 }
1579 
1580 
1581 status_t
1582 PositionPassingMenuItem::Invoke(BMessage* message)
1583 {
1584 	if (Menu() == NULL)
1585 		return B_ERROR;
1586 
1587 	if (!IsEnabled())
1588 		return B_ERROR;
1589 
1590 	if (message == NULL)
1591 		message = Message();
1592 
1593 	if (message == NULL)
1594 		return B_BAD_VALUE;
1595 
1596 	BMessage clone(*message);
1597 	clone.AddInt32("index", Menu()->IndexOf(this));
1598 	clone.AddInt64("when", system_time());
1599 	clone.AddPointer("source", this);
1600 
1601 	// embed the invoke location of the menu so that we can create
1602 	// a new folder, etc. on the spot
1603 	BMenu* menu = Menu();
1604 
1605 	for (;;) {
1606 		if (!menu->Supermenu())
1607 			break;
1608 
1609 		menu = menu->Supermenu();
1610 	}
1611 
1612 	// use the window position only, if the item was invoked from the menu
1613 	// menu->Window() points to the window the item was invoked from
1614 	if (dynamic_cast<BContainerWindow*>(menu->Window()) == NULL) {
1615 		LooperAutoLocker lock(menu);
1616 		if (lock.IsLocked()) {
1617 			BPoint invokeOrigin(menu->Window()->Frame().LeftTop());
1618 			clone.AddPoint("be:invoke_origin", invokeOrigin);
1619 		}
1620 	}
1621 
1622 	return BInvoker::Invoke(&clone);
1623 }
1624 
1625 
1626 //	#pragma mark - BPrivate functions
1627 
1628 
1629 bool
1630 BootedInSafeMode()
1631 {
1632 	const char* safeMode = getenv("SAFEMODE");
1633 	return (safeMode && strcmp(safeMode, "yes") == 0);
1634 }
1635 
1636 
1637 float
1638 ComputeTypeAheadScore(const char* text, const char* match, bool wordMode)
1639 {
1640 	// highest score: exact match
1641 	const char* found = strcasestr(text, match);
1642 	if (found != NULL) {
1643 		if (found == text)
1644 			return kExactMatchScore;
1645 
1646 		return 1.f / (found - text);
1647 	}
1648 
1649 	// there was no exact match
1650 
1651 	// second best: all characters at word beginnings
1652 	if (wordMode) {
1653 		float score = 0;
1654 		for (int32 j = 0, k = 0; match[j]; j++) {
1655 			while (text[k]
1656 				&& tolower(text[k]) != tolower(match[j])) {
1657 				k++;
1658 			}
1659 			if (text[k] == '\0') {
1660 				score = 0;
1661 				break;
1662 			}
1663 
1664 			bool wordStart = k == 0 || isspace(text[k - 1]);
1665 			if (wordStart)
1666 				score++;
1667 			if (j > 0) {
1668 				bool wordEnd = !text[k + 1] || isspace(text[k + 1]);
1669 				if (wordEnd)
1670 					score += 0.3;
1671 				if (match[j - 1] == text[k - 1])
1672 					score += 0.7;
1673 			}
1674 
1675 			score += 1.f / (k + 1);
1676 			k++;
1677 		}
1678 
1679 		return score;
1680 	}
1681 
1682 	return -1;
1683 }
1684 
1685 
1686 //	#pragma mark - throw on error functions.
1687 
1688 
1689 void
1690 _ThrowOnError(status_t result, const char* DEBUG_ONLY(file),
1691 	int32 DEBUG_ONLY(line))
1692 {
1693 	if (result != B_OK) {
1694 		PRINT(("%s at %s:%d\n", strerror(result), file, (int)line));
1695 		throw result;
1696 	}
1697 }
1698 
1699 
1700 void
1701 _ThrowIfNotSize(ssize_t size, const char* DEBUG_ONLY(file),
1702 	int32 DEBUG_ONLY(line))
1703 {
1704 	if (size < B_OK) {
1705 		PRINT(("%s at %s:%d\n", strerror((status_t)size), file, (int)line));
1706 		throw (status_t)size;
1707 	}
1708 }
1709 
1710 
1711 void
1712 _ThrowOnAssert(bool success, const char* DEBUG_ONLY(file),
1713 	int32 DEBUG_ONLY(line))
1714 {
1715 	if (!success) {
1716 		PRINT(("Assert failed at %s:%d\n", file, (int)line));
1717 		throw B_ERROR;
1718 	}
1719 }
1720 
1721 } // namespace BPrivate
1722