mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-05 04:27:55 +02:00
Dialogs: implement TOpenOptionEx.ofUseXPStyleAsFallBack (obviously Windows only). Resolves issue #40298.
This commit is contained in:
parent
8b937e6645
commit
7cfdbd74c8
@ -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;
|
||||||
|
@ -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.
|
||||||
|
Loading…
Reference in New Issue
Block a user