Android 自定义控件开发入门 (三)

上两次我们从如何自定义控件讲起,列举了View的一些Api,说明了一些在自定义的时候,可以进行重写的方法,然后通过一个例子的两种写法向大家展示了最基本的自定义控件和我们要充分了解并积极重写View方法的精神,这次我们将继续进行学习!

现在请大家回想一下我们使用安卓原生控件时的感受,一个好的控件是可以在xml中进行各种属性的操作的,而自定义控件往往有一些特殊的需求,今天我要讲的就是安卓给自定义控件添加自定义的属性。

下面再给大家具体介绍一下如果自定义的View需要有自定义的属性我们该如何处理:

我们还是用这个例子,继续扩展,比如我想在xml中设置球体的半径,我该怎么办呢?

让我们先来具体了解一下自定义属性的一些简单基础,然后通过实例大家就会很容易掌握了!

首先看一下属性文件中format可选项 

如果自定义控件需要自定义该控件的属性。在开始前,我们需要检查在values目录下是否有attrs.xml,如果没有则创建。下面我们先来了解一下format属性类型都有哪些。

format可选项

"reference" //引用

"color" //颜色

 "boolean" //布尔值

 "dimension" //尺寸值

 "float" //浮点值

 "integer" //整型值

 "string" //字符串

 "fraction" //百分数比如200%

这里我们的例子就多用几个自定义属性,帮助大家理解,我们就定义半径,颜色,以及小球刚开始停止的位置就好了,

我们先创建attrs.xml:

代码如下:

<?xml version="1.0" encoding="utf-8"?>
<resources>

    <declare-styleable name="MoveBallversion3">
        <attr name="BallColor" format="color" />
        <attr name="BallRadius" format="float" />
        <attr name="BallStartX" format="float" />
        <attr name="BallStartY" format="float" />
    </declare-styleable>

</resources>

MoveBallversion3是用来在java代码中找到这些数据的名字,下面每一项前部是属性名,后部是类型。

之后我们在创建自定义控件的时候就可以使用自定义属性了,但是设置的时候要注意:

1)其中com.example.customcomponentsdemo.component.DrawView3这句是需要显示的控件所代表的类。

2)com.example.customcomponentsdemo是类的包名,

就是manifast里面package地址

3)DrawView3是类名。这个类肯定是继承自View的自定义类,可以是在工程中直接源码添加xxxx.java的,也可以是在libs目录下自己新添加的jar包里面的。如果是jar包里面的一个类,则路径就是jar包里面,

Xml如下

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:myball="http://schemas.android.com/apk/res/com.example.customcomponentsdemo"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="这是MoveBall的Demo 版本3"
        android:textColor="@color/white" >
    </TextView>

    <com.example.customcomponentsdemo.component.DrawView3
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_margin="10dp"
        myball:BallColor = "@color/white"
        myball:BallRadius = "20"
        myball:BallStartX = "50"
        myball:BallStartY = "200" >
    </com.example.customcomponentsdemo.component.DrawView3>

</LinearLayout>

之后在重写的时候获取就可以了。

不过有时候会出现这种错误。

有时候使用控件的自定义属性,使用时填写命名空间。

按照老样子:http://schemas.android.com/apk/控件地址

在编译的时候报错error: No resource identifier found for attribute ‘xxxxt‘ in package

解决:代码看了n遍也没找出问题,最后再网上看了一个老外的回答。ADT升级以后,自定义控件的命名空间的路径访问给优化了改成

http://schemas.android.com/apk/res-auto 这样填充之后ADT会自动去相应的包路径下寻找。

我们在继续向下看:

我们在DrawView3.java 代码编写如下,其中下面的构造方法是重点,我们获取定义的属性,获取方法中后面通常设定默认值如:(floattextSize = a.getDimension(R.styleable.MyView_textSize, 36 ); ), 防止我们在xml 文件中没有定义.从而使用默认值!

MoveBallversion3就是定义在<declare-styleablename="MoveBallversion3 "></declare-styleable> 里的
名字,获取里面属性用 名字_ 属性 连接起来就可以.TypedArray 通常最后调用 .recycle() 方法,为了保持以后使用该属性一致性!

好,看一下具体代码:

package com.example.customcomponentsdemo.component;

import com.example.customcomponentsdemo.R;

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

public class DrawView3 extends View {

	private float CircleX;
	private float CircleY;
	private float CircleR;
	private int BallColor;

	public DrawView3(Context context, AttributeSet attrs) {
		super(context, attrs);

		// 获取自定义属性
		TypedArray a = context.obtainStyledAttributes(attrs,
				R.styleable.MoveBallversion3);

		// 获取开始的X位置属性值,默认大小为:10
		CircleX = a.getFloat(R.styleable.MoveBallversion3_BallStartX, 10);
		// 获取开始的位置属性值,默认大小为:10
		CircleY = a.getFloat(R.styleable.MoveBallversion3_BallStartY, 10);
		// 获取小球的半径属性值,默认大小为:10
		CircleR = a.getFloat(R.styleable.MoveBallversion3_BallRadius, 10);
		// 获取颜色属性值,默认颜色为:0x990000FF
		BallColor = a.getColor(R.styleable.MoveBallversion3_BallColor, 0x990000FF);

		a.recycle();

	}

	@Override
	public void onDraw(Canvas canves) {
		Paint paint = new Paint();

		paint.setColor(BallColor);
		canves.drawCircle(CircleX, CircleY, CircleR, paint);
	}

	@Override
	public boolean onTouchEvent(MotionEvent motionevent) {

		CircleX = motionevent.getX();
		CircleY = motionevent.getY();
		this.invalidate();

		return true;

	}

}

具体的注释也写了,大家好好看看代码就好了,接下来是activity的代码,也就没什么好说的了:

package com.example.customcomponentsdemo.Activity;

import com.example.customcomponentsdemo.R;

import android.app.Activity;
import android.content.Context;
import android.os.Bundle;

public class MoveBallActivity3 extends Activity{

	@Override
	protected void onCreate(Bundle savedInstanceState){

		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_moveball3);
	}

}

这样我们的第三个教程也就到这里结束了,这次给大家介绍一下如果自定义的View需要有自定义的属性我们该如何处理,这一讲也是这个系列完结篇,自定义View之路还有很远,我也没有举一些很难的例子,我认为基础知识设计流程就是这样了,学习了这些之后自定义控件的制作就剩下的大家在次基础之上发挥了!之后如果有比较好的例子我还会继续补充的!

源码的地址是:http://download.csdn.net/detail/sunmc1204953974/7738915

希望大家能学到东西!

另外我也是学生,如果有写的不好或者有错误的地方还请大家多多指教,谢谢!

Android 自定义控件开发入门 (三)

时间: 2024-10-02 03:00:29

Android 自定义控件开发入门 (三)的相关文章

Android 自定义控件开发入门(一)

作为一个有创意的开发者,或者软件对UI设计的要求比较高,你经常会遇到安卓自带的控件无法满足你的需求的情况,这种时候,我们只能去自己去实现适合项目的控件.同时,安卓也允许你去继承已经存在的控件或者实现你自己的控件以便优化界面和创造更加丰富的用户体验. 那么怎样来创建一个新的控件呢? 这得看需求是怎样的了. 1.需要在原生控件的基本功能上进行扩展,这个时候你只需要继承并对控件进行扩展.通过重写它的事件,onDraw ,但是始终都保持都父类方法的调用.如从已有的高级控件上继承,例如继承一个TextVi

Android 自定义控件开发入门(二)

上一次我们讲了一堆实现自定义控件的理论基础,列举了View类一些可以重写的方法,我们对这些方法的重写是我们继承View类来派生自定义控件的关键 我通过一个最简单的例子给大家展示了这一个过程,无论是多么复杂的自定义控件,思路总是这样子的,但是因为我们仅仅重写了onDraw方法使得大家觉得怪怪的,作为一个控件,我们居然还要为了他的实现为其增加麻烦的监听,这就不能叫做控件了. 下面再给大家介绍一个经常重写的方法法:publicboolean onTouchEvent (MotionEvent even

Android Wear 开发入门

大家好,我是陆嘉杰,我是一名Android开发者.我想和大家进行一些技术交流,希望越来越多的人能和我成为好朋友. 大家都知道,智能手表是下一个开发的风口,而这方面的技术又属于前沿,所以和大家分享下Android Wear的开发流程. 首先,我推荐大家使用Android Studio来进行Wear的开发,这也是谷歌推荐的,本次讲授过程也将以Android Studio作为开发集成环境进行. 下面我们来创建Android Wear项目. 请注意,Android Wear项目中同时包含mobile和w

AppleWatch开发入门三——代码交互与控制器生命周期

AppleWatch开发入门三--代码交互与控制器生命周期 一.引言 在前两篇博客中,讨论了关于watch开发中框架与界面布局相关,然而主要的逻辑,终究还是要通过代码来实现的,在我们创建了项目之后,就会生成InterfaceController这个文件,它就是我们storyBoard中的入口视图控制器. 二.代码交互与控制器声明周期 storyBoard中的控件我们可以通过拖拽的方式关联到文件中,Action和Outlet两种关联方式基本可以达到我们修改控件和处理业务逻辑的需求. WKInter

一看就懂的Android APP开发入门教程

一看就懂的Android APP开发入门教程 作者: 字体:[增加 减小] 类型:转载 这篇文章主要介绍了Android APP开发入门教程,从SDK下载.开发环境搭建.代码编写.APP打包等步骤一一讲解,非常简明的一个Android APP开发入门教程,需要的朋友可以参考下 工作中有做过手机App项目,前端和android或ios程序员配合完成整个项目的开发,开发过程中与ios程序配合基本没什么问题,而android各种机子和rom的问题很多,这也让我产生了学习android和ios程序开发的

Android Wear 开发入门——如何创建一个手机与可穿戴设备关联的通知(Notification)

创建通知 为了创建在手机与可穿戴设备中都能展现的通知,可以使用 NotificationCompat.Builder.通过该类创建的通知,系统会处理该通知是否展现在手机或者穿戴设备中. 导入必要的类库 在开发之前首先需要导入以下类库 importandroid.support.v4.app.NotificationCompat; importandroid.support.v4.app.NotificationManagerCompat; importandroid.support.v4.app

Android APP开发入门教程

这篇文章主要介绍了Android APP开发入门教程,从SDK下载.开发环境搭建.代码编写.APP打包等步骤一一讲解,非常简明的一个Android APP开发入门教程,需要的朋友可以参考下. 工作中有做过手机App项目,前端和android或ios程序员配合完成整个项目的开发,开发过程中与ios程序配合基本没什么问题,而android各种机子和rom的问题很多,这也让我产生了学习android和ios程序开发的兴趣.于是凌晨一点睡不着写了第一个android程序HelloAndroid,po出来

Android蓝牙开发入门

目录: 1. 蓝牙简史,现状 2. 蓝牙的应用场景 3. 蓝牙相关概念 4. Android蓝牙开发 1. 蓝牙简史: 蓝牙( Bluetooth)是一种无线技术标准,可以实现短距离(通常是几米范围之内)的无线通信.蓝牙技术始于1994年,迄今已经发展了超过20年.本质上它和其它几种射频通信技术类似,比如手机移动通信,近场通信技术(NFC),都是通过电磁波来实现不同设备的信息交换.区别在于无线电波的频率和发射功率不一样,从而传输距离也不一样. 2. 蓝牙的应用场景: l 移动电话和免提设备之间的

Android自定义控件View(三)组合控件

不少人应该见过小米手机系统音量控制UI,一个圆形带动画效果的音量加减UI,效果很好看.它是怎么实现的呢?这篇博客来揭开它的神秘面纱.先上效果图 相信很多人都知道Android自定义控件的三种方式,Android自定义控件View(一)自绘控件,Android自定义控件View(二)继承控件,还有就是这一节即将学习到的组合控件.我们通过实现圆形音量UI来讲解组合控件的定义和使用. 组合控件 所谓组合控件就是有多个已有的控件组合而成一个复杂的控件.比如上图的音量控件就是一个完美的组合控件.我们来分析