Masonry源碼解讀

Masonry這個框架是使用代碼進行自動布局使用的,它的使用非常廣泛,這段時間一直在學習這個框架,因此想把學到的東西記下來,方便以后查閱,也便于與人分享。

自動布局約束的等式:

item1.attribute1 = multiplier × item2.attribute2 + constant

Masonry中使用了大量的點鏈式語法,考慮到應該有些小伙伴不知道點鏈式語法的來龍去脈,因此這里先整理一下點鏈式語法。

點鏈式語法

我們先來看一下Masonry框架的一種使用:

    [view mas_makeConstraints:^(MASConstraintMaker *make) {
        make.left.equalTo(superview.mas_left).mas_offset(30);
    }];

上面的代碼是Masonry的簡單的使用,這里面就用到了點鏈式語法make.left.equalTo(superview.mas_left).mas_offset(30);,我們看一下這句點鏈式語法,這里面包括三個要素:

  • 點語法:我們在訪問屬性的時候會使用點語法。
  • 小括號調用:在Objective-C中使用[ ]來調用方法,只有在調用Block的時候會使用(),因此這里我們可以使用Block來實現點鏈式語法中的()。
  • 連續調用:Block是有返回值的,那么我們可以在每次調用完Block后返回調用者對象本身,那么我們就可以實現連續的調用了。
    總結起來就是:

我們可以聲明一些Block類型的屬性,讓block類型的屬性的返回值為其本身。

下面用一個計算器的例子來說明一下:

//Calculator.h
@interface Calculator : NSObject

//這里是創建一個屬性,屬性的類型是block類型,屬性名是add
@property (nonatomic, copy)Calculator * (^add)(NSInteger num);
@property (nonatomic, copy)Calculator * (^minus)(NSInteger num);
@property (nonatomic, copy)Calculator * (^multiply)(NSInteger num);
@property (nonatomic, copy)Calculator * (^divide)(NSInteger num);

@property (nonatomic, assign)NSInteger result;

@end
//Calculator.m
@implementation Calculator

- (instancetype)init
{
    self = [super init];
    if (self) {
        self.result = 0;
    }
    
    return self;
}

//這里實現的是add這個屬性的get方法,只不過屬性的類型是block類型的。
- (Calculator * (^)(NSInteger num))add
{
    return ^id(NSInteger num){
        self.result += num;
        return self;
    };
}

- (Calculator * (^)(NSInteger num))minus
{
    return ^id(NSInteger num){
        self.result -= num;
        return self;
    };
}

- (Calculator * (^)(NSInteger num))multiply
{
    return ^id(NSInteger num){
        self.result *= num;
        return self;
    };
}

- (Calculator * (^)(NSInteger num))divide
{
    return ^id(NSInteger num){
        self.result /= num;
        return self;
    };
}

@end

調用:

    Calculator *calculator = [[Calculator alloc] init];
    calculator.add(5).minus(8).multiply(8).divide(23);
  • 1.calculator.add是調用了add屬性的get方法,這個方法會返回一個block,block如下:
    return ^id(NSInteger num){
        self.result += num;
        return self;
    };
  • 2.calculator.add(5)會執行這個block,這個block的返回值是Calculator對象本身,所以calculator.add(5)執行完了得到的是一個Calculator對象。
  • 3.Calculator對象繼續訪問minus屬性,執行minus屬性的get方法。
更簡潔的實現

上面是通過聲明一系列的block類型的屬性,再實現block屬性的get方法來實現鏈式調用,但是Masonry的實現方式和這種方式還是有區別,我們在Masonry中并沒有發現Block類型的屬性的聲明,反而是看到了一些平時見的比較少的方法的聲明:

0D223676-9F5F-4D4E-AAE0-05FA01F25A5F.png
也就是說Masonry中是把Block類型的屬性改成了返回值為Block類型的方法,這樣也能成功實現鏈式語法,這是為什么呢?
回想一下,當我們通過點語法去訪問屬性的時候實質上就是訪問了get方法,那么當不存在一個名為name的屬性時,我們使用self.name去訪問時是不是也會跑去執行名為name的方法呢?答案是肯定的,也就是只要我們申明了一個xxx方法,那就可以放心的寫self.xxx。
所以最終Calculator.h文件就改成了這樣:

@interface Calculator : NSObject
/*
@property (nonatomic, copy)Calculator * (^add)(NSInteger num);
@property (nonatomic, copy)Calculator * (^minus)(NSInteger num);
@property (nonatomic, copy)Calculator * (^multiply)(NSInteger num);
@property (nonatomic, copy)Calculator * (^divide)(NSInteger num);
@property (nonatomic, assign)NSInteger result;
*/
- (Calculator * (^)(NSInteger num))add;
- (Calculator * (^)(NSInteger num))minus;
- (Calculator * (^)(NSInteger num))multiply;
- (Calculator * (^)(NSInteger num))divide;

@end

Masonry的使用方法

1.使用MASConstraintMaker創建約束
UIEdgeInsets padding = UIEdgeInsetsMake(10, 10, 10, 10);

[view1 mas_makeConstraints:^(MASConstraintMaker *make) {
    make.top.equalTo(superview.mas_top).with.offset(padding.top); //with is an optional semantic filler
    make.left.equalTo(superview.mas_left).with.offset(padding.left);
    make.bottom.equalTo(superview.mas_bottom).with.offset(-padding.bottom);
    make.right.equalTo(superview.mas_right).with.offset(-padding.right);
}];

或者更簡單的方法:

[view1 mas_makeConstraints:^(MASConstraintMaker *make) {
    make.edges.equalTo(superview).with.insets(padding);
}];
并不是只有equalTo即等于這一種關系,還可以有:

lessThanOrEqualTo:等同于NSLayoutRelationLessThanOrEqual
greaterThanOrEqualTo:等同于NSLayoutRelationGreaterThanOrEqual

2.MASViewAttribute

Masonry中有MASViewAttribute這個類,這個類就等同于NSLayoutAttribute這個類:

MASViewAttribute NSLayoutAttribute
view.mas_left NSLayoutAttributeLeft
view.mas_right NSLayoutAttributeRight
view.mas_top NSLayoutAttributeTop
view.mas_bottom NSLayoutAttributeBottom
view.mas_leading NSLayoutAttributeLeading
view.mas_trailing NSLayoutAttributeTrailing
view.mas_width NSLayoutAttributeWidth
view.mas_height NSLayoutAttributeHeight
view.mas_centerX NSLayoutAttributeCenterX
view.mas_centerY NSLayoutAttributeCenterY
view.mas_baseline NSLayoutAttributeBaseline
3.與常數有關的問題

自動布局不允許對齊的屬性如left,right,centerY等設置為常數,如果我們傳了一個常數給這些屬性,Masonry會自動把這些約束變為相對于父視圖的約束,即:

//creates view.left = view.superview.left + 10
make.left.equalTo(@10)

4.mas前綴相關

在使用Masonry的時候,有時候會比較迷糊什么時候使用帶有mas前綴的,什么時候使用不帶前綴的,我們看下面這句代碼:

make.top.mas_equalTo(42);

這句代碼也可以這樣寫:

make.top.equalTo(@42);

但是這樣寫就會報錯:

make.top.equalTo(42);

原因就在于這個括號里面的參數類型必須是id類型,如果括號里面的參數不傳id類型就傳常量類型也行,那么就必須要在equalTo前面加上mas,加上mas后,mas_equalTo會把傳進來的數值類型變成id類型。

5.MASCompositeConstraint類相關

Masonry給了我們幾個便利的方法來讓我們一次性創建多個約束,Masonry中與這個約束相關的類是MASCompositeConstraint類,簡單使用如下:
edges

// make top, left, bottom, right equal view2
make.edges.equalTo(view2);

// make top = superview.top + 5, left = superview.left + 10,
//      bottom = superview.bottom - 15, right = superview.right - 20
make.edges.equalTo(superview).insets(UIEdgeInsetsMake(5, 10, 15, 20))

size

// make width and height greater than or equal to titleLabel
make.size.greaterThanOrEqualTo(titleLabel)

// make width = superview.width + 100, height = superview.height - 50
make.size.equalTo(superview).sizeOffset(CGSizeMake(100, -50))

center

// make centerX and centerY = button1
make.center.equalTo(button1)

// make centerX = superview.centerX - 5, centerY = superview.centerY + 10
make.center.equalTo(superview).centerOffset(CGPointMake(-5, 10))
6.修改已經存在的約束

當我們只是修改約束的constant的時候,可以使用mas_updateConstraints:

// this is Apple's recommended place for adding/updating constraints
// this method can get called multiple times in response to setNeedsUpdateConstraints
// which can be called by UIKit internally or in your code if you need to trigger an update to your constraints
- (void)updateConstraints {
    [self.growingButton mas_updateConstraints:^(MASConstraintMaker *make) {
        make.center.equalTo(self);
        make.width.equalTo(@(self.buttonSize.width)).priorityLow();
        make.height.equalTo(@(self.buttonSize.height)).priorityLow();
        make.width.lessThanOrEqualTo(self);
        make.height.lessThanOrEqualTo(self);
    }];

    //according to apple super should be called at end of method
    [super updateConstraints];
}

當我們要修改的不止是約束的constant的時候,使用mas_updateConstraints就力不從心了,這時就需要使用mas_remakeConstraints:

- (void)changeButtonPosition {
    [self.button mas_remakeConstraints:^(MASConstraintMaker *make) {
        make.size.equalTo(self.buttonSize);

        if (topLeft) {
            make.top.and.left.offset(10);
        } else {
            make.bottom.and.right.offset(-10);
        }
    }];
}
休息一下

解讀源碼

我們在解讀源碼的時候先從最簡單最基礎的使用開始,然后由淺入深,逐漸深入。下面我們先分析一下整個框架的文件結構:

E4C13061-A282-4C44-9626-B797EE011C74.png

下面就從一個最簡單最基本的使用開始來探究源碼:

    [view mas_makeConstraints:^(MASConstraintMaker *make) {
        make.left.equalTo(superview.mas_left).mas_offset(30);
    }];

我們先不管外面的方法調用,只需要知道make是MASConstraintMaker類型的就行了,從make.left.equalTo(superview.mas_left).mas_offset(30);開始:

  • 1.make.left
    left是它的一個屬性,這里調用的是屬性的getter方法。
    57BE757D-EE05-47E3-9F2D-4A58A3F2712A.png
    繼續往下查看:
    668FACAD-700E-4583-82AB-867A76F46A5D.png
    再繼續:
    20ED5488-141A-442A-A8D5-3A7798A4486F.png
    我們來看一下MASViewConstraint對象的創建:
    2728C00E-9170-4905-BCB0-82CD9C91ACAF.png

那么現在我們來總結一下make.left做了哪些事:

make.left是調用了MASConstraintMaker類的left屬性的get方法,這里創建了一個MASViewAttribute對象,這個對象由一個UIView對象和一個NSLayoutAttribute來創建,這里UIView對象是view,NALayoutAttribute為NSLayoutAttributeLeft,所以這里MASViewAttribute對象也就是封裝了約束等號左邊的兩個元素。。然后使用創建的MASViewAttribute對象來創建了一個MASViewConstraint對象,這個對象代表這一行代碼所表示的整個約束。最終make.left返回了一個MASVIewConstraint對象。

需要注意的是,MASConstraintMaker對象有一個數組類型的consrtaints屬性,新創建的MASViewConstraint對象被加入到了這個屬性中,在最后添加約束的時候會遍歷這個數組。

  • 2.superview.mas_left
    mas_left是分類的一個屬性,所以superview.mas_left會調用分類的-(MASViewAttribute *)mas_left方法。
    9413264B-AB4F-4F1B-921A-153DE3B96E93.png

    這里通過代碼創建了一個MASViewAttribute對象,對象的view即superview,對象的attribute即NSLayoutAttributeLeft,我們看看是如何創建的:
    892F2158-3377-4352-8EFF-B5937A412744.png

    總結一下:

superview.mas_left返回了一個MASViewAttribute對象,這個對象封裝了約束等號右邊的兩個元素。

  • 3.make.left.equalTo(superview.mas_left)
    進入equalTo查看具體實現:
    F8189FD9-D511-48B3-95B5-CBA04C2DE78C.png
    也就是說我執行make.left.equalTo會得到一個Block,那么我執行make.left.equalTo(superview.mas_left)就是執行這個Block,即make.left.equalTo(superview.mas_left)會執行self.equalToWithRelation(attribute, NSLayoutRelationEqual)這一行核心代碼,并返回這一行核心代碼的返回值。
    由于make.left是MASViewConstraint對象,所以我們要去MASViewConstraint類中查看equalWithRelation的實現:
    4D84193E-0A50-425E-8CF8-297CCB1B53B8.png
    self.secondViewAttribute = attribute;會觸發secondViewAttribute這個屬性的set方法,我們看一下其set方法的實現:
    1C0A9CAA-0874-49C3-9E93-F126A4428219.png
    這里的意思就是make.left.equalTo()這個括號里面傳入的東西可能有三種情況,第一種是數字,第二種是一個UIView對象,如果是UIView對象,那就將其layoutAttribute設置為何firstAttribute一致,也就是我們也可以這樣寫:make.left.equalTo(superview),這樣Masonry也能成功識別。第三種是傳入的MASViewConstraint對象,如果傳入這種對象則可以直接賦值給secondViewAttribute屬性。
    下面我們再來看一下第一種情況傳入數字的處理方式,我們進入setLayoutConstantWithValue:這個方法:
    B7C31203-51F7-4E48-80ED-4F3B58AC9D31.png
    代碼里面設置offset,centerOffset等都不是真正的實現,真正的實現在offset屬性的set方法里,我們要其MASViewConstraint類中找offset屬性的set方法:
    C66A8756-CD7D-418C-838F-4130A72E8A0E.png
    總結起來就是如果傳入的是數值類型,那么就給MASViewConstraint的layoutConstant屬性賦值。

這樣我們就清楚了make.left.equalTo()這個括號中傳入各種不同類型的值會怎么操作。

總結一下make.left.equalTo(superview.mas_left)做的事情:

make.left創建了一個firstViewAttribute,firstViewAttribute的view屬性即為
mas_makeConstraint方法的調用者,其layoutAttribute屬性為NSLayoutAttributeLeft,firstViewAttribute封裝了約束等式左邊的兩個item。接著通過傳入firstViewAttribute創建了一個MASViewConstraint對象。superview.mas_left則是創建了一個secondViewAttribute對象,該對象的view即為superview,layoutAttributeNSLayoutAttributeLeft。make.left.equalTo(supervie.mas_left)則是將secondViewAttribute賦值給MASViewConstraint對象的secondViewAttribute屬性,并給MASViewConstraint對象的layoutRelation屬性賦值。

  • 4..mas_offset(30)
    mas_offset的顏色是土黃色,說明這是一個宏定義,我們點進去,發現這個宏定義是定義在MASConstraint.h文件中:
#define mas_offset(...)                  valueOffset(MASBoxValue((__VA_ARGS__)))

這不是一個簡單的宏定義,里面還進行了嵌套,我們看一下MASBoxValue()方法做了什么,在MASUtilities.h這個文件中找到了MASBoxValue()這個方法:

AF2FEF9E-81D3-4CAC-A6AB-79BBED81485C.png
從圖中我們可以看到,這個方法就是把一些數值類型的值轉為id類型。

Masonry的用法這部分我說過mas前綴的使用,這里我們就看到了其實現方法。如果我們括號里想要直接傳入數值類型而不是id類型的參數,那么前面使用的API就必須帶mas前綴,否則報錯,如果傳入的是id類型,則前面使用的API是否帶mas前綴均可。

所以.mas_offset(30)也就等于.valueOffset(@30),那么我們來查看一下MASConstraint.m中的實現:

9EDB3756-844B-48C8-BF55-1DF70D10A560.png
最終就是設置了MASViewConstraint的layoutConstant屬性:
85F60147-C2E5-4A0B-9AB7-E96B8BD51130.png

到這里make.left.equalTo(superview.mas_left).mas_offset(30);這行代碼就全解讀完了??偨Y一下就是:

make,left創建了一個MASViewConstraint對象,為這個對象的firstViewAttribute屬性賦值,superview.mas_left即創建了一個MASViewAttribute對象,equalTo()即把這個MASViewAttribute對象賦值給MASViewConstraint對象的secondViewAttribute屬性,.mas_offset(30)則是給MASViewConstraint對象的layoutConstant屬性賦值為30.

  • 5.mas_makeConstraints:
    mas_makeConstraints:這個方法是在UIView的分類中定義并實現的,下面我們看一下其具體實現:
    48B3AC46-BE1A-4FEE-98E2-252900F394F3.png
    72B1DEFF-9089-41C5-A215-C070C0781C33.png

    再來看[constraint install]:,這個方法的內容比較多,我們分兩部分來看:
    0735EF28-E952-4C4F-8F20-7DA742D42D64.png
    C50DD5C3-64C0-4E40-86F2-B18311F7979C.png

    到這里約束就添加完成了。 總結一下添加約束的大體流程:

首先根據MASViewConstraint對象的firstViewAttributesecondViewAttribute這兩個屬性,訪問這兩個屬性得到firstLayoutItem,firstLayoutAttribute。secondLayoutitemsecondLayoutAttribute。然后處理有時是對齊屬性如left沒有提供view的情況,這時就要設置view為其父視圖。然后尋找應該將約束添加到哪個視圖上,最后添加約束到對應的視圖上。

休息一下

組合約束(MASCompositeConstraint)

Masonry中可以直接約束size,center,edge這樣的組合約束。其本質也就是把它拆成多個單個約束,比如對size的約束,就是拆成width和height這兩個約束。下面我們看一下其具體的實現方法:

make.size.equalTo(superview).sizeOffset(CGSizeMake(10, 10));
  • 1.make.size
    44B0B9CD-B912-44A6-8C8B-F07275FDFAE9.png
    繼續往下看:- (MASConstraint *)addConstraintWithAttributes:(MASAttribute)attrs方法:
    081C430A-BD90-4FC8-9D62-4FDC07D0EA6B.png

    總結一下make.size做了哪些事:

把size這一個拆分成了width和height這兩個,根據這兩個約束創建了兩個MASViewConstraint對象,裝到一個數組里面,使用這個數組來創建了一個MASCompositeConstraint對象,最后返回這個MASCompositeConstraint對象。

  • 2..equalTo(superview)
    D5C22579-A50F-44BC-A79B-5815FB5FC96D.png
    繼續:
    5FDF1EB1-ED9A-4AD4-8EA8-B935A7AF5D5F.png

    總結一下make.size.equalTo(superview):

把size這一個約束拆分成了width和height這兩個約束,并根據此創建了兩個MASViewConstraint對象,根據這兩個對象組成的數組去創建一個MASCompositeConstraint對象。然后遍歷MASCompositeConstraint對象的childConstraints數組,取出數組里面的額每個MASViewConstraint對象,然后像處理單個MASViewConstraint一樣去處理。

  • 3..sizeOffset(CGSizeMake(10, 10))
    可以想象就是把它拆分,然后賦值給兩個MASViewConstraint的layoutConstant屬性,就不詳細說了。

mas_updateConstraints:和mas_remakeConstraints:

自動布局約束等式:
item1.attribute1 = multiplier × item2.attribute2 + constant

有時候我們有更改約束的需求,比如我們要做一個動畫,移動某個視圖,那就需要改變視圖約束,當我們只是改變約束的constant時,可以使用mas_updateConstraints:這個方法。而當我們需要改變的不止constant時,就需要調用mas_remakeConstraints:這個方法了。

先來看一下mas_updateConstraints:這個方法是怎么實現只改變約束的constant的:

CEDE51E1-CB87-42C7-9E40-A3AFA01A4755.png

進入[constraintMaker install]
99A0DCCC-96D2-455F-B0E2-EE71B5745FD0.png

看看[constraint install]中是怎么實現的:
2CE11985-5A8F-4701-939C-AB16EA8B56EA.png

總結起來,mas_updateConstraints:就是去self.installedView.constarints這個屬性數組中去遍歷,看有沒有這樣一個NSLayoutConstraint對象,它除了constant外,所有內容都和當前的NSLayoutConstraint對象一致,如果有這個對象,那么就把該對象的constant改為當前NSLayoutConstraint對象的constant。這樣來完成約束條件的更新。

再來看一下mas_remakeConstraints:
當我們要改變的約束條件不止是constant這么簡單時,使用mas_remakeConstraints:就不頂用了,就要使用mas_remakeConstraints:這個方法。

F6106353-8D44-4F85-BE0C-1E8E8CE7D9A0.png
再看[constraintMaker install]:
5D7151E9-43FE-4E11-8AB2-CE965A40B190.png

7AAA699D-C641-48AD-83D8-DA855ADE7437.png

這篇文章在簡書的地址:http://www.lxweimin.com/p/8990c5a98d29

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,578評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,701評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,691評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,974評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,694評論 6 413
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,026評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,015評論 3 450
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,193評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,719評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,442評論 3 360
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,668評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,151評論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,846評論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,255評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,592評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,394評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,635評論 2 380

推薦閱讀更多精彩內容