使用级联样式表 (CSS) 设置 Xamarin.Forms 应用的样式
Xamarin.Forms 支持使用级联样式表 (CSS) 对视觉对象元素设置样式。
Xamarin.Forms 应用程序可使用 CSS 进行样式设置。 样式表由规则列表组成,每个规则由一个或多个选择器和声明块组成。 声明块由放在大括号中的声明列表组成,每个声明由属性、冒号和值组成。 当块中有多个声明时,将插入分号作为分隔符。 下面的代码示例演示了一些符合 Xamarin.Forms 的 CSS:
navigationpage {
-xf-bar-background-color: lightgray;
}
^contentpage {
background-color: lightgray;
}
#listView {
background-color: lightgray;
}
stacklayout {
margin: 20;
}
.mainPageTitle {
font-style: bold;
font-size: medium;
}
.mainPageSubtitle {
margin-top: 15;
}
.detailPageTitle {
font-style: bold;
font-size: medium;
text-align: center;
}
.detailPageSubtitle {
text-align: center;
font-style: italic;
}
listview image {
height: 60;
width: 60;
}
stacklayout>image {
height: 200;
width: 200;
}
在 Xamarin.Forms 中,在运行时(而不是编译时)分析和计算 CSS 样式表,并且在使用时重新分析样式表。
注意
目前,XAML 样式设置可能的所有样式设置都无法通过 CSS 来实现。 但是,XAML 样式可用于为当前不受 Xamarin.Forms 支持的属性补充 CSS。 有关 XAML 样式的详细信息,请参阅使用 XAML 样式设置 Xamarin.Forms 应用的样式。
示例演示了使用 CSS 设置简单应用的样式,并显示在以下屏幕截图中:
使用样式表
向解决方案添加样式表的过程如下所示:
- 向 .NET Standard 库项目添加一个空 CSS 文件。
- 将 CSS 文件的生成操作设置为 EmbeddedResource。
加载样式表
有多种方法可用于加载样式表。
注意
目前不能在运行时更改样式表并应用新样式表。
XAML
将样式表添加到 ResourceDictionary
之前,可以使用 StyleSheet
类加载和分析样式表:
<Application ...>
<Application.Resources>
<StyleSheet Source="/Assets/styles.css" />
</Application.Resources>
</Application>
StyleSheet.Source
属性将样式表指定为相对于封闭 XAML 文件位置的 URI,如果 URI 以 /
开头,则将样式表指定为相对于项目根目录的 URI。
警告
如果 CSS 文件的生成操作未设置为 EmbeddedResource,则该文件将无法加载。
或者,在将样式表添加到 ResourceDictionary
之前,可以通过将其内联到 CDATA
部分中,使用 StyleSheet
类加载和分析样式表:
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet>
<![CDATA[
^contentpage {
background-color: lightgray;
}
]]>
</StyleSheet>
</ContentPage.Resources>
...
</ContentPage>
有关资源字典的详细信息,请参阅资源字典。
C#
在 C# 中,可以从 StringReader
加载样式表并将其添加到 ResourceDictionary
:
public partial class MyPage : ContentPage
{
public MyPage()
{
InitializeComponent();
using (var reader = new StringReader("^contentpage { background-color: lightgray; }"))
{
this.Resources.Add(StyleSheet.FromReader(reader));
}
}
}
StyleSheet.FromReader
方法的参数是已读取样式表的 TextReader
。
选择元素和应用属性
CSS 使用选择器来确定目标元素。 会按定义顺序连续应用具有匹配选择器的样式。 会在最后应用针对特定项定义的样式。 有关支持的选择器的详细信息,请参阅选择器引用。
CSS 使用属性为所选元素设置样式。 每个属性都有一组可能的值,一些属性可以影响任何类型的元素,而另一些属性则适用于元素组。 有关支持的属性的详细信息,请参阅属性引用。
如果子样式表设置相同的属性,则子样式表会始终替代父样式表。 因此,在应用设置了相同属性的样式时,将遵循以下优先规则:
- 如果在应用程序资源中定义的样式设置了与在页资源中定义的样式相同的属性,则前者将被后者覆盖。
- 页面资源中定义的样式将被控件资源中定义的样式覆盖(如果它们设置了相同的属性)。
- 如果在应用程序资源中定义的样式设置了与在控件资源中定义的样式相同的属性,则前者将被后者覆盖。
重要
不支持 CSS 变量。
按类型选择元素
可以使用不区分大小写的 element
选择器按类型选择可视化树中的元素:
stacklayout {
margin: 20;
}
此选择器会标识使用样式表的页面上的任何 StackLayout
元素,并将其边距设置为统一厚度 20。
注意
element
选择器不标识指定类型的子类。
按基类选择元素
可以使用不区分大小写的 ^base
选择器按基类选择可视化树中的元素:
^contentpage {
background-color: lightgray;
}
此选择器可标识使用样式表的任何 ContentPage
元素,并将其背景色设置为 lightgray
。
注意
^base
选择器特定于 Xamarin.Forms,且不是 CSS 规范的一部分。
按名称选择元素
可以使用区分大小写的 #id
选择器来选择可视化树中的单个元素:
#listView {
background-color: lightgray;
}
此选择器可标识其 StyleId
属性被设置为 listView
的元素。 但是,如果未设置 StyleId
属性,则选择器将回退以使用元素的 x:Name
。 因此,在以下 XAML 示例中,#listView
选择器将标识 x:Name
特性设置为 listView
的 ListView
,并将其背景色设置为 lightgray
。
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet Source="/Assets/styles.css" />
</ContentPage.Resources>
<StackLayout>
<ListView x:Name="listView" ...>
...
</ListView>
</StackLayout>
</ContentPage>
选择具有特定类特性的元素
可以使用区分大小写的 .class
选择器来选择具有特定类特性的元素:
.detailPageTitle {
font-style: bold;
font-size: medium;
text-align: center;
}
.detailPageSubtitle {
text-align: center;
font-style: italic;
}
可以通过将元素的 StyleClass
属性设置为 CSS 类名,将 CSS 类分配给 XAML 元素。 因此,在以下 XAML 示例中,将 .detailPageTitle
类定义的样式分配给第一个 Label
,而将 .detailPageSubtitle
类定义的样式将分配给第二个 Label
。
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet Source="/Assets/styles.css" />
</ContentPage.Resources>
<ScrollView>
<StackLayout>
<Label ... StyleClass="detailPageTitle" />
<Label ... StyleClass="detailPageSubtitle"/>
...
</StackLayout>
</ScrollView>
</ContentPage>
选择子元素
可以使用不区分大小写的 element element
选择器来选择可视化树中的子元素:
listview image {
height: 60;
width: 60;
}
此选择器可标识所有作为 ListView
元素的子元素的 Image
元素,并将其高度和宽度设置为 60。 因此,在以下 XAML 示例中,listview image
选择器将标识作为 ListView
子级的 Image
,并将其高度和宽度设置为 60。
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet Source="/Assets/styles.css" />
</ContentPage.Resources>
<StackLayout>
<ListView ...>
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
<Grid>
...
<Image ... />
...
</Grid>
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
</StackLayout>
</ContentPage>
注意
element element
选择器不要求子元素是父元素的直接子级 - 子元素可能有不同的父元素。 如果上级是指定的第一个元素,则会进行选择。
选择直接子元素
可以使用不区分大小写的 element>element
选择器来选择可视化树中的直接子元素:
stacklayout>image {
height: 200;
width: 200;
}
此选择器可识别所有作为 StackLayout
元素的直接子元素的 Image
元素,并将其高度和宽度设置为 200。 因此,在以下 XAML 示例中,stacklayout>image
选择器将标识作为 StackLayout
的直接子级的 Image
,并将其高度和宽度设置为 200。
<ContentPage ...>
<ContentPage.Resources>
<StyleSheet Source="/Assets/styles.css" />
</ContentPage.Resources>
<ScrollView>
<StackLayout>
...
<Image ... />
...
</StackLayout>
</ScrollView>
</ContentPage>
注意
element>element
选择器要求子元素是父元素的直接子级。
选择器引用
Xamarin.Forms 支持以下 CSS 选择器:
选择器 | 示例 | 说明 |
---|---|---|
.class |
.header |
选择其 StyleClass 属性中包含“header”的所有元素。 请注意,此选择器区分大小写。 |
#id |
#email |
选择其 StyleId 被设置为 email 的所有元素。 如果未设置 StyleId ,则回退到 x:Name 。 使用 XAML 时,首选 x:Name 而不是 StyleId 。 请注意,此选择器区分大小写。 |
* |
* |
选择所有元素。 |
element |
label |
选择类型为 Label 的所有元素,但不选择子类。 请注意,此选择器不区分大小写。 |
^base |
^contentpage |
选择其 ContentPage 被用作基类的所有元素,包括 ContentPage 本身。 请注意,此选择器不区分大小写,且不是 CSS 规范的一部分。 |
element,element |
label,button |
选择所有 Button 元素和所有 Label 元素。 请注意,此选择器不区分大小写。 |
element element |
stacklayout label |
选择 StackLayout 中的所有 Label 元素。 请注意,此选择器不区分大小写。 |
element>element |
stacklayout>label |
选择 StackLayout 的所有 Label 元素作为直接父级。 请注意,此选择器不区分大小写。 |
element+element |
label+entry |
直接选择 Label 之后的所有 Entry 元素。 请注意,此选择器不区分大小写。 |
element~element |
label~entry |
选择前面是 Label 的所有 Entry 元素。 请注意,此选择器不区分大小写。 |
会按定义顺序连续应用具有匹配选择器的样式。 会在最后应用针对特定项定义的样式。
提示
选择器可以不受限制地组合在一起,例如 StackLayout>ContentView>label.email
。
目前不支持以下选择器:
[attribute]
@media
和@supports
:
和::
注意
不支持特异性和特异性替代。
属性参考
Xamarin.Forms 支持以下 CSS 属性(在“值”列中,类型为斜体,而字符串文本为 gray
):
properties | 适用于 | 值 | 示例 |
---|---|---|---|
align-content |
FlexLayout |
stretch | center | start | end | spacebetween | spacearound | spaceevenly | flex-start | flex-end | space-between | space-around | initial |
align-content: space-between; |
align-items |
FlexLayout |
stretch | center | start | end | flex-start | flex-end | initial |
align-items: flex-start; |
align-self |
VisualElement |
auto | stretch | center | start | end | flex-start | flex-end | initial |
align-self: flex-end; |
background-color |
VisualElement |
color | initial |
background-color: springgreen; |
background-image |
Page |
字符串 | initial |
background-image: bg.png; |
border-color |
Button 、 Frame 、、 ImageButton |
color | initial |
border-color: #9acd32; |
border-radius |
BoxView 、、Button Frame 、、ImageButton |
double | initial |
border-radius: 10; |
border-width |
Button , ImageButton |
double | initial |
border-width: .5; |
color |
ActivityIndicator 、、BoxView 、CheckBox Button 、DatePicker 、Editor Picker Label ProgressBar Entry 、SearchBar 、、 Switch TimePicker |
color | initial |
color: rgba(255, 0, 0, 0.3); |
column-gap |
Grid |
double | initial |
column-gap: 9; |
direction |
VisualElement |
ltr | rtl | inherit | initial |
direction: rtl; |
flex-direction |
FlexLayout |
column | columnreverse | row | rowreverse | row-reverse | column-reverse | initial |
flex-direction: column-reverse; |
flex-basis |
VisualElement |
float | auto | initial 。 此外,可以使用 % 符号指定介于 0% 到 100% 之间的百分比。 |
flex-basis: 25%; |
flex-grow |
VisualElement |
float | initial |
flex-grow: 1.5; |
flex-shrink |
VisualElement |
float | initial |
flex-shrink: 1; |
flex-wrap |
VisualElement |
nowrap | wrap | reverse | wrap-reverse | initial |
flex-wrap: wrap-reverse; |
font-family |
Button 、、DatePicker 、Entry Editor 、Label 、Picker 、SearchBar 、、 TimePicker Span |
字符串 | initial |
font-family: Consolas; |
font-size |
Button 、、DatePicker 、Entry Editor 、Label 、Picker 、SearchBar 、、 TimePicker Span |
double | namedsize | initial |
font-size: 12; |
font-style |
Button 、、DatePicker 、Entry Editor 、Label 、Picker 、SearchBar 、、 TimePicker Span |
bold | italic | initial |
font-style: bold; |
height |
VisualElement |
double | initial |
min-height: 250; |
justify-content |
FlexLayout |
start | center | end | spacebetween | spacearound | spaceevenly | flex-start | flex-end | space-between | space-around | initial |
justify-content: flex-end; |
letter-spacing |
Button 、、DatePicker 、Entry Editor 、Label 、Picker 、SearchBar 、SearchHandler 、、 Span TimePicker |
double | initial |
letter-spacing: 2.5; |
line-height |
Label , Span |
double | initial |
line-height: 1.8; |
margin |
View |
thickness | initial |
margin: 6 12; |
margin-left |
View |
thickness | initial |
margin-left: 3; |
margin-top |
View |
thickness | initial |
margin-top: 2; |
margin-right |
View |
thickness | initial |
margin-right: 1; |
margin-bottom |
View |
thickness | initial |
margin-bottom: 6; |
max-lines |
Label |
int | initial |
max-lines: 2; |
min-height |
VisualElement |
double | initial |
min-height: 50; |
min-width |
VisualElement |
double | initial |
min-width: 112; |
opacity |
VisualElement |
double | initial |
opacity: .3; |
order |
VisualElement |
int | initial |
order: -1; |
padding |
Button 、、ImageButton Layout 、、Page |
thickness | initial |
padding: 6 12 12; |
padding-left |
Button 、、ImageButton Layout 、、Page |
double | initial |
padding-left: 3; |
padding-top |
Button 、、ImageButton Layout 、、Page |
double | initial |
padding-top: 4; |
padding-right |
Button 、、ImageButton Layout 、、Page |
double | initial |
padding-right: 2; |
padding-bottom |
Button 、、ImageButton Layout 、、Page |
double | initial |
padding-bottom: 6; |
position |
FlexLayout |
relative | absolute | initial |
position: absolute; |
row-gap |
Grid |
double | initial |
row-gap: 12; |
text-align |
Entry 、、EntryCell Label 、、SearchBar |
left | top | right | bottom | start | center | middle | end | initial 。 在从右到左的环境中应避免 left 和 right 。 |
text-align: right; |
text-decoration |
Label , Span |
none | underline | strikethrough | line-through | initial |
text-decoration: underline, line-through; |
text-transform |
Button 、Editor 、Entry 、Label 、、 SearchBar SearchHandler |
none | default | uppercase | lowercase | initial |
text-transform: uppercase; |
transform |
VisualElement |
none 、、rotate 、rotateY rotateX 、scale 、scaleX 、scaleY translate 、、translateX 、、 translateY initial |
transform: rotate(180), scaleX(2.5); |
transform-origin |
VisualElement |
double,double | initial |
transform-origin: 7.5, 12.5; |
vertical-align |
Label |
left | top | right | bottom | start | center | middle | end | initial |
vertical-align: bottom; |
visibility |
VisualElement |
true | visible | false | hidden | collapse | initial |
visibility: hidden; |
width |
VisualElement |
double | initial |
min-width: 320; |
注意
initial
是所有属性的有效值。 它可清除从另一种样式设置的值(重置为默认值)。
目前不支持以下属性:
all: initial
。- 布局属性(框或网格)。
- 速记属性,如
font
和border
。
此外,没有 inherit
值,因此不支持继承。 因此,无法在布局上设置 font-size
属性,也无法让布局中的所有 Label
实例都继承该值。 唯一例外是 direction
属性,其默认值为 inherit
。
目标 Span
元素存在一个已知问题,即通过元素和名称(使用 #
符号)阻止 span 成为 CSS 样式的目标。 Span
元素派生自 GestureElement
,它没有 StyleClass
属性,因此 span 不支持 CSS 类目标。 有关详细信息,请参阅无法将 CSS 样式设置应用于 Span 控件。
Xamarin.Forms 特定属性
以下 Xamarin.Forms 特定的 CSS 属性也受支持(在“值”列中,类型为斜体,而字符串文本为 gray
):
properties | 适用于 | 值 | 示例 |
---|---|---|---|
-xf-bar-background-color |
NavigationPage , TabbedPage |
color | initial |
-xf-bar-background-color: teal; |
-xf-bar-text-color |
NavigationPage , TabbedPage |
color | initial |
-xf-bar-text-color: gray |
-xf-horizontal-scroll-bar-visibility |
ScrollView |
default | always | never | initial |
-xf-horizontal-scroll-bar-visibility: never; |
-xf-max-length |
Entry 、 Editor 、、 SearchBar |
int | initial |
-xf-max-length: 20; |
-xf-max-track-color |
Slider |
color | initial |
-xf-max-track-color: red; |
-xf-min-track-color |
Slider |
color | initial |
-xf-min-track-color: yellow; |
-xf-orientation |
ScrollView , StackLayout |
horizontal | vertical | both | initial 。 both 仅在 ScrollView 上受支持。 |
-xf-orientation: horizontal; |
-xf-placeholder |
Entry 、 Editor 、、 SearchBar |
quoted text | initial |
-xf-placeholder: Enter name; |
-xf-placeholder-color |
Entry 、 Editor 、、 SearchBar |
color | initial |
-xf-placeholder-color: green; |
-xf-spacing |
StackLayout |
double | initial |
-xf-spacing: 8; |
-xf-thumb-color |
Slider , Switch |
color | initial |
-xf-thumb-color: limegreen; |
-xf-vertical-scroll-bar-visibility |
ScrollView |
default | always | never | initial |
-xf-vertical-scroll-bar-visibility: always; |
-xf-vertical-text-alignment |
Label |
start | center | end | initial |
-xf-vertical-text-alignment: end; |
-xf-visual |
VisualElement |
字符串 | initial |
-xf-visual: material; |
Xamarin.Forms Shell 特定的属性
以下 Xamarin.Forms Shell 特定的 CSS 属性也受支持(在“值”列中,类型为斜体,而字符串文本为 gray
):
properties | 适用于 | 值 | 示例 |
---|---|---|---|
-xf-flyout-background |
Shell |
color | initial |
-xf-flyout-background: red; |
-xf-shell-background |
Element |
color | initial |
-xf-shell-background: green; |
-xf-shell-disabled |
Element |
color | initial |
-xf-shell-disabled: blue; |
-xf-shell-foreground |
Element |
color | initial |
-xf-shell-foreground: yellow; |
-xf-shell-tabbar-background |
Element |
color | initial |
-xf-shell-tabbar-background: white; |
-xf-shell-tabbar-disabled |
Element |
color | initial |
-xf-shell-tabbar-disabled: black; |
-xf-shell-tabbar-foreground |
Element |
color | initial |
-xf-shell-tabbar-foreground: gray; |
-xf-shell-tabbar-title |
Element |
color | initial |
-xf-shell-tabbar-title: lightgray; |
-xf-shell-tabbar-unselected |
Element |
color | initial |
-xf-shell-tabbar-unselected: cyan; |
-xf-shell-title |
Element |
color | initial |
-xf-shell-title: teal; |
-xf-shell-unselected |
Element |
color | initial |
-xf-shell-unselected: limegreen; |
颜色
支持以下 color
值:
X11
与 CSS 颜色匹配的颜色、UWP 预定义的颜色和Xamarin.Forms颜色。 请注意,这些颜色值不区分大小写。- hex 颜色:
#rgb
、#argb
、#rrggbb
、#aarrggbb
- rgb 颜色:
rgb(255,0,0)
、rgb(100%,0%,0%)
。 值在 0-255 或 0%-100% 范围内。 - rgba 颜色:
rgba(255, 0, 0, 0.8)
、rgba(100%, 0%, 0%, 0.8)
。 不透明度值介于 0.0-1.0 之间。 - hsl 颜色:
hsl(120, 100%, 50%)
。 h 值介于 0-360 之间,而 s 和 l 值介于 0%-100% 之间。 - hsla 颜色:
hsla(120, 100%, 50%, .8)
。 不透明度值介于 0.0-1.0 之间。
Thickness
支持一个、两个、三个或四个 thickness
值,每个值用空格分隔:
- 单个值指示统一的厚度。
- 两个值指示垂直和水平厚度。
- 三个值指示顶部、水平(左和右),以及底部厚度。
- 四个值指示上、右、下、左的厚度。
注意
CSS thickness
值不同于 XAML Thickness
值。 例如,在 XAML 中,两个值 Thickness
指示水平和垂直厚度,而四个值 Thickness
表示左、上、右、下的厚度。 此外,XAML Thickness
值以逗号分隔。
NamedSize
支持以下不区分大小写的 namedsize
值:
default
micro
small
medium
large
每个 namedsize
值的确切含义取决于平台和视图。
函数
可以分别使用 linear-gradient()
和 radial-gradient()
CSS 函数指定线性和径向渐变。 应将这些函数的结果分配给控件的 background
属性。
Xamarin.Forms 中的 CSS 与 Xamarin.University
Xamarin.Forms 3.0 CSS 视频