文件管理(二)

  • 来源: 互联网 作者: 若水   2008-03-20/14:31
  • 6.2.7 记录的删除、插入、排序

    删除一条记录的基本思路是:获取当前记录的位置并把该位置后的记录逐个向前移动。 文件在最后一条记录前截断。

    for i:=CurrentRec+1 to Count-1 do

    begin

    seek(MethodFile,i);

    read(MethodFile,MethodRec);

    seek(MethodFile,i-1);

    Write(MethodFile,MethodRec);

    end;

    Truncate(MethodFile);

              为避免误删除,在进行删除操作前弹出一个消息框进行确认。删除后要更新全局变量的值和显示内容:

    Count := Count - 1;

    ChangeGrid;

               完整的程序如下:

    procedure TRecFileForm.DeleteButtonClick(Sender: TObject);

    var

    NewFile: MethodFileType;

    MethodRec: TMethod;

    NewFileName: String;

    i: Integer;

    begin

    if FileOpened = False then Exit;

    CurrentRec := StringGrid1.Row-1;

    if CurrentRec < 0 then Exit;

    if MessageDlg('Delete Current Record ?', mtConfirmation,

    [mbYes, mbNo], 0) = idYes then

    begin

    HazAttr.text := '';

    for I := CurrentRec+1 to Count-1 do

    begin

    seek(MethodFile,i);

    read(MethodFile,MethodRec);

    seek(MethodFile,i-1);

    Write(MethodFile,MethodRec);

    end;

    Truncate(MethodFile);

    Count := Count-1;

    ChangeGrid;

    end;

    end;

    这里所显示的删除操作简单明了。但在程序开始设计时我却走了一条弯路,后来发现虽然这种方法用于记录的删除操作显得笨拙、可笑,但却恰恰是记录插入、排序的思想。

    这种思想的核心是创建一个新文件保存更新后的内容。若新文件顺利创建,则删除原文件,否则恢复原来的文件。程序清单如下:

    procedure TRecFileForm.DeleteButtonClick(Sender: TObject);

    var

    NewFile: MethodFileType;

    MethodRec: TMethod;

    NewFileName: String;

    i: Integer;

    begin

    if FileOpened = False then Exit;

    CurrentRec := StringGrid1.Row-1;

    if CurrentRec < 0 then Exit;

    if MessageDlg('Delete Current Record ?', mtConfirmation,

    [mbYes, mbNo], 0) = idYes then

    begin

    HazAttr.text := '';

    NewFileName := ChangeFileExt(FileName,'.sav');

    try

    AssignFile(NewFile,FileName);

    ReWrite(NewFile);

    Except

    On EInOutError do

    begin

    Rename(MethodFile,FileName);

    Exit;

    end;

    end;

    for i := 1 to Count do

    if I <> CurrentRec+1 then

    begin

    MethodRec := GridToRec(i);

    Write(NewFile,MethodRec);

    end;

    closeFile(MethodFile);

    try

    AssignFile(MethodFile,Filename);

    Reset(MethodFile);

    except

    on EInOutError do

    begin

    DeleteFile(FileName);

    AssignFile(MethodFile,NewFileName);

    Reset(MethodFile);

    Rename(MethodFile,FileName);

    Exit;

    end;

    DeleteFile(NewFileName);

    Count:=Count-1;

    ChangeGrid;

    end;

    end;

    对于记录插入,方法基本同上。对于排序,可先将关键域读入排序,而后再按排序结果对应的记录号顺序重写文件。

    6.2.8 结果综合

    对不同方法的评估结果,可按一定的公式进行综合。当用户按下“计算”按钮时,系统进行计算并把综合结果写入HazAttr只读编辑框中。HazAttr编辑框清空。

    6.2.9 编辑对话框的输入检查

    当用户单击“增加”或“修改”按钮时系统将弹出一个编辑对话框,让用户输入或修改记录内容。其中的三个编辑框,一个组合列表框分别对应TMethod 的四个域。由于TMethodResult域必须是[0,1] 间的小数,因此当用户按OK键关闭对话框时应进行类型和范围检查。VB中我做过同样的工作,那时需要对用户输入的键码逐个进行判断。但这种方法很繁琐、很难做圆满(如不能很好地支持编辑键)。而Object Pascal提供了更好的方法。这种方法的关键就在于它的类型转换函数Val

    procedure Val(Str: String;var V; var Code: Integer)V是由Str转换成的整型或实型数。若字符串非法,则出错位置返至Code; 否则置Code为0 。字符串非法并不会引发一个转换异常。

    如果转换后的数超出了我们的范围,则显式把Code置为-1。最后统一通过检测Code是否为0来判断输入是否合法。

    我们把输入检查放在对话框的OnCloseQuery事件处理过程中。如输入非法,则禁止对话框关闭,并将输入焦点置于#p#分页标题#e#Result编辑框中。但假如用户按了Cancel按钮,则这种检查是多余的。为此定义一个布尔变量IsCancel,对话框生成时置为False。假如用户按下Cancel,则置为True,此时OnCloseQuery事件不进行输入检查。

    对话框的OnCloseQuery事件处理过程的程序清单如下:

    procedure TEditForm.FormCloseQuery(Sender: TObject; var CanClose: Boolean);

    var

    Res: Real;

    k: Integer;

    begin

    if IsCancel = False then

    begin

    val(Result.text,Res,k);

    if (Res > 1) or (Res < 0) then k := -1;

    if k <> 0 then

    begin

    MessageDlg(' 非法输入 ',mtWarning,[mbOK],0);

    Result.text := '';

    CanClose := False;

    Result.SetFocus;

    end;

    end;

    end;

    6.2.10 文件和系统的关闭

    文件关闭须调用CloseFile过程:

    CloseFile(MethodFile);

    并对系统的状态重新进行设置。

              系统关闭时首先检测当前是否有打开的文件。若有则先关闭文件。这在主窗口的OnCloseQuery事件中实现。

    实现文件关闭的程序清单如下:

    procedure TRecFileForm.CloseButtonClick(Sender: TObject);

    begin

    if FileOpened then

    begin

    CloseFile(MethodFile);

    FileOpened := False;

    ClearGrid;

    OpenButton.Enabled := True;

    NewButton.Enabled := True;

    CloseButton.Enabled := False;

    RecFileForm.Caption := FormCaption;

    end;

    end;

    实现系统关闭前检查的程序清单如下:

    procedure TRecFileForm.FormCloseQuery(Sender: TObject;

    var CanClose: Boolean);

    begin

    if FileOpened then

    closeFile(MethodFile);

    end;

    6.2.11 记录文件小结

    我们所举的例子虽然简单,但基本覆盖了记录文件操作的主要方面。这里关键问题在于灵活应用Delphi提供的文件管理函数。同时,为了保证程序的健壮性应对异常进行捕获并处理。在数据库应用技术发展的今天,记录文件的重要性也许有所下降,但对象我们这里所处理的简单问题它仍有用武之地。MDI程序。虽然对于这里的实际情况来说,似乎并无必要。

    6.3 文件控件的应用

    Delphi文件管理的最大特色是提供了一组文件操作控件。利用这些控件我们可以快速开发一个文件名浏览系统。其功能强大与其所需写代码之少所形成的强烈反差,正是Dephi生命力的体现。

    6.3.1 文件控件及其相互关系

    Delphi提供的专用文件控件如下表所示。

    表6.4 Delphi专用文件控件━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    控件名 功能

    ─────────────────────────────────────

    DriveComboBox 驱动器组合列表框。用于选择当前驱动器

    FileListBox 文件列表框。用于显示当前目录中的文件和选中当前文件

    FilterComboBox 文件类型组合列表框。用于选择显示文件的类型

    DirectoryOutline 目录树(6.4节专门介绍)

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    以上控件前四个在Component Palette( 部件选择板) 的System页中,DirectoryOutlineComponent Palette的Samples页中。(事实上是一般的编辑框、标签框) 就可以构成一个完整的文件操作系统。它们之间的联系几乎不用代码支持,只要设置好相应的属性就可以了。

    以上文件控件再加上文件编辑框、目录标签框

    FileEdit、DirLabelFileListBox、FileFilterComloList DirectoryListBox、DriveComboList六个控件间的属性联系如下:DriveComboList .DirList := DirectoryListBox;

    DirectoryListBox.DirLabel := DirLabel;

    DirectoryListBox.FileList := FileListBox;

    FileFilterComboList.FileList := FileListBox;

    FileListBox.FileEdit := FileEdit;

             以上联系可以在设计时完成。只要打开相应属性的选择列表框进行选择即可。也可以在运行时利用如上的赋值语句建立联系。

    文件控件的关键属性基本上都在以上联系中反映出来了。除此之外,FileFilterComboList有一个Filter属性,用来设置组合列表框的选择项;FileListBox 有一个Mask属性,用于设置显示文件的类型,这就允许FileListBox在脱离#p#分页标题#e#FileFilterComboList单独应用时仍能根据需要显示特定的文件。在6.4节中我们将应用这一功能。ListBoxComboBox中继承的。但FileListBox 中有一个ApplyFilePath方法很有用,我们将在后边给出其用法。

    6.3.2 文件名浏览查找系统的设计思路

    作为文件控件的应用实例,我们开发了一个简单的文件名浏览查找系统。这个系统可用于文件名的显示,把选中的文件写入列表框,并能按文件编辑框中输入的通配符对文件进行查找。

    表6.5 部件的设计

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    部件 属性 功能

    ─────────────────────────────────────

    FileCtrForm Position=poDefault 主窗口

    DirLabel 显示当前目录

    FileEdit TabOrder=0 显示当前文件/输入文件显示匹配符

    FileListBox1 FileEdit=FileEdit 显示当前目录文件

    DirectoryListBox1 DirLabel=DirLabel 显示当前驱动器目录

    FileList= FileListBox1

    DriveComboBox1 DirList= DirectoryListBox1 选择当前驱动器

    FilterComboBox1 FileList=FileListBox1 选择文件显示类型

    Filter='All Files(*.*)|*.*|

    Source Files(*.pas)|*.pas|

    Form Files(*.dfm)|*.dfm|

    Project Files(*.dpr)|*.dpr'

    ListBox1 显示选中或查找的文件

    Button1 Caption=' 查找' FileEdit 中的内容进行查找

    Button2 Caption='退出' 退出系统

    6.3.3 文件名浏览查找系统的功能和实现

    6.3.3.1 按指定后缀名显示当前目录中的文件

    实现这一功能只需要在控件间建立正确的联系即可,不需要代码支持。建立联系的方法如(6.3.1)中的介绍。

    6.3.3.2 把选中的文件添加到列表框中

    在FileListBox1的OnClick事件中:

    procedure TFileCtrForm.FileListBox1Click(Sender: TObject);

    begin

    if Searched then

    begin

    Searched := False;

    ListBox1.Items.Clear;

    Label5.Caption := 'Selected Files';

    end;

    if NotInList(ExtractFileName(FileListBox1.FileName),ListBox1.Items) then

    ListBox1.Items.Add(ExtractFileName(FileListBox1.FileName));

    end;

    Searched是一个全局变量,用于标明ListBox1当前显示内容是查找的结果还是从FileListBox1中选定的文件。

    函数NotInList用于判断待添加的字符串是否已存在于一个TStrings对象中。函数返回一个布尔型变量。NotInList的具体实现如下。

    Function TFileCtrForm.NotInList(FileName: String;Items: TStrings): Boolean;

    var

    i: Integer;

    begin

    for I := 0 to Items.Count-1 do

    if Items[i] = FileName then

    begin

    NotInList := False;

    Exit;

    end;

    NotInList := True;

    end;

    6.3.3.3 按指定匹配字符串显示当前目录中的文件

    当在FileEdit中输入一个匹配字符串,并回车,文件列表框将显示匹配结果。这一功能在FileEditOnKeyPress事件中实现。

    procedure TFileCtrForm.FileEditKeyPress(Sender: TObject; var Key: Char);

    begin

    if Key = #13 then

    begin

    FileListBox1.ApplyFilePath(FileEdit.Text);

    Key := #0;

    end;

    end;

    文件列表框提供的ApplyFilePath方法是解决这一问题的关键所在。

    6.3.3.4 按指定匹配字符串查找当前目录中的文件

    为了进行比较,我们用另一种方法来实现文件的查找功能,即利用标准过程FindFirstFindNext。FileList1ListBox1 中的内容完全一致。

            当用户单击“查找”按钮时,与FileEdit 中字符串相匹配的文件将显示在ListBox1中。下面是实现代码。

    procedure TFileCtrForm.Button1Click(Sender: TObject);

    var

    i: Integer;

    SearchRec: TSearchRec;

    begin

    Searched := True;

    Label5.Caption := 'Search Result';

    ListBox1.Items.Clear;

    FindFirst(FileEdit.text,faAnyFile,SearchRec);

    ListBox1.Items.Add(SearchRec.Name);

    Repeat

    i := FindNext(SearchRec);

    If i = 0 then

    ListBox1.Items.Add(SearchRec.Name);

    until i <> 0;

    end;

    SearchRec是一个TSearchRec类型的记录。TSearchRec的定义如下:

    TSearchRec = record

    Fill: array[1..21] of Byte;

    Attr: Byte;

    Time: Longint;

    Size: Longint;

    Name: string[12];

    end;

    在这一结构中提供了很多信息,灵活应用将给编程带来很大方便。下面我们举几个例子。

    1. 检测给定文件的大小。

    function GetFileSize(const FileName: String): LongInt;#p#分页标题#e#

    var

    SearchRec: TSearchRec;

    begin

    if FindFirst(ExpandFileName(FileName), faAnyFile, SearchRec) = 0 then

    Result := SearchRec.Size

    else

    Result := -1;

    end;

    这一程序将在下一节中应用。

    2. 获取给定文件的时间戳,事实上等价于FileAge函数。function GetFileTime(const FileName: String): Longint;

    var

    SearchRec: TSearchRec;

    begin

    if FindFirst(ExpandFileName(FileName),faAnyFile, SearchRec) = 0 then

    Result := SearchRec.Time

    else

    Result := -1;

    end;

    3. 检测文件的属性。如果文件具有某种属性,则

    SearchRec.Attr And GivenAttr > 0

    属性常量对应的值与意义如下表:

    表6.6 属性常量对应的值与意义

    ━━━━━━━━━━━━━━━━━━━━

    常量 值 描述

    ─────────────────────

    faReadOnly $01 只读文件

    faHidden $02 隐藏文件

    faSysFile $04 系统文件

    faVolumeID $08 卷标文件

    faDirectory $10 目录文件

    faArchive $20 档案文件

    faAnyFile $3F 任何文件

    ━━━━━━━━━━━━━━━━━━━━

    6.4 文件管理综合举例:文件管理器的实现

    在本章的最后,我们利用Delphi提供的文件控件和文件管理函数开发一个简单的文件管理器。虽然这一文件管理器还无法和Windows提供的文件管理器相比拟,但它也为一般的文件操作提供了足够多的功能,而且如果读者感兴趣,还可以对它做进一步的扩充。在后边的拖放操作一章中,我们就为它提供了拖放支持,使它看起来更象一个“文件管理器”。

    6.4.1 设计基本思路

    6.4.1.1 窗口设计

    文件管理器的主窗口是一个多文档界面(MDI) 。有关文件、目录的显示和文件管理功能的实现都放在子窗口中。在程序执行过程中将根据需要弹出一些完成不同操作的对话框。这些对话框都是在需要时动态生成的。表6.7给出了本程序所设计窗体的清单。6.7 FileManger窗体清单

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    窗体类 功能 用于创建该类窗体的菜单项

    ──────────────────────────────────────

    TFileManager 主窗口

    TFMForm 子窗口 Windows|New Window

    TFileAttrForm 显示文件属性 File|Properties;Function|Search

    TChangeForm 文件移动、拷贝、改名、改变 File|Move.Cope.Rename 当前目录等操作的输入对话框 Directory|change Directory

    TSearchForm 输入待查找文件的名称和路径 Function|Search

    TDiskViewForm 显示磁盘信息 Function|Disk View

    TViewDir 输入待创建的子目录 Directory|CreateDirectory

    TAboutBox 显示版权信息 Help|About

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    6.4.1.2 界面设计

    主窗口界面主要是主菜单和用于表示当前目录、当前文件的状态条。

    表6.8 主窗口界面设计

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    部件 属性 功能

    ─────────────────────────────

    FileManager Style=fsMDI 主窗口

    WindowMenu=Windows

    Position=poDefault

    MainMenu1 主菜单

    FilePanel Align=alBottom 显示当前选中文件

    BevelInner=bvLowered

    BevelWidth=2

    DirectoryPanel Align=alBottom 显示当前选中目录

    Alignment=taLeftJustify

    BevelInner=bvLowered

    BevelWidth=2

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    主窗口主菜单包括FileWIndows、Help三项。File菜单项在子窗口生成时被子窗口同名菜单项所取代。设置WindowsHelp的GroupIndex = 9,可以使子窗口生成时这两个菜单项仍存在。

    子窗口界面包括主菜单、目录树(DirectoryOutline) 、文件列表框、 用于显示驱动器的标签集(TabSet)以及三个用于显示驱动器类型的TImage部件。6.9 子窗口界面设计

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    部件 属性 功能

    ───────────────────────────────────────

    FMForm ActiveControl=DirectoryOutline 子窗口

    Position=poDefault

    Style=fsMDIChild

    MainMenu1 主菜单

    DriveTabSet Align=alTop 显示驱动器

    style=tsOwnerDraw

    DirectoryOutline Align=alLeft 显示当前驱动器的目录树

    options=[ooDrawTreeRoot,

    ooDrawFocusRect,ooStretchBitmaps]

    FileList Align=alClient 显示当前目录中的文件

    FileType=[ftReadOnly,

    ftHidden,ftSystem,ftArchive,ftNormal]

    ShowGlyphs=True

    Network(Image) Picture(Network.bmp) 标志网络驱动器

    Vsible=False

    Floppy(Image) Picture(Floppy.bmp) 标志软驱

    Visible=False

    Fixed(Image) Picture(Fixed.bmp) 标志硬驱

    #p#分页标题#e#

    Visible=False

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    子窗口主菜单包括FileFunction、Directory三个菜单项, 分别用于完成文件的基本管理功能、其它管理功能和目录管理功能。

    由于对话框界面设计很简单,这里不再进行赘述。 读者可直接参考后面将给出的对话框界面图(6.8---6.13) 进行设计。

    6.4.2 子窗口的创建、布置和关闭

    子窗口的创建、布置由父窗口的Windows菜单控制,其菜单项如下: New Windows : 创建新的子窗口

    ● Tile : 平铺

    ● Cascade : 层叠

    ● ArrangeIcon : 排列图标

    ● Minimized All : 极小化所有子窗口

    子窗口的创建只需要简单调用窗体的Create方法:FileMan := TFMForm.Create(Application);

    子窗口的标准排列方式直接调用MDI窗口的标准方法TileCascade和ArrangeIconsMDI窗口的两个属性:MDIChildCountMDIChildren:

    极小化所有子窗口的实现利用

    for i := 0 to MDICount - 1 do

    MDIChildren[i].Windowstate := wsMinimized;

    子窗口关闭时释放内存空间,为此在子窗口TFMFormOnClose事件中令

    Action := OnFree;

    为了保持和Windows的File Manager的一致性,我们也禁止关闭最后一个子窗口,这需要在子窗口的OnCloseQuery事件处理过程中实现:

    If FileManager.MDIChildCount <= 1 then

    CanClose := False;

    CanClose是OnCloseQuery事件过程返回的一个参数,用于判定窗口是否可以关闭。MDIChildCount前必须加上其对象名FileManagerMDI窗口关闭前首先关闭其所有的子窗口。如果子窗口不能关闭,MDI窗口也不能关闭。

    由于这一过程归子窗口所有,因而

    但不幸的是:这样一来我们的程序无法终止了!原来

    为此我们需要判断发出关闭消息的是子窗口的系统菜单还是菜单的Exit项。var

    定义一个全局变量

    ExitClick: Boolean;

    在子窗口的Exit1Click事件处理过程中:

    ExitClick := True;

    FileManager.Exit1Click(Sender);

    子窗口关闭前可以利用这一全局变量检测是否应关闭:

    If (FileManager.MDIChildCount <= 1) and (Not ExitClick) then

    CanClose := False;

    6.4.3 文件控件的联系

    在本例中我们使用了一组新的控件:TabSetDirectoryOutline、FileListBox,用于显示和选择驱动器、目录和文件。与(6.3)中所用方法相比,使用这一组控件需要少量的代码支持。TabSet与DirectoryOutline的联系在TabSet的Click事件处理过程中建立:With DriveTabSet do

    DirectoryOutline.Drive := Tabs[TabIndex][1];

    DirectoryOutline与FileListBox的联系在DirectoryOutlineChange事件处理过程中建立:

    FileList.Directory := DirectoryOutline.Directory;

    FileList.Update;

    6.4.4 DriveTabSet的自画风格显示

    Dephi为一些控件提供了自画风格的显示,如ListBoxComboBox、TabSet等。 在缺省情况下,这些控件自动显示文本。而在自画风格下,拥有控件的窗体在运行时间内自己画出控件的每一项目。

    自画风格显示通常的应用是为项目除文本外再添加图形显示。能以自画风格显示的控件有一个共同特点:都拥有一个TStrings类型的项目链。由于TStrings类的特点( 参第三章) ,它们都可以加入一个和对应文本相联系的对象。 而这正是自画风格显示的关键。

    通常情况下产生一个自画风格需要三个步骤:

    1.设置自画风格;

    2.向字符串链表添加图形对象;

    3. 画出自画项目。

    6.4.4.1 设置自画风格#p#分页标题#e#

    控件属性Style 用于设置自画风格。对于DriveTabSet,我们把Style 属性设置为tsOwnerDrawListBox、ComboBox等控件的设置与TabSet略有差异,读者可参阅联机帮助文档。

    6.4.4.2 向字符串链表添加图形对象

    1.在应用程序中添加图片部件

    在本程序中我们设置了三个图片部件NetWorkFloppy、Fixed,并分别与三个位图文件NetWork.bmpFloppy.bmp、Fixed.bmp相关联。2. 把图片添加到字符串链表中

    根据字符串链表的性质,我们可以把对象与已存在的字符串建立联系,也可以同时添加字符串和对象。这里我们采用后一种方法。

    在子窗口的OnCreate事件处理过程中,我们利用一个循环依次检测从az的驱动器是否存在以及驱动器的类型。这利用了Windwos API函数GetDrivetype, 如果驱动器不存在则返回0 ,否则返回驱动器的类型(DRIVE_REMOVABLEDRIVE_FIXED、DRIVE_REMOTE) 。根据驱动器类型我们可以判断与文本(驱动器名)同时添加到Tabs中的不同图形对象。在添加过程中,DriveTabSetTabIndex被设置为当前驱动器。

    程序清单如下:

    procedure TFMForm.FormCreate(Sender: TObject);

    var

    Drive, AddedIndex: Integer;

    DriveLetter: Char;

    begin

    for Drive := 0 to 25 do

    begin

    DriveLetter := Chr(Drive + ord('a'));

    case GetDrivetype(Drive) of

    DRIVE_REMOVABLE:

    AddedIndex := DriveTabSet.Tabs.AddObject(DriveLetter, Floppy.Picture.Graphic);

    DRIVE_FIXED:

    AddedIndex := DriveTabSet.Tabs.AddObject(DriveLetter, Fixed.Picture.Graphic);

    DRIVE_REMOTE:

    AddedIndex := DriveTabSet.Tabs.AddObject(DriveLetter, Network.Picture.Graphic);

    end;

    if UpCase(DriveLetter) = UpCase(FileList.Drive) then

    DriveTabSet.TAbIndex := AddedIndex;

    end;

    end;

    6.4.4.3 画出自画项目

    当把一个控件的风格设置为自画时,Windows不再负责往屏幕上画出控件的项目,而是为每个可见项目产生自画事件。应用程序可以通过处理自画事件画出控件的项目。

    1.确定自画项目的大小

    对于TabSet而言,这在OnMeasureTab事件处理过程中完成。我们需要把DriveTabSet每个标签的宽度增大到足以同时放下文本和位图。

    procedure TFMForm.DriveTabSetMeasureTab(Sender: TObject; Index: Integer;

    var TabWidth: Integer);

    var

    BitmapWidth: Integer;

    begin

    BitmapWidth := TBitmap(DriveTabSet.Tabs.Objects[Index]).Width;

    Inc(TabWidth, 2 + BitmapWidth);

    end;

    由于TStringsObjects属性中存放的对象都是TObject类型,并没有Width属性,因而需要再把它转化为TBitmap类型的对象: BitmapWidth := TBitmap(DriveTabSet.Tabs.Objects[Index]).Width;

    对于

    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

    文件控件的方法、事件基本是从

    这里所举的例子一次只能处理一个文件。但读者可以很容易把它改为一个

    为保证结果显示的正确性,每次增加、修改、删除操作确认后


    评论 {{userinfo.comments}}

    {{money}}

    {{question.question}}

    A {{question.A}}
    B {{question.B}}
    C {{question.C}}
    D {{question.D}}
    提交

    驱动号 更多