管理员
|
楼主#
更多
发布于:2011-11-02 03:28
| | | | 6.2.7 记录的删除、插入、排序 删除一条记录的基本思路是:获取当前记录的位置并把该位置后的记录逐个向前移动。 文件在最后一条记录前截断。 for i:=CurrentRec+1 to Count-1 dobeginseek(MethodFile,i);read(MethodFile,MethodRec);seek(MethodFile,i-1);Write(MethodFile,MethodRec);end;Truncate(MethodFile); 为避免误删除,在进行删除操作前弹出一个消息框进行确认。删除后要更新全局变量的值和显示内容: Count := Count - 1;ChangeGrid; 完整的程序如下: procedure TRecFileForm.DeleteButtonClick(Sender: TObject);varNewFile: MethodFileType;MethodRec: TMethod;NewFileName: String;i: Integer;beginif FileOpened = False then Exit;CurrentRec := StringGrid1.Row-1;if CurrentRec < 0 then Exit;if MessageDlg('Delete Current Record ?', mtConfirmation,[mbYes, mbNo], 0) = idYes thenbeginHazAttr.text := '';for I := CurrentRec+1 to Count-1 dobeginseek(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);varNewFile: MethodFileType;MethodRec: TMethod;NewFileName: String;i: Integer;beginif FileOpened = False then Exit;CurrentRec := StringGrid1.Row-1;if CurrentRec < 0 then Exit;if MessageDlg('Delete Current Record ?', mtConfirmation,[mbYes, mbNo], 0) = idYes thenbeginHazAttr.text := '';NewFileName := ChangeFileExt(FileName,'.sav');tryAssignFile(NewFile,FileName);ReWrite(NewFile);ExceptOn EInOutError dobeginRename(MethodFile,FileName);Exit;end;end;for i := 1 to Count doif I <> CurrentRec+1 thenbeginMethodRec := GridToRec(i);Write(NewFile,MethodRec);end;closeFile(MethodFile);tryAssignFile(MethodFile,Filename);Reset(MethodFile);excepton EInOutError dobeginDeleteFile(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 的四个域。由于TMethod的Result域必须是[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事件处理过程中。如输入非法,则禁止对话框关闭,并将输入焦点置于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页中,DirectoryOutline在Component Palette的Samples页中。 以上文件控件再加上文件编辑框、目录标签框(事实上是一般的编辑框、标签框)就可以构成一个完整的文件操作系统。它们之间的联系几乎不用代码支持,只要设置好相应的属性就可以了。 FileEdit、DirLabel、FileListBox、FileFilterComloList、 DirectoryListBox、DriveComboList六个控件间的属性联系如下: DriveComboList .DirList := DirectoryListBox; DirectoryListBox.DirLabel := DirLabel; DirectoryListBox.FileList := FileListBox; FileFilterComboList.FileList := FileListBox; FileListBox.FileEdit := FileEdit; 以上联系可以在设计时完成。只要打开相应属性的选择列表框进行选择即可。也可以在运行时利用如上的赋值语句建立联系。 文件控件的关键属性基本上都在以上联系中反映出来了。除此之外,FileFilterComboList有一个Filter属性,用来设置组合列表框的选择项;FileListBox 有一个Mask属性,用于设置显示文件的类型,这就允许FileListBox在脱离FileFilterComboList单独应用时仍能根据需要显示特定的文件。在6.4节中我们将应用这一功能。 文件控件的方法、事件基本是从ListBox和ComboBox中继承的。但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 = FileName then begin NotInList := False; Exit; end; NotInList := True; end; 6.3.3.3 按指定匹配字符串显示当前目录中的文件 当在FileEdit中输入一个匹配字符串,并回车,文件列表框将显示匹配结果。这一功能在FileEdit的OnKeyPress事件中实现。 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 按指定匹配字符串查找当前目录中的文件 为了进行比较,我们用另一种方法来实现文件的查找功能,即利用标准过程FindFirst、FindNext。FileList1与ListBox1 中的内容完全一致。 当用户单击“查找”按钮时,与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; 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 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 主窗口主菜单包括File、WIndows、Help三项。File菜单项在子窗口生成时被子窗口同名菜单项所取代。设置Windows、Help的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) 标志硬驱 Visible=False ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 子窗口主菜单包括File、Function、Directory三个菜单项, 分别用于完成文件的基本管理功能、其它管理功能和目录管理功能。 由于对话框界面设计很简单,这里不再进行赘述。 读者可直接参考后面将给出的对话框界面图(图6.8---6.13)进行设计。 6.4.2 子窗口的创建、布置和关闭 子窗口的创建、布置由父窗口的Windows菜单控制,其菜单项如下: ● New Windows : 创建新的子窗口 ● Tile : 平铺 ● Cascade : 层叠 ● ArrangeIcon : 排列图标 ● Minimized All : 极小化所有子窗口 子窗口的创建只需要简单调用窗体的Create方法: FileMan := TFMForm.Create(Application); 子窗口的标准排列方式直接调用MDI窗口的标准方法Tile、Cascade和ArrangeIcons。 极小化所有子窗口的实现利用MDI窗口的两个属性:MDIChildCount和MDIChildren: for i := 0 to MDICount - 1 do MDIChildren.Windowstate := wsMinimized; 子窗口关闭时释放内存空间,为此在子窗口TFMForm的OnClose事件中令 Action := OnFree; 为了保持和Windows的File Manager的一致性,我们也禁止关闭最后一个子窗口,这需要在子窗口的OnCloseQuery事件处理过程中实现: If FileManager.MDIChildCount <= 1 then CanClose := False; CanClose是OnCloseQuery事件过程返回的一个参数,用于判定窗口是否可以关闭。 由于这一过程归子窗口所有,因而MDIChildCount前必须加上其对象名FileManager。 但不幸的是:这样一来我们的程序无法终止了!原来MDI窗口关闭前首先关闭其所有的子窗口。如果子窗口不能关闭,MDI窗口也不能关闭。 为此我们需要判断发出关闭消息的是子窗口的系统菜单还是菜单的Exit项。 定义一个全局变量 var ExitClick: Boolean; 在子窗口的Exit1Click事件处理过程中: ExitClick := True; FileManager.Exit1Click(Sender); 子窗口关闭前可以利用这一全局变量检测是否应关闭: If (FileManager.MDIChildCount <= 1) and (Not ExitClick) then CanClose := False; 6.4.3 文件控件的联系 在本例中我们使用了一组新的控件:TabSet、DirectoryOutline、FileListBox,用于显示和选择驱动器、目录和文件。与(6.3)中所用方法相比,使用这一组控件需要少量的代码支持。 TabSet与DirectoryOutline的联系在TabSet的Click事件处理过程中建立: With DriveTabSet do DirectoryOutline.Drive := Tabs[TabIndex][1]; DirectoryOutline与FileListBox的联系在DirectoryOutline的Change事件处理过程中建立: FileList.Directory := DirectoryOutline.Directory; FileList.Update; 6.4.4 DriveTabSet的自画风格显示 Dephi为一些控件提供了自画风格的显示,如ListBox、ComboBox、TabSet等。 在缺省情况下,这些控件自动显示文本。而在自画风格下,拥有控件的窗体在运行时间内自己画出控件的每一项目。 自画风格显示通常的应用是为项目除文本外再添加图形显示。能以自画风格显示的控件有一个共同特点:都拥有一个TStrings类型的项目链。由于TStrings类的特点(参第三章),它们都可以加入一个和对应文本相联系的对象。 而这正是自画风格显示的关键。 通常情况下产生一个自画风格需要三个步骤: 1.设置自画风格; 2.向字符串链表添加图形对象; 3.画出自画项目。 6.4.4.1 设置自画风格 控件属性Style 用于设置自画风格。对于DriveTabSet,我们把Style 属性设置为tsOwnerDraw。 对于ListBox、ComboBox等控件的设置与TabSet略有差异,读者可参阅联机帮助文档。 6.4.4.2 向字符串链表添加图形对象 1.在应用程序中添加图片部件 在本程序中我们设置了三个图片部件NetWork、Floppy、Fixed,并分别与三个位图文件NetWork.bmp、Floppy.bmp、Fixed.bmp相关联。 2.把图片添加到字符串链表中 根据字符串链表的性质,我们可以把对象与已存在的字符串建立联系,也可以同时添加字符串和对象。这里我们采用后一种方法。 在子窗口的OnCreate事件处理过程中,我们利用一个循环依次检测从a到z的驱动器是否存在以及驱动器的类型。这利用了Windwos API函数GetDrivetype, 如果驱动器不存在则返回0,否则返回驱动器的类型(DRIVE_REMOVABLE、DRIVE_FIXED、DRIVE_REMOTE)。根据驱动器类型我们可以判断与文本(驱动器名)同时添加到Tabs中的不同图形对象。在添加过程中,DriveTabSet的TabIndex被设置为当前驱动器。 程序清单如下: 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; 由于TStrings的Objects属性中存放的对象都是TObject类型,并没有Width属性,因而需要再把它转化为TBitmap类型的对象: BitmapWidth := TBitmap(DriveTabSet.Tabs.Objects[Index]).Width
| | | | |
|