| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457 | //---------------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include "ScpCommander.h"#include "ScpExplorer.h"#include <CoreMain.h>#include <Common.h>#include <Exceptions.h>#include <Cryptography.h>#include "ProgParams.h"#include "VCLCommon.h"#include "WinConfiguration.h"#include "TerminalManager.h"#include "TextsWin.h"#include "WinInterface.h"#include "PasswordEdit.hpp"#include "ProgParams.h"#include "Tools.h"#include "Custom.h"#include "HelpWin.h"#include <Math.hpp>#include <PasTools.hpp>#include <GUITools.h>//---------------------------------------------------------------------------#pragma package(smart_init)//---------------------------------------------------------------------------const UnicodeString AppName = L"WinSCP";//---------------------------------------------------------------------------TConfiguration * __fastcall CreateConfiguration(){  WinConfiguration = new TWinConfiguration();  CustomWinConfiguration = WinConfiguration;  GUIConfiguration = CustomWinConfiguration;  TProgramParams * Params = TProgramParams::Instance();  UnicodeString IniFileName = Params->SwitchValue(INI_SWITCH);  if (!IniFileName.IsEmpty())  {    if (SameText(IniFileName, INI_NUL))    {      WinConfiguration->SetNulStorage();    }    else if (CheckSafe(Params))    {      IniFileName = ExpandFileName(ExpandEnvironmentVariables(IniFileName));      WinConfiguration->SetExplicitIniFileStorageName(IniFileName);    }  }  if (CheckSafe(Params))  {    std::unique_ptr<TStrings> RawConfig(new TStringList());    if (Params->FindSwitch(RAW_CONFIG_SWITCH, RawConfig.get()))    {      WinConfiguration->OptionsStorage = RawConfig.get();    }  }  return WinConfiguration;}//---------------------------------------------------------------------------TOptions * __fastcall GetGlobalOptions(){  return TProgramParams::Instance();}//---------------------------------------------------------------------------TCustomScpExplorerForm * __fastcall CreateScpExplorer(){  TCustomScpExplorerForm * ScpExplorer;  if (WinConfiguration->Interface == ifExplorer)  {    ScpExplorer = SafeFormCreate<TScpExplorerForm>();  }  else  {    ScpExplorer = SafeFormCreate<TScpCommanderForm>();  }  return ScpExplorer;}//---------------------------------------------------------------------------UnicodeString __fastcall SshVersionString(){  return FORMAT(L"WinSCP-release-%s", (Configuration->Version));}//---------------------------------------------------------------------------UnicodeString __fastcall AppNameString(){  return L"WinSCP";}//---------------------------------------------------------------------------UnicodeString __fastcall GetCompanyRegistryKey(){  return L"Software\\Martin Prikryl";}//---------------------------------------------------------------------------UnicodeString __fastcall GetRegistryKey(){  return GetCompanyRegistryKey() + L"\\WinSCP 2";}//---------------------------------------------------------------------------static bool ForcedOnForeground = false;void __fastcall SetOnForeground(bool OnForeground){  ForcedOnForeground = OnForeground;}//---------------------------------------------------------------------------void __fastcall FlashOnBackground(){  DebugAssert(Application);  if ((WinConfiguration != NULL) && WinConfiguration->FlashTaskbar && !ForcedOnForeground && !ForegroundTask())  {    FlashWindow(Application->MainFormHandle, true);  }}//---------------------------------------------------------------------------void __fastcall LocalSystemSettings(TForm * /*Control*/){  // noop}//---------------------------------------------------------------------------void __fastcall ShowExtendedException(Exception * E){  ShowExtendedExceptionEx(NULL, E);}//---------------------------------------------------------------------------void __fastcall TerminateApplication(){  Application->Terminate();}//---------------------------------------------------------------------------struct TOpenLocalPathHandler{  UnicodeString LocalPath;  UnicodeString LocalFileName;  void __fastcall Open(TObject * Sender, unsigned int & /*Answer*/)  {    TButton * Button = DebugNotNull(dynamic_cast<TButton *>(Sender));    // Reason for separate AMenu variable is given in TPreferencesDialog::EditorFontColorButtonClick    TPopupMenu * AMenu = new TPopupMenu(Application);    // Popup menu has to survive the popup as TBX calls click handler asynchronously (post).    Menu.reset(AMenu);    TMenuItem * Item;    Item = new TMenuItem(Menu.get());    Menu->Items->Add(Item);    Item->Caption = LoadStr(OPEN_TARGET_FOLDER);    Item->OnClick = OpenFolderClick;    if (!LocalFileName.IsEmpty())    {      Item = new TMenuItem(Menu.get());      Menu->Items->Add(Item);      Item->Caption = LoadStr(OPEN_DOWNLOADED_FILE);      Item->OnClick = OpenFileClick;    }    MenuPopup(Menu.get(), Button);  }private:  std::unique_ptr<TPopupMenu> Menu;  void __fastcall OpenFolderClick(TObject * /*Sender*/)  {    if (LocalFileName.IsEmpty())    {      OpenFolderInExplorer(LocalPath);    }    else    {      OpenFileInExplorer(LocalFileName);    }  }  void __fastcall OpenFileClick(TObject * /*Sender*/)  {    ExecuteShellChecked(LocalFileName, L"");  }};//---------------------------------------------------------------------------void __fastcall ShowExtendedExceptionEx(TTerminal * Terminal,  Exception * E){  bool Show = ShouldDisplayException(E);  bool DoNotDisplay = false;  try  {    // This is special case used particularly when called from .NET assembly    // (which always uses /nointeractiveinput),    // but can be useful for other console runs too    TProgramParams * Params = TProgramParams::Instance();    if (Params->FindSwitch(NOINTERACTIVEINPUT_SWITCH))    {      DoNotDisplay = true;      if (Show && CheckXmlLogParam(Params))      {        // The Started argument won't be used with .NET assembly, as it never uses patterns in XML log file name.        // But it theoretically can be used, when started manually.        std::unique_ptr<TActionLog> ActionLog(new TActionLog(Now(), Configuration));        ActionLog->AddFailure(E);        // unnecessary explicit release        ActionLog.reset(NULL);      }    }  }  catch (Exception & E)  {    // swallow  }  TTerminalManager * Manager = TTerminalManager::Instance(false);  if (!DoNotDisplay)  {    ESshTerminate * Terminate = dynamic_cast<ESshTerminate*>(E);    bool CloseOnCompletion = (Terminate != NULL);    bool ForActiveTerminal =      E->InheritsFrom(__classid(EFatal)) && (Terminal != NULL) &&      (Manager != NULL) && (Manager->ActiveTerminal == Terminal);    unsigned int Result;    if (CloseOnCompletion)    {      if (ForActiveTerminal)      {        DebugAssert(!Terminal->Active);        Manager->DisconnectActiveTerminal();      }      if (Terminate->Operation == odoSuspend)      {        // suspend, so that exit prompt is shown only after windows resume        SuspendWindows();      }      DebugAssert(Show);      bool ConfirmExitOnCompletion =        CloseOnCompletion &&        ((Terminate->Operation == odoDisconnect) || (Terminate->Operation == odoSuspend)) &&        WinConfiguration->ConfirmExitOnCompletion;      if (ConfirmExitOnCompletion)      {        TMessageParams Params(mpNeverAskAgainCheck);        unsigned int Answers = 0;        TQueryButtonAlias Aliases[1];        TOpenLocalPathHandler OpenLocalPathHandler;        if (!Terminate->TargetLocalPath.IsEmpty() && !ForActiveTerminal)        {          OpenLocalPathHandler.LocalPath = Terminate->TargetLocalPath;          OpenLocalPathHandler.LocalFileName = Terminate->DestLocalFileName;          Aliases[0].Button = qaIgnore;          Aliases[0].Alias = LoadStr(OPEN_BUTTON);          Aliases[0].OnSubmit = OpenLocalPathHandler.Open;          Aliases[0].MenuButton = true;          Answers |= Aliases[0].Button;          Params.Aliases = Aliases;          Params.AliasesCount = LENOF(Aliases);        }        if (ForActiveTerminal)        {          UnicodeString MessageFormat =            (Manager->Count > 1) ?              FMTLOAD(DISCONNECT_ON_COMPLETION, (Manager->Count - 1)) :              LoadStr(EXIT_ON_COMPLETION);          // Remove the leading "%s\n\n" (not to change the translation originals - previously the error message was prepended)          MessageFormat = FORMAT(MessageFormat, (UnicodeString())).Trim();          MessageFormat = MainInstructions(MessageFormat) + L"\n\n%s";          Result = FatalExceptionMessageDialog(E, qtInformation,            MessageFormat,            Answers | qaYes | qaNo, HELP_NONE, &Params);        }        else        {          Result =            ExceptionMessageDialog(E, qtInformation, L"", Answers | qaOK, HELP_NONE, &Params);        }      }      else      {        Result = qaYes;      }    }    else    {      if (Show)      {        if (ForActiveTerminal)        {          bool HookedDialog = false;          try          {            TMessageParams Params;            if (DebugAlwaysTrue(Manager->ActiveTerminal != NULL) &&                ((Configuration->SessionReopenTimeout == 0) ||                 ((double)Manager->ActiveTerminal->ReopenStart == 0) ||                 (int(double(Now() - Manager->ActiveTerminal->ReopenStart) * MSecsPerDay) < Configuration->SessionReopenTimeout)))            {              Params.Timeout = GUIConfiguration->SessionReopenAutoIdle;              Params.TimeoutAnswer = qaRetry;              Params.TimeoutResponse = Params.TimeoutAnswer;              HookedDialog = Manager->HookFatalExceptionMessageDialog(Params);            }            Result = FatalExceptionMessageDialog(E, qtError, EmptyStr, qaOK, EmptyStr, &Params);          }          __finally          {            if (HookedDialog)            {              Manager->UnhookFatalExceptionMessageDialog();            }          }        }        else        {          Result = ExceptionMessageDialog(E, qtError);        }      }      else      {        Result = qaOK;      }    }    if (Result == qaNeverAskAgain)    {      DebugAssert(CloseOnCompletion);      Result = qaYes;      WinConfiguration->ConfirmExitOnCompletion = false;    }    if (Result == qaYes)    {      DebugAssert(CloseOnCompletion);      DebugAssert(Terminate != NULL);      DebugAssert(Terminate->Operation != odoIdle);      TerminateApplication();      switch (Terminate->Operation)      {        case odoDisconnect:          break;        case odoSuspend:          // suspended before already          break;        case odoShutDown:          ShutDownWindows();          break;        default:          DebugFail();      }    }    else if (Result == qaRetry)    {      // qaRetry is used by FatalExceptionMessageDialog      if (DebugAlwaysTrue(ForActiveTerminal))      {        Manager->ReconnectActiveTerminal();      }    }    else    {      if (ForActiveTerminal)      {        Manager->DisconnectActiveTerminalIfPermanentFreeOtherwise();      }    }  }}//---------------------------------------------------------------------------void __fastcall ShowNotification(TTerminal * Terminal, const UnicodeString & Str,  TQueryType Type){  TTerminalManager * Manager = TTerminalManager::Instance(false);  DebugAssert(Manager != NULL);  Manager->ScpExplorer->PopupTrayBalloon(Terminal, Str, Type);}//---------------------------------------------------------------------------UnicodeString GetThemeName(bool Dark){  return Dark ? L"DarkOfficeXP" : L"OfficeXP";}//---------------------------------------------------------------------------void __fastcall ConfigureInterface(){  DebugAssert(WinConfiguration != NULL);  int BidiModeFlag =    AdjustLocaleFlag(LoadStr(BIDI_MODE), WinConfiguration->BidiModeOverride, false, bdRightToLeft, bdLeftToRight);  Application->BiDiMode = static_cast<TBiDiMode>(BidiModeFlag);  SetTBXSysParam(TSP_XPVISUALSTYLE, XPVS_AUTOMATIC);  UnicodeString Theme = GetThemeName(WinConfiguration->UseDarkTheme());  if (!SameText(TBXCurrentTheme(), Theme))  {    TBXSetTheme(Theme);  }  // Has any effect on Wine only  // (otherwise initial UserDocumentDirectory is equivalent to GetPersonalFolder())  UserDocumentDirectory = GetPersonalFolder();}//---------------------------------------------------------------------------void __fastcall DoAboutDialog(TConfiguration *Configuration){  DoAboutDialog(Configuration, true, NULL);}//---------------------------------------------------------------------void __fastcall DoProductLicense(){  DoLicenseDialog(lcWinScp);}//---------------------------------------------------------------------------const UnicodeString PixelsPerInchKey = L"PixelsPerInch";//---------------------------------------------------------------------int __fastcall GetToolbarLayoutPixelsPerInch(TStrings * Storage, TControl * Control){  int Result;  if (Storage->IndexOfName(PixelsPerInchKey))  {    Result = LoadPixelsPerInch(Storage->Values[PixelsPerInchKey], Control);  }  else  {    Result = -1;  }  return Result;}//---------------------------------------------------------------------UnicodeString __fastcall GetToolbarKey(const UnicodeString & ToolbarName){  UnicodeString Result = ToolbarName;  Result = RemoveSuffix(Result, L"Toolbar", true);  return Result;}//---------------------------------------------------------------------static inline void __fastcall GetToolbarKey(const UnicodeString & ToolbarName,  const UnicodeString & Value, UnicodeString & ToolbarKey){  ToolbarKey = GetToolbarKey(ToolbarName);  if (!Value.IsEmpty())  {    ToolbarKey += L"_" + Value;  }}//---------------------------------------------------------------------------static int __fastcall ToolbarReadInt(const UnicodeString ToolbarName,  const UnicodeString Value, const int Default, const void * ExtraData){  int Result;  if (Value == L"Rev")  {    Result = 2000;  }  else  {    TStrings * Storage = static_cast<TStrings *>(const_cast<void*>(ExtraData));    UnicodeString ToolbarKey;    GetToolbarKey(ToolbarName, Value, ToolbarKey);    if (Storage->IndexOfName(ToolbarKey) >= 0)    {      Result = StrToIntDef(Storage->Values[ToolbarKey], Default);      #if 0      // this does not work well, as it scales down the stretched      // toolbars (path toolbars) too much, it has to be reimplemented smarter      if (Value == L"DockPos")      {        int PixelsPerInch = GetToolbarLayoutPixelsPerInch(Storage);        // When DPI has decreased since the last time, scale down        // toolbar position to get rid of gaps caused by smaller labels.        // Do not do this when DPI has increased as it would introduce gaps,        // as toolbars consists mostly of icons only, that do not scale.        // The toolbars shift themselves anyway, when other toolbars to the left        // get wider. We also risk a bit that toolbar order changes,        // as with very small toolbars (History) we can get scaled down position        // of the following toolbar to the left of it.        // There's special handling (also for scaling-up) stretched toolbars        // in LoadToolbarsLayoutStr        if ((PixelsPerInch > 0) && (Screen->PixelsPerInch < PixelsPerInch))        {          Result = LoadDimension(Result, PixelsPerInch);        }      }      #endif    }    else    {      Result = Default;    }  }  return Result;}//---------------------------------------------------------------------------static UnicodeString __fastcall ToolbarReadString(const UnicodeString ToolbarName,  const UnicodeString Value, const UnicodeString Default, const void * ExtraData){  UnicodeString Result;  TStrings * Storage = static_cast<TStrings *>(const_cast<void*>(ExtraData));  UnicodeString ToolbarKey;  GetToolbarKey(ToolbarName, Value, ToolbarKey);  if (Storage->IndexOfName(ToolbarKey) >= 0)  {    Result = Storage->Values[ToolbarKey];  }  else  {    Result = Default;  }  return Result;}//---------------------------------------------------------------------------static void __fastcall ToolbarWriteInt(const UnicodeString ToolbarName,  const UnicodeString Value, const int Data, const void * ExtraData){  DebugFail();  if (Value != L"Rev")  {    TStrings * Storage = static_cast<TStrings *>(const_cast<void*>(ExtraData));    UnicodeString ToolbarKey;    GetToolbarKey(ToolbarName, Value, ToolbarKey);    DebugAssert(Storage->IndexOfName(ToolbarKey) < 0);    Storage->Values[ToolbarKey] = IntToStr(Data);  }}//---------------------------------------------------------------------------static void __fastcall ToolbarWriteString(const UnicodeString ToolbarName,  const UnicodeString Value, const UnicodeString Data, const void * ExtraData){  DebugAssert(Value.IsEmpty());  TStrings * Storage = static_cast<TStrings *>(const_cast<void*>(ExtraData));  UnicodeString ToolbarKey;  GetToolbarKey(ToolbarName, Value, ToolbarKey);  DebugAssert(Storage->IndexOfName(ToolbarKey) < 0);  Storage->Values[ToolbarKey] = Data;}//---------------------------------------------------------------------------UnicodeString __fastcall GetToolbarsLayoutStr(TControl * OwnerControl){  UnicodeString Result;  TStrings * Storage = new TStringList();  try  {    TBCustomSavePositions(OwnerControl, ToolbarWriteInt, ToolbarWriteString,      Storage);    Storage->Values[PixelsPerInchKey] = SavePixelsPerInch(OwnerControl);    Result = Storage->CommaText;  }  __finally  {    delete Storage;  }  return Result;}//---------------------------------------------------------------------------void __fastcall LoadToolbarsLayoutStr(TControl * OwnerControl, UnicodeString LayoutStr){  TStrings * Storage = CommaTextToStringList(LayoutStr);  try  {    TBCustomLoadPositions(OwnerControl, ToolbarReadInt, ToolbarReadString,      Storage);    int PixelsPerInch = GetToolbarLayoutPixelsPerInch(Storage, OwnerControl);    // Scale toolbars stretched to the first other toolbar to the right    if ((PixelsPerInch > 0) && (PixelsPerInch != GetControlPixelsPerInch(OwnerControl))) // optimization    {      for (int Index = 0; Index < OwnerControl->ComponentCount; Index++)      {        TTBXToolbar * Toolbar =          dynamic_cast<TTBXToolbar *>(OwnerControl->Components[Index]);        if ((Toolbar != NULL) && Toolbar->Stretch &&            (Toolbar->OnGetBaseSize != NULL) &&            // we do not support floating of stretched toolbars            DebugAlwaysTrue(!Toolbar->Floating))        {          TTBXToolbar * FollowingToolbar = NULL;          for (int Index2 = 0; Index2 < OwnerControl->ComponentCount; Index2++)          {            TTBXToolbar * Toolbar2 =              dynamic_cast<TTBXToolbar *>(OwnerControl->Components[Index2]);            if ((Toolbar2 != NULL) && !Toolbar2->Floating &&                (Toolbar2->Parent == Toolbar->Parent) &&                (Toolbar2->DockRow == Toolbar->DockRow) &&                (Toolbar2->DockPos > Toolbar->DockPos) &&                ((FollowingToolbar == NULL) || (FollowingToolbar->DockPos > Toolbar2->DockPos)))            {              FollowingToolbar = Toolbar2;            }          }          if (FollowingToolbar != NULL)          {            int NewWidth = LoadDimension(Toolbar->Width, PixelsPerInch, Toolbar);            FollowingToolbar->DockPos += NewWidth - Toolbar->Width;          }        }      }    }  }  __finally  {    delete Storage;  }}//---------------------------------------------------------------------------TTBXSeparatorItem * __fastcall AddMenuSeparator(TTBCustomItem * Menu){  TTBXSeparatorItem * Item = new TTBXSeparatorItem(Menu);  Menu->Add(Item);  return Item;}//---------------------------------------------------------------------------static TComponent * LastPopupComponent = NULL;static TRect LastPopupRect(-1, -1, -1, -1);static TDateTime LastCloseUp;//---------------------------------------------------------------------------static void __fastcall ConvertMenu(TMenuItem * AItems, TTBCustomItem * Items){  for (int Index = 0; Index < AItems->Count; Index++)  {    TMenuItem * AItem = AItems->Items[Index];    TTBCustomItem * Item;    if (!AItem->Enabled && !AItem->Visible && (AItem->Action == NULL) &&        (AItem->OnClick == NULL) && DebugAlwaysTrue(AItem->Count == 0))    {      TTBXLabelItem * LabelItem = new TTBXLabelItem(Items->Owner);      // TTBXLabelItem has its own Caption      LabelItem->Caption = AItem->Caption;      LabelItem->SectionHeader = true;      Item = LabelItem;    }    else    {      // see TB2DsgnConverter.pas DoConvert      if (AItem->Caption == L"-")      {        Item = new TTBXSeparatorItem(Items->Owner);      }      else      {        if (AItem->Count > 0)        {          Item = new TTBXSubmenuItem(Items->Owner);        }        else        {          Item = new TTBXItem(Items->Owner);        }        Item->Action = AItem->Action;        Item->AutoCheck = AItem->AutoCheck;        Item->Caption = AItem->Caption;        Item->Checked = AItem->Checked;        if (AItem->Default)        {          Item->Options = Item->Options << tboDefault;        }        Item->Enabled = AItem->Enabled;        Item->GroupIndex = AItem->GroupIndex;        Item->HelpContext = AItem->HelpContext;        Item->ImageIndex = AItem->ImageIndex;        Item->RadioItem = AItem->RadioItem;        Item->ShortCut = AItem->ShortCut;        Item->SubMenuImages = AItem->SubMenuImages;        Item->OnClick = AItem->OnClick;      }      Item->Hint = AItem->Hint;      Item->Tag = AItem->Tag;      Item->Visible = AItem->Visible;      if (AItem->Count > 0)      {        ConvertMenu(AItem, Item);      }    }    Items->Add(Item);  }}//---------------------------------------------------------------------------void __fastcall MenuPopup(TPopupMenu * AMenu, TRect Rect,  TComponent * PopupComponent){  // Pressing the same button within 200ms after closing its popup menu  // does nothing.  // It is to immitate close-by-click behavior. Note that menu closes itself  // before onclick handler of button occurs.  // To support content menu popups, we have to check for the popup location too,  // to allow poping menu on different location (such as different node of TTreeView),  // even if there's another popup opened already (so that the time interval  // below does not elapse).  TDateTime N = Now();  TDateTime Diff = N - LastCloseUp;  if ((PopupComponent == LastPopupComponent) &&      (Rect == LastPopupRect) &&      (Diff < TDateTime(0, 0, 0, 200)))  {    LastPopupComponent = NULL;  }  else  {    TTBXPopupMenu * Menu = dynamic_cast<TTBXPopupMenu *>(AMenu);    if (Menu == NULL)    {      Menu = CreateTBXPopupMenu(AMenu->Owner);      Menu->OnPopup = AMenu->OnPopup;      Menu->Items->SubMenuImages = AMenu->Images;      ConvertMenu(AMenu->Items, Menu->Items);    }    Menu->PopupComponent = PopupComponent;    Menu->PopupEx(Rect);    LastPopupComponent = PopupComponent;    LastPopupRect = Rect;    LastCloseUp = Now();  }}//---------------------------------------------------------------------------const int ColorCols = 8;const int StandardColorRows = 2;const int StandardColorCount = ColorCols * StandardColorRows;const int UserColorRows = 1;const int UserColorCount = UserColorRows * ColorCols;const wchar_t ColorSeparator = L',';//---------------------------------------------------------------------------static void __fastcall GetStandardSessionColorInfo(  int Col, int Row, TColor & Color, UnicodeString & Name){  #define COLOR_INFO(COL, ROW, NAME, COLOR) \    if ((Col == COL) && (Row == ROW)) { Name = NAME; Color = TColor(COLOR); } else  // bottom row of default TBX color set  COLOR_INFO(0, 0, L"Rose",              0xCC99FF)  COLOR_INFO(1, 0, L"Tan",               0x99CCFF)  COLOR_INFO(2, 0, L"Light Yellow",      0x99FFFF)  COLOR_INFO(3, 0, L"Light Green",       0xCCFFCC)  COLOR_INFO(4, 0, L"Light Turquoise",   0xFFFFCC)  COLOR_INFO(5, 0, L"Pale Blue",         0xFFCC99)  COLOR_INFO(6, 0, L"Lavender",          0xFF99CC)  // second row of Excel 2010 palette with second color (Lighter Black) skipped  COLOR_INFO(7, 0, L"Light Orange",      0xB5D5FB)  COLOR_INFO(0, 1, L"Darker White",      0xD8D8D8)  COLOR_INFO(1, 1, L"Darker Tan",        0x97BDC4)  COLOR_INFO(2, 1, L"Lighter Blue",      0xE2B38D)  COLOR_INFO(3, 1, L"Light Blue",        0xE4CCB8)  COLOR_INFO(4, 1, L"Lighter Red",       0xB7B9E5)  COLOR_INFO(5, 1, L"Light Olive Green", 0xBCE3D7)  COLOR_INFO(6, 1, L"Light Purple",      0xD9C1CC)  COLOR_INFO(7, 1, L"Light Aqua",        0xE8DDB7)  DebugFail();  #undef COLOR_INFO}//---------------------------------------------------------------------------static void __fastcall SessionColorSetGetColorInfo(  void * /*Data*/, TTBXCustomColorSet * /*Sender*/, int Col, int Row, TColor & Color, UnicodeString & Name){  GetStandardSessionColorInfo(Col, Row, Color, Name);}//---------------------------------------------------------------------------TColor __fastcall RestoreColor(const UnicodeString & CStr){  return TColor(StrToInt(UnicodeString(L"$") + CStr));}//---------------------------------------------------------------------------UnicodeString __fastcall StoreColor(TColor Color){  return IntToHex(Color, 6);}//---------------------------------------------------------------------------static UnicodeString __fastcall ExtractColorStr(UnicodeString & Colors){  return CutToChar(Colors, ColorSeparator, true);}//---------------------------------------------------------------------------static bool __fastcall IsStandardColor(bool SessionColors, TColor Color){  if (SessionColors)  {    for (int Row = 0; Row < StandardColorRows; Row++)    {      for (int Col = 0; Col < ColorCols; Col++)      {        TColor StandardColor;        UnicodeString Name; // unused        GetStandardSessionColorInfo(Col, Row, StandardColor, Name);        if (StandardColor == Color)        {          return true;        }      }    }    return false;  }  else  {    std::unique_ptr<TTBXColorPalette> DefaultColorPalette(new TTBXColorPalette(NULL));    return (DefaultColorPalette->FindCell(Color).X >= 0);  }}//---------------------------------------------------------------------------class TColorChangeData : public TComponent{public:  __fastcall TColorChangeData(TColorChangeEvent OnColorChange, TColor Color, bool SessionColors);  static TColorChangeData * __fastcall Retrieve(TObject * Object);  void __fastcall ColorChange(TColor Color);  __property TColor Color = { read = FColor };  __property bool SessionColors = { read = FSessionColors };private:  TColorChangeEvent FOnColorChange;  TColor FColor;  bool FSessionColors;};//---------------------------------------------------------------------------__fastcall TColorChangeData::TColorChangeData(    TColorChangeEvent OnColorChange, TColor Color, bool SessionColors) :  TComponent(NULL){  Name = QualifiedClassName();  FOnColorChange = OnColorChange;  FColor = Color;  FSessionColors = SessionColors;}//---------------------------------------------------------------------------TColorChangeData * __fastcall TColorChangeData::Retrieve(TObject * Object){  TComponent * Component = DebugNotNull(dynamic_cast<TComponent *>(Object));  TComponent * ColorChangeDataComponent = Component->FindComponent(QualifiedClassName());  return DebugNotNull(dynamic_cast<TColorChangeData *>(ColorChangeDataComponent));}//---------------------------------------------------------------------------static void SaveCustomColors(bool SessionColors, const UnicodeString & Colors){  if (SessionColors)  {    CustomWinConfiguration->SessionColors = Colors;  }  else  {    CustomWinConfiguration->FontColors = Colors;  }}//---------------------------------------------------------------------------static UnicodeString LoadCustomColors(bool SessionColors){  return SessionColors ? CustomWinConfiguration->SessionColors : CustomWinConfiguration->FontColors;}//---------------------------------------------------------------------------void __fastcall TColorChangeData::ColorChange(TColor Color){  // Color palette returns clNone when no color is selected,  // though it should not really happen.  // See also CreateColorPalette  if (Color == Vcl::Graphics::clNone)  {    Color = TColor(0);  }  if ((Color != TColor(0)) &&      !IsStandardColor(SessionColors, Color))  {    UnicodeString Colors = StoreColor(Color);    UnicodeString Temp = LoadCustomColors(SessionColors);    while (!Temp.IsEmpty())    {      UnicodeString CStr = ExtractColorStr(Temp);      if (RestoreColor(CStr) != Color)      {        Colors += UnicodeString(ColorSeparator) + CStr;      }    }    SaveCustomColors(SessionColors, Colors);  }  FOnColorChange(Color);}//---------------------------------------------------------------------------static void __fastcall ColorDefaultClick(void * /*Data*/, TObject * Sender){  TColorChangeData::Retrieve(Sender)->ColorChange(TColor(0));}//---------------------------------------------------------------------------static void __fastcall ColorPaletteChange(void * /*Data*/, TObject * Sender){  TTBXColorPalette * ColorPalette = DebugNotNull(dynamic_cast<TTBXColorPalette *>(Sender));  TColorChangeData::Retrieve(Sender)->ColorChange(GetNonZeroColor(ColorPalette->Color));}//---------------------------------------------------------------------------static UnicodeString __fastcall CustomColorName(int Index){  return UnicodeString(L"Color") + wchar_t(L'A' + Index);}//---------------------------------------------------------------------------static void __fastcall ColorPickClick(void * /*Data*/, TObject * Sender){  TColorChangeData * ColorChangeData = TColorChangeData::Retrieve(Sender);  std::unique_ptr<TColorDialog> Dialog(new TColorDialog(Application));  Dialog->Options = Dialog->Options << cdFullOpen << cdAnyColor;  Dialog->Color = (ColorChangeData->Color != 0 ? ColorChangeData->Color : clSkyBlue);  UnicodeString Temp = LoadCustomColors(ColorChangeData->SessionColors);  int StandardColorIndex = 0;  for (int Index = 0; Index < MaxCustomColors; Index++)  {    TColor CustomColor;    if (!Temp.IsEmpty())    {      CustomColor = RestoreColor(ExtractColorStr(Temp));    }    else    {      if (ColorChangeData->SessionColors)      {        if (StandardColorIndex < StandardColorCount)        {          UnicodeString Name; // not used          GetStandardSessionColorInfo(            StandardColorIndex % ColorCols, StandardColorIndex / ColorCols,            CustomColor, Name);          StandardColorIndex++;        }        else        {          break;        }      }      else      {        // no standard font colors        break;      }    }    Dialog->CustomColors->Values[CustomColorName(Index)] = StoreColor(CustomColor);  }  if (Dialog->Execute())  {    // so that we do not have to try to preserve the excess colors    DebugAssert(UserColorCount <= MaxCustomColors);    UnicodeString Colors;    for (int Index = 0; Index < MaxCustomColors; Index++)    {      UnicodeString CStr = Dialog->CustomColors->Values[CustomColorName(Index)];      if (!CStr.IsEmpty())      {        TColor CustomColor = RestoreColor(CStr);        if ((CustomColor != static_cast<TColor>(-1)) &&            !IsStandardColor(ColorChangeData->SessionColors, CustomColor))        {          AddToList(Colors, StoreColor(CustomColor), ColorSeparator);        }      }    }    SaveCustomColors(ColorChangeData->SessionColors, Colors);    // call color change only after copying custom colors back,    // so that it can add selected color to the user list    ColorChangeData->ColorChange(GetNonZeroColor(Dialog->Color));  }}//---------------------------------------------------------------------------TPopupMenu * __fastcall CreateSessionColorPopupMenu(TColor Color,  TColorChangeEvent OnColorChange){  std::unique_ptr<TTBXPopupMenu> PopupMenu(new TTBXPopupMenu(Application));  CreateSessionColorMenu(PopupMenu->Items, Color, OnColorChange);  return PopupMenu.release();}//---------------------------------------------------------------------------static void __fastcall UserCustomColorSetGetColorInfo(  void * /*Data*/, TTBXCustomColorSet * Sender, int Col, int Row, TColor & Color, UnicodeString & /*Name*/){  int Index = (Row * Sender->ColCount) + Col;  bool SessionColors = static_cast<bool>(Sender->Tag);  UnicodeString Temp = LoadCustomColors(SessionColors);  while ((Index > 0) && !Temp.IsEmpty())  {    ExtractColorStr(Temp);    Index--;  }  if (!Temp.IsEmpty())  {    Color = RestoreColor(ExtractColorStr(Temp));  }  else  {    // hide the trailing cells    Color = Vcl::Graphics::clNone;  }}//---------------------------------------------------------------------------void __fastcall CreateColorPalette(TTBCustomItem * Owner, TColor Color, int Rows,  TCSGetColorInfo OnGetColorInfo, TColorChangeEvent OnColorChange, bool SessionColors){  TTBXColorPalette * ColorPalette = new TTBXColorPalette(Owner);  if (OnGetColorInfo != NULL)  {    TTBXCustomColorSet * ColorSet = new TTBXCustomColorSet(Owner);    ColorPalette->InsertComponent(ColorSet);    ColorPalette->ColorSet = ColorSet;    // has to be set only after it's assigned to color palette    ColorSet->ColCount = ColorCols;    ColorSet->RowCount = Rows;    ColorSet->OnGetColorInfo = OnGetColorInfo;    ColorSet->Tag = static_cast<int>(SessionColors);  }  // clNone = no selection, see also ColorChange  ColorPalette->Color = (Color != 0) ? Color : Vcl::Graphics::clNone;  ColorPalette->OnChange = MakeMethod<TNotifyEvent>(NULL, ColorPaletteChange);  ColorPalette->InsertComponent(new TColorChangeData(OnColorChange, Color, SessionColors));  Owner->Add(ColorPalette);  Owner->Add(new TTBXSeparatorItem(Owner));}//---------------------------------------------------------------------------static void __fastcall CreateColorMenu(TComponent * AOwner, TColor Color,  TColorChangeEvent OnColorChange, bool SessionColors,  const UnicodeString & DefaultColorCaption, const UnicodeString & DefaultColorHint,  const UnicodeString & HelpKeyword,  const UnicodeString & ColorPickHint){  TTBCustomItem * Owner = dynamic_cast<TTBCustomItem *>(AOwner);  if (DebugAlwaysTrue(Owner != NULL))  {    Owner->Clear();    TTBCustomItem * Item;    Item = new TTBXItem(Owner);    Item->Caption = DefaultColorCaption;    Item->Hint = DefaultColorHint;    Item->HelpKeyword = HelpKeyword;    Item->OnClick = MakeMethod<TNotifyEvent>(NULL, ColorDefaultClick);    Item->Checked = (Color == TColor(0));    Item->InsertComponent(new TColorChangeData(OnColorChange, Color, SessionColors));    Owner->Add(Item);    Owner->Add(new TTBXSeparatorItem(Owner));    int CustomColorCount = 0;    UnicodeString Temp = LoadCustomColors(SessionColors);    while (!Temp.IsEmpty())    {      CustomColorCount++;      ExtractColorStr(Temp);    }    if (CustomColorCount > 0)    {      CustomColorCount = Min(CustomColorCount, UserColorCount);      int RowCount = ((CustomColorCount + ColorCols - 1) / ColorCols);      DebugAssert(RowCount <= UserColorRows);      CreateColorPalette(Owner, Color, RowCount,        MakeMethod<TCSGetColorInfo>(NULL, UserCustomColorSetGetColorInfo),        OnColorChange, SessionColors);    }    if (SessionColors)    {      CreateColorPalette(Owner, Color, StandardColorRows,        MakeMethod<TCSGetColorInfo>(NULL, SessionColorSetGetColorInfo),        OnColorChange, SessionColors);    }    else    {      CreateColorPalette(Owner, Color, -1, NULL, OnColorChange, SessionColors);    }    Owner->Add(new TTBXSeparatorItem(Owner));    Item = new TTBXItem(Owner);    Item->Caption = LoadStr(COLOR_PICK_CAPTION);    Item->Hint = ColorPickHint;    Item->HelpKeyword = HelpKeyword;    Item->OnClick = MakeMethod<TNotifyEvent>(NULL, ColorPickClick);    Item->InsertComponent(new TColorChangeData(OnColorChange, Color, SessionColors));    Owner->Add(Item);  }}//---------------------------------------------------------------------------void __fastcall CreateSessionColorMenu(TComponent * AOwner, TColor Color,  TColorChangeEvent OnColorChange){  CreateColorMenu(    AOwner, Color, OnColorChange, true,    LoadStr(COLOR_TRUE_DEFAULT_CAPTION), LoadStr(COLOR_DEFAULT_HINT),    HELP_COLOR, LoadStr(COLOR_PICK_HINT));}//---------------------------------------------------------------------------void __fastcall CreateEditorBackgroundColorMenu(TComponent * AOwner, TColor Color,  TColorChangeEvent OnColorChange){  CreateColorMenu(    AOwner, Color, OnColorChange, true,    LoadStr(COLOR_TRUE_DEFAULT_CAPTION), LoadStr(EDITOR_BACKGROUND_COLOR_HINT),    HELP_COLOR, LoadStr(EDITOR_BACKGROUND_COLOR_PICK_HINT));}//---------------------------------------------------------------------------TPopupMenu * __fastcall CreateColorPopupMenu(TColor Color,  TColorChangeEvent OnColorChange){  std::unique_ptr<TTBXPopupMenu> PopupMenu(new TTBXPopupMenu(Application));  CreateColorMenu(    PopupMenu->Items, Color, OnColorChange, false,    LoadStr(COLOR_TRUE_DEFAULT_CAPTION), UnicodeString(),    HELP_NONE, UnicodeString());  return PopupMenu.release();}//---------------------------------------------------------------------------struct TThreadParam{  TThreadFunc ThreadFunc;  void * Parameter;};//---------------------------------------------------------------------------static int __fastcall ThreadProc(void * AParam){  TThreadParam * Param = reinterpret_cast<TThreadParam *>(AParam);  unsigned int Result = Param->ThreadFunc(Param->Parameter);  delete Param;  EndThread(Result);  return Result;}//---------------------------------------------------------------------------int __fastcall StartThread(void * SecurityAttributes, unsigned StackSize,  TThreadFunc ThreadFunc, void * Parameter, unsigned CreationFlags,  TThreadID & ThreadId){  TThreadParam * Param = new TThreadParam;  Param->ThreadFunc = ThreadFunc;  Param->Parameter = Parameter;  return BeginThread(SecurityAttributes, StackSize, ThreadProc, Param,    CreationFlags, ThreadId);}//---------------------------------------------------------------------------static TShortCut FirstCtrlNumberShortCut = ShortCut(L'0', TShiftState() << ssCtrl);static TShortCut LastCtrlNumberShortCut = ShortCut(L'9', TShiftState() << ssCtrl);static TShortCut FirstCtrlKeyPadShortCut = ShortCut(VK_NUMPAD0, TShiftState() << ssCtrl);static TShortCut LastCtrlKeyPadShortCut = ShortCut(VK_NUMPAD9, TShiftState() << ssCtrl);static TShortCut FirstShiftCtrlAltLetterShortCut = ShortCut(L'A', TShiftState() << ssShift << ssCtrl << ssAlt);static TShortCut LastShiftCtrlAltLetterShortCut = ShortCut(L'Z', TShiftState() << ssShift << ssCtrl << ssAlt);//---------------------------------------------------------------------------void __fastcall InitializeShortCutCombo(TComboBox * ComboBox,  const TShortCuts & ShortCuts){  ComboBox->Items->BeginUpdate();  try  {    ComboBox->Items->Clear();    ComboBox->Items->AddObject(LoadStr(SHORTCUT_NONE), reinterpret_cast<TObject* >(0));    for (TShortCut AShortCut = FirstCtrlNumberShortCut; AShortCut <= LastCtrlNumberShortCut; AShortCut++)    {      if (!ShortCuts.Has(AShortCut))      {        ComboBox->Items->AddObject(ShortCutToText(AShortCut), reinterpret_cast<TObject* >(AShortCut));      }    }    for (TShortCut AShortCut = FirstShiftCtrlAltLetterShortCut; AShortCut <= LastShiftCtrlAltLetterShortCut; AShortCut++)    {      if (!ShortCuts.Has(AShortCut))      {        ComboBox->Items->AddObject(ShortCutToText(AShortCut), reinterpret_cast<TObject* >(AShortCut));      }    }  }  __finally  {    ComboBox->Items->EndUpdate();  }  ComboBox->Style = csDropDownList;  ComboBox->DropDownCount = Max(ComboBox->DropDownCount, 16);}//---------------------------------------------------------------------------void __fastcall SetShortCutCombo(TComboBox * ComboBox, TShortCut Value){  for (int Index = ComboBox->Items->Count - 1; Index >= 0; Index--)  {    TShortCut AShortCut = TShortCut(ComboBox->Items->Objects[Index]);    if (AShortCut == Value)    {      ComboBox->ItemIndex = Index;      break;    }    else if (AShortCut < Value)    {      DebugAssert(Value != 0);      ComboBox->Items->InsertObject(Index + 1, ShortCutToText(Value),        reinterpret_cast<TObject* >(Value));      ComboBox->ItemIndex = Index + 1;      break;    }    DebugAssert(Index > 0);  }}//---------------------------------------------------------------------------TShortCut __fastcall GetShortCutCombo(TComboBox * ComboBox){  return TShortCut(ComboBox->Items->Objects[ComboBox->ItemIndex]);}//---------------------------------------------------------------------------TShortCut __fastcall NormalizeCustomShortCut(TShortCut ShortCut){  if ((FirstCtrlKeyPadShortCut <= ShortCut) && (ShortCut <= LastCtrlKeyPadShortCut))  {    ShortCut = FirstCtrlNumberShortCut + (ShortCut - FirstCtrlKeyPadShortCut);  }  return ShortCut;}//---------------------------------------------------------------------------bool __fastcall IsCustomShortCut(TShortCut ShortCut){  return    ((FirstCtrlNumberShortCut <= ShortCut) && (ShortCut <= LastCtrlNumberShortCut)) ||    ((FirstShiftCtrlAltLetterShortCut <= ShortCut) && (ShortCut <= LastShiftCtrlAltLetterShortCut));}//---------------------------------------------------------------------------//---------------------------------------------------------------------------class TMasterPasswordDialog : public TCustomDialog{public:  __fastcall TMasterPasswordDialog(TComponent * AOwner);  void Init(bool Current);  bool __fastcall Execute(UnicodeString & CurrentPassword, UnicodeString & NewPassword);protected:  virtual void __fastcall DoValidate();  virtual void __fastcall DoChange(bool & CanSubmit);private:  TPasswordEdit * CurrentEdit;  TPasswordEdit * NewEdit;  TPasswordEdit * ConfirmEdit;};//---------------------------------------------------------------------------// Need to have an Owner argument for SafeFormCreate__fastcall TMasterPasswordDialog::TMasterPasswordDialog(TComponent *) :  TCustomDialog(EmptyStr){}//---------------------------------------------------------------------------void TMasterPasswordDialog::Init(bool Current){  HelpKeyword = Current ? HELP_MASTER_PASSWORD_CURRENT : HELP_MASTER_PASSWORD_CHANGE;  Caption = LoadStr(MASTER_PASSWORD_CAPTION);  CurrentEdit = new TPasswordEdit(this);  AddEdit(CurrentEdit, CreateLabel(LoadStr(MASTER_PASSWORD_CURRENT)));  EnableControl(CurrentEdit, Current || WinConfiguration->UseMasterPassword);  CurrentEdit->MaxLength = PasswordMaxLength();  if (!Current)  {    NewEdit = new TPasswordEdit(this);    AddEdit(NewEdit, CreateLabel(LoadStr(MASTER_PASSWORD_NEW)));    NewEdit->MaxLength = CurrentEdit->MaxLength;    if (!WinConfiguration->UseMasterPassword)    {      ActiveControl = NewEdit;    }    ConfirmEdit = new TPasswordEdit(this);    AddEdit(ConfirmEdit, CreateLabel(LoadStr(MASTER_PASSWORD_CONFIRM)));    ConfirmEdit->MaxLength = CurrentEdit->MaxLength;  }  else  {    NewEdit = NULL;    ConfirmEdit = NULL;  }}//---------------------------------------------------------------------------bool __fastcall TMasterPasswordDialog::Execute(  UnicodeString & CurrentPassword, UnicodeString & NewPassword){  bool Result = TCustomDialog::Execute();  if (Result)  {    if (CurrentEdit->Enabled)    {      CurrentPassword = CurrentEdit->Text;    }    if (NewEdit != NULL)    {      NewPassword = NewEdit->Text;    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TMasterPasswordDialog::DoChange(bool & CanSubmit){  CanSubmit =    (!WinConfiguration->UseMasterPassword || (IsValidPassword(CurrentEdit->Text) >= 0)) &&    ((NewEdit == NULL) || (IsValidPassword(NewEdit->Text) >= 0)) &&    ((ConfirmEdit == NULL) || (IsValidPassword(ConfirmEdit->Text) >= 0));  TCustomDialog::DoChange(CanSubmit);}//---------------------------------------------------------------------------void __fastcall TMasterPasswordDialog::DoValidate(){  TCustomDialog::DoValidate();  if (WinConfiguration->UseMasterPassword &&      !WinConfiguration->ValidateMasterPassword(CurrentEdit->Text))  {    CurrentEdit->SetFocus();    CurrentEdit->SelectAll();    throw Exception(MainInstructions(LoadStr(MASTER_PASSWORD_INCORRECT)));  }  if (NewEdit != NULL)  {    if (NewEdit->Text != ConfirmEdit->Text)    {      ConfirmEdit->SetFocus();      ConfirmEdit->SelectAll();      throw Exception(MainInstructions(LoadStr(MASTER_PASSWORD_DIFFERENT)));    }    int Valid = IsValidPassword(NewEdit->Text);    if (Valid <= 0)    {      DebugAssert(Valid == 0);      if (MessageDialog(LoadStr(MASTER_PASSWORD_SIMPLE2), qtWarning,            qaOK | qaCancel, HELP_MASTER_PASSWORD_SIMPLE) == qaCancel)      {        NewEdit->SetFocus();        NewEdit->SelectAll();        Abort();      }    }  }}//---------------------------------------------------------------------------static bool __fastcall DoMasterPasswordDialog(bool Current,  UnicodeString & NewPassword){  bool Result;  // This can be a standalone dialog when opening session from commandline  TMasterPasswordDialog * Dialog = SafeFormCreate<TMasterPasswordDialog>();  try  {    Dialog->Init(Current);    UnicodeString CurrentPassword;    Result = Dialog->Execute(CurrentPassword, NewPassword);    if (Result)    {      if ((Current || WinConfiguration->UseMasterPassword) &&          DebugAlwaysTrue(!CurrentPassword.IsEmpty()))      {        WinConfiguration->SetMasterPassword(CurrentPassword);      }    }  }  __finally  {    delete Dialog;  }  return Result;}//---------------------------------------------------------------------------bool __fastcall DoMasterPasswordDialog(){  UnicodeString NewPassword;  bool Result = DoMasterPasswordDialog(true, NewPassword);  DebugAssert(NewPassword.IsEmpty());  return Result;}//---------------------------------------------------------------------------bool __fastcall DoChangeMasterPasswordDialog(UnicodeString & NewPassword){  bool Result = DoMasterPasswordDialog(false, NewPassword);  return Result;}//---------------------------------------------------------------------------void __fastcall MessageWithNoHelp(const UnicodeString & Message){  TMessageParams Params;  Params.AllowHelp = false; // to avoid recursion  if (MessageDialog(LoadStr(HELP_SEND_MESSAGE2), qtConfirmation,        qaOK | qaCancel, HELP_NONE, &Params) == qaOK)  {    SearchHelp(Message);  }}//---------------------------------------------------------------------------void __fastcall CheckLogParam(TProgramParams * Params){  UnicodeString LogFile;  if (Params->FindSwitch(LOG_SWITCH, LogFile) && CheckSafe(Params))  {    Configuration->Usage->Inc(L"ScriptLog");    Configuration->TemporaryLogging(LogFile);  }}//---------------------------------------------------------------------------bool __fastcall CheckXmlLogParam(TProgramParams * Params){  UnicodeString LogFile;  bool Result =    Params->FindSwitch(L"XmlLog", LogFile) &&    CheckSafe(Params);  if (Result)  {    Configuration->Usage->Inc(L"ScriptXmlLog");    Configuration->TemporaryActionsLogging(LogFile);    if (Params->FindSwitch(L"XmlLogRequired"))    {      Configuration->LogActionsRequired = true;    }  }  return Result;}//---------------------------------------------------------------------------bool __fastcall CheckSafe(TProgramParams * Params){  // Originally we warned when the test didn't pass,  // but it would actually be helping hackers, so let's be silent.  // Earlier we tested presence of any URL on command-line.  // That was added to prevent abusing URL handler in 3.8.2 (2006).  // Later in 4.0.4 (2007) an /Unsafe switch was added to URL handler registration.  // So by now, we can check for the switch only and  // do not limit user from combining URL with say  // /rawconfig  return !Params->FindSwitch(UNSAFE_SWITCH);}
 |