MVVM的简单使用

2019-04-17  本文已影响0人  技术客栈

我们知道在今年MVP的设计模式非常的火。MVP 是由 MVC 演变而来,在代码的解耦层次更加的独到!避免了更多的代码耦合度。在MVP 中 将 M(modle)层 与 V(View) 层脱离开来,更好的应证了代码的解耦,更多关于MVC、MVP的设计模式的区别可以查看相关的资料,在这里我们简单的梳理下由MVP 演变至MVVM 的过程,MVP重要的核心思想是按业务进行划分,为了更好的解耦,我们队View 也就是Activity 进行拆分,由此而演变为MVVM。
在学习MVVM 之前我们应该先了解下Google 十分NB的 的一个东西 就是,Data Binding Library它的使用方式,我们在这里不做过多的解释,可以参考Google提供的API ,https://developer.android.com/topic/libraries/data-binding/index.html

我们先看Activity

/**
 * 有MVP 演变至 MVVM
 * 也就是对View Activity 进行拆分
 *
 */
public class MainActivity extends AppCompatActivity implements LoginView, ResultCallBack {

    private ProgressBar progressBar;
    private Login user;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //DataBinding 数据绑定的形式 ,根据API的实现书写即可
        ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
        progressBar = (ProgressBar) findViewById(R.id.progressBar);
        user = new Login();
        binding.setUser(user);
        LoginEvent event = new LoginEvent(user);
        binding.setEvent(event);


    }

    public void login(View view) {
        Presenter presenter = new Presenter();
        presenter.login(this,this);
    }


    @Override
    public String getUserName() {
        return user.userName;
    }

    @Override
    public String getPassword() {
        return user.password;
    }

    @Override
    public void showProgress() {
        progressBar.setVisibility(View.VISIBLE);
    }

    @Override
    public void dismissProgress() {
        progressBar.setVisibility(View.GONE);
    }

    @Override
    public void showToast(String toast) {
        Toast.makeText(MainActivity.this, toast, Toast.LENGTH_SHORT).show();

    }

    @Override
    public void Success(Login login) {
        Intent intent = new Intent(this,SuccessActivity.class);
        intent.setAction(this.getPackageName());
        Bundle bundle = new Bundle();
        bundle.putSerializable("javaBean",login);
        intent.putExtras(bundle);
        startActivity(intent);
    }

    @Override
    public void Failed(String error) {

    }
}

2、Interface 接口的定义

/**
 * Created by wangly on 2016/9/6.
 * 用于处理Thread+Handler+InterfaceCallBack 的形式
 */
public interface HandlerCallBack {

    void Success(Login login);

    void Failed(String error);
}
/**
 * Created by wangly on 2016/8/22.
 */
public interface LoginListener {
    void login(String username, String password, HandlerCallBack back);
}

/**
 * Created by wangly on 2016/8/22.
 *
 */
public interface ResultCallBack {
    /**
     * 登陆成功
     * @param login  JavaBean
     */
    void Success(Login login);

    /**
     * 登陆失败
     * @param error ErrorException
     */
    void Failed(String error);
}

3、DataBinding 中Event 时间的绑定中的内容


/**
 * Created by wangly on 2016/9/5.
 */
public class LoginEvent {

    private Login user;

    public LoginEvent(Login login) {
        this.user = login;
    }


    public TextWatcher userNameWatcher = 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) {
            user.userName = editable.toString();
        }
    };


    public TextWatcher passWordWatcher = 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) {
            user.password = editable.toString();
        }
    };


}

4、domain JavaBean


/**
 * Created by wangly on 2016/8/22.
 */
public class Login implements Serializable{
    public String userName;
    public String password;

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    @Override
    public String toString() {
        return "Login{" +
                "password='" + password + '\'' +
                ", userName='" + userName + '\'' +
                '}';
    }
}

5、Presenter 的内容

public class Presenter {
    private LoginUtils loginBiz;
    private ResultCallBack back;
    public void login(final LoginView loginView, final ResultCallBack back) {
        loginBiz = new LoginUtils();
        this.back = back;
        loginView.showProgress();
        loginBiz.login(loginView.getUserName(), loginView.getPassword(), new HandlerCallBack() {
            @Override
            public void Success(Login login) {
                loginView.showToast("登录成功");
                loginView.dismissProgress();
                if (back != null){
                    back.Success(login);
                }

            }

            @Override
            public void Failed(final String error) {
                loginView.showToast(error);
                loginView.dismissProgress();
                if (back != null){
                    back.Failed(error);
                }
            }
        });
    }
}

6、view (MVP 模式)


/**
 * Created by wangly on 2016/8/22.
 *
 * 负责 View 与 Presenter 通讯的通道
 */
public interface LoginView {
    public String getUserName();

    public String getPassword();

    public void showProgress();

    public void dismissProgress();

    void showToast(String toast);

}

7、utils

public class LoginUtils implements LoginListener {
    public final static int SENT_SUCCESS_MESSAGE = 1001;
    public final static int SENT_FAILED_MESSAGE = 1002;
    private HandlerCallBack back;
    private Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            switch (msg.what) {
                case SENT_SUCCESS_MESSAGE:
                    Login login = (Login) msg.obj;
                    back.Success(login);
                    break;

                case SENT_FAILED_MESSAGE:
                    String error = (String) msg.obj;
                    back.Failed(error);
                    break;
                default:
                    break;
            }


        }
    };

    @Override
    public void login(final String username, final String password, final HandlerCallBack back) {
        this.back = back;
        if ("".equals(username) || null == username) {
            back.Failed("请输入用户名");
            return;
        }
        if ("".equals(password) || null == password) {
            back.Failed("请输入密码");
            return;
        }
        new Thread(new Runnable() {

            @Override
            public void run() {
                try {
                    Thread.sleep(2000);
                    if ("张三".equals(username) && "111111".equals(password)) {
                        Login login = new Login();
                        login.setUserName(username);
                        login.setPassword(password);
                        handler.obtainMessage(SENT_SUCCESS_MESSAGE, login).sendToTarget();

                    } else {
                        handler.obtainMessage(SENT_FAILED_MESSAGE, "用户名或密码错误").sendToTarget();
                    }

                } catch (Exception e) {
                    handler.obtainMessage(SENT_FAILED_MESSAGE, "登录失败!" + e.getMessage()).sendToTarget();
                }
            }
        }).start();
    }
}

8、build.gradle 配置 具体的还是的需要参考 DataBinding 的API

  dataBinding{
        enabled = true
    }

9、layout activity_main布局

<?xml version="1.0" encoding="utf-8"?>

<layout xmlns:android="http://schemas.android.com/apk/res/android">

    <data>

        <variable
            name="user"
            type="com.wangly.mvpmodle.modle.Login">

        </variable>

        <variable
            name="event"
            type="com.wangly.mvpmodle.event.LoginEvent">

        </variable>
    </data>


    <LinearLayout xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical"
        tools:context=".activity.MainActivity">

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="50dp">

            <TextView
                android:id="@+id/tv_userName"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_alignParentStart="true"
                android:layout_centerVertical="true"
                android:layout_marginLeft="10dp"
                android:text="UserName" />

            <EditText
                android:id="@+id/et_user"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_alignParentEnd="true"
                android:layout_centerVertical="true"
                android:layout_marginLeft="10dp"
                android:layout_toEndOf="@+id/tv_userName"
                android:addTextChangedListener="@{event.userNameWatcher}"
                android:ems="10"
                android:hint="Please input username"
                android:inputType="textPersonName" />
        </RelativeLayout>

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="50dp">

            <TextView
                android:id="@+id/tv_password"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_alignParentStart="true"
                android:layout_centerVertical="true"
                android:layout_marginLeft="10dp"
                android:text="PassWorld" />

            <EditText
                android:id="@+id/et_password"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_alignParentEnd="true"
                android:layout_centerVertical="true"
                android:layout_marginLeft="10dp"
                android:layout_toEndOf="@+id/tv_password"
                android:addTextChangedListener="@{event.passWordWatcher}"
                android:ems="10"
                android:hint="Please input passworld"
                android:inputType="textPersonName" />
        </RelativeLayout>

        <Button
            android:id="@+id/bt_land"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center_horizontal"
            android:onClick="login"
            android:text="Log in" />

        <ProgressBar
            android:id="@+id/progressBar"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center_horizontal"
            android:layout_marginTop="100dp"
            android:indeterminate="false"
            android:visibility="gone" />

    </LinearLayout>
</layout>

当中更多的是怎么使用DataBinding
1、数据实体传输至控件
2、控件(如:EditText输入的内容) 传输至实体
两者的转换!在啰嗦一下就是当中的使用还是参考Google的API 写个DataBinding 的Demo,然后反过来修改MVP至MVVM。
OK!!

上一篇 下一篇

猜你喜欢

热点阅读