Delphi OpenDialog文件名显示问题

标签 delphi topendialog

将 OpenDialog 组件放在新表单上,并将此代码放在 OnCreate 中

opendialog1.filename:='This is a long filename.txt';
opendialog1.execute;

当应用程序运行时,会出现一个对话框,文件名显示在打开的对话框中,但文件名会突出显示并滚动到右侧(即使有足够的空间来显示完整的文件名)。

它显示的只是突出显示的“ng filename.txt”。

有没有办法“取消突出显示”文件名并将文本滚动回左侧,以便显示全名“这是一个长文件名.txt”?

如果我可以在显示 OpenDialog 后模拟/将 home 键按入 OpenDialog,则问题将得到解决,但以下选项似乎都不起作用。

keybd_event(VK_HOME, 0, 0, 0);
keybd_event(VK_HOME, 0, KEYEVENTF_KEYUP, 0);

input.Itype := INPUT_KEYBOARD;
input.ki.wVk := 47;
input.ki.wScan := 47;
input.ki.dwFlags := 0;
input.ki.time := 0;
input.ki.dwExtraInfo := 0;
SendInput(1, input, sizeof(input));
input.ki.dwFlags := KEYEVENTF_KEYUP;
SendInput(1, input, sizeof(input));

PostMessage(GetParent(OpenDialog1.Handle), WM_KEYDOWN, VK_HOME, 0);
PostMessage(GetParent(OpenDialog1.Handle), WM_KEYUP, VK_HOME, 0);

如果我将这些代码片段放在 openDialog1.execute 之前,它似乎可以在我的电脑上运行,但这是一个坏主意,因为对话框尚未打开,因此可能无法接收按键消息。

在 opendialog1.execute 调用之后尝试所有这些方法似乎没有做任何事情。

最佳答案

建议的解决方案

keybd_event(VK_HOME, 0, 0, 0);
keybd_event(VK_HOME, 0, KEYEVENTF_KEYUP, 0);

无论您何时执行它,都安全。

如果我在应用程序中按 Ctrl+O,并在等待对话框显示时切换到其他应用程序,会怎样?

然后这个其他应用程序将收到 HOME 键。那么任何事情都可能发生。另一个应用程序可能会显示一个控制患者静脉注入(inject)药物流速的轨迹栏,并且 HOME 键可能会将轨迹栏设置为 0 cc/min。

更有可能的是:您丢失了资源管理器窗口中的选择(可能包含一千个图像)、文档中的插入符位置(位于非常特定的位置)、 TreeView 中选定的节点等。或者您的媒体播放器重新开始当前轨道。

是的,很多人(比如我自己)确实以这种方式多任务!


这是一个可能的(安全的)解决方案。不过,我并不认为它是最优雅的。

方法如下:

  1. 我使用 OnSelectionChange 事件来获得在创建对话框窗口后运行一些代码的机会。但是,我必须手动确保仅在第一次触发此事件时运行“文件名修复”。我还确保仅在(默认)文件名非空时运行此代码。

  2. 第一次触发事件时,我在其中找到了编辑框。我“知道”它是正确的控件,因为(1)它是一个编辑框,并且(2)它的文本等于(默认)文件名(不为空)。

  3. 然后,我专门指示此编辑框将其插入符号移动到第一个字符,然后(重新)选择每个字符。

完整代码:

type
  TOpenDialogFileNameEditData = class
    FileName: string;
    Handle: HWND;
  end;

function EnumChildProc(h: HWND; lp: LPARAM): BOOL; stdcall;
var
  WndClass, WndTxt: array[0..1024] of Char;
begin
  Result := True;
  FillChar(WndClass, SizeOf(WndClass), 0);
  FillChar(WndTxt, SizeOf(WndTxt), 0);
  if GetClassName(h, WndClass, Length(WndClass)) <> 0 then
  begin
    if SameText(WndClass, 'Edit') then
    begin
      if GetWindowText(h, WndTxt, Length(WndTxt)) <> 0 then
      begin
        if WndTxt = TOpenDialogFileNameEditData(lp).FileName then
        begin
          TOpenDialogFileNameEditData(lp).Handle := h;
          Exit(False);
        end;
      end;
    end;
  end;
end;

procedure TForm1.FODE(Sender: TObject);
begin
  if Sender is TOpenDialog then
  begin
    var OpenDialog := TOpenDialog(Sender);
    if OpenDialog.Tag <> 0 then
      Exit;
    OpenDialog.Tag := 1;
    var Data := TOpenDialogFileNameEditData.Create;
    try
      Data.FileName := ExtractFileName(OpenDialog.FileName);
      if Data.FileName.IsEmpty then
        Exit;
      if OpenDialog.Handle <> 0 then
      begin
        EnumChildWindows(OpenDialog.Handle, @EnumChildProc, NativeInt(Data));
        if Data.Handle <> 0 then
        begin
          SendMessage(Data.Handle, EM_SETSEL, 0,  0); // set caret at first char
          SendMessage(Data.Handle, EM_SETSEL, 0, -1); // (re)select all
        end;
      end;
    finally
      Data.Free;
    end;
  end;
end;

procedure TForm1.FormCreate(Sender: TObject);
begin
  with TOpenDialog.Create(nil) do
    try
      FileName := 'This is the story of a horse that met a cat that met a dog the day before.txt';
      OnSelectionChange := FODE;
      Execute
    finally
      Free;
    end;
end;

之前:

Screenshot of unfixed open dialog. The file name reads "e day before.txt".

之后:

Screenshot of fixed open dialog. The file name is now fully visible.

更新

根据要求,我制作了一个易于重用的单元和函数来应用此修复程序。

这是完整的单元:

unit OpenDialogUpgrader;

interface

uses
  Windows, Messages, SysUtils, Types, Dialogs;

procedure FixOpenDialog(AOpenDialog: TOpenDialog);

implementation

type
  TOpenDialogFileNameEditData = class
    FileName: string;
    Handle: HWND;
    class procedure DialogSelectionChange(Sender: TObject);
  end;

procedure FixOpenDialog(AOpenDialog: TOpenDialog);
begin
  AOpenDialog.Tag := 0;
  AOpenDialog.OnSelectionChange := TOpenDialogFileNameEditData.DialogSelectionChange;
end;

{ TOpenDialogFileNameEditData }

function EnumChildProc(h: HWND; lp: LPARAM): BOOL; stdcall;
var
  WndClass, WndTxt: array[0..1024] of Char;
begin
  Result := True;
  FillChar(WndClass, SizeOf(WndClass), 0);
  FillChar(WndTxt, SizeOf(WndTxt), 0);
  if GetClassName(h, WndClass, Length(WndClass)) <> 0 then
  begin
    if SameText(WndClass, 'Edit') then
    begin
      if GetWindowText(h, WndTxt, Length(WndTxt)) <> 0 then
      begin
        if WndTxt = TOpenDialogFileNameEditData(lp).FileName then
        begin
          TOpenDialogFileNameEditData(lp).Handle := h;
          Exit(False);
        end;
      end;
    end;
  end;
end;


class procedure TOpenDialogFileNameEditData.DialogSelectionChange(Sender: TObject);
begin
  if Sender is TOpenDialog then
  begin
    var OpenDialog := TOpenDialog(Sender);
    if OpenDialog.Tag <> 0 then
      Exit;
    OpenDialog.Tag := 1;
    var Data := TOpenDialogFileNameEditData.Create;
    try
      Data.FileName := ExtractFileName(OpenDialog.FileName);
      if Data.FileName.IsEmpty then
        Exit;
      if OpenDialog.Handle <> 0 then
      begin
        EnumChildWindows(OpenDialog.Handle, @EnumChildProc, NativeInt(Data));
        if Data.Handle <> 0 then
        begin
          SendMessage(Data.Handle, EM_SETSEL, 0,  0); // set caret at first char
          SendMessage(Data.Handle, EM_SETSEL, 0, -1); // (re)select all
        end;
      end;
    finally
      Data.Free;
    end;
  end;
end;

end.

要在您自己的单元 X 中使用此单元和函数,只需将该单元 (OpenDialogUpgrader) 添加到您的 X 单元的实现部分 uses 子句并更改您的标准

var OpenDialog := TOpenDialog.Create(nil);
try
  OpenDialog.FileName := 'This is the story of a horse that met a cat that met a dog the day before.txt';
  OpenDialog.Execute;
finally
  OpenDialog.Free;
end;

进入

var OpenDialog := TOpenDialog.Create(nil);
try
  OpenDialog.FileName := 'This is the story of a horse that met a cat that met a dog the day before.txt';
  FixOpenDialog(OpenDialog); // <-- just call this prior to Execute
  OpenDialog.Execute;
finally
  OpenDialog.Free;
end;

当然,如果您的应用程序在 20 个不同的单元中有 73 个打开的对话框,则您只需要此 OpenDialogUpgrader 单元的一个副本,但需要将其添加到实现部分 uses 条款在您的 20 个单元中的每一个单元中。并且您需要在每个TOpenDialog.Execute之前调用FixOpenDialog

关于Delphi OpenDialog文件名显示问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66538427/

相关文章:

sorting - 如何完全按照选择的顺序从 TOpenDialog 获取选定的文件?

delphi - tOpenDialog 后面意外触发 tStringGrid.OnFixedCellClick

delphi - 如何在打开对话框中设置排序模式

delphi - 对于所有具有读/写权限的用户,我在 Vista/Win7 上的什么位置存储文件

delphi - FreePascal/Delphi以文本形式将文本加密时应使用哪种加密算法?

delphi - 为什么 T 类不允许出现在泛型类型中?

delphi - 有没有办法在 delphi 中禁用 TOpenDialog 的提示?

Delphi 7 - 表单中的嵌入式文件打开对话框

delphi - 新的 Delphi Rad Studio XE2 中的 FireMonkey (FMX) 包含哪些功能的详细信息?

delphi - 如何设置 TPageControl 的当前页面?