Android 框架之ButterKnife

ButterKnife 是一个Android系统的View注入框架 , 使用它我们再也不用写大量的findViewById 和 setOnClickListener 等代码 . 即使真有使用findViewById的需要也是可以省略强制类型转换 . 所有这个框架真的是很好 .

简单使用

正常情况下我们绑定控件使用findViewById方式如下.

class ExampleActivity extends Activity {
  TextView title;
  TextView subtitle;
  TextView footer;

  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.simple_activity);
    // 绑定控件 
    title = (TextView)findViewById(R.id.title);
    subtitle = (TextView)findViewById(R.id.subtitle);
    footer = (TextView)findViewById(R.id.footer);
  }
}

如果们使用ButterKnife代码如下.

class ExampleActivity extends Activity {
  @BindView(R.id.title) TextView title;
  @BindView(R.id.subtitle) TextView subtitle;
  @BindView(R.id.footer) TextView footer;

  @Override public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.simple_activity);
    // 绑定这个Activity,这是必须的否则无法实现绑定控件.和其他操作.
    ButterKnife.bind(this);
    // TODO Use fields...
  }
}

从上面可以看出我们只需要注解进行绑定即可.省略了大量的findViewById的代码.

配置ButterKnife(8.4.0 版本)

  • 安装ButterKnife插件.


    ButterKnife 插件
  • 配置 build.gradle 文件

    • Module的build.gradle添加如下依赖.
    compile 'com.jakewharton:butterknife:8.4.0'
    annotationProcessor 'com.jakewharton:butterknife-compiler:8.4.0'
    
配置一
  • 在项目的build.gradle 中添加如下代码
classpath 'com.jakewharton:butterknife-gradle-plugin:8.4.0'
配置二

同步项目即可正常是用ButterKnife.

资源绑定

使用ButterKnife的@Bindxxx系列注解(xxx 代表资源类型如 : bool, int, drawable 等)可以绑定预先定义好的资源.

class ExampleActivity extends Activity {
  @BindString(R.string.title) String title;
  @BindDrawable(R.drawable.ic_header) Drawable ic_header;
  @BindColor(R.color.black) int black; 
  @BindDimen(R.dimen.size) Float spacer; 
  @BindInt(R.int.number) Integer number;
}

绑定Fragment中的控件.

通过绑定自定义View可以实现绑定任意对象

// Fragment 中的控件的绑定.
public class MyFragment extends Fragment {
  @BindView(R.id.button1) Button button1;
  @BindView(R.id.button2) Button button2;

  @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View view = inflater.inflate(R.layout.fancy_fragment, container, false);
    ButterKnife.bind(this, view);
    // TODO Use fields...
    return view;
  }
}

简化ListView中的ViewHolder的绑定

public class MyAdapter extends BaseAdapter {
  @Override 
  public View getView(int position, View view, ViewGroup parent) {
    ViewHolder holder;
    if (view != null) {
      holder = (ViewHolder) view.getTag();
    } else {
      view = inflater.inflate(R.layout.whatever, parent, false);
      holder = new ViewHolder(view);
      view.setTag(holder);
    }

    holder.name.setText("ButterKnife 绑定了.");
    // ...
    return view;
  }

  static class ViewHolder {
    @BindView(R.id.title) 
    TextView name;
    @BindView(R.id.text) 
    TextView text;

    public ViewHolder(View view) {
      ButterKnife.bind(this, view);
    }
  }
}

绑定一组View到一个List或者Array中.

// 绑定三个控件到List中.
@BindViews({ R.id.first_view, R.id.second_view, R.id.third_view })
List<EditText> mViews;

使用 ButterKnife的apply 方法可以设置上面集合中的View属性.

// 将三个View都设置成Disable
ButterKnife.apply(mViews, DISABLE);
// 将三个View都设置成不可点击.
ButterKnife.apply(mViews, ENABLED, false);

使用Action 和 Setter 接口.

使用Action和Setter接口可以定义一些行为.

// 使用Action接口定义DISABLE的行为.
static final ButterKnife.Action<View> DISABLE = new ButterKnife.Action<View>() {
  @Override public void apply(View view, int index) {
    view.setEnabled(false);
  }
};
// 使用Setter接口定义ENABLED的行为.
static final ButterKnife.Setter<View, Boolean> ENABLED = new ButterKnife.Setter<View, Boolean>() {
  @Override public void set(View view, Boolean value, int index) {
    view.setEnabled(value);
  }
};

设置任意属性

// 设置所有View的透明度.
ButterKnife.apply(mViews, View.ALPHA, 0.0f);

绑定 listener

使用@OnClick注解可以绑定点击事件,关于绑定方法有如下几种形式

  • 没有参数 : 如果不需要操作控件可以不带参数
// 省略参数
@OnClick(R.id.btn)
public void btnClicked() {
  // do something
}
  • 带有一个参数 : 可以View ,也可以是具体的类型比如Button.
// 绑定按钮的点击事件
@OnClick(R.id.btn)
public void btnClicked(View view) {
  // do something
}
// 直接指定类型.
@OnClick(R.id.btn)
public void btnClicked(Button view) {
  // do something
}
  • 绑定多个控件到同一个方法上.
// 指定多个控件的到同一个方法上
@OnClick({R.id.btn,R.id.btn2,R.id.btn3})
public void btnClicked(View view) {
  // do something
}
  • 自定义控件可以直接绑定自身的Listener,不用指定ID
public class MyButton extends Button {
  @OnClick
  public void onClick() {
    // do something!
  }
}

取消绑定

ButterKnife在绑定时会返回一个Unbinder对象可以使用它来解除绑定.

public class MyFragment extends Fragment {
  @BindView(R.id.button1) 
  Button button1;
  @BindView(R.id.button2) 
  Button button2;
  private Unbinder unbinder;

  @Override 
  public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View view = inflater.inflate(R.layout.fancy_fragment, container, false);
    unbinder = ButterKnife.bind(this, view);
    // TODO Use fields...
    return view;
  }
  @Override 
  public void onDestroyView() {
    super.onDestroyView();
    // 解除绑定
    unbinder.unbind();
  }
}

处理异常

ButterKnife 在绑定控件是如果没有找到个指定的控件,则会抛出异常, 可以使用@Nullable 和 @optional注解来解决.

// 绑定控件使用@Nullable
@Nullable 
@BindView(R.id.btn) 
TextView btn;
// 绑定Listener , @Optional 
@Optional 
@OnClick(R.id.btn) 
void onBtnClicked() {
  // TODO ...
}

Item点击事件绑定.

// ListView Item 点击事件绑定
@OnItemSelected(R.id.list_view)
void onItemSelected(int position) {
  // TODO ...
}
// 绑定两个.Item的listener
@OnItemSelected(value = R.id.maybe_missing, callback = NOTHING_SELECTED)
void onNothingSelected() {
  // TODO ...
}

findById 方法

使用findById可以和findViewById用法一样,但是ButterKnife会自动进行类型转化.

View view = LayoutInflater.from(context).inflate(R.layout.thing, null);
TextView firstName = ButterKnife.findById(view, R.id.first_name);
TextView lastName = ButterKnife.findById(view, R.id.last_name);
ImageView photo = ButterKnife.findById(view, R.id.photo);

OK ButterKnife 8.4.0 简单使用结束.

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

推荐阅读更多精彩内容