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

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

偶爾來避難的地方~

  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>
            日韩视频精品| 国产精品久久综合| 欧美成人性生活| 欧美在线视频一区| 午夜精品久久久久久久99热浪潮 | 一本色道久久综合亚洲精品小说| 国产精品私房写真福利视频| 欧美午夜一区二区三区免费大片| 欧美精品福利| 欧美日韩日日夜夜| 国产精品久久久一本精品| 欧美午夜精品久久久久免费视 | 麻豆av一区二区三区久久| 久久精品中文| 久久一区亚洲| 欧美日本不卡高清| 国产美女高潮久久白浆| 狠狠色2019综合网| 亚洲精品乱码久久久久久蜜桃91| 日韩一级裸体免费视频| 亚洲影视在线播放| 久久精品99久久香蕉国产色戒| 久久久久久久久久看片| 玖玖国产精品视频| 亚洲欧洲日韩女同| 亚洲高清不卡| 亚洲欧美日韩精品综合在线观看| 久久久国产亚洲精品| 欧美精品一区二区精品网| 国产精品激情电影| 亚洲国产日韩欧美| 亚洲欧洲av一区二区| 欧美va日韩va| 日韩亚洲欧美中文三级| 午夜精品免费视频| 免费的成人av| 国产区精品视频| 亚洲电影在线| 欧美一区二区三区免费在线看| 欧美激情中文字幕一区二区| 亚洲欧美日韩网| 欧美日韩国产色视频| 欧美三区免费完整视频在线观看| 亚洲人成久久| 久久精品九九| 洋洋av久久久久久久一区| 久久久久国产精品一区三寸| 欧美视频一区在线观看| 亚洲高清免费视频| 久久精品夜色噜噜亚洲a∨ | 亚洲一区二区三区精品在线观看| 美女福利精品视频| 韩国在线一区| 久久动漫亚洲| 午夜一区不卡| 国产精品欧美在线| 亚洲无线观看| 亚洲美女免费精品视频在线观看| 久久亚洲一区| 在线观看视频一区二区欧美日韩| 久久成人精品一区二区三区| 一本色道久久综合| 欧美色欧美亚洲高清在线视频| 91久久极品少妇xxxxⅹ软件| 久久久久久有精品国产| 欧美中文字幕在线播放| 国产日韩一区二区三区在线播放| 香蕉久久国产| 亚洲欧美韩国| 国产一区二区三区四区五区美女| 欧美亚洲尤物久久| 香蕉久久精品日日躁夜夜躁| 国产日产精品一区二区三区四区的观看方式 | 亚洲午夜精品国产| 9色精品在线| 欧美日韩专区| 小嫩嫩精品导航| 欧美一区二区三区视频在线| 韩日欧美一区二区| 欧美国产日韩一二三区| 欧美黄色网络| 亚洲欧美日韩综合aⅴ视频| 亚洲影院色在线观看免费| 国产婷婷成人久久av免费高清| 久久精品噜噜噜成人av农村| 久久久久久尹人网香蕉| 亚洲精品影视| 制服丝袜亚洲播放| 国产综合在线看| 亚洲国产成人av| 欧美丝袜一区二区| 久久久久这里只有精品| 牛夜精品久久久久久久99黑人 | 亚洲乱码一区二区| 国产精品久久久久久久久搜平片| 中文一区字幕| 欧美一区二区三区成人 | 日韩午夜免费视频| 日韩一级视频免费观看在线| 国产精品亚洲精品| 嫩模写真一区二区三区三州| 欧美日韩视频在线| 久久人人97超碰精品888| 欧美久久视频| 久久影院亚洲| 欧美日韩一区二区在线| 久久久久久久久一区二区| 欧美国内亚洲| 久久久久久综合网天天| 欧美日韩高清一区| 久久久xxx| 欧美日韩视频| 欧美黑人多人双交| 国产日韩欧美综合一区| 亚洲精品乱码| 影音先锋国产精品| 亚洲免费影院| 亚洲色图在线视频| 免费亚洲电影| 久久婷婷影院| 国产精品久久一级| 亚洲电影自拍| 在线免费观看日韩欧美| 亚洲欧美综合| av成人国产| 欧美成人国产va精品日本一级| 亚洲一区二区三区免费观看| 欧美大色视频| 欧美激情国产日韩| 在线精品亚洲一区二区| 欧美在线视频观看| 欧美专区在线| 国产日本欧美在线观看| 亚洲综合电影| 欧美一区二区三区在线| 国产精品卡一卡二卡三| 一本色道久久综合一区| 99精品视频网| 欧美国产综合一区二区| 亚洲国产欧美在线 | 欧美一区二区在线观看| 国产精品久久久久久影视 | 99精品视频免费在线观看| 蜜臀91精品一区二区三区| 久热re这里精品视频在线6| 国产网站欧美日韩免费精品在线观看| 亚洲一区免费网站| 欧美一级大片在线观看| 国产精品影视天天线| 午夜久久一区| 麻豆精品在线观看| 亚洲欧洲日本专区| 欧美久久久久久久久久| 亚洲精品日韩精品| 亚洲人线精品午夜| 欧美第一黄网免费网站| 国产区精品视频| 亚洲精品国产欧美| 在线欧美不卡| 久久久久国内| 欧美91福利在线观看| 亚洲第一偷拍| 欧美成人一区二区三区片免费| 欧美va亚洲va香蕉在线| 亚洲激情女人| 欧美日韩99| 亚洲免费观看高清完整版在线观看熊 | 一本一本久久| 亚洲欧美国产精品桃花| 国产精品美女视频网站| 亚洲综合视频网| 久久久久久久一区二区三区| 一区二区在线视频播放| 欧美+亚洲+精品+三区| 99国产一区二区三精品乱码| 欧美亚洲午夜视频在线观看| 在线观看91精品国产麻豆| 欧美日本一区| 久久激情综合网| 最新高清无码专区| 欧美在线观看网址综合| 亚洲国产裸拍裸体视频在线观看乱了中文 | 亚洲欧美国产一区二区三区| 久久色中文字幕| 在线综合欧美| 在线成人欧美| 国产精品一区二区三区免费观看| 久久亚洲捆绑美女| 亚洲欧美另类国产| 亚洲人成人99网站| 久久久久国产精品午夜一区| 亚洲免费av观看| 国内自拍视频一区二区三区| 欧美日本一区二区三区| 久久婷婷综合激情| 亚洲欧美经典视频| 亚洲国产成人久久综合| 欧美中文字幕在线| 亚洲综合99| 一区二区三区视频免费在线观看|