| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744 | //---------------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include <Common.h>#include "Authenticate.h"#include <VCLCommon.h>#include <TextsWin.h>#include <Terminal.h>#include <CoreMain.h>#include <PasTools.hpp>#include <CustomWinConfiguration.h>#include <Character.hpp>#include <Tools.h>//---------------------------------------------------------------------------#pragma package(smart_init)#pragma link "PasswordEdit"#pragma resource "*.dfm"//---------------------------------------------------------------------------__fastcall TAuthenticateForm::TAuthenticateForm(TComponent * Owner)  : TForm(Owner), FSessionData(NULL), FTerminal(NULL){}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::Init(TTerminal * Terminal){  FTerminal = Terminal;  FSessionData = Terminal->SessionData;  UseSystemSettings(this);  FShowAsModalStorage = NULL;  FFocusControl = NULL;  UseDesktopFont(LogView);  FAnimationPainted = false;  FShowNoActivate = false;  FPromptParent = InstructionsLabel->Parent;  FPromptLeft = InstructionsLabel->Left;  FPromptTop = InstructionsLabel->Top;  FPromptRight = FPromptParent->ClientWidth - InstructionsLabel->Width - FPromptLeft;  FPromptEditGap = PromptEdit1->Top - PromptLabel1->Top - PromptLabel1->Height;  FPromptsGap = PromptLabel2->Top - PromptEdit1->Top - PromptEdit1->Height;  ClientHeight = ScaleByTextHeight(this, 270);  FHorizontalLogPadding = ScaleByTextHeight(this, 4);  FVerticalLogPadding = ScaleByTextHeight(this, 3);  FLogTextFormat << tfNoPrefix << tfWordBreak << tfVerticalCenter;  FHintIndex = -1;}//---------------------------------------------------------------------------__fastcall TAuthenticateForm::~TAuthenticateForm(){  Application->CancelHint();  if (ReleaseAsModal(this, FShowAsModalStorage))  {    UnhookFormActivation(this);  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::ShowAsModal(){  if (IsApplicationMinimized())  {    FShowNoActivate = true;  }  // Do not call BringToFront when minimized, so that we do not have to use the same hack as in TMessageForm::SetZOrder  ::ShowAsModal(this, FShowAsModalStorage, !FShowNoActivate, true);  HookFormActivation(this);}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::CMShowingChanged(TMessage & Message){  if (Showing && FShowNoActivate)  {    ShowFormNoActivate(this);  }  else  {    TForm::Dispatch(&Message);  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::WMNCCreate(TWMNCCreate & Message){  // bypass TForm::WMNCCreate to prevent disabling "resize"  // (which is done for bsDialog, see comments in CreateParams)  DefaultHandler(&Message);}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::DoCancel(){  if (FOnCancel != NULL)  {    FOnCancel(this);  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::Dispatch(void * AMessage){  TMessage & Message = *reinterpret_cast<TMessage *>(AMessage);  if (Message.Msg == WM_NCCREATE)  {    WMNCCreate(*reinterpret_cast<TWMNCCreate *>(AMessage));  }  else if (Message.Msg == WM_CLOSE)  {    DoCancel();    TForm::Dispatch(AMessage);  }  else if (Message.Msg == WM_MANAGES_CAPTION)  {    // caption managed in TAuthenticateForm::AdjustControls()    Message.Result = 1;  }  else if (Message.Msg == CM_SHOWINGCHANGED)  {    CMShowingChanged(Message);  }  else  {    TForm::Dispatch(AMessage);  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::CreateParams(TCreateParams & Params){  TForm::CreateParams(Params);  // Allow resizing of the window, even if this is bsDialog.  // This makes it more close to bsSizeable, but bsSizeable cannot for some  // reason receive focus, if window is shown atop non-main window  // (like editor)  Params.Style = Params.Style | WS_THICKFRAME;}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::FormShow(TObject * /*Sender*/){  AdjustControls();  if (FFocusControl != NULL)  {    ActiveControl = FFocusControl;  }  UnicodeString AnimationName = FSessionData->IsSecure() ? L"ConnectingSecure" : L"ConnectingInsecure";  FFrameAnimation.Init(AnimationPaintBox, AnimationName);  FFrameAnimation.Start();}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::Log(const UnicodeString & Message, const UnicodeString & Additional){  // HACK  // The first call to Repaint from TFrameAnimation happens  // even before the form is showing. After it is shown it takes sometime too long  // before the animation is painted, so that the form is closed before the first frame even appers  if (!FAnimationPainted && Showing)  {    AnimationPaintBox->Repaint();    FAnimationPainted = true;  }  int Index = LogView->Items->Add(Message);  FHints.push_back(Additional);  MakeLogItemVisible(Index);  LogView->Repaint();}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::MakeLogItemVisible(int Index){  if (Index < LogView->TopIndex)  {    LogView->TopIndex = Index;  }  else  {    int TotalHeight = 0;    int Index2 = Index;    while ((Index2 >= 0) && TotalHeight < LogView->ClientHeight)    {      TotalHeight += LogItemHeight(Index2);      Index2--;    }    // Index2 is the last item above the first fully visible,    // were the Index on the very bottom.    if (LogView->TopIndex <= Index2 + 1)    {      LogView->TopIndex = Index2 + 2;    }  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::AdjustControls(){  UnicodeString ACaption;  if (FStatus.IsEmpty())  {    ACaption = FSessionData->SessionName;  }  else  {    ACaption = FStatus + TitleSeparator + FSessionData->SessionName;  }  Caption = FormatFormCaption(this, ACaption);}//---------------------------------------------------------------------------TLabel * __fastcall TAuthenticateForm::GenerateLabel(int Current, UnicodeString Caption){  TLabel * Result = new TUIStateAwareLabel(this);  Result->Parent = FPromptParent;  Result->Anchors = TAnchors() << akLeft << akTop << akRight;  Result->WordWrap = true;  Result->AutoSize = false;  Result->Top = Current;  Result->Left = FPromptLeft;  Result->Caption = Caption;  int Width = FPromptParent->ClientWidth - FPromptLeft - FPromptRight;  Result->Width = Width;  Result->AutoSize = true;  Result->AutoSize = false;  Result->Width = Width;  return Result;}//---------------------------------------------------------------------------TCustomEdit * __fastcall TAuthenticateForm::GenerateEdit(int Current, bool Echo){  TCustomEdit * Result = (Echo ? static_cast<TCustomEdit *>(new TEdit(this)) :    static_cast<TCustomEdit *>(new TPasswordEdit(this)));  Result->Parent = FPromptParent;  Result->Anchors = TAnchors() << akLeft << akTop << akRight;  Result->Top = Current;  Result->Left = FPromptLeft;  Result->Width = FPromptParent->ClientWidth - FPromptLeft - FPromptRight;  return Result;}//---------------------------------------------------------------------------void TAuthenticateForm::ExternalLabel(TLabel * Label){  Label->OnContextPopup = LabelContextPopup;  Label->PopupMenu = LabelPopupMenu;  UnicodeString Url;  if (ExtractUrl(Label->Caption, Url))  {    Label->Cursor = crHandPoint;    Label->OnClick = LinkClick;  }}//---------------------------------------------------------------------------TList * __fastcall TAuthenticateForm::GeneratePrompt(  TPromptKind Kind, const UnicodeString & Instructions, TStrings * Prompts){  while (FPromptParent->ControlCount > 0)  {    delete FPromptParent->Controls[0];  }  TList * Result = new TList;  int Current = FPromptTop;  if (!Instructions.IsEmpty())  {    TLabel * Label = GenerateLabel(Current, Instructions);    ExternalLabel(Label);    Current += Label->Height + FPromptsGap;  }  for (int Index = 0; Index < Prompts->Count; Index++)  {    if (Index > 0)    {      Current += FPromptEditGap;    }    TLabel * Label = GenerateLabel(Current, Prompts->Strings[Index]);    if (Kind == pkKeybInteractive)    {      ExternalLabel(Label);    }    Current += Label->Height + FPromptEditGap;    bool Echo = FLAGSET(int(Prompts->Objects[Index]), pupEcho);    TCustomEdit * Edit = GenerateEdit(Current, Echo);    Result->Add(Edit);    Label->FocusControl = Edit;    Current += Edit->Height;  }  FPromptParent->ClientHeight = Current;  return Result;}//---------------------------------------------------------------------------bool __fastcall TAuthenticateForm::PromptUser(TPromptKind Kind, UnicodeString Name,  UnicodeString Instructions, TStrings * Prompts, TStrings * Results, bool ForceLog,  bool StoredCredentialsTried){  bool Result;  TList * Edits = GeneratePrompt(Kind, Instructions, Prompts);  try  {    bool ShowSessionRememberPasswordPanel = false;    bool ShowSavePasswordPanel = false;    TSessionData * Data = NULL;    if (IsPasswordPrompt(Kind, Prompts) && StoredCredentialsTried)    {      Data = StoredSessions->FindSame(FSessionData);      ShowSavePasswordPanel = (Data != NULL) && !Data->Password.IsEmpty();    }    // do not offer to remember password,    // if we are offering to save the password to stored site    if (!ShowSavePasswordPanel &&        (Prompts->Count == 1) &&        FLAGSET(int(Prompts->Objects[0]), pupRemember) &&        DebugAlwaysTrue(IsPasswordOrPassphrasePrompt(Kind, Prompts)))    {      ShowSessionRememberPasswordPanel = true;    }    SavePasswordCheck->Checked = false;    SavePasswordPanel->Visible = ShowSavePasswordPanel;    SessionRememberPasswordCheck->Checked = false;    SessionRememberPasswordPanel->Visible = ShowSessionRememberPasswordPanel;    if (PasswordPanel->AutoSize)    {      PasswordPanel->AutoSize = false;      PasswordPanel->AutoSize = true;    }    PasswordPanel->Realign();    DebugAssert(Results->Count == Edits->Count);    for (int Index = 0; Index < Edits->Count; Index++)    {      TCustomEdit * Edit = reinterpret_cast<TCustomEdit *>(Edits->Items[Index]);      Edit->Text = Results->Strings[Index];    }    Result = Execute(Name, PasswordPanel,      ((Edits->Count > 0) ?         reinterpret_cast<TWinControl *>(Edits->Items[0]) :         static_cast<TWinControl *>(PasswordOKButton)),      PasswordOKButton, PasswordCancelButton, true, false, ForceLog);    if (Result)    {      for (int Index = 0; Index < Edits->Count; Index++)      {        TCustomEdit * Edit = reinterpret_cast<TCustomEdit *>(Edits->Items[Index]);        Results->Strings[Index] = Edit->Text;        Prompts->Objects[Index] = (TObject *)          ((int(Prompts->Objects[Index]) & ~pupRemember) |           FLAGMASK(((Index == 0) && SessionRememberPasswordCheck->Checked), pupRemember));      }      if (SavePasswordCheck->Checked)      {        DebugAssert(Data != NULL);        DebugAssert(Results->Count >= 1);        FSessionData->Password = Results->Strings[0];        Data->Password = Results->Strings[0];        // modified only, explicit        StoredSessions->Save(false, true);      }    }  }  __finally  {    delete Edits;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::UpdateBannerFont(){  if (BannerMonospacedFontAction->Checked)  {    BannerMemo->Font->Name = CustomWinConfiguration->DefaultFixedWidthFontName;    BannerMemo->Font->Size = CustomWinConfiguration->DefaultFixedWidthFontSize;  }  else  {    BannerMemo->ParentFont = true;  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::Banner(const UnicodeString & Banner,  bool & NeverShowAgain, int Options, unsigned int & Params){  BannerMemo->Lines->Text = Banner;  NeverShowAgainCheck->Visible = FLAGCLEAR(Options, boDisableNeverShowAgain);  NeverShowAgainCheck->Checked = NeverShowAgain;  BannerMonospacedFontAction->Checked = FLAGSET(Params, bpMonospacedFont);  UpdateBannerFont();  bool Result = Execute(LoadStr(AUTHENTICATION_BANNER), BannerPanel, BannerCloseButton,    BannerCloseButton, BannerCloseButton, false, true, false);  if (Result)  {    NeverShowAgain = NeverShowAgainCheck->Checked;    Params =      (Params & ~bpMonospacedFont) |      FLAGMASK(BannerMonospacedFontAction->Checked, bpMonospacedFont);  }}//---------------------------------------------------------------------------bool __fastcall TAuthenticateForm::Execute(UnicodeString Status, TPanel * Panel,  TWinControl * FocusControl, TButton * DefaultButton, TButton * CancelButton,  bool FixHeight, bool Zoom, bool ForceLog){  TModalResult DefaultButtonResult;  TAlign Align = Panel->Align;  try  {    // If not visible yet, the animation was not even initialized yet    if (Visible)    {      FFrameAnimation.Stop();    }    DebugAssert(FStatus.IsEmpty());    FStatus = Status;    DefaultButton->Default = true;    CancelButton->Cancel = true;    DefaultButtonResult = DefaultResult(this);    if (Zoom)    {      Panel->Align = alClient;    }    if (ForceLog || Visible)    {      if (ClientHeight < Panel->Height)      {        ClientHeight = Panel->Height;      }      // Panel being hidden gets not realigned automatically, even if it      // has Align property set      Panel->Top = ClientHeight - Panel->Height;      Panel->Show();      TCursor PrevCursor = Screen->Cursor;      try      {        if (Zoom)        {          TopPanel->Hide();        }        else        {          if (LogView->Items->Count > 0)          {            // To avoid the scrolling effect when setting TopIndex            LogView->Items->BeginUpdate();            try            {              MakeLogItemVisible(LogView->Items->Count - 1);            }            __finally            {              LogView->Items->EndUpdate();              RedrawLog();            }          }        }        Screen->Cursor = crDefault;        if (!Visible)        {          DebugAssert(ForceLog);          ShowAsModal();        }        ActiveControl = FocusControl;        ModalResult = mrNone;        AdjustControls();        do        {          Application->HandleMessage();        }        while (!Application->Terminated && (ModalResult == mrNone));      }      __finally      {        Panel->Hide();        Screen->Cursor = PrevCursor;        if (Zoom)        {          TopPanel->Show();        }        Repaint();      }    }    else    {      int PrevHeight = ClientHeight;      int PrevMinHeight = Constraints->MinHeight;      int PrevMaxHeight = Constraints->MaxHeight;      try      {        Constraints->MinHeight = 0;        ClientHeight = Panel->Height;        if (FixHeight)        {          Constraints->MinHeight = Height;          Constraints->MaxHeight = Height;        }        TopPanel->Hide();        Panel->Show();        FFocusControl = FocusControl;        ShowModal();      }      __finally      {        FFocusControl = NULL;        ClientHeight = PrevHeight;        Constraints->MinHeight = PrevMinHeight;        Constraints->MaxHeight = PrevMaxHeight;        Panel->Hide();        TopPanel->Show();      }    }  }  __finally  {    Panel->Align = Align;    DefaultButton->Default = false;    CancelButton->Cancel = false;    FStatus = L"";    AdjustControls();    FFrameAnimation.Start();  }  bool Result = (ModalResult == DefaultButtonResult);  if (!Result)  {    // This is not nice as it may ultimately route to    // TTerminalThread::Cancel() and throw fatal exception,    // what actually means that any PromptUser call during authentication never    // return false and their fall back/alternative code never occurs.    // It probably needs fixing.    DoCancel();  }  return Result;}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::HelpButtonClick(TObject * /*Sender*/){  FormHelp(this);}//---------------------------------------------------------------------------int __fastcall TAuthenticateForm::LogItemHeight(int Index){  UnicodeString S = LogView->Items->Strings[Index];  TRect TextRect(0, 0, LogView->ClientWidth - (2 * FHorizontalLogPadding), 0);  LogView->Canvas->Font = LogView->Font;  LogView->Canvas->TextRect(TextRect, S, FLogTextFormat + (TTextFormat() << tfCalcRect));  int Result = TextRect.Height() + (2 * FVerticalLogPadding);  return Result;}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::LogViewMeasureItem(TWinControl * /*Control*/, int Index,  int & Height){  Height = LogItemHeight(Index);}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::LogViewDrawItem(TWinControl * /*Control*/, int Index,  TRect & Rect, TOwnerDrawState /*State*/){  // Reset back to base colors. We do not want to render selection.  // + At initial phases the canvas font will not yet reflect the desktop font.  LogView->Canvas->Font = LogView->Font;  LogView->Canvas->Brush->Color = LogView->Color;  LogView->Canvas->FillRect(Rect);  // tfVerticalCenter does not seem to center the text,  // so we need to deflate the vertical size too  Rect.Inflate(-FHorizontalLogPadding,  -FVerticalLogPadding);  UnicodeString S = LogView->Items->Strings[Index];  LogView->Canvas->TextRect(Rect, S, FLogTextFormat);}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::RedrawLog(){  // Redraw including the scrollbar (RDW_FRAME)  RedrawWindow(LogView->Handle, NULL, NULL, RDW_FRAME | RDW_INVALIDATE | RDW_UPDATENOW);}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::FormResize(TObject * /*Sender*/){  if (LogView->Showing)  {    // Mainly to avoid the scrolling effect when setting TopIndex    LogView->Items->BeginUpdate();    try    {      // Rebuild the list view to force Windows to resend WM_MEASUREITEM      int ItemIndex = LogView->ItemIndex;      int TopIndex = LogView->TopIndex;      std::unique_ptr<TStringList> Items(new TStringList);      Items->Assign(LogView->Items);      LogView->Items->Assign(Items.get());      LogView->ItemIndex = ItemIndex;      LogView->TopIndex = TopIndex;    }    __finally    {      LogView->Items->EndUpdate();      RedrawLog();    }  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::ChangeScale(int M, int D){  TForm::ChangeScale(M, D);  // Recreate the list to re-measure the items according to the new font  if (DebugAlwaysTrue(LogView->HandleAllocated()) &&      (LogView->Items->Count > 0))  {    std::unique_ptr<TStrings> Items(new TStringList());    Items->AddStrings(LogView->Items);    LogView->Items->Clear();    LogView->Items->AddStrings(Items.get());    MakeLogItemVisible(LogView->Items->Count - 1);  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::BannerMemoContextPopup(TObject * Sender, TPoint & MousePos, bool & Handled){  MenuPopup(Sender, MousePos, Handled);}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::BannerMonospacedFontActionExecute(TObject * /*Sender*/){  BannerMonospacedFontAction->Checked = !BannerMonospacedFontAction->Checked;  UpdateBannerFont();}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::LogViewMouseMove(TObject *, TShiftState, int X, int Y){  int Index = LogView->ItemAtPos(TPoint(X, Y), true);  if (FHintIndex != Index)  {    Application->CancelHint();    FHintIndex = Index;    if (Index >= 0)    {      LogView->Hint = FHints[Index];    }  }}//---------------------------------------------------------------------------bool TAuthenticateForm::ExtractUrl(const UnicodeString & Text, UnicodeString & Url){  bool Result = false;  UnicodeString Prefix = HttpsProtocol + ProtocolSeparator;  int P = Text.Pos(Prefix);  if (P == 0)  {    Prefix = HttpProtocol + ProtocolSeparator;    P = Text.Pos(Prefix);  }  if (P > 0)  {    Url = Text.SubString(P, Text.Length() - P + 1);    P = 1;    while (P <= Url.Length())    {      #pragma warn -dpr      if (TCharacter::IsWhiteSpace(Url[P]))      #pragma warn .dpr      {        Url.SetLength(P - 1);      }      P++;    }    // At least one letter/digit after the prefix    if ((Url.Length() > Prefix.Length()) &&        (::IsLetter(Url[Prefix.Length() + 1]) || ::IsDigit(Url[Prefix.Length() + 1])))    {      Result = true;    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::LabelContextPopup(TObject * Sender, const TPoint & MousePos, bool & Handled){  TLabel * Label = dynamic_cast<TLabel *>(Sender);  UnicodeString Url;  LabelOpenLinkAction2->Visible = ExtractUrl(Label->Caption, Url);  FContextLabel = Label;  MenuPopup(Sender, MousePos, Handled);}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::LabelCopyActionExecute(TObject *){  if (DebugAlwaysTrue(FContextLabel != NULL))  {    TInstantOperationVisualizer Visualizer;    CopyToClipboard(FContextLabel->Caption);  }}//---------------------------------------------------------------------------void TAuthenticateForm::LabelOpen(TLabel * Label){  UnicodeString Url;  if (ExtractUrl(Label->Caption, Url))  {    OpenBrowser(Url);  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::LabelOpenLinkAction2Execute(TObject *){  if (DebugAlwaysTrue(FContextLabel != NULL))  {    LabelOpen(FContextLabel);  }}//---------------------------------------------------------------------------void __fastcall TAuthenticateForm::LinkClick(TObject * Sender){  LabelOpen(DebugNotNull(dynamic_cast<TLabel *>(Sender)));}//---------------------------------------------------------------------------
 |