這是今天csdn的咨詢報道,我覺得非常實用,而且之前沒有關注過shell還有圖形的潛質,所以特此作個筆記。
http://sd.csdn.net/a/20110420/296194.html ,作者:Martin Streicher
對話UNIX:使用shell腳本創建好的圖形應用程序
簡介: 命令行不適合于每一位用戶。事實上,一些用戶可能僅在握著鼠標時才感到舒服。要僅使用 shell 來滿足這些用戶或構建桌面應用程序,可以向您的腳本添加一些 GUI。這里是一些具體做法。
如果您走進一個擁擠的機房,可能會聽到有關 “shebangs”、斜線、點、根、管道、端口等等這個那個的閑聊。如果講到 UNIX?,您無疑會理解本地術語 — 有關 UNIX 的縮略詞、命令名、快捷鍵、選項、文件名和方言 — 且有賓至如歸的感覺。與其他藝術工作者一樣,UINX 用戶擁有廣泛的術語來描述其工作細節。
并非每個人都探討 UNIX;事實上,有些人可能發現命令行很復雜,令人卻步。此外,您可能不希望將全部命令行寄托給臨時或無經驗的用戶。要幫助那些不習慣使用命令行的人,或構建圍繞 shell 的自定義解決方案,您可以為您的腳本構建 GUI。有了這樣的工具 — dialog 和 Zenity 是兩個值得一提的工具(參見 參考資料 ) — 您就可以使用對話框、文件瀏覽器和其他常見的 “windowing” 控件和技術來與您的用戶交互。事實上,對話框提供更多自然對話:您提出問題,請求響應,并相應地予以響應。
本期的 “對話 UNIX” 探討 dialog 和 Zenity,并展示如何將任何腳本轉化成一個令人信服的 GUI 應用程序。對于傳統的、基于文本的界面使用 dialog,Zenity 提供現代風格的視窗化桌面。
向任何 shell 腳本添加對話框
一個命令行實用程序通常提供足夠的選項來完全控制每個調用。一些 DOS 命令可能啟用或禁用一個特性,而其他 DOS 命令可能處理參數,比如名稱列表。在命令行,您將(幾乎)所有信息呈現在前面,然后執行任務。圖形應用程序很不同。選擇是通過菜單、復選框和文件瀏覽器做出的。一個圖形應用程序接受一點信息,處理它,然后通常要求獲得更多信息。據說 GUI 應用程序是事件驅動的。
dialog 實用程序跨越兩個世界。當您需要來自用戶的輸入時調用該實用程序,然后返回到您的腳本繼續處理提供的任何數據。換言之,如果您寫一個腳本來使用 dialog,就有可能忽略命令行參數,而是使用 dialog 在必要時發出提示信息。
如果您的系統缺少 dialog 實用程序,您可以輕松使用當前版本自帶的包管理器來安裝它,或者您可以直接通過源代碼編譯它。例如,如果您的系統使用 Aptitude,您可以通過如下命令安裝 dialog:
- sudoapt-getinstalldialog
否則就要通過源代碼編譯,可以下載維護人員 Thomas Dickey 的 Web 站點上的代碼(參見 參考資料 )并運行典型的三個命令:./configure && make && make install:
- $wgethttp: //invisible-island.net/datafiles/release/dialog.tar.gz
- $tarxzfdialog.tar.gz
- $cddialog-1.1-20100428
- $./configure
- $make
- $sudomakeinstall
安裝完成之后,您的路徑中應當會有一個名為 dialog 的新實用程序。輸入 man dialog 來查看捆綁文檔。
dialog 使用起來很簡單:它僅是另一個 UNIX 命令。您使用命令選項顯示您選擇的對話框,然后捕獲結果并基于該值執行一些邏輯。dialog 的一些變體直接將命令結果放在特殊的 shell 狀態變量 $? 中,您應當在 dialog 命令退出后立即保存或詢問該變量(因為隨后的一個命令會立即改變其值)。另外,通常更為復雜的 dialog 命令變體同時設置 shell 狀態變量并生成其他結果。為將事情簡單化,dialog 提供 --stdout 選項來將其結果發出到標準輸出,因而便于通過命令求值捕獲數據(帶左引號的命令和賦值語句的組合)。
例如,dialog --yesno 命令是最簡單的變體之一。它提出一個問題,提示做出是或否的響應,并返回 $? 中的 0 或 1,具體取決于用戶選擇了 “Yes” 還是 “No”。您可以測試 $? 的值并執行一些條件代碼。這里是您可以添加到 shell 腳本的一個工作代碼段:
- dialog--yesno "Doyouwanttocontinue?" 00
- rc=$?
- if [ "${rc}" == "0" ];then
- echoYes
- else
- echoNo
- fi
--yesno 選項需要至少三個參數:問題文本以及對話框本身的高度和寬度,后者用行和列度量。如果您不需要特定尺寸,總是可以為高度或寬度使用 0,以自動調整對話框大小。(還有相對于窗口左下角放置窗口的選項。)圖 1 展示運行中的 --yesno。
圖 1. --yesno 操作
dialog 選項 --calendar 呈現一個日歷來允許用戶選擇特定日期。如果用戶選擇一個日期,然后單擊 OK,命令返回 0。但是,如果用戶單擊 Cancel,命令返回 1。此外,如果用戶單擊 OK,命令將選定日期發出為標準輸出。這里是使用命令求值產生日期的一個例子:
- RESULT=`dialog--stdout--title "CALENDAR"
- --calendar "Pleasechooseadate..." 00912010`
- retval=$?
--title 選項使用下一個參數來將一個標題添加到對話框,且可用于任何 dialog 命令。非常像 --yesno,您提供一些文本來提示用戶。接下來,選項 0 0 再次指定自動高度和寬度,選項 9 1 2010 分別指示日歷中顯示的初始日、月和年。選項卡和箭頭鍵改變日歷并選擇一個日期。對話框退出后,如果 retval 是 0,RESULT 的值就是選定的日期。圖 2 顯示日歷對話框。
圖 2. 日歷對話框
dialog 命令提供通常在圖形應用程序中找到的大部分控件:
--infobox 僅僅展示信息:它不要求任何輸入。信息框仍然只是簡單地在屏幕上。要延長其顯示,在它和下一個命令之前置入一個 sleep 命令。
--input 收集單一輸入響應。您可能會使用該命令來收集您的用戶的姓名或郵政編碼。
--textbox 顯示一個文本文件的內容。如果文件超出對話框的垂直高度,一個控件支持簡單的向上和向下滾動。
--menu 和 --radiolist 提供一個選擇列表,供用戶進行選擇。兩種對話框在功能上是等同的,但是略有不同的視覺風格,以更好地模擬一個 GUI 可能展示的東西。特別地,--radiolist 命令呈現 ( ) 來模擬單選按鈕。
--checklist 顯示用戶可單獨啟用或禁用的一個項目列表。
每個 dialog 變體的輸出不同,或是一個單一值,或是一列由空格分隔的帶引號值。例如,--checklist 是用于選擇一個或多個選項的一個不錯的控件,它發出一列帶引號值,其中每個值與一個啟用的選項相關。下面演示了一個操作示例:
- RESULT=`dialog--stdout
- --checklist "Enabletheaccountoptionsyouwant:" 10403/
- 1 "Homedirectory" on/
- 2 "Signaturefile" off/
- 3 "Simplepassword" off`
行 1、2 和 3 結尾的反斜杠(/)是延續標記;從 RESULT 到 off` 的一切內容是一個命令。如果用戶啟用了 Home directory 和 Simple password,$RESULT 將會是 "1" "3"。--checklist 的參數是高度和寬度,任何時間內的列表元數量(如果有些項目被擋住,您可以通過滾動查看這些項目),以及清單選項(其中每個選項是一個值)、一個描述、在最初啟用或禁用該選項。
您可以隨時輸入 dialog --help 來查看常規列表,輸入 dialog 來查看特定選項。dialog 有無數用法。
有像素?使用 Zenity。
Zenity 是 UNIX 桌面,如同 dialog 是簡單的終端窗口。您可以使用 Zenity 從任何 shell 腳本打開 GTK+ 對話框。事實上,Zenity 與 dialog 有著許多相同的功能;惟一的區別在于,Zenity 在一個 X Window System 環境中工作。Zenity 與 GNOME 相捆綁。如果您不運行 GNOME,可以單獨安裝 Zenity(但是,也要安裝大量 GTK+ 庫)。您還可以從 GNOME 項目頁面下載 Zenity 的源代碼(參見 參考資料 獲取鏈接)。
下面是一個簡單的例子。命令為:
- zenity--question--text "Doyouwanttocontinue?"
生成的結果如 圖 3 所示。(用于演示的機器在運行 Ubuntu 10。)如果您單擊 OK,命令返回 0。否則,它返回 1。
圖 3. 一個簡單問題
如同 dialog,Zenity 有很多選項 — 甚至比 dialog 還多 — 但是選項命名貼切,因而不言自明。您可能發現 Zenity 比 dialog 更有優勢,特別是由于大部分計算機用戶都有某種 X 桌面。
Zenity 提供與 dialog 相同的許多控件。這里是收集名稱的一個代碼段:
- ENTRY=`zenity--entry--text "Pleaseenteryourname"
- --entry-text "Yourname" --title "Enteryourname"
- if [$?==0];then
- zenity--info--text "Hello$ENTRY/!"
- fi
再次說明,如果 zenity 的退出代碼是 0,那么 ENTRY 有某人的姓名。這里是為使用 Zenity 而重寫后的日歷示例:
- DATE=`zenity--calendar--day "9" --month "1" --year "2010" --format "%Y-%m-%d"
- if [$?==0];then
- echo$DATE
- fi
盡管 Zenity 更詳細一點 — 例如,對于年、月、日有單獨的選項 — 其他 DOS 命令使您免于記住精確的參數使用順序。Zenity 的日歷還允許您指定輸出格式,即使用標準 strftime() 代碼。該命令的結果類似于 2010-1-9,它表示 2010 年 1 月 9 日。
Zenity 還提供一個過程表來展示一個操作的狀態。它從標準輸入逐行讀取數據。如果一個行的前綴是井號(#),文本被更新為該行文本。如果一個行僅包含一個數字,百分比被更新為該數字。清單 1 展示 Zenity 文檔中的一個示例。
清單 1. Zenity 過程表
- #!/bin/sh
- (
- echo "10" ;sleep1
- echo "#Updatingmaillogs" ;sleep1
- echo "20" ;sleep1
- echo "#Resettingcronjobs" ;sleep1
- echo "50" ;sleep1
- echo "Thislinewilljustbeignored" ;sleep1
- echo "75" ;sleep1
- echo "#Rebootingsystem" ;sleep1
- echo "100" ;sleep1
- )|
- zenity--progress/
- --title= "UpdateSystemLogs" /
- --text= "Scanningmaillogs..." /
- --percentage=0
- if [ "$?" =-1];then
- zenity--error/
- --text= "Updatecanceled."
- fi
sub-shell(包含在括號中)執行一系列任務 — 在這個人為例子中 albeit sleep 延遲 — 且通過一個管道將輸出發出到一個 Zenity 過程表。在每一步之前,sub-shell 發出一個數字來推進過程表,每個 --percentage 0 起始于 0,然后發出一個以 # 開頭的字符串來改變狀態消息。因此,過程表沿著步驟標記腳本工作。如果 Zenity 的退出代碼是 -1,單擊的是 Cancel 按鈕。
再次說明,要使用 dialog 或 Zenity,用對話框替換您之前引用過命令行參數的代碼。用一個小創意,您可以將您的 shell 腳本轉化為一等桌面公民。
其他高級工具
有些時候,您可能發現您的需求超過了 shell 腳本以及 dialog 和 Zenity 工具的功能范圍之外。在那些實例中,您可能轉向 C/C++ 并為桌面構建本機應用程序,但是您還可以使用高級腳本語言和許多強大的 GUI 框架的語言綁定。
一個組合是 Ruby 腳本語言和 wxWidgets 框架的 Ruby 綁定。Ruby 是面向對象的、富于表現力的且簡潔的,運行于大部分操作系統之上。wxWidgets 框架還可用于每個主流平臺,包括 Mac OS X、Windows?、Linux? 和 UNIX。由于兩者都是可移植的,您可以用 Ruby 編寫一個應用程序一次,然后隨處運行它。另一個更簡單的選擇是 Shoes。盡管不如 wxWidgets 豐富,Shoes 學習和使用起來相當簡單。清單 2 使用 70 行代碼實現了一個計算器。
清單 2. 用 Shoes 實現的一個計算器
- class Calc
- definitialize
- @number=0
- @previous=nil
- @op=nil
- end
- defto_s
- @number.to_s
- end
- (0..9).each do |n|
- define_method "press_#{n}" do
- @number=@number.to_i*10+n
- end
- end
- defpress_clear
- @number=0
- end
- { 'add' => '+' , 'sub' => '-' , 'times' => '*' , 'div' => '/' }.each do |meth,op|
- define_method "press_#{meth}" do
- if @op
- press_equals
- end
- @op=op
- @previous,@number=@number,nil
- end
- end
- defpress_equals
- @number=@previous.send(@op,@number.to_i)
- @op=nil
- end
- end
- number_field=nil
- number=Calc. new
- Shoes.app:height=>250,:width=>200,:resizable=> false do
- background "#EEC" .. "#996" ,:curve=>5,:margin=>2
- stack:margin=>2 do
- stack:margin=>8 do
- number_field=parastrong(number)
- end
- flow:width=>218,:margin=>4 do
- %w(789/456*123-0Clr=+).each do |btn|
- buttonbtn,:width=>46,:height=>46 do
- method= case btn
- when/[0-9]/; 'press_' +btn
- when 'Clr' ; 'press_clear'
- when '=' ; 'press_equals'
- when '+' ; 'press_add'
- when '-' ; 'press_sub'
- when '*' ; 'press_times'
- when '/' ; 'press_div'
- end
- number.send(method)
- number_field.replacestrong(number)
- end
- end
- end
- end
- end
對 Ruby 和 Shoes 的介紹不在本文討論范圍之內,但是這里是一些最重要的構造:
大多數 Ruby 類 Calc 使用 Ruby 的元編程功能,在運行時為所有數字鍵和數學操作鍵定義功能。
代碼開頭 Shoes.app... 創建計算器的 GUI,為其呈現布局和按鈕。Shoes 提供兩個容器來裝配布局:stack 和 flow。一個 stack 是元素的一個垂直堆棧,其中每個元素直接放在前一個元素下面。一個 flow 盡量緊密地包裹元素,直至它達到其邊框局限,然后包裝其余的元素。(您可以將一個堆??醋魇且粋€ HTML <div>,將一個流看作 HTML <p>。)您可以使用 Ruby 塊創建一個堆?;蛞粋€流。
最里面的 flow 快循環創建應用程序中的所有按鈕,并有效地將每個按鈕綁定到其方法。(case 語句返回一個方法名稱;number.send(method) 行調用實例化計算器上的那個方法。)
number_field.replace strong(number) 行通過最新計算結果更新計算器顯示。發出 number 致使類調用其自己的 to_s (“to string”) 方法。
其他腳本語言擁有類似的庫,且 Ruby 本身有更多選擇,包括 Ruby Cocoa,可使用 Ruby 在 Mac OS X 上開發 Cocoa 應用程序。選擇您喜歡的開源腳本語言,找到一個輕量級 GUI 工具包,然后開始編碼。
您不需要討厭的編譯器!
如果您已經掌握了 shell 腳本編寫,將您的工作與 dialog 或 Zenity 結合起來,以增加互操作性。如果您需要的編程功能比 shell 提供的更多,考慮 Ruby 或 Python 這樣的語言以及任何窗口工具包。您無需一個編譯器來編寫良好的桌面應用程序。
關于作者
Martin Streicher 是一位 Ruby on Rails 的自由開發人員和 Linux Magazine 的前任主編。Martin 畢業于 Purdue University 并獲得計算機科學學位,從 1986 年起他一直從事 UNIX 類系統的編程工作。他喜歡收集藝術品和玩具。
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
