/*
* Copyright 2017 Hannes Dorfmann.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*/
package com.hannesdorfmann.mosby3.sample.mvi.view.shoppingcartoverview;
import android.graphics.Canvas;
import android.graphics.drawable.ColorDrawable;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.v4.content.ContextCompat;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.support.v7.widget.helper.ItemTouchHelper;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.Unbinder;
import com.hannesdorfmann.mosby3.mvi.MviFragment;
import com.hannesdorfmann.mosby3.sample.mvi.R;
import com.hannesdorfmann.mosby3.sample.mvi.SampleApplication;
import com.hannesdorfmann.mosby3.sample.mvi.businesslogic.model.Product;
import io.reactivex.Observable;
import io.reactivex.subjects.PublishSubject;
import java.util.List;
import java.util.concurrent.TimeUnit;
import timber.log.Timber;
/**
* This class doesn't neccessarily has to be a fragment. It's just a fragment because I want to
* demonstrate that mosby works with fragments in xml layouts too.
*
* @author Hannes Dorfmann
*/
public class ShoppingCartOverviewFragment
extends MviFragment<ShoppingCartOverviewView, ShoppingCartOverviewPresenter>
implements ShoppingCartOverviewView {
private ShoppingCartOverviewAdapter adapter;
private PublishSubject<Product> removeRelay = PublishSubject.create();
private Unbinder unbinder;
@BindView(R.id.shoppingCartRecyclerView) RecyclerView recyclerView;
@Nullable @Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
@Nullable Bundle savedInstanceState) {
View v = inflater.inflate(R.layout.fragment_shopping_cart, container, false);
unbinder = ButterKnife.bind(this, v);
adapter = new ShoppingCartOverviewAdapter(getActivity());
recyclerView.setAdapter(adapter);
recyclerView.setLayoutManager(new LinearLayoutManager(getActivity()));
setUpItemTouchHelper();
return v;
}
@Override public void onDestroyView() {
super.onDestroyView();
unbinder.unbind();
}
@NonNull @Override public ShoppingCartOverviewPresenter createPresenter() {
Timber.d("Create Presenter");
return SampleApplication.getDependencyInjection(getActivity()).getShoppingCartPresenter();
}
@Override public Observable<Boolean> loadItemsIntent() {
return Observable.just(true);
}
@Override public Observable<List<Product>> selectItemsIntent() {
return adapter.selectedItemsObservable();
}
private void setUpItemTouchHelper() {
//
// Borrowed from https://github.com/nemanja-kovacevic/recycler-view-swipe-to-delete/blob/master/app/src/main/java/net/nemanjakovacevic/recyclerviewswipetodelete/MainActivity.java
//
ItemTouchHelper.SimpleCallback simpleItemTouchCallback =
new ItemTouchHelper.SimpleCallback(0, ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT) {
// we want to cache these and not allocate anything repeatedly in the onChildDraw method
Drawable background;
Drawable xMark;
int xMarkMargin;
boolean initiated;
private void init() {
background =
new ColorDrawable(ContextCompat.getColor(getActivity(), R.color.delete_background));
xMark = ContextCompat.getDrawable(getActivity(), R.drawable.ic_remove);
xMarkMargin = (int) getActivity().getResources().getDimension(R.dimen.ic_clear_margin);
initiated = true;
}
// not important, we don't want drag & drop
@Override public boolean onMove(RecyclerView recyclerView,
RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) {
return false;
}
@Override public void onSwiped(RecyclerView.ViewHolder viewHolder, int swipeDir) {
int swipedPosition = viewHolder.getAdapterPosition();
Product productAt = adapter.getProductAt(swipedPosition);
removeRelay.onNext(productAt);
}
@Override public void onChildDraw(Canvas c, RecyclerView recyclerView,
RecyclerView.ViewHolder viewHolder, float dX, float dY, int actionState,
boolean isCurrentlyActive) {
View itemView = viewHolder.itemView;
// not sure why, but this method get's called for viewholder that are already swiped away
if (viewHolder.getAdapterPosition() == -1) {
// not interested in those
return;
}
if (!initiated) {
init();
}
// draw red background
background.setBounds(itemView.getRight() + (int) dX, itemView.getTop(),
itemView.getRight(), itemView.getBottom());
background.draw(c);
// draw x mark
int itemHeight = itemView.getBottom() - itemView.getTop();
int intrinsicWidth = xMark.getIntrinsicWidth();
int intrinsicHeight = xMark.getIntrinsicWidth();
int xMarkLeft = itemView.getRight() - xMarkMargin - intrinsicWidth;
int xMarkRight = itemView.getRight() - xMarkMargin;
int xMarkTop = itemView.getTop() + (itemHeight - intrinsicHeight) / 2;
int xMarkBottom = xMarkTop + intrinsicHeight;
xMark.setBounds(xMarkLeft, xMarkTop, xMarkRight, xMarkBottom);
// xMark.draw(c);
super.onChildDraw(c, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive);
}
};
ItemTouchHelper mItemTouchHelper = new ItemTouchHelper(simpleItemTouchCallback);
mItemTouchHelper.attachToRecyclerView(recyclerView);
}
@Override public Observable<Product> removeItemIntent() {
// DELEAY it for 500 miliseconds so that the user sees the swipe to delete animation
return removeRelay.delay(500, TimeUnit.MILLISECONDS);
}
@Override public void render(List<ShoppingCartOverviewItem> itemsInShoppingCart) {
Timber.d("Render %s ", itemsInShoppingCart);
adapter.setItems(itemsInShoppingCart);
}
}