In the usual development process, it is inevitable to catch up with the schedule or write a function in a relatively short time, we are generally simple and rough to solve the problem for the purpose, I think for such code, and then think carefully, maybe there will be a new discovery, today I met such a small demand.
Requirements are as follows:
As shown in the figure below, there are two input fields, one button, and the requirement is that the button lights up when both EditTexts enter something.
At that time, it was close to work, and to immediately send the package, time is tight, and must solve this problem, so simply use the simplest way to limit the implementation is king, so hurried to write the code is like this:
et1.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { } @Override public void onTextChanged(CharSequence s, int start, int before, Int count) {} @override public void afterTextChanged(Editable s) {// Check whether the text of both editText is emptyif(Util.checkEmpty(et1.getText().toString().trim())
&& Util.checkEmpty(et2.getText().toString().trim())){
btnActive.setEnabled(true);
}else{
btnActive.setEnabled(false); }}}); et2.addTextChangedListener(newTextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { } @Override public void onTextChanged(CharSequence s, int start, int before, Int count) {} @override public void afterTextChanged(Editable s) {// Check whether the text of both editText is emptyif(Util.checkEmpty(et1.getText().toString().trim())
&& Util.checkEmpty(et2.getText().toString().trim())){
btnActive.setEnabled(true);
}else{
btnActive.setEnabled(false); }}});Copy the code
A CTRL + C and CTRL + V, achieved, when the heart was broken, in fact, feel where uncomfortable, if there are 5, will it feel a little long, at that time also so thought, the day first finished the package, sent out.
The second day
If you want to add an addTextChanged method to each EditText, you need to add an addTextChanged method to your current Activity. If you want to add an addTextChanged method to each EditText, you need to add an addTextChanged method to your current Activity.
class MainActivity extends Activity implements TextWatcher{ @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState);setContentView(R.layout.activity_mian);
et1.addTextChangedListener(this);
et2.addTextChangedListener(this);
}
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
}
@Override
public void afterTextChanged(Editable s) {
btnActive.setEnabled(checkState());
}
public boolean checkState() {
returnUtil.checkEmpty(et1.getText().toString().trim()) && Util.checkEmpty(et2getText().toString().trim()); }}Copy the code
So this looks a lot cleaner, at least a little bit better than before, and then it feels a lot like the observer mode, why, you see the Button is a lot like the observer, the EditText is a lot like the observed, the Button is looking at the changes in the EditText text, As soon as it changes, the Button should be notified. The Button loops through all the observed objects to see if it meets the requirements. Based on this idea, I created my third version, which is to customize a Button. It only takes one line of code, so the end result looks like this:
btnActive.observer(et1,et2);Copy the code
Customize Button code as follows:
public class ButtonObserver extends android.support.v7.widget.AppCompatButton implements TextWatcher { ArrayList<EditText> list = new ArrayList<>(); public ButtonObserver(Context context) { this(context,null); } public ButtonObserver(Context context, AttributeSet attrs) { this(context, attrs,0); } public ButtonObserver(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); } /** * watch * @param ets */ public void observer(EditText... Ets){// Iterate over all etsfor(EditText et : ets){
et.addTextChangedListener(this);
list.add(et);
}
}
@Override
public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {
}
@Override
public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {
}
@Override
public void afterTextChanged(Editable editable) {
setEnabled(checkEmpty());
}
public boolean checkEmpty(){
boolean isFlag = true;
for(EditText et : list){
if(TextUtils.isEmpty(et.getText().toString().trim())){
isFlag = false;
break; }}returnisFlag; }}Copy the code
Ok, now, only need to write one line of code in the activity can be finished, I think it should be like this, in order to facilitate the next time for my own use or others with simple point, with one line of code to solve things, don’t write two lines of code, can encapsulate encapsulation, code looks is also very clean, at this point, I think should be ok.
If only we could do something like ButterKnife, we could do it by adding a comment to the Button observer, passing in the id of the Edittext we want to see, and registering it in the onCreate method.
@MyTextWatcher({R.id.et1,R.id.et2})
Button btnActive;
AnnotateUtils.register(this);Copy the code
It looks very good, but it seems that I have not learned the annotation for many years, so I began to look for information on the Internet to learn the knowledge about annotation first, and during the reflection of relevant knowledge, well, then a piece of it.
After a whole afternoon of study, OK, I probably know how to use it, about the knowledge of annotations, I will write a separate article, here is not much to say, first of all, let’s define a custom annotation:
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyTextWatcher {
int[] value();
}Copy the code
Target identifies the Target range to which I apply this annotation. ElementType.FIELD is applied to the property variable. Retention is the Retention period when code is run after it’s compiled. In this case, runtime
The logic implementation of this ButterKnife requires a different class. Recall that when the ButterKnife was used, it was not necessary to call a ButterKnife,bind(this), and similarly: Eventbus.getdefault ().register(this); annotateUtils.register (this); eventbus.getDefault ().register(this);
public class AnnotateUtils { public static void register(final Activity activity) { final ArrayList<EditText> list = new ArrayList<>(); // Get the activity Class<? extends Activity> object = activity.getClass(); Field[] Fields = object.getdeclAredFields (); // Iterate over all fieldsfor(final Field Field: fields) {MyTextWatcher MyTextWatcher = Field. GetAnnotation (myTextwatcher.class);if(myTextWatcher ! = null) {int[] viewId = myTextwatcher.value (); // Get the parameter for the field annotation, which is the control Id we pass infor (int id : viewId) {
EditText et = activity.findViewById(id);
list.add(et);
et.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) { } @Override public void onTextChanged(CharSequence charSequence, int i, int i1, Int i2) {} @override public void afterTextChanged(Editable Editable) {try {// Field is the field of the current annotation field.true); Button btn = (Button) field.get(activity); btn.setEnabled(checkEmpty(list)); } catch (Exception e) { e.printStackTrace(); }}}); } } } } }Copy the code
This allows us to add a ButterKnife annotation like @bindview (R.I.D.tv1) to the Button in our code, which solves the problem we started with. Are automatically generated in the compilation period of the code, will not affect the program run after the impact of performance, and I this is in the runtime, is a little performance consumption, if you can dynamically generate that feel perfect, I have not thought of a good solution, there are friends who know, you can private messages or comments oh.