再學(xué)Blazor——組件建造者

使用 RenderTreeBuilder 創(chuàng)建組件是 Blazor 的一種高級(jí)方案。前幾篇文中有這樣創(chuàng)建組件的示例 builder.Component<MyComponent>().Build(); 付呕,本文主要介紹該高級(jí)方案的具體實(shí)現(xiàn)胯甩,我們采用測(cè)試驅(qū)動(dòng)開發(fā)(TDD)方法入桂,大致思路如下:

  • 從測(cè)試示例入手
  • 擴(kuò)展一個(gè)RenderTreeBuilder類的泛型擴(kuò)展方法哟旗,泛型類型為組件類型
  • 創(chuàng)建組件建造者類(ComponentBuilder)提供方法來構(gòu)建組件
  • 通過組件的屬性選擇器來設(shè)置組件參數(shù)
  • 構(gòu)建時(shí)能返回組件的對(duì)象實(shí)例

1. 示例

首頁我們從一個(gè)我們預(yù)想的高級(jí)方案示例入手箱吕,然后逐漸分析并實(shí)現(xiàn)我們預(yù)想的方案缸棵。下面是預(yù)想的示例代碼:

class MyComponent : ComponentBase
{
    private MyTest test; //MyTest組件的對(duì)象實(shí)例

    //覆寫構(gòu)建呈現(xiàn)樹方法
    protected override void BuildRenderTree(RenderTreeBuilder builder)
    {
        builder.Component<MyTest>()
               .Set(c => c.Title, "Hello")    //設(shè)置MyTest組件Title參數(shù)
               .Build(value => test = value); //建造組件并給MyTest實(shí)例賦值
    }
}

2. 擴(kuò)展方法

下面實(shí)現(xiàn)builder.Component<MyTest>()這行代碼舟茶,這是RenderTreeBuilder的一個(gè)擴(kuò)展方法,該方法返回組件建造者類(ComponentBuilder)堵第。

public static class Extension
{
    //泛型T是Blazor組件類型
    public static ComponentBuilder<T> Component<T>(this RenderTreeBuilder builder) where T : notnull, IComponent
    {
        //返回一個(gè)組件建造者類對(duì)象吧凉,將builder傳遞給建造者
        //其內(nèi)部方法需要通過builder來構(gòu)建組件
        return new ComponentBuilder<T>(builder);
    }
}

3. 建造者類

接下來實(shí)現(xiàn)組件建造者類(ComponentBuilder),該類是手動(dòng)構(gòu)建組件的核心代碼踏志,提供設(shè)置組件參數(shù)以及構(gòu)建方法阀捅。

public class ComponentBuilder<T> where T : IComponent
{
    //手動(dòng)構(gòu)建呈現(xiàn)器
    private readonly RenderTreeBuilder builder;
    //組件參數(shù)字典,設(shè)置組件參數(shù)時(shí)针余,先存入字典饲鄙,在構(gòu)建時(shí)批量添加
    internal readonly Dictionary<string, object> Parameters = new(StringComparer.Ordinal);

    //構(gòu)造函數(shù)
    internal ComponentBuilder(RenderTreeBuilder builder)
    {
        this.builder = builder;
    }

    //添加組件參數(shù)方法,name為組件參數(shù)名稱圆雁,value為組件參數(shù)值
    //提供Add方法可以添加非組件定義的屬性忍级,例如html屬性
    public ComponentBuilder<T> Add(string name, object value)
    {
        Parameters[name] = value; //將參數(shù)存入字典
        return this;              //返回this對(duì)象,可以流式操作
    }

    //設(shè)置組件參數(shù)方法伪朽,selector為組件參數(shù)屬性選擇器表達(dá)式轴咱,value為組件參數(shù)值
    //使用選擇器有如下優(yōu)點(diǎn):
    // - 當(dāng)組件屬性名稱更改時(shí),可自動(dòng)替換
    // - 通過表達(dá)式 c => c. 可以直接調(diào)出組件定義的屬性,方便閱讀
    // - 可通過TValue直接限定屬性的類型嗦玖,開發(fā)時(shí)即可編譯檢查
    public ComponentBuilder<T> Set<TValue>(Expression<Func<T, TValue>> selector, TValue value)
    {
        var property = TypeHelper.Property(selector); //通過屬性選擇器表達(dá)式獲取組件參數(shù)屬性
        return Add(property.Name, value);             //添加組件參數(shù)
    }

    //組件構(gòu)建方法患雇,action為返回組件對(duì)象實(shí)例的委托跃脊,默認(rèn)為空不返回實(shí)例
    public void Build(Action<T> action = null)
    {
        builder.OpenComponent<T>(0); //開始附加組件
        if (Parameters.Count > 0)
            builder.AddMultipleAttributes(1, Parameters); //批量添加組件參數(shù)
        if (action != null)
            builder.AddComponentReferenceCapture(2, value => action.Invoke((T)value)); //返回組件對(duì)象實(shí)例
        builder.CloseComponent();   //結(jié)束附加組件
    }
}

4. 屬性選擇器

為什么要用屬性選擇器宇挫,組件建造者類中已經(jīng)提到,下面介紹如何通過屬性選擇器表達(dá)式來獲取組件類型的屬性對(duì)象酪术。

public class TypeHelper
{
    //通過屬性選擇器表達(dá)式來獲取指定類型的屬性
    public static PropertyInfo Property<T, TValue>(Expression<Func<T, TValue>> selector)
    {
        if (selector is null)
            throw new ArgumentNullException(nameof(selector));

        if (selector.Body is not MemberExpression expression || expression.Member is not PropertyInfo propInfoCandidate)
            throw new ArgumentException($"The parameter selector '{selector}' does not resolve to a public property on the type '{typeof(T)}'.", nameof(selector));

        var type = typeof(T);
        var propertyInfo = propInfoCandidate.DeclaringType != type
                         ? type.GetProperty(propInfoCandidate.Name, propInfoCandidate.PropertyType)
                         : propInfoCandidate;
        if (propertyInfo is null)
            throw new ArgumentException($"The parameter selector '{selector}' does not resolve to a public property on the type '{typeof(T)}'.", nameof(selector));

        return propertyInfo;
    }
}

5. 總結(jié)

以上就是組件建造者的完整實(shí)現(xiàn)過程器瘪,代碼不長(zhǎng),但這些功能足以完成手動(dòng)構(gòu)建Blazor組件的需求绘雁。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末橡疼,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子庐舟,更是在濱河造成了極大的恐慌欣除,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,602評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件挪略,死亡現(xiàn)場(chǎng)離奇詭異历帚,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)杠娱,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,442評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門挽牢,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人摊求,你說我怎么就攤上這事禽拔。” “怎么了室叉?”我有些...
    開封第一講書人閱讀 152,878評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵睹栖,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我茧痕,道長(zhǎng)磨淌,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,306評(píng)論 1 279
  • 正文 為了忘掉前任凿渊,我火速辦了婚禮梁只,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘埃脏。我一直安慰自己搪锣,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,330評(píng)論 5 373
  • 文/花漫 我一把揭開白布彩掐。 她就那樣靜靜地躺著构舟,像睡著了一般。 火紅的嫁衣襯著肌膚如雪堵幽。 梳的紋絲不亂的頭發(fā)上狗超,一...
    開封第一講書人閱讀 49,071評(píng)論 1 285
  • 那天弹澎,我揣著相機(jī)與錄音,去河邊找鬼努咐。 笑死苦蒿,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的渗稍。 我是一名探鬼主播砌些,決...
    沈念sama閱讀 38,382評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼荐类,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起洒扎,我...
    開封第一講書人閱讀 37,006評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤父叙,失蹤者是張志新(化名)和其女友劉穎碎浇,沒想到半個(gè)月后腾它,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,512評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡碗誉,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,965評(píng)論 2 325
  • 正文 我和宋清朗相戀三年召嘶,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片诗充。...
    茶點(diǎn)故事閱讀 38,094評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡苍蔬,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出蝴蜓,到底是詐尸還是另有隱情碟绑,我是刑警寧澤,帶...
    沈念sama閱讀 33,732評(píng)論 4 323
  • 正文 年R本政府宣布茎匠,位于F島的核電站格仲,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏诵冒。R本人自食惡果不足惜凯肋,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,283評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望汽馋。 院中可真熱鬧侮东,春花似錦、人聲如沸豹芯。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,286評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽铁蹈。三九已至宽闲,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背容诬。 一陣腳步聲響...
    開封第一講書人閱讀 31,512評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工娩梨, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人览徒。 一個(gè)月前我還...
    沈念sama閱讀 45,536評(píng)論 2 354
  • 正文 我出身青樓狈定,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親吱殉。 傳聞我的和親對(duì)象是個(gè)殘疾皇子掸冤,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,828評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容