自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

用Scala實(shí)現(xiàn)Qt QWidget對(duì)象的Eventable接口

移動(dòng)開發(fā) 后端
今天我們來講講用Scala實(shí)現(xiàn)Qt QWidget對(duì)象的Eventable接口。Scala強(qiáng)調(diào)FP,但是Qt Jambi本身是基于OOP的,事件重載需要在類里面進(jìn)行。

今天我們來講講用Scala實(shí)現(xiàn)Qt QWidget對(duì)象的Eventable接口。這個(gè)Eventable接口是我項(xiàng)目中常用的一個(gè)東西,Scala強(qiáng)調(diào)FP,但是Qt Jambi本身是基于OOP的,事件重載需要在類里面進(jìn)行。在前面展示的例子中,大家可以看到經(jīng)常會(huì)這樣展開一個(gè)類去重載:

  1. new QLabel {  
  2.   override def xxxxEvent  
  3. }  

這種聲明的方法多了其實(shí)很容易讓人覺得不規(guī)范,而且閱讀也是不易。所以我萌生了讓將js那種聲明事件風(fēng)格的代碼加入至此,js是一個(gè)可以很fp的語言,而scala也是,這不是一個(gè)很好的決定嗎?獻(xiàn)上具體的代碼:

  1. package yourporject.package  
  2.  
  3. import scala.collection.mutable.{ ArrayBuffer, HashMap }  
  4. import com.trolltech.qt.gui._  
  5. import com.trolltech.qt.core._  
  6. import com.trolltech.qt.core.QEvent  
  7. import com.trolltech.qt.QSignalEmitter._  
  8. import com.agiers.mvc.Base  
  9. /*  
  10.  * Base類里面封裝了的是對(duì)于Java和Scala既有類的方法擴(kuò)展,使用的是隱式混入的方式,不會(huì)改變對(duì)象本身。  
  11.  * 如  
  12.  * "onClick".toEventName => click  
  13.  * "中文字".encode => url encode  
  14.  * "繁體字".encodeSys => 這個(gè)是根據(jù)客戶端操作系統(tǒng)默認(rèn)的字符編碼進(jìn)行urlencode  
  15.  * "繁體字".toSimplified => 繁體轉(zhuǎn)簡體  
  16.  * "簡體字".toTraditional => 簡體轉(zhuǎn)繁體  
  17.  * "hello_world".toCamelCase => HelloWorld  
  18.  * "good guys".dump("temp.txt") => 將字符串內(nèi)容輸入到一個(gè)io文件中  
  19.  * "hello world".md5 => 將字符串md5加密  
  20.  */  
  21.  
  22. trait Eventable[T <: QWidget] extends QWidget with Base {  
  23.     
  24.   // 定義閉包的格式聲明  
  25.   // 凡是在Eventable里使用閉包的類型,應(yīng)該首先使用Fn類型  
  26.   // 修改閉包類型,應(yīng)該在此修改,而不在具體聲明的地方修改  
  27.   type Fn = EventHandle => Unit  
  28.   // 定義一個(gè)event的類型組合  
  29.   // 這個(gè)代表的實(shí)際上是String -> Fn或者(String, Fn)  
  30.   type Ev = (String, Fn)  
  31.     
  32.   /**  
  33.    * 事件接管對(duì)象  
  34.    * 用于接管聲明事件時(shí)的閉包處理,并臨時(shí)寄存該閉包中的各種狀態(tài)和變量  
  35.    * @TODO 要逐漸增加他的寄存和讀取的接口  
  36.    * @author Janpoem  
  37.    */  
  38.   sealed case class EventHandle(val widget : T, val event : QEvent) {  
  39.       
  40.     // 這個(gè)是用來獲取該widget執(zhí)行event時(shí)的狀態(tài)的  
  41.     private var _break = false 
  42.       
  43.     // 以下  
  44.     def isBreak = _break 
  45.     def isBreak_=(is : Boolean) = _break = is 
  46.     def break(fn : EventHandle => Boolean) = isBreak = fn(this)  
  47.   }  
  48.     
  49.   /**  
  50.    * 閉包的存放容器  
  51.    * 允許將閉包作為一個(gè)隊(duì)列存放,并在fire的時(shí),按照隊(duì)列先后順序執(zhí)行。  
  52.    * @author Janpoem  
  53.    */  
  54.   sealed case class FnContainer(fn : Fn) {  
  55.       
  56.     private var fns = ArrayBuffer[Fn](fn)  
  57.       
  58.     def +(fn : Fn) : this.type = {  
  59.       fns += fn  
  60.       this  
  61.     }  
  62.       
  63.     def fire(widget : T, event : QEvent) : EventHandle = {  
  64.       val handle = EventHandle(widget, event)  
  65.       fns.foreach(_(handle))  
  66.       handle  
  67.     }  
  68.   }  
  69.     
  70.   // 定義Qt標(biāo)準(zhǔn)時(shí)間類型轉(zhuǎn)換到當(dāng)前類的助記名  
  71.   // name統(tǒng)一使用小寫  
  72.   // @TODO 要不斷增加QEvent.Type的內(nèi)容  
  73.   private val _eventsMap = HashMap[QEvent.Type, String](  
  74.     QEvent.Type.Show                -> "show",  
  75.     QEvent.Type.MouseButtonPress    -> "click",  
  76.     QEvent.Type.MouseButtonDblClick -> "doubleclick",  
  77.     QEvent.Type.FocusIn             -> "focus",  
  78.     QEvent.Type.FocusOut            -> "blur",  
  79.     QEvent.Type.Enter               -> "enter",  
  80.     QEvent.Type.Leave               -> "leave"  
  81.   )  
  82.     
  83.   // 事件  
  84.   private val _events = HashMap[String, FnContainer]()  
  85.  
  86.   // 傳入Qt的QEvent.Type,獲取其在Eventable內(nèi)部的快捷助記名  
  87.   def eventType2Name(_type : QEvent.Type) : Option[String] = _eventsMap.get(_type)  
  88.     
  89.   // 裝載事件  
  90.   // w.addEvent("show", handle => { /*  */ })  
  91.   def addEvent(s : String, fn : Fn) : this.type = {  
  92.     val name = s.toEventName  
  93.     if (!this.hasEvent(name))  
  94.       _events(name) = FnContainer(fn)  
  95.     else  
  96.       _events(name) + fn  
  97.     this  
  98.   }  
  99.     
  100.   // w.addEvent("click" -> { handle => println(handle.event) })  
  101.   def addEvent(event : Ev) : thisthis.type = this.addEvent(event._1, event._2)  
  102.     
  103.   def addEvents(events : Ev*) : this.type = {  
  104.     events.foreach(this.addEvent(_))  
  105.     this  
  106.   }  
  107.     
  108.   // 判斷是否存在事件  
  109.   def hasEvent(name : String) : Boolean = _events.contains(name.toEventName)  
  110.     
  111.   // Qt事件覆蓋  
  112.   override def event(event : QEvent) : Boolean = {  
  113.     eventType2Name(event.`type`()) match {  
  114.       case Some(name) => 
  115.         if (this.hasEvent(name)) {  
  116.           val handle = _events(name).fire(this.asInstanceOf[T], event)  
  117.         }  
  118.       case _ => 
  119.     }  
  120.     super.event(event)  
  121.   }  
  122. }  

這個(gè)Eventable只是一個(gè)很初步的封裝,只是針對(duì)所有的QWidget適用,我還有好些想法,比如延時(shí)事件激活,定時(shí)事件循環(huán)。并且希望能對(duì)QObject進(jìn)行全部的適用,而對(duì)于Qt的信號(hào)槽,自然也要兼容。唉,想法太多,可惜時(shí)間太有限。先用著吧,能好像寫js一樣寫事件聲明,該知足了。

下面奉上使用的代碼:

  1. class Widget extends QWidget with Eventable[QWidget]  
  2.  
  3. val w = new Widget()  
  4. w.addEvent("onClick", handle => {  
  5.   println("單擊了!")  
  6. })  
  7. w.addEvents(  
  8.   "show" -> { handle =>  
  9.     println("窗口顯示了")  
  10.   },  
  11.   "doubleClick" -> { handle =>  
  12.     println("雙擊了!")  
  13.   }  
  14. )  
  15.  

【編輯推薦】

  1. 用Qt實(shí)現(xiàn)類似QQ截圖的工具
  2. Qt 4使用MySQL的中文問題解決方法
  3. QML教程:構(gòu)建和安裝QtComponents
  4. QML教程:Qt-Quick六大開源組件
  5. Necessitas 讓Qt應(yīng)用解脫回家的誘惑
責(zé)任編輯:佚名 來源: oschina
相關(guān)推薦

2011-06-28 11:05:19

Qt QWidget Eventable

2011-07-04 16:12:00

QT QWidget

2011-06-16 11:13:13

QtQWidget

2011-03-18 19:50:32

ScalaJVMQt

2011-03-18 19:05:31

QQQtWidget

2009-07-21 08:21:46

Scala對(duì)象相等性

2009-07-21 12:18:37

ScalaRational對(duì)象toString

2009-07-22 09:22:20

Scala工廠對(duì)象

2009-09-09 14:09:35

Scala Trait

2011-06-15 17:28:23

Qt 多視圖 架構(gòu)

2023-06-12 15:33:52

Scalafor循環(huán)語句

2009-09-09 11:14:16

Scala對(duì)象

2009-08-13 10:35:05

Scala數(shù)組排序

2010-11-17 11:31:22

Scala基礎(chǔ)面向?qū)ο?/a>Scala

2019-05-14 09:05:16

SerializablJava對(duì)象

2010-03-11 10:34:22

Scala

2009-06-22 14:26:12

ScalaXML對(duì)象

2011-06-28 11:06:16

Scala

2011-06-24 10:05:51

QT 對(duì)象 父對(duì)象

2011-06-21 18:02:14

Qt 動(dòng)態(tài) 鏈接庫
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)