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

隨筆-341  評論-2670  文章-0  trackbacks-0

    趁此機(jī)會做個廣告,http://www.gaclib.net終于上線啦!

    GacUI的列表控件的第二個Demo是關(guān)于列表項的多選的。跟Windows一樣,我們可以通過鼠標(biāo)和方向鍵,配合CTRL和SHIFT選擇列表的多個內(nèi)容。因此這次我實現(xiàn)了一個簡單的“名字選擇窗口”,就跟QQ邀請好友入群的界面一樣,兩個列表,兩個按鈕。先看圖:



    列表內(nèi)容始終是排序的。當(dāng)我們選中一邊的內(nèi)容之后,可以按按鈕把內(nèi)容復(fù)制到另一邊。現(xiàn)在我們先來看一看創(chuàng)建和排版這些控件的代碼。這里我用了一個五行三列的表格。左右方列表,中間的第二個第四行放按鈕,第三行64個像素高,按鈕32*32,第一行和第五行平均地充滿剩下的空間:

#include "..\..\Public\Source\GacUIIncludes.h"
#include 
<Windows.h>

// for SortedList, CopyFrom and Select
using namespace vl::collections;

int CALLBACK WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int CmdShow)
{
    
return SetupWindowsDirect2DRenderer();
}

class NameSelectorWindow : public GuiWindow
{
private:
    GuiTextList
*                    listSource;
    GuiTextList
*                    listDestination;
    GuiButton
*                        buttonAdd;
    GuiButton
*                        buttonRemove;

    (略)
public:
    NameSelectorWindow()
        :GuiWindow(GetCurrentTheme()
->CreateWindowStyle())
    {
        
this->SetText(L"Controls.ListBox.NameSelector");

        GuiTableComposition
* table=new GuiTableComposition;
        table
->SetRowsAndColumns(53);
        table
->SetCellPadding(3);
        table
->SetAlignmentToParent(Margin(0000));

        table
->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
        table
->SetRowOption(1, GuiCellOption::MinSizeOption());
        table
->SetRowOption(2, GuiCellOption::AbsoluteOption(64));
        table
->SetRowOption(3, GuiCellOption::MinSizeOption());
        table
->SetRowOption(4, GuiCellOption::PercentageOption(0.5));

        table
->SetColumnOption(0, GuiCellOption::PercentageOption(0.5));
        table
->SetColumnOption(1, GuiCellOption::MinSizeOption());
        table
->SetColumnOption(2, GuiCellOption::PercentageOption(0.5));

        
this->GetContainerComposition()->AddChild(table);

        {
            GuiCellComposition
* cell=new GuiCellComposition;
            table
->AddChild(cell);
            cell
->SetSite(0051);

            listSource
=g::NewTextList();
            listSource
->GetBoundsComposition()->SetAlignmentToParent(Margin(0000));
            
// make listSource's horizontal scroll bar disappeared when it is not needed.
            listSource->SetHorizontalAlwaysVisible(false);
            listSource
->SetMultiSelect(true);
            listSource
->SelectionChanged.AttachMethod(this&NameSelectorWindow::listSource_SelectionChanged);
            cell
->AddChild(listSource->GetBoundsComposition());
        }
        {
            GuiCellComposition
* cell=new GuiCellComposition;
            table
->AddChild(cell);
            cell
->SetSite(0251);

            listDestination
=g::NewTextList();
            listDestination
->GetBoundsComposition()->SetAlignmentToParent(Margin(0000));
            
// make listSource's horizontal scroll bar disappeared when it is not needed.
            listDestination->SetHorizontalAlwaysVisible(false);
            listDestination
->SetMultiSelect(true);
            listDestination
->SelectionChanged.AttachMethod(this&NameSelectorWindow::listDestination_SelectionChanged);
            cell
->AddChild(listDestination->GetBoundsComposition());
        }
        {
            GuiCellComposition
* cell=new GuiCellComposition;
            table
->AddChild(cell);
            cell
->SetSite(1111);

            buttonAdd
=g::NewButton();
            buttonAdd
->SetText(L">>");
            buttonAdd
->GetBoundsComposition()->SetAlignmentToParent(Margin(0000));
            buttonAdd
->GetBoundsComposition()->SetPreferredMinSize(Size(3232));
            buttonAdd
->Clicked.AttachMethod(this&NameSelectorWindow::buttonAdd_Clicked);
            buttonAdd
->SetEnabled(false);
            cell
->AddChild(buttonAdd->GetBoundsComposition());
        }
        {
            GuiCellComposition
* cell=new GuiCellComposition;
            table
->AddChild(cell);
            cell
->SetSite(3111);

            buttonRemove
=g::NewButton();
            buttonRemove
->SetText(L"<<");
            buttonRemove
->GetBoundsComposition()->SetAlignmentToParent(Margin(0000));
            buttonRemove
->GetBoundsComposition()->SetPreferredMinSize(Size(3232));
            buttonRemove
->Clicked.AttachMethod(this&NameSelectorWindow::buttonRemove_Clicked);
            buttonRemove
->SetEnabled(false);
            cell
->AddChild(buttonRemove->GetBoundsComposition());
        }

        
// Add names into listSource
        LoadNames(listSource);

        
// 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();
    }
};

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


    剩下的內(nèi)容分為三部分。首先是如何在列表沒有選中內(nèi)容的時候把按鈕變灰。在上面的代碼中我們知道listSource和listDestination都監(jiān)聽了SelectionChanged事件。事件處理函數(shù)的內(nèi)容如下:

    void listSource_SelectionChanged(GuiGraphicsComposition* sender, GuiEventArgs& arguments)
    {
        buttonAdd
->SetEnabled(listSource->GetSelectedItems().Count()>0);
    }

    
void listDestination_SelectionChanged(GuiGraphicsComposition* sender, GuiEventArgs& arguments)
    {
        buttonRemove
->SetEnabled(listDestination->GetSelectedItems().Count()>0);
    }


    代碼內(nèi)容十分簡潔明了。第二部分是NameSelectorWindow構(gòu)造函數(shù)中調(diào)用的LoadNames函數(shù)。我們首先把所有的名字都放在一個const wchar_t* DataSource[]數(shù)組里面,內(nèi)容是沒有排過序的:

const wchar_t* DataSource[]=
{
    L
"Weierstrass",
    L
"Cantor",
    L
"Bernoulli",
    L
"Fatou",
    L
"Green",
    L
"S.Lie",
    L
"Euler",
    L
"Gauss",
    L
"Sturm",
    L
"Riemann",
    L
"Neumann",
    L
"Caratheodory",
    L
"Newton",
    L
"Jordan",
    L
"Laplace",
    L
"Wiener",
    L
"Thales",
    L
"Maxwell",
    L
"Riesz",
    L
"Fourier",
    L
"Noether",
    L
"Kepler",
    L
"Kolmogorov",
    L
"Borel",
    L
"Sobolev",
    L
"Dirchlet",
    L
"Lebesgue",
    L
"Leibniz",
    L
"Abel",
    L
"Lagrange",
    L
"Ramanujan",
    L
"Ljapunov",
    L
"Holder",
    L
"Poisson",
    L
"Nikodym",
    L
"H.Hopf",
    L
"Pythagoras",
    L
"Baire",
    L
"Haar",
    L
"Fermat",
    L
"Kronecker",
    L
"E.Laudau",
    L
"Markov",
    L
"Wronski",
    L
"Zermelo",
    L
"Rouche",
    L
"Taylor",
    L
"Urysohn",
    L
"Frechet",
    L
"Picard",
    L
"Schauder",
    L
"Lipschiz",
    L
"Liouville",
    L
"Lindelof",
    L
"de Moivre",
    L
"Klein",
    L
"Bessel",
    L
"Euclid",
    L
"Kummer",
    L
"Ascoli",
    L
"Chebyschev",
    L
"Banach",
    L
"Hilbert",
    L
"Minkowski",
    L
"Hamilton",
    L
"Poincare",
    L
"Peano",
    L
"Zorn",
};


    然后LoadNames函數(shù)如下所示:

    void LoadNames(GuiTextList* list)
    {
        
// Use linq for C++ to create sorted TextItem(s) from DataSource
        CopyFrom(
            list
->GetItems(),
            FromArray(DataSource)
                
>>OrderBy(_wcsicmp)
                
>>Select<const wchar_t*, list::TextItem>(
                    [](
const wchar_t* name){return list::TextItem(name);}
                )
            );
    }


    首先我們用FromArray函數(shù)從const wchar_t* DataSource[]數(shù)組創(chuàng)建出一個IEnumerable<const wchar_t*>,然后用_wcsicmp進(jìn)行排序,最后把每一個const wchar_t* name都用list::TextItem(name)轉(zhuǎn)變成一個列表項。

    最后一步比較復(fù)雜,就是如何在移動列表的內(nèi)容的時候還保持兩個列表的內(nèi)容是排序的。首先,從一個排序列表中刪除東西,結(jié)果肯定是排序的。其次,把一堆新的名字插入一個列表,最簡單的方法就是把所有的東西連起來重新排序一遍,然后放回去:

    static list::TextItem GetTextItem(GuiTextList* list, int index)
    {
        
return list->GetItems()[index];
    }

    
static int CompareTextItem(list::TextItem a, list::TextItem b)
    {
        
return _wcsicmp(a.GetText().Buffer(), b.GetText().Buffer());
    }

    
static int ReverseCompareInt(int a, int b)
    {
        
return b-a;
    }

    
void MoveNames(GuiTextList* from, GuiTextList* to)
    {
        CopyFrom(
            to
->GetItems(),
            to
->GetItems()
                
>>Concat(
                    from
->GetSelectedItems()>>Select(Curry(GetTextItem)(from))
                    )
                
>>OrderBy(CompareTextItem)
            );

        List
<int> selectedItems;
        CopyFrom(
            selectedItems.Wrap(),
            from
->GetSelectedItems()
                
>>OrderBy(ReverseCompareInt)
            );
        FOREACH(
int, index, selectedItems.Wrap())
        {
            from
->GetItems().RemoveAt(index);
        }
    }

    
void buttonAdd_Clicked(GuiGraphicsComposition* sender, GuiEventArgs& arguments)
    {
        MoveNames(listSource, listDestination);
    }

    
void buttonRemove_Clicked(GuiGraphicsComposition* sender, GuiEventArgs& arguments)
    {
        MoveNames(listDestination, listSource);
    }


    我們可以看到MoveNames函數(shù)里面做了三件事情。

    第一件事情就是把to列表中的內(nèi)容,和from列表中選中的內(nèi)容Concat起來,也就是一個連著一個組成一個沒有排序過的IEnumerable<list::TextItem>,然后用CompareTextItem進(jìn)行排序。其中Curry(GetTextItem)(item)的意思是,將item作為GetTextItem的第一個參數(shù),把“填補(bǔ)剩下的參數(shù)”的這個過程產(chǎn)生一個新的函數(shù),就跟bind1st函數(shù)的意思是一樣的。所以經(jīng)過這個Select,就可以吧GetSelectedItems()返回的所有選中的item的下標(biāo),給轉(zhuǎn)換成原原本本的列表項。這樣將to里面的列表項和from里面選中的列表項所有的內(nèi)容合起來排序,最后放回to列表里面,這樣?xùn)|西就挪過去了。

    第二件事情就是要把from->GetSelectedItems()的內(nèi)容都復(fù)制下來,然后逆序。

    第三件事情,就是遍歷逆序后的GetSelectedItems的結(jié)果,一個一個刪除掉選中的項目了。列表控件只要內(nèi)容一變化,選中就會全部清空,因此如果不先保留GetSelectedItems的結(jié)果的話,刪除了一個列表項之后,GetSelectedItems就會變空,這樣就會出bug。

    這個Demo就介紹到這里了。GacUI下一個列表控件的Demo將是關(guān)于虛擬模式的。也就是說,這次我們不一個一個Add進(jìn)去了,而是直接告訴列表控件一共有多少個項目,然后列表要顯示的時候回調(diào)一下獲得真正的內(nèi)容。這個功能是很有用的,特別是當(dāng)內(nèi)容特別多,我們沒辦法一下子準(zhǔn)備好的時候,可以讓界面很快的出來,讓用戶感覺起來,程序運行的很流暢。

posted on 2012-05-25 21:54 陳梓瀚(vczh) 閱讀(3675) 評論(12)  編輯 收藏 引用 所屬分類: GacUI

評論:
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-26 01:42 | phoenixbing
我是來支持老大的  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-26 06:35 | 溪流
linq 能合起來寫不?哈~  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-26 07:29 | 朱峰everettjf
學(xué)習(xí)。
請問,能否說,gaclib是一個.NET WinForm風(fēng)格的c++界面庫?
GPU accelerated 是針對顯示做了優(yōu)化(相對于使用sdk或mfc、wtl)?  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-26 09:19 | 陳梓瀚(vczh)
@朱峰everettjf
我覺得,應(yīng)該是WPF風(fēng)格的吧。GPU是針對WinForm、MFC、WTL這種傳統(tǒng)的界面而言的。  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-26 09:20 | 陳梓瀚(vczh)
@溪流
能啊,里面不是合了嗎,還嵌套了  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-26 18:23 | 朱峰everettjf
@陳梓瀚(vczh)
嗯。向博主學(xué)習(xí)。  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-27 16:42 | tb
向博主學(xué)習(xí)  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-27 20:49 | 溪流
@陳梓瀚(vczh)
我是說類sql的寫法:)  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-27 23:09 | TianJun
單擊列表控件里面的名字有時候會選擇到其它名字, 不是我單擊的名字。  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-28 01:23 | 陳梓瀚(vczh)
@TianJun
有沒有重現(xiàn)方法?然后我去查一查  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-28 01:23 | 陳梓瀚(vczh)
@溪流
沒有那種寫法,C++搞不定這種……  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-28 03:44 | Scan
在C++中倒是可以弄出a.OrderBy().Select([](){}).ToList()這種語法的linq,借助一個代理類,用.來連接操作,至少在代碼補(bǔ)全方面更好用。
我以前倒是也試過在C++11中借助閉包實現(xiàn)linq to object,閉包用于延遲迭代。  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++ 2012-05-29 19:13 | TianJun
@陳梓瀚(vczh)
剛打開程序時不會, 把一些名字移動到右邊后, 就會出現(xiàn)單擊左邊列表控件會選擇不對的名字。  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++[未登錄] 2012-05-29 19:35 | 陳梓瀚(vczh)
@TianJun
我去看一看  回復(fù)  更多評論
  
# re: GacUI Demo:列表控件內(nèi)容的排序和移動,以及Linq for C++[未登錄] 2012-05-29 20:13 | 陳梓瀚(vczh)
@TianJun
refactor的時候犯了點小錯誤,這個問題已經(jīng)解決了。感謝提供信息。  回復(fù)  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲视频一二三| 一本久久a久久免费精品不卡| 亚洲午夜在线视频| 亚洲另类黄色| 欧美日韩在线播放三区四区| 亚洲手机成人高清视频| 亚洲一区二区三区色| 国产日本欧美一区二区| 久久精品国产亚洲精品| 久久麻豆一区二区| 亚洲黄色av一区| 99人久久精品视频最新地址| 国产精品久久久久影院亚瑟 | 久久久久久久网| 亚洲第一视频网站| 亚洲精选视频免费看| 国产精品综合网站| 欧美顶级艳妇交换群宴| 国产精品爱久久久久久久| 久久精品一区二区国产| 欧美国产日韩精品| 欧美自拍偷拍午夜视频| 欧美18av| 久久精品国产亚洲高清剧情介绍| 久久婷婷国产综合国色天香| 这里只有精品在线播放| 久久国产欧美| 亚洲一区在线视频| 鲁大师成人一区二区三区| 亚洲综合首页| 免费欧美在线视频| 久久国产精品99精品国产| 免费不卡欧美自拍视频| 久久精品123| 欧美日韩的一区二区| 久久综合久久美利坚合众国| 欧美四级电影网站| 亚洲高清精品中出| 欧美亚洲第一页| 亚洲激情视频| 亚洲国产精品va在线看黑人动漫| 亚洲午夜精品| 国产精品99久久久久久人| 久热国产精品| 久久精品国产91精品亚洲| 欧美三区美女| 亚洲精品一区二区三区不| 在线看不卡av| 久久精品人人做人人综合| 亚洲欧美中文字幕| 欧美巨乳在线| 亚洲欧洲另类| 亚洲另类在线一区| 欧美成人国产一区二区| 免费观看成人网| 激情国产一区| 久久久久久久综合色一本| 久久久久国产精品www| 国产精品乱码| 亚洲综合精品一区二区| 亚洲欧美日韩一区| 国产精品激情电影| 亚洲图片在线| 欧美一区二区三区免费大片| 国产精品激情| 欧美一级午夜免费电影| 欧美在线观看视频| 国产婷婷色一区二区三区四区| 99精品视频免费观看| 亚洲视频每日更新| 国产精品毛片在线| 欧美一区二区视频在线| 久久久久久欧美| 亚洲国产精品一区二区第一页| 欧美综合77777色婷婷| 久久字幕精品一区| 亚洲激情视频网站| 欧美久久久久久蜜桃| 一本色道久久综合狠狠躁篇怎么玩| 一区二区三区精品久久久| 国产精品a久久久久| 午夜视频精品| 欧美激情第一页xxx| 99精品热视频| 国产精品五月天| 久久久精品国产免费观看同学 | 亚洲另类视频| 欧美三级日本三级少妇99| 亚洲午夜极品| 欧美大片免费久久精品三p| 亚洲精品视频在线播放| 国产精品v亚洲精品v日韩精品| 亚洲一区二区av电影| 久久香蕉国产线看观看av| 亚洲激情婷婷| 国产农村妇女精品一二区| 久久男女视频| 亚洲视频1区2区| 另类av一区二区| 亚洲一区二区少妇| 影音欧美亚洲| 国产精品夫妻自拍| 免费在线看一区| 亚洲欧美电影院| 欧美国产日韩视频| 篠田优中文在线播放第一区| 亚洲激情在线视频| 国产欧美一区在线| 欧美日韩国产精品一区二区亚洲 | 亚洲精品资源| 久久久综合香蕉尹人综合网| 夜夜爽夜夜爽精品视频| 一区国产精品| 国产精品尤物| 国产精品成人在线| 欧美国产三区| 久久久人成影片一区二区三区| 亚洲少妇诱惑| 日韩亚洲视频在线| 亚洲国产成人精品女人久久久| 久久国产免费| 午夜影院日韩| 亚洲一区二区三区视频播放| 亚洲日本va午夜在线影院| 国产一区二区三区网站| 国产精品v片在线观看不卡| 欧美激情1区2区| 玖玖玖国产精品| 久久久精品国产免费观看同学| 亚洲欧美日韩一区在线| 99国产精品私拍| 日韩一区二区福利| 亚洲精品免费网站| 亚洲经典在线| 亚洲三级视频| 亚洲精品国产精品乱码不99| 欧美激情综合| 91久久久久久| 亚洲国产高清自拍| 亚洲国产精品久久久久婷婷884| 久久先锋影音| 欧美顶级少妇做爰| 欧美国产综合视频| 亚洲国产一区视频| 亚洲精品久久久久久久久久久久| 亚洲国产成人精品久久| 亚洲国产精品久久久久秋霞不卡| 欧美肥婆bbw| 亚洲国产cao| 亚洲伦伦在线| 亚洲婷婷综合色高清在线| 亚洲女人天堂成人av在线| 午夜欧美大尺度福利影院在线看| 亚洲免费在线视频一区 二区| 亚洲综合激情| 久久久久久尹人网香蕉| 免费成人美女女| 欧美日韩专区| 国产日韩欧美一区| 亚洲福利视频免费观看| 日韩视频永久免费观看| 亚洲一区免费视频| 久久久精品国产一区二区三区| 久久亚洲一区二区| 91久久久国产精品| 亚洲男人第一av网站| 久久人人爽人人爽爽久久| 欧美成人午夜剧场免费观看| 欧美视频在线不卡| 合欧美一区二区三区| 亚洲美女黄色片| 小辣椒精品导航| 欧美不卡视频一区| 国产精品99久久久久久久久久久久 | 日韩午夜在线播放| 欧美一区二区| 亚洲国产高清在线观看视频| 在线亚洲精品福利网址导航| 久久精品综合一区| 欧美日韩一区在线观看| 国产尤物精品| 亚洲图片欧洲图片日韩av| 久久免费国产精品| 一本久久精品一区二区| 久久精品视频在线免费观看| 欧美日韩在线免费观看| 国语精品中文字幕| 亚洲一级二级| 亚洲缚视频在线观看| 欧美一级视频精品观看| 欧美日韩免费观看中文| 海角社区69精品视频| 亚洲伊人网站| 亚洲国产欧美在线人成| 欧美一区二区视频在线观看| 欧美午夜精品| 亚洲精品自在久久| 欧美 日韩 国产 一区| 亚洲在线中文字幕| 欧美色图一区二区三区|