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

隨筆-341  評論-2670  文章-0  trackbacks-0
    GacUI的高亮Demo做了一半。現在的進度是,可以手寫著色器的狀態轉換函數,但是自動從正則表達式產生著色器的狀態轉換函數部分還沒有集成進GacUI。這篇博客還是照舊,看圖、看Demo代碼,說著色原理。

    這次的Demo要做一個可以動態切換著色器的小程序,里面包含INI、XML和C++三種著色器。現在只實現了INI一中,手寫的。另外兩種將要通過正則表達式來做。窗口包含一個表格,來排版label、選著色器用的下拉框和一個大大的文本框。先看圖:



    在選擇了INI Colorizer之后,里面的INI內容被識別了出來并正確著色。現在我們來看Demo的代碼怎么寫。首先是窗口自己的代碼:

class TextBoxColorizerWindow : public GuiWindow
{
private:
    GuiMultilineTextBox
*                textBox;
    GuiComboBoxListControl
*                comboSelector;

    
void comboSelector_SelectedIndexChanged(GuiGraphicsComposition* sender, GuiEventArgs& arguments)
    {
        
switch(comboSelector->GetSelectedIndex())
        {
        
case 0:
            textBox
->SetColorizer(new IniColorizer);
            textBox
->SetText(
                L
";This is a comment\r\n"
                L
"[Section1]\r\n"
                L
"Name=John Smith\r\n"
                L
"ID=008\r\n"
                L
"\r\n"
                L
"[Section2]\r\n"
                L
"Name=Kyon\r\n"
                L
"ID=009\r\n"
                );
            
break;
        
default:
            textBox
->SetColorizer(0);
        }
    }
public:
    TextBoxColorizerWindow()
        :GuiWindow(GetCurrentTheme()
->CreateWindowStyle())
    {
        
this->SetText(L"Controls.TextBox.Colorizer");
        
this->GetContainerComposition()->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);

        GuiTableComposition
* table=new GuiTableComposition;
        table
->SetAlignmentToParent(Margin(0000));
        table
->SetCellPadding(2);
        table
->SetRowsAndColumns(23);
        table
->SetRowOption(0, GuiCellOption::MinSizeOption());
        table
->SetRowOption(1, GuiCellOption::PercentageOption(1.0));
        table
->SetColumnOption(0, GuiCellOption::MinSizeOption());
        table
->SetColumnOption(1, GuiCellOption::MinSizeOption());
        table
->SetColumnOption(2, GuiCellOption::PercentageOption(1.0));
        
this->GetContainerComposition()->AddChild(table);
        
        {
            GuiCellComposition
* cell=new GuiCellComposition;
            table
->AddChild(cell);
            cell
->SetSite(0011);

            GuiLabel
* label=g::NewLabel();
            label
->SetText(L"Select a colorizer: ");
            label
->GetBoundsComposition()->SetAlignmentToParent(Margin(0-100));
            cell
->AddChild(label->GetBoundsComposition());
        }
        {
            GuiCellComposition
* cell=new GuiCellComposition;
            table
->AddChild(cell);
            cell
->SetSite(0111);
            
// combo box doesn't have a minimum width, so set it to 150.
            cell->SetPreferredMinSize(Size(1500));

            
// create a text list control.
            GuiTextList* listContent=g::NewTextList();
            
// insert text items.
            listContent->GetItems().Add(L"INI colorizer");
            listContent
->GetItems().Add(L"XML colorizer");
            listContent
->GetItems().Add(L"C++ colorizer");

            
// use the text list control to create a combo box.
            
// items in the text list control will be the data source.
            
// the text list control will be displayed in the combo box dropdown.
            comboSelector=g::NewComboBox(listContent);
            comboSelector
->GetBoundsComposition()->SetAlignmentToParent(Margin(0000));
            comboSelector
->SelectedIndexChanged.AttachMethod(this&TextBoxColorizerWindow::comboSelector_SelectedIndexChanged);
            cell
->AddChild(comboSelector->GetBoundsComposition());
        }
        {
            GuiCellComposition
* cell=new GuiCellComposition;
            table
->AddChild(cell);
            cell
->SetSite(1013);

            textBox
=g::NewMultilineTextBox();
            textBox
->GetBoundsComposition()->SetAlignmentToParent(Margin(0000));
            cell
->AddChild(textBox->GetBoundsComposition());
        }

        comboSelector
->SetSelectedIndex(0);

        
// set the preferred minimum client size
        this->GetBoundsComposition()->SetPreferredMinSize(Size(640480));
        
// call this to calculate the size immediately if any indirect content in the table changes
        
// so that the window can calcaulte its correct size before calling the MoveToScreenCenter()
        this->ForceCalculateSizeImmediately();
        
// move to the screen center
        this->MoveToScreenCenter();
    }

    
~TextBoxColorizerWindow()
    {
    }
};

void GuiMain()
{
    GuiWindow
* window=new TextBoxColorizerWindow();
    GetApplication()
->Run(window);
    delete window;
}

    對于這段代碼現在已經不需要多加解釋了。首先創造了一個table,然后第一行放label和combobox,第二行撐開了放textbox。關鍵就在與combobox的SelectedIndexChanged事件。一開始是-1,在這個函數的最后我把它改成了0,這樣就出發了事件,安裝了IniColorizer著色器。然后我們來看IniColorizer的代碼:

class IniColorizer : public GuiTextBoxColorizerBase
{
    typedef collections::Array
<text::ColorEntry> ColorArray;
private:
    
static const int    NORMAL_COLOR=0;
    
static const int    SECTION_COLOR=1;
    
static const int    ATTRIBUTE_COLOR=2;
    
static const int    OPERATOR_COLOR=3;
    
static const int    COMMENT_COLOR=4;

    ColorArray            colors;
public:
    IniColorizer()
    {
        text::ColorEntry entry
=win7::Win7GetTextBoxTextColor();
        colors.Resize(
5);
        
        
// text color
        colors[NORMAL_COLOR]=entry;

        
// section color
        entry.normal.text=Color(1632121);
        colors[SECTION_COLOR]
=entry;

        
// attribute color
        entry.normal.text=Color(25500);
        colors[ATTRIBUTE_COLOR]
=entry;

        
// operator color
        entry.normal.text=Color(00255);
        colors[OPERATOR_COLOR]
=entry;

        
// operator color
        entry.normal.text=Color(2412824);
        colors[COMMENT_COLOR]
=entry;
    }

    
int GetStartState()
    {
        
return 0;
    }

    
int ColorizeLine(const wchar_t* text, unsigned __int32* colors, int length, int startState)
    {
        
if(length>0)
        {
            
if(text[0]==L';')
            {
                
for(int i=0;i<length;i++)
                {
                    colors[i]
=COMMENT_COLOR;
                }
            }
            
else if(text[0]==L'[')
            {
                
for(int i=0;i<length;i++)
                {
                    colors[i]
=(text[i]==L'[' || text[i]==L']')?OPERATOR_COLOR:SECTION_COLOR;
                }
            }
            
else
            {
                
bool afterAssign=false;
                
for(int i=0;i<length;i++)
                {
                    
if(text[i]==L'=' && !afterAssign)
                    {
                        afterAssign
=true;
                        colors[i]
=OPERATOR_COLOR;
                    }
                    
else
                    {
                        colors[i]
=afterAssign?NORMAL_COLOR:ATTRIBUTE_COLOR;
                    }
                }
            }
        }
        
return 0;
    }

    
const ColorArray& GetColors()
    {
        
return colors;
    }
};

    著色器的規格是這樣的。首先有一個起始狀態。其次對每一行進行著色的時候,文本框會把文本的內容和上一行的中介狀態傳給著色器。因為第一行并沒有“上一行”,所以第一行的起始狀態就是需要著色器自己給出的。第三個是指定所有支持的顏色。在這里ColorizeLine函數是異步的,也就是說不會在GUI線程里面運行。這樣哪怕是著色器寫得很慢,用戶看起來也就是一行一行的顏色依次出現,對于編輯文字則毫無影響。不過上面這個著色器因為相當簡單,所以他的速度其實是非常快的。我測試了一下,貼了10M的代碼進去(這個過程會很慢,因為每一行都new了兩個buffer,下個星期我會修掉他。不過一旦貼進去了,以后編輯相當流暢,這這個速度跟文本框有多少行字符是獨立的。),大概有30多萬行。在所有的文字最終都被貼進去之后,著色器會開始異步執行。黏貼之后光標直接跳到了最后,然后顏色也立刻就出來了。10M的代碼著色花的時間相當的短。

    每一次文字修改之后,著色器會從修改的那一行開始著色。如果那一行以前還沒有被著色過,則會從上一次的最后一行開始著色。這樣總是可以一邊打字,著色器一邊額外運算。在著色器開始運算的時候,會把當前著色的行復制一份,然后異步算好顏色。這個時候就開始對文本框進行加鎖,看看著色的這一行和上面的所有行是不是沒有被修改過(修改的時候記錄一個最小行號就可以了),如果有則丟掉結果從上一次修改的第一行開始著色,否則就把顏色復制進文本框。這個過程的代碼基本上如下所示:

    (下面這段代碼屬于GacUI,使用GacUI開發著色器的時候,只要直接從GuiTextBoxColorizerBase繼承即可)
            void GuiTextBoxColorizerBase::ColorizerThreadProc(void* argument)
            {
                GuiTextBoxColorizerBase
* colorizer=(GuiTextBoxColorizerBase*)argument;
                
while(!colorizer->isFinalizing)
                {
                    
int lineIndex=-1;
                    wchar_t
* text=0;
                    unsigned __int32
* colors=0;
                    
int length=0;
                    
int startState=-1;

                    {
                        SpinLock::Scope scope(
*colorizer->elementModifyLock);
                        
if(colorizer->colorizedLineCount>=colorizer->element->GetLines().GetCount())
                        {
                            colorizer
->isColorizerRunning=false;
                            
break;
                        }

                        lineIndex
=colorizer->colorizedLineCount++;
                        TextLine
& line=colorizer->element->GetLines().GetLine(lineIndex);
                        length
=line.dataLength;
                        text
=new wchar_t[length];
                        colors
=new unsigned __int32[length];
                        memcpy(text, line.text, 
sizeof(wchar_t)*length);
                        startState
=lineIndex==0?colorizer->GetStartState():colorizer->element->GetLines().GetLine(lineIndex-1).lexerFinalState;
                    }

                    
int finalState=colorizer->ColorizeLine(text, colors, length, startState);

                    {
                        SpinLock::Scope scope(
*colorizer->elementModifyLock);
                        
if(lineIndex<colorizer->colorizedLineCount)
                        {
                            TextLine
& line=colorizer->element->GetLines().GetLine(lineIndex);
                            line.lexerFinalState
=finalState;
                            
for(int i=0;i<length;i++)
                            {
                                line.att[i].colorIndex
=colors[i];
                            }
                        }
                        delete[] text;
                        delete[] colors;
                    }
                }
                colorizer
->colorizerRunningEvent.Leave();
            }

            
void GuiTextBoxColorizerBase::StartColorizer()
            {
                
if(!isColorizerRunning)
                {
                    isColorizerRunning
=true;
                    colorizerRunningEvent.Enter();
                    ThreadPoolLite::Queue(
&GuiTextBoxColorizerBase::ColorizerThreadProc, this);
                }
            }

            
void GuiTextBoxColorizerBase::StopColorizer()
            {
                isFinalizing
=true;
                colorizerRunningEvent.Enter();
                colorizerRunningEvent.Leave();
                colorizedLineCount
=0;
                isFinalizing
=false;
            }

            GuiTextBoxColorizerBase::GuiTextBoxColorizerBase()
                :element(
0)
                ,elementModifyLock(
0)
                ,colorizedLineCount(
0)
                ,isColorizerRunning(
false)
                ,isFinalizing(
false)
            {
            }

            GuiTextBoxColorizerBase::
~GuiTextBoxColorizerBase()
            {
                StopColorizer();
            }

            
void GuiTextBoxColorizerBase::Attach(elements::GuiColorizedTextElement* _element, SpinLock& _elementModifyLock)
            {
                
if(_element)
                {
                    SpinLock::Scope scope(_elementModifyLock);
                    element
=_element;
                    elementModifyLock
=&_elementModifyLock;
                    StartColorizer();
                }
            }

            
void GuiTextBoxColorizerBase::Detach()
            {
                
if(element && elementModifyLock)
                {
                    StopColorizer();
                    SpinLock::Scope scope(
*elementModifyLock);
                    element
=0;
                    elementModifyLock
=0;
                }
            }

            
void GuiTextBoxColorizerBase::TextEditNotify(TextPos originalStart, TextPos originalEnd, const WString& originalText, TextPos inputStart, TextPos inputEnd, const WString& inputText)
            {
                
if(element && elementModifyLock)
                {
                    SpinLock::Scope scope(
*elementModifyLock);
                    
int line=originalStart.row<originalEnd.row?originalStart.row:originalEnd.row;
                    
if(colorizedLineCount>line)
                    {
                        colorizedLineCount
=line;
                    }
                    StartColorizer();
                }
            }

    由于每一次加鎖做的事情都是檢查幾個int,復制幾個buffer,花的時間相當的短,所以這里都用spinlock來做。由于spinlock的內在性質,就算按鍵盤每一次都wait了一次spinlock,其性能損失也是不可察覺的。因此整個編輯過程顯得相當的流暢。

    接下來我要繼續補充這個Demo,寫一下如何通過正則表達式綁定顏色,用GacUI輕松制造上下文無關文本框著色效果。
posted on 2012-05-11 08:58 陳梓瀚(vczh) 閱讀(3273) 評論(1)  編輯 收藏 引用 所屬分類: GacUI

評論:
# re: GacUI Demo:文本框高亮(一) 2012-05-11 22:38 | ArthasLee
強烈期待期待“通過正則表達式綁定顏色”;  回復  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美日韩情趣电影| 日韩写真在线| 欧美在线免费| 黄色成人在线网址| 久久久久国色av免费看影院| 亚洲欧美日韩系列| 国产美女精品| 久久精选视频| 欧美99在线视频观看| 91久久在线观看| 亚洲美洲欧洲综合国产一区| 欧美日韩xxxxx| 亚洲免费一级电影| 欧美一区二区三区在线免费观看| 伊人久久成人| 亚洲精品在线三区| 国产农村妇女精品| 欧美二区在线播放| 欧美日韩一区二区高清| 欧美在线观看视频一区二区三区| 久久国产成人| 亚洲精选国产| 午夜精品久久| 亚洲老司机av| 亚洲欧美成人| 亚洲精品乱码久久久久久按摩观| 一区二区三区国产在线| 狠狠色噜噜狠狠狠狠色吗综合| 欧美激情精品久久久| 国产精品国产a级| 美乳少妇欧美精品| 国产精品福利在线| 牛人盗摄一区二区三区视频| 欧美午夜一区| 欧美大片免费久久精品三p| 欧美日韩国产二区| 免费成人性网站| 国产精品乱码一区二三区小蝌蚪| 欧美高清视频一区| 国产欧美一区二区精品性色| 最新日韩在线| 国产一区二区三区高清 | 亚洲精品午夜| 香蕉av777xxx色综合一区| 日韩午夜在线观看视频| 久久超碰97人人做人人爱| 在线一区二区三区四区| 久久久九九九九| 香蕉乱码成人久久天堂爱免费| 欧美国产日本韩| 欧美超级免费视 在线| 国产一区二区三区四区三区四 | 麻豆成人综合网| 国产精品久久999| 99v久久综合狠狠综合久久| 在线观看视频免费一区二区三区| 亚洲综合色丁香婷婷六月图片| 日韩午夜在线| 欧美激情精品久久久久久免费印度| 久久―日本道色综合久久| 国产精品美女久久久| 99精品国产热久久91蜜凸| 日韩一级视频免费观看在线| 蜜臀av性久久久久蜜臀aⅴ四虎| 久久久久成人精品| 国产主播喷水一区二区| 欧美一区二区成人| 亚洲女性裸体视频| 国产精品乱看| 亚洲欧美另类中文字幕| 亚洲欧美影音先锋| 国产精品久久久久永久免费观看| 一区二区三区国产在线| 亚洲专区一二三| 国产精品一区二区久激情瑜伽| 亚洲视频自拍偷拍| 久久99在线观看| 黑人巨大精品欧美一区二区小视频| 亚洲欧美日韩区| 久久久久天天天天| 亚洲电影免费在线观看| 免费黄网站欧美| 亚洲激情自拍| 亚洲综合精品| 国产一区二区三区在线观看免费| 久久激情视频久久| 欧美成人免费全部| 一本一本久久a久久精品牛牛影视| 欧美激情一区二区三区蜜桃视频| 亚洲精选在线观看| 性做久久久久久久免费看| 韩国成人福利片在线播放| 久久久久久夜精品精品免费| 欧美国产免费| 亚洲在线黄色| 亚洲电影av| 国产精品成人va在线观看| 久久精品99无色码中文字幕| 欧美高清在线| 亚洲欧美日本日韩| 在线欧美日韩| 国产精品福利片| 久久午夜精品一区二区| 日韩一级视频免费观看在线| 欧美在线日韩精品| 亚洲日本中文字幕区| 国产精品久久| 欧美成人精品高清在线播放| 亚洲视频999| 欧美激情欧美激情在线五月| 羞羞色国产精品| 亚洲欧洲日本mm| 国产婷婷一区二区| 欧美日韩aaaaa| 久久五月婷婷丁香社区| 亚洲一区二区视频在线观看| 欧美成人午夜激情视频| 欧美一级视频一区二区| 亚洲精品色婷婷福利天堂| 国产一区二区视频在线观看| 欧美激情二区三区| 久久亚洲精选| 欧美影院一区| 亚洲在线观看视频| 99国产精品| 亚洲国产第一页| 麻豆精品一区二区av白丝在线| 午夜精品偷拍| 亚洲私人影院| 一区二区av| 亚洲每日更新| 亚洲日本中文字幕区| 在线观看国产日韩| 国产在线一区二区三区四区 | 国产精品每日更新| 欧美伦理91| 欧美成人自拍| 另类激情亚洲| 巨乳诱惑日韩免费av| 久久精品视频va| 欧美在线啊v| 久久精品国产亚洲一区二区| 亚洲欧美国产高清| 亚洲一级免费视频| 亚洲一区二区在线播放| 一本色道久久88亚洲综合88| 亚洲人成网在线播放| 亚洲精品久久久久| 亚洲日本欧美日韩高观看| 亚洲国产成人一区| 亚洲激情第一区| 日韩午夜电影av| 一区二区三区精密机械公司| 在线一区免费观看| 亚洲无线观看| 欧美在线视频一区二区三区| 欧美在线视频网站| 噜噜噜久久亚洲精品国产品小说| 老司机aⅴ在线精品导航| 乱码第一页成人| 欧美激情一区二区三区不卡| 欧美伦理影院| 国产欧美三级| 亚洲电影视频在线| 一区二区福利| 欧美一区不卡| 欧美aa国产视频| 99国内精品久久| 亚洲欧美日韩另类精品一区二区三区| 午夜精品999| 牛牛影视久久网| 国产精品成人一区二区三区夜夜夜 | 久久激情综合网| 牛牛影视久久网| 欧美性猛交一区二区三区精品| 国产农村妇女毛片精品久久麻豆| 黄色在线一区| 一本色道久久综合狠狠躁篇怎么玩| 亚洲一区二区在线| 久久久久久久久久久久久久一区| 美玉足脚交一区二区三区图片| 亚洲激情另类| 午夜精品亚洲| 欧美乱人伦中文字幕在线| 国产精品一区二区三区久久久| 一区二区亚洲欧洲国产日韩| 一本色道久久综合一区| 久久婷婷激情| 亚洲毛片av| 久久欧美中文字幕| 国产精品卡一卡二卡三| 亚洲国产美女| 欧美在线视频一区| 亚洲美女在线看| 免费黄网站欧美| 国产一区日韩二区欧美三区| 亚洲色诱最新| 亚洲福利在线观看| 久久精品一区二区三区不卡| 国产精品久久久久999|