//--------------------------------------------------------------------------- #include #pragma hdrstop #include #include "NamedObjs.h" #include "Bookmarks.h" #include "Configuration.h" #include "HierarchicalStorage.h" #include "TextsCore.h" //--------------------------------------------------------------------------- #pragma package(smart_init) //--------------------------------------------------------------------------- __fastcall TBookmarks::TBookmarks(): TObject() { FSharedKey = TNamedObjectList::HiddenPrefix + L"shared"; FBookmarkLists = CreateSortedStringList(false, Types::dupError); } //--------------------------------------------------------------------------- __fastcall TBookmarks::~TBookmarks() { Clear(); SAFE_DESTROY(FBookmarkLists); } //--------------------------------------------------------------------------- void __fastcall TBookmarks::Clear() { for (int i = 0; i < FBookmarkLists->Count; i++) { delete FBookmarkLists->Objects[i]; } FBookmarkLists->Clear(); } //--------------------------------------------------------------------------- UnicodeString TBookmarks::Keys[] = { L"Local", L"Remote", L"ShortCuts", L"Options" }; //--------------------------------------------------------------------------- void __fastcall TBookmarks::Load(THierarchicalStorage * Storage) { for (int i = 0; i <= 3; i++) { if (Storage->OpenSubKey(Keys[i], false)) { TStrings * BookmarkKeys = new TStringList(); try { Storage->GetSubKeyNames(BookmarkKeys); for (int Index = 0; Index < BookmarkKeys->Count; Index++) { UnicodeString Key = BookmarkKeys->Strings[Index]; if (Storage->OpenSubKey(Key, false)) { TBookmarkList * BookmarkList = Bookmarks[Key]; if (!BookmarkList) { BookmarkList = new TBookmarkList(); FBookmarkLists->AddObject(Key, BookmarkList); } if (i < 3) { LoadLevel(Storage, L"", i, BookmarkList); } else { BookmarkList->LoadOptions(Storage); } Storage->CloseSubKey(); } } } __finally { delete BookmarkKeys; } Storage->CloseSubKey(); } } ModifyAll(false); } //--------------------------------------------------------------------------- void __fastcall TBookmarks::LoadLevel(THierarchicalStorage * Storage, const UnicodeString Key, int Index, TBookmarkList * BookmarkList) { TStrings * Names = new TStringList(); try { Storage->GetValueNames(Names); UnicodeString Name; UnicodeString Directory; TShortCut ShortCut = TShortCut(); for (int i = 0; i < Names->Count; i++) { Name = Names->Strings[i]; bool IsDirectory = (Index == 0) || (Index == 1); if (IsDirectory) { Directory = Storage->ReadString(Name, L""); } else { Directory = L""; // use only in case of malformed config ShortCut = (TShortCut)Storage->ReadInteger(Name, 0); } TBookmark * Bookmark; if (IsNumber(Name)) { DebugAssert(IsDirectory); // unless malformed Name = Directory; } if (!Name.IsEmpty()) { Bookmark = BookmarkList->FindByName(Key, Name); bool New; New = (Bookmark == NULL); if (New) { Bookmark = new TBookmark(); Bookmark->Node = Key; Bookmark->Name = Name; } switch (Index) { case 0: Bookmark->Local = Directory; break; case 1: Bookmark->Remote = Directory; break; case 2: Bookmark->ShortCut = ShortCut; break; } if (New) { BookmarkList->Add(Bookmark); } } } Storage->GetSubKeyNames(Names); for (int i = 0; i < Names->Count; i++) { Name = Names->Strings[i]; if (Storage->OpenSubKey(Name, false)) { LoadLevel(Storage, Key + (Key.IsEmpty() ? L"" : L"/") + Name, Index, BookmarkList); Storage->CloseSubKey(); } } } __finally { delete Names; } } //--------------------------------------------------------------------------- void __fastcall TBookmarks::Save(THierarchicalStorage * Storage, bool All) { for (int i = 0; i <= 3; i++) { if (Storage->OpenSubKey(Keys[i], true)) { for (int Index = 0; Index < FBookmarkLists->Count; Index++) { TBookmarkList * BookmarkList = dynamic_cast(FBookmarkLists->Objects[Index]); if (All || BookmarkList->Modified) { UnicodeString Key = FBookmarkLists->Strings[Index]; Storage->RecursiveDeleteSubKey(Key); if (Storage->OpenSubKey(Key, true)) { if (i < 3) { for (int IndexB = 0; IndexB < BookmarkList->Count; IndexB++) { TBookmark * Bookmark = BookmarkList->Bookmarks[IndexB]; // avoid creating empty subfolder if there's no shortcut if ((i == 0) || (i == 1) || ((i == 2) && (Bookmark->ShortCut != 0))) { bool HasNode = !Bookmark->Node.IsEmpty(); if (!HasNode || Storage->OpenSubKey(Bookmark->Node, true)) { switch (i) { case 0: Storage->WriteString(Bookmark->Name, Bookmark->Local); break; case 1: Storage->WriteString(Bookmark->Name, Bookmark->Remote); break; case 2: DebugAssert(Bookmark->ShortCut != 0); Storage->WriteInteger(Bookmark->Name, Bookmark->ShortCut); break; } if (HasNode) { Storage->CloseSubKey(); } } } } } else { BookmarkList->SaveOptions(Storage); } Storage->CloseSubKey(); } } } Storage->CloseSubKey(); } } if (!All) { ModifyAll(false); } } //--------------------------------------------------------------------------- void __fastcall TBookmarks::ModifyAll(bool Modify) { TBookmarkList * BookmarkList; for (int i = 0; i < FBookmarkLists->Count; i++) { BookmarkList = dynamic_cast(FBookmarkLists->Objects[i]); DebugAssert(BookmarkList); BookmarkList->Modified = Modify; } } //--------------------------------------------------------------------------- TBookmarkList * __fastcall TBookmarks::GetBookmarks(UnicodeString Index) { int I = FBookmarkLists->IndexOf(Index); if (I >= 0) { return dynamic_cast(FBookmarkLists->Objects[I]); } else { return NULL; } } //--------------------------------------------------------------------------- void __fastcall TBookmarks::SetBookmarks(UnicodeString Index, TBookmarkList * value) { int I = FBookmarkLists->IndexOf(Index); if (I >= 0) { TBookmarkList * BookmarkList; BookmarkList = dynamic_cast(FBookmarkLists->Objects[I]); BookmarkList->Assign(value); } else { TBookmarkList * BookmarkList = new TBookmarkList(); BookmarkList->Assign(value); FBookmarkLists->AddObject(Index, BookmarkList); } } //--------------------------------------------------------------------------- TBookmarkList * __fastcall TBookmarks::GetSharedBookmarks() { return GetBookmarks(FSharedKey); } //--------------------------------------------------------------------------- void __fastcall TBookmarks::SetSharedBookmarks(TBookmarkList * value) { SetBookmarks(FSharedKey, value); } //--------------------------------------------------------------------------- //--------------------------------------------------------------------------- __fastcall TBookmarkList::TBookmarkList(): TPersistent() { FModified = false; FBookmarks = new TStringList(); FBookmarks->CaseSensitive = false; FOpenedNodes = CreateSortedStringList(); } //--------------------------------------------------------------------------- __fastcall TBookmarkList::~TBookmarkList() { Clear(); SAFE_DESTROY(FBookmarks); SAFE_DESTROY(FOpenedNodes); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::Clear() { for (int i = 0; i < FBookmarks->Count; i++) { delete FBookmarks->Objects[i]; } FBookmarks->Clear(); FOpenedNodes->Clear(); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::Assign(TPersistent * Source) { TBookmarkList * SourceList; SourceList = dynamic_cast(Source); if (SourceList) { Clear(); for (int i = 0; i < SourceList->FBookmarks->Count; i++) { TBookmark * Bookmark = new TBookmark(); Bookmark->Assign(dynamic_cast(SourceList->FBookmarks->Objects[i])); Add(Bookmark); } FOpenedNodes->Assign(SourceList->FOpenedNodes); Modified = SourceList->Modified; } else { TPersistent::Assign(Source); } } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::LoadOptions(THierarchicalStorage * Storage) { FOpenedNodes->CommaText = Storage->ReadString(L"OpenedNodes", L""); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::SaveOptions(THierarchicalStorage * Storage) { Storage->WriteString(L"OpenedNodes", FOpenedNodes->CommaText); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::Add(TBookmark * Bookmark) { Insert(Count, Bookmark); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::InsertBefore(TBookmark * BeforeBookmark, TBookmark * Bookmark) { DebugAssert(BeforeBookmark); int I = FBookmarks->IndexOf(BeforeBookmark->Key); DebugAssert(I >= 0); Insert(I, Bookmark); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::MoveTo(TBookmark * ToBookmark, TBookmark * Bookmark, bool Before) { DebugAssert(ToBookmark != NULL); int NewIndex = FBookmarks->IndexOf(ToBookmark->Key); DebugAssert(Bookmark != NULL); int OldIndex = FBookmarks->IndexOf(Bookmark->Key); if (Before && (NewIndex > OldIndex)) { // otherwise item is moved after the item in the target index NewIndex--; } else if (!Before && (NewIndex < OldIndex)) { NewIndex++; } FModified = true; FBookmarks->Move(OldIndex, NewIndex); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::Insert(int Index, TBookmark * Bookmark) { DebugAssert(Bookmark); DebugAssert(!Bookmark->FOwner); DebugAssert(!Bookmark->Name.IsEmpty()); FModified = true; Bookmark->FOwner = this; if (FBookmarks->IndexOf(Bookmark->Key) >= 0) { throw Exception(FMTLOAD(DUPLICATE_BOOKMARK, (Bookmark->Name))); } FBookmarks->InsertObject(Index, Bookmark->Key, Bookmark); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::Delete(TBookmark * Bookmark) { DebugAssert(Bookmark); DebugAssert(Bookmark->FOwner == this); int I = IndexOf(Bookmark); DebugAssert(I >= 0); FModified = true; Bookmark->FOwner = NULL; FBookmarks->Delete(I); delete Bookmark; } //--------------------------------------------------------------------------- int __fastcall TBookmarkList::IndexOf(TBookmark * Bookmark) { return FBookmarks->IndexOf(Bookmark->Key); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::KeyChanged(int Index) { DebugAssert(Index < Count); TBookmark * Bookmark = dynamic_cast(FBookmarks->Objects[Index]); DebugAssert(FBookmarks->Strings[Index] != Bookmark->Key); if (FBookmarks->IndexOf(Bookmark->Key) >= 0) { throw Exception(FMTLOAD(DUPLICATE_BOOKMARK, (Bookmark->Name))); } FBookmarks->Strings[Index] = Bookmark->Key; } //--------------------------------------------------------------------------- TBookmark * __fastcall TBookmarkList::FindByName(const UnicodeString Node, const UnicodeString Name) { int I = FBookmarks->IndexOf(TBookmark::BookmarkKey(Node, Name)); TBookmark * Bookmark = I >= 0 ? dynamic_cast(FBookmarks->Objects[I]) : NULL; DebugAssert(!Bookmark || (Bookmark->Node == Node && Bookmark->Name == Name)); return Bookmark; } //--------------------------------------------------------------------------- TBookmark * __fastcall TBookmarkList::FindByShortCut(TShortCut ShortCut) { for (int Index = 0; Index < FBookmarks->Count; Index++) { if (Bookmarks[Index]->ShortCut == ShortCut) { return Bookmarks[Index]; } } return NULL; } //--------------------------------------------------------------------------- int __fastcall TBookmarkList::GetCount() { return FBookmarks->Count; } //--------------------------------------------------------------------------- TBookmark * __fastcall TBookmarkList::GetBookmarks(int Index) { TBookmark * Bookmark = dynamic_cast(FBookmarks->Objects[Index]); DebugAssert(Bookmark); return Bookmark; } //--------------------------------------------------------------------------- bool __fastcall TBookmarkList::GetNodeOpened(UnicodeString Index) { return (FOpenedNodes->IndexOf(Index) >= 0); } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::SetNodeOpened(UnicodeString Index, bool value) { int I = FOpenedNodes->IndexOf(Index); if ((I >= 0) != value) { if (value) { FOpenedNodes->Add(Index); } else { FOpenedNodes->Delete(I); } FModified = true; } } //--------------------------------------------------------------------------- void __fastcall TBookmarkList::ShortCuts(TShortCuts & ShortCuts) { for (int Index = 0; Index < Count; Index++) { TBookmark * Bookmark = Bookmarks[Index]; if (Bookmark->ShortCut != 0) { ShortCuts.Add(Bookmark->ShortCut); } } } //--------------------------------------------------------------------------- //--------------------------------------------------------------------------- __fastcall TBookmark::TBookmark() { FOwner = NULL; } //--------------------------------------------------------------------------- void __fastcall TBookmark::Assign(TPersistent * Source) { TBookmark * SourceBookmark; SourceBookmark = dynamic_cast(Source); if (SourceBookmark) { Name = SourceBookmark->Name; Local = SourceBookmark->Local; Remote = SourceBookmark->Remote; Node = SourceBookmark->Node; ShortCut = SourceBookmark->ShortCut; } else { TPersistent::Assign(Source); } } //--------------------------------------------------------------------------- void __fastcall TBookmark::SetName(const UnicodeString value) { if (Name != value) { int OldIndex = FOwner ? FOwner->IndexOf(this) : -1; UnicodeString OldName = FName; FName = value; try { Modify(OldIndex); } catch(...) { FName = OldName; throw; } } } //--------------------------------------------------------------------------- void __fastcall TBookmark::SetLocal(const UnicodeString value) { if (Local != value) { FLocal = value; Modify(-1); } } //--------------------------------------------------------------------------- void __fastcall TBookmark::SetRemote(const UnicodeString value) { if (Remote != value) { FRemote = value; Modify(-1); } } //--------------------------------------------------------------------------- void __fastcall TBookmark::SetNode(const UnicodeString value) { if (Node != value) { int OldIndex = FOwner ? FOwner->IndexOf(this) : -1; FNode = value; Modify(OldIndex); } } //--------------------------------------------------------------------------- void __fastcall TBookmark::SetShortCut(TShortCut value) { if (ShortCut != value) { FShortCut = value; Modify(-1); } } //--------------------------------------------------------------------------- void __fastcall TBookmark::Modify(int OldIndex) { if (FOwner) { FOwner->Modified = true; if (OldIndex >= 0) { FOwner->KeyChanged(OldIndex); } } } //--------------------------------------------------------------------------- UnicodeString __fastcall TBookmark::BookmarkKey(const UnicodeString Node, const UnicodeString Name) { return FORMAT(L"%s\1%s", (Node, Name)); } //--------------------------------------------------------------------------- UnicodeString __fastcall TBookmark::GetKey() { return BookmarkKey(Node, Name); } //--------------------------------------------------------------------------- UnicodeString __fastcall TBookmark::GetSideDirectory(TOperationSide Side) { return (Side == osLocal) ? Local : Remote; }