//--------------------------------------------------------------------------- #include #pragma hdrstop #include "Common.h" #include "WinConfiguration.h" #include "Exceptions.h" #include "Bookmarks.h" #include "Terminal.h" #include "TextsWin.h" #include "WinInterface.h" #include "GUITools.h" #include "Tools.h" #include "Setup.h" #include "Security.h" #include "TerminalManager.h" #include "Cryptography.h" #include #include #include #include #include #include #include "FileInfo.h" //--------------------------------------------------------------------------- #pragma package(smart_init) //--------------------------------------------------------------------------- TWinConfiguration * WinConfiguration = NULL; //--------------------------------------------------------------------------- static UnicodeString NotepadName(L"notepad.exe"); //--------------------------------------------------------------------------- __fastcall TEditorData::TEditorData() : FileMask(L"*.*"), Editor(edInternal), ExternalEditor(L""), ExternalEditorText(false), SDIExternalEditor(false), DetectMDIExternalEditor(false) { } //--------------------------------------------------------------------------- __fastcall TEditorData::TEditorData(const TEditorData & Source) : FileMask(Source.FileMask), Editor(Source.Editor), ExternalEditor(Source.ExternalEditor), ExternalEditorText(Source.ExternalEditorText), SDIExternalEditor(Source.SDIExternalEditor), DetectMDIExternalEditor(Source.DetectMDIExternalEditor) { } //--------------------------------------------------------------------------- #define C(Property) (Property == rhd.Property) bool __fastcall TEditorData::operator==(const TEditorData & rhd) const { return C(FileMask) && C(Editor) && C(ExternalEditor) && C(ExternalEditorText) && C(SDIExternalEditor) && C(DetectMDIExternalEditor) && true; } #undef C //--------------------------------------------------------------------------- bool __fastcall TEditorData::DecideExternalEditorText(UnicodeString ExternalEditor) { bool Result = false; // By default we use default transfer mode (binary), // as all reasonable 3rd party editors support all EOL styles. // A notable exception is Windows Notepad, so here's an exception for it. ReformatFileNameCommand(ExternalEditor); UnicodeString ProgramName = ExtractProgramName(ExternalEditor); // We explicitly do not use TEditorPreferences::GetDefaultExternalEditor(), // as we need to explicitly refer to the Notepad, even if the default external // editor ever changes UnicodeString NotepadProgramName = ExtractProgramName(NotepadName); if (SameText(ProgramName, NotepadProgramName)) { Result = true; } return Result; } //--------------------------------------------------------------------------- void __fastcall TEditorData::DecideExternalEditorText() { if (DecideExternalEditorText(ExternalEditor)) { ExternalEditorText = true; } } //--------------------------------------------------------------------------- __fastcall TEditorPreferences::TEditorPreferences() { } //--------------------------------------------------------------------------- __fastcall TEditorPreferences::TEditorPreferences(const TEditorData & Data) : FData(Data) { } //--------------------------------------------------------------------------- bool __fastcall TEditorPreferences::operator==(const TEditorPreferences & rhp) const { return (FData == rhp.FData); } #undef C //--------------------------------------------------------------------------- bool __fastcall TEditorPreferences::Matches(const UnicodeString FileName, bool Local, const TFileMasks::TParams & Params) const { return FData.FileMask.Matches(FileName, Local, false, &Params); } //--------------------------------------------------------------------------- UnicodeString __fastcall TEditorPreferences::GetDefaultExternalEditor() { return NotepadName; } //--------------------------------------------------------------------------- void __fastcall TEditorPreferences::LegacyDefaults() { FData.ExternalEditor = GetDefaultExternalEditor(); FData.DecideExternalEditorText(); } //--------------------------------------------------------------------------- UnicodeString __fastcall TEditorPreferences::ExtractExternalEditorName() const { DebugAssert(FData.Editor == edExternal); UnicodeString ExternalEditor = FData.ExternalEditor; ReformatFileNameCommand(ExternalEditor); // Trim is a workaround for unknown problem with "notepad " (2 trailing spaces) return ExtractProgramName(ExternalEditor).Trim(); } //--------------------------------------------------------------------------- void __fastcall TEditorPreferences::Load(THierarchicalStorage * Storage, bool Legacy) { if (!Legacy) { FData.FileMask = Storage->ReadString(L"FileMask", FData.FileMask.Masks); } FData.Editor = (TEditor)Storage->ReadInteger(L"Editor", FData.Editor); FData.ExternalEditor = Storage->ReadString(L"ExternalEditor", FData.ExternalEditor); FData.ExternalEditorText = Storage->ReadBool(L"ExternalEditorText", FData.ExternalEditorText); FData.SDIExternalEditor = Storage->ReadBool(L"SDIExternalEditor", FData.SDIExternalEditor); FData.DetectMDIExternalEditor = Storage->ReadBool(L"DetectMDIExternalEditor", FData.DetectMDIExternalEditor); } //--------------------------------------------------------------------------- void __fastcall TEditorPreferences::Save(THierarchicalStorage * Storage) const { Storage->WriteString(L"FileMask", FData.FileMask.Masks); Storage->WriteInteger(L"Editor", FData.Editor); Storage->WriteString(L"ExternalEditor", FData.ExternalEditor); Storage->WriteBool(L"ExternalEditorText", FData.ExternalEditorText); Storage->WriteBool(L"SDIExternalEditor", FData.SDIExternalEditor); Storage->WriteBool(L"DetectMDIExternalEditor", FData.DetectMDIExternalEditor); } //--------------------------------------------------------------------------- TEditorData * __fastcall TEditorPreferences::GetData() { // returning non-const data, possible data change, invalidate cached name FName = L""; return &FData; }; //--------------------------------------------------------------------------- UnicodeString __fastcall TEditorPreferences::GetName() const { if (FName.IsEmpty()) { if (FData.Editor == edInternal) { // StripHotkey is relic from times when INTERNAL_EDITOR_NAME was used // also for the menu item caption FName = StripHotkey(LoadStr(INTERNAL_EDITOR_NAME)); } else if (FData.Editor == edOpen) { FName = StripHotkey(LoadStr(OPEN_EDITOR_NAME)); } else { UnicodeString Program, Params, Dir; UnicodeString ExternalEditor = FData.ExternalEditor; ReformatFileNameCommand(ExternalEditor); SplitCommand(ExternalEditor, Program, Params, Dir); FName = ExtractFileName(Program); int P = FName.LastDelimiter(L"."); if (P > 0) { FName.SetLength(P - 1); } if (FName.ByteType(1) == mbSingleByte) { if (FName.UpperCase() == FName) { FName = FName.LowerCase(); } if (FName.LowerCase() == FName) { FName = FName.SubString(1, 1).UpperCase() + FName.SubString(2, FName.Length() - 1); } } } } return FName; } //--------------------------------------------------------------------------- //--------------------------------------------------------------------------- __fastcall TEditorList::TEditorList() { Init(); } //--------------------------------------------------------------------------- void __fastcall TEditorList::Init() { FEditors = new TList(); FModified = false; } //--------------------------------------------------------------------------- __fastcall TEditorList::~TEditorList() { Clear(); delete FEditors; } //--------------------------------------------------------------------- void __fastcall TEditorList::Modify() { FModified = true; } //--------------------------------------------------------------------------- void __fastcall TEditorList::Saved() { FModified = false; } //--------------------------------------------------------------------------- TEditorList & __fastcall TEditorList::operator=(const TEditorList & rhl) { Clear(); for (int Index = 0; Index < rhl.Count; Index++) { Add(new TEditorPreferences(*rhl.Editors[Index])); } // there should be comparison of with the assigned list, but we rely on caller // to do it instead (TWinConfiguration::SetEditorList) Modify(); return *this; } //--------------------------------------------------------------------------- bool __fastcall TEditorList::operator==(const TEditorList & rhl) const { bool Result = (Count == rhl.Count); if (Result) { int i = 0; while ((i < Count) && Result) { Result = (*Editors[i]) == (*rhl.Editors[i]); i++; } } return Result; } //--------------------------------------------------------------------------- void __fastcall TEditorList::Clear() { for (int i = 0; i < Count; i++) { delete Editors[i]; } FEditors->Clear(); } //--------------------------------------------------------------------------- void __fastcall TEditorList::Add(TEditorPreferences * Editor) { Insert(Count, Editor); } //--------------------------------------------------------------------------- void __fastcall TEditorList::Insert(int Index, TEditorPreferences * Editor) { FEditors->Insert(Index, reinterpret_cast(Editor)); Modify(); } //--------------------------------------------------------------------------- void __fastcall TEditorList::Change(int Index, TEditorPreferences * Editor) { if (!((*Editors[Index]) == *Editor)) { delete Editors[Index]; FEditors->Items[Index] = (reinterpret_cast(Editor)); Modify(); } else { delete Editor; } } //--------------------------------------------------------------------------- void __fastcall TEditorList::Move(int CurIndex, int NewIndex) { if (CurIndex != NewIndex) { FEditors->Move(CurIndex, NewIndex); Modify(); } } //--------------------------------------------------------------------------- void __fastcall TEditorList::Delete(int Index) { DebugAssert((Index >= 0) && (Index < Count)); delete Editors[Index]; FEditors->Delete(Index); Modify(); } //--------------------------------------------------------------------------- const TEditorPreferences * __fastcall TEditorList::Find( const UnicodeString FileName, bool Local, const TFileMasks::TParams & Params) const { const TEditorPreferences * Result = NULL; int i = 0; while ((i < FEditors->Count) && (Result == NULL)) { Result = Editors[i]; if (!Result->Matches(FileName, Local, Params)) { Result = NULL; } i++; } return Result; } //--------------------------------------------------------------------------- void __fastcall TEditorList::Load(THierarchicalStorage * Storage) { int Index = 0; bool Next; do { UnicodeString Name = IntToStr(Index); TEditorPreferences * Editor = NULL; try { Next = Storage->OpenSubKey(Name, false); if (Next) { try { Editor = new TEditorPreferences(); Editor->Load(Storage, false); } __finally { Storage->CloseSubKey(); } } } catch(...) { delete Editor; throw; } if (Editor != NULL) { FEditors->Add(reinterpret_cast(Editor)); } Index++; } while (Next); FModified = false; } //--------------------------------------------------------------------------- void __fastcall TEditorList::Save(THierarchicalStorage * Storage) const { Storage->ClearSubKeys(); for (int Index = 0; Index < Count; Index++) { if (Storage->OpenSubKey(IntToStr(Index), true)) { try { Editors[Index]->Save(Storage); } __finally { Storage->CloseSubKey(); } } } } //--------------------------------------------------------------------------- int __fastcall TEditorList::GetCount() const { int X = FEditors->Count; return X; } //--------------------------------------------------------------------------- const TEditorPreferences * __fastcall TEditorList::GetEditor(int Index) const { return reinterpret_cast(FEditors->Items[Index]); } //--------------------------------------------------------------------------- bool __fastcall TEditorList::IsDefaultList() const { bool Result = true; for (int Index = 0; Result && (Index < Count); Index++) { const TEditorPreferences * Editor = GetEditor(Index); if (Editor->Data->Editor == edInternal) { // noop (keeps Result true) } else if (Editor->Data->Editor == edExternal) { UnicodeString ExternalEditor = Editor->ExtractExternalEditorName(); UnicodeString DefaultExternalEditor = ExtractProgramName(TEditorPreferences::GetDefaultExternalEditor()); Result = SameText(ExternalEditor, DefaultExternalEditor); } else { Result = false; } } return Result; } //--------------------------------------------------------------------------- //--------------------------------------------------------------------------- __fastcall TWinConfiguration::TWinConfiguration(): TCustomWinConfiguration() { FInvalidDefaultTranslationMessage = L""; FDDExtInstalled = -1; FBookmarks = new TBookmarks(); FCustomCommandList = new TCustomCommandList(); FExtensionList = new TCustomCommandList(); FEditorList = new TEditorList(); FDefaultUpdatesPeriod = 0; FDontDecryptPasswords = 0; FMasterPasswordSession = 0; FMasterPasswordSessionAsked = false; FSystemIconFont.reset(new TFont()); FCustomCommandOptions.reset(new TStringList()); FCustomCommandOptionsModified = false; UpdateSystemIconFont(); Default(); try { CheckTranslationVersion( GetResourceModuleName(Application->Name, ModuleFileName().c_str()), true); } catch(Exception & E) { FInvalidDefaultTranslationMessage = E.Message; } } //--------------------------------------------------------------------------- __fastcall TWinConfiguration::~TWinConfiguration() { if (!FTemporarySessionFile.IsEmpty()) DeleteFile(ApiPath(FTemporarySessionFile)); ClearTemporaryLoginData(); delete FBookmarks; delete FCustomCommandList; delete FExtensionList; delete FEditorList; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::Default() { FCustomCommandsDefaults = true; FCustomCommandOptionsModified = false; TCustomWinConfiguration::Default(); int WorkAreaWidthScaled = DimensionToDefaultPixelsPerInch(Screen->WorkAreaWidth); int WorkAreaHeightScaled = DimensionToDefaultPixelsPerInch(Screen->WorkAreaHeight); UnicodeString PixelsPerInchToolbarValue = "PixelsPerInch=" + SaveDefaultPixelsPerInch(); FDDAllowMove = false; FDDAllowMoveInit = false; FDDTransferConfirmation = asAuto; FDDTemporaryDirectory = L""; FDDWarnLackOfTempSpace = true; FDDWarnLackOfTempSpaceRatio = 1.1; FDDExtEnabled = DDExtInstalled; FDDExtTimeout = MSecsPerSec; FDeleteToRecycleBin = true; FSelectDirectories = false; FSelectMask = L"*.*"; FShowHiddenFiles = false; FFormatSizeBytes = fbKilobytes; FShowInaccesibleDirectories = true; FConfirmTransferring = true; FConfirmDeleting = true; FConfirmRecycling = true; FConfirmClosingSession = true; FDoubleClickAction = dcaEdit; FCopyOnDoubleClickConfirmation = false; FDimmHiddenFiles = true; FRenameWholeName = false; FAutoStartSession = L""; FExpertMode = true; FUseLocationProfiles = false; FUseSharedBookmarks = false; FDefaultDirIsHome = true; FDDDeleteDelay = 120; FTemporaryDirectoryAppendSession = false; FTemporaryDirectoryDeterministic = false; FTemporaryDirectoryAppendPath = true; FTemporaryDirectoryCleanup = true; FConfirmTemporaryDirectoryCleanup = true; FPreservePanelState = true; FLastStoredSession = L""; // deliberately not being saved, so that when saving ad-hoc workspace, // we do not offer to overwrite the last saved workspace, what may be undesirable FLastWorkspace = L""; FAutoSaveWorkspace = false; FAutoSaveWorkspacePasswords = false; FAutoWorkspace = L""; FPathInCaption = picShort; FMinimizeToTray = false; FMinimizeToTrayOnce = false; FBalloonNotifications = true; FNotificationsTimeout = 10; FNotificationsStickTime = 2; FCopyParamAutoSelectNotice = true; FLockToolbars = false; FSelectiveToolbarText = true; FAutoOpenInPutty = false; FRefreshRemotePanel = false; FRefreshRemotePanelInterval = TDateTime(0, 1, 0, 0); FPanelFont.FontName = L""; FPanelFont.FontSize = 0; FPanelFont.FontStyle = 0; FPanelFont.FontCharset = DEFAULT_CHARSET; FFullRowSelect = false; FOfferedEditorAutoConfig = false; FVersionHistory = L""; AddVersionToHistory(); FUseMasterPassword = false; FPlainMasterPasswordEncrypt = L""; FPlainMasterPasswordDecrypt = L""; FMasterPasswordVerifier = L""; FOpenedStoredSessionFolders = L""; FAutoImportedFromPuttyOrFilezilla = false; FGenerateUrlComponents = -1; FGenerateUrlCodeTarget = guctUrl; FGenerateUrlScriptFormat = sfScriptFile; FGenerateUrlAssemblyLanguage = alCSharp; FExternalSessionInExistingInstance = true; FKeepOpenWhenNoSession = false; FLocalIconsByExt = false; FBidiModeOverride = lfoLanguageIfRecommended; FFlipChildrenOverride = lfoLanguageIfRecommended; FShowTips = true; FTipsSeen = L""; FTipsShown = Now(); FRunsSinceLastTip = 0; FExtensionsDeleted = L""; HonorDrivePolicy = true; FEditor.Font.FontName = DefaultFixedWidthFontName; FEditor.Font.FontSize = DefaultFixedWidthFontSize; FEditor.Font.FontStyle = 0; FEditor.Font.FontCharset = DEFAULT_CHARSET; FEditor.FontColor = TColor(0); FEditor.BackgroundColor = TColor(0); FEditor.WordWrap = false; FEditor.FindText = L""; FEditor.ReplaceText = L""; FEditor.FindMatchCase = false; FEditor.FindWholeWord = false; FEditor.FindDown = true; FEditor.TabSize = 8; FEditor.MaxEditors = 500; FEditor.EarlyClose = 2; // seconds FEditor.SDIShellEditor = false; FEditor.WindowParams = L""; FEditor.Encoding = CP_ACP; FEditor.WarnOnEncodingFallback = true; FEditor.WarnOrLargeFileSize = true; FQueueView.Height = 140; FQueueView.HeightPixelsPerInch = USER_DEFAULT_SCREEN_DPI; // with 1000 pixels wide screen, both interfaces are wide enough to fit wider queue FQueueView.Layout = UnicodeString((WorkAreaWidthScaled > 1000) ? L"70,250,250,80,80,80,100" : L"70,160,160,80,80,80,100") + // WORKAROUND (the comma), see GetListViewStr L",;" + SaveDefaultPixelsPerInch(); FQueueView.Show = qvHideWhenEmpty; FQueueView.LastHideShow = qvHideWhenEmpty; FQueueView.ToolBar = true; FQueueView.Label = true; FEnableQueueByDefault = true; FUpdates.Period = FDefaultUpdatesPeriod; FUpdates.LastCheck = 0; FUpdates.HaveResults = false; FUpdates.ShownResults = false; FUpdates.BetaVersions = asAuto; FUpdates.ShowOnStartup = true; FUpdates.AuthenticationEmail = L""; // for backward compatibility the default is decided based on value of ProxyHost FUpdates.ConnectionType = (TConnectionType)-1; FUpdates.ProxyHost = L""; // keep empty (see above) FUpdates.ProxyPort = 8080; FUpdates.Results.Reset(); FUpdates.DotNetVersion = L""; FUpdates.ConsoleVersion = L""; FLogWindowOnStartup = true; FLogWindowParams = FormatDefaultWindowParams(500, 400); int ExplorerWidth = Min(WorkAreaWidthScaled - 40, 960); int ExplorerHeight = Min(WorkAreaHeightScaled - 30, 720); FScpExplorer.WindowParams = FormatDefaultWindowParams(ExplorerWidth, ExplorerHeight); // WORKAROUND (the semicolon, see TCustomListViewColProperties.GetParamsStr, and see other instances below) FScpExplorer.DirViewParams = L"0;1;0|150,1;70,1;150,1;79,1;62,1;55,0;20,0;150,0;125,0;@" + SaveDefaultPixelsPerInch() + L"|6;7;8;0;1;2;3;4;5"; FScpExplorer.ToolbarsLayout = UnicodeString( L"Queue_Visible=1,Queue_LastDock=QueueDock,Queue_DockRow=0,Queue_DockPos=-1,Queue_FloatLeft=0,Queue_FloatTop=0,Queue_FloatRightX=0," "Menu_Visible=1,Menu_DockedTo=TopDock,Menu_LastDock=TopDock,Menu_DockRow=0,Menu_DockPos=0,Menu_FloatLeft=0,Menu_FloatTop=0,Menu_FloatRightX=0," "Buttons_Visible=1,Buttons_DockedTo=TopDock,Buttons_LastDock=TopDock,Buttons_DockRow=2,Buttons_DockPos=0,Buttons_FloatLeft=0,Buttons_FloatTop=0,Buttons_FloatRightX=0," "Selection_Visible=0,Selection_DockedTo=TopDock,Selection_LastDock=TopDock,Selection_DockRow=3,Selection_DockPos=0,Selection_FloatLeft=227,Selection_FloatTop=445,Selection_FloatRightX=0," "Session_Visible=0,Session_DockedTo=TopDock,Session_LastDock=TopDock,Session_DockRow=6,Session_DockPos=0,Session_FloatLeft=39,Session_FloatTop=160,Session_FloatRightX=0," "Preferences_Visible=1,Preferences_DockedTo=TopDock,Preferences_LastDock=TopDock,Preferences_DockRow=4,Preferences_DockPos=0,Preferences_FloatLeft=0,Preferences_FloatTop=0,Preferences_FloatRightX=0," "Sort_Visible=0,Sort_DockedTo=TopDock,Sort_LastDock=TopDock,Sort_DockRow=5,Sort_DockPos=0,Sort_FloatLeft=0,Sort_FloatTop=0,Sort_FloatRightX=0," "Address_Visible=1,Address_DockedTo=TopDock,Address_LastDock=TopDock,Address_DockRow=1,Address_DockPos=0,Address_FloatLeft=0,Address_FloatTop=0,Address_FloatRightX=0," "Updates_Visible=1,Updates_DockedTo=TopDock,Updates_LastDock=TopDock,Updates_DockRow=4,Updates_DockPos=416,Updates_FloatLeft=0,Updates_FloatTop=0,Updates_FloatRightX=0," "Transfer_Visible=1,Transfer_DockedTo=TopDock,Transfer_LastDock=TopDock,Transfer_DockRow=4,Transfer_DockPos=194,Transfer_FloatLeft=0,Transfer_FloatTop=0,Transfer_FloatRightX=0," "CustomCommands_Visible=0,CustomCommands_DockedTo=TopDock,CustomCommands_LastDock=TopDock,CustomCommands_DockRow=7,CustomCommands_DockPos=0,CustomCommands_FloatLeft=0,CustomCommands_FloatTop=0,CustomCommands_FloatRightX=0,") + PixelsPerInchToolbarValue; FScpExplorer.SessionsTabs = true; FScpExplorer.StatusBar = true; FScpExplorer.LastLocalTargetDirectory = GetPersonalFolder(); FScpExplorer.ViewStyle = 0; /* vsIcon */ FScpExplorer.ShowFullAddress = true; FScpExplorer.DriveView = true; FScpExplorer.DriveViewWidth = 180; FScpExplorer.DriveViewWidthPixelsPerInch = USER_DEFAULT_SCREEN_DPI; int CommanderWidth = Min(WorkAreaWidthScaled - 40, 1090); int CommanderHeight = Min(WorkAreaHeightScaled - 30, 700); FScpCommander.WindowParams = FormatDefaultWindowParams(CommanderWidth, CommanderHeight); FScpCommander.LocalPanelWidth = 0.5; FScpCommander.SwappedPanels = false; FScpCommander.SessionsTabs = true; FScpCommander.StatusBar = true; FScpCommander.NortonLikeMode = nlKeyboard; FScpCommander.PreserveLocalDirectory = false; // Toolbar2_FloatRightX=1 makes keybar apper initially "in column" when undocked FScpCommander.ToolbarsLayout = UnicodeString( L"Queue_Visible=1,Queue_LastDock=QueueDock,Queue_DockRow=0,Queue_DockPos=-1,Queue_FloatLeft=0,Queue_FloatTop=0,Queue_FloatRightX=0," "Menu_Visible=1,Menu_DockedTo=TopDock,Menu_LastDock=TopDock,Menu_DockRow=0,Menu_DockPos=0,Menu_FloatLeft=0,Menu_FloatTop=0,Menu_FloatRightX=0," "Preferences_Visible=1,Preferences_DockedTo=TopDock,Preferences_LastDock=TopDock,Preferences_DockRow=1,Preferences_DockPos=228,Preferences_FloatLeft=0,Preferences_FloatTop=0,Preferences_FloatRightX=0," "Session_Visible=0,Session_DockedTo=TopDock,Session_LastDock=TopDock,Session_DockRow=1,Session_DockPos=602,Session_FloatLeft=380,Session_FloatTop=197,Session_FloatRightX=0," "Sort_Visible=0,Sort_DockedTo=TopDock,Sort_LastDock=TopDock,Sort_DockRow=2,Sort_DockPos=0,Sort_FloatLeft=0,Sort_FloatTop=0,Sort_FloatRightX=0," "Commands_Visible=1,Commands_DockedTo=TopDock,Commands_LastDock=TopDock,Commands_DockRow=1,Commands_DockPos=0,Commands_FloatLeft=0,Commands_FloatTop=0,Commands_FloatRightX=0," "Updates_Visible=1,Updates_DockedTo=TopDock,Updates_LastDock=TopDock,Updates_DockRow=1,Updates_DockPos=619,Updates_FloatLeft=0,Updates_FloatTop=0,Updates_FloatRightX=0," "Transfer_Visible=1,Transfer_DockedTo=TopDock,Transfer_LastDock=TopDock,Transfer_DockRow=1,Transfer_DockPos=364,Transfer_FloatLeft=0,Transfer_FloatTop=0,Transfer_FloatRightX=0," "CustomCommands_Visible=0,CustomCommands_DockedTo=TopDock,CustomCommands_LastDock=TopDock,CustomCommands_DockRow=3,CustomCommands_DockPos=0,CustomCommands_FloatLeft=0,CustomCommands_FloatTop=0,CustomCommands_FloatRightX=0," "RemoteHistory_Visible=1,RemoteHistory_DockedTo=RemoteTopDock,RemoteHistory_LastDock=RemoteTopDock,RemoteHistory_DockRow=0,RemoteHistory_DockPos=172,RemoteHistory_FloatLeft=0,RemoteHistory_FloatTop=0,RemoteHistory_FloatRightX=0," "RemoteNavigation_Visible=1,RemoteNavigation_DockedTo=RemoteTopDock,RemoteNavigation_LastDock=RemoteTopDock,RemoteNavigation_DockRow=0,RemoteNavigation_DockPos=252,RemoteNavigation_FloatLeft=0,RemoteNavigation_FloatTop=0,RemoteNavigation_FloatRightX=0," "RemotePath_Visible=1,RemotePath_DockedTo=RemoteTopDock,RemotePath_LastDock=RemoteTopDock,RemotePath_DockRow=0,RemotePath_DockPos=0,RemotePath_FloatLeft=0,RemotePath_FloatTop=0,RemotePath_FloatRightX=0," "RemoteFile_Visible=1,RemoteFile_DockedTo=RemoteTopDock,RemoteFile_LastDock=RemoteTopDock,RemoteFile_DockRow=1,RemoteFile_DockPos=0,RemoteFile_FloatLeft=0,RemoteFile_FloatTop=0,RemoteFile_FloatRightX=0," "RemoteSelection_Visible=1,RemoteSelection_DockedTo=RemoteTopDock,RemoteSelection_LastDock=RemoteTopDock,RemoteSelection_DockRow=1,RemoteSelection_DockPos=345,RemoteSelection_FloatLeft=0,RemoteSelection_FloatTop=0,RemoteSelection_FloatRightX=0," "LocalHistory_Visible=1,LocalHistory_DockedTo=LocalTopDock,LocalHistory_LastDock=LocalTopDock,LocalHistory_DockRow=0,LocalHistory_DockPos=207,LocalHistory_FloatLeft=0,LocalHistory_FloatTop=0,LocalHistory_FloatRightX=0," "LocalNavigation_Visible=1,LocalNavigation_DockedTo=LocalTopDock,LocalNavigation_LastDock=LocalTopDock,LocalNavigation_DockRow=0,LocalNavigation_DockPos=287,LocalNavigation_FloatLeft=0,LocalNavigation_FloatTop=0,LocalNavigation_FloatRightX=0," "LocalPath_Visible=1,LocalPath_DockedTo=LocalTopDock,LocalPath_LastDock=LocalTopDock,LocalPath_DockRow=0,LocalPath_DockPos=0,LocalPath_FloatLeft=0,LocalPath_FloatTop=0,LocalPath_FloatRightX=0," "LocalFile_Visible=1,LocalFile_DockedTo=LocalTopDock,LocalFile_LastDock=LocalTopDock,LocalFile_DockRow=1,LocalFile_DockPos=0,LocalFile_FloatLeft=0,LocalFile_FloatTop=0,LocalFile_FloatRightX=0," "LocalSelection_Visible=1,LocalSelection_DockedTo=LocalTopDock,LocalSelection_LastDock=LocalTopDock,LocalSelection_DockRow=1,LocalSelection_DockPos=329,LocalSelection_FloatLeft=0,LocalSelection_FloatTop=0,LocalSelection_FloatRightX=0," "Toolbar2_Visible=0,Toolbar2_DockedTo=BottomDock,Toolbar2_LastDock=BottomDock,Toolbar2_DockRow=1,Toolbar2_DockPos=0,Toolbar2_FloatLeft=0,Toolbar2_FloatTop=0,Toolbar2_FloatRightX=1," "CommandLine_Visible=0,CommandLine_DockedTo=BottomDock,CommandLine_LastDock=BottomDock,CommandLine_DockRow=0,CommandLine_DockPos=0,CommandLine_FloatLeft=0,CommandLine_FloatTop=0,CommandLine_FloatRightX=0,") + PixelsPerInchToolbarValue; FScpCommander.CurrentPanel = osLocal; FScpCommander.CompareByTime = true; FScpCommander.CompareBySize = false; FScpCommander.TreeOnLeft = false; FScpCommander.ExplorerKeyboardShortcuts = false; FScpCommander.SystemContextMenu = false; FScpCommander.RemotePanel.DirViewParams = L"0;1;0|150,1;70,1;150,1;79,1;62,1;55,0;20,0;150,0;125,0;@" + SaveDefaultPixelsPerInch() + L"|6;7;8;0;1;2;3;4;5"; FScpCommander.RemotePanel.StatusBar = true; FScpCommander.RemotePanel.DriveView = false; FScpCommander.RemotePanel.DriveViewHeight = 100; FScpCommander.RemotePanel.DriveViewHeightPixelsPerInch = USER_DEFAULT_SCREEN_DPI; FScpCommander.RemotePanel.DriveViewWidth = 100; FScpCommander.RemotePanel.DriveViewWidthPixelsPerInch = USER_DEFAULT_SCREEN_DPI; FScpCommander.LocalPanel.DirViewParams = L"0;1;0|150,1;70,1;120,1;150,1;55,0;55,0;@" + SaveDefaultPixelsPerInch() + L"|5;0;1;2;3;4"; FScpCommander.LocalPanel.StatusBar = true; FScpCommander.LocalPanel.DriveView = false; FScpCommander.LocalPanel.DriveViewHeight = 100; FScpCommander.LocalPanel.DriveViewHeightPixelsPerInch = USER_DEFAULT_SCREEN_DPI; FScpCommander.LocalPanel.DriveViewWidth = 100; FScpCommander.LocalPanel.DriveViewWidthPixelsPerInch = USER_DEFAULT_SCREEN_DPI; FBookmarks->Clear(); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::DefaultLocalized() { TGUIConfiguration::DefaultLocalized(); if (FCustomCommandsDefaults) { FCustomCommandList->Clear(); FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_EXECUTE), L"\"./!\"", 0); FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_TOUCH), L"touch \"!\"", ccApplyToDirectories | ccRecursive); FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_TAR), FORMAT(L"tar -cz -f \"!?%s?archive.tgz!\" !&", (LoadStr(CUSTOM_COMMAND_TAR_ARCHIVE))), ccApplyToDirectories); FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_UNTAR), FORMAT(L"tar -xz --directory=\"!?%s?.!\" -f \"!\"", (LoadStr(CUSTOM_COMMAND_UNTAR_DIRECTORY))), 0); FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_GREP), FORMAT(L"grep \"!?%s?!\" !&", (LoadStr(CUSTOM_COMMAND_GREP_PATTERN))), ccShowResults); FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_FC), L"cmd /c fc \"!\" \"\!^!\" | more && pause", ccLocal); FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_PRINT), L"notepad.exe /p \"!\"", ccLocal); FCustomCommandList->Reset(); FCustomCommandsDefaults = true; } } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::DetectRegistryStorage(HKEY RootKey) { bool Result = false; TRegistryStorage * Storage = new TRegistryStorage(RegistryStorageKey, RootKey); try { if (Storage->OpenRootKey(false)) { Result = true; Storage->CloseSubKey(); } } __finally { delete Storage; } return Result; } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::CanWriteToStorage() { bool Result = false; try { THierarchicalStorage * Storage = CreateConfigStorage(); try { Storage->AccessMode = smReadWrite; // This is actually not very good test, as we end up potentially with // the very same config, and TIniFileStorage file won't even try to // write the file then. Lucliky, we use this for empty config only, // so we end up with at least an empty section. if (Storage->OpenSubKey(ConfigurationSubKey, true)) { Storage->WriteBool(L"WriteTest", true); Storage->DeleteValue(L"WriteTest"); } Storage->Flush(); } __finally { delete Storage; } Result = true; } catch(...) { } return Result; } //--------------------------------------------------------------------------- TStorage __fastcall TWinConfiguration::GetStorage() { if (FStorage == stDetect) { if (FindResourceEx(NULL, RT_RCDATA, L"WINSCP_SESSION", MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL))) { FTemporarySessionFile = IncludeTrailingBackslash(SystemTemporaryDirectory()) + L"winscps.tmp"; DumpResourceToFile(L"WINSCP_SESSION", FTemporarySessionFile); FEmbeddedSessions = true; FTemporaryKeyFile = IncludeTrailingBackslash(SystemTemporaryDirectory()) + L"winscpk.tmp"; if (!DumpResourceToFile(L"WINSCP_KEY", FTemporaryKeyFile)) { FTemporaryKeyFile = L""; } } FStorage = stIniFile; if (!FileExists(ApiPath(IniFileStorageNameForReading))) { if (DetectRegistryStorage(HKEY_CURRENT_USER) || DetectRegistryStorage(HKEY_LOCAL_MACHINE) || // FStorage is now stIniFile, so this tests writing to an INI file. // As we fall back to user profile folder, when application folder // is not writtable, it is actually unlikely that the below test ever fails. !CanWriteToStorage()) { FStorage = stRegistry; } } } TStorage Result = TCustomWinConfiguration::GetStorage(); return Result; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::Saved() { TCustomWinConfiguration::Saved(); FBookmarks->ModifyAll(false); FCustomCommandList->Reset(); FEditorList->Saved(); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::RecryptPasswords(TStrings * RecryptPasswordErrors) { TCustomWinConfiguration::RecryptPasswords(RecryptPasswordErrors); try { TTerminalManager * Manager = TTerminalManager::Instance(false); DebugAssert(Manager != NULL); if (Manager != NULL) { Manager->RecryptPasswords(); } } catch (Exception & E) { UnicodeString Message; if (ExceptionMessage(&E, Message)) { // we do not expect this really to happen, // so we do not bother providing context RecryptPasswordErrors->Add(Message); } } } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::GetUseMasterPassword() { return FUseMasterPassword; } //--------------------------------------------------------------------------- THierarchicalStorage * TWinConfiguration::CreateScpStorage(bool & SessionList) { // Detect embedded session, if not checked yet GetStorage(); THierarchicalStorage * Result; if (SessionList && !FTemporarySessionFile.IsEmpty()) { Result = new TIniFileStorage(FTemporarySessionFile); // This is session-list specific store, so the only instance, // we do not reset the SessionList argument // (compare TConfiguration::CreateScpStorage) } else { Result = TCustomWinConfiguration::CreateScpStorage(SessionList); } return Result; } //--------------------------------------------------------------------------- // duplicated from core\configuration.cpp #define LASTELEM(ELEM) \ ELEM.SubString(ELEM.LastDelimiter(L".>")+1, ELEM.Length() - ELEM.LastDelimiter(L".>")) #define BLOCK(KEY, CANCREATE, BLOCK) \ if (Storage->OpenSubKey(KEY, CANCREATE, true)) try { BLOCK } __finally { Storage->CloseSubKey(); } #define KEY(TYPE, VAR) KEYEX(TYPE, VAR, PropertyToKey(TEXT(#VAR))) #define REGCONFIG(CANCREATE) \ BLOCK(L"Interface", CANCREATE, \ KEYEX(Integer,DoubleClickAction, L"CopyOnDoubleClick"); \ KEY(Bool, CopyOnDoubleClickConfirmation); \ KEY(Bool, DDAllowMove); \ KEY(Bool, DDAllowMoveInit); \ KEYEX(Integer, DDTransferConfirmation, L"DDTransferConfirmation2"); \ KEY(String, DDTemporaryDirectory); \ KEY(Bool, DDWarnLackOfTempSpace); \ KEY(Float, DDWarnLackOfTempSpaceRatio); \ KEY(Bool, DeleteToRecycleBin); \ KEY(Bool, DimmHiddenFiles); \ KEY(Bool, RenameWholeName); \ KEY(Bool, SelectDirectories); \ KEY(String, SelectMask); \ KEY(Bool, ShowHiddenFiles); \ KEY(Integer, FormatSizeBytes); \ KEY(Bool, ShowInaccesibleDirectories); \ KEY(Bool, ConfirmTransferring); \ KEY(Bool, ConfirmDeleting); \ KEY(Bool, ConfirmRecycling); \ KEY(Bool, ConfirmClosingSession); \ KEY(String, AutoStartSession); \ KEY(Bool, UseLocationProfiles); \ KEY(Bool, UseSharedBookmarks); \ KEY(Integer, LocaleSafe); \ KEY(Bool, DDExtEnabled); \ KEY(Integer, DDExtTimeout); \ KEY(Bool, DefaultDirIsHome); \ KEY(Bool, TemporaryDirectoryAppendSession); \ KEY(Bool, TemporaryDirectoryAppendPath); \ KEY(Bool, TemporaryDirectoryDeterministic); \ KEY(Bool, TemporaryDirectoryCleanup); \ KEY(Bool, ConfirmTemporaryDirectoryCleanup); \ KEY(Bool, PreservePanelState); \ KEY(String, LastStoredSession); \ KEY(Bool, AutoSaveWorkspace); \ KEY(Bool, AutoSaveWorkspacePasswords); \ KEY(String, AutoWorkspace); \ KEY(Integer, PathInCaption); \ KEY(Bool, MinimizeToTray); \ KEY(Bool, BalloonNotifications); \ KEY(Integer, NotificationsTimeout); \ KEY(Integer, NotificationsStickTime); \ KEY(Bool, CopyParamAutoSelectNotice); \ KEY(Bool, LockToolbars); \ KEY(Bool, SelectiveToolbarText); \ KEY(Bool, AutoOpenInPutty); \ KEY(Bool, RefreshRemotePanel); \ KEY(DateTime, RefreshRemotePanelInterval); \ KEYEX(String, PanelFont.FontName, L"PanelFontName"); \ KEYEX(Integer,PanelFont.FontSize, L"PanelFontSize"); \ KEYEX(Integer,PanelFont.FontStyle, L"PanelFontStyle"); \ KEYEX(Integer,PanelFont.FontCharset, L"PanelFontCharset"); \ KEY(Bool, FullRowSelect); \ KEY(Bool, OfferedEditorAutoConfig); \ KEY(Integer, LastMonitor); \ KEY(String, VersionHistory); \ KEY(Bool, EnableQueueByDefault); \ KEY(String, OpenedStoredSessionFolders); \ KEY(Bool, AutoImportedFromPuttyOrFilezilla); \ KEY(Integer, GenerateUrlComponents); \ KEY(Integer, GenerateUrlCodeTarget); \ KEY(Integer, GenerateUrlScriptFormat); \ KEY(Integer, GenerateUrlAssemblyLanguage); \ KEY(Bool, ExternalSessionInExistingInstance); \ KEY(Bool, KeepOpenWhenNoSession); \ KEY(Bool, LocalIconsByExt); \ KEY(Integer, BidiModeOverride); \ KEY(Integer, FlipChildrenOverride); \ KEY(Bool, ShowTips); \ KEY(String, TipsSeen); \ KEY(DateTime, TipsShown); \ KEY(Integer, RunsSinceLastTip); \ KEY(Bool, HonorDrivePolicy); \ KEY(Integer, LastMachineInstallations); \ KEYEX(String, FExtensionsDeleted, L"ExtensionsDeleted"); \ KEYEX(String, FExtensionsOrder, L"ExtensionsOrder"); \ ); \ BLOCK(L"Interface\\Editor", CANCREATE, \ KEYEX(String, Editor.Font.FontName, L"FontName2"); \ KEY(Integer, Editor.Font.FontSize); \ KEY(Integer, Editor.Font.FontStyle); \ KEY(Integer, Editor.Font.FontCharset); \ KEY(Integer, Editor.FontColor); \ KEY(Integer, Editor.BackgroundColor); \ KEY(Bool, Editor.WordWrap); \ KEY(String, Editor.FindText); \ KEY(String, Editor.ReplaceText); \ KEY(Bool, Editor.FindMatchCase); \ KEY(Bool, Editor.FindWholeWord); \ KEY(Bool, Editor.FindDown); \ KEY(Integer, Editor.TabSize); \ KEY(Integer, Editor.MaxEditors); \ KEY(Integer, Editor.EarlyClose); \ KEY(Bool, Editor.SDIShellEditor); \ KEY(String, Editor.WindowParams); \ KEY(Integer, Editor.Encoding); \ KEY(Bool, Editor.WarnOnEncodingFallback); \ KEY(Bool, Editor.WarnOrLargeFileSize); \ ); \ BLOCK(L"Interface\\QueueView", CANCREATE, \ KEY(Integer, QueueView.Height); \ KEY(Integer, QueueView.HeightPixelsPerInch); \ KEY(String, QueueView.Layout); \ KEY(Integer, QueueView.Show); \ KEY(Integer, QueueView.LastHideShow); \ KEY(Bool, QueueView.ToolBar); \ KEY(Bool, QueueView.Label); \ ); \ BLOCK(L"Interface\\Updates", CANCREATE, \ KEY(Integer, FUpdates.Period); \ KEY(DateTime, FUpdates.LastCheck); \ KEY(Integer, FUpdates.HaveResults); \ KEY(Integer, FUpdates.ShownResults); \ KEY(Integer, FUpdates.BetaVersions); \ KEY(Bool, FUpdates.ShowOnStartup); \ KEY(String, FUpdates.AuthenticationEmail); \ KEY(Integer, FUpdates.ConnectionType); \ KEY(String, FUpdates.ProxyHost); \ KEY(Integer, FUpdates.ProxyPort); \ KEY(Integer, FUpdates.Results.ForVersion); \ KEY(Integer, FUpdates.Results.Version); \ KEY(String, FUpdates.Results.Message); \ KEY(Integer, FUpdates.Results.Critical); \ KEY(String, FUpdates.Results.Release); \ KEY(Bool, FUpdates.Results.Disabled); \ KEY(String, FUpdates.Results.Url); \ KEY(String, FUpdates.Results.UrlButton); \ KEY(String, FUpdates.Results.NewsUrl); \ KEYEX(Integer,FUpdates.Results.NewsSize.Width, L"NewsWidth"); \ KEYEX(Integer,FUpdates.Results.NewsSize.Height, L"NewsHeight"); \ KEY(String, FUpdates.Results.DownloadUrl); \ KEY(Int64, FUpdates.Results.DownloadSize); \ KEY(String, FUpdates.Results.DownloadSha256); \ KEY(String, FUpdates.Results.AuthenticationError); \ KEY(Bool, FUpdates.Results.OpenGettingStarted); \ KEY(String, FUpdates.Results.DownloadingUrl); \ KEYEX(Integer,FUpdates.Results.TipsSize.Width, L"TipsWidth"); \ KEYEX(Integer,FUpdates.Results.TipsSize.Height, L"TipsHeight"); \ KEY(String, FUpdates.Results.TipsUrl); \ KEY(String, FUpdates.Results.Tips); \ KEY(Integer, FUpdates.Results.TipsIntervalDays); \ KEY(Integer, FUpdates.Results.TipsIntervalRuns); \ KEY(String, FUpdates.DotNetVersion); \ KEY(String, FUpdates.ConsoleVersion); \ ); \ BLOCK(L"Interface\\Explorer", CANCREATE, \ KEY(String, ScpExplorer.ToolbarsLayout); \ KEY(String, ScpExplorer.DirViewParams); \ KEY(String, ScpExplorer.LastLocalTargetDirectory); \ KEY(Bool, ScpExplorer.SessionsTabs); \ KEY(Bool, ScpExplorer.StatusBar); \ KEY(String, ScpExplorer.WindowParams); \ KEY(Integer, ScpExplorer.ViewStyle); \ KEY(Bool, ScpExplorer.ShowFullAddress); \ KEY(Bool, ScpExplorer.DriveView); \ KEY(Integer, ScpExplorer.DriveViewWidth); \ KEY(Integer, ScpExplorer.DriveViewWidthPixelsPerInch); \ ); \ BLOCK(L"Interface\\Commander", CANCREATE, \ KEY(String, ScpCommander.ToolbarsLayout); \ KEY(Integer, ScpCommander.CurrentPanel); \ KEY(Float, ScpCommander.LocalPanelWidth); \ KEY(Bool, ScpCommander.SwappedPanels); \ KEY(Bool, ScpCommander.SessionsTabs); \ KEY(Bool, ScpCommander.StatusBar); \ KEY(String, ScpCommander.WindowParams); \ KEYEX(Integer, ScpCommander.NortonLikeMode, L"ExplorerStyleSelection"); \ KEY(Bool, ScpCommander.PreserveLocalDirectory); \ KEY(Bool, ScpCommander.CompareByTime); \ KEY(Bool, ScpCommander.CompareBySize); \ KEY(Bool, ScpCommander.TreeOnLeft); \ KEY(Bool, ScpCommander.ExplorerKeyboardShortcuts); \ KEY(Bool, ScpCommander.SystemContextMenu); \ ); \ BLOCK(L"Interface\\Commander\\LocalPanel", CANCREATE, \ KEY(String, ScpCommander.LocalPanel.DirViewParams); \ KEY(Bool, ScpCommander.LocalPanel.StatusBar); \ KEY(Bool, ScpCommander.LocalPanel.DriveView); \ KEY(Integer, ScpCommander.LocalPanel.DriveViewHeight); \ KEY(Integer, ScpCommander.LocalPanel.DriveViewHeightPixelsPerInch); \ KEY(Integer, ScpCommander.LocalPanel.DriveViewWidth); \ KEY(Integer, ScpCommander.LocalPanel.DriveViewWidthPixelsPerInch); \ ); \ BLOCK(L"Interface\\Commander\\RemotePanel", CANCREATE, \ KEY(String, ScpCommander.RemotePanel.DirViewParams); \ KEY(Bool, ScpCommander.RemotePanel.StatusBar); \ KEY(Bool, ScpCommander.RemotePanel.DriveView); \ KEY(Integer, ScpCommander.RemotePanel.DriveViewHeight); \ KEY(Integer, ScpCommander.RemotePanel.DriveViewHeightPixelsPerInch); \ KEY(Integer, ScpCommander.RemotePanel.DriveViewWidth); \ KEY(Integer, ScpCommander.RemotePanel.DriveViewWidthPixelsPerInch); \ ); \ BLOCK(L"Logging", CANCREATE, \ KEY(Bool, LogWindowOnStartup); \ KEY(String, LogWindowParams); \ ); \ BLOCK(L"Security", CANCREATE, \ KEYEX(Bool, FUseMasterPassword, L"UseMasterPassword"); \ KEYEX(String,FMasterPasswordVerifier, L"MasterPasswordVerifier"); \ ); //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SaveData(THierarchicalStorage * Storage, bool All) { TCustomWinConfiguration::SaveData(Storage, All); // duplicated from core\configuration.cpp #define KEYEX(TYPE, VAR, NAME) Storage->Write ## TYPE(NAME, VAR) REGCONFIG(true); #undef KEYEX if (Storage->OpenSubKey(L"Bookmarks", true)) { FBookmarks->Save(Storage, All); Storage->CloseSubKey(); } if ((All && !FCustomCommandsDefaults) || FCustomCommandList->Modified) { FCustomCommandList->Save(Storage); } if ((All || FCustomCommandOptionsModified) && Storage->OpenSubKey(L"CustomCommandOptions", true)) { Storage->ClearValues(); Storage->WriteValues(FCustomCommandOptions.get(), true); Storage->CloseSubKey(); FCustomCommandOptionsModified = false; } if ((All || FEditorList->Modified) && Storage->OpenSubKey(L"Interface\\Editor", true, true)) try { FEditorList->Save(Storage); } __finally { Storage->CloseSubKey(); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::LoadFrom(THierarchicalStorage * Storage) { FLegacyEditor = new TEditorPreferences(); try { FLegacyEditor->LegacyDefaults(); TCustomWinConfiguration::LoadFrom(Storage); // This needs to be done even if there's no Configuration key in the storage, // so it cannot be in LoadData int EditorCount = FEditorList->Count; if (EditorCount == 0) { TEditorPreferences * AlternativeEditor = NULL; try { if (FLegacyEditor->Data->Editor == edInternal) { if (!FLegacyEditor->Data->ExternalEditor.IsEmpty()) { AlternativeEditor = new TEditorPreferences(*FLegacyEditor); AlternativeEditor->GetData()->Editor = edExternal; FLegacyEditor->GetData()->ExternalEditor = L""; } } else { if (FLegacyEditor->Data->ExternalEditor.IsEmpty()) { FLegacyEditor->GetData()->Editor = edInternal; } else { AlternativeEditor = new TEditorPreferences(*FLegacyEditor); AlternativeEditor->GetData()->Editor = edInternal; } } } catch(...) { delete AlternativeEditor; throw; } FEditorList->Add(FLegacyEditor); FLegacyEditor = NULL; if (AlternativeEditor != NULL) { FEditorList->Add(AlternativeEditor); } } } __finally { delete FLegacyEditor; FLegacyEditor = NULL; } if (FUpdates.ConnectionType == -1) { FUpdates.ConnectionType = (FUpdates.ProxyHost.IsEmpty() ? ctAuto : ctProxy); } AddVersionToHistory(); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::DoLoadExtensionList( const UnicodeString & Path, const UnicodeString & PathId, TStringList * DeletedExtensions) { TSearchRecChecked SearchRec; int FindAttrs = faReadOnly | faArchive; if (FindFirstUnchecked(IncludeTrailingBackslash(Path) + L"*.*", FindAttrs, SearchRec) == 0) { try { do { UnicodeString Id = TCustomCommandType::GetExtensionId(SearchRec.Name); if (!Id.IsEmpty()) { Id = IncludeTrailingBackslash(PathId) + Id; if (DeletedExtensions->IndexOf(Id) >= 0) { // reconstruct the list, so that we remove the commands that no longer exists AddToList(FExtensionsDeleted, Id, L"|"); } else { std::unique_ptr CustomCommand(new TCustomCommandType()); CustomCommand->Id = Id; try { CustomCommand->LoadExtension(IncludeTrailingBackslash(Path) + SearchRec.Name); FExtensionList->Add(CustomCommand.release()); } catch (...) { // skip invalid extension files } } } } while (FindNextChecked(SearchRec) == 0); } __finally { FindClose(SearchRec); } } } //--------------------------------------------------------------------------- void __fastcall ParseExtensionList(TStrings * Strings, UnicodeString S) { while (!S.IsEmpty()) { UnicodeString Extension = CutToChar(S, L'|', false); Strings->Add(Extension); } } //--------------------------------------------------------------------------- const UnicodeString ExtensionsSubFolder(L"Extensions"); const UnicodeString ExtensionsCommonPathId(L"common"); const UnicodeString ExtensionsCommonExtPathId(L"commonext"); const UnicodeString ExtensionsUserExtPathId(L"userext"); //--------------------------------------------------------------------------- UnicodeString __fastcall TWinConfiguration::GetUserExtensionsPath() { return IncludeTrailingBackslash(GetShellFolderPath(CSIDL_APPDATA)) + L"WinSCP\\" + ExtensionsSubFolder; } //--------------------------------------------------------------------------- TStrings * __fastcall TWinConfiguration::GetExtensionsPaths() { std::unique_ptr Result(new TStringList()); UnicodeString ExeParentPath = ExcludeTrailingBackslash(ExtractFilePath(Application->ExeName)); Result->Values[ExtensionsCommonPathId] = ExeParentPath; UnicodeString CommonExtensions = IncludeTrailingBackslash(ExeParentPath) + ExtensionsSubFolder; Result->Values[ExtensionsCommonExtPathId] = CommonExtensions; Result->Values[ExtensionsUserExtPathId] = GetUserExtensionsPath(); return Result.release(); } //--------------------------------------------------------------------------- UnicodeString __fastcall TWinConfiguration::GetExtensionId(const UnicodeString & ExtensionPath) { UnicodeString Path = ExcludeTrailingBackslash(ExtractFilePath(ExtensionPath)); UnicodeString NameId = TCustomCommandType::GetExtensionId(ExtractFileName(ExtensionPath)); if (!NameId.IsEmpty()) { std::unique_ptr ExtensionsPaths(GetExtensionsPaths()); for (int Index = 0; Index < ExtensionsPaths->Count; Index++) { if (CompareFileName(Path, ExtensionsPaths->ValueFromIndex[Index])) { return IncludeTrailingBackslash(ExtensionsPaths->Names[Index]) + NameId; } } } return L""; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::LoadExtensionList() { FExtensionList->Clear(); std::unique_ptr DeletedExtensions(CreateSortedStringList()); ParseExtensionList(DeletedExtensions.get(), FExtensionsDeleted); // will reconstruct the list in DoLoadExtensionList FExtensionsDeleted = L""; std::unique_ptr ExtensionsPaths(GetExtensionsPaths()); for (int Index = 0; Index < ExtensionsPaths->Count; Index++) { DoLoadExtensionList(ExtensionsPaths->ValueFromIndex[Index], ExtensionsPaths->Names[Index], DeletedExtensions.get()); } std::unique_ptr OrderedExtensions(new TStringList()); ParseExtensionList(OrderedExtensions.get(), FExtensionsOrder); FExtensionList->SortBy(OrderedExtensions.get()); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::LoadData(THierarchicalStorage * Storage) { TCustomWinConfiguration::LoadData(Storage); // duplicated from core\configuration.cpp #define KEYEX(TYPE, VAR, NAME) VAR = Storage->Read ## TYPE(NAME, VAR) #pragma warn -eas REGCONFIG(false); #pragma warn +eas #undef KEYEX // Load after the ExtensionsDeleted and ExtensionsOrder LoadExtensionList(); // to reflect changes to PanelFont UpdateIconFont(); if (Storage->OpenSubKey(L"Bookmarks", false)) { FBookmarks->Load(Storage); Storage->CloseSubKey(); } if (Storage->HasSubKey(L"CustomCommands")) { FCustomCommandList->Load(Storage); FCustomCommandsDefaults = false; } else if (FCustomCommandList->Modified) { // can this (=reloading of configuration) even happen? // if it does, shouldn't we reset default commands? DebugFail(); FCustomCommandList->Clear(); FCustomCommandsDefaults = false; } FCustomCommandList->Reset(); if (Storage->OpenSubKey(L"CustomCommandOptions", false)) { Storage->ReadValues(FCustomCommandOptions.get(), true); Storage->CloseSubKey(); FCustomCommandOptionsModified = false; } if (Storage->OpenSubKey(L"Interface\\Editor", false, true)) try { FEditorList->Clear(); FEditorList->Load(Storage); } __finally { Storage->CloseSubKey(); } // load legacy editor configuration DebugAssert(FLegacyEditor != NULL); if (Storage->OpenSubKey(L"Interface\\Editor", false, true)) { try { FLegacyEditor->Load(Storage, true); } __finally { Storage->CloseSubKey(); } } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::LoadAdmin(THierarchicalStorage * Storage) { TCustomWinConfiguration::LoadAdmin(Storage); FDisableOpenEdit = Storage->ReadBool(L"DisableOpenEdit", FDisableOpenEdit); FDefaultUpdatesPeriod = Storage->ReadInteger(L"DefaultUpdatesPeriod", FDefaultUpdatesPeriod); FMachineInstallations = Storage->ReadInteger(L"Installations", 0); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::CopyData(THierarchicalStorage * Source, THierarchicalStorage * Target) { TCustomWinConfiguration::CopyData(Source, Target); if (Source->OpenSubKey(ConfigurationSubKey, false)) { if (Target->OpenSubKey(ConfigurationSubKey, true)) { Target->WriteString(L"JumpList", Source->ReadString(L"JumpList", L"")); Target->WriteString(L"JumpListWorkspaces", Source->ReadString(L"JumpListWorkspaces", L"")); Target->CloseSubKey(); } Source->CloseSubKey(); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::ClearTemporaryLoginData() { if (!FTemporaryKeyFile.IsEmpty()) { DeleteFile(ApiPath(FTemporaryKeyFile)); FTemporaryKeyFile = L""; } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::AddVersionToHistory() { int CurrentVersion = CompoundVersion; int From = 1; bool CurrentVersionPresent = false; while (!CurrentVersionPresent && (From < FVersionHistory.Length())) { UnicodeString VersionInfo = CopyToChars(FVersionHistory, From, L";", true); UnicodeString VersionStr = CutToChar(VersionInfo, L',', true); int Version; if (TryStrToInt(VersionStr, Version) && (Version == CurrentVersion)) { CurrentVersionPresent = true; } } if (!CurrentVersionPresent) { UnicodeString CurrentVersionInfo = IntToStr(CurrentVersion) + L"," + GetReleaseType(); AddToList(FVersionHistory, CurrentVersionInfo, L';'); } Usage->Set(L"AnyBetaUsed", AnyBetaInVersionHistory); } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::DoIsBeta(const UnicodeString & ReleaseType) { // What about "Development" release type? return AnsiSameText(ReleaseType, L"beta") || AnsiSameText(ReleaseType, L"rc"); } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::GetIsBeta() { return DoIsBeta(GetReleaseType()); } //--------------------------------------------------------------------------- TFont * __fastcall TWinConfiguration::GetSystemIconFont() { // We should do live update from Screen->IconFont when custom panel font is not set return FSystemIconFont.get(); } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::GetAnyBetaInVersionHistory() { int From = 1; bool AnyBeta = false; while (!AnyBeta && (From < VersionHistory.Length())) { UnicodeString VersionInfo = CopyToChars(VersionHistory, From, L";", true); CutToChar(VersionInfo, L',', true); UnicodeString ReleaseType = CutToChar(VersionInfo, ',', true); if (DoIsBeta(ReleaseType)) { AnyBeta = true; } } return AnyBeta; } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::GetDDExtInstalled() { if (FDDExtInstalled < 0) { if (IsWin64()) { // WORKAROUND // We cannot load 64-bit COM class into 32-bit process, // so we fallback to querying registration keys #define CLSID_SIZE 39 wchar_t ClassID[CLSID_SIZE]; StringFromGUID2(CLSID_ShellExtension, ClassID, LENOF(ClassID)); NULL_TERMINATE(ClassID); UnicodeString SubKey = UnicodeString(L"CLSID\\") + ClassID; HKEY HKey; LONG Result = RegOpenKeyEx(HKEY_CLASSES_ROOT, SubKey.c_str(), 0, KEY_READ | KEY_WOW64_64KEY, &HKey); if (Result == ERROR_SUCCESS) { RegCloseKey(HKey); FDDExtInstalled = 1; } else { FDDExtInstalled = 0; } } else { void * DragExtRef; int CreateResult = CoCreateInstance(CLSID_ShellExtension, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_IUnknown, &DragExtRef); bool Result = (CreateResult == S_OK); FDDExtInstalled = (Result ? 1 : 0); if (Result) { reinterpret_cast(DragExtRef)->Release(); CoFreeUnusedLibraries(); } } } return (FDDExtInstalled > 0); } //--------------------------------------------------------------------------- RawByteString __fastcall TWinConfiguration::StronglyRecryptPassword(RawByteString Password, UnicodeString Key) { RawByteString Dummy; RawByteString Result; if (GetExternalEncryptedPassword(Password, Dummy) || !FUseMasterPassword) { // already-strongly encrypted // or no master password set, so we cannot strongly-encrypt it Result = Password; } else { UnicodeString PasswordText = TCustomWinConfiguration::DecryptPassword(Password, Key); if (!PasswordText.IsEmpty()) { // can be not set for instance, when editing=>saving site with no prior password AskForMasterPasswordIfNotSet(); Password = ScramblePassword(PasswordText); AES256EncyptWithMAC(Password, FPlainMasterPasswordEncrypt, Result); Result = SetExternalEncryptedPassword(Result); } } return Result; } //--------------------------------------------------------------------------- UnicodeString __fastcall TWinConfiguration::DecryptPassword(RawByteString Password, UnicodeString Key) { UnicodeString Result; RawByteString Buf; if (GetExternalEncryptedPassword(Password, Buf)) { if (FDontDecryptPasswords == 0) { // As opposite to AskForMasterPasswordIfNotSet, we test here // for decrypt password. This is important while recrypting password, // when clearing master password, when encrypt password is already empty. if (FPlainMasterPasswordDecrypt.IsEmpty()) { AskForMasterPassword(); } if (!AES256DecryptWithMAC(Buf, FPlainMasterPasswordDecrypt, Buf) || !UnscramblePassword(Buf, Result)) { throw Exception(LoadStr(DECRYPT_PASSWORD_ERROR)); } } else { Abort(); } } else { Result = TCustomWinConfiguration::DecryptPassword(Password, Key); } return Result; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetMasterPassword(UnicodeString value) { // just stores the plain-text version of the password // we can get here even if master password is off, // when we encounter stray encrypted password (e.g. manually imported site), // make sure we do not set encrypt password to avoid starting encrypting // new passwords // (this is bit of edge case, not really well tested) if (!FUseMasterPassword) { FPlainMasterPasswordDecrypt = value; } else if (DebugAlwaysTrue(FUseMasterPassword) && DebugAlwaysTrue(ValidateMasterPassword(value))) { FPlainMasterPasswordEncrypt = value; FPlainMasterPasswordDecrypt = value; } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::ChangeMasterPassword( UnicodeString value, TStrings * RecryptPasswordErrors) { RawByteString Verifier; AES256CreateVerifier(value, Verifier); FMasterPasswordVerifier = BytesToHex(Verifier); FPlainMasterPasswordEncrypt = value; FUseMasterPassword = true; try { RecryptPasswords(RecryptPasswordErrors); } __finally { FPlainMasterPasswordDecrypt = value; } } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::ValidateMasterPassword(UnicodeString value) { DebugAssert(UseMasterPassword); DebugAssert(!FMasterPasswordVerifier.IsEmpty()); bool Result = AES256Verify(value, HexToBytes(FMasterPasswordVerifier)); return Result; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::ClearMasterPassword(TStrings * RecryptPasswordErrors) { FMasterPasswordVerifier = L""; FUseMasterPassword = false; Shred(FPlainMasterPasswordEncrypt); try { RecryptPasswords(RecryptPasswordErrors); } __finally { Shred(FPlainMasterPasswordDecrypt); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::AskForMasterPassword() { if (FMasterPasswordSession > 0) { if (FMasterPasswordSessionAsked) { Abort(); } // set before call to OnMasterPasswordPrompt as it may abort FMasterPasswordSessionAsked = true; } if (FOnMasterPasswordPrompt == NULL) { throw Exception(L"Master password handler not set"); } else { FOnMasterPasswordPrompt(); DebugAssert(!FPlainMasterPasswordDecrypt.IsEmpty()); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::AskForMasterPasswordIfNotSet() { if (FPlainMasterPasswordEncrypt.IsEmpty()) { AskForMasterPassword(); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::BeginMasterPasswordSession() { // We do not expect nesting DebugAssert(FMasterPasswordSession == 0); DebugAssert(!FMasterPasswordSessionAsked || (FMasterPasswordSession > 0)); // This should better be thread-specific FMasterPasswordSession++; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::EndMasterPasswordSession() { if (DebugAlwaysTrue(FMasterPasswordSession > 0)) { FMasterPasswordSession--; } FMasterPasswordSessionAsked = false; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetLogWindowOnStartup(bool value) { SET_CONFIG_PROPERTY(LogWindowOnStartup); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetLogWindowParams(UnicodeString value) { SET_CONFIG_PROPERTY(LogWindowParams); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDDAllowMove(bool value) { SET_CONFIG_PROPERTY(DDAllowMove); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDDAllowMoveInit(bool value) { SET_CONFIG_PROPERTY(DDAllowMoveInit); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDDTransferConfirmation(TAutoSwitch value) { SET_CONFIG_PROPERTY(DDTransferConfirmation); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDDTemporaryDirectory(UnicodeString value) { SET_CONFIG_PROPERTY(DDTemporaryDirectory); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDDExtEnabled(bool value) { SET_CONFIG_PROPERTY(DDExtEnabled); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDDExtTimeout(int value) { SET_CONFIG_PROPERTY(DDExtTimeout); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDDWarnLackOfTempSpace(bool value) { SET_CONFIG_PROPERTY(DDWarnLackOfTempSpace); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDDWarnLackOfTempSpaceRatio(double value) { SET_CONFIG_PROPERTY(DDWarnLackOfTempSpaceRatio); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetScpExplorer(TScpExplorerConfiguration value) { SET_CONFIG_PROPERTY(ScpExplorer); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetScpCommander(TScpCommanderConfiguration value) { SET_CONFIG_PROPERTY(ScpCommander); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetEditor(TEditorConfiguration value) { SET_CONFIG_PROPERTY(Editor); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetQueueView(TQueueViewConfiguration value) { SET_CONFIG_PROPERTY(QueueView); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetEnableQueueByDefault(bool value) { SET_CONFIG_PROPERTY(EnableQueueByDefault); } //--------------------------------------------------------------------------- TUpdatesConfiguration __fastcall TWinConfiguration::GetUpdates() { TUpdatesConfiguration Result; { TGuard Guard(FCriticalSection); Result = FUpdates; } return Result; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetUpdates(TUpdatesConfiguration value) { TGuard Guard(FCriticalSection); // do not use SET_CONFIG_PROPERTY to avoid OnChange handler call (not synchronized) FUpdates = value; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetVersionHistory(UnicodeString value) { SET_CONFIG_PROPERTY(VersionHistory); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDeleteToRecycleBin(bool value) { SET_CONFIG_PROPERTY(DeleteToRecycleBin); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetSelectDirectories(bool value) { SET_CONFIG_PROPERTY(SelectDirectories); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetShowHiddenFiles(bool value) { SET_CONFIG_PROPERTY(ShowHiddenFiles); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetFormatSizeBytes(TFormatBytesStyle value) { SET_CONFIG_PROPERTY(FormatSizeBytes); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetShowInaccesibleDirectories(bool value) { SET_CONFIG_PROPERTY(ShowInaccesibleDirectories); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetConfirmTransferring(bool value) { SET_CONFIG_PROPERTY(ConfirmTransferring); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetConfirmDeleting(bool value) { SET_CONFIG_PROPERTY(ConfirmDeleting); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetConfirmRecycling(bool value) { SET_CONFIG_PROPERTY(ConfirmRecycling); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetUseLocationProfiles(bool value) { SET_CONFIG_PROPERTY(UseLocationProfiles); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetUseSharedBookmarks(bool value) { SET_CONFIG_PROPERTY(UseSharedBookmarks); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetConfirmClosingSession(bool value) { SET_CONFIG_PROPERTY(ConfirmClosingSession); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDoubleClickAction(TDoubleClickAction value) { SET_CONFIG_PROPERTY(DoubleClickAction); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetCopyOnDoubleClickConfirmation(bool value) { SET_CONFIG_PROPERTY(CopyOnDoubleClickConfirmation); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDimmHiddenFiles(bool value) { SET_CONFIG_PROPERTY(DimmHiddenFiles); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetRenameWholeName(bool value) { SET_CONFIG_PROPERTY(RenameWholeName); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetAutoStartSession(UnicodeString value) { SET_CONFIG_PROPERTY(AutoStartSession); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetExpertMode(bool value) { SET_CONFIG_PROPERTY(ExpertMode); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetDefaultDirIsHome(bool value) { SET_CONFIG_PROPERTY(DefaultDirIsHome); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetTemporaryDirectoryAppendSession(bool value) { SET_CONFIG_PROPERTY(TemporaryDirectoryAppendSession); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetTemporaryDirectoryAppendPath(bool value) { SET_CONFIG_PROPERTY(TemporaryDirectoryAppendPath); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetTemporaryDirectoryDeterministic(bool value) { SET_CONFIG_PROPERTY(TemporaryDirectoryDeterministic); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetTemporaryDirectoryCleanup(bool value) { SET_CONFIG_PROPERTY(TemporaryDirectoryCleanup); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetConfirmTemporaryDirectoryCleanup(bool value) { SET_CONFIG_PROPERTY(ConfirmTemporaryDirectoryCleanup); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetPreservePanelState(bool value) { SET_CONFIG_PROPERTY(PreservePanelState); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetLastStoredSession(UnicodeString value) { SET_CONFIG_PROPERTY(LastStoredSession); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetAutoSaveWorkspace(bool value) { SET_CONFIG_PROPERTY(AutoSaveWorkspace); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetAutoSaveWorkspacePasswords(bool value) { SET_CONFIG_PROPERTY(AutoSaveWorkspacePasswords); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetAutoWorkspace(UnicodeString value) { SET_CONFIG_PROPERTY(AutoWorkspace); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetPathInCaption(TPathInCaption value) { SET_CONFIG_PROPERTY(PathInCaption); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetMinimizeToTray(bool value) { SET_CONFIG_PROPERTY(MinimizeToTray); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::MinimizeToTrayOnce() { FMinimizeToTrayOnce = true; } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::GetMinimizeToTray() { bool Result = FMinimizeToTrayOnce || FMinimizeToTray; FMinimizeToTrayOnce = false; return Result; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetBalloonNotifications(bool value) { SET_CONFIG_PROPERTY(BalloonNotifications); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetNotificationsTimeout(unsigned int value) { SET_CONFIG_PROPERTY(NotificationsTimeout); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetNotificationsStickTime(unsigned int value) { SET_CONFIG_PROPERTY(NotificationsStickTime); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetCopyParamAutoSelectNotice(bool value) { SET_CONFIG_PROPERTY(CopyParamAutoSelectNotice); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetLockToolbars(bool value) { SET_CONFIG_PROPERTY(LockToolbars); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetSelectiveToolbarText(bool value) { SET_CONFIG_PROPERTY(SelectiveToolbarText); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetAutoOpenInPutty(bool value) { SET_CONFIG_PROPERTY(AutoOpenInPutty); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetRefreshRemotePanel(bool value) { SET_CONFIG_PROPERTY(RefreshRemotePanel); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetRefreshRemotePanelInterval(TDateTime value) { SET_CONFIG_PROPERTY(RefreshRemotePanelInterval); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::UpdateSystemIconFont() { FSystemIconFont->Assign(Screen->IconFont); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::UpdateIconFont() { if (!PanelFont.FontName.IsEmpty()) { UpdateSystemIconFont(); std::unique_ptr IconFont(new TFont()); RestoreFont(PanelFont, IconFont.get()); Screen->IconFont->Assign(IconFont.get()); // When using non-standard icon font, prevent resetting it back // to standard one on WM_WININICHANGE. Unfortunatelly this prevents // updating all other fonts. Application->UpdateMetricSettings = false; } else { if (DebugAlwaysTrue(FSystemIconFont.get() != NULL) && !SameFont(Screen->IconFont, FSystemIconFont.get())) { Screen->IconFont->Assign(FSystemIconFont.get()); } Application->UpdateMetricSettings = true; } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetPanelFont(const TFontConfiguration & value) { SET_CONFIG_PROPERTY_EX(PanelFont, UpdateIconFont()); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetFullRowSelect(bool value) { SET_CONFIG_PROPERTY(FullRowSelect); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetOfferedEditorAutoConfig(bool value) { SET_CONFIG_PROPERTY(OfferedEditorAutoConfig); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetOpenedStoredSessionFolders(UnicodeString value) { SET_CONFIG_PROPERTY(OpenedStoredSessionFolders); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetAutoImportedFromPuttyOrFilezilla(bool value) { SET_CONFIG_PROPERTY(AutoImportedFromPuttyOrFilezilla); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetGenerateUrlComponents(int value) { SET_CONFIG_PROPERTY(GenerateUrlComponents); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetGenerateUrlCodeTarget(TGenerateUrlCodeTarget value) { SET_CONFIG_PROPERTY(GenerateUrlCodeTarget); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetGenerateUrlScriptFormat(TScriptFormat value) { SET_CONFIG_PROPERTY(GenerateUrlScriptFormat); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetGenerateUrlAssemblyLanguage(TAssemblyLanguage value) { SET_CONFIG_PROPERTY(GenerateUrlAssemblyLanguage); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetExternalSessionInExistingInstance(bool value) { SET_CONFIG_PROPERTY(ExternalSessionInExistingInstance); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetKeepOpenWhenNoSession(bool value) { SET_CONFIG_PROPERTY(KeepOpenWhenNoSession); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetLocalIconsByExt(bool value) { SET_CONFIG_PROPERTY(LocalIconsByExt); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetBidiModeOverride(TLocaleFlagOverride value) { SET_CONFIG_PROPERTY(BidiModeOverride); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetFlipChildrenOverride(TLocaleFlagOverride value) { SET_CONFIG_PROPERTY(FlipChildrenOverride); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetShowTips(bool value) { SET_CONFIG_PROPERTY(ShowTips); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetTipsSeen(UnicodeString value) { SET_CONFIG_PROPERTY(TipsSeen); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetTipsShown(TDateTime value) { SET_CONFIG_PROPERTY(TipsShown); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetRunsSinceLastTip(int value) { SET_CONFIG_PROPERTY(RunsSinceLastTip); } //--------------------------------------------------------------------------- bool __fastcall TWinConfiguration::GetHonorDrivePolicy() { return DriveInfo->HonorDrivePolicy; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetHonorDrivePolicy(bool value) { if (HonorDrivePolicy != value) { DriveInfo->HonorDrivePolicy = value; Changed(); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetCustomCommandList(TCustomCommandList * value) { DebugAssert(FCustomCommandList); if (!FCustomCommandList->Equals(value)) { FCustomCommandList->Assign(value); FCustomCommandsDefaults = false; } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetExtensionList(TCustomCommandList * value) { if (!ExtensionList->Equals(value)) { std::unique_ptr DeletedExtensions(CreateSortedStringList()); ParseExtensionList(DeletedExtensions.get(), FExtensionsDeleted); for (int Index = 0; Index < ExtensionList->Count; Index++) { const TCustomCommandType * CustomCommand = ExtensionList->Commands[Index]; int Index = value->FindIndexByFileName(CustomCommand->FileName); if (Index < 0) { if (FileExists(CustomCommand->FileName) && !DeleteFile(CustomCommand->FileName)) { DeletedExtensions->Add(CustomCommand->Id); } } } FExtensionsOrder = L""; for (int Index = 0; Index < value->Count; Index++) { const TCustomCommandType * CustomCommand = value->Commands[Index]; AddToList(FExtensionsOrder, CustomCommand->Id, L"|"); int DeletedIndex = DeletedExtensions->IndexOf(CustomCommand->Id); if (DeletedIndex >= 0) { DeletedExtensions->Delete(DeletedIndex); } } FExtensionList->Assign(value); FExtensionsDeleted = L""; for (int Index = 0; Index < DeletedExtensions->Count; Index++) { AddToList(FExtensionsDeleted, DeletedExtensions->Strings[Index], L"|"); } Changed(); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::CustomCommandShortCuts(TShortCuts & ShortCuts) const { CustomCommandList->ShortCuts(ShortCuts); ExtensionList->ShortCuts(ShortCuts); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetBookmarks(UnicodeString Key, TBookmarkList * value) { FBookmarks->Bookmarks[Key] = value; Changed(); } //--------------------------------------------------------------------------- TBookmarkList * __fastcall TWinConfiguration::GetBookmarks(UnicodeString Key) { return FBookmarks->Bookmarks[Key]; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetSharedBookmarks(TBookmarkList * value) { FBookmarks->SharedBookmarks = value; Changed(); } //--------------------------------------------------------------------------- TBookmarkList * __fastcall TWinConfiguration::GetSharedBookmarks() { return FBookmarks->SharedBookmarks; } //--------------------------------------------------------------------------- UnicodeString __fastcall TWinConfiguration::GetDefaultKeyFile() { return (!FDefaultKeyFile.IsEmpty() ? FDefaultKeyFile : FTemporaryKeyFile); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetLastMonitor(int value) { ::SetLastMonitor(value); } //--------------------------------------------------------------------------- int __fastcall TWinConfiguration::GetLastMonitor() { return ::GetLastMonitor(); } //--------------------------------------------------------------------------- UnicodeString __fastcall TWinConfiguration::ExpandedTemporaryDirectory() { UnicodeString Result = ExpandFileName(ExpandEnvironmentVariables(DDTemporaryDirectory)); if (Result.IsEmpty()) { Result = SystemTemporaryDirectory(); } return Result; } //--------------------------------------------------------------------------- UnicodeString __fastcall TWinConfiguration::TemporaryDir(bool Mask) { return UniqTempDir(ExpandedTemporaryDirectory(), L"scp", Mask); } //--------------------------------------------------------------------------- TStrings * __fastcall TWinConfiguration::FindTemporaryFolders() { TStrings * Result = new TStringList(); try { TSearchRecChecked SRec; UnicodeString Mask = TemporaryDir(true); UnicodeString Directory = ExtractFilePath(Mask); if (FindFirstUnchecked(Mask, faDirectory, SRec) == 0) { do { if (FLAGSET(SRec.Attr, faDirectory)) { Result->Add(Directory + SRec.Name); } } while (FindNextChecked(SRec) == 0); } if (Result->Count == 0) { delete Result; Result = NULL; } } catch(...) { delete Result; throw; } return Result; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::CleanupTemporaryFolders(TStrings * Folders) { UnicodeString ErrorList; TStrings * F; if (Folders == NULL) { F = FindTemporaryFolders(); } else { F = Folders; } if (F != NULL) { try { if (DebugAlwaysTrue(F->Count > 0)) { Usage->Inc(L"TemporaryDirectoryCleanups"); } for (int i = 0; i < F->Count; i++) { if (!DeleteDirectory(F->Strings[i])) { if (!ErrorList.IsEmpty()) { ErrorList += L"\n"; } ErrorList += F->Strings[i]; } } } __finally { if (Folders == NULL) { delete F; } } if (!ErrorList.IsEmpty()) { throw ExtException(LoadStr(CLEANUP_TEMP_ERROR), ErrorList); } } } //--------------------------------------------------------------------------- LCID __fastcall TWinConfiguration::GetLocale() { if (!FLocale) { UnicodeString ResourceModule = GetResourceModuleName(Application->Name, ModuleFileName().c_str()); if (!ResourceModule.IsEmpty()) { UnicodeString ResourceExt = ExtractFileExt(ResourceModule).UpperCase(); ResourceExt.Delete(1, 1); TLanguages * Langs = Languages(); int Index, Count; Count = Langs->Count; Index = 0; while ((Index < Count) && (FLocale == 0)) { if (Langs->Ext[Index] == ResourceExt) { SetInitialLocale(Langs->LocaleID[Index]); } else if (Langs->Ext[Index].SubString(1, 2) == ResourceExt) { SetInitialLocale( MAKELANGID(PRIMARYLANGID(Langs->LocaleID[Index]), SUBLANG_DEFAULT)); } if (FLocale != 0) { FLocaleModuleName = ResourceModule; } Index++; } } } return TCustomWinConfiguration::GetLocale(); } //--------------------------------------------------------------------------- HINSTANCE __fastcall TWinConfiguration::LoadNewResourceModule(LCID ALocale, UnicodeString & FileName) { HINSTANCE Instance = TCustomWinConfiguration::LoadNewResourceModule(ALocale, FileName); if (Instance != NULL) { try { CheckTranslationVersion(FileName, false); } catch(...) { FreeResourceModule(Instance); throw; } } return Instance; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::CheckTranslationVersion(const UnicodeString FileName, bool InternalLocaleOnError) { UnicodeString TranslationProductVersion = GetFileProductVersion(FileName); UnicodeString TranslationProductName = GetFileProductName(FileName); if ((ProductName != TranslationProductName) || (ProductVersion != TranslationProductVersion)) { if (InternalLocaleOnError) { LocaleSafe = InternalLocale(); } if (TranslationProductName.IsEmpty() || TranslationProductVersion.IsEmpty()) { throw Exception(FMTLOAD(UNKNOWN_TRANSLATION, (FileName))); } else { throw Exception(FMTLOAD(INCOMPATIBLE_TRANSLATION, (FileName, TranslationProductName, TranslationProductVersion))); } } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::CheckDefaultTranslation() { if (!FInvalidDefaultTranslationMessage.IsEmpty()) { MoreMessageDialog(FMTLOAD(INVALID_DEFAULT_TRANSLATION, (FInvalidDefaultTranslationMessage)), NULL, qtWarning, qaOK, HELP_NONE); } } //--------------------------------------------------------------------------- const TEditorPreferences * __fastcall TWinConfiguration::DefaultEditorForFile( const UnicodeString FileName, bool Local, const TFileMasks::TParams & MaskParams) { return FEditorList->Find(FileName, Local, MaskParams); } //--------------------------------------------------------------------------- const TEditorList * __fastcall TWinConfiguration::GetEditorList() { return FEditorList; } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetEditorList(const TEditorList * value) { if (!(*FEditorList == *value)) { *FEditorList = *value; Changed(); } } //--------------------------------------------------------------------------- TStrings * __fastcall TWinConfiguration::GetCustomCommandOptions() { return FCustomCommandOptions.get(); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SetCustomCommandOptions(TStrings * value) { if (!FCustomCommandOptions->Equals(value)) { FCustomCommandOptions->Assign(value); FCustomCommandOptionsModified = true; } } //--------------------------------------------------------------------------- TStringList * __fastcall TWinConfiguration::LoadJumpList( THierarchicalStorage * Storage, UnicodeString Name) { UnicodeString JumpList = Storage->ReadString(Name, L""); std::unique_ptr List(new TStringList()); List->CaseSensitive = false; List->CommaText = JumpList; return List.release(); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::SaveJumpList( THierarchicalStorage * Storage, UnicodeString Name, TStringList * List) { UnicodeString JumpList = Storage->ReadString(Name, L""); UnicodeString NewJumpList = List->CommaText; if (NewJumpList != JumpList) { Storage->WriteString(Name, NewJumpList); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::TrimJumpList(TStringList * List) { while (List->Count > 30) { List->Delete(List->Count - 1); } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::UpdateEntryInJumpList( bool Session, const UnicodeString & Name, bool Add) { THierarchicalStorage * Storage = CreateConfigStorage(); try { FDontDecryptPasswords++; Storage->AccessMode = smReadWrite; if (Storage->OpenSubKey(ConfigurationSubKey, true)) { std::unique_ptr ListSessions(LoadJumpList(Storage, L"JumpList")); std::unique_ptr ListWorkspaces(LoadJumpList(Storage, L"JumpListWorkspaces")); if (!Name.IsEmpty()) { TStringList * List = (Session ? ListSessions : ListWorkspaces).get(); int Index = List->IndexOf(Name); if (Index >= 0) { List->Delete(Index); } if (Add) { List->Insert(0, Name); } } TrimJumpList(ListSessions.get()); TrimJumpList(ListWorkspaces.get()); ::UpdateJumpList(ListSessions.get(), ListWorkspaces.get()); SaveJumpList(Storage, L"JumpList", ListSessions.get()); SaveJumpList(Storage, L"JumpListWorkspaces", ListWorkspaces.get()); } } __finally { FDontDecryptPasswords--; delete Storage; } } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::AddSessionToJumpList(UnicodeString SessionName) { UpdateEntryInJumpList(true, SessionName, true); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::DeleteSessionFromJumpList(UnicodeString SessionName) { UpdateEntryInJumpList(true, SessionName, false); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::AddWorkspaceToJumpList(UnicodeString Workspace) { UpdateEntryInJumpList(false, Workspace, true); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::DeleteWorkspaceFromJumpList(UnicodeString Workspace) { UpdateEntryInJumpList(false, Workspace, false); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::UpdateJumpList() { AddSessionToJumpList(L""); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::UpdateStaticUsage() { // This is here because of TStoredSessionList::UpdateStaticUsage() call // from TConfiguration::UpdateStaticUsage() TAutoNestingCounter DontDecryptPasswordsAutoCounter(FDontDecryptPasswords); TCustomWinConfiguration::UpdateStaticUsage(); Usage->Set(L"Beta", IsBeta); Usage->Set(L"Interface", Interface); Usage->Set(L"CustomCommandsCount", (FCustomCommandsDefaults ? 0 : FCustomCommandList->Count)); Usage->Set(L"UsingLocationProfiles", UseLocationProfiles); Usage->Set(L"UsingMasterPassword", UseMasterPassword); Usage->Set(L"UsingAutoSaveWorkspace", AutoSaveWorkspace); Usage->Set(L"TreeVisible", (Interface == ifExplorer) ? ScpExplorer.DriveView : (ScpCommander.LocalPanel.DriveView || ScpCommander.RemotePanel.DriveView)); Usage->Set(L"MinimizeToTray", MinimizeToTray); Usage->Set(L"ShowingTips", ShowTips); TipsUpdateStaticUsage(); Usage->Set(L"CommanderNortonLikeMode", int(ScpCommander.NortonLikeMode)); Usage->Set(L"CommanderExplorerKeyboardShortcuts", ScpCommander.ExplorerKeyboardShortcuts); Usage->Set(L"ExplorerViewStyle", ScpExplorer.ViewStyle); Usage->Set(L"LastMonitor", LastMonitor); UnicodeString ExternalEditors; for (int Index = 0; Index < EditorList->Count; Index++) { const TEditorPreferences * Editor = EditorList->Editors[Index]; if (Editor->Data->Editor == edExternal) { AddToList(ExternalEditors, Editor->ExtractExternalEditorName(), ","); } } Usage->Set(L"ExternalEditors", ExternalEditors); if (LastMachineInstallations < FMachineInstallations) { Usage->Inc(L"InstallationsMachine", (FMachineInstallations - LastMachineInstallations)); LastMachineInstallations = FMachineInstallations; } int ExtensionsPortable = 0; int ExtensionsInstalled = 0; int ExtensionsUser = 0; for (int Index = 0; Index < FExtensionList->Count; Index++) { const TCustomCommandType * CustomCommand = FExtensionList->Commands[Index]; UnicodeString PathId = ExcludeTrailingBackslash(ExtractFilePath(CustomCommand->Id)); if (SameText(PathId, ExtensionsCommonPathId)) { ExtensionsPortable++; } else if (SameText(PathId, ExtensionsCommonExtPathId)) { ExtensionsInstalled++; } else if (SameText(PathId, ExtensionsUserExtPathId)) { ExtensionsUser++; } } Usage->Set(L"ExtensionsPortableCount", (ExtensionsPortable)); Usage->Set(L"ExtensionsInstalledCount", (ExtensionsInstalled)); Usage->Set(L"ExtensionsUserCount", (ExtensionsUser)); std::unique_ptr DeletedExtensions(CreateSortedStringList()); ParseExtensionList(DeletedExtensions.get(), FExtensionsDeleted); Usage->Set(L"ExtensionsDeleted", (DeletedExtensions->Count)); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::RestoreFont( const TFontConfiguration & Configuration, TFont * Font) { Font->Name = Configuration.FontName; Font->Size = Configuration.FontSize; Font->Charset = Configuration.FontCharset; Font->Style = IntToFontStyles(Configuration.FontStyle); } //--------------------------------------------------------------------------- void __fastcall TWinConfiguration::StoreFont( TFont * Font, TFontConfiguration & Configuration) { Configuration.FontName = Font->Name; Configuration.FontSize = Font->Size; Configuration.FontCharset = Font->Charset; Configuration.FontStyle = FontStylesToInt(Font->Style); } //--------------------------------------------------------------------------- //--------------------------------------------------------------------------- __fastcall TCustomCommandType::TCustomCommandType() : FParams(0), FShortCut(0) { } //--------------------------------------------------------------------------- __fastcall TCustomCommandType::TCustomCommandType(const TCustomCommandType & Other) : FName(Other.FName), FCommand(Other.FCommand), FParams(Other.FParams), FShortCut(Other.FShortCut), FId(Other.FId), FFileName(Other.FFileName), FDescription(Other.FDescription), FHomePage(Other.FHomePage), FOptionsPage(Other.FOptionsPage), FOptions(Other.FOptions) { } //--------------------------------------------------------------------------- TCustomCommandType & TCustomCommandType::operator=(const TCustomCommandType & Other) { FName = Other.FName; FCommand = Other.FCommand; FParams = Other.FParams; FShortCut = Other.FShortCut; FId = Other.FId; FFileName = Other.FFileName; FDescription = Other.FDescription; FHomePage = Other.FHomePage; FOptionsPage = Other.FOptionsPage; FOptions = Other.FOptions; return *this; } //--------------------------------------------------------------------------- bool __fastcall TCustomCommandType::Equals(const TCustomCommandType * Other) const { return (FName == Other->FName) && (FCommand == Other->FCommand) && (FParams == Other->FParams) && (FShortCut == Other->FShortCut) && (FId == Other->FId) && (FFileName == Other->FFileName) && (FDescription == Other->FDescription) && (FHomePage == Other->FHomePage) && (FOptionsPage == Other->FOptionsPage) && (FOptions == Other->FOptions); } //--------------------------------------------------------------------------- const UnicodeString ExtensionNameDirective(L"name"); const UnicodeString ExtensionCommandDirective(L"command"); const wchar_t ExtensionMark = L'@'; const UnicodeString WinSCPExtensionExt(L".WinSCPextension"); //--------------------------------------------------------------------------- UnicodeString __fastcall TCustomCommandType::GetExtensionId(const UnicodeString & Name) { UnicodeString Result; int P = Pos(UpperCase(WinSCPExtensionExt), UpperCase(Name)); // Ends with Ext or there's another extension after it if ((P > 1) && ((Name.Length() == (P + WinSCPExtensionExt.Length() - 1)) || (Name[P + WinSCPExtensionExt.Length()] == L'.'))) { Result = Name.SubString(1, P - 1); } return Result; } //--------------------------------------------------------------------------- void __fastcall TCustomCommandType::LoadExtension(const UnicodeString & Path) { std::unique_ptr Lines(new TStringList()); FileName = Path; LoadScriptFromFile(Path, Lines.get()); LoadExtension(Lines.get()); Command = ReplaceStr(Command, L"%EXTENSION_PATH%", Path); } //--------------------------------------------------------------------------- void __fastcall TCustomCommandType::LoadExtension(TStrings * Lines) { Params = ccLocal; bool AnythingFound = false; std::set OptionIds; for (int Index = 0; Index < Lines->Count; Index++) { UnicodeString Line = Lines->Strings[Index].Trim(); if (!Line.IsEmpty()) { bool IsComment = false; if (StartsText(ExtensionMark, Line)) { IsComment = true; } else if (StartsText(L"rem ", Line)) { IsComment = true; Line.Delete(1, 4); } else if (StartsText(L"#", Line) || StartsText(L";", Line) || StartsText(L"'", Line)) { IsComment = true; Line.Delete(1, 1); } else if (StartsText(L"//", Line)) { IsComment = true; Line.Delete(1, 2); } if (!IsComment) { break; } else { Line = Line.Trim(); int P; if (!Line.IsEmpty() && (Line[1] == ExtensionMark) && ((P = Pos(L" ", Line)) >= 2)) { UnicodeString Key = Line.SubString(2, P - 2).LowerCase(); UnicodeString Directive = UnicodeString(ExtensionMark) + Key; UnicodeString Value = Line.SubString(P + 1, Line.Length() - P).Trim(); bool KnownKey = true; if (Key == ExtensionNameDirective) { Name = Value; } else if (Key == ExtensionCommandDirective) { Command = Value; } else if (Key == L"require") { UnicodeString DependencyVersion = Value; UnicodeString Dependency = CutToChar(Value, L' ', true).LowerCase(); Value = Value.Trim(); bool Failed; if (Dependency == L"winscp") { int Version = StrToCompoundVersion(Value); Failed = (Version > WinConfiguration->CompoundVersion); } else if (Dependency == L".net") { Failed = (CompareVersion(Value, GetNetVersionStr()) > 0); } else if (Dependency == L"powershell") { Failed = (CompareVersion(Value, GetPowerShellVersionStr()) > 0); } else if (Dependency == L"windows") { Failed = (CompareVersion(Value, WindowsVersion()) > 0); } else { Failed = true; } if (Failed) { throw Exception(MainInstructions(FMTLOAD(EXTENSION_DEPENDENCY_ERROR, (DependencyVersion)))); } } else if (Key == L"side") { if (SameText(Value, L"Local")) { Params |= ccLocal; } else if (SameText(Value, L"Remote")) { Params &= ~ccLocal; } else { throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_ERROR, (Value, Directive)))); } } else if (Key == L"flag") { if (SameText(Value, L"ApplyToDirectories")) { Params |= ccApplyToDirectories; } else if (SameText(Value, L"Recursive")) { Params |= ccRecursive; } else if (SameText(Value, L"ShowResults")) { Params |= ccShowResults; } else if (SameText(Value, L"CopyResults")) { Params |= ccCopyResults; } else if (SameText(Value, L"RemoteFiles")) { Params |= ccRemoteFiles; } else { throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_ERROR, (Value, Directive)))); } } else if (Key == L"shortcut") { TShortCut AShortCut = TextToShortCut(Value); if (IsCustomShortCut(AShortCut)) { ShortCut = AShortCut; } else { throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_ERROR, (Value, Directive)))); } } else if (Key == L"option") { TOption Option; if (!ParseOption(Value, Option) || (Option.IsControl && (OptionIds.find(Option.Id.LowerCase()) != OptionIds.end()))) { throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_ERROR, (Value, Directive)))); } else { FOptions.push_back(Option); if (!Option.IsControl) { OptionIds.insert(Option.Id.LowerCase()); } } } else if (Key == L"description") { Description = Value; } else if (Key == L"author") { // noop } else if (Key == L"version") { // noop } else if (Key == L"homepage") { HomePage = Value; } else if (Key == L"optionspage") { OptionsPage = Value; } else if (Key == L"source") { // noop } else { KnownKey = false; } if (KnownKey) { AnythingFound = true; } } } } } if (!AnythingFound) { throw Exception(MainInstructions(LoadStr(EXTENSION_NOT_FOUND))); } if (Name.IsEmpty()) { throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_MISSING, (UnicodeString(ExtensionMark) + ExtensionNameDirective)))); } if (Command.IsEmpty()) { throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_MISSING, (UnicodeString(ExtensionMark) + ExtensionCommandDirective)))); } } //--------------------------------------------------------------------------- bool __fastcall TCustomCommandType::ParseOption(const UnicodeString & Value, TOption & Option) { UnicodeString Buf = Value; UnicodeString KindName; bool Result = CutToken(Buf, Option.Id) && CutToken(Buf, KindName); if (Result) { KindName = KindName.LowerCase(); if (KindName == L"label") { Option.Kind = okLabel; Result = !Option.IsControl; } else if (KindName == L"link") { Option.Kind = okLink; Result = !Option.IsControl; } else if (KindName == L"separator") { Option.Kind = okSeparator; Result = !Option.IsControl; } else if (KindName == L"textbox") { Option.Kind = okTextBox; Result = Option.IsControl; } else if (KindName == L"file") { Option.Kind = okFile; Result = Option.IsControl; } else if (KindName == L"dropdownlist") { Option.Kind = okDropDownList; Result = Option.IsControl; } else if (KindName == L"combobox") { Option.Kind = okComboBox; Result = Option.IsControl; } else if (KindName == L"checkbox") { Option.Kind = okCheckBox; Result = Option.IsControl; } else { Option.Kind = okUnknown; } if ((Option.Kind != okUnknown) && (Option.Kind != okSeparator)) { Result = CutToken(Buf, Option.Caption); if (Result && Option.IsControl) { if (CutToken(Buf, Option.Default)) { UnicodeString Param; while (CutToken(Buf, Param)) { Option.Params.push_back(Param); } } } } } return Result; } //--------------------------------------------------------------------------- int __fastcall TCustomCommandType::GetOptionsCount() const { return FOptions.size(); } //--------------------------------------------------------------------------- const TCustomCommandType::TOption & __fastcall TCustomCommandType::GetOption(int Index) const { return FOptions[Index]; } //--------------------------------------------------------------------------- UnicodeString __fastcall TCustomCommandType::GetOptionKey(const TCustomCommandType::TOption & Option) const { return Id + L"\\" + Option.Id; } //--------------------------------------------------------------------------- UnicodeString __fastcall TCustomCommandType::GetCommandWithExpandedOptions(TStrings * CustomCommandOptions) const { UnicodeString Result = Command; for (int Index = 0; Index < OptionsCount; Index++) { const TCustomCommandType::TOption & Option = GetOption(Index); if (Option.IsControl) { UnicodeString OptionKey = GetOptionKey(Option); UnicodeString OptionValue; if (CustomCommandOptions->IndexOfName(OptionKey) >= 0) { OptionValue = CustomCommandOptions->Values[OptionKey]; } else { OptionValue = Option.Default; } UnicodeString OptionCommand = GetOptionCommand(Option, OptionValue); Result = ReplaceText(Result, FORMAT(L"%%%s%%", (Option.Id)), OptionCommand); } } return Result; } //--------------------------------------------------------------------------- UnicodeString __fastcall TCustomCommandType::GetOptionCommand(const TOption & Option, const UnicodeString & Value) const { UnicodeString Result = Value; switch (Option.Kind) { case okUnknown: case okTextBox: case okDropDownList: case okComboBox: case okCheckBox: // noop break; case okFile: Result = ExpandEnvironmentVariables(Result); break; case okLabel: case okLink: case okSeparator: default: DebugFail(); } return Result; } //--------------------------------------------------------------------------- //--------------------------------------------------------------------------- bool __fastcall TCustomCommandType::TOption::GetIsControl() const { return (Id != L"-"); } //--------------------------------------------------------------------------- bool TCustomCommandType::TOption::operator==(const TCustomCommandType::TOption & Other) const { return (Id == Other.Id) && (Kind == Other.Kind) && (Caption == Other.Caption) && (Default == Other.Default) && (Params == Other.Params); } //--------------------------------------------------------------------------- //--------------------------------------------------------------------------- __fastcall TCustomCommandList::TCustomCommandList() { FCommands = new TList(); FModified = false; } //--------------------------------------------------------------------------- __fastcall TCustomCommandList::~TCustomCommandList() { Clear(); delete FCommands; } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Reset() { FModified = false; } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Modify() { FModified = true; } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Load(THierarchicalStorage * Storage) { Clear(); if (Storage->OpenSubKey(L"CustomCommands", false)) { TStrings * Names = new TStringList(); try { Storage->ReadValues(Names, true); for (int Index = 0; Index < Names->Count; Index++) { TCustomCommandType * Command = new TCustomCommandType(); Command->Name = Names->Names[Index]; Command->Command = Names->Values[Names->Names[Index]]; FCommands->Add(Command); } Storage->CloseSubKey(); } __finally { delete Names; } } if (Storage->OpenSubKey(L"CustomCommandsParams", false)) { for (int Index = 0; Index < FCommands->Count; Index++) { TCustomCommandType * Command = GetCommand(Index); Command->Params = Storage->ReadInteger(Command->Name, Command->Params); } Storage->CloseSubKey(); } if (Storage->OpenSubKey(L"CustomCommandsShortCuts", false)) { for (int Index = 0; Index < FCommands->Count; Index++) { TCustomCommandType * Command = GetCommand(Index); Command->ShortCut = (Word)Storage->ReadInteger(Command->Name, Command->ShortCut); } Storage->CloseSubKey(); } Reset(); } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Save(THierarchicalStorage * Storage) { if (Storage->OpenSubKey(L"CustomCommands", true)) { Storage->ClearValues(); for (int Index = 0; Index < FCommands->Count; Index++) { const TCustomCommandType * Command = Commands[Index]; Storage->WriteString(Command->Name, Command->Command); } Storage->CloseSubKey(); } if (Storage->OpenSubKey(L"CustomCommandsParams", true)) { Storage->ClearValues(); for (int Index = 0; Index < FCommands->Count; Index++) { const TCustomCommandType * Command = Commands[Index]; Storage->WriteInteger(Command->Name, Command->Params); } Storage->CloseSubKey(); } if (Storage->OpenSubKey(L"CustomCommandsShortCuts", true)) { Storage->ClearValues(); for (int Index = 0; Index < FCommands->Count; Index++) { const TCustomCommandType * Command = Commands[Index]; if (Command->ShortCut != 0) { Storage->WriteInteger(Command->Name, Command->ShortCut); } } Storage->CloseSubKey(); } } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Clear() { for (int Index = 0; Index < FCommands->Count; Index++) { delete Commands[Index]; } FCommands->Clear(); } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Add(const UnicodeString Name, const UnicodeString ACommand, int Params) { TCustomCommandType * Command = new TCustomCommandType(); Command->Name = Name; Command->Command = ACommand; Command->Params = Params; Add(Command); } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Add(TCustomCommandType * Command) { Insert(Count, Command); } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Insert(int Index, TCustomCommandType * Command) { FCommands->Insert(Index, Command); Modify(); } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Change(int Index, TCustomCommandType * ACommand) { TCustomCommandType * Command = GetCommand(Index); if (!Command->Equals(ACommand)) { delete Command; FCommands->Items[Index] = ACommand; Modify(); } else { delete ACommand; } } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Move(int CurIndex, int NewIndex) { if (CurIndex != NewIndex) { FCommands->Move(CurIndex, NewIndex); Modify(); } } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Delete(int Index) { DebugAssert((Index >= 0) && (Index < Count)); delete GetCommand(Index); FCommands->Delete(Index); Modify(); } //--------------------------------------------------------------------------- class TCustomCommandCompareFunc : public TCppInterfacedObject { public: TCustomCommandCompareFunc(TStrings * Ids) { FIds = Ids; } virtual int __fastcall Invoke(void * Item1, void * Item2) { TCustomCommandType * CustomCommand1 = static_cast(Item1); TCustomCommandType * CustomCommand2 = static_cast(Item2); int Index1 = FIds->IndexOf(CustomCommand1->Id); int Index2 = FIds->IndexOf(CustomCommand2->Id); int Result; // new items to the end if ((Index1 < 0) && (Index2 >= 0)) { Result = 1; } else if ((Index2 < 0) && (Index1 >= 0)) { Result = -1; } // fallback to comparing by name else if ((Index1 < 0) && (Index2 < 0)) { Result = TComparer__1::Default()->Compare(CustomCommand1->Name, CustomCommand2->Name); } else { Result = TComparer__1::Default()->Compare(Index1, Index2); } return Result; } private: TStrings * FIds; }; //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::SortBy(TStrings * Ids) { TCustomCommandCompareFunc * Func = new TCustomCommandCompareFunc(Ids); FCommands->SortList(Func); } //--------------------------------------------------------------------------- int __fastcall TCustomCommandList::GetCount() const { return FCommands->Count; } //--------------------------------------------------------------------------- const TCustomCommandType * __fastcall TCustomCommandList::GetConstCommand(int Index) const { return static_cast(FCommands->Items[Index]); } //--------------------------------------------------------------------------- TCustomCommandType * __fastcall TCustomCommandList::GetCommand(int Index) { return static_cast(FCommands->Items[Index]); } //--------------------------------------------------------------------------- bool __fastcall TCustomCommandList::Equals(const TCustomCommandList * Other) const { bool Result = (Count == Other->Count); for (int Index = 0; Result && (Index < Count); Index++) { Result = Commands[Index]->Equals(Other->Commands[Index]); } return Result; } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::Assign(const TCustomCommandList * Other) { Clear(); for (int Index = 0; Index < Other->Count; Index++) { Add(new TCustomCommandType(*Other->Commands[Index])); } // there should be comparison of with the assigned list, be we rely on caller // to do it instead (TGUIConfiguration::SetCopyParamList) Modify(); } //--------------------------------------------------------------------------- const TCustomCommandType * TCustomCommandList::Find(const UnicodeString Name) const { for (int Index = 0; Index < FCommands->Count; Index++) { if (Commands[Index]->Name == Name) { return Commands[Index]; } } return NULL; } //--------------------------------------------------------------------------- const TCustomCommandType * TCustomCommandList::Find(TShortCut ShortCut) const { for (int Index = 0; Index < FCommands->Count; Index++) { if (Commands[Index]->ShortCut == ShortCut) { return Commands[Index]; } } return NULL; } //--------------------------------------------------------------------------- int TCustomCommandList::FindIndexByFileName(const UnicodeString & FileName) const { for (int Index = 0; Index < FCommands->Count; Index++) { if (CompareFileName(Commands[Index]->FileName, FileName)) { return Index; } } return -1; } //--------------------------------------------------------------------------- void __fastcall TCustomCommandList::ShortCuts(TShortCuts & ShortCuts) const { for (int Index = 0; Index < FCommands->Count; Index++) { const TCustomCommandType * Command = Commands[Index]; if (Command->ShortCut != 0) { ShortCuts.Add(Command->ShortCut); } } }