12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256 |
- //---------------------------------------------------------------------------
- #include <vcl.h>
- #pragma hdrstop
- #include <Consts.hpp>
- #include <GUITools.h>
- #include <Common.h>
- #include <VCLCommon.h>
- #include <CoreMain.h>
- #include <WinInterface.h>
- #include <Tools.h>
- #include <TextsWin.h>
- #include <TextsCore.h>
- #include <Vcl.Imaging.pngimage.hpp>
- #include <StrUtils.hpp>
- #include <PasTools.hpp>
- #include <Math.hpp>
- #include <vssym32.h>
- #include <WebBrowserEx.hpp>
- #include <Setup.h>
- //---------------------------------------------------------------------------
- #pragma package(smart_init)
- //---------------------------------------------------------------------------
- class TMessageButton : public TButton
- {
- public:
- __fastcall TMessageButton(TComponent * Owner);
- protected:
- virtual void __fastcall Dispatch(void * Message);
- private:
- void __fastcall WMGetDlgCode(TWMGetDlgCode & Message);
- };
- //---------------------------------------------------------------------------
- __fastcall TMessageButton::TMessageButton(TComponent * Owner) :
- TButton(Owner)
- {
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageButton::Dispatch(void * Message)
- {
- TMessage * M = reinterpret_cast<TMessage*>(Message);
- if (M->Msg == WM_GETDLGCODE)
- {
- WMGetDlgCode(*((TWMGetDlgCode *)Message));
- }
- else
- {
- TButton::Dispatch(Message);
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageButton::WMGetDlgCode(TWMGetDlgCode & Message)
- {
- TButton::Dispatch(&Message);
- // WORKAROUND
- // Windows default handler returns DLGC_WANTARROWS for split buttons,
- // what prevent left/right keys from being used for focusing next/previous buttons/controls.
- // Overrwide that. Though note that we need to pass the up/down keys back to button
- // to allow drop down, see TMessageForm::CMDialogKey
- Message.Result = Message.Result & ~DLGC_WANTARROWS;
- }
- //---------------------------------------------------------------------------
- class TMessageForm : public TForm
- {
- public:
- static TForm * __fastcall Create(const UnicodeString & Msg, TStrings * MoreMessages,
- TMsgDlgType DlgType, unsigned int Answers,
- const TQueryButtonAlias * Aliases, unsigned int AliasesCount,
- unsigned int TimeoutAnswer, TButton ** TimeoutButton, const UnicodeString & ImageName,
- const UnicodeString & NeverAskAgainCaption, const UnicodeString & MoreMessagesUrl,
- TSize MoreMessagesSize);
- virtual int __fastcall ShowModal();
- protected:
- __fastcall TMessageForm(TComponent * AOwner);
- virtual __fastcall ~TMessageForm();
- DYNAMIC void __fastcall KeyDown(Word & Key, TShiftState Shift);
- DYNAMIC void __fastcall KeyUp(Word & Key, TShiftState Shift);
- UnicodeString __fastcall GetFormText();
- UnicodeString __fastcall GetReportText();
- UnicodeString __fastcall NormalizeNewLines(UnicodeString Text);
- virtual void __fastcall CreateParams(TCreateParams & Params);
- DYNAMIC void __fastcall DoShow();
- virtual void __fastcall Dispatch(void * Message);
- void __fastcall MenuItemClick(TObject * Sender);
- void __fastcall ButtonDropDownClick(TObject * Sender);
- void __fastcall UpdateForShiftStateTimer(TObject * Sender);
- DYNAMIC void __fastcall SetZOrder(bool TopMost);
- private:
- typedef std::map<unsigned int, TButton *> TAnswerButtons;
- UnicodeString MessageText;
- TMemo * MessageMemo;
- TPanel * MessageBrowserPanel;
- TWebBrowserEx * MessageBrowser;
- UnicodeString MessageBrowserUrl;
- TShiftState FShiftState;
- TTimer * FUpdateForShiftStateTimer;
- TForm * FDummyForm;
- bool FShowNoActivate;
- void __fastcall HelpButtonClick(TObject * Sender);
- void __fastcall ReportButtonClick(TObject * Sender);
- void __fastcall CMDialogKey(TWMKeyDown & Message);
- void __fastcall CMShowingChanged(TMessage & Message);
- void __fastcall UpdateForShiftState();
- TButton * __fastcall CreateButton(
- UnicodeString Name, UnicodeString Caption, unsigned int Answer,
- TNotifyEvent OnClick, bool IsTimeoutButton,
- int GroupWith, TShiftState GrouppedShiftState,
- TAnswerButtons & AnswerButtons, bool HasMoreMessages, int & ButtonWidths);
- bool __fastcall ApplicationHook(TMessage & Message);
- };
- //---------------------------------------------------------------------------
- __fastcall TMessageForm::TMessageForm(TComponent * AOwner) : TForm(AOwner, 0)
- {
- FShowNoActivate = false;
- MessageMemo = NULL;
- MessageBrowserPanel = NULL;
- MessageBrowser = NULL;
- FUpdateForShiftStateTimer = NULL;
- Position = poOwnerFormCenter;
- UseSystemSettingsPre(this);
- FDummyForm = new TForm(this);
- UseSystemSettings(FDummyForm);
- }
- //---------------------------------------------------------------------------
- __fastcall TMessageForm::~TMessageForm()
- {
- SAFE_DESTROY(FDummyForm);
- SAFE_DESTROY(FUpdateForShiftStateTimer);
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::HelpButtonClick(TObject * /*Sender*/)
- {
- if (HelpKeyword != HELP_NONE)
- {
- FormHelp(this);
- }
- else
- {
- MessageWithNoHelp(GetReportText());
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::ReportButtonClick(TObject * /*Sender*/)
- {
- UnicodeString Url =
- FMTLOAD(ERROR_REPORT_URL,
- (EncodeUrlString(GetReportText()), Configuration->ProductVersion,
- IntToHex(__int64(GUIConfiguration->Locale), 4)));
- OpenBrowser(Url);
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::UpdateForShiftState()
- {
- TShiftState ShiftState =
- KeyboardStateToShiftState() *
- (TShiftState() << ssShift << ssCtrl << ssAlt);
- if (FShiftState != ShiftState)
- {
- FShiftState = ShiftState;
- for (int ComponentIndex = 0; ComponentIndex < ComponentCount - 1; ComponentIndex++)
- {
- TButton * Button = dynamic_cast<TButton*>(Components[ComponentIndex]);
- if ((Button != NULL) && (Button->DropDownMenu != NULL))
- {
- TMenuItem * MenuItems = Button->DropDownMenu->Items;
- for (int ItemIndex = 0; ItemIndex < MenuItems->Count; ItemIndex++)
- {
- TMenuItem * Item = MenuItems->Items[ItemIndex];
- TShiftState GrouppedShiftState(Item->Tag >> 16);
- if (Item->Enabled &&
- ((ShiftState.Empty() && Item->Default) ||
- (!ShiftState.Empty() && (ShiftState == GrouppedShiftState))))
- {
- Button->Caption = CopyToChar(Item->Caption, L'\t', false);
- Button->ModalResult = Item->Tag & 0xFFFF;
- assert(Button->OnClick == NULL);
- assert(Item->OnClick == MenuItemClick);
- break;
- }
- }
- }
- }
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::KeyUp(Word & Key, TShiftState Shift)
- {
- UpdateForShiftState();
- TForm::KeyUp(Key, Shift);
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::KeyDown(Word & Key, TShiftState Shift)
- {
- if (Shift.Contains(ssCtrl) && (Key == L'C'))
- {
- TInstantOperationVisualizer Visualizer;
- CopyToClipboard(GetFormText());
- }
- else
- {
- if (!Shift.Contains(ssCtrl))
- {
- for (int ComponentIndex = 0; ComponentIndex < ComponentCount - 1; ComponentIndex++)
- {
- TButton * Button = dynamic_cast<TButton*>(Components[ComponentIndex]);
- if ((Button != NULL) && (Button->DropDownMenu != NULL))
- {
- TMenuItem * MenuItems = Button->DropDownMenu->Items;
- for (int ItemIndex = 0; ItemIndex < MenuItems->Count; ItemIndex++)
- {
- TMenuItem * Item = MenuItems->Items[ItemIndex];
- if (IsAccel(Key, MenuItems->Items[ItemIndex]->Caption))
- {
- Item->OnClick(Item);
- Key = 0;
- break;
- }
- }
- }
- if (Key == 0)
- {
- break;
- }
- }
- }
- UpdateForShiftState();
- TForm::KeyDown(Key, Shift);
- }
- }
- //---------------------------------------------------------------------------
- UnicodeString __fastcall TMessageForm::NormalizeNewLines(UnicodeString Text)
- {
- Text = ReplaceStr(Text, L"\r", L"");
- Text = ReplaceStr(Text, L"\n", L"\r\n");
- return Text;
- }
- //---------------------------------------------------------------------------
- UnicodeString __fastcall TMessageForm::GetFormText()
- {
- UnicodeString DividerLine, ButtonCaptions;
- DividerLine = UnicodeString::StringOfChar(L'-', 27) + sLineBreak;
- for (int i = 0; i < ComponentCount - 1; i++)
- {
- if (dynamic_cast<TButton*>(Components[i]) != NULL)
- {
- ButtonCaptions += dynamic_cast<TButton*>(Components[i])->Caption +
- UnicodeString::StringOfChar(L' ', 3);
- }
- }
- ButtonCaptions = ReplaceStr(ButtonCaptions, L"&", L"");
- UnicodeString MoreMessages;
- if (MessageMemo != NULL)
- {
- MoreMessages = MessageMemo->Text + DividerLine;
- }
- else if (MessageBrowser != NULL)
- {
- MessageBrowser->SelectAll();
- MessageBrowser->CopyToClipBoard();
- if (TextFromClipboard(MoreMessages, true))
- {
- if (!EndsStr(sLineBreak, MoreMessages))
- {
- MoreMessages += sLineBreak;
- }
- MoreMessages += DividerLine;
- }
- // http://www.ssicom.org/js/x277333.htm
- MessageBrowser->DoCommand(L"UNSELECT");
- }
- UnicodeString MessageCaption = NormalizeNewLines(MessageText);
- UnicodeString Result = FORMAT(L"%s%s%s%s%s%s%s%s%s%s%s", (DividerLine, Caption, sLineBreak,
- DividerLine, MessageCaption, sLineBreak, DividerLine, MoreMessages,
- ButtonCaptions, sLineBreak, DividerLine));
- return Result;
- }
- //---------------------------------------------------------------------------
- UnicodeString __fastcall TMessageForm::GetReportText()
- {
- UnicodeString Text = MessageText;
- Text = Text.TrimRight();
- if (MessageMemo != NULL)
- {
- Text += L"\n\n" + MessageMemo->Text;
- }
- // Currently we use browser for updates box only and it has help context,
- // and does not have Report button, so we cannot get here.
- assert(MessageBrowser == NULL);
- Text = NormalizeNewLines(Text);
- UnicodeString ReportErrorText = NormalizeNewLines(FMTLOAD(REPORT_ERROR, (L"")));
- Text = ReplaceStr(Text, ReportErrorText, L"");
- Text = Trim(Text);
- return Text;
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::CMDialogKey(TWMKeyDown & Message)
- {
- // this gets used in WinInterface.cpp SetTimeoutEvents
- if (OnKeyDown != NULL)
- {
- OnKeyDown(this, Message.CharCode, KeyDataToShiftState(Message.KeyData));
- }
- if (Message.CharCode == VK_MENU)
- {
- bool AnyButtonWithGrouppedCommandsWithShiftState = false;
- for (int ComponentIndex = 0; ComponentIndex < ComponentCount - 1; ComponentIndex++)
- {
- TButton * Button = dynamic_cast<TButton*>(Components[ComponentIndex]);
- if ((Button != NULL) && (Button->DropDownMenu != NULL))
- {
- // we should check if there are any commands with shift state,
- // but it's bit overkill
- AnyButtonWithGrouppedCommandsWithShiftState = true;
- break;
- }
- }
- // this is to make Alt only alter button meaning (if there is any
- // alternable button) and not popup system menu
- if (AnyButtonWithGrouppedCommandsWithShiftState)
- {
- Message.Result = 1;
- UpdateForShiftState();
- }
- else
- {
- TForm::Dispatch(&Message);
- }
- }
- else if ((Message.CharCode == VK_UP) || (Message.CharCode == VK_DOWN))
- {
- // WORKAROUND
- // noop to make up/down be passed back to button to allow drop down,
- // see TMessageButton::WMGetDlgCode
- }
- else
- {
- TForm::Dispatch(&Message);
- }
- }
- //---------------------------------------------------------------------------
- int __fastcall TMessageForm::ShowModal()
- {
- if (IsApplicationMinimized())
- {
- FShowNoActivate = true;
- }
- int Result = TForm::ShowModal();
- Application->UnhookMainWindow(ApplicationHook);
- return Result;
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::SetZOrder(bool TopMost)
- {
- // WORKAROUND: If application is minimized,
- // swallow call to BringToFront() from TForm::ShowModal()
- if (FShowNoActivate && TopMost)
- {
- // noop
- }
- else
- {
- TForm::SetZOrder(TopMost);
- }
- }
- //---------------------------------------------------------------------------
- bool __fastcall TMessageForm::ApplicationHook(TMessage & Message)
- {
- bool Result = false;
- // If application is restored, message box is not activated, do it manually.
- // We cannot do this from TApplication::OnActivate because
- // TApplication.WndProc resets focus to the last active window afterwards.
- // So we override CM_ACTIVATE implementation here completelly.
- if ((Message.Msg == CM_ACTIVATE) && FShowNoActivate)
- {
- ::SetFocus(Handle);
- // VCLCOPY
- if (Application->OnActivate != NULL)
- {
- Application->OnActivate(Application);
- }
- Result = true;
- }
- return Result;
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::CMShowingChanged(TMessage & Message)
- {
- if (Showing && FShowNoActivate)
- {
- // With is same as SendToBack, except for added SWP_NOACTIVATE (VCLCOPY)
- SetWindowPos(WindowHandle, HWND_BOTTOM, 0, 0, 0, 0,
- SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
- // This replaces TCustomForm::CMShowingChanged()
- // which calls ShowWindow(Handle, SW_SHOWNORMAL).
- ShowWindow(Handle, SW_SHOWNOACTIVATE);
- // - so we have to call DoShow explicitly.
- DoShow();
- // - also we skip applying TForm::Position (VCLCOPY)
- if (ALWAYS_TRUE(Position == poOwnerFormCenter))
- {
- TCustomForm * CenterForm = Application->MainForm;
- TCustomForm * OwnerForm = dynamic_cast<TCustomForm *>(Owner);
- if (OwnerForm != NULL)
- {
- CenterForm = OwnerForm;
- }
- int X, Y;
- if ((CenterForm != NULL) && (CenterForm != this))
- {
- TRect Bounds = CenterForm->BoundsRect;
- X = ((Bounds.Width() - Width) / 2) + CenterForm->Left;
- Y = ((Bounds.Height() - Height) / 2) + CenterForm->Top;
- }
- else
- {
- X = (Screen->Width - Width) / 2;
- Y = (Screen->Height - Height) / 2;
- }
- if (X < Screen->DesktopLeft)
- {
- X = Screen->DesktopLeft;
- }
- if (Y < Screen->DesktopTop)
- {
- Y = Screen->DesktopTop;
- }
- SetBounds(X, Y, Width, Height);
- // We cannot call SetWindowToMonitor().
- // We cannot set FPosition = poDesigned, so worlarea-checking code
- // in DoFormWindowProc is not triggered
- }
- // wait for application to be activate to activate ourself
- Application->HookMainWindow(ApplicationHook);
- }
- else
- {
- TForm::Dispatch(&Message);
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::Dispatch(void * Message)
- {
- TMessage * M = reinterpret_cast<TMessage*>(Message);
- if (M->Msg == CM_DIALOGKEY)
- {
- CMDialogKey(*((TWMKeyDown *)Message));
- }
- else if (M->Msg == CM_SHOWINGCHANGED)
- {
- CMShowingChanged(*M);
- }
- else
- {
- TForm::Dispatch(Message);
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::CreateParams(TCreateParams & Params)
- {
- TForm::CreateParams(Params);
- if ((Screen != NULL) && (Screen->ActiveForm != NULL) &&
- Screen->ActiveForm->HandleAllocated())
- {
- Params.WndParent = Screen->ActiveForm->Handle;
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::DoShow()
- {
- UseSystemSettingsPost(this);
- TForm::DoShow();
- if (!MessageBrowserUrl.IsEmpty() &&
- // Guard against repeated calls to DoOpen()
- (MessageBrowser == NULL))
- {
- // Web Browser component does not seem to work,
- // when created before any window is shown.
- // I.e. when the message dialog is the first window (like when /update is used).
- // So we have to delay its creation until at least the dialog box is shown.
- MessageBrowser = CreateBrowserViewer(MessageBrowserPanel, LoadStr(MESSAGE_LOADING));
- MessageBrowser->SendToBack();
- MessageBrowser->Navigate(MessageBrowserUrl.c_str());
- }
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::MenuItemClick(TObject * Sender)
- {
- TMenuItem * Item = NOT_NULL(dynamic_cast<TMenuItem *>(Sender));
- ModalResult = (Item->Tag & 0xFFFF);
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::UpdateForShiftStateTimer(TObject * /*Sender*/)
- {
- // this is needed to reflect shift state, even when we do not have a keyboard
- // focus, what happens when drop down menu is popped up
- UpdateForShiftState();
- }
- //---------------------------------------------------------------------------
- void __fastcall TMessageForm::ButtonDropDownClick(TObject * /*Sender*/)
- {
- // as optimization, do not waste time running timer, unless
- // user pops up drop down menu. we do not have a way to stop timer, once
- // it closes, but functionaly it does not matter
- if (FUpdateForShiftStateTimer == NULL)
- {
- FUpdateForShiftStateTimer = new TTimer(this);
- FUpdateForShiftStateTimer->Interval = 50;
- FUpdateForShiftStateTimer->OnTimer = UpdateForShiftStateTimer;
- }
- }
- //---------------------------------------------------------------------------
- const ResourceString * Captions[] = { &_SMsgDlgWarning, &_SMsgDlgError, &_SMsgDlgInformation,
- &_SMsgDlgConfirm, NULL };
- const wchar_t * IconIDs[] = { IDI_EXCLAMATION, IDI_HAND, IDI_ASTERISK,
- IDI_QUESTION, NULL };
- const int mcHorzMargin = 10;
- const int mcVertMargin = 13;
- const int mcHorzSpacing = 12;
- const int mcButtonVertMargin = 7;
- const int mcButtonSpacing = 5;
- // includes mcVertMargin
- const int mcMoreMessageHeight = 86;
- // approximately what Windows Vista task dialogs use,
- // actually they probably has fixed width
- const int mcMaxDialogWidth = 340;
- const int mcMinDialogWidth = 310;
- const int mcMinDialogwithMoreMessagesWidth = 400;
- //---------------------------------------------------------------------------
- static UnicodeString __fastcall GetKeyNameStr(int Key)
- {
- wchar_t Buf[MAX_PATH];
- LONG VirtualKey = MapVirtualKey(Key, MAPVK_VK_TO_VSC);
- VirtualKey <<= 16;
- if (GetKeyNameText(VirtualKey, Buf, LENOF(Buf)) > 0)
- {
- NULL_TERMINATE(Buf);
- }
- else
- {
- Buf[0] = L'\0';
- }
- return Buf;
- }
- //---------------------------------------------------------------------------
- TButton * __fastcall TMessageForm::CreateButton(
- UnicodeString Name, UnicodeString Caption, unsigned int Answer,
- TNotifyEvent OnClick, bool IsTimeoutButton,
- int GroupWith, TShiftState GrouppedShiftState,
- TAnswerButtons & AnswerButtons, bool HasMoreMessages, int & ButtonWidths)
- {
- UnicodeString MeasureCaption = Caption;
- if (IsTimeoutButton)
- {
- MeasureCaption = FMTLOAD(TIMEOUT_BUTTON, (MeasureCaption, 99));
- }
- TRect TextRect;
- DrawText(Canvas->Handle,
- UnicodeString(MeasureCaption).c_str(), -1,
- &TextRect, DT_CALCRECT | DT_LEFT | DT_SINGLELINE |
- DrawTextBiDiModeFlagsReadingOnly());
- int CurButtonWidth = TextRect.Right - TextRect.Left + ScaleByTextHeightRunTime(this, 16);
- TButton * Button = NULL;
- if (SupportsSplitButton() &&
- (GroupWith >= 0) &&
- ALWAYS_TRUE(AnswerButtons.find(GroupWith) != AnswerButtons.end()))
- {
- TButton * GroupWithButton = AnswerButtons[GroupWith];
- if (GroupWithButton->DropDownMenu == NULL)
- {
- GroupWithButton->Style = TCustomButton::bsSplitButton;
- GroupWithButton->DropDownMenu = new TPopupMenu(this);
- // cannot handle subitems with shift state,
- // if the button has its own handler
- // (though it may not be the case still here)
- assert(GroupWithButton->OnClick == NULL);
- TMenuItem * Item = new TMenuItem(GroupWithButton->DropDownMenu);
- GroupWithButton->DropDownMenu->Items->Add(Item);
- GroupWithButton->OnDropDownClick = ButtonDropDownClick;
- Item->Caption = GroupWithButton->Caption;
- Item->OnClick = MenuItemClick;
- assert(GroupWithButton->ModalResult <= 0xFFFF);
- Item->Tag = GroupWithButton->ModalResult;
- Item->Default = true;
- }
- TMenuItem * Item = new TMenuItem(GroupWithButton->DropDownMenu);
- GroupWithButton->DropDownMenu->Items->Add(Item);
- // See ShortCutToText in Vcl.Menus.pas
- if (GrouppedShiftState == (TShiftState() << ssAlt))
- {
- Caption = Caption + L"\t" + GetKeyNameStr(VK_MENU);
- }
- else if (GrouppedShiftState == (TShiftState() << ssCtrl))
- {
- Caption = Caption + L"\t" + GetKeyNameStr(VK_CONTROL);
- }
- else if (GrouppedShiftState == (TShiftState() << ssShift))
- {
- Caption = Caption + L"\t" + GetKeyNameStr(VK_SHIFT);
- }
- else
- {
- // do not support combined shift states yet
- assert(GrouppedShiftState == TShiftState());
- }
- Item->Caption = Caption;
- if (OnClick != NULL)
- {
- Item->OnClick = OnClick;
- }
- else
- {
- Item->OnClick = MenuItemClick;
- assert((Answer <= 0xFFFF) && (GrouppedShiftState.ToInt() <= 0xFFFF));
- Item->Tag = Answer + (GrouppedShiftState.ToInt() << 16);
- }
- // Hard-coded drop down button width (do not know how to ask for system width).
- // Also we do not update the max button width for the default groupped
- // button caption. We just blindly hope that captions of advanced commands
- // are always longer than the caption of simple default command
- CurButtonWidth += ScaleByTextHeightRunTime(this, 15);
- // never shrink buttons below their default width
- if (GroupWithButton->Width < CurButtonWidth)
- {
- ButtonWidths += CurButtonWidth - GroupWithButton->Width;
- GroupWithButton->Width = CurButtonWidth;
- }
- }
- else
- {
- Button = new TMessageButton(this);
- Button->Name = Name;
- Button->Parent = this;
- Button->Caption = Caption;
- // Scale buttons using regular font, so that they are as large as buttons
- // on other dialogs (note that they are still higher than Windows Task dialog
- // buttons)
- Button->Height = ScaleByTextHeightRunTime(FDummyForm, Button->Height);
- Button->Width = ScaleByTextHeightRunTime(FDummyForm, Button->Width);
- if (OnClick != NULL)
- {
- Button->OnClick = OnClick;
- }
- else
- {
- Button->ModalResult = Answer;
- }
- if (HasMoreMessages)
- {
- Button->Anchors = TAnchors() << akBottom << akLeft;
- }
- // never shrink buttons below their default width
- if (Button->Width < CurButtonWidth)
- {
- Button->Width = CurButtonWidth;
- }
- ButtonWidths += Button->Width;
- }
- return Button;
- }
- //---------------------------------------------------------------------------
- void __fastcall AnswerNameAndCaption(
- unsigned int Answer, UnicodeString & Name, UnicodeString & Caption)
- {
- switch (Answer)
- {
- case qaYes:
- Caption = LoadStr(_SMsgDlgYes.Identifier);
- Name = L"Yes";
- break;
- case qaNo:
- Caption = LoadStr(_SMsgDlgNo.Identifier);
- Name = L"No";
- break;
- case qaOK:
- Caption = LoadStr(_SMsgDlgOK.Identifier);
- Name = L"OK";
- break;
- case qaCancel:
- Caption = LoadStr(_SMsgDlgCancel.Identifier);
- Name = L"Cancel";
- break;
- case qaAbort:
- Caption = LoadStr(_SMsgDlgAbort.Identifier);
- Name = L"Abort";
- break;
- case qaRetry:
- Caption = LoadStr(_SMsgDlgRetry.Identifier);
- Name = L"Retry";
- break;
- case qaIgnore:
- Caption = LoadStr(_SMsgDlgIgnore.Identifier);
- Name = L"Ignore";
- break;
- // Own variant to avoid accelerator conflict with "Abort" button.
- // Note that as of now, ALL_BUTTON is never actually used,
- // because qaAll is always aliased
- case qaAll:
- Caption = LoadStr(ALL_BUTTON);
- Name = L"All";
- break;
- case qaNoToAll:
- Caption = LoadStr(_SMsgDlgNoToAll.Identifier);
- Name = L"NoToAll";
- break;
- // Own variant to avoid accelerator conflict with "Abort" button.
- case qaYesToAll:
- Caption = LoadStr(YES_TO_ALL_BUTTON);
- Name = L"YesToAll";
- break;
- case qaHelp:
- Caption = LoadStr(_SMsgDlgHelp.Identifier);
- Name = L"Help";
- break;
- case qaSkip:
- Caption = LoadStr(SKIP_BUTTON);
- Name = L"Skip";
- break;
- case qaReport:
- Caption = LoadStr(REPORT_BUTTON);
- Name = L"Report";
- break;
- default:
- FAIL;
- throw Exception(L"Undefined answer");
- }
- }
- //---------------------------------------------------------------------------
- static int __fastcall CalculateWidthOnCanvas(UnicodeString Text, void * Arg)
- {
- TCanvas * Canvas = static_cast<TCanvas *>(Arg);
- return Canvas->TextWidth(Text);
- }
- //---------------------------------------------------------------------------
- TForm * __fastcall TMessageForm::Create(const UnicodeString & Msg,
- TStrings * MoreMessages, TMsgDlgType DlgType, unsigned int Answers,
- const TQueryButtonAlias * Aliases, unsigned int AliasesCount,
- unsigned int TimeoutAnswer, TButton ** TimeoutButton, const UnicodeString & ImageName,
- const UnicodeString & NeverAskAgainCaption, const UnicodeString & MoreMessagesUrl,
- TSize MoreMessagesSize)
- {
- unsigned int DefaultAnswer;
- if (FLAGSET(Answers, qaOK))
- {
- DefaultAnswer = qaOK;
- }
- else if (FLAGSET(Answers, qaYes))
- {
- DefaultAnswer = qaYes;
- }
- else
- {
- DefaultAnswer = qaRetry;
- }
- unsigned int CancelAnswer = ::CancelAnswer(Answers);
- if (TimeoutButton != NULL)
- {
- *TimeoutButton = NULL;
- }
- TColor MainInstructionColor = Graphics::clNone;
- HFONT MainInstructionFont = 0;
- HFONT InstructionFont = 0;
- HTHEME Theme = OpenThemeData(0, L"TEXTSTYLE");
- if (Theme != NULL)
- {
- LOGFONT AFont;
- COLORREF AColor;
- memset(&AFont, 0, sizeof(AFont));
- if (GetThemeFont(Theme, NULL, TEXT_MAININSTRUCTION, 0, TMT_FONT, &AFont) == S_OK)
- {
- MainInstructionFont = CreateFontIndirect(&AFont);
- }
- if (GetThemeColor(Theme, TEXT_MAININSTRUCTION, 0, TMT_TEXTCOLOR, &AColor) == S_OK)
- {
- MainInstructionColor = (TColor)AColor;
- }
- memset(&AFont, 0, sizeof(AFont));
- if (GetThemeFont(Theme, NULL, TEXT_INSTRUCTION, 0, TMT_FONT, &AFont) == S_OK)
- {
- InstructionFont = CreateFontIndirect(&AFont);
- }
- CloseThemeData(Theme);
- }
- TMessageForm * Result = SafeFormCreate<TMessageForm>();
- if (InstructionFont != 0)
- {
- Result->Font->Handle = InstructionFont;
- }
- else
- {
- Result->Font->Assign(Screen->MessageFont);
- }
- // Can be possibly nul when error occurs before configuration is created
- if (Configuration != NULL)
- {
- Configuration->Usage->Set(L"ThemeMessageFontSize", Result->Font->Size);
- }
- // make sure we consider sizes of the monitor,
- // that is set in DoFormWindowProc(CM_SHOWINGCHANGED) later.
- Forms::TMonitor * Monitor = FormMonitor(Result);
- bool HasMoreMessages = (MoreMessages != NULL) || !MoreMessagesUrl.IsEmpty();
- Result->BiDiMode = Application->BiDiMode;
- Result->BorderStyle = bsDialog;
- Result->Canvas->Font = Result->Font;
- Result->KeyPreview = true;
- int HorzMargin = ScaleByTextHeightRunTime(Result, mcHorzMargin);
- int VertMargin = ScaleByTextHeightRunTime(Result, mcVertMargin);
- int HorzSpacing = ScaleByTextHeightRunTime(Result, mcHorzSpacing);
- int ButtonVertMargin = ScaleByTextHeightRunTime(Result, mcButtonVertMargin);
- int ButtonWidths = 0;
- int ButtonHeight = -1;
- std::vector<TButton *> ButtonControls;
- TAnswerButtons AnswerButtons;
- for (unsigned int Answer = qaFirst; Answer <= qaLast; Answer = Answer << 1)
- {
- if (FLAGSET(Answers, Answer))
- {
- assert(Answer != mrCancel);
- UnicodeString Caption;
- UnicodeString Name;
- AnswerNameAndCaption(Answer, Name, Caption);
- TNotifyEvent OnClick = NULL;
- int GroupWith = -1;
- TShiftState GrouppedShiftState;
- if (Aliases != NULL)
- {
- for (unsigned int i = 0; i < AliasesCount; i++)
- {
- if (Answer == Aliases[i].Button)
- {
- if (!Aliases[i].Alias.IsEmpty())
- {
- Caption = Aliases[i].Alias;
- }
- OnClick = Aliases[i].OnClick;
- GroupWith = Aliases[i].GroupWith;
- GrouppedShiftState = Aliases[i].GrouppedShiftState;
- assert((OnClick == NULL) || (GrouppedShiftState == TShiftState()));
- break;
- }
- }
- }
- // we hope that all grouped-with buttons are for answer with greater
- // value that the answer to be grouped with
- if (GroupWith >= 0)
- {
- if (ALWAYS_FALSE(GroupWith >= static_cast<int>(Answer)) ||
- ALWAYS_FALSE(Answer == TimeoutAnswer) &&
- ALWAYS_FALSE(Answer == DefaultAnswer) &&
- ALWAYS_FALSE(Answer == CancelAnswer))
- {
- GroupWith = -1;
- }
- }
- bool IsTimeoutButton = (TimeoutButton != NULL) && (Answer == TimeoutAnswer);
- if (Answer == qaHelp)
- {
- assert(OnClick == NULL);
- OnClick = Result->HelpButtonClick;
- }
- if (Answer == qaReport)
- {
- assert(OnClick == NULL);
- OnClick = Result->ReportButtonClick;
- }
- TButton * Button = Result->CreateButton(
- Name, Caption, Answer,
- OnClick, IsTimeoutButton, GroupWith, GrouppedShiftState,
- AnswerButtons, HasMoreMessages, ButtonWidths);
- if (Button != NULL)
- {
- ButtonControls.push_back(Button);
- Button->Default = (Answer == DefaultAnswer);
- Button->Cancel = (Answer == CancelAnswer);
- if (ButtonHeight < 0)
- {
- ButtonHeight = Button->Height;
- }
- assert(ButtonHeight == Button->Height);
- AnswerButtons.insert(TAnswerButtons::value_type(Answer, Button));
- if (IsTimeoutButton)
- {
- *TimeoutButton = Button;
- }
- }
- }
- }
- int NeverAskAgainWidth = 0;
- if (!NeverAskAgainCaption.IsEmpty())
- {
- NeverAskAgainWidth =
- ScaleByTextHeightRunTime(Result, 16) + // checkbox
- Result->Canvas->TextWidth(NeverAskAgainCaption) +
- ScaleByTextHeightRunTime(Result, 16); // margin
- }
- int ButtonSpacing = ScaleByTextHeightRunTime(Result, mcButtonSpacing);
- int ButtonGroupWidth = NeverAskAgainWidth;
- if (!ButtonControls.empty())
- {
- ButtonGroupWidth += ButtonWidths +
- ButtonSpacing * (ButtonControls.size() - 1);
- }
- assert((ButtonHeight > 0) && (ButtonWidths > 0));
- TPanel * Panel = new TPanel(Result);
- Panel->Name = L"Panel";
- Panel->Parent = Result;
- Panel->Color = clWindow;
- Panel->ParentBackground = false;
- Panel->Anchors = TAnchors() << akLeft << akRight << akTop;
- Panel->BevelOuter = bvNone;
- Panel->BevelKind = bkNone;
- Panel->Caption = L"";
- int IconWidth = 0;
- int IconHeight = 0;
- const wchar_t * IconID = IconIDs[DlgType];
- if ((IconID != NULL) || !ImageName.IsEmpty())
- {
- TImage * Image = new TImage(Panel);
- Image->Name = L"Image";
- Image->Parent = Panel;
- if (!ImageName.IsEmpty())
- {
- LoadResourceImage(Image, ImageName);
- }
- else
- {
- // Until Windows 8, LoadIcon for IDI_XXX always returns 32x32 image.
- // Since Windows 8.1, it returns image adjusted for DPI.
- // For 125%, it's scaled version. For 150%, there's native larger version.
- Image->Picture->Icon->Handle = LoadIcon(0, IconID);
- }
- Image->SetBounds(HorzMargin, VertMargin, Image->Picture->Width, Image->Picture->Height);
- IconWidth = Image->Width + HorzSpacing;
- IconHeight = Image->Height;
- }
- int MaxTextWidth = ScaleByTextHeightRunTime(Result, mcMaxDialogWidth);
- // if the dialog would be wide anyway (overwrite confirmation on Windows XP),
- // to fit the buttons, do not restrict the text
- if (MaxTextWidth < ButtonGroupWidth - IconWidth)
- {
- MaxTextWidth = ButtonGroupWidth - IconWidth;
- }
- UnicodeString BodyMsg = Msg;
- BodyMsg = RemoveInteractiveMsgTag(BodyMsg);
- UnicodeString MainMsg;
- if (ExtractMainInstructions(BodyMsg, MainMsg))
- {
- Result->MessageText = MainMsg + BodyMsg;
- BodyMsg = BodyMsg.TrimLeft();
- }
- else
- {
- Result->MessageText = BodyMsg;
- }
- ApplyTabs(Result->MessageText, L' ', NULL, NULL);
- // Windows XP (not sure about Vista) does not support Hair space.
- // For Windows XP, we still keep the existing hack by using hard-coded spaces
- // in resource string
- if (CheckWin32Version(6, 1))
- {
- // Have to be padding with spaces (the smallest space defined, hair space = 1px),
- // as tabs actually do not tab, just expand to 8 spaces.
- // Otherwise we would have to do custom drawing
- // (using GetTabbedTextExtent and TabbedTextOut)
- const wchar_t HairSpace = L'\x200A';
- ApplyTabs(BodyMsg, HairSpace, CalculateWidthOnCanvas, Result->Canvas);
- }
- assert(MainMsg.Pos(L"\t") == 0);
- int IconTextWidth = -1;
- int IconTextHeight = 0;
- int ALeft = IconWidth + HorzMargin;
- for (int MessageIndex = 0; MessageIndex <= 1; MessageIndex++)
- {
- UnicodeString LabelMsg;
- UnicodeString LabelName;
- TColor LabelColor = Graphics::clNone;
- HFONT LabelFont = 0;
- switch (MessageIndex)
- {
- case 0:
- LabelMsg = MainMsg;
- LabelName = L"MainMessage";
- LabelColor = MainInstructionColor;
- LabelFont = MainInstructionFont;
- break;
- case 1:
- LabelMsg = BodyMsg;
- LabelName = L"Message";
- break;
- default:
- FAIL;
- break;
- }
- if (!LabelMsg.IsEmpty())
- {
- TLabel * Message = new TLabel(Panel);
- Message->Name = LabelName;
- Message->Parent = Panel;
- Message->WordWrap = true;
- Message->Caption = LabelMsg;
- Message->BiDiMode = Result->BiDiMode;
- // added to show & as & for messages containing !& pattern of custom commands
- // (suppose that we actually never want to use & as accel in message text)
- Message->ShowAccelChar = false;
- if (LabelFont != 0)
- {
- Message->Font->Handle = LabelFont;
- if (ALWAYS_TRUE(LabelFont == MainInstructionFont))
- {
- Configuration->Usage->Set(L"ThemeMainInstructionFontSize", Message->Font->Size);
- }
- }
- if (LabelColor != Graphics::clNone)
- {
- Message->Font->Color = LabelColor;
- }
- TRect TextRect;
- SetRect(&TextRect, 0, 0, MaxTextWidth, 0);
- DrawText(Message->Canvas->Handle, LabelMsg.c_str(), LabelMsg.Length() + 1, &TextRect,
- DT_EXPANDTABS | DT_CALCRECT | DT_WORDBREAK | DT_NOPREFIX |
- Result->DrawTextBiDiModeFlagsReadingOnly());
- int MaxWidth = Monitor->Width - HorzMargin * 2 - IconWidth - 30;
- if (TextRect.right > MaxWidth)
- {
- // this will truncate the text, we should implement something smarter eventually
- TextRect.right = MaxWidth;
- }
- IconTextWidth = Max(IconTextWidth, IconWidth + TextRect.Right);
- if (IconTextHeight > 0)
- {
- IconTextHeight += VertMargin;
- }
- Message->SetBounds(ALeft, VertMargin + IconTextHeight, TextRect.Right, TextRect.Bottom);
- IconTextHeight += TextRect.Bottom;
- }
- }
- assert((IconTextWidth > 0) && (IconTextHeight > 0));
- IconTextHeight = Max(IconTextHeight, IconHeight);
- int MoreMessageHeight =
- (HasMoreMessages ?
- ScaleByTextHeightRunTime(Result, (MoreMessagesSize.Height > 0 ? MoreMessagesSize.Height : mcMoreMessageHeight)) : 0);
- Panel->SetBounds(0, 0, Result->ClientWidth, VertMargin + IconTextHeight + VertMargin + MoreMessageHeight);
- TControl * MoreMessagesControl = NULL;
- if (HasMoreMessages)
- {
- if (MoreMessages != NULL)
- {
- assert(MoreMessagesUrl.IsEmpty());
- TMemo * MessageMemo = new TMemo(Panel);
- MoreMessagesControl = MessageMemo;
- MessageMemo->Name = L"MessageMemo";
- MessageMemo->Parent = Panel;
- MessageMemo->ReadOnly = true;
- MessageMemo->WantReturns = False;
- MessageMemo->ScrollBars = ssVertical;
- MessageMemo->Anchors = TAnchors() << akLeft << akRight << akTop;
- MessageMemo->Lines->Text = MoreMessages->Text;
- Result->MessageMemo = MessageMemo;
- }
- else if (ALWAYS_TRUE(!MoreMessagesUrl.IsEmpty()))
- {
- TPanel * MessageBrowserPanel = new TPanel(Panel);
- MessageBrowserPanel->Parent = Panel;
- MessageBrowserPanel->Anchors = TAnchors() << akLeft << akRight << akTop;
- MessageBrowserPanel->BevelOuter = bvNone;
- MessageBrowserPanel->BevelInner = bvNone; // default
- MessageBrowserPanel->BevelKind = bkTile;
- Result->MessageBrowserPanel = MessageBrowserPanel;
- MoreMessagesControl = Result->MessageBrowserPanel;
- UnicodeString FontSizeParam = FORMAT(L"fontsize=%d", (Result->Font->Size));
- UnicodeString Url = AppendUrlParams(CampaignUrl(MoreMessagesUrl), FontSizeParam);
- Result->MessageBrowserUrl = Url;
- }
- }
- int MinClientWidth =
- ScaleByTextHeightRunTime(Result,
- HasMoreMessages ? (MoreMessagesSize.Width > 0 ? MoreMessagesSize.Width : mcMinDialogwithMoreMessagesWidth) : mcMinDialogWidth);
- int AClientWidth =
- Max(
- (IconTextWidth > ButtonGroupWidth ? IconTextWidth : ButtonGroupWidth) +
- HorzMargin * 2,
- MinClientWidth);
- Result->ClientWidth = AClientWidth;
- Result->ClientHeight =
- Panel->Height + ButtonVertMargin + ButtonHeight + ButtonVertMargin;
- Result->Left = (Monitor->Width / 2) - (Result->Width / 2);
- Result->Top = (Monitor->Height / 2) - (Result->Height / 2);
- if (DlgType != mtCustom)
- {
- Result->Caption = LoadResourceString(Captions[DlgType]);
- }
- else
- {
- Result->Caption = Application->Title;
- }
- if (MoreMessagesControl != NULL)
- {
- MoreMessagesControl->SetBounds(
- ALeft,
- Panel->Height - MoreMessageHeight,
- Result->ClientWidth - ALeft - HorzMargin,
- MoreMessageHeight - VertMargin);
- }
- int ButtonTop = Panel->Height + ButtonVertMargin;
- int X = Result->ClientWidth - ButtonGroupWidth + NeverAskAgainWidth - HorzMargin;
- for (unsigned int i = 0; i < ButtonControls.size(); i++)
- {
- ButtonControls[i]->SetBounds(
- X, ButtonTop, ButtonControls[i]->Width, ButtonControls[i]->Height);
- X += ButtonControls[i]->Width + ButtonSpacing;
- }
- if (!NeverAskAgainCaption.IsEmpty() &&
- !ButtonControls.empty())
- {
- TCheckBox * NeverAskAgainCheck = new TCheckBox(Result);
- NeverAskAgainCheck->Name = L"NeverAskAgainCheck";
- NeverAskAgainCheck->Parent = Result;
- NeverAskAgainCheck->Caption = NeverAskAgainCaption;
- NeverAskAgainCheck->Anchors = TAnchors() << akBottom << akLeft;
- TButton * FirstButton = ButtonControls[0];
- int NeverAskAgainHeight = ScaleByTextHeightRunTime(Result, NeverAskAgainCheck->Height);
- int NeverAskAgainTop = FirstButton->Top + ((FirstButton->Height - NeverAskAgainHeight) / 2);
- int NeverAskAgainLeft = HorzMargin;
- NeverAskAgainCheck->SetBounds(
- NeverAskAgainLeft, NeverAskAgainTop, NeverAskAgainWidth, NeverAskAgainHeight);
- }
- return Result;
- }
- //---------------------------------------------------------------------------
- TForm * __fastcall CreateMoreMessageDialog(const UnicodeString & Msg,
- TStrings * MoreMessages, TMsgDlgType DlgType, unsigned int Answers,
- const TQueryButtonAlias * Aliases, unsigned int AliasesCount,
- unsigned int TimeoutAnswer, TButton ** TimeoutButton, const UnicodeString & ImageName,
- const UnicodeString & NeverAskAgainCaption, const UnicodeString & MoreMessagesUrl,
- TSize MoreMessagesSize)
- {
- return TMessageForm::Create(Msg, MoreMessages, DlgType, Answers,
- Aliases, AliasesCount, TimeoutAnswer, TimeoutButton, ImageName,
- NeverAskAgainCaption, MoreMessagesUrl, MoreMessagesSize);
- }
|