Android官方DataBinding(十二):双向绑定之反向绑定的InverseBindingMethods改造和实现

简介: Android官方DataBinding(十二):双向绑定之反向绑定的InverseBindingMethods改造和实现在附录文章十、十一的基础上,使用InverseBindingMethod进行双向绑定和反向绑定操作。
Android官方DataBinding(十二):双向绑定之反向绑定的InverseBindingMethods改造和实现


在附录文章十、十一的基础上,使用InverseBindingMethod进行双向绑定和反向绑定操作。


以下是完整示例代码。
建立model:
package zhangphil.test;

import android.databinding.BaseObservable;
import android.databinding.ObservableBoolean;

/**
 * Created by Phil on 2017/8/25.
 */

public class ViewModel extends BaseObservable {
    public final ObservableBoolean isRefreshing = new ObservableBoolean();
}



写布局activity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto">

    <data>

        <import type="android.view.View" />

        <variable
            name="model"
            type="zhangphil.test.ViewModel" />
    </data>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">

        <ProgressBar
            android:layout_width="100dp"
            android:layout_height="100dp"
            android:layout_gravity="center_horizontal"
            android:visibility="@{model.isRefreshing ? View.VISIBLE:View.GONE}" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="50dp"
            android:background="@android:color/holo_red_light"
            android:gravity="center"
            android:text="加载中..."
            android:visibility="@{model.isRefreshing ? View.VISIBLE:View.GONE}" />

        <zhangphil.test.PhilView
            android:id="@+id/philview"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:background="@android:color/holo_blue_light"
            app:refreshing="@={model.isRefreshing}">

            <LinearLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:orientation="vertical">

                <TextView
                    android:layout_width="match_parent"
                    android:layout_height="150dp"
                    android:gravity="center"
                    android:text="z" />

                <TextView
                    android:layout_width="match_parent"
                    android:layout_height="150dp"
                    android:gravity="center"
                    android:text="h" />

                <TextView
                    android:layout_width="match_parent"
                    android:layout_height="150dp"
                    android:gravity="center"
                    android:text="a" />

                <TextView
                    android:layout_width="match_parent"
                    android:layout_height="150dp"
                    android:gravity="center"
                    android:text="n" />

                <TextView
                    android:layout_width="match_parent"
                    android:layout_height="150dp"
                    android:gravity="center"
                    android:text="g" />
            </LinearLayout>

        </zhangphil.test.PhilView>
    </LinearLayout>
</layout>





测试的MainActivity.java:

package zhangphil.test;

import android.databinding.DataBindingUtil;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.View;

import zhangphil.test.databinding.ActivityMainBinding;

public class MainActivity extends AppCompatActivity {
    private final String TAG = "MainActivity调试";
    private ViewModel mViewModel;
    private ActivityMainBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        final GestureDetector mGestureDetector = createGestureDetector();

        binding = DataBindingUtil.setContentView(this, R.layout.activity_main);

        mViewModel = new ViewModel();
        binding.setModel(mViewModel);

        binding.philview.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View view, MotionEvent motionEvent) {
                mGestureDetector.onTouchEvent(motionEvent);
                return false;
            }
        });
    }

    private GestureDetector createGestureDetector() {
        // mGestureDetector用于监测用户在手机屏幕上的上滑和下滑事件。
        GestureDetector mGestureDetector = new GestureDetector(this, new GestureDetector.SimpleOnGestureListener() {
            @Override
            public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
                //处于顶部且用户再下拉
                if ((distanceY < 0) && (binding.philview.getScrollY() == 0)) {
                    if (mViewModel.isRefreshing.get()) {
                        Log.d(TAG, "加载中,请勿重复加载");
                    } else {
                        Log.d(TAG, "开始下拉刷新...");

                        //执行下拉刷新/加载更多事务
                        loadMore();
                    }
                }

                return super.onScroll(e1, e2, distanceX, distanceY);
            }
        });

        return mGestureDetector;
    }

    private void loadMore() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                //通过设置布尔值改变View
                mViewModel.isRefreshing.set(true);

                try {
                    //假设这里做了一个长时间的耗时操作
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                //通过设置布尔值改变View
                mViewModel.isRefreshing.set(false);
            }
        }).start();
    }
}



关键的PhilView.java:
package zhangphil.test;

import android.content.Context;
import android.databinding.InverseBindingListener;
import android.databinding.InverseBindingMethod;
import android.databinding.InverseBindingMethods;
import android.support.v4.widget.NestedScrollView;
import android.util.AttributeSet;
import android.util.Log;

/**
 * Created by Phil on 2017/9/1.
 */

@InverseBindingMethods({@InverseBindingMethod(type = PhilView.class, attribute = "refreshing", event = "refreshingAttrChanged")})

public class PhilView extends NestedScrollView {

    private String TAG = "调试";
    private boolean isRefreshing = false;

    public PhilView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public void setRefreshing(boolean refreshing) {
        if (isRefreshing == refreshing) {
            //防止死循环
            Log.d(TAG, "重复设置");
            return;
        } else {
            Log.d(TAG, "setRefreshing " + refreshing);
            isRefreshing = refreshing;
        }
    }

    public boolean getRefreshing() {
        return isRefreshing;
    }

    public void setRefreshingAttrChanged(InverseBindingListener inverseBindingListener) {
        if (inverseBindingListener == null) {
            Log.e(TAG, "InverseBindingListener为空!");
        } else {
            Log.d(TAG, "setRefreshingAttrChanged");
            inverseBindingListener.onChange();
        }
    }
}




对比本篇文章的PhilView.java代码和附录十、十一的异同,最关键的地方是用InverseBindingMethod重新实现。InverseBindingMethod在注解定义的地方,有几点需要特别注意:
(1)InverseBindingMethod里面的event事件不是必须的,如果在InverseBindingMethod里面没有定义,那么Android系统自己会自动匹配查找。Android系统自动匹配查找的原则:根据定义的attribute值后面追加”AttrChanged”形成默认方法名进行匹配查找。比如,如果自己定义了attribute=”xxx”,那么Android系统自动会匹配查找xxxAttrChanged方法,该方法是set开头,那么就最终变成:setXxxAttrChanged
(2)如果开发者在event里面自己随意定义了一个方法名,那么必须严格一致确保类里面有这个方法,比如,如果用户在InverseBindingMethod的event里面任意定义了一个方法“abcdefg”,那么必须在该注解类有一个同名方法如setAbcdefg(),然后在这里面调用InverseBindingListener的onChange()。
(3)InverseBindingMethod中定义的attribute值,即为开发者想要和xml布局里面的app:xxx绑定的值。
(4)如果基于InverseBindingMethod,在绑定注解类里面get和set方法,后面的方法名即为attribute的值,举例,如果,attribute=”xxx”,那么set和get方法即为setXxx()和getXxx()。



附录:
1,《Android官方DataBinding简例(一)》链接:http://blog.csdn.net/zhangphil/article/details/77322530    
2,《Android官方DataBinding(二):动态数据更新notifyPropertyChanged》链接:http://blog.csdn.net/zhangphil/article/details/77328688   
3,《Android官方DataBinding(三):RecyclerView 使用ViewDataBinding更新数据》链接:http://blog.csdn.net/zhangphil/article/details/77367432  
4,《Android官方DataBinding(四):BindingAdapter》链接:http://blog.csdn.net/zhangphil/article/details/77374211 
5,《Android官方DataBinding(五):ObservableMap,ObservableArrayMap》链接:http://blog.csdn.net/zhangphil/article/details/77448933
6,《Android官方DataBinding(六): @= 操作符进行双向绑定》链接:http://blog.csdn.net/zhangphil/article/details/77454045 
7,《Android官方DataBinding(七):BindingMethods与BindingMethod》链接:http://blog.csdn.net/zhangphil/article/details/77479843 
8,《Android官方DataBinding(八):Lambda长表达式事件处理》链接:http://blog.csdn.net/zhangphil/article/details/77503013 
9,《Android官方DataBinding(九):反向绑定,View变化结果回写进数据模型中》链接:http://blog.csdn.net/zhangphil/article/details/77649256  
10,《Android官方DataBinding(十):双向绑定之基于InverseBindingAdapter的反向绑定》链接:http://blog.csdn.net/zhangphil/article/details/77767513 
11,《Android官方DataBinding(十一):对于双向绑定之反向绑定的改进和简化》链接:http://blog.csdn.net/zhangphil/article/details/77772671 

相关文章
|
4月前
|
存储 数据库 Android开发
🔥Android Jetpack全解析!拥抱Google官方库,让你的开发之旅更加顺畅无阻!🚀
【7月更文挑战第28天】在Android开发中追求高效稳定的路径?Android Jetpack作为Google官方库集合,是你的理想选择。它包含多个独立又协同工作的库,覆盖UI到安全性等多个领域,旨在减少样板代码,提高开发效率与应用质量。Jetpack核心组件如LiveData、ViewModel、Room等简化了数据绑定、状态保存及数据库操作。引入Jetpack只需在`build.gradle`中添加依赖。例如,使用Room进行数据库操作变得异常简单,从定义实体到实现CRUD操作,一切尽在掌握之中。拥抱Jetpack,提升开发效率,构建高质量应用!
65 4
|
5月前
|
Android开发 Kotlin
Android面试题 之 Kotlin DataBinding 图片加载和绑定RecyclerView
本文介绍了如何在Android中使用DataBinding和BindingAdapter。示例展示了如何创建`MyBindingAdapter`,包含一个`setImage`方法来设置ImageView的图片。布局文件使用`&lt;data&gt;`标签定义变量,并通过`app:image`调用BindingAdapter。在Activity中设置变量值传递给Adapter处理。此外,还展示了如何在RecyclerView的Adapter中使用DataBinding,如`MyAdapter`,在子布局`item.xml`中绑定User对象到视图。关注公众号AntDream阅读更多内容。
93 1
|
6月前
|
前端开发 Android开发
Android架构组件JetPack之DataBinding玩转MVVM开发实战(四)
Android架构组件JetPack之DataBinding玩转MVVM开发实战(四)
Android架构组件JetPack之DataBinding玩转MVVM开发实战(四)
|
6月前
|
JSON Android开发 数据格式
Android框架-Google官方Gson解析,android开发实验报告总结
Android框架-Google官方Gson解析,android开发实验报告总结
|
6月前
|
存储 前端开发 测试技术
Android 官方架构中的 UseCase 该怎么写?
Android 官方架构中的 UseCase 该怎么写?
164 0
|
6月前
|
Java 开发工具 Android开发
SLS:使用 OTel 官方 SDK 采集 Android、iOS Trace 数据实践
本文介绍了使用 OTel 官方 SDK 采集 Android、iOS Trace 数据实践。
422 7
SLS:使用 OTel 官方 SDK 采集 Android、iOS Trace 数据实践
|
Android开发 数据格式 XML
Android官方DataBinding(十一):对双向绑定之反向绑定的改进和简化
Android官方DataBinding(十一):对于双向绑定之反向绑定的改进和简化 本文接附录十文章,对附录十文章的代码和逻辑进行改进和简化,使得代码更易于阅读和理解,进一步清晰化文章十的下拉刷新实现代码和逻辑。
1160 0
|
4天前
|
编解码 Java Android开发
通义灵码:在安卓开发中提升工作效率的真实应用案例
本文介绍了通义灵码在安卓开发中的应用。作为一名97年的聋人开发者,我在2024年Google Gemma竞赛中获得了冠军,拿下了很多项目竞赛奖励,通义灵码成为我的得力助手。文章详细展示了如何安装通义灵码插件,并通过多个实例说明其在适配国际语言、多种分辨率、业务逻辑开发和编程语言转换等方面的应用,显著提高了开发效率和准确性。
|
2天前
|
Android开发 开发者 UED
安卓开发中自定义View的实现与性能优化
【10月更文挑战第28天】在安卓开发领域,自定义View是提升应用界面独特性和用户体验的重要手段。本文将深入探讨如何高效地创建和管理自定义View,以及如何通过代码和性能调优来确保流畅的交互体验。我们将一起学习自定义View的生命周期、绘图基础和事件处理,进而探索内存和布局优化技巧,最终实现既美观又高效的安卓界面。
13 5
|
1天前
|
JSON Java Android开发
探索安卓开发之旅:打造你的第一个天气应用
【10月更文挑战第30天】在这个数字时代,掌握移动应用开发技能无疑是进入IT行业的敲门砖。本文将引导你开启安卓开发的奇妙之旅,通过构建一个简易的天气应用来实践你的编程技能。无论你是初学者还是有一定经验的开发者,这篇文章都将成为你宝贵的学习资源。我们将一步步地深入到安卓开发的世界中,从搭建开发环境到实现核心功能,每个环节都充满了发现和创造的乐趣。让我们开始吧,一起在代码的海洋中航行!