I am trying to develop e-commerce type of app in which user can add item to cart. But I am unable to update the cart badge counter from adapter. Here is screenshot of my app.
I am using Recycler view inside fragment. And I am referring this below link for menu item count.
https://mobikul.com/adding-badge-count-on-menu-items-like-cart-notification-etc/
My problem is I cannot access menu item from adapter.
Here is my fragment code.
PriceListFragment.java
public class PriceListFragment extends Fragment {
public ArrayList<Design> designList;
private Single_DesignAdapter adapter;
private RecyclerView recyclerView;
public PriceListFragment() {
// Required empty public constructor
}
@Override
public View onCreateView(LayoutInflater inflater, final ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
setHasOptionsMenu(true);
View v = inflater.inflate(R.layout.fragment_price_list, container, false);
recyclerView = (RecyclerView)v.findViewById(R.id.pl_recycler_view);
designList = new ArrayList<Design>();
designList = list();
recyclerView.setLayoutManager(new LinearLayoutManager(getContext()));
recyclerView.setHasFixedSize(true);
adapter = new Single_DesignAdapter(getContext(), designList);
recyclerView.setAdapter(adapter);
return v;
}
@Override
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
inflater.inflate(R.menu.cart_menu, menu);
super.onCreateOptionsMenu(menu, inflater);
MenuItem itemCart = menu.findItem(R.id.menu_cart);
LayerDrawable icon = (LayerDrawable) itemCart.getIcon();
setBadgeCount(getContext(), icon, 0);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case R.id.menu_cart:
Toast.makeText(getActivity(), "cart", Toast.LENGTH_SHORT). show();
return false;
default:
break;
}
return false;
}
public ArrayList<Design> list() {
ArrayList<Design> arrayList = new ArrayList<Design>();
Design design = new Design();
design.name = "Black";
design.image = "http://website/Demo/images/1.jpg";
design.designName = "11001";
design.qualityName = "Cotton";
design.amount = "1000";
design.discPercent = "5";
Design design2 = new Design();
design2.name = "Green";
design2.image = "http://website/Demo/images/2.jpg";
design2.designName = "11001";
design2.qualityName = "Cotton";
design2.amount = "900";
design2.discPercent = "9";
arrayList.add(design);
arrayList.add(design2);
return arrayList;
}
public static void setBadgeCount(Context context, LayerDrawable icon, int count) {
BadgeDrawable badge;
// Reuse drawable if possible
Drawable reuse = icon.findDrawableByLayerId(R.id.ic_badge);
if (reuse != null && reuse instanceof BadgeDrawable) {
badge = (BadgeDrawable) reuse;
} else {
badge = new BadgeDrawable(context);
}
badge.setCount(count);
icon.mutate();
icon.setDrawableByLayerId(R.id.ic_badge, badge);
}
}
Design is my model class
Design.java
public class Design implements Serializable{
public String id;
public String qualityId;
public String qualityName;
public String designId;
public String designName;
public String image;
public String name; //use as shade name
public String discPercent;
public String amount;
public String pcs;
public String qty;
}
Single_DesignAdapter.java
public class Single_DesignAdapter extends RecyclerView.Adapter<Single_DesignAdapter.ViewHolder> {
public ArrayList<Design> designList;
private Context context;
private LayoutInflater layoutInflater;
public Single_DesignAdapter(Context context, ArrayList<Design> designList) {
this.designList = designList;
this.context = context;
this.layoutInflater = LayoutInflater.from(context);
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View view = layoutInflater.inflate(R.layout.single_design, parent, false);
return new ViewHolder(view);
}
@Override
public void onBindViewHolder(final ViewHolder holder, int position) {
Design design = designList.get(position);
holder.tvQuality.setText(design.qualityName);
holder.tvDesign.setText(design.designName);
holder.tvShade.setText(design.name);
if (!design.discPercent.equals("0")) {
holder.tvDisPer.setText("-"+design.discPercent+"%");
holder.tvAmount.setPaintFlags(holder.tvAmount.getPaintFlags() | Paint.STRIKE_THRU_TEXT_FLAG);
int a = Integer.parseInt(design.amount);
float b = Float.parseFloat(design.discPercent) / 100;
holder.tvAmount.setText("₹ " + design.amount);
int c = (int) (a * b);
int d = a-c;
holder.tvDiscAmt.setText("₹ " +d);
Picasso.with(context).load(design.image).into(holder.ivDesign);
} else {
holder.tvAmount.setText("₹ " + design.amount);
Picasso.with(context).load(design.image).into(holder.ivDesign);
holder.tvDisPer.setVisibility(View.INVISIBLE);
holder.tvDiscAmt.setVisibility(View.INVISIBLE);
}
holder.btAdd.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
holder.llAdd.setVisibility(View.GONE);
holder.llPlusMinus.setVisibility(View.VISIBLE);
}
});
holder.btPlus.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (Integer.parseInt(holder.btPcsCount.getText().toString()) >= 1) {
int a = Integer.parseInt(holder.btPcsCount.getText().toString());
int x = a++;
holder.btPcsCount.setText(Integer.toString(a));
}
}
});
holder.btMinus.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (Integer.parseInt(holder.btPcsCount.getText().toString()) <= 1) {
holder.llAdd.setVisibility(View.VISIBLE);
holder.llPlusMinus.setVisibility(View.GONE);
} else {
int a = Integer.parseInt(holder.btPcsCount.getText().toString());
a--;
holder.btPcsCount.setText(Integer.toString(a));
}
}
});
}
@Override
public int getItemCount() {
return designList.size();
}
public class ViewHolder extends RecyclerView.ViewHolder {
public LinearLayout llPer, llAdd, llPlusMinus;
public Button btPlus, btMinus, btPcsCount, btAdd;
public ImageView ivDesign;
public TextView tvDesign, tvQuality, tvShade, tvAmount, tvDiscAmt, tvDisPer;
public ViewHolder(View itemView) {
super(itemView);
btPlus = (Button) itemView.findViewById(R.id.btPlus);
//other find view by ids
}
}
}
Now when user add item to cart it should increment counter. I will add every cart items in ArrayList<Design>
and then I will save this list into shared preference. I will manage everything, but I am unable understand that How can I update counter of cart from adapter. I cannot access menu in adapter. If there is other way to do this, Please help.
Since you already have a Context
in your Adapter
. You can cast that to Activity
. This way you have access to the menu.
//when item is added or removed
((Activity)this.context).invalidateOptionsMenu()
In your Fragment.onCreateOptionsMenu
you can go ahead and set the item count
In your fragment click event:
((MainActivity) getActivity()).createCartBadge("value");
Your activity contains this method:
public void createCartBadge(int i) {
MenuItem cartItem = mToolbarMenu.findItem(R.id.cart);
LayerDrawable localLayerDrawable = (LayerDrawable) cartItem.getIcon();
Drawable cartBadgeDrawable = localLayerDrawable
.findDrawableByLayerId(R.id.ic_badge);
BadgeDrawable badgeDrawable;
if ((cartBadgeDrawable != null)
&& ((cartBadgeDrawable instanceof BadgeDrawable))
&& (i < 10)) {
badgeDrawable = (BadgeDrawable) cartBadgeDrawable;
} else {
badgeDrawable = new BadgeDrawable(MainActivity.this);
}
badgeDrawable.setCount(i);
localLayerDrawable.mutate();
localLayerDrawable.setDrawableByLayerId(R.id.ic_badge, badgeDrawable);
cartItem.setIcon(localLayerDrawable);
}
You can use Interface
to communicate between the Fragment
and the Adapter
.
for example:
public interface CallbackInterface {
void onCall();
}
and in your PriceListFragment
in onCreateView
do that:
CallbackInterface callback = new CallbackInterface{
@Override
public void onCall() {
if(itemCart!=null){
//code to increment badge
}
}
};
and pass callback
to the adapter's constructor and then use it there.
You can use badge style in TextView
now, All you need to do is adjust your textview position to make it look like badge.
<TextView
android:id="@+id/fabCounter"
style="@style/Widget.Design.FloatingActionButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentEnd="true"
android:layout_centerVertical="true"
android:layout_marginEnd="10dp"
android:padding="5dp"
android:text="-9"
android:textColor="@android:color/black"
android:textSize="14sp" />
来源:https://stackoverflow.com/questions/43015908/android-change-cart-menu-badge-counter-from-recycler-adapter