• <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) 閱讀(2244) 評論(2)  編輯 收藏 引用 所屬分類: GacUI

            評論:
            # re: GacUI Demo:文本框高亮(三,完)上下文相關著色 2012-05-20 01:40 | 陳梓瀚(vczh)
            發現xml里面把C++他爹的名字寫錯了……  回復  更多評論
              
            # re: GacUI Demo:文本框高亮(三,完)上下文相關著色 2012-05-20 04:06 | iloveprogramme
            嚴重佩服啊!期待lz完成的那天。  回復  更多評論
              
            久久成人国产精品一区二区| 亚洲欧美日韩精品久久亚洲区| 久久精品视频一| 欧美亚洲色综久久精品国产| 91精品国产9l久久久久| 国内精品久久久久久中文字幕| 亚洲精品tv久久久久| 97久久香蕉国产线看观看| 99久久综合狠狠综合久久| 久久精品无码一区二区WWW| 91精品国产高清久久久久久io| 欧美日韩精品久久久免费观看| 久久精品aⅴ无码中文字字幕不卡 久久精品aⅴ无码中文字字幕重口 | 久久精品国产黑森林| 久久久黄色大片| 国产精品综合久久第一页 | AV无码久久久久不卡网站下载 | 亚洲国产欧洲综合997久久| 99久久精品这里只有精品| 波多野结衣久久精品| 国产精品久久久天天影视香蕉 | 久久久久亚洲Av无码专| 久久婷婷色综合一区二区| 久久99精品国产麻豆| 久久精品国产男包| 久久婷婷五月综合色99啪ak| 久久亚洲国产午夜精品理论片| 久久夜色精品国产噜噜噜亚洲AV| 久久伊人精品一区二区三区| 国产国产成人久久精品| 久久91精品久久91综合| 久久久高清免费视频| 久久久久久精品免费免费自慰| 亚洲精品乱码久久久久久按摩 | 久久国产精品成人免费| 国产99久久久国产精品~~牛| 国产精品伦理久久久久久| 狠狠综合久久AV一区二区三区 | 久久精品卫校国产小美女| 老司机国内精品久久久久| 99久久久精品免费观看国产|