這個 sample 的簡單需求:
獲取數據庫中的數據, 并展示為 list, 點擊后進入詳情頁
數據庫設計
字段名 | 說明 |
---|---|
id | primary key |
name | 名稱 |
description | 描述 |
price | 價格 |
1.entity table表的實體類
使用 arch.room 類進行綁定
@Entity(tableName = "products")
public class ProductEntity implements Product {
@PrimaryKey
private int id;
private String name;
private String description;
private int price;
}
2.DAO android 調用的接口
使用 arch.room 進行數據庫的使用, 類似retrofit
CRUD
功能1: 獲取全部products
功能2: 插入所有
功能3: 根據 id 取1個 product
@Dao
public interface ProductDao {
@Query("SELECT * FROM products")
LiveData<List<ProductEntity>> loadAllProducts();
@Insert(onConflict = OnConflictStrategy.REPLACE)
void insertAll(List<ProductEntity> products);
@Query("select * from products where id = :productId")
LiveData<ProductEntity> loadProduct(int productId);
}
3.viewModel
arch.lifecycle 的 viewModel
public class ProductViewModel extends AndroidViewModel {
public ProductViewModel(@NonNull Application application,
final int productId) {
super(application);
mProductId = productId;
final DatabaseCreator databaseCreator = DatabaseCreator.getInstance(this.getApplication());
mObservableComments = Transformations.switchMap(databaseCreator.isDatabaseCreated(), new Function<Boolean, LiveData<List<CommentEntity>>>() {
@Override
public LiveData<List<CommentEntity>> apply(Boolean isDbCreated) {
if (!isDbCreated) {
//noinspection unchecked
return ABSENT;
} else {
//noinspection ConstantConditions
return databaseCreator.getDatabase().commentDao().loadComments(mProductId);
}
}
});
databaseCreator.createDb(this.getApplication());
}
}
4.dataBinding
在 gradle 中使用 dataBingding 進行 xml 和 java 的綁定
dataBinding {
enabled = true
}
5.list_fragment.xml 和 ProductListFragment.xml 之間的關系
在 listfragment 這頁面, 根據數據是否獲取成功, 展示 loading 頁面和 list 頁面
使用@ BindingAdapter("visibleOrGone")
用 java代碼實現app:visibleOrGone="@{isLoading}" 邏輯
public class BindingAdapters {
@BindingAdapter("visibleOrGone")
public static void showHide(View view, boolean show) {
view.setVisibility(show ? View.VISIBLE : View.GONE);
}
}
在.xml 中使用app:visibleOrGone="@{isLoading}"動態(tài)判斷
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="isLoading"
type="boolean" />
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/cardview_light_background"
android:orientation="vertical">
<TextView
android:id="@+id/loading_tv"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center_vertical|center_horizontal"
android:text="@string/loading_products"
android:textAlignment="center"
app:visibleOrGone="@{isLoading}"
/>
<android.support.v7.widget.RecyclerView
android:id="@+id/products_list"
android:contentDescription="@string/cd_products_list"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layoutManager="LinearLayoutManager"
app:visibleOrGone="@{!isLoading}"/>
</LinearLayout>
</layout>
6.如何把 entity 和頁面對應:
使用
<variable name="product"
type="com.example.android.persistence.model.Product"/>
android:text="@{product.name}"
和 textview 進行一一對應
然后使用
<variable name="callback" type="com.example.android.persistence.ui.ProductClickCallback"/>
android:onClick="@{() -> callback.onClick(product)}"
和點擊事件綁定
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable name="product"
type="com.example.android.persistence.model.Product"/>
<variable name="callback"
type="com.example.android.persistence.ui.ProductClickCallback"/>
</data>
<android.support.v7.widget.CardView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:minHeight="@dimen/product_item_min_height"
android:onClick="@{() -> callback.onClick(product)}"
android:orientation="horizontal"
android:layout_marginStart="@dimen/item_horizontal_margin"
android:layout_marginEnd="@dimen/item_horizontal_margin"
app:cardUseCompatPadding="true">
<RelativeLayout
android:layout_marginStart="@dimen/item_horizontal_margin"
android:layout_marginEnd="@dimen/item_horizontal_margin"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:id="@+id/name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:contentDescription="@string/cd_product_name"
android:text="@{product.name}"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentEnd="true"
android:layout_marginEnd="5dp"
android:text="@{@string/product_price(product.price)}"/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_below="@id/name"
android:text="@{product.description}"/>
</RelativeLayout>
</android.support.v7.widget.CardView>
</layout>