使用MonoDevelop開發GTK#圖形界面
Mono一直到現在的2.8已經完全可以勝任一些比較小的項目了,但相關的開發文檔與教程一直比較匱乏,中文材料更是屈指可數。雖然Mono與.net很多類庫都是通用的,但Mono仍然有為數不少的擴充類庫,其中最重要的就是對于GTK二次封裝的GTK#圖形界面類庫了,目前想要開發原生的Mono圖形界面程序此類庫必不可少。在MonoDevelop里已經支持對于GTK#界面設計的直接拖放,有相關經驗的開發人員可以很快上手。
Mono官方網站上也有一篇GTK#開發的入門文檔:Stetic GUI Designer,現在結合此教程簡要概述一下如何在MonoDevelop下使用GTK#。
首先,我們需要做的就是創建一個工程……(這是一句廢話)。在MonoDevelop主界面選擇文件->New->Solution(話說目前MonoDevelop漢化工作還未徹底完成,菜單中中文夾英文……),在彈出來的對話框選擇C#->GTK#2.0工程。
輸入想建立的工程名,我們這里用“SteticTutorial”,然后點新建。再下面的一個對話框中根據自己的需求選擇,然后就可以點擊OK了。
稍等一會之后,可以看到MonoDevelop已經幫我們做好了一個GTK#工程所需要做的最基本的工作了,點擊運行菜單里的運行按鈕或者工具欄上的齒輪按鈕就可以編譯執行當前工程了,因為我們現在什么也沒做,只是單純彈出了一個空白的窗體。
嗯嗯,很漂亮的界面……咳咳……
現在我們要往這個界面中添加一些控件,雙擊工程目錄中的MainWindow.cs,這里可以看到窗體事件的代碼。注意看MonoDevelop的下方有兩個選項,目前選中的是“源代碼”,現在讓我們點擊“設計器”。
界面右側有隱藏的工具欄,其中有常用的控件,現在我們可以使用常用的拖放操作來畫窗體了,萬歲~
好了,現在我們先不急,首先選中主窗體(MainWindow),然后再右側的屬性欄可以查看當前選中控件的屬性。
我們先把Window Tile一項改成需要的名字,這一項決定了窗體運行時標題欄上的文字。
好了,剩下的就是往窗體中拖放控件。這里GTK#與WinForms有一些不同,WinForms下直接往窗體上拖放需要使用的控件就好,而基于GTK的GTK#,大部分控件都需要放在“容器”中。容器就是專門用來盛放控件的控件,在工具欄的第一部分全是這些容器。GTK#就是依賴這些容器來進行控件的定位的,學習過Swing或者QT的同學應該很容易理解。
根據我們的需要,我們往主窗體拖放一個VBox。VBox是一個縱向排列的容器,將窗體分割為了一個個縱向的單元,可以往這些單元中添加控件或者其他容器來滿足需求。
默認的VBox有三欄,我們需要兩欄就足夠了。右擊第三個,選擇“Delete”。大部分窗體應用中菜單欄是不可或缺的一部分,所以我們首先要往窗體中VBox頂欄中添加一個MenuBar。VBox的一個很有意思的特性就是可以根據它其中的控件而改變容器的大小,所以當我們把MenuBar拖進第一欄的時候可以看到MenuBar并沒有變得像整個窗體一半大小那樣難看,而是自動將VBox縮成單行的寬度了。
接下來添加菜單選項就很簡單了,直接點擊“Click to create menu”就可以添加按鈕,還可以選擇按鈕的圖標。選中菜單,在屬性欄還可以設置Accelerator項來設定此菜單的快捷鍵。
建立完菜單后,MainWindow.cs的窗體下多了一個選項“行為”(英文叫Action),點開,看到我們之前在菜單欄中所有的定義。選中“Exit”,然后查看屬性欄,注意看屬性欄上方有一個“信號”(Signal)的選項看,選擇它。
在“Activated”項后面的Hander一欄點擊“Click here to add a new hander”,需要注意的是很多時候因為工具欄默認寬度有些窄而導致Handler項顯示不出來,我們可以拉長一下工具欄的長度。點擊后添加一個“OnExit”的事件,雙擊它進入源代碼編輯窗口。
這里添加一句代碼:
{
Application.Quit();
}
嘗試編譯運行一下,在程序菜單欄的File菜單欄中選擇Exit,看看程序是否如期關閉。如果你的代碼沒有問題的話,讓我們繼續往下走。現在VBox的下欄還空著,我們往里添加一個TextView控件,命名為logTextView。原教程中提到先要添加一個ScrolledWindow容器,再往其中填充TextView控件才能實現滾動條,但實際發現ScrolledWindow容器并非必需的,大概是GTK#版本不同的緣故吧。
好了,現在我們的程序界面已經完成,已經可以在窗體寫入文字。不過你會發現有一個小問題,TextView并不能自動換行,每當一行文字的長度超過窗體的長度時程序只是添加了一個水平的滾動條。解決這個問題的方法很簡單,只要將textview控件的Wrap Mode改為Word or Charactor就可以了。實際編程中最好將此屬性最為可選項提供為用戶。
剩下的就是在每個菜單添加相應時間代碼:
protected virtual void OnOpen(object sender, System.EventArgs e) {
// Reset the logTreeView and change the window back to original size
int width, height;
this.GetDefaultSize( out width, out height );
this.Resize( width, height );
logTextView.Buffer.Text = "";
// Create and display a fileChooserDialog
FileChooserDialog chooser = new FileChooserDialog(
"Please select a logfile to view ...", this, FileChooserAction.Open,
"Cancel", ResponseType.Cancel,
"Open", ResponseType.Accept );
if( chooser.Run() == ( int )ResponseType.Accept )
{
// Open the file for reading.
System.IO.StreamReader file =
System.IO.File.OpenText( chooser.Filename );
// Copy the contents into the logTextView logTextView.Buffer.Text = file.ReadToEnd();
// Set the MainWindow Title to the filename.
this.Title = "Nate's Log Viewer -- " + chooser.Filename.ToString();
// Make the MainWindow bigger to accomodate the text in the logTextView this.Resize( 640, 480 );
// Close the file so as to not leave a mess.
file.Close();
} // end if
chooser.Destroy();
} // end method OnOpen
For the OnClose method:
protected virtual void OnClose(object sender, System.EventArgs e) {
// Reset the logTreeView and change the window back to original size
int width, height;
this.GetDefaultSize( out width, out height );
this.Resize( width, height );
logTextView.Buffer.Text = "";
// Change the MainWindow Title back to the default.
this.Title = "Nate's Log Viewer";
} // end method OnClose
For the OnAbout method:
protected virtual void OnAbout(object sender, System.EventArgs e) {
// Create a new About dialog
AboutDialog about = new AboutDialog();
// Change the Dialog's properties to the appropriate values. about.Name = "Nate's Log Viewer";
about.Version = "1.0.0";
// Show the Dialog and pass it control
about.Run();
// Destroy the dialog
about.Destroy(); } //
end method OnAbout
其實可以更加完善,比如再新建文檔或者關閉文檔的時候檢測textview內容是否改變,提示用戶保存等等功能。
編譯運行界面: