Butter Knife 8

  1. // 部分代码省略…
  1. @Override
  1. public View getView(int position, View view, ViewGroup parent) {
  1. ViewHolder holder;
  1. if (view != null) {
  1. holder = (ViewHolder) view.getTag();
  1. } else {
  1. view = inflater.inflate(R.layout.testlayout, parent, false);
  1. holder = new ViewHolder(view);
  1. view.setTag(holder);
  1. }
  1. holder.name.setText(”诺诺”);
  1. // etc…
  1. return view;
  1. }
  1. static class ViewHolder {
  1. @BindView(R.id.title) TextView name;
  1. @BindView(R.id.job_title) TextView jobTitle;
  1. public ViewHolder(View view) {
  1. ButterKnife.bind(this, view);
  1. }
  1. }
  1. }

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.testlayout, parent, false);
  holder = new ViewHolder(view);
  view.setTag(holder);
}
holder.name.setText("诺诺");
// etc...
return view;

}

static class ViewHolder {

@BindView(R.id.title) TextView name;
@BindView(R.id.job_title) TextView jobTitle;
public ViewHolder(View view) {
  ButterKnife.bind(this, view);
}

}

}

六.Butter Knife的使用


1.快速生成注解view和onclick代码

操作流程如图:(AS上操作)

首先将鼠标放置在初始化view布局文件的位置上,然后按快捷键Shift+Alt+S –> Alt+Insert –>Ctrl+Shif+F12,最后选择生成view和点击事件。

第一步:

第二步:

第三步:(注:上面的选项,自己勾选试试自动生成的代码效果就明白了。非常简单。)

2.使用注意事项

1)ButterKinfe的注解标签因版本不同而有所变化。8.0.0之前的Bind标签在8.0.0之后变成了BindView,

而8.7.0之后在绑定view时,要用R2.id.XXX,而不再是常用的R.id.XXX。

Note:我在实际使用时用R.id.XXX和R2.id.XXX都可以,目前本人并未发现有什么异常。自动生成代码时是R.id.XXX;建议使用R2.id.XXX。因为网上都说使用R2是因为library中的R字段的id值不是final类型的,但是你的应用module中确是final类型的……

2)默认情况下,@bind和 listener 的绑定是必需的。如果无法找到目标视图,将抛出一个异常。

要抑制此行为并创建可选绑定,可以将@Nullable注解添加到字段中,或将@Optional注解添加到方法。

**Note:**任何被命名为@Nullable的注解都可以用于成员变量。鼓励使用Android的”support-annotations”库中的@Nullable注解。

eg:

[java] view plain copy

print ?

  1. @Nullable
  1. @BindView(R.id.might_not_be_there)
  1. TextView mightNotBeThere;
  1. @Optional
  1. @OnClick(R.id.maybe_missing)
  1. public void onMaybeMissingClicked() {
  1. // TODO …
  1. }
@Nullable
@BindView(R.id.might_not_be_there) 
TextView mightNotBeThere;
@Optional
@OnClick(R.id.maybe_missing) 
public void onMaybeMissingClicked() {
    // TODO ...
}

3.绑定注解

(注:此处作个统计,后面会有详细解释)

@BindView—->绑定一个view;id为一个view 变量

@BindViews  —-> 绑定多个view;id为一个view的list变量

@BindArray—-> 绑定string里面array数组;@BindArray(R.array.city ) String[] citys ;

@BindBitmap—->绑定图片资源为Bitmap;@BindBitmap( R.mipmap.wifi ) Bitmap bitmap;

@BindBool —->绑定boolean值

@BindColor —->绑定color;@BindColor(R.color.colorAccent) int black;

@BindDimen —->绑定Dimen;@BindDimen(R.dimen.borth_width) int mBorderWidth;

@BindDrawable —-> 绑定Drawable;@BindDrawable(R.drawable.test_pic) Drawable mTestPic;

@BindFloat —->绑定float

@BindInt —->绑定int

@BindString —->绑定一个String id为一个String变量;@BindString( R.string.app_name ) String meg;

4.事件注解

(注:此处作个统计,后面会有详细解释)

@OnClick—->点击事件

@OnCheckedChanged —->选中,取消选中

@OnEditorAction —->软键盘的功能键

@OnFocusChange —->焦点改变

@OnItemClick item—->被点击(注意这里有坑,如果item里面有Button等这些有点击的控件事件的,需要设置这些控件属性focusable为false)

@OnItemLongClick item—->长按(返回真可以拦截onItemClick)

@OnItemSelected —->item被选择事件

@OnLongClick —->长按事件

@OnPageChange —->页面改变事件

@OnTextChanged —->EditText里面的文本变化事件

@OnTouch —->触摸事件

@Optional —->

选择性注入,如果当前对象不存在,就会抛出一个异常,为了压制这个异常,可以在变量或者方法上加入一下注解,让注入变成选择性的,如果目标View存在,则注入, 不存在,则什么事情都不做=如下代码

eg:

[java] view plain copy

print ?

  1. @Optional
  1. @OnCheckedChanged(R.id.cb_test)
  1. public void onCheckedChanged(CompoundButton buttonView,boolean isChecked){
  1. if(isChecked){
  1. tvTest.setText(”被选中…”);
  1. }else{
  1. tvTest.setText(”被取消…”);
  1. }
  1. }

@Optional

@OnCheckedChanged(R.id.cb_test)

public void onCheckedChanged(CompoundButton buttonView,boolean isChecked){

if(isChecked){
    tvTest.setText("被选中...");
}else{
    tvTest.setText("被取消...");
}

}

5.绑定view

1)@BindView() : 布局id 注解

eg:

[java] view plain copy

print ?

  1. @BindView( R2.id.button)
  1. public Button button;

@BindView( R2.id.button)

public Button button;

Note:

button 的修饰类型不能是:private 或者 static 。 否则会报错:错误: @BindView fields must not be private or static. (com.ButterknifeActivity.button)

2)@BindViews(): 布局内多个控件 id 注解

eg:

[java] view plain copy

print ?

  1. public class MainActivity extends AppCompatActivity {
  1. @BindViews({ R2.id.button1, R2.id.button2,  R2.id.button3})
  1. public List buttonList ;
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_main);
  1. ButterKnife.bind(this);
  1. buttonList.get( 0 ).setText( “hello 1 ”);
  1. buttonList.get( 1 ).setText( “hello 2 ”);
  1. buttonList.get( 2 ).setText( “hello 3 ”);
  1. }
  1. }

public class MainActivity extends AppCompatActivity {

@BindViews({ R2.id.button1, R2.id.button2,  R2.id.button3})
public List<Button> buttonList ;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    ButterKnife.bind(this);
    buttonList.get( 0 ).setText( "hello 1 ");
    buttonList.get( 1 ).setText( "hello 2 ");
    buttonList.get( 2 ).setText( "hello 3 ");
}

}

6.绑定资源

(注:只举例常用的几个,使用方式的都是类似的,有不会用的可以给我评论留言。)

1)@BindString() :绑定string 字符串

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @BindView(R2.id.button) //绑定button 控件
  1. public Button button ;
  1. @BindString(R2.string.app_name)  //绑定资源文件中string字符串
  1. String str;
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. button.setText( str );
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@BindView(R2.id.button) //绑定button 控件
public Button button ;
@BindString(R2.string.app_name)  //绑定资源文件中string字符串
String str;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_butterknife);
    //绑定activity
    ButterKnife.bind( this ) ;
    button.setText( str );
}

}

2)@BindArray() : 绑定string里面array数组

eg:

// 资源文件R.array.city

[html] view plain copy

print ?

  1. <resources>
  1. <string name=“app_name”>城市</string>
  1. <string-array name=“city”>
  1. <item>北京市</item>
  1. <item>天津市</item>
  1. <item>哈尔滨市</item>
  1. <item>大连市</item>
  1. <item>香港市</item>
  1. </string-array>
  1. </resources>
<string name="app_name">城市</string>
<string-array name="city">
    <item>北京市</item>
    <item>天津市</item>
    <item>哈尔滨市</item>
    <item>大连市</item>
    <item>香港市</item>
</string-array>

//示例

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @BindView(R2.id.button) //绑定button 控件
  1. public Button button ;
  1. @BindString(R2.string.app_name)  //绑定资源文件中string字符串
  1. String str;
  1. @BindArray(R2.array.city)  //绑定string里面array数组
  1. String [] citys ;
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. button.setText(citys[0]);
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@BindView(R2.id.button) //绑定button 控件
public Button button ;
@BindString(R2.string.app_name)  //绑定资源文件中string字符串
String str;
@BindArray(R2.array.city)  //绑定string里面array数组
String [] citys ;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_butterknife);
    //绑定activity
    ButterKnife.bind( this ) ;
    button.setText(citys[0]);
}

}

3)@BindBitmap( ) : 绑定Bitmap 资源

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @BindView( R2.id.imageView ) //绑定ImageView 控件
  1. public ImageView imageView ;
  1. @BindBitmap(R2.drawable.bm)//绑定Bitmap 资源
  1. public Bitmap bitmap ;
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. imageView.setImageBitmap(bitmap);
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@BindView( R2.id.imageView ) //绑定ImageView 控件
public ImageView imageView ;
@BindBitmap(R2.drawable.bm)//绑定Bitmap 资源
public Bitmap bitmap ;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_butterknife);
    //绑定activity
    ButterKnife.bind( this ) ;
    imageView.setImageBitmap(bitmap);
}

}

**Note:**这里的图片资源引用存在一个小问题:drawable文件夹下的图片使用R或R2引用都没问题;但是mipmap文件夹下的图片资源引用目前只能使用R。

4)@BindColor( ) : 绑定一个颜色值

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @BindView( R2.id.button)  //绑定一个控件
  1. public Button button;
  1. @BindColor( R2.color.colorAccent ) //具体色值在color文件中
  1. int black ;  //绑定一个颜色值
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. button.setTextColor(  black );
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@BindView( R2.id.button)  //绑定一个控件
public Button button;
@BindColor( R2.color.colorAccent ) //具体色值在color文件中
int black ;  //绑定一个颜色值
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_butterknife);
    //绑定activity
    ButterKnife.bind( this ) ;
    button.setTextColor(  black );
}

}

7.点击事件绑定

(注:只举例常用的几个,使用方式的都是类似的,有不会用的可以给我评论留言。)

Note:不用声明view,不用setOnClickLisener()就可以绑定点击事件

1)绑定事件

@OnClick( ) : 绑定控件点击事件

@OnLongClick( ) : 绑定控件长按事件

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @OnClick(R2.id.button1 )   //给 button1 设置一个点击事件
  1. public void showToast(){
  1. Toast.makeText(this, “is a click”, Toast.LENGTH_SHORT).show();
  1. }
  1. @OnLongClick( R2.id.button1 )    //给 button1 设置一个长按事件
  1. public boolean showToast2(){
  1. Toast.makeText(this, “is a long click”, Toast.LENGTH_SHORT).show();
  1. return true ;
  1. }
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@OnClick(R2.id.button1 )   //给 button1 设置一个点击事件
public void showToast(){
    Toast.makeText(this, "is a click", Toast.LENGTH_SHORT).show();
}
@OnLongClick( R2.id.button1 )    //给 button1 设置一个长按事件
public boolean showToast2(){
    Toast.makeText(this, "is a long click", Toast.LENGTH_SHORT).show();
    return true ;
}
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_butterknife);
    //绑定activity
    ButterKnife.bind( this ) ;
}

}

2)指定多个id绑定事件

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. //提示:这是可以自动生成的,大家不要什么都手动敲哦
  1. @OnClick({R.id.ll_product_name, R.id.ll_product_lilv, R.id.ll_product_qixian, R.id.ll_product_repayment_methods})
  1. public void onViewClicked(View view) {
  1. switch (view.getId()) {
  1. case R.id.ll_product_name:
  1. System.out.print(”我是点击事件1”);
  1. break;
  1. case R.id.ll_product_lilv:
  1. System.out.print(”我是点击事件2”);
  1. break;
  1. case R.id.ll_product_qixian:
  1. System.out.print(”我是点击事件3”);
  1. break;
  1. case R.id.ll_product_repayment_methods:
  1. System.out.print(”我是点击事件4”);
  1. break;
  1. }
  1. }
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

//提示:这是可以自动生成的,大家不要什么都手动敲哦
@OnClick({R.id.ll_product_name, R.id.ll_product_lilv, R.id.ll_product_qixian, R.id.ll_product_repayment_methods})
public void onViewClicked(View view) {
    switch (view.getId()) {
        case R.id.ll_product_name:
            System.out.print("我是点击事件1");
            break;
        case R.id.ll_product_lilv:
            System.out.print("我是点击事件2");
            break;
        case R.id.ll_product_qixian:
            System.out.print("我是点击事件3");
            break;
        case R.id.ll_product_repayment_methods:
            System.out.print("我是点击事件4");
            break;
    }
}
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_butterknife);
    //绑定activity
    ButterKnife.bind( this ) ;
}

}

Note:下面是错误的写法,多条点击事件不可以用R2的方式:

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @OnClick({R2.id.ll_product_name, R2.id.ll_product_lilv, R2.id.ll_product_qixian, R2.id.ll_product_repayment_methods})
  1. public void onViewClicked(View view) {
  1. switch (view.getId()) {
  1. case R2.id.ll_product_name:
  1. System.out.print(”我是点击事件1”);
  1. break;
  1. case R2.id.ll_product_lilv:
  1. System.out.print(”我是点击事件2”);
  1. break;
  1. case R2.id.ll_product_qixian:
  1. System.out.print(”我是点击事件3”);
  1. break;
  1. case R2.id.ll_product_repayment_methods:
  1. System.out.print(”我是点击事件4”);
  1. break;
  1. }
  1. }
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@OnClick({R2.id.ll_product_name, R2.id.ll_product_lilv, R2.id.ll_product_qixian, R2.id.ll_product_repayment_methods})  
public void onViewClicked(View view) {  
    switch (view.getId()) {  
        case R2.id.ll_product_name:  
            System.out.print("我是点击事件1");  
            break;  
        case R2.id.ll_product_lilv:  
            System.out.print("我是点击事件2");  
            break;  
        case R2.id.ll_product_qixian:  
            System.out.print("我是点击事件3");  
            break;  
        case R2.id.ll_product_repayment_methods:  
            System.out.print("我是点击事件4");  
            break;  
    }  
}  
@Override  
protected void onCreate(Bundle savedInstanceState) {  
    super.onCreate(savedInstanceState);  
    setContentView(R.layout.activity_butterknife);  
    //绑定activity  
    ButterKnife.bind( this ) ;  
}  

}

如果一定要使用R2的写法,可以单一逐次写,下面写法是正确的

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @OnClick(R2.id.ll_product_name)
  1. public void onViewClicked1(View view) {
  1. System.out.print(”我是点击事件1”);
  1. }
  1. @OnClick(R2.id.ll_product_lilv)
  1. public void onViewClicked2(View view) {
  1. System.out.print(”我是点击事件2”);
  1. }
  1. @OnClick(R2.id.ll_product_qixian)
  1. public void onViewClicked3(View view) {
  1. System.out.print(”我是点击事件3”);
  1. }
  1. @OnClick(R2.id.ll_product_repayment_methods)
  1. public void onViewClicked4(View view) {
  1. System.out.print(”我是点击事件4”);
  1. }
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@OnClick(R2.id.ll_product_name)  
public void onViewClicked1(View view) {  
   System.out.print("我是点击事件1");             
}  
@OnClick(R2.id.ll_product_lilv)  
public void onViewClicked2(View view) {  
   System.out.print("我是点击事件2");   
} 
@OnClick(R2.id.ll_product_qixian)  
public void onViewClicked3(View view) {  
   System.out.print("我是点击事件3");             
}  
@OnClick(R2.id.ll_product_repayment_methods)  
public void onViewClicked4(View view) {  
   System.out.print("我是点击事件4");             
}  
@Override  
protected void onCreate(Bundle savedInstanceState) {  
    super.onCreate(savedInstanceState);  
    setContentView(R.layout.activity_butterknife);  
    //绑定activity  
    ButterKnife.bind( this ) ;  
}  

}

3)自定义View使用注解事件

Note:不用指定id,直接注解OnClick

eg:

[java] view plain copy

print ?

  1. public class MyButton extends Button {
  1. @OnClick
  1. public void onClick() {}
  1. }

public class MyButton extends Button {

@OnClick

public void onClick() {}

}

8.绑定监听

(**注:**本来不打算详细贴这段代码的,英文说明文档中就有,想想为了便于新生理解,还是贴出来吧。前后对比学习,很快就容易理解了。)

1)Listeners可以自动配置到方法中。

eg:

[java] view plain copy

print ?

  1. @OnClick(R.id.submit)
  1. public void submit(View view) {
  1. // TODO submit data to server…
  1. }

@OnClick(R.id.submit)

public void submit(View view) {

// TODO submit data to server…

}

2)对监听器方法的所有参数都是可选的。

eg:

[java] view plain copy

print ?

  1. @OnClick(R.id.submit)
  1. public void submit() {
  1. // TODO submit data to server…
  1. }

@OnClick(R.id.submit)

public void submit() {

// TODO submit data to server…

}

3)定义一个特定类型,它将自动被转换。

eg:

[java] view plain copy

print ?

  1. @OnClick(R.id.submit)
  1. public void sayHi(Button button) {//看参数这里的变化就明白了
  1. button.setText(”Hello!”);
  1. }

@OnClick(R.id.submit)

public void sayHi(Button button) {//看参数这里的变化就明白了
  button.setText("Hello!");
}
4)在单个绑定中指定多个id,用于公共事件处理。

(注:其实这里讲的就是上面的指定多个id绑定点击事件处理,不只是点击事件,其他事件监听也是可以的。很容易理解,不贴代码了。)

5)自定义视图可以通过不指定ID来绑定到它们自己的监听器。

eg:

[java] view plain copy

print ?

  1. public class FancyButton extends Button {
  1. @OnClick
  1. public void onClick() {
  1. // TODO do something!
  1. }
  1. }

public class FancyButton extends Button {

@OnClick

public void onClick() {

// TODO do something!

}

}

6) Listener中多方法注解

方法注解,其对应的监听器有多个回调,可用于绑定到其中任何一个。每个注解都有一个它绑定的默认回调。使用回调参数指定一个替换。

eg:以Spinner为例

原始方式:

[java] view plain copy

print ?

  1. Spinner s=new Spinner(this);
  1. //原始方法:Spinner 条目选择监听事件 正常写法
  1. s.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener(){
  1. @Override
  1. public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
  1. }
  1. @Override
  1. public void onNothingSelected(AdapterView<?> parent) {
  1. }
  1. });

Spinner s=new Spinner(this);

    //原始方法:Spinner 条目选择监听事件 正常写法
    s.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener(){
        @Override
        public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
        }
        @Override
        public void onNothingSelected(AdapterView<?> parent) {
        }
    });

Butter Knife 注解方式:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. /*利用注解对Spinner item  作选择监听事件处理方式*/
  1. @OnItemSelected(R.id.my_spiner)//默认callback为ITEM_SELECTED
  1. void onItemSelected(int position) {
  1. Toast.makeText(this, “position: ” + position, Toast.LENGTH_SHORT).show();
  1. }
  1. /*
  1. * 注解onNothingSelected,需要在注解参数添加一个callback,
  1. * 注意的是Spinner中只要有数据,默认都会选中第0个数据,所以想进入到onNothingSelected()方法,就需要把Adapter中的数据都清空
  1. */
  1. @OnItemSelected(value = R.id.my_spiner, callback = OnItemSelected.Callback.NOTHING_SELECTED)
  1. void onNothingSelected() {
  1. Toast.makeText(this, “Nothing”, Toast.LENGTH_SHORT).show();
  1. }
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind( this ) ;
  1. Spinner s=new Spinner(this);
  1. }
  1. }

public class ButterknifeActivity extends AppCompatActivity {

/*利用注解对Spinner item  作选择监听事件处理方式*/
@OnItemSelected(R.id.my_spiner)//默认callback为ITEM_SELECTED
void onItemSelected(int position) {
    Toast.makeText(this, "position: " + position, Toast.LENGTH_SHORT).show();
}
/*
* 注解onNothingSelected,需要在注解参数添加一个callback,
* 注意的是Spinner中只要有数据,默认都会选中第0个数据,所以想进入到onNothingSelected()方法,就需要把Adapter中的数据都清空
*/
@OnItemSelected(value = R.id.my_spiner, callback = OnItemSelected.Callback.NOTHING_SELECTED)
void onNothingSelected() {
    Toast.makeText(this, "Nothing", Toast.LENGTH_SHORT).show();
}
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_butterknife);
    //绑定activity
    ButterKnife.bind( this ) ;
    Spinner s=new Spinner(this);
}

}

7) @OnCheckedChanged监听的使用

原方法应是:setOnCheckedChangeListener()

在使用注解@OnCheckedChanged的时候,出现了CompoundButton这个参数,因为理解有误,传参有误,还出现了无法转换成CompoundButton的bug。正常使用该注解时,格式如下:

[java] view plain copy

print ?

  1. @OnCheckedChanged({R.id.XXX})
  1. public void OnCheckedChangeListener(CompoundButton view, boolean ischanged ){
  1. }

@OnCheckedChanged({R.id.XXX})

public void OnCheckedChangeListener(CompoundButton view, boolean ischanged ){

}

参数是CompoundButton是变化的按钮控件事件id,ischanged是指该控件是否被点击改变。

具体使用请看示例:

eg:

xml文件:

(注:这里只是举例,方便大家理解,其他相关按钮控件也是通用的。不要纠结具体布局,有看不明白的地方可以给我留言。)

[java] view plain copy

print ?

  1. <?xml version=“1.0” encoding=“utf-8”?>  
  1. <LinearLayout xmlns:android=”http://schemas.android.com/apk/res/android”
  1. android:layout_width=”match_parent”
  1. android:layout_height=”match_parent”
  1. android:orientation=”vertical”>
  1. <RadioGroup
  1. android:id=”@+id/rg_main”
  1. android:layout_width=”fill_parent”
  1. android:layout_height=”48dp”
  1. android:layout_alignParentBottom=”true”
  1. android:background=”@color/white”
  1. android:orientation=”horizontal”>
  1. <RadioButton
  1. android:id=”@+id/rg_home”
  1. android:layout_width=”match_parent”
  1. android:layout_height=”match_parent”
  1. android:focusable=”false”
  1. android:text=”@string/nav_one” />
  1. <RadioButton
  1. android:id=”@+id/rg_wealth”
  1. android:layout_width=”match_parent”
  1. android:layout_height=”match_parent”
  1. android:focusable=”false”
  1. android:text=”@string/nav_two” />
  1. <RadioButton
  1. android:id=”@+id/rg_account”
  1. android:layout_width=”match_parent”
  1. android:layout_height=”match_parent”
  1. android:focusable=”false”
  1. android:text=”@string/nav_four” />
  1.   
<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<RadioGroup
    android:id="@+id/rg_main"
    android:layout_width="fill_parent"
    android:layout_height="48dp"
    android:layout_alignParentBottom="true"
    android:background="@color/white"
    android:orientation="horizontal">
    <RadioButton
        android:id="@+id/rg_home"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:focusable="false"
        android:text="@string/nav_one" />
    <RadioButton
        android:id="@+id/rg_wealth"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:focusable="false"
        android:text="@string/nav_two" />
    <RadioButton
        android:id="@+id/rg_account"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:focusable="false"
        android:text="@string/nav_four" />
</RadioGroup>

类文件中的监听写法:

[java] view plain copy

print ?

  1. @OnCheckedChanged({R.id.rg_home,R.id.rg_wealth,R.id.rg_account})
  1. public void OnCheckedChangeListener(CompoundButton view, boolean ischanged ){
  1. switch (view.getId()) {
  1. case R.id.rg_home:
  1. if (ischanged){//注意:这里一定要有这个判断,只有按钮被点击了,ischanged状态发生改变,才会执行下面的内容
  1. //这里写你的按钮变化状态的UI及相关逻辑
  1. }
  1. break;
  1. case R.id.rg_wealth:
  1. if (ischanged) {
  1. //这里写你的按钮变化状态的UI及相关逻辑
  1. }
  1. break;
  1. case R.id.rg_account:
  1. if (ischanged) {
  1. //这里写你的按钮变化状态的UI及相关逻辑
  1. }
  1. break;
  1. default:
  1. break;
  1. }
  1. }

@OnCheckedChanged({R.id.rg_home,R.id.rg_wealth,R.id.rg_account})

public void OnCheckedChangeListener(CompoundButton view, boolean ischanged ){
    switch (view.getId()) {
        case R.id.rg_home:
            if (ischanged){//注意:这里一定要有这个判断,只有按钮被点击了,ischanged状态发生改变,才会执行下面的内容
                //这里写你的按钮变化状态的UI及相关逻辑
            }
            break;
        case R.id.rg_wealth:
            if (ischanged) {
                //这里写你的按钮变化状态的UI及相关逻辑
            }
            break;
        case R.id.rg_account:
            if (ischanged) {
                //这里写你的按钮变化状态的UI及相关逻辑
            }
            break;
        default:
            break;
    }
}

9.使用findById

Butter Knife仍然包含了findById()方法,用于仍需从一个view ,Activity,或者Dialog上初始化view的时候,并且它可以自动转换类型。

eg:

[java] view plain copy

print ?

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

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 iv = ButterKnife.findById(view, R.id.iv);

10.设置多个view的属性

apply()

作用:允许您立即对列表中的所有视图进行操作。

Action和Setter接口

作用:Action和Setter接口允许指定简单的行为。

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @BindViews({R2.id.first_name, R2.id.middle_name, R2.id.last_name})
  1. List nameViews;
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind(this);
  1. //看不懂的小伙伴,多敲几遍代码,实际操作一下自然就懂了。相信我!如果你重复敲了10遍以上的代码,并认真思考了还没懂,欢迎给我评论留言。
  1. //设置多个view的属性
  1. //方式1:传递值
  1. ButterKnife.apply(nameViews, DISABLE);
  1. //方式2:指定值
  1. ButterKnife.apply(nameViews, ENABLED, false);
  1. 方式3 设置View的Property
  1. ButterKnife.apply(nameViews, View.ALPHA, 0.0f);//一个Android属性也可以用于应用的方法。
  1. }
  1. /*
  1. * Action接口设置属性
  1. */
  1. static final ButterKnife.Action DISABLE = new ButterKnife.Action() {
  1. @Override
  1. public void apply(View view, int index) {
  1. view.setEnabled(false);//目的是使多个view都具备此属性
  1. }
  1. };
  1. /*
  1. * Setter接口设置属性
  1. */
  1. static final ButterKnife.Setter<View, Boolean> ENABLED = new ButterKnife.Setter<View, Boolean>() {
  1. @Override
  1. public void set(View view, Boolean value, int index) {
  1. view.setEnabled(value);//目的是使多个view都具备此属性,可变boolean值是可以传递的
  1. }
  1. };
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@BindViews({R2.id.first_name, R2.id.middle_name, R2.id.last_name})

最后

小编这些年深知大多数初中级Android工程师,想要提升自己,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

资料⬅专栏获取
/80007661)10.设置多个view的属性

apply()

作用:允许您立即对列表中的所有视图进行操作。

Action和Setter接口

作用:Action和Setter接口允许指定简单的行为。

eg:

[java] view plain copy

print ?

  1. public class ButterknifeActivity extends AppCompatActivity {
  1. @BindViews({R2.id.first_name, R2.id.middle_name, R2.id.last_name})
  1. List nameViews;
  1. @Override
  1. protected void onCreate(Bundle savedInstanceState) {
  1. super.onCreate(savedInstanceState);
  1. setContentView(R.layout.activity_butterknife);
  1. //绑定activity
  1. ButterKnife.bind(this);
  1. //看不懂的小伙伴,多敲几遍代码,实际操作一下自然就懂了。相信我!如果你重复敲了10遍以上的代码,并认真思考了还没懂,欢迎给我评论留言。
  1. //设置多个view的属性
  1. //方式1:传递值
  1. ButterKnife.apply(nameViews, DISABLE);
  1. //方式2:指定值
  1. ButterKnife.apply(nameViews, ENABLED, false);
  1. 方式3 设置View的Property
  1. ButterKnife.apply(nameViews, View.ALPHA, 0.0f);//一个Android属性也可以用于应用的方法。
  1. }
  1. /*
  1. * Action接口设置属性
  1. */
  1. static final ButterKnife.Action DISABLE = new ButterKnife.Action() {
  1. @Override
  1. public void apply(View view, int index) {
  1. view.setEnabled(false);//目的是使多个view都具备此属性
  1. }
  1. };
  1. /*
  1. * Setter接口设置属性
  1. */
  1. static final ButterKnife.Setter<View, Boolean> ENABLED = new ButterKnife.Setter<View, Boolean>() {
  1. @Override
  1. public void set(View view, Boolean value, int index) {
  1. view.setEnabled(value);//目的是使多个view都具备此属性,可变boolean值是可以传递的
  1. }
  1. };
  1. }

public class ButterknifeActivity extends AppCompatActivity {

@BindViews({R2.id.first_name, R2.id.middle_name, R2.id.last_name})

最后

小编这些年深知大多数初中级Android工程师,想要提升自己,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。

[外链图片转存中…(img-efHg69gN-1719050367990)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

资料⬅专栏获取

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/733149.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Ansible 自动化运维实践

随着 IT 基础设施的复杂性不断增加&#xff0c;手动运维已无法满足现代企业对高效、可靠的 IT 运维需求。Ansible 作为一款开源的自动化运维工具&#xff0c;通过简洁易用的 YAML 语法和无代理&#xff08;agentless&#xff09;架构&#xff0c;极大简化了系统配置管理、应用部…

物理层(二)

2.2 传输介质 2.2.1 双绞线、同轴电缆、光纤和无线传输介质 传输介质也称传输媒体&#xff0c;是数据传输系统中发送器和接收器之间的物理通路。传输介质可分为:①导向传输介质&#xff0c;指铜线或光纤等&#xff0c;电磁波被导向为沿着固体介质传播:②)非导向传输介质&…

6月27日云技术研讨会 | 中央集中架构新车型功能和网络测试解决方案

会议摘要 “软件定义汽车”新时代下&#xff0c;整车电气电气架构向中央-区域集中式发展已成为行业共识&#xff0c;车型架构的变革带来更复杂的整车功能定义、更多的新技术的应用&#xff08;如SOA服务化、TSN等&#xff09;和更短的车型研发周期&#xff0c;对整车和新产品研…

【yolov8语义分割】跑通:下载yolov8+预测图片+预测视频

1、下载yolov8到autodl上 git clone https://github.com/ultralytics/ultralytics 下载到Yolov8文件夹下面 另外&#xff1a;现在yolov8支持像包一样导入&#xff0c;pip install就可以 2、yolov8 语义分割文档 看官方文档&#xff1a;主页 -Ultralytics YOLO 文档 还能切…

GLM-4V模型学习

智谱AI引领技术前沿&#xff0c;推出了新一代预训练模型GLM-4系列&#xff0c;其中的GLM-4-9B作为开源版本&#xff0c;展现了其在人工智能领域的深厚实力。在语义理解、数学运算、逻辑推理、代码编写以及广泛知识领域的数据集测评中&#xff0c;GLM-4-9B及其人类偏好对齐的版本…

AI音乐模型:创新还是颠覆?

文章目录 AI音乐大模型的崛起音乐创作门槛的降低与兴奋AI音乐作品的版权归属问题创意产业在AI阴影下的生长结语 &#x1f389;欢迎来到AIGC人工智能专栏~探索Java中的静态变量与实例变量 ☆* o(≧▽≦)o *☆嗨~我是IT陈寒&#x1f379;✨博客主页&#xff1a;IT陈寒的博客&…

Apple - Text System Storage Layer Overview

本文翻译整理自&#xff1a;Text System Storage Layer Overview&#xff08;更新日期&#xff1a;2012-09-19 https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/TextStorageLayer/TextStorageLayer.html#//apple_ref/doc/uid/10000087i 文章目录 …

java:JWT的简单例子

【pom.xml】 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><version>2.3.12.RELEASE</version> </dependency> <dependency><groupId>org.springf…

程序猿成长之路之数据挖掘篇——决策树分类算法(1)——信息熵和信息增益

决策树不仅在人工智能领域发挥着他的作用&#xff0c;而且在数据挖掘中也在分类领域中独占鳌头。了解决策树的思想是学习数据挖掘中的分类算法的关键&#xff0c;也是学习分类算法的基础。 什么是决策树 用术语来说&#xff0c;决策树&#xff08;Decision Tree&#xff09;是…

STM32驱动-ads1112

汇总一系列AD/DA的驱动程序 ads1112.c #include "ads1112.h" #include "common.h"void AD5726_Init(void) {GPIO_InitTypeDef GPIO_InitStructure;RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOC, ENABLE );//PORTA、D时钟使能 G…

SQLite数据库(数据库和链表双向转换)

文章目录 SQLite数据库一、SQLite简介1、SQLite和MySQL2、基于嵌入式的数据库 二、SQLite数据库安装三、SQLite的常用命令四、SQLite的编程操作1、SQLite数据库相关API&#xff08;1&#xff09;头文件&#xff08;2&#xff09;sqlite3_open()&#xff08;3&#xff09;sqlite…

Springboot拓展之整合邮件 JavaMail的使用与实操

邮件 电子邮件仍然是我们企业间交往的一种非常常见的方式 发送简单邮件 第一步首先导入坐标 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-mail</artifactId><version>2.6.13</version&…

架构师指南:现代 Datalake 参考架构

这篇文章的缩写版本于 2024 年 3 月 26 日出现在 The New Stack 上。 旨在最大化其数据资产的企业正在采用可扩展、灵活和统一的数据存储和分析方法。这一趋势是由企业架构师推动的&#xff0c;他们的任务是制定符合不断变化的业务需求的基础设施。现代数据湖体系结构通过将数…

设计模式——设计模式原则

设计模式 设计模式原则 单一职责原则&#xff08;SPS&#xff09;&#xff1a; 又称单一功能原则&#xff0c;面向对象五个基本原则&#xff08;SOLID&#xff09;之一 原则定义&#xff1a;一个类应该只有一个发生变化的原因 使用if else进行判断实现不好维护 模式场景&a…

ruoyi添加自己的菜单

先把自己自定义的view填写好 在菜单管理模块 因为我已经新增过&#xff0c;所以就看看我填的啥就行了 我发现一个问题&#xff0c;路由地址可以填index2或者scooldemo/index2都可以&#xff08;这个包含了文件夹路径&#xff09;&#xff0c;反正组件路径一定要填对就可以了。 …

刷代码随想录有感(112):动态规划——组合总和IV

题干&#xff1a; 代码&#xff1a; class Solution { public:int combinationSum4(vector<int>& nums, int target) {vector<int>dp(target 1, 0);dp[0] 1;for(int j 0; j < target; j){for(int i 0; i < nums.size(); i){if(j > nums[i] &…

CATIA_DELMIA_V5R2019安装包下载及安装教程破解

以下为V5-6R2019安装说明 1.将两卷安装文件解压到同一目录内&#xff0c;互相覆盖即可 &#xff08;按用户需要下载 CATIA 或者DELMIA&#xff09; 以上为 CATIA 的安装包 以上为 DELMIA 的安装包 两者合并到一起&#xff0c;同一目录 2.解压后运行setup.exe 如遇到报错&…

【代码随想录】【算法训练营】【第45天】 [198]打家劫舍 [213]打家劫舍II [337]打家劫舍III

前言 思路及算法思维&#xff0c;指路 代码随想录。 题目来自 LeetCode。 day 45&#xff0c;周五&#xff0c;坚持不了一点~ 题目详情 [198] 打家劫舍 题目描述 198 打家劫舍 解题思路 前提&#xff1a; 思路&#xff1a; 重点&#xff1a; 代码实现 C语言 虚拟头…

Python | Leetcode Python题解之第165题比较版本号

题目&#xff1a; 题解&#xff1a; class Solution:def compareVersion(self, version1: str, version2: str) -> int:n, m len(version1), len(version2)i, j 0, 0while i < n or j < m:x 0while i < n and version1[i] ! .:x x * 10 ord(version1[i]) - o…

VBA技术资料MF164:列出文件夹中的所有文件和创建日期

我给VBA的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高自己的工作效率&#xff0c;而且可以提高数据的准确度。“VBA语言専攻”提供的教程一共九套&#xff0c;分为初级、中级、高级三大部分&#xff0c;教程是对VBA的系统讲解&#…