限制條件和修飾符順序

在 Compose 中,您可以將多個修飾符鏈結在一起,以變更可組合函式的外觀和風格。這些修飾符鏈結可能會影響傳遞至可組合項的限制,這些限制會定義寬度和高度邊界。

本頁說明鏈結修飾符如何影響限制條件,進而影響可組合項的測量和放置方式。

UI 樹狀結構中的修飾符

如要瞭解修飾符如何相互影響,建議您以視覺化方式呈現修飾符在 UI 樹狀結構中的顯示方式,這類樹狀結構會在組合階段產生。詳情請參閱「組合」一節。

在 UI 樹狀結構中,您可以將修飾符視為版面配置節點的包裝函式節點:

可組合項和修飾符的程式碼,以及以 UI 樹狀結構呈現的視覺化效果。
圖 1. 在 UI 樹狀結構中包裝版面配置節點的修飾符。

在可組合項中新增多個修飾符,即可建立修飾符鏈結。鏈結多個修飾符時,每個修飾符節點都會包裝其餘鏈結和內部版面配置節點。舉例來說,當您鏈結 clipsize 修飾符時,clip 修飾符節點會包裝 size 修飾符節點,而 size 修飾符節點會包裝 Image 版面配置節點。

在版面配置階段,遍歷樹狀結構的演算法會維持不變,但也會造訪每個修飾符節點。這樣一來,修飾符就能變更所包裝修飾符或版面配置節點的大小需求和位置。

如圖 2 所示,ImageText 可組合項的實作內容包含一連串包裝單一版面配置節點的修飾符。RowColumn 的實作項目只是版面配置節點,用於說明如何版面配置子項。

先前的樹狀結構,但現在每個節點都只是簡單的版面配置,並且有許多修飾符包裝節點。
圖 2. 與圖 1 相同的樹狀結構,但 UI 樹狀結構中的可組合項以修飾符鏈結呈現。

總結來說:

  • 修飾符會包裝單一修飾符或版面配置節點。
  • 版面配置節點可排版多個子節點。

以下各節將說明如何使用這個概念模型,推論修飾符鏈結的情形,以及鏈結會如何影響可組合項的大小。

版面配置階段中的限制

版面配置階段會按照三步驟演算法找出每個版面配置節點的寬度、高度和 x、y 座標:

  1. 測量子項:節點會測量其子項 (如果有)。
  2. 決定自身大小:節點會根據這些測量值決定自身大小。
  3. 放置子項:每個子項節點都會相對於節點本身的位置放置。

Constraints 可在演算法的前兩個步驟中,協助找出節點的正確大小。限制會定義節點寬度和高度的最小和最大邊界。當節點決定其大小時,其測量大小應落在這個大小範圍內。

限制條件的類型

限制可以是下列任一項目:

  • 有邊界:節點有寬度和高度的上限和下限。
容器內不同大小的受限約束條件。
圖 3. 受限的限制條件。
  • 無限制:節點不受任何大小限制。寬度和高度上限範圍設為無限。
無邊界限制,寬度和高度設為無限。約束條件超出容器範圍。
圖 4. 無界限的限制條件。
  • Exact:要求節點遵循確切大小規定。最小和最大邊界都設為相同的值。
符合容器內確切大小需求的確切限制。
圖 5. 完全限制條件。
  • 組合:節點會遵循上述約束條件類型的組合。舉例來說,限制條件可以限制寬度,同時允許無上限的高度,或是設定確切的寬度,但提供受限的高度。
兩個容器,顯示受限和無限的限制條件組合,以及確切的寬度和高度。
圖 6. 結合有界和無界限制條件,以及確切的寬度和高度。

下一節將說明如何將這些限制條件從父項傳遞至子項。

限制如何從父項傳遞至子項

在「版面配置階段中的限制條件」一文中所述演算法的首個步驟中,限制條件會從 UI 樹狀結構中的父項傳遞至子項。

父項節點測量子項時,會向每個子項提供這些限制,讓子項知道允許的大小範圍。接著,當它決定自己的大小時,也會遵循其父項傳入的限制條件。

大致來說,演算法的運作方式如下:

  1. 為了決定實際要佔用的大小,UI 樹狀結構中的根節點會測量其子項,並將相同的限制條件轉送至第一個子項。
  2. 如果子項是不會影響測量值的修飾符,則會將限制轉送至下一個修飾符。除非遇到會影響測量值的修飾符,否則限制條件會原封不動地傳遞至修飾符鏈結。然後再根據計算結果重新調整限制。
  3. 一旦到達沒有任何子項的節點 (稱為「葉節點」),系統就會根據傳入的限制決定其大小,並將這個已解析的大小傳回至父項。
  4. 父項會根據這個子項的測量值調整限制條件,並使用這些調整後的限制條件呼叫下一個子項。
  5. 父項的所有子項都完成測量後,父項節點會決定自身的大小,並將該大小傳達給自身的父項。
  6. 這樣一來,整個樹狀結構就會以深度優先方式進行逐層搜尋。最後,所有節點都已決定其大小,測量步驟也已完成。

如需深入瞭解範例,請觀看「限制和修飾符順序」影片。

影響限制條件的修飾符

您在上一節中瞭解到,某些修飾符可能會影響約束條件大小。以下各節說明會影響限制條件的特定修飾符。

size 修飾符

size 修飾符會宣告內容的偏好大小。

舉例來說,下列 UI 樹狀結構應由 200dp 轉換為 300dp 的容器,並進行轉譯。限制條件設有邊界,寬度介於 100dp300dp 之間,高度介於 100dp200dp 之間:

UI 樹狀結構的一部分,其中包含包裝版面配置節點的大小修飾符,以及容器中由大小修飾符設定的受限約束條件表示法。
圖 7. UI 樹狀結構中的邊界限制條件,以及在容器中的表示方式。

size 修飾符會調整傳入的限制條件,以符合傳遞給它的值。在這個範例中,這個值為 150dp

與圖 7 相同,但大小修飾符會調整傳入的限制,以符合傳遞給它的值。
圖 8.size 修飾符會將限制調整為 150dp

如果寬度和高度小於最小限制邊界,或大於最大限制邊界,修飾符會盡可能與傳入的限制相符,同時仍遵循以下傳入的限制:

兩個 UI 樹狀結構及其在容器中的對應表示法。在第一個情況中,大小修飾符會接受傳入的限制條件;在第二個情況中,大小修飾符會盡可能調整為最接近過大限制條件的大小,進而產生填滿容器的限制條件。
圖 9.size 修飾符會盡可能貼近傳遞的限制條件。

請注意,連接多個 size 修飾符不會生效。第一個 size 修飾符會將最小和最大限制設為固定值。即使第二個大小修飾符要求較小或較大的大小,仍需遵循傳入的確切邊界,因此不會覆寫這些值:

在 UI 樹狀結構中,兩個大小修飾符的鏈結,以及在容器中的表示方式。這是傳入的第一個值的結果,而非第二個值。
圖 10. 由兩個 size 修飾符所組成的鏈結,其中傳入的第二個值 (50dp) 不會覆寫第一個值 (100dp)。

requiredSize 修飾符

如果需要節點覆寫輸入的限制,請使用 requiredSize 修飾符,而非 sizerequiredSize 修飾符會取代傳入的限制條件,並將您指定的大小傳遞為確切邊界。

當大小傳回樹狀結構時,子節點會置中於可用空間:

在 UI 樹狀結構中鏈結的 size 和 requiredSize 修飾符,以及容器中的對應表示法。requiredSize 修飾符限制會覆寫 size 修飾符限制。
圖 11. requiredSize 修飾符會覆寫來自 size 修飾符的輸入限制。

widthheight 修飾符

size 修飾符會調整限制條件的寬度和高度。您可以使用 width 修飾符來設定固定寬度,但高度則不需指定。同樣地,您可以使用 height 修飾符來設定固定高度,但寬度則不需指定:

兩個 UI 樹狀結構,一個包含寬度修飾符及其容器表示法,另一個則包含高度修飾符及其表示法。
圖 12. width 修飾符和 height 修飾符分別設定固定寬度和高度。

sizeIn 修飾符

sizeIn 修飾符可讓您設定寬度和高度的確切最小和最大限制。如果您需要精細控制限制,請使用 sizeIn 修飾符。

包含 sizeIn 輔助鍵的 UI 樹狀結構,其中包含設定的最小和最大寬度和高度,以及在容器內的表示方式。
圖 13.sizeIn 修飾符,其中設定了 minWidthmaxWidthminHeightmaxHeight

範例

本節會顯示並說明多個含有鏈結修飾符的程式碼片段輸出內容。

Image(
    painterResource(R.drawable.hero),
    contentDescription = null,
    Modifier
        .fillMaxSize()
        .size(50.dp)
)

這個程式碼片段會產生以下輸出內容:

  • fillMaxSize 修飾符會變更限制,將最小寬度和高度都設為最大值:寬度為 300dp,高度為 200dp
  • 即使 size 修飾符想使用 50dp 的大小,仍需遵循輸入的最小限制。因此,size 修飾符也會透過 200 輸出 300 的確切限制邊界,有效地忽略 size 修飾符中提供的值。
  • Image 會遵循這些邊界,並透過 200 回報 300 的大小,而這會一直傳遞至樹狀結構的頂端。

Image(
    painterResource(R.drawable.hero),
    contentDescription = null,
    Modifier
        .fillMaxSize()
        .wrapContentSize()
        .size(50.dp)
)

這個程式碼片段會產生以下輸出內容:

  • fillMaxSize 修飾符會調整限制條件,將最小寬度和高度都設為最大值,也就是寬度為 300dp,高度為 200dp
  • wrapContentSize 修飾符會重設最小限制。因此,當 fillMaxSize 導致固定限制時,wrapContentSize將其重設為受限限制。以下節點現在可以再次占用整個空間,或小於整個空間。
  • size 修飾符會將限制設為 50 的最小和最大邊界。
  • Image 會透過 50 解析為 50 的大小,而 size 修飾符會將其傳遞出去。
  • wrapContentSize 修飾符具有特殊屬性。它會取得子項,並將其置於傳遞的可用最小邊界的中心。因此,傳達給父項的大小等於傳入的邊界下限。

只要結合三個修飾符,即可為可組合項定義大小,並將其置於父項的中央。

Image(
    painterResource(R.drawable.hero),
    contentDescription = null,
    Modifier
        .clip(CircleShape)
        .padding(10.dp)
        .size(100.dp)
)

這個程式碼片段會產生以下輸出內容:

  • clip 修飾符不會變更限制條件。
    • padding 修飾符會降低最大限制。
    • size 修飾符會將所有限制設為 100dp
    • Image 會遵循這些限制,並透過 100dp 回報 100 的大小。
    • padding 修飾符會在所有大小上加上 10dp,因此會將回報的寬度和高度增加 20dp
    • 在繪圖階段,clip 修飾符會透過 120dp120 的畫布執行動作。因此,它會建立該大小的圓形遮罩
    • 接著,padding 修飾符會在所有大小上將內容內嵌 10dp,因此會將畫布大小降低 100dp100
    • Image 會在該畫布中繪製。系統會根據 120dp 的原始圓形裁剪圖片,因此輸出結果並非圓形。