Android编辑信息界面,组合控件的封装

Github地址(完整Demo,欢迎下载)
https://github.com/zhouxu88/ItemGroup

效果图

组合控件1.png
组合控件2.PNG

在很多APP当中,有些页面需要填写很多资料,比如个人中心,还有一些页面,点击每个Item都要跳转页面或者弹框选择。这样的组合控件,相信大家都见过,说到这里,可能有人不服了,不就是一个线性布局或者相对布局吗,我分分钟写出来。是的,我承认你可以搞定,因为我以前也是一个个布局嵌套写出来的,但是我想说,这么多相似的重复布局,你难道真的要一个个去写吗?复制粘贴一个个改,而且有些功能是重复的,比如输入框那个清除输入内容按钮,你也去写,你累吗?这样的代码看上去可读性高吗?所以,我今天就是来和大家一起解决这个问题的,我们完全可以把这种组合控件封装起来,然后专注于处理我们的业务逻辑就好。

实现步骤

1、分析布局(以组合控件1为例子来分析)
从上面图我们可以看出,每一个Item的布局从左往右依次是:标题,输入框,清除输入的按钮,向右的箭头。前2个Item,我们点击中间输入框的时候,直接跳出来软键盘,直接输入文字内容,当有内容的时候,显示清除按钮,没有内容的时候,隐藏清除按钮。后2个Item,是带箭头的,是不可以输入的,提示我们这一项是用来跳转或者点击弹出选择框等。

选择城市.jpg

上面这些个Item的布局很简单,大致结构是这样的

<LinearLayout>
    <LinearLayout>
        <TextView />
        <EditText />
        <!--清除输入内容-->
        <ImageView />
        <!--点击跳转或者弹出选择框-->
        <ImageView />
    </LinearLayout>

    <View
        android:layout_width="match_parent"
        android:layout_height="1dp"
        android:background="@color/item_group_divider" />
</LinearLayout>

然后每一项都这么写一个相似的LinearLayout,然后如果没有右边的箭头,我们又把它去掉,如果我们的项目中也有这种类似的item,而且有很多,那么这个界面的layout的代码会显得很臃肿,维护起来也不清晰。

所以我根据我的项目把这种组合控件封装了一个,可能扩展性不是那么好,但是代码少,思路清晰,符合我的项目需求,你拿过去做适当调整也能用。

下面来介绍怎样封装成为一个通用的控件

2、Item的布局 item_group_layout.xml

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

    <LinearLayout
        android:id="@+id/item_group_layout"
        android:layout_width="match_parent"
        android:layout_height="45dp"
        android:background="@color/white"
        android:gravity="center_vertical"
        android:orientation="horizontal">

        <TextView
            android:id="@+id/title_tv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerVertical="true"
            android:textColor="@color/item_group_title"
            android:textSize="15sp"
            tools:text="姓名" />

        <EditText
            android:id="@+id/content_edt"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:background="@null"
            android:gravity="center_vertical|end"
            android:hint="请输入姓名"
            android:singleLine="true"
            android:paddingStart="10dp"
            android:textColor="@color/item_group_edt"
            android:textSize="13sp" />
        
        <!--清除输入内容-->
        <ImageView
            android:id="@+id/clear_iv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginRight="10dp"
            android:padding="8dp"
            android:scaleType="centerCrop"
            android:src="@mipmap/close" />
        
        <!--点击跳转或者弹出选择框-->
        <ImageView
            android:id="@+id/jt_right_iv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginRight="7dp"
            android:padding="8dp"
            android:scaleType="centerCrop"
            android:src="@mipmap/jiantou_right" />
    </LinearLayout>

    <View
        android:layout_width="match_parent"
        android:layout_height="1dp"
        android:background="@color/item_group_divider" />
</LinearLayout>

3.首先新建一个类,继承自FrameLayout,实现对应的构造方法
我们引入刚才Item的布局,把它添加到这个FrameLayout

public class ItemGroup extends FrameLayout implements View.OnClickListener {

    private LinearLayout itemGroupLayout; //组合控件的布局
    private TextView titleTv; //标题
    private EditText contentEdt; //输入框
    private ImageView clearIv; //清楚输入内容
    private ImageView jtRightIv; //向右的箭头
    private ItemOnClickListener itemOnClickListener; //Item的点击事件
  
  public ItemGroup(@NonNull Context context) {
        super(context);
        initView(context);
    }

    public ItemGroup(@NonNull Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        initView(context);
    }

    public ItemGroup(@NonNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initView(context);
    }

    //初始化View
    private void initView(Context context) {
        View view = LayoutInflater.from(context).inflate(R.layout.item_group_layout, null);
        itemGroupLayout = (LinearLayout) view.findViewById(R.id.item_group_layout);
        titleTv = (TextView) view.findViewById(R.id.title_tv);
        contentEdt = (EditText) view.findViewById(R.id.content_edt);
        clearIv = (ImageView) view.findViewById(R.id.clear_iv);
        jtRightIv = (ImageView) view.findViewById(R.id.jt_right_iv);
        addView(view); //把自定义的这个组合控件的布局加入到当前FramLayout
}
}

4.属性的定义,在完成类的创建后,来自定义相关属性
首先需要在values目录下面新建一个attrs.xml文件 , 自定义相关属性,

其定义格式如下:

<declare-styleable name="自定义属性名称">
    <attr name="属性名称" format="属性类型"/>
</declare-styleable>
<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="ItemGroup">
        <!--标题的文字-->
        <attr name="title" format="string" />
        <!--标题的字体大小-->
        <attr name="title_size" format="dimension" />
        <!--标题的字体颜色-->
        <attr name="title_color" format="color" />
        <!--输入框的内容-->
        <attr name="edt_content" format="string" />
        <!--输入框的字体大小-->
        <attr name="edt_text_size" format="dimension" />
        <!--输入框的字体颜色-->
        <attr name="edt_text_color" format="color" />
        <!--输入框提示的内容-->
        <attr name="edt_hint_content" format="string" />
        <!--输入框的提示字体的字体颜色-->
        <attr name="edt_hint_text_color" format="color" />
        <!--输入框是否可以编辑内容-->
        <attr name="isEditable" format="boolean"/>
        <!--向的右箭头图标是否可见-->
        <attr name="jt_visible" format="boolean"/>
        <!--item布局的内边距-->
        <attr name="paddingLeft" format="dimension"/>
        <attr name="paddingRight" format="dimension"/>
        <attr name="paddingTop" format="dimension"/>
        <attr name="paddingBottom" format="dimension"/>
        
        <attr name="drawable_left" format="reference" />
        <attr name="drawable_right" format="reference" />
        <attr name="line_color" format="color" />
        <attr name="line_height" format="integer" />
    </declare-styleable>
</resources>

属性类型主要包括:
reference 引用
color 颜色
boolean 布尔值
dimension 尺寸值
float 浮点值
integer 整型值
string 字符串
enum 枚举值

5.属性的引入,在定义完属性后,接下来将定义的属性值引入到类中
先获取到各属性,然后将引入的属性值设置到布局控件上。

 /**
     * 初始化相关属性,引入相关属性
     *
     * @param context
     * @param attrs
     */
    private void initAttrs(Context context, AttributeSet attrs) {
        //标题的默认字体颜色
        int defaultTitleColor = context.getResources().getColor(R.color.item_group_title);
        //输入框的默认字体颜色
        int defaultEdtColor = context.getResources().getColor(R.color.item_group_edt);
        //输入框的默认的提示内容的字体颜色
        int defaultHintColor = context.getResources().getColor(R.color.item_group_edt);

        TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.ItemGroup);
        String title = typedArray.getString(R.styleable.ItemGroup_title);
        float paddingLeft = typedArray.getDimension(R.styleable.ItemGroup_paddingLeft, 15);
        float paddingRight = typedArray.getDimension(R.styleable.ItemGroup_paddingRight, 15);
        float paddingTop = typedArray.getDimension(R.styleable.ItemGroup_paddingTop, 5);
        float paddingBottom = typedArray.getDimension(R.styleable.ItemGroup_paddingTop, 5);
        float titleSize = typedArray.getDimension(R.styleable.ItemGroup_title_size, 15);
        int titleColor = typedArray.getColor(R.styleable.ItemGroup_title_color, defaultTitleColor);
        String content = typedArray.getString(R.styleable.ItemGroup_edt_content);
        float contentSize = typedArray.getDimension(R.styleable.ItemGroup_edt_text_size, 13);
        int contentColor = typedArray.getColor(R.styleable.ItemGroup_edt_text_color, defaultEdtColor);
        String hintContent = typedArray.getString(R.styleable.ItemGroup_edt_hint_content);
        int hintColor = typedArray.getColor(R.styleable.ItemGroup_edt_hint_text_color, defaultHintColor);
        //默认输入框可以编辑
        boolean isEditable = typedArray.getBoolean(R.styleable.ItemGroup_isEditable, true);
        //向右的箭头图标是否可见,默认可见
        boolean showJtIcon = typedArray.getBoolean(R.styleable.ItemGroup_jt_visible, true);
        typedArray.recycle();

        //设置数据
        //设置item的内边距
        itemGroupLayout.setPadding((int) paddingLeft, (int) paddingTop, (int) paddingRight, (int) paddingBottom);
        titleTv.setText(title);
        titleTv.setTextSize(titleSize);
        titleTv.setTextColor(titleColor);

        contentEdt.setText(content);
        contentEdt.setTextSize(contentSize);
        contentEdt.setTextColor(contentColor);
        contentEdt.setHint(hintContent);
        contentEdt.setHintTextColor(hintColor);
        contentEdt.setFocusableInTouchMode(isEditable); //设置输入框是否可以编辑
        contentEdt.setLongClickable(false); //输入框不允许长按
        jtRightIv.setVisibility(showJtIcon ? View.VISIBLE : View.GONE);  //设置向右的箭头图标是否可见
}

这里主要通过obtainStyledAttributes方法获取到一个TypedArray对象,然后通过TypedArray对象就可以获取到相对应定义的属性值,相关自定义View的内容,自行查阅

6、在完成上面的步骤以后,就算大体上封装好了,就可以在布局文件里面使用了

调用的xml布局文件

<?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"
    android:orientation="vertical"
    tools:context="com.zx.itemgroup.MainActivity">

    <com.zx.itemgroup.ItemGroup
        android:id="@+id/name_ig"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:edt_hint_content="请输入姓名"
        app:jt_visible="false"
        app:paddingLeft="15dp"
        app:title="姓名" />

    <com.zx.itemgroup.ItemGroup
        android:id="@+id/id_card_ig"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:edt_hint_content="请输入身份证号"
        app:jt_visible="false"
        app:paddingLeft="15dp"
        app:title="身份证" />

    <com.zx.itemgroup.ItemGroup
        android:id="@+id/select_birthday_ig"
        android:layout_width="match_parent"
        android:layout_height="46dp"
        app:edt_hint_content="请选择出生日期"
        app:isEditable="false"
        app:paddingLeft="15dp"
        app:title="出生日期" />

    <com.zx.itemgroup.ItemGroup
        android:id="@+id/select_city_ig"
        android:layout_width="match_parent"
        android:layout_height="46dp"
        app:edt_hint_content="请选择您所在的城市"
        app:isEditable="false"
        app:paddingLeft="15dp"
        app:title="所在城市" />
</LinearLayout>

调用的activity

/**
 * 组合控件封装(提交信息及编辑信息界面及功能)
 */
public class MainActivity extends AppCompatActivity {

    private Context mContext;
    private ItemGroup nameIG, idCardIG, birthdayIG, cityIG;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mContext = this;
        initView();
    }

    private void initView() {
        nameIG = (ItemGroup) findViewById(R.id.name_ig);
        idCardIG = (ItemGroup) findViewById(R.id.id_card_ig);
        birthdayIG = (ItemGroup) findViewById(R.id.select_birthday_ig);
        cityIG = (ItemGroup) findViewById(R.id.select_city_ig);
        birthdayIG.setItemOnClickListener(new ItemGroup.ItemOnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(mContext, "点击了选择出生日期", Toast.LENGTH_SHORT).show();
            }
        });
        cityIG.setItemOnClickListener(new ItemGroup.ItemOnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(mContext, "点击了选择城市", Toast.LENGTH_SHORT).show();
            }
        });
    }
}

扫描下面二维码或者公众号搜索Android老鸟开发经验谈,即可关注【公众号】,持续推出优秀文章

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

推荐阅读更多精彩内容