如何給用戶保存下來的圖片文件命名也是個問題。我們可以設置一個全局整型變量,每當保存一個圖片文件時,就令這個變量增加1,將這個整型變量轉換成字符串做為文件名。如果指定的文件名已經存在,就要給文件重命名。最簡單的辦法就是在文件名之前(或之後)加上一個字符串(比如'new'),如果加上這個字符串後還是存在重名的文件呢?這就要用到學編程的人在一開始就學到的一個小技巧:遞歸。這個問題的解決辦法見下面的代碼:
procedure SaveToPic(APic: TJPegImage; AFileName: string);
Const PICPLUSSTR = 'new';
begin
if FileExists(AFileName) then
savetopic(ABmp, PICPLUSSTR+AFileName)
else
SaveBmpAsJpg(APic, AFileName);
end;
在實際應用的時候,還應該加上異常處理(如磁盤空間已滿,文件名過長等)。圖片的保存的基本問題已經解決,我們再來看看文字的保存。為了增強程序的靈活性,我們應該使用用戶能方便地將不同地文字保存到不同的文件。繼續沿用上面保存圖片的方式用數字做文件名嗎?當然不可以。一是因為文本文件不像圖片那樣在資源管理器中可以預覽,用戶必須打開文件才能知道文件中保存的是什麼內容,如果用戶想在一大堆“1.txt”、“2.txt”……中找自己想要的內容就太麻煩了;二是因為用戶並不要求每次復制下來的內容都保存到單一的文件中,而是要將相關的內容保存到一個文件中。我對這個問題的解決方法是這樣的:
用戶可以先復制一段文字,然後再按一個熱鍵(比如Ctrl+Alt+S,為什麼要選Ctrl+Alt+S做熱鍵呢?後面再說!),這樣用戶以後復制下的文字就保存到以用戶復制的文字做為文件名的文件中。
記得無數位大師說過:“要將用戶界面與業務邏輯分開。”好吧,就將上面的東西封裝一下,也算是我向OO邁進的第一步吧!(下面之列出了類的部分成員)
TWebPageSaver = class(TObject)
private
FImagePath: string;
FTextPath: string;
FImageCount: Integer;
FTextFileName: string;
procedure SetImagePath(const Value: string);
procedure SetTextPath(const Value: string);
public
function Save: Boolean;//result is whether the content is saved
procedure NewTextFile(AFileName:string);
property ImagePath: string read FImagePath write SetImagePath;
property TextPath: string read FTextPath write SetTextPath;
end;
在用戶界面中,當用戶按下熱鍵Ctrl+Alt+S時,就調用TWebPageSaver.NewTextFile更改文字保存的文件名FTextFileName;當收到剪貼板變化的消息時就調用TWebPageSaver.Save保存剪貼板中的內容。另外還有ImagePath、TextPath等屬性,可以由用戶來更改圖片、文字的保存路徑。
核心代碼已經完成,來做一下用戶界面吧!仿照著“Windows優化大師”我做了界面,左邊我用的是TSpeedButton組件,右邊是TNotePage組件。當用戶點擊一個TSpeedButton時,調用TNotePage.ActivePage := '頁面的代號'就可以激活相應的配置界面。這個軟件需要在後台運行,那麼就讓它在平時縮小到系統托盤吧!將程序縮小到系統托盤很容易做到,網上有很多這樣的示例代碼。我手頭有一個控件cooltray4.3可以用來實現系統托盤的功能,我就懶得自己再去寫代碼了。
軟件運行一切良好。不過一直令我耿耿於懷的就是網上那種防復制的網頁:不管你怎麼拖動鼠標,那些文字就是無法被選定。仔細想一想,既然文字能夠在IE上顯示就一定可以得到它們。在MSDN中找了半天,才找到解決方法。可以通過ShellWindows集合來代表屬於shell 的當前打開的窗口的集合,而IE就是屬於shell的一個應用程序。用CoShellWindows.Create得到當前打開的shell的接口(IShellWindows),調用接口的Count屬性得到當前打開的shell的數量,然後遍歷這些窗口,嘗試從接口中取出IWebbrowser2接口(通過ShellWindow.Item(I) as IWebbrowser2這樣的接口類型轉換方式),如果結果不為nil說明這個窗口是IE窗口。之後只要調用IWebBrowser2接口的相應方法即可得到窗口中的文字、URL、標題等內容了。
示例代碼如下:
{需要使用msHtml,SHdocvw兩個單元}
var
ShellWindow : IShellWindows;
WebBrowser : IWebBrowser2;
I, ShellWindowCount: integer;
HTMLdocument : IHtmldocument2;
URL, Title, Text:string;
begin
ShellWindow := CoShellWindows.Create;
ShellWindowCount := ShellWindow.Count;
for I := 0 to ShellWindowCount-1 do
begin
WebBrowser := ShellWindow.Item(I) as IWebbrowser2;
if WebBrowser <> nil then
begin
HTMLDocument := WebBrowser.Document as IHtmlDocument2;
URL := URL;
Title := HtmlDocument.title;
Text := HtmlDocument.body.outerText ;
ShowMessage(URL+Title+Text);
end;
end;
ShellWindow := nil;
end;
我們定義一個記錄類型:
TWebPageRecord = record
URL: string; file://保存網頁的URL
Title: string;//保存網頁的標題
Text: string; file://保存網頁的文字
end;
然後定義一個TWebPageRecord類型的數組FWebPageRecordArray,大小定位20吧(我想一般人不會打開20個以上的IE吧):
Const MAXPAGECOUNT = 20;
……
FWebPageRecordArray : array [0..MAXPAGECOUNT-1] of TWebPageRecord;
在遍歷IE窗口時,向數組中的元素的相應字段復制即可。
對這個復制防復制(好拗口呀:))網頁的功能也封裝成一個類吧!
type
TWebCracker = class(TObject)
private
FWebPageRecordArray : array [0..MAXPAGECOUNT-1] of TWebPageRecord;
FWebPageCount: Integer;
public
procedure SnapShot;
function GetWebText(AIndex:integer): string;
function GetWebTitle(AIndex:integer): string;
function GetWebURL(AIndex:integer): string;
procedure Clear;
procedure Refresh;
function GetWebPageCount: Integer;
end;
在用戶界面中,可以通過調用TWebCracker.SnapShot;來對打開的IE窗口進行遍歷,並保存到FWebPageRecordArray這個數組中。通過TWebCracker.GetWebPageCount方法可以得到FWebPageRecordArray中保存的頁面的個數,通過GetWebText、GetWebTitle、GetWebURL就可以得到指定頁面的文字、標題或是URL。
一切都已經搞定了!爽!
通過編寫這個小軟件,我是收獲頗豐呀!除了學到了上邊這些技巧外,我還有一些小的經驗,願意與大家分享:
1、為用戶著想,讓用戶舒服
用戶是上帝嘛!以那個Ctrl+Alt+S熱鍵來說吧:一般用戶上網都是右手握鼠標,空下來的只有左手。小拇指按Ctrl,大拇指按Alt,食指剛好能按到S鍵,不費一點力氣!
2、 良好的編碼習慣
(1)不要出現魔術數
以TWebCracker定義的那個FWebPageRecordArray數組來說:
Const MAXPAGECOUNT = 20;
……
FWebPageRecordArray : array [0..MAXPAGECOUNT-1] of TWebPageRecord;
別人一看MAXPAGECOUNT就知道是什麼意思,而如果你寫成:
FWebPageRecordArray : array [0..19] of TWebPageRecord;
估計除了你自己沒有人能夠知道19到底是什麼意思。
(2)用sender的方式增強代碼的健壯性
procedure TMainfrm.CBAutoRunClick(Sender: TObject);
Const
SIGNINREGISTRY = 'WebSuction';
begin
if (Sender as TCheckBox).Checked then
AddToAutoRun(Application.ExeName,SIGNINREGISTRY)
else DelAutoRun(SIGNINREGISTRY);
end;
這樣即使Checkbox1改了名字也不怕。
又如:
procedure TMainfrm.N1Click(Sender: TObject);
begin
if (Sender as TMenuItem).Caption = '暫停(&S)' then
begin
(Sender as TMenuItem).Caption := '開始(&R)';
FWebPageSaver.Pause;
end
else
begin
(Sender as TMenuItem).Caption := '暫停(&S)';
FWebPageSaver.ReStart;
end;
end;
(3)不要直接使用Tform2單元的全局Form2變量,那樣就破壞了封裝性
procedure TMainfrm.SBNextClick(Sender: TObject);
var
LSelectedIndex : integer;
FormDisplay : Tform2;
begin
LSelectedIndex := LBWebPage.ItemIndex;
if LSelectedIndex <> -1 then
begin
FormDisplay := Tform2.Create(self);
FormDisplay.SetContent(FWebCracker.GetWebText(LSelectedIndex));
FormDisplay.Show;
end;
end;
在TForm2中定義 SetContent方法
procedure TWebCrackfrm.SetContent(AText:string);
begin
Memo.Clear;
Memo.Lines.Add(AText);
end;