• <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>
            隨筆-341  評論-2670  文章-0  trackbacks-0
                這是GacUI關于文本框高亮的最后一個Demo了。這個Demo是關于XML著色的。XML著色比起C++著色更難,主要是因為在類似<book name="C++ Primer">這樣的代碼里面,book和name的顏色不一樣,<和book的顏色也不一樣(參考Visual Studio)。這種時候單純依靠正則表達式來區分顏色是不夠的,我們還需要引入一個新的狀態機。這個狀態機只有三個狀態,用來區分tag name,attribute name和text三種顏色。狀態機是手寫的,并且GacUI提供了一個回調來寫這個只有寥寥幾行狀態機。先看圖:



                跟C++一樣,XML著色首先是由正則表達式組成的。XML的正則表達式比較簡單,只有符號、name、字符串、CData、注釋和其它一些簡單的東西:

             1 class XmlColorizer : public GuiTextBoxRegexColorizer
             2 {
             3 public:
             4     XmlColorizer()
             5     {
             6         text::ColorEntry entry=win7::Win7GetTextBoxTextColor();
             7         SetDefaultColor(entry);
             8 
             9         entry.normal.text=Color(01280);
            10         AddToken(L"/<!--([^/-]|-[^/-]|--[^>])*--/>", entry);
            11 
            12         entry.normal.text=Color(1280255);
            13         AddToken(L"/<!/[CDATA/[([^/]]|/][^/]]|/]/][^>])*/]/]/>", entry);
            14 
            15         entry.normal.text=Color(000);
            16         AddToken(L"\"[^\"]*\"", entry);
            17 
            18         entry.normal.text=Color(00255);
            19         AddToken(L"[<>=]", entry);
            20 
            21         entry.normal.text=Color(25500);
            22         AddToken(L"[a-zA-Z0-9_/-:]+", entry);
            23 
            24         entry.normal.text=Color(1632121);
            25         AddExtraToken(entry);
            26 
            27         Setup();
            28     }
            29 
            30     void ColorizeTokenContextSensitive(const wchar_t* text, vint start, vint length, vint& token, int& contextState)override
            31     {
            32         
            33     }
            34 
            35     int GetContextStartState()override
            36     {
            37         
            38     }
            39 };

                其次要對三種地方的[a-zA-Z0-9_/-:]進行著色。Tag的名字用褐色,attribute的名字用紅色,而普通文本用黑色。因此我們可以做一個狀態機,初始狀態為0。如果讀進了<,狀態變成1。1遇到了一個Tag名字之后變為2。從2開始所有的名字就只能是attribute的名字了。我們只考慮正確的情況,錯誤的代碼著色出了問題不僅沒有壞處,還可以提醒程序員有什么地方寫錯了。之后遇到了>變回0,在0的狀態下所有的東西都是普通文本,所以名字就都是黑色的。因此上面ColorizeTokenContextSensitive函數中就需要填入這個邏輯。GetContextStartState返回0,作為第一行的起始狀態。代碼如下:

                void ColorizeTokenContextSensitive(const wchar_t* text, vint start, vint length, vint& token, int& contextState)override
                {
                    
            // 0 < 1 name 2 att > 0
                    switch(token)
                    {
                    
            case 3:
                        
            if(length==1)
                        {
                            
            switch(text[start])
                            {
                            
            case '<':
                                contextState
            =1;
                                
            break;
                            
            case '>':
                                contextState
            =0;
                                
            break;
                            }
                        }
                        
            break;
                    
            case 4:
                        
            switch(contextState)
                        {
                        
            case 0:
                            token
            =-1;
                            
            break;
                        
            case 1:
                            token
            =5;
                            contextState
            =2;
                            
            break;
                        }
                        
            break;
                    }
                }

                
            int GetContextStartState()override
                {
                    
            return 0;
                }

                這個函數里面有幾個魔法數字,其實都是關于Token的編號的。構造函數里面我們使用AddToken將一個顏色關聯到正則表達式上,使用AddExtraToken創建一個沒有正則表達式關聯的顏色。所以在這個狀態機里面,所有的顏色都用Token的序號來表示。無論是使用AddToken還是AddExtraToken,第一個顏色編號為0,第二個顏色編號為1。因此case 3指的是[<>=],而case 4指的是[a-zA-Z0-9_/-:]+。而case 4里面的token=5則表示在狀態為1的時候,名字都用AddExtraToken指定的那個褐色進行染色。缺省的名字(也就是id為4的token)是紅色,所以不需要對contextState為2的時候進行處理。

                這樣我們就完成了XML的著色。GacUI接下來的幾個Demo將會是關于ListBox、ListView和TreeView的,敬請期待。
            posted on 2012-05-20 00:41 陳梓瀚(vczh) 閱讀(2250) 評論(2)  編輯 收藏 引用 所屬分類: GacUI

            評論:
            # re: GacUI Demo:文本框高亮(三,完)上下文相關著色 2012-05-20 01:40 | 陳梓瀚(vczh)
            發現xml里面把C++他爹的名字寫錯了……  回復  更多評論
              
            # re: GacUI Demo:文本框高亮(三,完)上下文相關著色 2012-05-20 04:06 | iloveprogramme
            嚴重佩服啊!期待lz完成的那天。  回復  更多評論
              
            久久天天躁夜夜躁狠狠躁2022| 久久久久国产亚洲AV麻豆| 亚洲欧美另类日本久久国产真实乱对白| 久久精品国产只有精品2020| 久久精品国产免费一区| 久久久久99精品成人片三人毛片| 亚洲国产综合久久天堂| 欧美大香线蕉线伊人久久| 无码AV波多野结衣久久| 精品久久久久久99人妻| 亚洲熟妇无码另类久久久| 亚洲嫩草影院久久精品| 久久婷婷五月综合国产尤物app| 久久国产高清字幕中文| 久久天天躁狠狠躁夜夜avapp | 精品久久久久久综合日本| 久久精品国产91久久综合麻豆自制| 日产久久强奸免费的看| 色成年激情久久综合| 性欧美大战久久久久久久久| 久久久久免费视频| 91久久精品无码一区二区毛片| 国产偷久久久精品专区| 一本一本久久a久久精品综合麻豆| 久久99国产精品久久99| 91精品国产色综合久久| 久久天天躁狠狠躁夜夜96流白浆 | 一本伊大人香蕉久久网手机| 久久综合狠狠综合久久综合88| 久久亚洲精品成人无码网站| 久久久WWW免费人成精品| 久久精品国产秦先生| 一本大道久久a久久精品综合| 国产精品禁18久久久夂久| 少妇内射兰兰久久| 无码国内精品久久人妻蜜桃| 精品多毛少妇人妻AV免费久久| 精品久久久久成人码免费动漫| 性做久久久久久久久久久| 久久午夜无码鲁丝片午夜精品| 久久久久久久久久免免费精品|