MVVM 和 Android Data Binding 續(xù)

上一篇文章我們回顧了 MVVM 模式并了解了 Android Data Binding 的相關(guān)知識(shí)添瓷。那么這一章我們來(lái)實(shí)際做一個(gè)簡(jiǎn)單的基于 Android Data Binding 的 MVVM 架構(gòu)的 app。

創(chuàng)建一個(gè) Android 項(xiàng)目

我們先用 Android Studio 創(chuàng)建一個(gè)帶有 Empty Activity 的項(xiàng)目:

new android project
new android 2

自動(dòng)生成的 MainActivity 如下:

package li.fyun.mvvm.view;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;

import li.fyun.mvvm.R;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
}

activity_main.xml:

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

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
                xmlns:tools="http://schemas.android.com/tools"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:paddingBottom="@dimen/activity_vertical_margin"
                android:paddingLeft="@dimen/activity_horizontal_margin"
                android:paddingRight="@dimen/activity_horizontal_margin"
                android:paddingTop="@dimen/activity_vertical_margin"
                tools:context=".view.MainActivity">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello Word"
        />
</RelativeLayout>

增加對(duì) Data Binding 的支持

在 app 的 build.gradle 中增加:

...
android {
    ...
    dataBinding {
        enabled = true
    }
    ...
}

創(chuàng)建 Model

在子 package .model 中創(chuàng)建 Model 類 User:

package li.fyun.mvvm.model;

import java.util.Random;

/**
 * Created by fyunli on 15/12/28.
 */
public class User {

    private static final User[] users = new User[]{
            new User("Andrea"), new User("Betty"), new User("Calvin"), new User("Daisy"), new User("Eason"),
            new User("Frank"), new User("Gary"), new User("Halen"), new User("Iris"), new User("Java")
    };


    private String username;
    private String portrait = "http://ww3.sinaimg.cn/large/620f1e7egw1ey8bjsbmr2j20b40b475e.jpg";

    public User(String username) {
        this.username = username;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPortrait() {
        return portrait;
    }

    public void setPortrait(String portrait) {
        this.portrait = portrait;
    }

    // just a sample logic
    public static User getUser() {
        int index = new Random().nextInt(3);
        return users[index];
    }

}

創(chuàng)建 ViewModel

然后在子 package .viewmodel 中創(chuàng)建 ViewModel 類 UserModel:

package li.fyun.mvvm.viewmodel;

/**
 * Created by fyunli on 15/12/28.
 */
public interface ViewModel {
    void destroy();
}

package li.fyun.mvvm.viewmodel;

import android.databinding.ObservableField;
import android.os.Parcel;
import android.os.Parcelable;

import li.fyun.mvvm.model.User;

/**
 * Created by fyunli on 15/12/28.
 */
public class UserModel implements ViewModel, Parcelable {

    public ObservableField<String> username = new ObservableField<>();
    public ObservableField<String> portrait = new ObservableField<>();

    public UserModel(User user) {
        this.username.set(user.getUsername());
        this.portrait.set(user.getPortrait());
    }

    @Override
    public void destroy() {

    }

    // just a sample logic
    public static UserModel laodUser() {
        User user = User.getUser();
        return new UserModel(user);
    }

    // use android parceable generator plugin to gen the code
    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeSerializable(this.username);
        dest.writeSerializable(this.portrait);
    }

    protected UserModel(Parcel in) {
        this.username = (ObservableField<String>) in.readSerializable();
        this.portrait = (ObservableField<String>) in.readSerializable();
    }

    public static final Creator<UserModel> CREATOR = new Creator<UserModel>() {
        public UserModel createFromParcel(Parcel source) {
            return new UserModel(source);
        }

        public UserModel[] newArray(int size) {
            return new UserModel[size];
        }
    };
}

綁定 View 和 ViewModel

在 activity_main.xml 中聲明 UserModel 變量 user, 并在 TextView 中用 @{user.username} 綁定屬性:

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools">

    <data>

        <variable
            name="user"
            type="li.fyun.mvvm.viewmodel.UserModel"/>
    </data>

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:paddingBottom="@dimen/activity_vertical_margin"
        android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        tools:context=".view.MainActivity">

        <TextView
            android:id="@+id/textview_username_label"
            android:layout_width="80dp"
            android:layout_height="wrap_content"
            android:text="Username:"
            />
        <TextView
            android:id="@+id/textview_username_field"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@{user.username}"
            android:layout_toRightOf="@id/textview_username_label"
            tools:text="Vincent"
            />
    </RelativeLayout>
</layout>

然后在 MainActivity 中綁定 View 和 ViewModel:

package li.fyun.mvvm.view;

import android.databinding.DataBindingUtil;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;

import li.fyun.mvvm.R;
import li.fyun.mvvm.databinding.ActivityMainBinding;
import li.fyun.mvvm.viewmodel.UserModel;

public class MainActivity extends AppCompatActivity {

    ActivityMainBinding binding;
    UserModel userModel;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
        userModel = UserModel.laodUser();
        binding.setUser(userModel);
    }

}

展示一下成果

在模擬器中運(yùn)行蛆封,效果如下:

showtime

增加一點(diǎn)新東西

我們希望在這個(gè)程序中能顯示用戶的頭像耘柱。

首先我們?cè)?layout 中先聲明一個(gè) ImageView 并綁定到 UserModel 的 portrait:

<ImageView
    android:id="@+id/imageview_portraint"
    android:layout_width="64dp"
    android:layout_height="64dp"
    android:layout_marginRight="32dp"
    app:imageUrl="@{user.portrait}"/>

然后梢莽,我們要在 UserModel 中寫一個(gè) BindingAdapter 用于加載圖像:

@BindingAdapter({"bind:imageUrl"})
public static void loadImage(ImageView view, String imageUrl) {
    Glide.with(view.getContext())
            .load(imageUrl)
            .into(view);
}

別忘了在 app 的 build.gralde 中引入依賴:

compile 'com.github.bumptech.glide:glide:3.6.1'

并在 AndroidManifest.xml 中聲明 INTERNET 訪問(wèn)權(quán)限:

<uses-permission android:name="android.permission.INTERNET"/>

運(yùn)行程序评抚,看一下效果:

showtime 2

處理屏幕旋轉(zhuǎn)

這時(shí)候發(fā)現(xiàn)有什么問(wèn)題嗎熟空?當(dāng)你旋轉(zhuǎn)屏幕的時(shí)候,顯示的用戶名會(huì)隨機(jī)變化竟纳,這是應(yīng)為 Activity Instance State 沒(méi)有保存撵溃,在 MVVM 模式下,Instance State 的保存變得簡(jiǎn)單許多:

首先在 onSaveInstanceState 中保存:

@Override
protected void onSaveInstanceState(Bundle outState) {
    super.onSaveInstanceState(outState);
    outState.putParcelable(USER_MODEL, userModel);
}

然后在 onRestoreInstanceState(Bundle savedInstanceState) 或者 onCreate(Bundle savedInstanceState) 方法中取回保存的數(shù)據(jù)即可:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
    if(savedInstanceState == null) {
        userModel = UserModel.laodUser();
    }else{
        userModel = savedInstanceState.getParcelable(USER_MODEL);
    }
    binding.setUser(userModel);
}

打完锥累,收工缘挑!

源代碼在 Github 可找到。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末桶略,一起剝皮案震驚了整個(gè)濱河市卖哎,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌删性,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,695評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件焕窝,死亡現(xiàn)場(chǎng)離奇詭異蹬挺,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)它掂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,569評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門巴帮,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)溯泣,“玉大人,你說(shuō)我怎么就攤上這事榕茧±伲” “怎么了?”我有些...
    開封第一講書人閱讀 168,130評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵用押,是天一觀的道長(zhǎng)肢簿。 經(jīng)常有香客問(wèn)我,道長(zhǎng)蜻拨,這世上最難降的妖魔是什么池充? 我笑而不...
    開封第一講書人閱讀 59,648評(píng)論 1 297
  • 正文 為了忘掉前任,我火速辦了婚禮缎讼,結(jié)果婚禮上收夸,老公的妹妹穿的比我還像新娘。我一直安慰自己血崭,他們只是感情好卧惜,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,655評(píng)論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著夹纫,像睡著了一般咽瓷。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上捷凄,一...
    開封第一講書人閱讀 52,268評(píng)論 1 309
  • 那天忱详,我揣著相機(jī)與錄音,去河邊找鬼跺涤。 笑死匈睁,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的桶错。 我是一名探鬼主播航唆,決...
    沈念sama閱讀 40,835評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼院刁!你這毒婦竟也來(lái)了糯钙?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,740評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤退腥,失蹤者是張志新(化名)和其女友劉穎任岸,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體狡刘,經(jīng)...
    沈念sama閱讀 46,286評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡享潜,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,375評(píng)論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了嗅蔬。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片剑按。...
    茶點(diǎn)故事閱讀 40,505評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡疾就,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出艺蝴,到底是詐尸還是另有隱情猬腰,我是刑警寧澤,帶...
    沈念sama閱讀 36,185評(píng)論 5 350
  • 正文 年R本政府宣布猜敢,位于F島的核電站姑荷,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏锣枝。R本人自食惡果不足惜厢拭,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,873評(píng)論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望撇叁。 院中可真熱鬧供鸠,春花似錦、人聲如沸陨闹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,357評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)趋厉。三九已至寨闹,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間君账,已是汗流浹背繁堡。 一陣腳步聲響...
    開封第一講書人閱讀 33,466評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留乡数,地道東北人椭蹄。 一個(gè)月前我還...
    沈念sama閱讀 48,921評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像净赴,于是被迫代替她去往敵國(guó)和親绳矩。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,515評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容