Steam皮肤制作指导 进阶篇

前言

本文可能涉及到一些“专业”的知识,为了不影响阅读,可以先阅读入门篇或者看看我的自制皮肤Glass

如果有按照我的步骤一步一步来,入门篇的最后一定会出现一个“bug”,放一张图回忆回忆:

一个渲染错误的界面

其实这不是bug啦,只是基本操作。其原理是客户端的渲染机制导致的,这里不做过多的讲解。只需要了解:

最底层(最下面)的控件一定要有背景绘制,实际上我们的壁纸就是最底层那层绘制。后来我们加上壁纸的绘制,客户端就表现正常了,就是这个原理。

概览

本文将会主要讲解这些知识点,放张图压压惊:


image.png

主要参考官方文档以及自己的经验。

简单介绍

layout : 控件该在哪里出现

controls : 控件该干些什么

styles : 控件该长什么样

color : 描述样式中的颜色或某些特定的属性值

一个完整的 .layout 文件由上面4部分构成,每个部分都被一对大括号包裹。但这4部分都不是必须的,即使一个部分都没有,只要格式正确(如果有下一篇,会解释这个格式),就会被steam客户端加载。

详细介绍

这里会对上面4部分做详细介绍,包括说明。和相关属性的默认值、可选值怎么使用可以参考入门篇的styles

需要注意的是,这些属性在某些地方可能有特殊用法(比如下载进度条),在使用特殊用法的地方,v社开发人员都贴心的留下注释了,这里我只关注一般情况。

语法

先了解一下语法,吐槽一下layout文件里边什么格式都有,强迫症快要死了。

1.1 符号不敏感

变量中对 = 不敏感,下面两段代码的效果一样的,不过我建议使用 =

color1="0 0 0 255"
color1 "0 0 0 255" // 有空格代替等号是允许的

空格不敏感,在空格用作两个属性之间的间隔变量名和变量值的连接时,1个空格和100个空格是一样的

// 在变量中
// 下面两种情况效果一样
color1="0 0 0 255"
color1    =           "0 0 0 255"

// 下面两种情况效果一样
color1      "0 0 0 255"
color1  "0 0 0 255"
// 在style、control、place、region中,以place为例,其他类似
// 下面两种情况效果一样
place{x=50 y=40 width=max} // 在这里对等号敏感
place{   x  =  50    y =  40   width    =  max  }

换行不敏感,找不到合适的语言来描述,放代码自行体会

// 在style、control、place、region中,以place为例,其他类似
// 下面的情况效果一样
place{x=50 y=40 width=max}

place{
    x=50 y=40 width=max
}

place
{x=50 y=40 width=max}

place
{
    x=50 y=40 width=max
}

1.2 样式

controlslayoutstyles内部使用

button {
    textcolor="0 0 0 255"
    bgcolor="255 255 255 0"
    padding=10
}

其中button是名称,textcolorbgcolor是属性"0 0 0 255"10是属性值

2 style

定义一个样式,供controlplaceregion 使用。

button {
    textcolor="0 0 0 255"
    bgcolor="255 255 255 0"
}

注意:所有的属性都不是必须的

2.1 字体相关

属性 类型 默认值 预设值 说明
font-family 字符串 Tahoma - 字体名称
font-size 整数 - - 字体大小,单位像素
font-weight 整数 400 0~1000 字体粗细。400是标准,700是粗体
font-style 字符串 normal normal(正常),italic(斜体),underline(下划线),strikeout(删除线),uppercase(大写),lowercase(小写),symbol,antialias,dropshadow,outline,rotary,additive 字体样式,其中一些可以叠加
  • 字体样式:

    1. normal(正常)
    normal(正常)
    1. italic(斜体)
    italic(斜体)
    1. underline(下划线)
    underline(下划线)
    1. strikeout(删除线)
    strikeout(删除线)
    1. uppercase(大写)
    image.png
    1. lowercase(小写)
    image.png
    1. symbol(未知)
    image.png
    1. antialias(未知)
    image.png
    1. dropshadow(未知)
    image.png
    1. outline(未知)
    image.png
    1. rotary(未知)
    image.png
    1. additive(未知)
    image.png

2.2 颜色相关

属性 类型 说明
bgcolor 字符串或颜色 背景色
textcolor 字符串颜色 文字颜色
selectedtextcolor 字符串颜色 选中文字颜色
selectedbgcolor 字符串颜色 选中文字背景色
shadowtextcolor 字符串颜色 在不同的控件中有不同的意义

2.3 布局相关

属性 类型 默认值 说明
padding 整数 0 内边距,只是用与类型为LabelButtoncontrol
insert 整数 "0 0 0 0" 子控件在控件中如何定位,4个数字分别代表距离左、上、右、下的距离,对bgcolor起作用
minimum-width 整数 0 最小宽度,不适用与所有控件
minimum-height 整数 0 最小高度,不适用与所有控件
  • padding(内边距)
padding=30
image.png

假如紫框为定义的控件,绿框+紫框就是实际显然效果的效果,当然颜色还是原来定义的颜色,这里只是方便区分

padding-left=30
padding-top=30
padding-right=30
padding-bottom=30

这段代码和上面那段可以达到一样的效果

  • insert
insert="30 30 30 30"
image.png

假如绿框是定义的控件,紫框就是加上上面代码后的效果,显示不全的部分会被隐藏

insert-left=30
insert-top=30
insert-right=30
insert-bottom=30

这段代码和上面那段可以达到一样的效果

2.4 image

用图片替代文字,使用于LabelButton控件(控件类型属于control部分内容)

image="graphics/broadcast/icon_close_default"

值为一个图片的相对于皮肤文件夹根目录的相对位置,不要带文件后缀。当然这里的图片指的是tag格式的图片

2.5 render、render_bg(渲染)

这两个属性都是用于渲染的,前者在控件绘制之后开始绘制后者在控件绘制之前开始绘制,说的通俗一点就是前者是绘制遮罩,后者是绘制背景。

做个比喻方便理解,bgcolor是你的皮肤颜色,render_bg是衣服颜色,textcolor衣服上字的颜色,render是围裙的颜色。

  • 注意: render_bgbgcolor绘制之后绘制,这意味着bgcolor的效果可能被覆盖。

每一个完整的渲染由一组渲染指令完成(一条指令也能完成一个完整的渲染),每条渲染指令可以指定一种渲染方式,包括fillimageimage_tiledimage_scaleimage_proportionalgradientgradient_horizontaldashedrect

无论哪种方式渲染都需要至少4个参数,这4个参数实际上是2个坐标,以控件占据的区域为准 (x0,y0)表示左上角,(x1,y1)表示右下角,假设一个控件占据的区域长100,宽70,那么有:

// ==两边代码指定的区域相同
(x0, y0)==(x1-100, y0-70)
(x1, y1)==(x0+100, y0+70)
(x0+20, y0+30)==(x1-80, y1-40)

描述坐标参数时需要遵从先上后下,先左后右的原则:

// 前两个参数为第一个点,后两个参数为第二个点
0="fill(x0, y0, x0, y0+10, red)" // 正确,先上后下
0="fill(x0, y0+10, x0, y0, red)" // 错误,先下后上

0="fill(x0, y0, x0+10, y0, red)" // 正确,先左后右
0="fill(x0+10, y0, x0, y0, red)" // 错误,先右后左

这要求我们在确定渲染区域是要选取左上角和右下角来确定渲染区域,且需要先写左上角在写右下角。

观察上面的指令,每条指令前面都有一个数字先看下官方解释:

The numbers on each line are just an artifact of our common parsing library; it doesn't matter what you put there, and it can all be the same.

大概意思是:每条指令上的数字只是我们公共解析库中的一部分;他们并不重要,甚至完全一样也没问题。

  • 注意:这里我特意引用了官方文档,其原因是我在制作皮肤的过程中发现:在某些控件中,将渲染指令中的数字使用一样的,会导致渲染效果达不到预期。当我将数字替换为成不一样的之后。渲染和我预期的一样。

现在来详细解释下这些渲染方式:

  • 注意:前4个坐标参数为必须,下表中就不在列举

2.5.1 fill(填充)

  • 必要参数
    一个颜色,必须是一个定义好的颜色变量

  • 例子

    0="fill(x0, y0, x0+10, y0, red)"
    
  • 说明
    单色填充所选区域

2.5.2 dashedrect(虚线框)

  • 必要参数
    一个颜色,必须是一个定义好的颜色变量

  • 例子

    0="dashedrect(x0, y0, x0+10, y0, red)"
    
  • 说明
    绘制具有指定颜色的虚线矩形

2.5.3 gradient_horizontal(纵向渐变填充)

  • 必要参数
    使用两个颜色渐变填充指定区域,渐变有上到下,第一个颜色为开始渐变,第一个为结束渐变颜色

  • 例子

    0="gradient_horizontal(x0, y0, x0+10, y0, red, blue)"
    
  • 说明
    渐变填充区域,方向从上到下

2.5.4 gradient_horizontal(横向渐变填充)

  • 必要参数
    使用两个颜色渐变填充指定区域,渐变有上到下,第一个颜色为开始渐变,第一个为结束渐变颜色

  • 例子

    0="gradient_horizontal(x0, y0, x0+10, y0, red, blue)"
    
  • 说明
    渐变填充区域,方向从左到右

2.5.5 image(图片)

  • 必要参数
    一个图片的相对于皮肤文件夹根目录的相对位置,不要带文件后缀。当然这里的图片指的是tag格式的图片,与image属性一样

  • 例子

    0="image(x0, y0, x0+10, y0, graphics/broadcast/icon_close_default)"
    
  • 说明
    使用图片渲染指定区域,需要注意的是,如果图片的尺寸不渲染区域要大,可能图片会超出指定的渲染区域

2.5.6 image_tiled(图片,拓展)

  • 必要参数
    同上

  • 例子

    0="image_tiled(x0, y0, x0+10, y0, graphics/broadcast/icon_close_default)"
    
  • 说明
    从指定区域的左上角开始,重复渲染该图片,直到铺满整个区域。
    假设你现在新建一个文本文档,你的指定你的记事本区域为渲染区域,数字1为渲染的图片,使用使用此渲染方式就是不停的按1,直到整个文档全是1

2.5.7 image_scale(图片,缩放,不等比例)

  • 必要参数
    同上

  • 例子

    0="image_scale(x0, y0, x0+10, y0, graphics/broadcast/icon_close_default)"
    
  • 说明
    缩放图片,铺满整个区域,图片可能会变形

2.5.8 image_proportional

  • 必要参数
    同上

  • 例子

    0="image_proportional(x0, y0, x0+10, y0, graphics/broadcast/icon_close_default)"
    
  • 说明
    缩放图片,铺满整个区域,会等比例缩放,图片不会变形

3 control

control在controls区域内被定义,它赋予控件样式和具体作用

  • 注意: 这control部分内容并没有文档作为支撑,绝大部分内容靠我自己脑补的,所以此部分巨大多数情况下我不建议去调整
// 一个样式
fileimage {
    controlname="ImagePanel"
    image=resource/icon_file
    zpos=1
    style="fileimagestyle"
}

其中fileimage为该控件的名称,用在其他地方引用这个控件,{}内的为该控件的属性

3.1 属性

属性|类型|预设值|说明
-|-|-|-|-
controlname|string|Label、URLLabel、ImagePanel、Button、CheckButton、RadioButton|控件类型
style|string|-|styles中定义的样式

恩,没了(

其他的属性,根据属性名大概能猜出来什么意思,但在实际使用中,有时有用,有时又没有用,索性就不说了。

control一般都不去动他,只是用来确定控件使用的style的。

4 layout

4.1 说明

这里的layout并非指.layout文件,而是.layout文件中的layout区域,这个区域是用来确定控件的在客户端上位置的。

在这个区域中有两种确定控件位置的方式:regionplace,前者多用于指定包含多个控件的一个区域,后者则用来单独指定某个控件的位置,当然前者用来单独指定一个区域也是完全没有问题的。

placeregion命令有很多参数,其中绝大部分都是通用的,并且这些参数全是可选的。

4.2 place和region的属性(参数)简要说明

属性 类型 必须 默认值 预设值 说明
control 字符串 - - 为该区域指定一个control,可以一次指定多个control
region 字符串 - - 为该区域指定一个父容器,并且该区域的相对于父级定位
name 字符串 - - 为该区域指定一个名称,方便其他区域应用
x 整数 0 - 相对于父容器的绝对定位,原点在父容器的左上角。若没有指定父容器为.layout文件指定的区域
y 整数 0 - 同x,x、y构成一个坐标确定该区域的左上角
width 整数 - - 该区域的宽度,单位像素,父容器参考x
height 整数 - - 该区域的高度,单位像素,父容器参考x
start 字符串 - - 用来参考定位的其他区域的名称
dir 字符串 right down、right 相对于参考区域的方向
align 字符串 top-left bottom、top、left、right、top-left、top-right 该区域的对其方式
margin 整数 0 - 该区域的外边距(上下左右),单位像素,可以使用margin-left, margin-right, margin-top, margin-bottom的方式指定单独每一边的边距
padding 整数 0 - 该区域的内边距(上下左右),单位像素,可以使用padding-left, padding-right, padding-top, padding-bottom的方式指定单独每一边的边距
spacing 整数 0 - 区域中控件的间距,单位像素
overflow 字符串 - allow-horizo​​ntal、allow-vertical、allow-both、scroll-horizo​​ntal、scroll-vertical、scroll-both region独有,区域内的内容超出该区域后的处理方式,带有allow及时内容超出,依旧会显示,并且不会改变定义好的样式,带有scroll的内容超出后会为该区域添加一个滚动条

4.3 属性详细说明

我会挑出我认为不容易理解的属性,详细说明一下

4.3.1 region、place

layout {

    // 区域1
    region {
        name="aRegion"
        width=100
        height=200
        x=50
        y=50
    }

    // 区域2
    place {
        region="aRegion"
        width=20
        height=20
        x=50
        y=40
    }

    // 区域3
    region {
        region="aRegion"
        width=50
        height=100
    }

    // 区域4
    place {
        width=130
        height=140
    }
}

上面的4个区域,实际上不会有任何显示。只定义了区域,没有定义要展示的内容。

假如每一个区域都是有边框的,他们一定是这样的:

image.png

先看区域1(黑框):

  1. 定义了一个名称为aRegion的矩形区域,矩形区域的长为100像素,高为200像素
  2. 没有指定region属性,默认父容器为.layout控制的区域
  3. 指定xy,所以从(50,50)出开始渲染

区域2(绿框):

  1. 定义一个20 x 20的区域
  2. 指定aRegion为父容器
  3. 指定从(50,40)开始渲染,对于.layout文件就是从(100,90)开始渲染

区域3(红框):

  1. 定义一个50 x 100的区域
  2. 指定aRegion为父容器
  3. 指定从(0,0)开始渲染,对于.layout文件就是从(50,50)开始渲染

先看区域4(紫框):

  1. 定义一个130 x 140的区域
  2. 没有指定region属性,默认父容器为.layout控制的区域
  3. 指定xy,所以从(50,50)出开始渲染

4.3.2 control

// 指定单个control
place {
    control=DetailsBorder
    y=1
    x=1 
    width=max 
    height=max 
}

// 指定多个control
// 使用逗号隔开
place {
    control=welcomedetails,rentaldetails,turnnotifications,friendsdetails,achievementsdetails,dlcdetails,clouddetails,communityfilesdetails,newsdetails,nonsteamdetails,screenshotsdetails
    region=detailsbody
    spacing=30
    dir=down
    width=max
    margin-left=25
}

指定一个或者多个control使用这个区域

dir

place {
    control=welcomedetails,rentaldetails,turnnotifications,friendsdetails,achievementsdetails,dlcdetails,clouddetails,communityfilesdetails,newsdetails,nonsteamdetails,screenshotsdetails
    region=detailsbody
    spacing=30
    dir=down
    width=max
    margin-left=25
}

在指定多个control时使用dir来指定这些control的排列方式,right水平向右,down垂直向下

spacing

place {
    control=welcomedetails,rentaldetails,turnnotifications,friendsdetails,achievementsdetails,dlcdetails,clouddetails,communityfilesdetails,newsdetails,nonsteamdetails,screenshotsdetails
    region=detailsbody
    spacing=30
    dir=down
    width=max
    margin-left=25
}

在指定多个control时使用dir来指定这些control之间的间隔,单位像素,水平向右

4.3.3 margin(外边距)

layout {
    place {
        width=100
        height=100
        x=10
        y=40
        margin=30
    }
}

图:


image.png

图中紫色为最终渲染结果,绿色虚线框仅仅为辅助线,margin为外边距,即外边的边距,只影响实际渲染的位置,不影响内容

margin=10

maring-left=10
maring-top=10
maring-right=10
maring-bottom=10

效果相同

4.3.4 padding(外边距)

这部分和style中的margin一样

4.3.5 start

place { 
    control="change_user_button"  
    align=left 
    height=24 
}

place { 
    control="account_name_label"  
    start="change_user_button" 
    dir="right" 
    height=24 
    margin-left=-4 
}

start用来给定一个参考,给出的参考是control不是region,start能和dir联合使用,指定dir为right或者down来让当前布局生成在参考控件的右边或者下面。

设备标识符

这个是用来做适配的,随便找一句

ClientTitle:FrameFocus [!$OSX] {
    font-family=basefont
    font-size=14
    font-weight=400
    textcolor="label"
    bgcolor="none"
    inset="12 6 0 0"
}
    
ClientTitle:FrameFocus [$OSX] {
    font-family=basefont
    font-size=15
    font-weight=400
    textcolor="texthover"
    bgcolor="none"
    inset="0 6 0 0"
}

第一个适用于非OSX,第二个适用于OSX,其他的基本看到都能明白

结语

祝,各位玩的愉快~

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351

推荐阅读更多精彩内容