1 /*
2  * Copyright (C) 2022 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.systemui.volume;
18 
19 import static android.app.slice.Slice.HINT_ERROR;
20 import static android.app.slice.SliceItem.FORMAT_SLICE;
21 
22 import android.content.Context;
23 import android.net.Uri;
24 import android.view.LayoutInflater;
25 import android.view.View;
26 import android.view.ViewGroup;
27 
28 import androidx.annotation.NonNull;
29 import androidx.lifecycle.LifecycleOwner;
30 import androidx.lifecycle.LiveData;
31 import androidx.recyclerview.widget.RecyclerView;
32 import androidx.slice.Slice;
33 import androidx.slice.SliceItem;
34 import androidx.slice.widget.SliceView;
35 
36 import com.android.systemui.res.R;
37 
38 import java.util.ArrayList;
39 import java.util.List;
40 import java.util.Map;
41 
42 /**
43  * RecyclerView adapter for Slices in Settings Panels.
44  */
45 public class VolumePanelSlicesAdapter extends
46         RecyclerView.Adapter<VolumePanelSlicesAdapter.SliceRowViewHolder> {
47 
48     private final List<LiveData<Slice>> mSliceLiveData;
49     private final LifecycleOwner mLifecycleOwner;
50     private SliceView.OnSliceActionListener mOnSliceActionListener;
51 
VolumePanelSlicesAdapter(LifecycleOwner lifecycleOwner, Map<Uri, LiveData<Slice>> sliceLiveData)52     public VolumePanelSlicesAdapter(LifecycleOwner lifecycleOwner,
53             Map<Uri, LiveData<Slice>> sliceLiveData) {
54         mLifecycleOwner = lifecycleOwner;
55         mSliceLiveData = new ArrayList<>(sliceLiveData.values());
56     }
57 
58     @NonNull
59     @Override
onCreateViewHolder(@onNull ViewGroup viewGroup, int viewType)60     public SliceRowViewHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int viewType) {
61         final Context context = viewGroup.getContext();
62         final LayoutInflater inflater = LayoutInflater.from(context);
63         View view = inflater.inflate(R.layout.volume_panel_slice_slider_row, viewGroup, false);
64         return new SliceRowViewHolder(view);
65     }
66 
67     @Override
onBindViewHolder(@onNull SliceRowViewHolder sliceRowViewHolder, int position)68     public void onBindViewHolder(@NonNull SliceRowViewHolder sliceRowViewHolder, int position) {
69         sliceRowViewHolder.onBind(mSliceLiveData.get(position), position);
70     }
71 
72     @Override
getItemCount()73     public int getItemCount() {
74         return mSliceLiveData.size();
75     }
76 
77     @Override
getItemViewType(int position)78     public int getItemViewType(int position) {
79         return position;
80     }
81 
setOnSliceActionListener(SliceView.OnSliceActionListener listener)82     void setOnSliceActionListener(SliceView.OnSliceActionListener listener) {
83         mOnSliceActionListener = listener;
84     }
85 
updateDataSet(ArrayList<LiveData<Slice>> list)86     void updateDataSet(ArrayList<LiveData<Slice>> list) {
87         mSliceLiveData.clear();
88         mSliceLiveData.addAll(list);
89         notifyDataSetChanged();
90     }
91 
92     /**
93      * ViewHolder for binding Slices to SliceViews.
94      */
95     public class SliceRowViewHolder extends RecyclerView.ViewHolder {
96 
97         private final SliceView mSliceView;
98 
SliceRowViewHolder(View view)99         public SliceRowViewHolder(View view) {
100             super(view);
101             mSliceView = view.findViewById(R.id.slice_view);
102             mSliceView.setMode(SliceView.MODE_LARGE);
103             mSliceView.setShowTitleItems(true);
104             mSliceView.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO);
105             mSliceView.setOnSliceActionListener(mOnSliceActionListener);
106         }
107 
108         /**
109          * Called when the view is displayed.
110          */
onBind(LiveData<Slice> sliceLiveData, int position)111         public void onBind(LiveData<Slice> sliceLiveData, int position) {
112             sliceLiveData.observe(mLifecycleOwner, mSliceView);
113 
114             // Do not show the divider above media devices switcher slice per request
115             final Slice slice = sliceLiveData.getValue();
116 
117             // Hides slice which reports with error hint or not contain any slice sub-item.
118             if (slice == null || !isValidSlice(slice)) {
119                 mSliceView.setVisibility(View.GONE);
120             } else {
121                 mSliceView.setVisibility(View.VISIBLE);
122             }
123         }
124 
isValidSlice(Slice slice)125         private boolean isValidSlice(Slice slice) {
126             if (slice.getHints().contains(HINT_ERROR)) {
127                 return false;
128             }
129             for (SliceItem item : slice.getItems()) {
130                 if (item.getFormat().equals(FORMAT_SLICE)) {
131                     return true;
132                 }
133             }
134             return false;
135         }
136     }
137 }
138