/* * Copyright 2018 The Android Open Source Project * * 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.android.pump.fragment; import android.content.Context; import android.graphics.Bitmap; import android.net.Uri; import android.os.Bundle; import android.util.SparseIntArray; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.annotation.UiThread; import androidx.fragment.app.Fragment; import androidx.recyclerview.widget.GridLayoutManager; import androidx.recyclerview.widget.GridLayoutManager.SpanSizeLookup; import androidx.recyclerview.widget.RecyclerView; import androidx.recyclerview.widget.RecyclerView.Adapter; import androidx.recyclerview.widget.RecyclerView.ViewHolder; import com.android.pump.R; import com.android.pump.activity.OtherDetailsActivity; import com.android.pump.db.MediaDb; import com.android.pump.db.Other; import com.android.pump.util.Globals; import com.android.pump.util.ImageLoader; import com.android.pump.util.Orientation; import com.android.pump.widget.UriImageView; import java.util.List; @UiThread public class OtherFragment extends Fragment { private static final int SPAN_COUNT = 6; private RecyclerView mRecyclerView; public static @NonNull Fragment newInstance() { return new OtherFragment(); } @Override public @NonNull View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { View view = inflater.inflate(R.layout.fragment_other, container, false); mRecyclerView = view.findViewById(R.id.fragment_other_recycler_view); mRecyclerView.setHasFixedSize(true); OtherAdapter otherAdapter = new OtherAdapter(requireContext()); mRecyclerView.setAdapter(otherAdapter); GridLayoutManager gridLayoutManager = (GridLayoutManager) mRecyclerView.getLayoutManager(); gridLayoutManager.setSpanSizeLookup(otherAdapter.getSpanSizeLookup()); if (gridLayoutManager.getSpanCount() != SPAN_COUNT) { throw new IllegalArgumentException("Expected a span count of " + SPAN_COUNT + ", found a span count of " + gridLayoutManager.getSpanCount() + "."); } mRecyclerView.setItemAnimator(null); // TODO Re-enable add/remove animations // TODO(b/123707260) Enable view caching //mRecyclerView.setItemViewCacheSize(0); //mRecyclerView.setRecycledViewPool(Globals.getRecycledViewPool(requireContext())); return view; } private static class OtherAdapter extends Adapter implements MediaDb.UpdateCallback, ImageLoader.Callback { private final ImageLoader mImageLoader; private final MediaDb mMediaDb; private final List mOthers; // TODO(b/123710968) Use android.support.v7.util.SortedList/android.support.v7.widget.util.SortedListAdapterCallback instead private final SparseIntArray mSpanSize = new SparseIntArray(); private OtherAdapter(@NonNull Context context) { setHasStableIds(true); mImageLoader = Globals.getImageLoader(context); mMediaDb = Globals.getMediaDb(context); mOthers = mMediaDb.getOthers(); recalculateSpans(); } public void onAttachedToRecyclerView(@NonNull RecyclerView recyclerView) { mMediaDb.addOtherUpdateCallback(this); mImageLoader.addCallback(this); } public void onDetachedFromRecyclerView(@NonNull RecyclerView recyclerView) { mMediaDb.removeOtherUpdateCallback(this); mImageLoader.removeCallback(this); } @Override public @NonNull ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { if (viewType == R.layout.header) { return new ViewHolder(LayoutInflater.from(parent.getContext()) .inflate(viewType, parent, false)) { }; } else { return new OtherViewHolder(LayoutInflater.from(parent.getContext()) .inflate(viewType, parent, false)); } } @Override public void onBindViewHolder(@NonNull ViewHolder holder, int position) { if (position == 0) { // TODO Handle header view } else { Other other = mOthers.get(position - 1); mMediaDb.loadData(other); // TODO Where should we call this? In bind()? ((OtherViewHolder) holder).bind(other); } } @Override public int getItemCount() { return mOthers.size() + 1; } @Override public long getItemId(int position) { return position == 0 ? -1 : mOthers.get(position - 1).getId(); } @Override public int getItemViewType(int position) { return position == 0 ? R.layout.header : R.layout.other; } @Override public void onImageLoaded(@NonNull Uri uri, @Nullable Bitmap bitmap) { // TODO Optimize this (only update necessary parts -- not the whole world) recalculateSpans(); notifyItemRangeChanged(1, mOthers.size()); } @Override public void onItemsInserted(int index, int count) { notifyItemRangeInserted(index + 1, count); } @Override public void onItemsUpdated(int index, int count) { notifyItemRangeChanged(index + 1, count); } @Override public void onItemsRemoved(int index, int count) { notifyItemRangeRemoved(index + 1, count); } private void recalculateSpans() { // TODO Recalculate when an image is loaded // TODO Recalculate when notifyXxx is called // TODO Optimize mSpanSize.clear(); int current = 0; while (current < mOthers.size()) { int orientation = getOrientation(current); if (orientation == Orientation.LANDSCAPE) { orientation = getOrientation(current + 1); if (orientation == Orientation.LANDSCAPE) { // L L mSpanSize.append(current++, SPAN_COUNT / 2); mSpanSize.append(current++, SPAN_COUNT / 2); } else if (orientation == Orientation.PORTRAIT) { // L P mSpanSize.append(current++, SPAN_COUNT * 2 / 3); mSpanSize.append(current++, SPAN_COUNT * 1 / 3); } else { // L mSpanSize.append(current++, SPAN_COUNT); } } else if (orientation == Orientation.PORTRAIT) { orientation = getOrientation(current + 1); if (orientation == Orientation.LANDSCAPE) { // P L mSpanSize.append(current++, SPAN_COUNT * 1 / 3); mSpanSize.append(current++, SPAN_COUNT * 2 / 3); } else if (orientation == Orientation.PORTRAIT && getOrientation(current + 2) == Orientation.PORTRAIT) { // P P P mSpanSize.append(current++, SPAN_COUNT / 3); mSpanSize.append(current++, SPAN_COUNT / 3); mSpanSize.append(current++, SPAN_COUNT / 3); } else { // P mSpanSize.append(current++, SPAN_COUNT); } } else { // unknown mSpanSize.append(current++, SPAN_COUNT); } } } private @Orientation int getOrientation(int index) { Uri thumbUri = index >= mOthers.size() ? null : mOthers.get(index).getThumbnailUri(); if (thumbUri == null) { return Orientation.UNKNOWN; } return mImageLoader.getOrientation(thumbUri); } private @NonNull SpanSizeLookup getSpanSizeLookup() { return new SpanSizeLookup() { @Override public int getSpanSize(int position) { return position == 0 ? SPAN_COUNT : mSpanSize.get(position - 1); } @Override public int getSpanIndex(int position, int spanCount) { // TODO Optimize return super.getSpanIndex(position, spanCount); } @Override public int getSpanGroupIndex(int adapterPosition, int spanCount) { // TODO Optimize return super.getSpanGroupIndex(adapterPosition, spanCount); } }; } } private static class OtherViewHolder extends ViewHolder { private OtherViewHolder(@NonNull View itemView) { super(itemView); } private void bind(@NonNull Other other) { UriImageView imageView = itemView.findViewById(R.id.other_image); imageView.setImageURI(other.getThumbnailUri()); itemView.setOnClickListener((view) -> OtherDetailsActivity.start(view.getContext(), other)); } } }