最近在使用知乎Android客户端的时候发现一个十分好玩的UI。如下图:


图1

其实不难看出,知乎app使用了大量原生的Android Material Design控件,包括ToolBar、DrawerLayout、NavigationView、CardView、SwipeRefreshLayout、FloatingActionButton、BottomNavigationBar等等等等。
顺着这个思路,我很快找到上图这个EditText获取到焦点后,hint文本的动画效果实际上也是来源于Material Design库中的一个控件——TextInputLayout。下面简单介绍一下这个控件的用法以及用途。

1 使用

1.1 添加库依赖

要使用这个控件,需要引入appcompat-v7以及Design Support Library两个库。

dependencies {    compile 'com.android.support:appcompat-v7:23.1.1'    compile 'com.android.support:design:23.1.1'}

1.2 布局中的使用

首先我们必须了解的是,TextInputLayout继承于LinearLayout,只能拥有一个直接的ChildView(类似于ScrollView),且这个ChildView只能是EditText

    <android.support.design.widget.TextInputLayout        android:layout_width="match_parent"        android:layout_height="wrap_content">        <EditText            android:id="@+id/edt_password"            android:layout_width="match_parent"            android:layout_height="wrap_content"            android:hint="password"/>    android.support.design.widget.TextInputLayout>

一般来说,EditText有一个hint属性,当Edittext中没有内容时,就会显示文字提示。一旦用户开始输入时,这个文字提示就会消失,取而代之地显示用户的输入。这样有一个坏处就是用户就无法了解到当前自己输入的是关于什么的信息。
而TextInputLayout解决了这个问题,用户开始输入时,hint文字提示会变成EditText上方的标签,并伴随一个向上平移+缩放的优雅动画。
就这样,就可以实现图1的效果。

1.3 其他

  1. Google将 Design Support Library 设计得尽善尽美。库中每个控件的设计颜色都来自style.xmltheme指定的各种颜色。在工程应用的主题中,修改colorAccent属性便可以指定TextInputLayout的标签字体颜色以及EditText的下划线颜色。
    <style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar"> <item name="colorAccent">#3498dbitem>style>
  2. 实际项目中,我们需要对用户的输入进行校验(例如 邮箱格式、密码长度、手机号码等),对用户的输入(即mTextInputLayout.getEditText().getText().toString())在后台进行验证,如果检查到用户输入不合法,可以通过setError()显示输入错误提示,以及setErrorEnabled(fasle)清除错误提示。

    public void onClick(View v) { hideKeyboard(); String username = usernameWrapper.getEditText().getText().toString(); String password = usernameWrapper.getEditText().getText().toString(); if (!validateEmail(username)) {     usernameWrapper.setError("Not a valid email address!"); } else if (!validatePassword(password)) {     passwordWrapper.setError("Not a valid password!"); } else {     usernameWrapper.setErrorEnabled(false);     passwordWrapper.setErrorEnabled(false);     doLogin(); }}`

正常状态.png

错误提示的效果是:标签字体颜色变红,且在EditText下方出现错误信息的标签,这时整个TextInputLayout的高度也会发生变化。如下图:


错误提示.png

2 内部实现

/** * Layout which wraps an {@link android.widget.EditText} (or descendant) to show a floating label * when the hint is hidden due to the user inputting text. * * Also supports showing an error via {@link #setErrorEnabled(boolean)} and * {@link #setError(CharSequence)}. */public class TextInputLayout extends LinearLayout {      ......    @Override    public void addView(View child, int index, ViewGroup.LayoutParams params) {        if (child instanceof EditText) {            setEditText((EditText) child);            super.addView(child, 0, updateEditTextMargin(params));        } else {            // Carry on adding the View...            super.addView(child, index, params);        }    }   private void setEditText(EditText editText) {// Add a TextWatcher so that we know when the text input has changed        mEditText.addTextChangedListener(new TextWatcher() {            @Override            public void afterTextChanged(Editable s) {                updateLabelVisibility(true);                if (mCounterEnabled) {                    updateCounter(s.length());                }            }            @Override            public void beforeTextChanged(CharSequence s, int start, int count, int after) {}            @Override            public void onTextChanged(CharSequence s, int start, int before, int count) {}        });      ......}  }

可以看出,TextInputLayout为内部的EditText设置了一个监听器,监听到有文本输入的时候,通过updateLabelVisibility(true);将EditText的hint转变成上方的标签。

private void updateLabelVisibility(boolean animate) {......        if (hasText || isFocused || isErrorShowing) {            // We should be showing the label so do so if it isn't already            collapseHint(animate);        } else {            // We should not be showing the label so hide it             expandHint(animate);        }}

而在collapseHint(boolean animate)expandHint(boolean animate)内部都是执行animateToExpansionFraction(final float target)方法,通过属性动画来控制TextInputLayout里面EditText上方hint标签的显示。

private void animateToExpansionFraction(final float target) {......        mAnimator.setFloatValues(mCollapsingTextHelper.getExpansionFraction(), target);        mAnimator.start();    }

3 Conclusion

TextInputLayout的简单使用,是Google推出的整个Material Design库的一个缩影:Google将UI视觉效果设计得华丽且流畅,同时代码封装更为优雅,开发者只需要在layout.xml中写好布局文件,就可以轻松在手机屏幕上展现出魔法般的动画效果,实在是妙不可言。

相关链接:

  1. CodeCreating a Login Screen Using TextInputLayout


文/天才木木(简书作者)
原文链接:http://www.jianshu.com/p/dad372912dd5
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

更多相关文章

  1. Android布局详解
  2. Android(安卓)- ToDoList 详解
  3. Android(OPhone) 学习笔记 - 地图(2)
  4. Android网易有道词典案例源码分享
  5. Mac OS下配置 ADB环境变量
  6. Android中Preference的使用以及监听事件分析
  7. Android(安卓)touch 事件的分发和消费机制
  8. android 常见布局及控件的属性详解
  9. Android用户界面UI组件--AdapterView及其子类(三) ExpandableLis

随机推荐

  1. Android驱动使用JNI调用
  2. Your project contains error(s),please f
  3. android数据库编程----SqLiteOpenHelper
  4. Android中FMRadio实现
  5. android 实现home键效果
  6. [Android] 文件夹下文件的个数限制
  7. android 播放音乐示例
  8. Android(安卓)Custom Dialog错误Unable t
  9. Android(安卓)换肤(三) -通过Style来完成
  10. android中使用定时任务