青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

飯中淹的避難所~~~~~

偶爾來避難的地方~

  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
  94 隨筆 :: 0 文章 :: 257 評論 :: 0 Trackbacks

【簡述】

本文講述了一個簡單的平臺無關的RICHTEXT的實現方法。

這個RICHTEXT特性如下:

-          使用UTF-16作為字符編碼

-          使用行來排版,文字從左到右顯示

-          支持可獨立設置字體顏色的文字和鏈接

-          支持自定義元素用來實現圖像和動畫

 

【平臺無關】

 

平臺無關實際上是使用統一的接口來封裝不同平臺的實現方法來做到的。在RICHTEXT中使用到的平臺相關的有兩個:

1-       文字大小獲取。

2-       文字的繪制。

 

我們把它封裝到一個字體的純虛接口類中去:

class IFont {

public:

// 獲取字體的高度

virtual float GetHeight() const = 0;

// 獲取文字的橫向步進

virtual float GetCharsAdvance( const UTF16_CHAR * pChars, float * pAdvanceArray, size_t uCount ) const = 0;

// 繪制文字

virtual void DrawChars( const UTF16_CHAR * pChars, size_t uCount, float fX, float fY, unsigned long ulColor ) const = 0;

};

 

對于自定義的元素,也是一個純虛的接口類:

         class IRichTextCustomElement

         {

         public:

                   //      獲取元素寬度

                   virtual float GetWidth() const = 0;

                   //      獲取元素的高度

                   virtual float GetHeight() const = 0;

                   //      繪制元素

                   virtual void Draw( float fX, float fY ) const = 0;

         };

 

【實現】

 

1-       模塊劃分

RICHTEXT在這里劃分為兩個模塊:一個稱為RichTextDoc,用來存儲內容的,稱為文檔;一個稱為RichTextView,用來存儲表現的,稱為視圖。

 

2-       模塊實現:RichTextDoc

 

RichTextDoc主要實現了內容管理。

 

RichTextDoc內部存儲兩項內容

1)       字符

2)       元素(不同的元素類型,或者同種元素類型但屬性不同)

 

字符存儲了文字和鏈接的原始字符,而元素存儲了同屬性的一組字符、鏈接或者一個自定義元素。他們使用idxlen關聯到字符存儲中的原始字符。對于一個圖片,在字符中使用了一個空格作為占位符。

元素中同時存儲了是否作為一個段落ID,這用來描述一組元素是否在同一個段落里,這個ID為一個不為0的正整數。

 

         RichTextDoc提供了以下接口來添加內容以及訪問元素。

 

         class IRichTextDoc

         {

         public:

                   //      添加一段文本

                  virtual void AddText( const UTF16_CHAR * pText, size_t uTextLen ) = 0;

                   //      添加一個鏈接

                   virtual void AddLink( const UTF16_CHAR * pText, size_t uTextLen, unsigned long ulLinkID ) = 0;

                   //      添加一個自定義的元素

                   virtual void AddCustom( IRichTextCustomElement * pElement ) = 0;

                   //      添加一個段落

                   virtual unsigned long AddParagraph() = 0;

                   //      設置文字顏色

                   virtual void SetTextColor( unsigned long ulColor ) = 0;

                   //      設置文字字體

                   virtual void SetTextFont( IFont * pFont ) = 0;

                   //      獲取元素的數量

                   virtual void GetElementCount() const = 0;

                   //      獲取元素類型

                   //      result: -1 = 非法索引 0=文字 1=鏈接 2=自定義元素

                   virtual int GetElementType( size_t uElementIndex ) const = 0;

                   //      獲取元素的字體和顏色

                   //      result: -1 = 失敗 0=成功

                   //      pFont: 返回字體接口

                   //      ulColor: 返回顏色值

                   virtual int GetElementFontAndColor( size_t uElementIndex, IFont *& ppFont, unsigned long & ulColor ) const = 0;

                   //      獲取元素的字符

                   virtual void GetElementChars( size_t uElementIndex, const UTF16_CHAR * &pChars, size_t & uCount ) const = 0;

                   //      獲取自定義元素

                   virtual IRichTextCustomElement * GetCustomElement( size_t uElementIndex ) const = 0;

                   //      獲取元素的段落ID

                   virtual unsigned long GetElementParagraphID( size_t uElementIndex ) const = 0;

                   //      獲取元素的鏈接ID

                   virtual unsigned long GetElementLinkID( size_t uElementIndex ) const = 0;

         };

 

3-       模塊實現:RichTextView

RichTextView 主要實現了排版和繪制。

 

A        排版功能

它的基本排版單位是LINE(行),也就是顯示行。在LINE的內部存儲了數個RUN。每個RUN僅對應一個DOC中的元素,但是一個DOC中的元素可以對應多個RUN(被拆分成多行的情況)。

 

RichTextView中排版是通過拆分DOC中的每個元素實現的。因為有IFont接口以及IRichTextCustomElement接口,就可以獲取到文字和自定義元素的大小,依次累加到元素結束或者LINE寬度溢出,就可以結束一個RUN,開始下一個RUN。

 

在這個模塊的實現中,需要注意下面幾個問題:

 

1)       如何確定一個LINE的高度:在實現里,是根據每個RUN對應的元素的高度取MAX來實現的。

2)       根據段落來適時的換行。

3)       LINK根據需求來決定是否可以拆分成多個LINE中的多個RUN。(實際需求里是禁止拆分LINK

4)       行間距與RUNLINEHITTEST。

 

RUN的結構是這樣的:

struct RUN_S {

                 size_t uElementIndex;                       //      元素的索引

                 size_t uInElementCharIndex;           //      在元素的字符中的索引

                 size_t uInElementCharCount;                   //      在元素中的字符數量

                 float fWidth;                                          //      RUN的寬度

                 float fHeight;                                        //      RUN的高度

};

 

LINE 的結構是這樣的:

struct LINE_S {

                   vector<RUN_S*> vecRuns;               //      行內的RUN

                   float fPosY;                                            //      LINE在整個VIEW中的Y坐標。

                   float fHeight;                                        //      LINE的高度

};

B- 繪制功能

繪制功能和拆分排版差不多,主要就是繪制坐標根據RUNLINE的寬度和高度的累計。

然后調用IFontIRichTextCustomElement的繪制方法。

C- HITTEST

除了排版和繪制之外,VIEW還提供了HITTEST,用來檢測點擊命中了哪個LINE、RUN、或者對應到DOC中的元素,從而實現點擊鏈接的檢測。

 

RichTextView接口如下:

 

 

class IRichTextView

{

public:

           //      獲取行數

           virtual size_t GetLineCount() const = 0;

           //      獲取行的RUN數量

           virtual size_t GetRunCount( size_t uLineIndex ) const = 0;

           //      獲取RUN對應的元素索引

           virtual size_t GetRunElementIndex( size_t uLineIndex, size_t uRunIndex ) const = 0;

           //      DOC,行寬和行間距建立排版內容。

           virtual void Build( IRichTextDoc * pDoc, float fLineWidth, float fLineGap ) = 0;

           //      檢測點擊的行

           virtual size_t LineHitTest( float fX, float fY ) const = 0;

           //      檢測點擊的RUN

           virtual size_t RunHitTest( size_t uLineIndex, float fX, float fY ) const = 0;

           //      檢測點擊的元素索引

           virtual size_t ElementHitTest( float fX, float fY ) const = 0;

           //      獲取VIEW的高度。

           virtual float GetHeight() const = 0;         

           //      繪制

           virtual void Draw(float fX, float fY, float fWidth, float fHeight) const = 0;

           //      從某行開始繪制

           virtual void Draw(size_t uBeginLineIndex, float fX, float fY, float fWidth, float fHeight) const = 0;

};

 

 

【應用】

         目前應用在一個手機網游的項目中,來顯示聊天內容。

       平臺目前是IOSWIN32IOS下字體使用的是CORETEXT+COREGRAPHICS來實現的。WIN32下用的是GetGlyphOutline API。渲染使用的OPENGLES 1.1,內部用glTexSubImage2D來實現了一個字形的貼圖緩沖。

 

         在項目中,View被綁定在一個RichTextUI控件中。

 

【擴展】

         目前只能顯示富文本,后面需要擴展為RICHEDIT使用。

         需要增加光標的位置判定和光標的顯示位置和大小的獲取。

         考慮在DOC上增加存儲文字寬度,以便于VIEW上進行CHARHITTEST時的快速取用。

 

 

 

 

posted on 2012-11-07 15:48 飯中淹 閱讀(3374) 評論(0)  編輯 收藏 引用 所屬分類: 游戲客戶端 、手機開發(ios)
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲国产裸拍裸体视频在线观看乱了中文 | 亚洲黄色一区二区三区| 午夜精品久久久久久久蜜桃app| 欧美日韩国产美| 亚洲在线免费| 欧美大片在线观看一区| 国产欧美日韩在线播放| 久久精品亚洲一区| 亚洲国产欧美一区二区三区同亚洲| 免费在线观看一区二区| 欧美a级在线| 亚洲一区二区免费| 久久av一区二区三区| 国产精品系列在线播放| 久久精品视频免费| 蜜乳av另类精品一区二区| 一区二区成人精品 | 亚洲精品视频一区二区三区| 国产精品理论片| 老司机午夜精品| 欧美日韩国产在线| 久久大综合网| 欧美女同视频| 久久久久免费观看| 欧美日韩国产一区精品一区| 亚洲欧洲av一区二区三区久久| 久久精品国产一区二区三区| 亚洲免费不卡| 久久久视频精品| 亚洲香蕉伊综合在人在线视看| 久久精品国产综合| 亚洲一线二线三线久久久| 99在线视频精品| 亚洲免费在线看| 亚洲乱码国产乱码精品精98午夜| 午夜精品福利一区二区蜜股av| 亚洲日本黄色| 久久成人18免费观看| 亚洲图中文字幕| 免费一区二区三区| 久久午夜国产精品| 国产精品色午夜在线观看| 亚洲激情视频在线播放| 国产资源精品在线观看| 亚洲无玛一区| 一区二区三区欧美亚洲| 免费的成人av| 久久夜色精品一区| 国产美女精品人人做人人爽| 99国产精品一区| 一本大道久久a久久综合婷婷| 久久久久综合| 久热re这里精品视频在线6| 国产精品亚洲网站| 亚洲一区二区三区在线看| 一区二区高清视频| 欧美日韩免费观看一区| 亚洲黄色三级| 一区二区三区www| 欧美国产日本| 亚洲毛片一区二区| 一区二区三区高清在线| 欧美人与禽性xxxxx杂性| 亚洲高清在线观看一区| 亚洲高清三级视频| 欧美成人免费播放| 亚洲黄色影院| 正在播放欧美一区| 欧美日韩妖精视频| 一区二区三区.www| 亚洲欧美伊人| 国产一区二区三区在线免费观看 | 亚洲午夜小视频| 欧美一区二区三区日韩| 国产精品美女久久久久久久| 亚洲欧美中文另类| 久久久久久久波多野高潮日日| 激情成人中文字幕| 欧美激情91| 一二三区精品| 久久国产精品免费一区| 永久免费视频成人| 欧美xx视频| 亚洲网友自拍| 免费成人在线视频网站| 亚洲日韩成人| 国产精品视频一| 久久午夜精品| 亚洲图片你懂的| 快射av在线播放一区| 日韩亚洲欧美一区二区三区| 国产精品久久久久久久午夜 | 久久久久九九九| 亚洲精品美女久久7777777| 亚洲欧美日韩一区在线观看| 国产一区二区三区四区| 欧美精品尤物在线| 香蕉久久a毛片| 亚洲欧洲日本一区二区三区| 性欧美大战久久久久久久免费观看 | 欧美日精品一区视频| 午夜精品美女自拍福到在线 | 久久久久久久成人| 一本色道综合亚洲| 狠狠色2019综合网| 欧美视频不卡| 欧美大片18| 欧美在线日韩精品| 一本大道av伊人久久综合| 免费不卡亚洲欧美| 亚洲欧美综合v| 日韩一级裸体免费视频| 国产一区二区三区日韩欧美| 亚洲国产精品久久久久| 日韩视频免费观看高清在线视频 | 亚洲欧美日本另类| 亚洲激情在线激情| 国产视频综合在线| 欧美丝袜一区二区三区| 猛干欧美女孩| 美女国产精品| 久久久久国产精品午夜一区| 亚洲线精品一区二区三区八戒| 亚洲精品一区二区三区在线观看 | 久久视频这里只有精品| 校园春色国产精品| 亚洲婷婷在线| 99精品国产高清一区二区| 在线成人激情黄色| 激情久久一区| 在线观看亚洲视频啊啊啊啊| 国产亚洲制服色| 国产视频在线观看一区| 欧美午夜在线| 欧美一区二区视频97| 亚洲国产成人av在线| 欧美国产欧美亚州国产日韩mv天天看完整| 午夜免费日韩视频| 亚洲午夜国产成人av电影男同| 99热免费精品| 亚洲精品视频在线看| 激情五月婷婷综合| 国产亚洲欧洲997久久综合| 国产精品欧美精品| 欧美性猛交xxxx乱大交退制版| 欧美1级日本1级| 麻豆精品视频在线观看| 欧美亚洲午夜视频在线观看| 在线亚洲精品| 亚洲免费一区二区| 亚洲影院免费观看| 亚洲天堂av高清| av成人天堂| 在线视频亚洲欧美| 亚洲少妇诱惑| 欧美中文字幕在线观看| 亚洲女女做受ⅹxx高潮| 一区二区激情视频| 一本色道久久88亚洲综合88| 最新国产拍偷乱拍精品| 亚洲精品欧美日韩| 亚洲免费av片| 亚洲天堂激情| 亚洲一区二区三区在线看| 亚洲欧美网站| 久久久亚洲高清| 国产综合色精品一区二区三区| 午夜影视日本亚洲欧洲精品| 亚洲婷婷免费| 欧美一级黄色网| 一二三区精品| 久久精品视频播放| 亚洲欧美日韩在线观看a三区| 国模精品一区二区三区| 国内精品久久久久久| 国产精品第一页第二页第三页| 国产一区二区0| 亚洲国产国产亚洲一二三| 亚洲欧洲一区二区三区| 一本色道久久综合亚洲精品小说| 性感少妇一区| 美女久久一区| 亚洲精品久久久久久久久久久| 亚洲天堂成人在线视频| 欧美成人黄色小视频| 欧美三级不卡| 激情欧美日韩| 99综合视频| 欧美高清视频一区二区| 亚洲欧洲日韩综合二区| 亚洲欧美日韩国产一区| 久久蜜桃香蕉精品一区二区三区| 久久午夜国产精品| 国产精品va在线播放| 亚洲精品永久免费| 91久久精品国产91久久性色tv| 在线亚洲免费视频| 午夜精品在线看| 国产精品大片wwwwww| 揄拍成人国产精品视频|