| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792 | //---------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include "WinInterface.h"#include "Properties.h"#include <VCLCommon.h>#include <Common.h>#include <Terminal.h>#include <TextsWin.h>#include <GUITools.h>#include <CoreMain.h>#include <Tools.h>#include <BaseUtils.hpp>//---------------------------------------------------------------------#pragma link "PathLabel"#pragma link "Rights"#pragma resource "*.dfm"//---------------------------------------------------------------------bool __fastcall DoPropertiesDialog(TStrings * FileList,  const UnicodeString Directory, const TRemoteTokenList * GroupList,  const TRemoteTokenList * UserList, TStrings * ChecksumAlgs,  TRemoteProperties * Properties,  int AllowedChanges, bool UserGroupByID, TCalculateSizeEvent OnCalculateSize,  TCalculateChecksumEvent OnCalculateChecksum){  bool Result;  TPropertiesDialog * PropertiesDialog = new TPropertiesDialog(Application,    FileList, Directory, GroupList, UserList, ChecksumAlgs, AllowedChanges, UserGroupByID,    OnCalculateSize, OnCalculateChecksum);  try  {    Result = PropertiesDialog->Execute(*Properties);  }  __finally  {    delete PropertiesDialog;  }  return Result;}//---------------------------------------------------------------------__fastcall TPropertiesDialog::TPropertiesDialog(TComponent* AOwner,  TStrings * FileList, const UnicodeString Directory,  const TRemoteTokenList * GroupList, const TRemoteTokenList * UserList,  TStrings * ChecksumAlgs,  int AllowedChanges, bool UserGroupByID, TCalculateSizeEvent OnCalculateSize,  TCalculateChecksumEvent OnCalculateChecksum)  : TForm(AOwner){  FOnCalculateSize = OnCalculateSize;  FOnCalculateChecksum = OnCalculateChecksum;  RightsFrame->OnChange = ControlChange;  FFileList = new TStringList();  FFileList->Assign(FileList);  FAllowedChanges = AllowedChanges;  FUserGroupByID = UserGroupByID;  FAllowCalculateStats = false;  FStatsNotCalculated = false;  FChecksumLoaded = false;  FMultipleChecksum = false;  LocationLabel->Caption = Directory;  FGroupList = GroupList;  FUserList = UserList;  FChecksumAlgs = ChecksumAlgs;  ReadOnlyControl(ChecksumEdit);  ChecksumUnknownLabel->Caption = LoadStr(PROPERTIES_CHECKSUM_UNKNOWN);  LoadInfo();  UseSystemSettings(this);}//---------------------------------------------------------------------------__fastcall TPropertiesDialog::~TPropertiesDialog(){  delete FFileList;  FFileList = NULL;}//---------------------------------------------------------------------bool __fastcall TPropertiesDialog::Execute(TRemoteProperties & Properties){  SetFileProperties(Properties);  PageControl->ActivePage = CommonSheet;  if (FAllowedChanges & cpGroup) ActiveControl = GroupComboBox;    else  if (FAllowedChanges & cpOwner) ActiveControl = OwnerComboBox;    else  if (FAllowedChanges & cpMode) ActiveControl = RightsFrame;    else ActiveControl = CancelButton;  if (DebugAlwaysTrue(FChecksumAlgs != NULL))  {    ChecksumAlgEdit->Items->Assign(FChecksumAlgs);    int ChecksumIndex = FChecksumAlgs->IndexOf(GUIConfiguration->ChecksumAlg);    if (ChecksumIndex < 0)    {      ChecksumIndex = 0;    }    ChecksumAlgEdit->ItemIndex = ChecksumIndex;  }  ResetChecksum();  UpdateControls();  bool Result = (ShowModal() == DefaultResult());  if (Result)  {    Properties = GetFileProperties();  }  return Result;}//---------------------------------------------------------------------------TModalResult __fastcall TPropertiesDialog::DefaultResult(){  // Fallback when ChecksumButton is default  return ::DefaultResult(this, OkButton);}//---------------------------------------------------------------------------UnicodeString __fastcall TPropertiesDialog::LoadRemoteToken(  const TRemoteToken & Token){  UnicodeString Result;  if (FUserGroupByID)  {    if (Token.IDValid)    {      if (Token.NameValid)      {        Result = FORMAT(L"%s [%d]", (Token.Name, int(Token.ID)));      }      else      {        Result = FORMAT(L"[%d]", (int(Token.ID)));      }    }    else    {      // be it valid or not      Result = Token.Name;    }  }  else  {    // what if name is not filled in?    Result = Token.Name;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::LoadRemoteToken(  TComboBox * ComboBox, bool Valid, const TRemoteToken & Token){  if (Valid)  {    ComboBox->Text = LoadRemoteToken(Token);  }  else  {    ComboBox->Text = L"";  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::LoadRemoteTokens(TComboBox * ComboBox,  const TRemoteTokenList * List){  TStrings * Items = ComboBox->Items;  Items->BeginUpdate();  try  {    Items->Clear();    if (List != NULL)    {      int Count = List->Count();      for (int Index = 0; Index < Count; Index++)      {        Items->Add(LoadRemoteToken(*List->Token(Index)));      }    }  }  __finally  {    Items->EndUpdate();  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::LoadInfo(){  DebugAssert(FFileList->Count > 0);  FMultiple = FFileList->Count > 1;  FMultipleChecksum = FMultiple;  FAllowCalculateStats = false;  FStatsNotCalculated = false;  __int64 FilesSize = 0;  TCalculateSizeStats Stats;  for (int Index = 0; Index < FFileList->Count; Index++)  {    TRemoteFile * File = (TRemoteFile *)(FFileList->Objects[Index]);    if (File->IsDirectory)    {      Stats.Directories++;      // we should use TTerminal::CanRecurseToDirectory instead      if (!File->IsSymLink)      {        FAllowCalculateStats = true;        FStatsNotCalculated = true;        FMultipleChecksum = true;      }    }    else    {      Stats.Files++;    }    if (File->IsSymLink)    {      Stats.SymLinks++;    }    FilesSize += File->Size;  }  if (!FMultiple)  {    // Show only file name, if we have only single file/directory.    // For directory, this changes, once "Calculate" button is pressed    Stats = TCalculateSizeStats();  }  LoadRemoteTokens(GroupComboBox, FGroupList);  LoadRemoteTokens(OwnerComboBox, FUserList);  RightsFrame->AllowAddXToDirectories = (Stats.Directories > 0);  RecursiveCheck->Visible = (Stats.Directories > 0);  RecursiveBevel->Visible = (Stats.Directories > 0);  LoadStats(FilesSize, Stats);  RightsFrame->AllowUndef = FMultiple;  if (!FMultiple)  {    TRemoteFile * File = (TRemoteFile *)(FFileList->Objects[0]);    DebugAssert(File);    UpdateFileImage();    LinksToLabelLabel->Visible = File->IsSymLink;    LinksToLabel->Visible = File->IsSymLink;    if (File->IsSymLink)    {      LinksToLabel->Caption = File->LinkTo;    }    RightsFrame->AllowAddXToDirectories = File->IsDirectory;    Caption = FMTLOAD(PROPERTIES_FILE_CAPTION, (File->FileName));    RecursiveCheck->Visible = File->IsDirectory;    RecursiveBevel->Visible = File->IsDirectory;  }  else  {    Caption = FMTLOAD(PROPERTIES_FILES_CAPTION, (FFileList->Strings[0]));    LinksToLabelLabel->Hide();    LinksToLabel->Hide();    LoadDialogImage(FileIconImage, L"Multiple Files");  }  ChecksumGroup->Visible = !FMultipleChecksum;  ChecksumView->Visible = FMultipleChecksum;}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::UpdateFileImage(){  TImageList * ImageList = ShellImageListForControl(this, ilsLarge);  FileIconImage->Picture->Bitmap = NULL;  TRemoteFile * File = (TRemoteFile *)(FFileList->Objects[0]);  // shell image list does not have fixed large icon size  // (it is probably 32x32 min, but can be larger, on xp it is 48x48 if  // large icons are enabled, on vista can be even larger).  // so we stretch (shrink) the icon to 32x32 here to be sure.  Graphics::TBitmap * Bitmap = new Graphics::TBitmap;  try  {    ImageList->GetBitmap(File->IconIndex, Bitmap);    int Size = DialogImageSize(this);    // Use exact DPI-scaled size, not approximate scaling by font size.    // Otherwise we stretch icons unnecessarily because the canvas    // is one or two pixels off the icon size    FileIconImage->Width = Size;    FileIconImage->Height = Size;    FileIconImage->Picture->Bitmap->Width = Size;    FileIconImage->Picture->Bitmap->Height = Size;    FileIconImage->Picture->Bitmap->Canvas->StretchDraw(      TRect(0, 0, Size, Size),      Bitmap);  }  __finally  {    delete Bitmap;  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::LoadStats(__int64 FilesSize,  const TCalculateSizeStats & Stats){  UnicodeString SizeStr;  UnicodeString FilesStr;  if (FStatsNotCalculated)  {    SizeStr = LoadStr(PROPERTIES_UNKNOWN_SIZE);  }  else  {    SizeStr = FormatBytes(FilesSize);    UnicodeString SizeUnorderedStr = FormatBytes(FilesSize, fbNone);    if (SizeStr != SizeUnorderedStr)    {      SizeStr = FORMAT(L"%s (%s)", (SizeStr, SizeUnorderedStr));    }  }  if (((Stats.Files + Stats.Directories) == 0) && !FMultiple)  {    TRemoteFile * File = (TRemoteFile *)(FFileList->Objects[0]);    DebugAssert(File != NULL);    FilesStr = File->FileName;  }  else  {    if (Stats.Files > 0)    {      FilesStr = (Stats.Files == 1) ? FMTLOAD(PROPERTIES_FILE, (Stats.Files)) :        FMTLOAD(PROPERTIES_FILES, (Stats.Files));      if (Stats.Directories > 0)      {        FilesStr = FORMAT(L"%s, ", (FilesStr));      }    }    if (Stats.Directories > 0)    {      FilesStr += (Stats.Directories == 1) ? FMTLOAD(PROPERTIES_DIRECTORY, (Stats.Directories)) :        FMTLOAD(PROPERTIES_DIRECTORIES, (Stats.Directories));    }    if (Stats.SymLinks > 0)    {      UnicodeString SymLinksStr;      SymLinksStr = (Stats.SymLinks == 1) ? FMTLOAD(PROPERTIES_SYMLINK, (Stats.SymLinks)) :        FMTLOAD(PROPERTIES_SYMLINKS, (Stats.SymLinks));      FilesStr = FORMAT(L"%s (%s)", (FilesStr, SymLinksStr));    }  }  SizeLabel->Caption = SizeStr;  FileLabel->Caption = FilesStr;}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::SetFileProperties(const TRemoteProperties & value){  TValidProperties Valid;  if (value.Valid.Contains(vpRights) && FAllowedChanges & cpMode) Valid << vpRights;  if (value.Valid.Contains(vpOwner) && FAllowedChanges & cpOwner) Valid << vpOwner;  if (value.Valid.Contains(vpGroup) && FAllowedChanges & cpGroup) Valid << vpGroup;  FOrigProperties = value;  FOrigProperties.Valid = Valid;  FOrigProperties.Recursive = false;  if (value.Valid.Contains(vpRights))  {    RightsFrame->Rights = value.Rights;    RightsFrame->AddXToDirectories = value.AddXToDirectories;  }  else  {    RightsFrame->Rights = TRights();    RightsFrame->AddXToDirectories = false;  }  LoadRemoteToken(GroupComboBox, value.Valid.Contains(vpGroup), value.Group);  LoadRemoteToken(OwnerComboBox, value.Valid.Contains(vpOwner), value.Owner);  RecursiveCheck->Checked = value.Recursive;  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::StoreRemoteToken(unsigned int ID,  const UnicodeString & Text, const TRemoteTokenList * List, TRemoteToken & Result){  DebugAssert(List != NULL);  const TRemoteToken * Token = List->Find(ID);  if (Token == NULL)  {    Result.ID = ID;    Result.Name = Text;  }  else  {    Result = *Token;  }}//---------------------------------------------------------------------------TRemoteToken __fastcall TPropertiesDialog::StoreRemoteToken(const TRemoteToken & Orig,  UnicodeString Text, int Message, const TRemoteTokenList * List){  TRemoteToken Result;  Text = Text.Trim();  if (!Text.IsEmpty())  {    if (FUserGroupByID)    {      DebugAssert(List != NULL);      int IDStart = Text.LastDelimiter(L"[");      if (!Text.IsEmpty() && (IDStart >= 0) && (Text[Text.Length()] == L']'))      {        int ID;        UnicodeString IDStr = Text.SubString(IDStart + 1, Text.Length() - IDStart - 1);        if (!TryStrToInt(IDStr, ID))        {          throw Exception(Message);        }        else        {          StoreRemoteToken(ID, Text.SubString(1, IDStart - 1).Trim(), List, Result);        }      }      else      {        const TRemoteToken * Token = List->Find(Text);        if (Token == NULL)        {          int ID;          if (TryStrToInt(Text, ID))          {            StoreRemoteToken(ID, Text, List, Result);          }          else          {            throw Exception(MainInstructions(FMTLOAD(PROPERTIES_UNKNOWN_TOKEN, (Text))));          }        }        else        {          Result = *Token;        }      }    }    else    {      Result.Name = Text;    }  }  if (LoadRemoteToken(Result) == LoadRemoteToken(Orig))  {    Result = Orig;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::StoreRemoteToken(TComboBox * ComboBox,  int ChangeFlag, TValidProperty PropertyFlag, const TRemoteToken & Orig,  TRemoteToken & Token, int Message, const TRemoteTokenList * List,  TRemoteProperties & Properties){  UnicodeString Text = ComboBox->Text.Trim();  if (FLAGSET(FAllowedChanges, ChangeFlag))  {    Token = StoreRemoteToken(Orig, Text, Message, List);    if (Token.IsSet)    {      Properties.Valid << PropertyFlag;    }  }}//---------------------------------------------------------------------------TRemoteProperties __fastcall TPropertiesDialog::GetFileProperties(){  TRemoteProperties Result;  if (FAllowedChanges & cpMode)  {    Result.Valid << vpRights;    Result.Rights = RightsFrame->Rights;    Result.AddXToDirectories = RightsFrame->AddXToDirectories;  }  StoreRemoteToken(GroupComboBox, cpGroup, vpGroup, FOrigProperties.Group,    Result.Group, PROPERTIES_INVALID_GROUP, FGroupList, Result);  StoreRemoteToken(OwnerComboBox, cpOwner, vpOwner, FOrigProperties.Owner,    Result.Owner, PROPERTIES_INVALID_OWNER, FUserList, Result);  Result.Recursive = RecursiveCheck->Checked;  return Result;}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::ControlChange(TObject * /*Sender*/){  if (Visible)  {    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::UpdateControls(){  // No point enabling recursive check if there's no change allowed (supported),  // i.e. with WebDAV.  EnableControl(RecursiveCheck, ((FAllowedChanges & (cpGroup | cpOwner | cpMode)) != 0));  bool Allow;  try  {    Allow =      !TRemoteProperties::ChangedProperties(FOrigProperties, GetFileProperties()).Valid.Empty() ||      (RecursiveCheck->Enabled && RecursiveCheck->Checked);  }  catch(...)  {    // when properties are invalid allow submitting the form,    // because that reveals the cause to the user, otherwise he/she    // may not be able to tell what is wrong    Allow = true;  }  EnableControl(OkButton, Allow);  EnableControl(GroupComboBox, FAllowedChanges & cpGroup);  EnableControl(OwnerComboBox, FAllowedChanges & cpOwner);  EnableControl(RightsFrame, FAllowedChanges & cpMode);  CalculateSizeButton->Visible = FAllowCalculateStats;  if (!FMultiple)  {    // when setting properties for one file only, allow undef state    // only when the input right explicitly requires it or    // when "recursive" is on (possible for directory only).    bool AllowUndef =      (FOrigProperties.Valid.Contains(vpRights) &&       FOrigProperties.Rights.AllowUndef) ||      (RecursiveCheck->Checked);    if (!AllowUndef)    {      // when disallowing undef state, make sure, all undef are turned into unset      RightsFrame->Rights = TRights(RightsFrame->Rights.NumberSet);    }    RightsFrame->AllowUndef = AllowUndef;  }  EnableControl(ChecksumSheet, ChecksumSupported());  EnableControl(ChecksumButton, ChecksumSheet->Enabled &&    !ChecksumAlgEdit->Text.IsEmpty());  ChecksumEdit->Visible = !ChecksumEdit->Text.IsEmpty();  ChecksumUnknownLabel->Visible = !ChecksumEdit->Visible;  DefaultButton(ChecksumButton, ChecksumAlgEdit->Focused());  DefaultButton(OkButton, !ChecksumAlgEdit->Focused());}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::FormCloseQuery(TObject * /*Sender*/,      bool & /*CanClose*/){  if (ModalResult == DefaultResult())  {    ExitActiveControl(this);  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::CalculateSizeButtonClick(      TObject * /*Sender*/){  DebugAssert(FOnCalculateSize != NULL);  bool DoClose = false;  Enabled = false;  try  {    __int64 Size;    TCalculateSizeStats Stats;    FOnCalculateSize(FFileList, Size, Stats, DoClose);    FStatsNotCalculated = false;    LoadStats(Size, Stats);  }  __finally  {    Enabled = true;    if (DoClose)    {      Close();    }  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::HelpButtonClick(TObject * /*Sender*/){  FormHelp(this);}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::ResetChecksum(){  ChecksumView->Items->Clear();  ChecksumEdit->Text = UnicodeString();  AutoSizeListColumnsWidth(ChecksumView);}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::CalculateChecksum(){  DebugAssert(FOnCalculateChecksum != NULL);  ResetChecksum();  FChecksumLoaded = true;  FAlgUsed = UnicodeString();  bool DoClose = false;  try  {    FOnCalculateChecksum(ChecksumAlgEdit->Text, FFileList, CalculatedChecksum, DoClose);  }  __finally  {    if (DoClose)    {      Close();    }  }  // If we successfully used the selected checksum, remember it (in normalized form)  if (!FAlgUsed.IsEmpty())  {    GUIConfiguration->ChecksumAlg = FAlgUsed;  }  AutoSizeListColumnsWidth(ChecksumView);}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::CalculatedChecksum(  const UnicodeString & FileName, const UnicodeString & Alg,  const UnicodeString & Hash){  if (FMultipleChecksum)  {    TListItem * Item = ChecksumView->Items->Add();    Item->Caption = FileName;    Item->SubItems->Add(Hash);    // optimization    int TopIndex = ListView_GetTopIndex(ChecksumView->Handle);    int Index = Item->Index;    if ((TopIndex <= Index) &&        (Index <= TopIndex + ChecksumView->VisibleRowCount))    {      AutoSizeListColumnsWidth(ChecksumView);    }  }  else  {    ChecksumEdit->Text = Hash;  }  FAlgUsed = Alg;  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::NeedChecksum(){  if (!FChecksumLoaded && ChecksumSupported())  {    CalculateChecksum();  }}//---------------------------------------------------------------------------bool __fastcall TPropertiesDialog::ChecksumSupported(){  return (FOnCalculateChecksum != NULL);}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::ChecksumButtonClick(TObject * /*Sender*/){  CalculateChecksum();}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::PageControlChange(TObject * /*Sender*/){  if (PageControl->ActivePage == ChecksumSheet)  {    NeedChecksum();  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::ChecksumAlgEditChange(TObject * /*Sender*/){  ResetChecksum();  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::CopyClick(TObject * Sender){  TInstantOperationVisualizer Visualizer;  TListView * ListView = dynamic_cast<TListView *>(GetPopupComponent(Sender));  DebugAssert(ListView != NULL);  int Count = 0;  UnicodeString SingleText;  UnicodeString Text;  TListItem * Item = ListView->GetNextItem(NULL, sdAll, TItemStates() << isSelected);  while (Item != NULL)  {    DebugAssert(Item->Selected);    SingleText = Item->SubItems->Strings[0];    Text += FORMAT(L"%s = %s\r\n", (Item->Caption, Item->SubItems->Strings[0]));    Count++;    Item = ListView->GetNextItem(Item, sdAll, TItemStates() << isSelected);  }  CopyToClipboard(Count == 1 ? SingleText : Text);}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::ChecksumViewContextPopup(  TObject * Sender, TPoint & MousePos, bool & Handled){  MenuPopup(Sender, MousePos, Handled);}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::ValidateRemoteToken(  const TRemoteToken & Orig, int Message, TComboBox * ComboBox,  const TRemoteTokenList * List){  if (!IsCancelButtonBeingClicked(this))  {    try    {      ComboBox->Text =        LoadRemoteToken(StoreRemoteToken(Orig, ComboBox->Text, Message, List));    }    catch(...)    {      ComboBox->SetFocus();      throw;    }  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::GroupComboBoxExit(TObject * Sender){  ValidateRemoteToken(FOrigProperties.Group, PROPERTIES_INVALID_GROUP,    dynamic_cast<TComboBox *>(Sender), FGroupList);}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::OwnerComboBoxExit(TObject * Sender){  ValidateRemoteToken(FOrigProperties.Owner, PROPERTIES_INVALID_OWNER,    dynamic_cast<TComboBox *>(Sender), FUserList);}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::FormShow(TObject * /*Sender*/){  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::CMDpiChanged(TMessage & Message){  TForm::Dispatch(&Message);  if (!FMultiple)  {    UpdateFileImage();  }}//---------------------------------------------------------------------------void __fastcall TPropertiesDialog::Dispatch(void * Message){  TMessage * M = reinterpret_cast<TMessage*>(Message);  if (M->Msg == CM_DPICHANGED)  {    CMDpiChanged(*M);  }  else  {    TForm::Dispatch(Message);  }}//---------------------------------------------------------------------------
 |