Browse Source

Bug 1727: Failure when starting GUI

https://winscp.net/tracker/1727

Source commit: f3096a73ba80657f7bce396c5e1fc7553970c6a9
Martin Prikryl 6 years ago
parent
commit
c08a3e7c02

+ 7 - 2
source/packages/filemng/CustomDirView.pas

@@ -520,6 +520,8 @@ function CompareLogicalTextPas(const S1, S2: string; NaturalOrderNumericalSortin
 
 
 function OverlayImageList(Size: Integer): TImageList;
 function OverlayImageList(Size: Integer): TImageList;
 
 
+procedure InitFileControls;
+
 var
 var
   StdDirIcon: Integer;
   StdDirIcon: Integer;
   StdDirSelIcon: Integer;
   StdDirSelIcon: Integer;
@@ -569,12 +571,15 @@ begin
   end;
   end;
 end; {GetshFileInfo}
 end; {GetshFileInfo}
 
 
-procedure InitGlobals;
+procedure InitFileControls;
 begin
 begin
   if not GlobalsInitialized then
   if not GlobalsInitialized then
   begin
   begin
     GlobalsInitialized := True;
     GlobalsInitialized := True;
 
 
+    // See the comment in the call from Execute()
+    NeedShellImageLists;
+
     // Calling GetshFileInfo in Windows Session 0 sometime cause crash
     // Calling GetshFileInfo in Windows Session 0 sometime cause crash
     // (not immediately, but very shortly afterwards [few ms]).
     // (not immediately, but very shortly afterwards [few ms]).
     // So this code was moved from initialization section to avoid it
     // So this code was moved from initialization section to avoid it
@@ -786,7 +791,7 @@ end;
 
 
 constructor TCustomDirView.Create(AOwner: TComponent);
 constructor TCustomDirView.Create(AOwner: TComponent);
 begin
 begin
-  InitGlobals;
+  InitFileControls;
 
 
   inherited;
   inherited;
 
 

+ 13 - 5
source/packages/my/PasTools.pas

@@ -66,6 +66,7 @@ function GetSystemMetricsForControl(Control: TControl; nIndex: Integer): Integer
 type
 type
   TImageListSize = (ilsSmall, ilsLarge);
   TImageListSize = (ilsSmall, ilsLarge);
 
 
+procedure NeedShellImageLists;
 function ShellImageListForControl(Control: TControl; Size: TImageListSize): TImageList;
 function ShellImageListForControl(Control: TControl; Size: TImageListSize): TImageList;
 
 
 function ControlHasRecreationPersistenceData(Control: TControl): Boolean;
 function ControlHasRecreationPersistenceData(Control: TControl): Boolean;
@@ -486,6 +487,7 @@ var
   Height, Width: Integer;
   Height, Width: Integer;
   ShellImageList: TImageList;
   ShellImageList: TImageList;
   SHGetImageList: TSHGetImageList;
   SHGetImageList: TSHGetImageList;
+  HR: HRESULT;
 begin
 begin
   Lib := LoadLibrary('shell32');
   Lib := LoadLibrary('shell32');
   SHGetImageList := GetProcAddress(Lib, 'SHGetImageList');
   SHGetImageList := GetProcAddress(Lib, 'SHGetImageList');
@@ -493,7 +495,8 @@ begin
   for ImageList := 0 to SHIL_LAST do
   for ImageList := 0 to SHIL_LAST do
   begin
   begin
     // VCL have declaration for SHGetImageList in ShellAPI, but it does not link
     // VCL have declaration for SHGetImageList in ShellAPI, but it does not link
-    if (SHGetImageList(ImageList, IID_IImageList, Pointer(Handle)) = S_OK) and
+    HR := SHGetImageList(ImageList, IID_IImageList, Pointer(Handle));
+    if (HR = S_OK) and
        ImageList_GetIconSize(Handle, Width, Height) then
        ImageList_GetIconSize(Handle, Width, Height) then
     begin
     begin
 
 
@@ -511,6 +514,14 @@ begin
   end;
   end;
 end;
 end;
 
 
+procedure NeedShellImageLists;
+begin
+  if ShellImageLists = nil then
+  begin
+    InitializeShellImageLists;
+  end;
+end;
+
 function ShellImageListForControl(Control: TControl; Size: TImageListSize): TImageList;
 function ShellImageListForControl(Control: TControl; Size: TImageListSize): TImageList;
 var
 var
   ImageListPair: TPair<Integer, TImageList>;
   ImageListPair: TPair<Integer, TImageList>;
@@ -518,10 +529,7 @@ var
   Diff, BestDiff: Integer;
   Diff, BestDiff: Integer;
 begin
 begin
   // Delay load image lists, not to waste resources in console/scripting mode
   // Delay load image lists, not to waste resources in console/scripting mode
-  if ShellImageLists = nil then
-  begin
-    InitializeShellImageLists;
-  end;
+  NeedShellImageLists;
 
 
   case Size of
   case Size of
     ilsSmall: Width := 16;
     ilsSmall: Width := 16;

+ 4 - 0
source/windows/WinMain.cpp

@@ -964,6 +964,10 @@ int __fastcall Execute()
       }
       }
 
 
       WinConfiguration->CheckDefaultTranslation();
       WinConfiguration->CheckDefaultTranslation();
+      // Loading shell image lists here (rather than only on demand when file controls are being created)
+      // reduces risk of an occasional crash.
+      // It seems that the point is to load the lists before any call to SHGetFileInfoWithTimeout.
+      InitFileControls();
 
 
       if (!Params->Empty)
       if (!Params->Empty)
       {
       {