样式和主题

样式——style

概述

在API中对Style的描述:

A style resource defines the format and look for a UI. A style can be applied to an individual View (from within a layout file) or to an entire or Activity application (from within the manifest file).

简单来说,就是类似web页面设计中的css( 层叠样式表)

定义好style文件,在组件中通过"@style/"来轻松的使用,不需要再为每一个组件重复书写属性内容,提高了代码复用性。

使用

传统activity_main.xml

如果我们需要写两个TextView,而它们的属性几乎相同,我们需要写成下面这样。

android 主题控制颜色分组 安卓组件主题_xml

如果我们要写的属性相似的组件不止两个,是四个五个甚至是十余个,这时候,虽然ctrl+c,ctrl+v也不慢,但是会让代码看起来很“蠢”

这时候我们就需要使用style了,具体如何使用呢?

打开res/values目录,一般默认情况下会发现一个style.xml文件

android 主题控制颜色分组 安卓组件主题_xml_02

我们打开它,它默认情况下大概是下面这样的(版本不一样可能会有差异,不过没关系)

android 主题控制颜色分组 安卓组件主题_自定义组件_03

android 主题控制颜色分组 安卓组件主题_android 主题控制颜色分组_04

如何自定义我们的第一个style呢?我们发现其实默认代码就是一个很好的参考,它就已经是一个完整的style了。

所以我们的自定义style由外到里分别是

 

接下来我们把上面截图中的代码转换为style

  • <style> </style> 我们还需要为style加一个name(这个name可以随便起,用于在引用时唯一确定和标识)除此之外,还有一个parents,代表继承关系,写上之后会继承父级的样式效果。可以继承系统内部的样式。如果想要继承自己的样式,直接在name的值前面加上"你要继承的样式名."即可。如下图,这样MyStyle就继承了NewStyle,虽然我的NewStyle里啥也没有...
  • <item> </item> 也需要一个name,即属性的具体内容,例如android:layout_width。
  • 然后在两个尖括号内,需要为这个书写这个属性具体的值,例如wrap_content
<resources>

    <!-- Base application theme. -->
    <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
        <!-- Customize your theme here. -->
        <item name="colorPrimary">@color/colorPrimary</item>
        <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
        <item name="colorAccent">@color/colorAccent</item>
    </style>

    <style name="MyStyle">
        <item name="android:layout_width">wrap_content</item>
        <item name="android:layout_height">wrap_content</item>
        <item name="android:background">@android:color/holo_blue_bright</item>
        <item name="android:textColor">@android:color/holo_blue_light</item>
        <item name="android:textSize">35sp</item>
    </style>
</resources>

回到activity_main.xml,我们去引用定义好的style,可以看到效果和刚刚完全一样,但是代码清爽了很多,这在组件属性重复性高的时候,会变得非常有用。

android 主题控制颜色分组 安卓组件主题_xml_05

android 主题控制颜色分组 安卓组件主题_android 主题控制颜色分组_06

除此之外,样式表是允许被覆盖的,如图,虽然样式表已经定义了textSize=35sp,但是你依旧可以定义textSize=66sp来对其进行覆盖,从而满足了个性化的需求

android 主题控制颜色分组 安卓组件主题_android 主题控制颜色分组_07

主题——Themes

概述

首先要明确Themes和Style的区别和关系。

主题是应用于整个Acitivity或应用程序的样式,而不是一个独立的View对象。

(即Themes依旧是一种样式,但应用的场景不一样,Themes应用于应用或Activity)

然后有一点需要明确,也很重要。

当Themes与样式Style发生冲突时,Style的优先级更高。

使用

首先,和之前一样,我们还是在style文件里创建一个style

<style name="MyThemes">
        <item name="android:layout_width">wrap_content</item>
        <item name="android:layout_height">wrap_content</item>
        <item name="android:background">@android:color/holo_orange_dark</item>
        <item name="android:textColor">@android:color/holo_orange_light</item>
        <item name="android:textSize">35sp</item>
    </style>

区别在于,主题的应用需要在清单里进行(manifests/AndroidManifest.xml)

android 主题控制颜色分组 安卓组件主题_Android_08

打开文件,我们可以发现有一行 android:theme,如果修改这里,那么整个程序所有的Activity将会被修改样式。

我们应该做的是在我们的Activity里加上一行android:theme

如下图:

android 主题控制颜色分组 安卓组件主题_自定义组件_09


插一条,可能会遇到如下报错

android 主题控制颜色分组 安卓组件主题_android 主题控制颜色分组_10

建议修改继承关系如下图

android 主题控制颜色分组 安卓组件主题_android 主题控制颜色分组_11


效果如图:

android 主题控制颜色分组 安卓组件主题_android 主题控制颜色分组_12

可以看到之前提到的优先级,确实是Style高,而且Themes确实是作用于整个Activity。

 

自定义组件

概述

在实际开发中,安卓提供的UI组件可能并不能满足所有的需求,当我们要构建更复杂的UI视图,就需要自定义组件了。

实现方式

常有以下三种方式

  1. 将Android现有的控件进行组合,继承ViewGroup或其子Layout类等布局类进行组合
  2. 调整Android现有的控件,继承View的子类具体类
  3. 完全自定义组件,继承View积累,里面的界面及事件由自己控制

 具体实现

首先我们需要一个XML属性资源文件来进行配合。

所以在配置资源文件目录下建立文件attrs.xml

android 主题控制颜色分组 安卓组件主题_android 主题控制颜色分组_13

代码:attr即属性,需要为其定义属性名及格式

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="MyView">
        <attr name="textColor" format="color"></attr>
        <attr name="textSize" format="dimension"></attr>
        <attr name="text" format="string"></attr>
    </declare-styleable>
</resources>

 

第二步,我们新建一个类继承View,即我们的自定义视图类

然后我们去实现(重写)里面的构造方法

android 主题控制颜色分组 安卓组件主题_自定义组件_14

package com.example.a3_23custom_componen;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.view.View;

import com.example.a3_23custom_componen.R;

public class MyView extends View {

    //刚才attrs里定义的三个...
    private int textColor;
    private float textSize;
    private String text;
    //画笔
    private Paint paint;

    public MyView(Context context) {
        super(context);
    }

    public MyView(Context context, AttributeSet attrs) {
        super(context,attrs);
        //实例化画笔
        paint=new Paint();
        //获取配置文件中的属性值
        TypedArray array=context.obtainStyledAttributes(attrs,R.styleable.MyView);
        textColor=array.getColor(R.styleable.MyView_textColor,0xFFFFFF);
        textSize=array.getDimension(R.styleable.MyView_textSize,24);
        text=array.getString(R.styleable.MyView_text);

    }

    public MyView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }
    //视图的绘制事件方法
    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        paint.setColor(textColor);
        paint.setTextSize(textSize);
        canvas.drawText(text,100,400,paint);
    }
}

接下来是使用它,回到我们的activity_main.xml布局文件,组件名即你的类名(路径要写全)然后通过app:你的属性来进行设置,用法和普通的组件一样。

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="自定义组件!"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <com.example.a3_23custom_componen.MyView
        android:id="@+id/myView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:text="233333333"
        app:textColor="#233333"
        app:textSize="23sp" />

</android.support.constraint.ConstraintLayout>

android 主题控制颜色分组 安卓组件主题_android 主题控制颜色分组_15

效果如上,“2333333”即为自定义组件。

Scrollview嵌套Listview、GridView的冲突

由于Listview、GridView自带滚动条,而Scrollview本身就是滚动条的组件,所以嵌套会出现冲突。

然后我们简单测试一下,首先在string.xml里准备一个数组

<resources>
    <string name="app_name">3_23custom_debug</string>
    <string-array name="test">
        <item>test1</item>
        <item>test2</item>
        <item>test3</item>
        <item>test5</item>
        <item>test6</item>
        <item>test7</item>
        <item>test8</item>
        <item>test9</item>
        <item>test0</item>
        <item>test1</item>
        <item>test2</item>
        <item>test3</item>
        <item>test4</item>
        <item>test5</item>
        <item>test6</item>
    </string-array>
</resources>

接着在布局里写一个listview和若干按钮

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity"
    android:orientation="vertical">

    <ListView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:entries="@array/test"></ListView>

    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button3"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button4"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button5"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button6"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button7"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

</LinearLayout>

运行,发现无法显示按钮。 

android 主题控制颜色分组 安卓组件主题_xml_16

为了能让按钮显示,我们很自然的想到使用scrollview,由于scrollview只能放一个东西,所以我们再嵌套一个linearlayout

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity"
    android:orientation="vertical">

    <ScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent">

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="vertical">

    <ListView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:entries="@array/test"></ListView>

    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button3"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button4"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button5"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button6"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />

    <Button
        android:id="@+id/button7"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button" />
        </LinearLayout>
    </ScrollView>
</LinearLayout>

可以看到,虽然两个组件都显示了,但是listview显然显示不正常。这就是题中所提到的冲突,如何解决这个冲突呢?

android 主题控制颜色分组 安卓组件主题_android_17

这时候我们就可以用自定组件了,自定义一个MyListView类

package com.example.a3_23custom_debug;


import android.content.Context;
import android.util.AttributeSet;
import android.widget.ListView;

public class MyListView extends ListView {


    public MyListView(Context context) {
        super(context);
    }

    public MyListView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public MyListView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }
    //重写该方法,达到使ListView适应ScrollView的效果
    
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        //给高度重新赋值  利用Spec传值  内容有多少就显示多少,
        heightMeasureSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE>>2,MeasureSpec.AT_MOST);
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    }
}

在activity_main.xml中将ListView替换成我们自定义的ListView(即由上图改成下图)

android 主题控制颜色分组 安卓组件主题_Android_18

android 主题控制颜色分组 安卓组件主题_Android_19

效果如下:可以看到,已经恢复正常了

android 主题控制颜色分组 安卓组件主题_Android_20