/* Open Tracker License Terms and Conditions Copyright (c) 1991-2000, Be Incorporated. All rights reserved. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice applies to all licensees and shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF TITLE, MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL BE INCORPORATED BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF, OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. Except as contained in this notice, the name of Be Incorporated shall not be used in advertising or otherwise to promote the sale, use or other dealings in this Software without prior written authorization from Be Incorporated. Tracker(TM), Be(R), BeOS(R), and BeIA(TM) are trademarks or registered trademarks of Be Incorporated in the United States and other countries. Other brand product names are registered trademarks or trademarks of their respective holders. All rights reserved. */ #include "Attributes.h" #include "MimeTypes.h" #include "Model.h" #include "PoseView.h" #include "Utilities.h" #include "ContainerWindow.h" #ifdef __HAIKU__ # include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifndef _IMPEXP_BE # define _IMPEXP_BE #endif extern _IMPEXP_BE const uint32 LARGE_ICON_TYPE; extern _IMPEXP_BE const uint32 MINI_ICON_TYPE; FILE *logFile = NULL; static const float kMinSeparatorStubX = 10; static const float kStubToStringSlotX = 5; namespace BPrivate { const float kExactMatchScore = INFINITY; const rgb_color kBlack = {0, 0, 0, 255}; const rgb_color kWhite = {255, 255, 255, 255}; uint32 HashString(const char *string, uint32 seed) { char ch; uint32 result = seed; while((ch = *string++) != 0) { result = (result << 7) ^ (result >> 24); result ^= ch; } result ^= result << 12; return result; } uint32 AttrHashString(const char *string, uint32 type) { char c; uint32 hash = 0; while((c = *string++) != 0) { hash = (hash << 7) ^ (hash >> 24); hash ^= c; } hash ^= hash << 12; hash &= ~0xff; hash |= type; return hash; } bool ValidateStream(BMallocIO *stream, uint32 key, int32 version) { uint32 testKey; int32 testVersion; if (stream->Read(&testKey, sizeof(uint32)) <= 0 || stream->Read(&testVersion, sizeof(int32)) <=0) return false; return testKey == key && testVersion == version; } void DisallowFilenameKeys(BTextView *textView) { textView->DisallowChar(':'); textView->DisallowChar('/'); } void DisallowMetaKeys(BTextView *textView) { textView->DisallowChar(B_TAB); textView->DisallowChar(B_ESCAPE); textView->DisallowChar(B_INSERT); textView->DisallowChar(B_DELETE); textView->DisallowChar(B_HOME); textView->DisallowChar(B_END); textView->DisallowChar(B_PAGE_UP); textView->DisallowChar(B_PAGE_DOWN); textView->DisallowChar(B_FUNCTION_KEY); } PeriodicUpdatePoses::PeriodicUpdatePoses() : fPoseList(20, true) { fLock = new Benaphore("PeriodicUpdatePoses"); } PeriodicUpdatePoses::~PeriodicUpdatePoses() { fLock->Lock(); fPoseList.MakeEmpty(); delete fLock; } void PeriodicUpdatePoses::AddPose(BPose *pose, BPoseView *poseView, PeriodicUpdateCallback callback, void *cookie) { periodic_pose *periodic = new periodic_pose; periodic->pose = pose; periodic->pose_view = poseView; periodic->callback = callback; periodic->cookie = cookie; fPoseList.AddItem(periodic); } bool PeriodicUpdatePoses::RemovePose(BPose *pose, void **cookie) { int32 count = fPoseList.CountItems(); for (int32 index = 0; index < count; index++) { if (fPoseList.ItemAt(index)->pose == pose) { if (!fLock->Lock()) return false; periodic_pose *periodic = fPoseList.RemoveItemAt(index); if (cookie) *cookie = periodic->cookie; delete periodic; fLock->Unlock(); return true; } } return false; } void PeriodicUpdatePoses::DoPeriodicUpdate(bool forceRedraw) { if (!fLock->Lock()) return; int32 count = fPoseList.CountItems(); for (int32 index = 0; index < count; index++) { periodic_pose *periodic = fPoseList.ItemAt(index); if (periodic->callback(periodic->pose, periodic->cookie) || forceRedraw) { periodic->pose_view->LockLooper(); periodic->pose_view->UpdateIcon(periodic->pose); periodic->pose_view->UnlockLooper(); } } fLock->Unlock(); } PeriodicUpdatePoses gPeriodicUpdatePoses; } // namespace BPrivate void PoseInfo::EndianSwap(void *castToThis) { PoseInfo *self = (PoseInfo *)castToThis; PRINT(("swapping PoseInfo\n")); STATIC_ASSERT(sizeof(ino_t) == sizeof(int64)); self->fInitedDirectory = SwapInt64(self->fInitedDirectory); swap_data(B_POINT_TYPE, &self->fLocation, sizeof(BPoint), B_SWAP_ALWAYS); // do a sanity check on the icon position if (self->fLocation.x < -20000 || self->fLocation.x > 20000 || self->fLocation.y < -20000 || self->fLocation.y > 20000) { // position out of range, force autoplcemement PRINT((" rejecting icon position out of range\n")); self->fInitedDirectory = -1LL; self->fLocation = BPoint(0, 0); } } void PoseInfo::PrintToStream() { PRINT(("%s, inode:%Lx, location %f %f\n", fInvisible ? "hidden" : "visible", fInitedDirectory, fLocation.x, fLocation.y)); } // #pragma mark - size_t ExtendedPoseInfo::Size() const { return sizeof(ExtendedPoseInfo) + fNumFrames * sizeof(FrameLocation); } size_t ExtendedPoseInfo::Size(int32 count) { return sizeof(ExtendedPoseInfo) + count * sizeof(FrameLocation); } size_t ExtendedPoseInfo::SizeWithHeadroom() const { return sizeof(ExtendedPoseInfo) + (fNumFrames + 1) * sizeof(FrameLocation); } size_t ExtendedPoseInfo::SizeWithHeadroom(size_t oldSize) { int32 count = (ssize_t)oldSize - (ssize_t)sizeof(ExtendedPoseInfo); if (count > 0) count /= sizeof(FrameLocation); else count = 0; return Size(count + 1); } bool ExtendedPoseInfo::HasLocationForFrame(BRect frame) const { for (int32 index = 0; index < fNumFrames; index++) { if (fLocations[index].fFrame == frame) return true; } return false; } BPoint ExtendedPoseInfo::LocationForFrame(BRect frame) const { for (int32 index = 0; index < fNumFrames; index++) { if (fLocations[index].fFrame == frame) return fLocations[index].fLocation; } TRESPASS(); return BPoint(0, 0); } bool ExtendedPoseInfo::SetLocationForFrame(BPoint newLocation, BRect frame) { for (int32 index = 0; index < fNumFrames; index++) { if (fLocations[index].fFrame == frame) { if (fLocations[index].fLocation == newLocation) return false; fLocations[index].fLocation = newLocation; return true; } } fLocations[fNumFrames].fFrame = frame; fLocations[fNumFrames].fLocation = newLocation; fLocations[fNumFrames].fWorkspaces = 0xffffffff; fNumFrames++; return true; } void ExtendedPoseInfo::EndianSwap(void *castToThis) { ExtendedPoseInfo *self = (ExtendedPoseInfo *)castToThis; PRINT(("swapping ExtendedPoseInfo\n")); self->fWorkspaces = SwapUInt32(self->fWorkspaces); self->fNumFrames = SwapInt32(self->fNumFrames); for (int32 index = 0; index < self->fNumFrames; index++) { swap_data(B_POINT_TYPE, &self->fLocations[index].fLocation, sizeof(BPoint), B_SWAP_ALWAYS); if (self->fLocations[index].fLocation.x < -20000 || self->fLocations[index].fLocation.x > 20000 || self->fLocations[index].fLocation.y < -20000 || self->fLocations[index].fLocation.y > 20000) { // position out of range, force autoplcemement PRINT((" rejecting icon position out of range\n")); self->fLocations[index].fLocation = BPoint(0, 0); } swap_data(B_RECT_TYPE, &self->fLocations[index].fFrame, sizeof(BRect), B_SWAP_ALWAYS); } } void ExtendedPoseInfo::PrintToStream() { } // #pragma mark - OffscreenBitmap::OffscreenBitmap(BRect frame) : fBitmap(NULL) { NewBitmap(frame); } OffscreenBitmap::OffscreenBitmap() : fBitmap(NULL) { } OffscreenBitmap::~OffscreenBitmap() { delete fBitmap; } void OffscreenBitmap::NewBitmap(BRect bounds) { delete fBitmap; fBitmap = new BBitmap(bounds, B_RGB32, true); if (fBitmap->Lock()) { BView *view = new BView(fBitmap->Bounds(), "", B_FOLLOW_NONE, 0); fBitmap->AddChild(view); BRect clipRect = view->Bounds(); BRegion newClip; newClip.Set(clipRect); view->ConstrainClippingRegion(&newClip); fBitmap->Unlock(); } else { delete fBitmap; fBitmap = NULL; } } BView * OffscreenBitmap::BeginUsing(BRect frame) { if (!fBitmap || fBitmap->Bounds() != frame) NewBitmap(frame); fBitmap->Lock(); return View(); } void OffscreenBitmap::DoneUsing() { fBitmap->Unlock(); } BBitmap * OffscreenBitmap::Bitmap() const { ASSERT(fBitmap); ASSERT(fBitmap->IsLocked()); return fBitmap; } BView * OffscreenBitmap::View() const { ASSERT(fBitmap); return fBitmap->ChildAt(0); } // #pragma mark - namespace BPrivate { /** Changes the alpha value of the given bitmap to create a nice * horizontal fade out in the specified region. * "from" is always transparent, "to" opaque. */ void FadeRGBA32Horizontal(uint32 *bits, int32 width, int32 height, int32 from, int32 to) { // check parameters if (width < 0 || height < 0 || from < 0 || to < 0) return; float change = 1.f / (to - from); if (from > to) { int32 temp = from; from = to; to = temp; } for (int32 y = 0; y < height; y++) { float alpha = change > 0 ? 0.0f : 1.0f; for (int32 x = from; x <= to; x++) { if (bits[x] & 0xff000000) { uint32 a = uint32((bits[x] >> 24) * alpha); bits[x] = (bits[x] & 0x00ffffff) | (a << 24); } alpha += change; } bits += width; } } /** Changes the alpha value of the given bitmap to create a nice * vertical fade out in the specified region. * "from" is always transparent, "to" opaque. */ void FadeRGBA32Vertical(uint32 *bits, int32 width, int32 height, int32 from, int32 to) { // check parameters if (width < 0 || height < 0 || from < 0 || to < 0) return; if (from > to) bits += width * (height - (from - to)); float change = 1.f / (to - from); if (from > to) { int32 temp = from; from = to; to = temp; } float alpha = change > 0 ? 0.0f : 1.0f; for (int32 y = from; y <= to; y++) { for (int32 x = 0; x < width; x++) { if (bits[x] & 0xff000000) { uint32 a = uint32((bits[x] >> 24) * alpha); bits[x] = (bits[x] & 0x00ffffff) | (a << 24); } } alpha += change; bits += width; } } } // namespace BPrivate // #pragma mark - DraggableIcon::DraggableIcon(BRect rect, const char *name, const char *mimeType, icon_size size, const BMessage *message, BMessenger target, uint32 resizeMask, uint32 flags) : BView(rect, name, resizeMask, flags), fMessage(*message), fTarget(target) { fBitmap = new BBitmap(Bounds(), kDefaultIconDepth); BMimeType mime(mimeType); status_t error = mime.GetIcon(fBitmap, size); ASSERT(mime.IsValid()); if (error != B_OK) { PRINT(("failed to get icon for %s, %s\n", mimeType, strerror(error))); BMimeType mime(B_FILE_MIMETYPE); ASSERT(mime.IsInstalled()); mime.GetIcon(fBitmap, size); } } DraggableIcon::~DraggableIcon() { delete fBitmap; } void DraggableIcon::SetTarget(BMessenger target) { fTarget = target; } BRect DraggableIcon::PreferredRect(BPoint offset, icon_size size) { BRect result(0, 0, size - 1, size - 1); result.OffsetTo(offset); return result; } void DraggableIcon::AttachedToWindow() { BView *parent = Parent(); if (parent) { SetViewColor(parent->ViewColor()); SetLowColor(parent->LowColor()); } } void DraggableIcon::MouseDown(BPoint point) { if (!DragStarted(&fMessage)) return; BRect rect(Bounds()); BBitmap *dragBitmap = new BBitmap(rect, B_RGBA32, true); dragBitmap->Lock(); BView *view = new BView(dragBitmap->Bounds(), "", B_FOLLOW_NONE, 0); dragBitmap->AddChild(view); view->SetOrigin(0, 0); BRect clipRect(view->Bounds()); BRegion newClip; newClip.Set(clipRect); view->ConstrainClippingRegion(&newClip); // Transparent draw magic view->SetHighColor(0, 0, 0, 0); view->FillRect(view->Bounds()); view->SetDrawingMode(B_OP_ALPHA); view->SetHighColor(0, 0, 0, 128); // set the level of transparency by value view->SetBlendingMode(B_CONSTANT_ALPHA, B_ALPHA_COMPOSITE); view->DrawBitmap(fBitmap); view->Sync(); dragBitmap->Unlock(); DragMessage(&fMessage, dragBitmap, B_OP_ALPHA, point, fTarget.Target(0)); } bool DraggableIcon::DragStarted(BMessage *) { return true; } void DraggableIcon::Draw(BRect) { #ifdef __HAIKU__ SetDrawingMode(B_OP_ALPHA); SetBlendingMode(B_PIXEL_ALPHA, B_ALPHA_OVERLAY); #else SetDrawingMode(B_OP_OVER); #endif DrawBitmap(fBitmap); } // #pragma mark - FlickerFreeStringView::FlickerFreeStringView(BRect bounds, const char *name, const char *text, uint32 resizeFlags, uint32 flags) : BStringView(bounds, name, text, resizeFlags, flags), fBitmap(NULL), fOrigBitmap(NULL) { } FlickerFreeStringView::FlickerFreeStringView(BRect bounds, const char *name, const char *text, BBitmap *inBitmap, uint32 resizeFlags, uint32 flags) : BStringView(bounds, name, text, resizeFlags, flags), fBitmap(NULL), fOrigBitmap(inBitmap) { } FlickerFreeStringView::~FlickerFreeStringView() { delete fBitmap; } void FlickerFreeStringView::Draw(BRect) { BRect bounds(Bounds()); if (!fBitmap) fBitmap = new OffscreenBitmap(Bounds()); BView *offscreen = fBitmap->BeginUsing(bounds); if (Parent()) { fViewColor = Parent()->ViewColor(); fLowColor = Parent()->ViewColor(); } offscreen->SetViewColor(fViewColor); offscreen->SetHighColor(HighColor()); offscreen->SetLowColor(fLowColor); BFont font; GetFont(&font); offscreen->SetFont(&font); offscreen->Sync(); if (fOrigBitmap) offscreen->DrawBitmap(fOrigBitmap, Frame(), bounds); else offscreen->FillRect(bounds, B_SOLID_LOW); if (Text()) { BPoint loc; font_height height; GetFontHeight(&height); edge_info eInfo; switch (Alignment()) { case B_ALIGN_LEFT: #ifdef HAIKU_TARGET_PLATFORM_HAIKU case B_ALIGN_HORIZONTAL_UNSET: case B_ALIGN_USE_FULL_WIDTH: #endif { // If the first char has a negative left edge give it // some more room by shifting that much more to the right. font.GetEdges(Text(), 1, &eInfo); loc.x = bounds.left + (2 - eInfo.left); break; } case B_ALIGN_CENTER: { float width = StringWidth(Text()); float center = (bounds.right - bounds.left) / 2; loc.x = center - (width/2); break; } case B_ALIGN_RIGHT: { float width = StringWidth(Text()); loc.x = bounds.right - width - 2; break; } } loc.y = bounds.bottom - (1 + height.descent); offscreen->DrawString(Text(), loc); } offscreen->Sync(); SetDrawingMode(B_OP_COPY); DrawBitmap(fBitmap->Bitmap()); fBitmap->DoneUsing(); } void FlickerFreeStringView::AttachedToWindow() { _inherited::AttachedToWindow(); if (Parent()) { fViewColor = Parent()->ViewColor(); fLowColor = Parent()->ViewColor(); } SetViewColor(B_TRANSPARENT_32_BIT); SetLowColor(B_TRANSPARENT_32_BIT); } void FlickerFreeStringView::SetViewColor(rgb_color color) { if (fViewColor != color) { fViewColor = color; Invalidate(); } _inherited::SetViewColor(B_TRANSPARENT_32_BIT); } void FlickerFreeStringView::SetLowColor(rgb_color color) { if (fLowColor != color) { fLowColor = color; Invalidate(); } _inherited::SetLowColor(B_TRANSPARENT_32_BIT); } // #pragma mark - TitledSeparatorItem::TitledSeparatorItem(const char *label) : BMenuItem(label, 0) { _inherited::SetEnabled(false); } TitledSeparatorItem::~TitledSeparatorItem() { } void TitledSeparatorItem::SetEnabled(bool) { // leave disabled } void TitledSeparatorItem::GetContentSize(float *width, float *height) { _inherited::GetContentSize(width, height); } inline rgb_color ShiftMenuBackgroundColor(float by) { return tint_color(ui_color(B_MENU_BACKGROUND_COLOR), by); } void TitledSeparatorItem::Draw() { BRect frame(Frame()); BMenu *parent = Menu(); ASSERT(parent); menu_info minfo; get_menu_info(&minfo); if (minfo.separator > 0) { frame.left += 10; frame.right -= 10; } else { frame.left += 1; frame.right -= 1; } float startX = frame.left; float endX = frame.right; float maxStringWidth = endX - startX - (2 * kMinSeparatorStubX + 2 * kStubToStringSlotX); // ToDo: // handle case where maxStringWidth turns out negative here BString truncatedLabel(Label()); parent->TruncateString(&truncatedLabel, B_TRUNCATE_END, maxStringWidth); maxStringWidth = parent->StringWidth(truncatedLabel.String()); // first calculate the length of the stub part of the // divider line, so we can use it for secondStartX float firstEndX = ((endX - startX) - maxStringWidth) / 2 - kStubToStringSlotX; if (firstEndX < 0) firstEndX = 0; float secondStartX = endX - firstEndX; // now finish calculating firstEndX firstEndX += startX; parent->PushState(); int32 y = (int32) (frame.top + (frame.bottom - frame.top) / 2); parent->BeginLineArray(minfo.separator == 2 ? 6 : 4); parent->AddLine(BPoint(frame.left, y), BPoint(firstEndX, y), ShiftMenuBackgroundColor(B_DARKEN_1_TINT)); parent->AddLine(BPoint(secondStartX, y), BPoint(frame.right, y), ShiftMenuBackgroundColor(B_DARKEN_1_TINT)); if (minfo.separator == 2) { y++; frame.left++; frame.right--; parent->AddLine(BPoint(frame.left,y), BPoint(firstEndX, y), ShiftMenuBackgroundColor(B_DARKEN_1_TINT)); parent->AddLine(BPoint(secondStartX,y), BPoint(frame.right, y), ShiftMenuBackgroundColor(B_DARKEN_1_TINT)); } y++; if (minfo.separator == 2) { frame.left++; frame.right--; } parent->AddLine(BPoint(frame.left, y), BPoint(firstEndX, y), ShiftMenuBackgroundColor(B_DARKEN_1_TINT)); parent->AddLine(BPoint(secondStartX, y), BPoint(frame.right, y), ShiftMenuBackgroundColor(B_DARKEN_1_TINT)); parent->EndLineArray(); font_height finfo; parent->GetFontHeight(&finfo); parent->SetLowColor(parent->ViewColor()); BPoint loc(firstEndX + kStubToStringSlotX, ContentLocation().y + finfo.ascent); parent->MovePenTo(loc + BPoint(1, 1)); parent->SetHighColor(ShiftMenuBackgroundColor(B_DARKEN_1_TINT)); parent->DrawString(truncatedLabel.String()); parent->MovePenTo(loc); parent->SetHighColor(ShiftMenuBackgroundColor(B_DISABLED_LABEL_TINT)); parent->DrawString(truncatedLabel.String()); parent->PopState(); } // #pragma mark - ShortcutFilter::ShortcutFilter(uint32 shortcutKey, uint32 shortcutModifier, uint32 shortcutWhat, BHandler *target) : BMessageFilter(B_KEY_DOWN), fShortcutKey(shortcutKey), fShortcutModifier(shortcutModifier), fShortcutWhat(shortcutWhat), fTarget(target) { } filter_result ShortcutFilter::Filter(BMessage *message, BHandler **) { if (message->what == B_KEY_DOWN) { uint32 modifiers; uint32 rawKeyChar = 0; uint8 byte = 0; int32 key = 0; if (message->FindInt32("modifiers", (int32 *)&modifiers) != B_OK || message->FindInt32("raw_char", (int32 *)&rawKeyChar) != B_OK || message->FindInt8("byte", (int8 *)&byte) != B_OK || message->FindInt32("key", &key) != B_OK) return B_DISPATCH_MESSAGE; modifiers &= B_SHIFT_KEY | B_COMMAND_KEY | B_CONTROL_KEY | B_OPTION_KEY | B_MENU_KEY; // strip caps lock, etc. if (modifiers == fShortcutModifier && rawKeyChar == fShortcutKey) { fTarget->Looper()->PostMessage(fShortcutWhat, fTarget); return B_SKIP_MESSAGE; } } // let others deal with this return B_DISPATCH_MESSAGE; } // #pragma mark - namespace BPrivate { void EmbedUniqueVolumeInfo(BMessage *message, const BVolume *volume) { BDirectory rootDirectory; time_t created; fs_info info; if (volume->GetRootDirectory(&rootDirectory) == B_OK && rootDirectory.GetCreationTime(&created) == B_OK && fs_stat_dev(volume->Device(), &info) == 0) { message->AddInt32("creationDate", created); message->AddInt64("capacity", volume->Capacity()); message->AddString("deviceName", info.device_name); message->AddString("volumeName", info.volume_name); message->AddString("fshName", info.fsh_name); } } status_t MatchArchivedVolume(BVolume *result, const BMessage *message, int32 index) { time_t created; off_t capacity; if (message->FindInt32("creationDate", index, &created) != B_OK || message->FindInt64("capacity", index, &capacity) != B_OK) return B_ERROR; BVolumeRoster roster; BVolume volume; BString deviceName, volumeName, fshName; if (message->FindString("deviceName", &deviceName) == B_OK && message->FindString("volumeName", &volumeName) == B_OK && message->FindString("fshName", &fshName) == B_OK) { // New style volume identifiers: We have a couple of characteristics, // and compute a score from them. The volume with the greatest score // (if over a certain threshold) is the one we're looking for. We // pick the first volume, in case there is more than one with the // same score. dev_t foundDevice = -1; int foundScore = -1; roster.Rewind(); while (roster.GetNextVolume(&volume) == B_OK) { if (volume.IsPersistent() && volume.KnowsQuery()) { // get creation time and fs_info BDirectory root; volume.GetRootDirectory(&root); time_t cmpCreated; fs_info info; if (root.GetCreationTime(&cmpCreated) == B_OK && fs_stat_dev(volume.Device(), &info) == 0) { // compute the score int score = 0; // creation time if (created == cmpCreated) score += 5; // capacity if (capacity == volume.Capacity()) score += 4; // device name if (deviceName == info.device_name) score += 3; // volume name if (volumeName == info.volume_name) score += 2; // fsh name if (fshName == info.fsh_name) score += 1; // check score if (score >= 9 && score > foundScore) { foundDevice = volume.Device(); foundScore = score; } } } } if (foundDevice >= 0) return result->SetTo(foundDevice); } else { // Old style volume identifiers: We have only creation time and // capacity. Both must match. roster.Rewind(); while (roster.GetNextVolume(&volume) == B_OK) if (volume.IsPersistent() && volume.KnowsQuery()) { BDirectory root; volume.GetRootDirectory(&root); time_t cmpCreated; root.GetCreationTime(&cmpCreated); if (created == cmpCreated && capacity == volume.Capacity()) { *result = volume; return B_OK; } } } return B_DEV_BAD_DRIVE_NUM; } void StringFromStream(BString *string, BMallocIO *stream, bool endianSwap) { int32 length; stream->Read(&length, sizeof(length)); if (endianSwap) length = SwapInt32(length); if (length < 0 || length > 10000) { // TODO: should fail here PRINT(("problems instatiating a string, length probably wrong %d\n", length)); return; } char *buffer = string->LockBuffer(length + 1); stream->Read(buffer, (size_t)length + 1); string->UnlockBuffer(length); } void StringToStream(const BString *string, BMallocIO *stream) { int32 length = string->Length(); stream->Write(&length, sizeof(int32)); stream->Write(string->String(), (size_t)string->Length() + 1); } int32 ArchiveSize(const BString *string) { return string->Length() + 1 + (ssize_t)sizeof(int32); } int32 CountRefs(const BMessage *message) { uint32 type; int32 count; message->GetInfo("refs", &type, &count); return count; } static entry_ref * EachEntryRefCommon(BMessage *message, entry_ref *(*func)(entry_ref *, void *), void *passThru, int32 maxCount) { uint32 type; int32 count; message->GetInfo("refs", &type, &count); if (maxCount >= 0 && count > maxCount) count = maxCount; for (int32 index = 0; index < count; index++) { entry_ref ref; message->FindRef("refs", index, &ref); entry_ref *result = (func)(&ref, passThru); if (result) return result; } return NULL; } bool ContainsEntryRef(const BMessage *message, const entry_ref *ref) { entry_ref match; for (int32 index = 0; (message->FindRef("refs", index, &match) == B_OK); index++) { if (*ref == match) return true; } return false; } entry_ref * EachEntryRef(BMessage *message, entry_ref *(*func)(entry_ref *, void *), void *passThru) { return EachEntryRefCommon(message, func, passThru, -1); } typedef entry_ref *(*EachEntryIteratee)(entry_ref *, void *); const entry_ref * EachEntryRef(const BMessage *message, const entry_ref *(*func)(const entry_ref *, void *), void *passThru) { return EachEntryRefCommon(const_cast(message), (EachEntryIteratee)func, passThru, -1); } entry_ref * EachEntryRef(BMessage *message, entry_ref *(*func)(entry_ref *, void *), void *passThru, int32 maxCount) { return EachEntryRefCommon(message, func, passThru, maxCount); } const entry_ref * EachEntryRef(const BMessage *message, const entry_ref *(*func)(const entry_ref *, void *), void *passThru, int32 maxCount) { return EachEntryRefCommon(const_cast(message), (EachEntryIteratee)func, passThru, maxCount); } void TruncateLeaf(BString *string) { for (int32 index = string->Length(); index >= 0; index--) { if ((*string)[index] == '/') { string->Truncate(index + 1); return; } } } int64 StringToScalar(const char *text) { char *end; int64 val; char *buffer = new char [strlen(text) + 1]; strcpy(buffer, text); if (strstr(buffer, "k") || strstr(buffer, "K")) { val = strtoll(buffer, &end, 10); val *= kKBSize; } else if (strstr(buffer, "mb") || strstr(buffer, "MB")) { val = strtoll(buffer, &end, 10); val *= kMBSize; } else if (strstr(buffer, "gb") || strstr(buffer, "GB")) { val = strtoll(buffer, &end, 10); val *= kGBSize; } else if (strstr(buffer, "byte") || strstr(buffer, "BYTE")) { val = strtoll(buffer, &end, 10); val *= kGBSize; } else { // no suffix, try plain byte conversion val = strtoll(buffer, &end, 10); } delete [] buffer; return val; } #if B_BEOS_VERSION <= B_BEOS_VERSION_MAUI && !defined(__HAIKU__) bool operator==(const rgb_color &a, const rgb_color &b) { return a.red == b.red && a.green == b.green && a.blue == b.blue && a.alpha == b.alpha; } bool operator!=(const rgb_color &a, const rgb_color &b) { return !operator==(a, b); } #endif static BRect LineBounds(BPoint where, float length, bool vertical) { BRect result; result.SetLeftTop(where); result.SetRightBottom(where + BPoint(2, 2)); if (vertical) result.bottom = result.top + length; else result.right = result.left + length; return result; } SeparatorLine::SeparatorLine(BPoint where, float length, bool vertical, const char *name) : BView(LineBounds(where, length, vertical), name, B_FOLLOW_LEFT | B_FOLLOW_TOP, B_WILL_DRAW) { SetViewColor(ui_color(B_PANEL_BACKGROUND_COLOR)); SetLowColor(ui_color(B_PANEL_BACKGROUND_COLOR)); } void SeparatorLine::Draw(BRect) { BRect bounds(Bounds()); rgb_color hiliteColor = tint_color(ViewColor(), 1.5f); bool vertical = (bounds.left > bounds.right - 3); BeginLineArray(2); if (vertical) { AddLine(bounds.LeftTop(), bounds.LeftBottom(), hiliteColor); AddLine(bounds.LeftTop() + BPoint(1, 0), bounds.LeftBottom() + BPoint(1, 0), kWhite); } else { AddLine(bounds.LeftTop(), bounds.RightTop(), hiliteColor); AddLine(bounds.LeftTop() + BPoint(0, 1), bounds.RightTop() + BPoint(0, 1), kWhite); } EndLineArray(); } void HexDump(const void *buf, int32 length) { const int32 kBytesPerLine = 16; int32 offset; unsigned char *buffer = (unsigned char *)buf; for (offset = 0; ; offset += kBytesPerLine, buffer += kBytesPerLine) { int32 remain = length; int32 index; printf( "0x%06x: ", (int)offset); for (index = 0; index < kBytesPerLine; index++) { if (remain-- > 0) printf("%02x%c", buffer[index], remain > 0 ? ',' : ' '); else printf(" "); } remain = length; printf(" \'"); for (index = 0; index < kBytesPerLine; index++) { if (remain-- > 0) printf("%c", buffer[index] > ' ' ? buffer[index] : '.'); else printf(" "); } printf("\'\n"); length -= kBytesPerLine; if (length <= 0) break; } fflush(stdout); } void EnableNamedMenuItem(BMenu *menu, const char *itemName, bool on) { BMenuItem *item = menu->FindItem(itemName); if (item) item->SetEnabled(on); } void MarkNamedMenuItem(BMenu *menu, const char *itemName, bool on) { BMenuItem *item = menu->FindItem(itemName); if (item) item->SetMarked(on); } void EnableNamedMenuItem(BMenu *menu, uint32 commandName, bool on) { BMenuItem *item = menu->FindItem(commandName); if (item) item->SetEnabled(on); } void MarkNamedMenuItem(BMenu *menu, uint32 commandName, bool on) { BMenuItem *item = menu->FindItem(commandName); if (item) item->SetMarked(on); } void DeleteSubmenu(BMenuItem *submenuItem) { if (!submenuItem) return; BMenu *menu = submenuItem->Submenu(); if (!menu) return; for (;;) { BMenuItem *item = menu->RemoveItem((int32)0); if (!item) return; delete item; } } status_t GetAppSignatureFromAttr(BFile *file, char *result) { // This call is a performance improvement that // avoids using the BAppFileInfo API when retrieving the // app signature -- the call is expensive because by default // the resource fork is scanned to read the attribute #ifdef B_APP_FILE_INFO_IS_FAST BAppFileInfo appFileInfo(file); return appFileInfo.GetSignature(result); #else ssize_t readResult = file->ReadAttr(kAttrAppSignature, B_MIME_STRING_TYPE, 0, result, B_MIME_TYPE_LENGTH); if (readResult <= 0) return (status_t)readResult; return B_OK; #endif // B_APP_FILE_INFO_IS_FAST } status_t GetAppIconFromAttr(BFile *file, BBitmap *result, icon_size size) { // This call is a performance improvement that // avoids using the BAppFileInfo API when retrieving the // app icons -- the call is expensive because by default // the resource fork is scanned to read the icons //#ifdef B_APP_FILE_INFO_IS_FAST BAppFileInfo appFileInfo(file); return appFileInfo.GetIcon(result, size); //#else // // const char *attrName = kAttrIcon; // uint32 type = B_VECTOR_ICON_TYPE; // // // try vector icon // attr_info ainfo; // status_t ret = file->GetAttrInfo(attrName, &ainfo); // // if (ret == B_OK) { // uint8 buffer[ainfo.size]; // ssize_t readResult = file->ReadAttr(attrName, type, 0, buffer, // ainfo.size); // if (readResult == ainfo.size) { // if (BIconUtils::GetVectorIcon(buffer, ainfo.size, result) == B_OK) // return B_OK; // } // } // // // try again with R5 icons // attrName = size == B_LARGE_ICON ? kAttrLargeIcon : kAttrMiniIcon; // type = size == B_LARGE_ICON ? LARGE_ICON_TYPE : MINI_ICON_TYPE; // // ret = file->GetAttrInfo(attrName, &ainfo); // if (ret < B_OK) // return ret; // // uint8 buffer[ainfo.size]; // // ssize_t readResult = file->ReadAttr(attrName, type, 0, buffer, ainfo.size); // if (readResult <= 0) // return (status_t)readResult; // // if (result->ColorSpace() != B_CMAP8) { // ret = BIconUtils::ConvertFromCMAP8(buffer, size, size, size, result); // } else { // result->SetBits(buffer, result->BitsLength(), 0, B_CMAP8); // } // // return ret; //#endif // B_APP_FILE_INFO_IS_FAST } status_t GetFileIconFromAttr(BNode *file, BBitmap *result, icon_size size) { BNodeInfo fileInfo(file); return fileInfo.GetIcon(result, size); } void PrintToStream(rgb_color color) { printf("r:%x, g:%x, b:%x, a:%x\n", color.red, color.green, color.blue, color.alpha); } extern BMenuItem * EachMenuItem(BMenu *menu, bool recursive, BMenuItem *(*func)(BMenuItem *)) { int32 count = menu->CountItems(); for (int32 index = 0; index < count; index++) { BMenuItem *item = menu->ItemAt(index); BMenuItem *result = (func)(item); if (result) return result; if (recursive) { BMenu *submenu = menu->SubmenuAt(index); if (submenu) return EachMenuItem(submenu, true, func); } } return NULL; } extern const BMenuItem * EachMenuItem(const BMenu *menu, bool recursive, BMenuItem *(*func)(const BMenuItem *)) { int32 count = menu->CountItems(); for (int32 index = 0; index < count; index++) { BMenuItem *item = menu->ItemAt(index); BMenuItem *result = (func)(item); if (result) return result; if (recursive) { BMenu *submenu = menu->SubmenuAt(index); if (submenu) return EachMenuItem(submenu, true, func); } } return NULL; } PositionPassingMenuItem::PositionPassingMenuItem(const char *title, BMessage *message, char shortcut, uint32 modifiers) : BMenuItem(title, message, shortcut, modifiers) { } PositionPassingMenuItem::PositionPassingMenuItem(BMenu *menu, BMessage *message) : BMenuItem(menu, message) { } status_t PositionPassingMenuItem::Invoke(BMessage *message) { if (!Menu()) return B_ERROR; if (!IsEnabled()) return B_ERROR; if (!message) message = Message(); if (!message) return B_BAD_VALUE; BMessage clone(*message); clone.AddInt32("index", Menu()->IndexOf(this)); clone.AddInt64("when", system_time()); clone.AddPointer("source", this); // embed the invoke location of the menu so that we can create // a new folder, etc. on the spot BMenu *menu = Menu(); for (;;) { if (!menu->Supermenu()) break; menu = menu->Supermenu(); } // use the window position only, if the item was invoked from the menu // menu->Window() points to the window the item was invoked from if (dynamic_cast(menu->Window()) == NULL) { LooperAutoLocker lock(menu); if (lock.IsLocked()) { BPoint invokeOrigin(menu->Window()->Frame().LeftTop()); clone.AddPoint("be:invoke_origin", invokeOrigin); } } return BInvoker::Invoke(&clone); } bool BootedInSafeMode() { const char *safeMode = getenv("SAFEMODE"); return (safeMode && strcmp(safeMode, "yes") == 0); } float ComputeTypeAheadScore(const char *text, const char *match, size_t matchLength, bool wordMode) { // highest score: exact match const char* found = strcasestr(text, match); if (found != NULL) { if (found == text) return kExactMatchScore; return 1.f / (found - text); } // there was no exact match // second best: all characters at word beginnings if (wordMode) { float score = 0; for (int32 j = 0, k = 0; match[j]; j++) { while (text[k] && tolower(text[k]) != tolower(match[j])) { k++; } if (text[k] == '\0') { score = 0; break; } bool wordStart = k == 0 || isspace(text[k - 1]); if (wordStart) score++; if (j > 0) { bool wordEnd = !text[k + 1] || isspace(text[k + 1]); if (wordEnd) score += 0.3; if (match[j - 1] == text[k - 1]) score += 0.7; } score += 1.f / (k + 1); k++; } return score; } return -1; } void _ThrowOnError(status_t error, const char *DEBUG_ONLY(file), int32 DEBUG_ONLY(line)) { if (error != B_OK) { PRINT(("failing %s at %s:%d\n", strerror(error), file, line)); throw error; } } void _ThrowIfNotSize(ssize_t size, const char *DEBUG_ONLY(file), int32 DEBUG_ONLY(line)) { if (size < B_OK) { PRINT(("failing %s at %s:%d\n", strerror(size), file, line)); throw (status_t)size; } } void _ThrowOnError(status_t error, const char *DEBUG_ONLY(debugString), const char *DEBUG_ONLY(file), int32 DEBUG_ONLY(line)) { if (error != B_OK) { PRINT(("failing %s, %s at %s:%d\n", debugString, strerror(error), file, line)); throw error; } } } // namespace BPrivate