LCL: Add a new function SelectInFolder in LclIntf. Use it in source editor's StatusBar.

This commit is contained in:
Juha 2025-02-01 18:28:34 +02:00
parent 6d7ce926a7
commit 28ba5306c3
7 changed files with 108 additions and 9 deletions

View File

@ -54,7 +54,11 @@ type
end; end;
{$ENDIF} {$ENDIF}
procedure RunCmdFromPath(const ProgramFilename, CmdLineParameters: string); // poWaitOnExit prevents a zombie process but locks the calling program until the process
// terminates. When runnning a GUI application you may want to use [] as ProcessOpts.
procedure RunCmdFromPath(const ProgramFilename, CmdLineParameters: string;
ProcessOpts: TProcessOptions = [poWaitOnExit]);
function FindFilenameOfCmd(ProgramFilename: string): string; function FindFilenameOfCmd(ProgramFilename: string): string;
function GetSystemThreadCount: integer; // guess number of cores function GetSystemThreadCount: integer; // guess number of cores
@ -156,7 +160,8 @@ end;
// For example: ProgramFilename='ls' CmdLineParameters='-l /home' // For example: ProgramFilename='ls' CmdLineParameters='-l /home'
// Will locate and execute the file '/bin/ls' // Will locate and execute the file '/bin/ls'
// If the command isn't found, an exception will be raised // If the command isn't found, an exception will be raised
procedure RunCmdFromPath(const ProgramFilename, CmdLineParameters: string); procedure RunCmdFromPath(const ProgramFilename, CmdLineParameters: string;
ProcessOpts: TProcessOptions);
var var
NewProgramFilename: String; NewProgramFilename: String;
BrowserProcess: TProcessUTF8; BrowserProcess: TProcessUTF8;
@ -172,7 +177,7 @@ begin
BrowserProcess := TProcessUTF8.Create(nil); BrowserProcess := TProcessUTF8.Create(nil);
try try
BrowserProcess.InheritHandles:=false; BrowserProcess.InheritHandles:=false;
BrowserProcess.Options := [poWaitOnExit]; // Prevent a zombie process. BrowserProcess.Options := ProcessOpts;
// Encloses the executable with "" if its name has spaces // Encloses the executable with "" if its name has spaces
if Pos(' ',NewProgramFilename)>0 then if Pos(' ',NewProgramFilename)>0 then
NewProgramFilename:='"'+NewProgramFilename+'"'; NewProgramFilename:='"'+NewProgramFilename+'"';

View File

@ -9033,7 +9033,7 @@ end;
procedure TSourceNotebook.OpenFolderMenuItemClick(Sender: TObject); procedure TSourceNotebook.OpenFolderMenuItemClick(Sender: TObject);
begin begin
OpenDocument(ExtractFilePath(Statusbar.Panels[CStatusPanelFile].Text)); SelectInFolder(Statusbar.Panels[CStatusPanelFile].Text);
end; end;
procedure TSourceNotebook.ExecuteEditorItemClick(Sender: TObject); procedure TSourceNotebook.ExecuteEditorItemClick(Sender: TObject);

View File

@ -90,14 +90,21 @@ begin
Result := False; Result := False;
end; end;
// Open a given URL with whatever Windows thinks is appropriate // ToDo: Implement
function OpenURL(AURL: String): Boolean; function OpenURL(AURL: String): Boolean;
begin begin
Result := False; Result := False;
end; end;
// Open a document with the default application associated with it in the system // ToDo: Implement
function OpenDocument(APath: String): Boolean; function OpenDocument(APath: String): Boolean;
begin begin
Result := OpenURL(APath); Result := OpenURL(APath);
end; end;
// ToDo: Implement
function SelectInFolder(AFullPath: String): Boolean;
begin
Result := OpenDocument(ExtractFilePath(AFullPath));
end;

View File

@ -51,3 +51,11 @@ begin
ResultingPath:=APath; ResultingPath:=APath;
RunCmdFromPath('open',ResultingPath); RunCmdFromPath('open',ResultingPath);
end; end;
// ToDo: Implement
// Now just open the folder in system filemanager.
function SelectInFolder(AFullPath: String): Boolean;
begin
Result := OpenDocument(ExtractFilePath(AFullPath));
end;

View File

@ -6,7 +6,6 @@ begin
Result := FindPredefinedBrowser(ABrowser, AParams); Result := FindPredefinedBrowser(ABrowser, AParams);
end; end;
// Open a given URL with the default browser
function OpenURL(AURL: String): Boolean; function OpenURL(AURL: String): Boolean;
var var
ABrowser, AParams: String; ABrowser, AParams: String;
@ -22,7 +21,6 @@ begin
RunCmdFromPath(ABrowser,Format(AParams, [AURL])); RunCmdFromPath(ABrowser,Format(AParams, [AURL]));
end; end;
// Open a document with the default application associated with it in the system
function OpenDocument(APath: String): Boolean; function OpenDocument(APath: String): Boolean;
var var
lApp: string; lApp: string;
@ -32,13 +30,15 @@ begin
Exit(OpenDocumentWidgetsetImplementation(APath)); Exit(OpenDocumentWidgetsetImplementation(APath));
Result := True; Result := True;
if not (FileExistsUTF8(APath) or DirectoryExistsUTF8(APath)) then if not (FileExists(APath) or DirectoryExists(APath)) then
Exit(false); Exit(false);
lApp:=FindFilenameOfCmd('xdg-open'); // Portland OSDL/FreeDesktop standard on Linux lApp:=FindFilenameOfCmd('xdg-open'); // Portland OSDL/FreeDesktop standard on Linux
if lApp='' then if lApp='' then
// Does not work. Tested in Manjaro Linux + KDE. Could be removed.
lApp:=FindFilenameOfCmd('kfmclient'); // KDE command lApp:=FindFilenameOfCmd('kfmclient'); // KDE command
if lApp='' then if lApp='' then
// Does this work? Please test somebody.
lApp:=FindFilenameOfCmd('gnome-open'); // GNOME command lApp:=FindFilenameOfCmd('gnome-open'); // GNOME command
if lApp='' then if lApp='' then
Exit(False); Exit(False);
@ -47,3 +47,65 @@ begin
APath:=QuotedStr(APath); APath:=QuotedStr(APath);
RunCmdFromPath(lApp,APath); RunCmdFromPath(lApp,APath);
end; end;
function SelectInFolder(AFullPath: String): Boolean;
var
lApp, DesktopEnv: string;
// Parameters for a filename or directory, and for filemanager selection.
FileParam, SelParam: string;
procedure SelectNautilus;
begin
lApp := 'nautilus';
SelParam := '--select ';
end;
procedure SelectDolphin;
begin
lApp := 'dolphin';
SelParam := '--select ';
end;
begin
if not (FileExists(AFullPath) or DirectoryExists(AFullPath)) then
Exit(false);
FileParam := AFullPath;
SelParam := '';
DesktopEnv := GetEnvironmentVariable('XDG_CURRENT_DESKTOP');
//pcmanfm does not have a --select option i can find but may be another way
//caja has a --select option but i cannot work it out. keeps trying to open file as a path issue
//try to get desktop env first. but could be any file manager... seems like a good order
if Pos('GNOME', DesktopEnv) > 0 then
SelectNautilus
else if Pos('KDE', DesktopEnv) > 0 then
SelectDolphin
else if FileExists('/usr/bin/nautilus') then
SelectNautilus
else if FileExists('/usr/bin/dolphin') then
SelectDolphin
else if FileExists('/usr/bin/thunar') then
lApp := 'thunar' // Unable to select but opens directory
else if FileExists('/usr/bin/nemo') then
lApp := 'nemo'
else if FileExists('/usr/bin/caja') then
// Can't figure out the --select option which help says is there but does not behave
lApp := 'caja'
else if FileExists('/usr/bin/krusader') then
begin
// Not possible to select the file but better open the dir in the left pane
lApp := 'krusader';
SelParam := '--left ';
FileParam := ExtractFilePath(AFullPath); // Just the directory
end
else begin // No known file manager found. Trying xdg-open for the directory
lApp := FindFilenameOfCmd('xdg-open');
FileParam := ExtractFilePath(AFullPath);
end;
if (FileParam <> '') and (FileParam[1] <> '"') then
FileParam := QuotedStr(FileParam);
RunCmdFromPath(lApp, SelParam + FileParam, []);
Result := True;
end;

View File

@ -352,3 +352,11 @@ function OpenDocument(APath: String): Boolean;
begin begin
Result := OpenURL(APath); Result := OpenURL(APath);
end; end;
// ToDo: Implement
// Now just open the folder in system filemanager.
function SelectInFolder(AFullPath: String): Boolean;
begin
Result := OpenDocument(ExtractFilePath(AFullPath));
end;

View File

@ -79,10 +79,19 @@ function GetTickStep: DWord;
{$ENDIF} {$ENDIF}
function FindDefaultBrowser(out ABrowser, AParams: String): Boolean; function FindDefaultBrowser(out ABrowser, AParams: String): Boolean;
// Open a given URL with the system's default browser.
// Spaces in URLs need to be encoded as %20 Read http://www.ietf.org/rfc/rfc1738.txt // Spaces in URLs need to be encoded as %20 Read http://www.ietf.org/rfc/rfc1738.txt
function OpenURL(AURL: String): Boolean; function OpenURL(AURL: String): Boolean;
// Open a document in the system's default application associated with it.
function OpenDocument(APath: String): Boolean; function OpenDocument(APath: String): Boolean;
// Open a system filemanager and select the given file there.
// AFullPath contains a directory and a file name.
// Works also with directory names, then nothing gets selected.
function SelectInFolder(AFullPath: String): Boolean;
type type
TOpenParamStringProc = function (AString: string): Boolean of object; TOpenParamStringProc = function (AString: string): Boolean of object;