Dialogs: implement TOpenOptionEx.ofUseXPStyleAsFallBack (obviously Windows only). Resolves issue #40298.

This commit is contained in:
Bart 2025-01-05 15:02:11 +01:00
parent 8b937e6645
commit 7cfdbd74c8
2 changed files with 76 additions and 19 deletions

View File

@ -222,8 +222,9 @@ type
ofPickFolders, //Windows Vista+ Turns the dialog into a TSelectDirectoryDialog ofPickFolders, //Windows Vista+ Turns the dialog into a TSelectDirectoryDialog
ofOkButtonNeedsInteraction, //Windows Vista+ The OK button will be disabled until the user navigates the view or edits the filename (if applicable). ofOkButtonNeedsInteraction, //Windows Vista+ The OK button will be disabled until the user navigates the view or edits the filename (if applicable).
ofForceFileSystem, //Windows Vista+ Ensures that returned items are file system items ofForceFileSystem, //Windows Vista+ Ensures that returned items are file system items
ofAllNonStorageItems //Windows Vista+ Enables the user to choose any item in the Shell namespace, not just those with SFGAO_STREAM or SFAGO_FILESYSTEM attributes. ofAllNonStorageItems, //Windows Vista+ Enables the user to choose any item in the Shell namespace, not just those with SFGAO_STREAM or SFAGO_FILESYSTEM attributes.
// This flag cannot be combined with FOS_FORCEFILESYSTEM. // This flag cannot be combined with FOS_FORCEFILESYSTEM.
ofUseXPStyleAsFallBack //Windows Vista+ Use XP style dialog if creating Vista style dialog fails (e.g. when running under Windows Recovery)
// Intentionally not supported: ofDefaultNoMiniMode, ofHideMruPlaces: these values are not supported as of Windows 7. // Intentionally not supported: ofDefaultNoMiniMode, ofHideMruPlaces: these values are not supported as of Windows 7.
); );
TOpenOptionsEx = set of TOpenOptionEx; TOpenOptionsEx = set of TOpenOptionEx;

View File

@ -193,7 +193,7 @@ var
implementation implementation
uses uses
CommCtrl, TaskDlgEmulation; CommCtrl, TaskDlgEmulation, contnrs;
function SaveApplicationState: TApplicationState; function SaveApplicationState: TApplicationState;
begin begin
@ -354,15 +354,6 @@ begin
ACommonDialog.UserChoice := mrCancel; ACommonDialog.UserChoice := mrCancel;
end; end;
function CanUseVistaDialogs(const AOpenDialog: TOpenDialog): Boolean;
begin
{$IFnDEF DisableVistaDialogs}
Result := (WindowsVersion >= wvVista) and not (ofOldStyleDialog in AOpenDialog.Options);
{$ELSE}
Result := False;
{$ENDIF}
end;
{ TWin32WSColorDialog } { TWin32WSColorDialog }
@ -788,6 +779,47 @@ end;
{ TWin32WSOpenDialog } { TWin32WSOpenDialog }
var
XPStyleFallBackList: TFPObjectList = nil;
procedure MaybeInitXPStyleFallBackList;
begin
if not Assigned(XPStyleFallBackList) then
XPStyleFallBackList := TFPObjectList.Create(False); //don't free objects
end;
procedure FreeXPStyleFallBackList;
begin
if Assigned(XPStyleFallBackList) then
FreeAndNil(XPStyleFallBackList);
end;
function IsXPStyleFallBack(const AOpenDialog: TOpenDialog): Boolean;
var
Idx: Integer;
begin
if not Assigned(XPStyleFallBackList) or not (ofUseXPStyleAsFallBack in AOpenDialog.OptionsEx) then
Exit(False);
Idx := XPStyleFallBackList.IndexOf(AOpenDialog);
Result := (Idx <> -1);
end;
function IsXPStyleFallBack(const AOpenDialog: TOpenDialog; out Idx: Integer): Boolean;
begin
if not Assigned(XPStyleFallBackList) or not (ofUseXPStyleAsFallBack in AOpenDialog.OptionsEx) then
Exit(False);
Idx := XPStyleFallBackList.IndexOf(AOpenDialog);
Result := (Idx <> -1);
end;
function CanUseVistaDialogs(const AOpenDialog: TOpenDialog): Boolean;
begin
{$IFnDEF DisableVistaDialogs}
Result := (WindowsVersion >= wvVista) and not (ofOldStyleDialog in AOpenDialog.Options);
{$ELSE}
Result := False;
{$ENDIF}
end;
class procedure TWin32WSOpenDialog.SetupVistaFileDialog(ADialog: IFileDialog; const AOpenDialog: TOpenDialog); class procedure TWin32WSOpenDialog.SetupVistaFileDialog(ADialog: IFileDialog; const AOpenDialog: TOpenDialog);
var var
@ -998,8 +1030,9 @@ var
HRes: HRESULT; HRes: HRESULT;
DlgType: TIID; DlgType: TIID;
CLS_ID: TGUID; CLS_ID: TGUID;
AOpenDialog: TOpenDialog absolute ACommonDialog;
begin begin
if CanUseVistaDialogs(TOpenDialog(ACommonDialog)) then if CanUseVistaDialogs(AOpenDialog) then
begin begin
if (ACommonDialog is TSaveDialog) then if (ACommonDialog is TSaveDialog) then
begin begin
@ -1015,29 +1048,49 @@ begin
if Succeeded(HRes) and Assigned(Dialog) then if Succeeded(HRes) and Assigned(Dialog) then
begin begin
Dialog._AddRef; Dialog._AddRef;
SetupVistaFileDialog(Dialog, TOpenDialog(ACommonDialog)); SetupVistaFileDialog(Dialog, AOpenDialog);
Result := THandle(Dialog); Result := THandle(Dialog);
end end
else else
Result := INVALID_HANDLE_VALUE; begin
end if (ofUseXPStyleAsFallback in AOpenDialog.OptionsEx) then
begin
MaybeInitXPStyleFallBackList;
XPStyleFallbackList.Add(ACommonDialog);
//debugln(['TWin32WSOpenDialog.CreateHandle: Added ',DbgSName(AOpenDialog),' to XPStyleFallbackList']);
Result := CreateFileDialogHandle(AOpenDialog);
end
else
Result := INVALID_HANDLE_VALUE;
end;
end//CanUseVistaDialogs
else else
Result := CreateFileDialogHandle(TOpenDialog(ACommonDialog)); Result := CreateFileDialogHandle(AOpenDialog);
end; end;
class procedure TWin32WSOpenDialog.DestroyHandle(const ACommonDialog: TCommonDialog); class procedure TWin32WSOpenDialog.DestroyHandle(const ACommonDialog: TCommonDialog);
var var
Dialog: IFileDialog; Dialog: IFileDialog;
Idx: Integer;
begin begin
if (ACommonDialog.Handle <> 0) and (ACommonDialog.Handle <> INVALID_HANDLE_VALUE) then if (ACommonDialog.Handle <> 0) and (ACommonDialog.Handle <> INVALID_HANDLE_VALUE) then
if CanUseVistaDialogs(TOpenDialog(ACommonDialog)) then begin
if CanUseVistaDialogs(TOpenDialog(ACommonDialog)) and not IsXPStyleFallBack(TOpenDialog(ACommonDialog), Idx) then
begin begin
Dialog := IFileDialog(ACommonDialog.Handle); Dialog := IFileDialog(ACommonDialog.Handle);
Dialog._Release; Dialog._Release;
Dialog := nil; Dialog := nil;
end end
else else
begin
if (Idx <> -1) then
begin
//debugln(['TWin32WSOpenDialog.CreateHandle: Removing ',DbgSName(ACommonDialog),' from XPStyleFallbackList']);
XPStyleFallBackList.Delete(Idx);
end;
DestroyFileDialogHandle(ACommonDialog.Handle) DestroyFileDialogHandle(ACommonDialog.Handle)
end;
end;
end; end;
class procedure TWin32WSOpenDialog.ShowModal(const ACommonDialog: TCommonDialog); class procedure TWin32WSOpenDialog.ShowModal(const ACommonDialog: TCommonDialog);
@ -1054,7 +1107,7 @@ begin
lInitialDir := TOpenDialog(ACommonDialog).InitialDir; lInitialDir := TOpenDialog(ACommonDialog).InitialDir;
if lInitialDir <> '' then if lInitialDir <> '' then
SetCurrentDirUTF8(lInitialDir); SetCurrentDirUTF8(lInitialDir);
if CanUseVistaDialogs(TOpenDialog(ACommonDialog)) then if CanUseVistaDialogs(TOpenDialog(ACommonDialog)) and not IsXPStyleFallBack(TOpenDialog(ACommonDialog)) then
begin begin
Dialog := IFileOpenDialog(ACommonDialog.Handle); Dialog := IFileOpenDialog(ACommonDialog.Handle);
VistaDialogShowModal(Dialog, TOpenDialog(ACommonDialog)); VistaDialogShowModal(Dialog, TOpenDialog(ACommonDialog));
@ -1136,7 +1189,7 @@ begin
lInitialDir := TSaveDialog(ACommonDialog).InitialDir; lInitialDir := TSaveDialog(ACommonDialog).InitialDir;
if lInitialDir <> '' then if lInitialDir <> '' then
SetCurrentDirUTF8(lInitialDir); SetCurrentDirUTF8(lInitialDir);
if CanUseVistaDialogs(TOpenDialog(ACommonDialog)) then if CanUseVistaDialogs(TOpenDialog(ACommonDialog)) and not IsXPStyleFallBack(TOpenDialog(ACommonDialog)) then
begin begin
Dialog := IFileSaveDialog(ACommonDialog.Handle); Dialog := IFileSaveDialog(ACommonDialog.Handle);
TWin32WSOpenDialog.VistaDialogShowModal(Dialog, TOpenDialog(ACommonDialog)); TWin32WSOpenDialog.VistaDialogShowModal(Dialog, TOpenDialog(ACommonDialog));
@ -2080,4 +2133,7 @@ initialization
else else
OpenFileNameSize := SizeOf(OPENFILENAME); OpenFileNameSize := SizeOf(OPENFILENAME);
InitTaskDialogIndirect; InitTaskDialogIndirect;
finalization
FreeXPStyleFallBackList
end. end.