SyncedFolderAdapter.java 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  1. /*
  2. * Nextcloud Android client application
  3. *
  4. * @author Andy Scherzinger
  5. * Copyright (C) 2016 Andy Scherzinger
  6. * Copyright (C) 2016 Nextcloud
  7. *
  8. * This program is free software; you can redistribute it and/or
  9. * modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
  10. * License as published by the Free Software Foundation; either
  11. * version 3 of the License, or any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
  17. *
  18. * You should have received a copy of the GNU Affero General Public
  19. * License along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. */
  21. package com.owncloud.android.ui.adapter;
  22. import android.content.Context;
  23. import android.view.LayoutInflater;
  24. import android.view.MenuItem;
  25. import android.view.View;
  26. import android.view.ViewGroup;
  27. import android.widget.FrameLayout;
  28. import android.widget.ImageButton;
  29. import android.widget.ImageView;
  30. import android.widget.LinearLayout;
  31. import android.widget.PopupMenu;
  32. import android.widget.RelativeLayout;
  33. import android.widget.TextView;
  34. import com.afollestad.sectionedrecyclerview.SectionedRecyclerViewAdapter;
  35. import com.afollestad.sectionedrecyclerview.SectionedViewHolder;
  36. import com.nextcloud.client.core.Clock;
  37. import com.owncloud.android.R;
  38. import com.owncloud.android.datamodel.MediaFolderType;
  39. import com.owncloud.android.datamodel.SyncedFolderDisplayItem;
  40. import com.owncloud.android.datamodel.ThumbnailsCacheManager;
  41. import com.owncloud.android.utils.ThemeUtils;
  42. import java.io.File;
  43. import java.util.ArrayList;
  44. import java.util.List;
  45. import java.util.Locale;
  46. import androidx.annotation.NonNull;
  47. import butterknife.BindView;
  48. import butterknife.ButterKnife;
  49. /**
  50. * Adapter to display all auto-synced folders and/or instant upload media folders.
  51. */
  52. public class SyncedFolderAdapter extends SectionedRecyclerViewAdapter<SectionedViewHolder> {
  53. private final Context context;
  54. private final Clock clock;
  55. private final int gridWidth;
  56. private final int gridTotal;
  57. private final ClickListener clickListener;
  58. private final List<SyncedFolderDisplayItem> syncFolderItems;
  59. private final List<SyncedFolderDisplayItem> filteredSyncFolderItems;
  60. private final boolean light;
  61. private final int VIEW_TYPE_EMPTY = Integer.MAX_VALUE;
  62. private boolean hideItems;
  63. public SyncedFolderAdapter(Context context, Clock clock, int gridWidth, ClickListener listener, boolean light) {
  64. this.context = context;
  65. this.clock = clock;
  66. this.gridWidth = gridWidth;
  67. gridTotal = gridWidth * 2;
  68. clickListener = listener;
  69. syncFolderItems = new ArrayList<>();
  70. filteredSyncFolderItems = new ArrayList<>();
  71. this.light = light;
  72. this.hideItems = true;
  73. shouldShowHeadersForEmptySections(true);
  74. shouldShowFooters(true);
  75. }
  76. public void toggleHiddenItemsVisibility() {
  77. hideItems = !hideItems;
  78. filteredSyncFolderItems.clear();
  79. filteredSyncFolderItems.addAll(filterHiddenItems(syncFolderItems, hideItems));
  80. notifyDataSetChanged();
  81. }
  82. public void setSyncFolderItems(List<SyncedFolderDisplayItem> syncFolderItems) {
  83. this.syncFolderItems.clear();
  84. this.syncFolderItems.addAll(syncFolderItems);
  85. this.filteredSyncFolderItems.clear();
  86. this.filteredSyncFolderItems.addAll(filterHiddenItems(this.syncFolderItems, hideItems));
  87. }
  88. public void setSyncFolderItem(int location, SyncedFolderDisplayItem syncFolderItem) {
  89. if (hideItems && syncFolderItem.isHidden() && filteredSyncFolderItems.contains(syncFolderItem)) {
  90. filteredSyncFolderItems.remove(location);
  91. } else {
  92. if (filteredSyncFolderItems.contains(syncFolderItem)) {
  93. filteredSyncFolderItems.set(filteredSyncFolderItems.indexOf(syncFolderItem), syncFolderItem);
  94. } else {
  95. filteredSyncFolderItems.add(syncFolderItem);
  96. }
  97. }
  98. if (syncFolderItems.contains(syncFolderItem)) {
  99. syncFolderItems.set(syncFolderItems.indexOf(syncFolderItem), syncFolderItem);
  100. } else {
  101. syncFolderItems.add(syncFolderItem);
  102. }
  103. notifyDataSetChanged();
  104. }
  105. public void addSyncFolderItem(SyncedFolderDisplayItem syncFolderItem) {
  106. syncFolderItems.add(syncFolderItem);
  107. // add item for display when either all items should be shown (!hideItems)
  108. // or if item should be shown (!.isHidden())
  109. if (!hideItems || !syncFolderItem.isHidden()) {
  110. filteredSyncFolderItems.add(syncFolderItem);
  111. notifyDataSetChanged();
  112. }
  113. }
  114. public void removeItem(int section) {
  115. if (filteredSyncFolderItems.contains(syncFolderItems.get(section))) {
  116. filteredSyncFolderItems.remove(syncFolderItems.get(section));
  117. notifyDataSetChanged();
  118. }
  119. syncFolderItems.remove(section);
  120. }
  121. /**
  122. * Filter for hidden items
  123. *
  124. * @param items Collection of items to filter
  125. * @return Non-hidden items
  126. */
  127. private List<SyncedFolderDisplayItem> filterHiddenItems(List<SyncedFolderDisplayItem> items, boolean hide) {
  128. if (!hide) {
  129. return items;
  130. } else {
  131. List<SyncedFolderDisplayItem> result = new ArrayList<>();
  132. for (SyncedFolderDisplayItem item : items) {
  133. if (!item.isHidden() && !result.contains(item)) {
  134. result.add(item);
  135. }
  136. }
  137. return result;
  138. }
  139. }
  140. @Override
  141. public int getSectionCount() {
  142. if (filteredSyncFolderItems.size() > 0) {
  143. return filteredSyncFolderItems.size() + 1;
  144. } else {
  145. return 0;
  146. }
  147. }
  148. public int getUnfilteredSectionCount() {
  149. if (syncFolderItems.size() > 0) {
  150. return syncFolderItems.size() + 1;
  151. } else {
  152. return 0;
  153. }
  154. }
  155. @Override
  156. public int getItemCount(int section) {
  157. if (section < filteredSyncFolderItems.size()) {
  158. List<String> filePaths = filteredSyncFolderItems.get(section).getFilePaths();
  159. if (filePaths != null) {
  160. return filteredSyncFolderItems.get(section).getFilePaths().size();
  161. } else {
  162. return 1;
  163. }
  164. } else {
  165. return 1;
  166. }
  167. }
  168. public SyncedFolderDisplayItem get(int section) {
  169. return filteredSyncFolderItems.get(section);
  170. }
  171. @Override
  172. public int getItemViewType(int section, int relativePosition, int absolutePosition) {
  173. if (isLastSection(section)) {
  174. return VIEW_TYPE_EMPTY;
  175. } else {
  176. return VIEW_TYPE_ITEM;
  177. }
  178. }
  179. @Override
  180. public int getHeaderViewType(int section) {
  181. if (isLastSection(section)) {
  182. return VIEW_TYPE_EMPTY;
  183. } else {
  184. return VIEW_TYPE_HEADER;
  185. }
  186. }
  187. @Override
  188. public int getFooterViewType(int section) {
  189. if (isLastSection(section) && showFooter()) {
  190. return VIEW_TYPE_FOOTER;
  191. } else {
  192. // only show footer after last item and only if folders have been hidden
  193. return VIEW_TYPE_EMPTY;
  194. }
  195. }
  196. private boolean showFooter() {
  197. return syncFolderItems.size() > filteredSyncFolderItems.size();
  198. }
  199. /**
  200. * returns the section of a synced folder for the given local path and type.
  201. *
  202. * @param localPath the local path of the synced folder
  203. * @param type the of the synced folder
  204. * @return the section index of the looked up synced folder, <code>-1</code> if not present
  205. */
  206. public int getSectionByLocalPathAndType(String localPath, int type) {
  207. for (int i = 0; i < filteredSyncFolderItems.size(); i++) {
  208. if (filteredSyncFolderItems.get(i).getLocalPath().equalsIgnoreCase(localPath) &&
  209. filteredSyncFolderItems.get(i).getType().getId().equals(type)) {
  210. return i;
  211. }
  212. }
  213. return -1;
  214. }
  215. @Override
  216. public void onBindHeaderViewHolder(SectionedViewHolder commonHolder, final int section, boolean expanded) {
  217. if (section < filteredSyncFolderItems.size()) {
  218. HeaderViewHolder holder = (HeaderViewHolder) commonHolder;
  219. holder.mainHeaderContainer.setVisibility(View.VISIBLE);
  220. holder.title.setText(filteredSyncFolderItems.get(section).getFolderName());
  221. if (MediaFolderType.VIDEO == filteredSyncFolderItems.get(section).getType()) {
  222. holder.type.setImageResource(R.drawable.video_32dp);
  223. } else if (MediaFolderType.IMAGE == filteredSyncFolderItems.get(section).getType()) {
  224. holder.type.setImageResource(R.drawable.image_32dp);
  225. } else {
  226. holder.type.setImageResource(R.drawable.folder_star_32dp);
  227. }
  228. holder.syncStatusButton.setVisibility(View.VISIBLE);
  229. holder.syncStatusButton.setTag(section);
  230. holder.syncStatusButton.setOnClickListener(v -> {
  231. filteredSyncFolderItems.get(section).setEnabled(
  232. !filteredSyncFolderItems.get(section).isEnabled(),
  233. clock.getCurrentTime()
  234. );
  235. setSyncButtonActiveIcon(holder.syncStatusButton, filteredSyncFolderItems.get(section).isEnabled());
  236. clickListener.onSyncStatusToggleClick(section, filteredSyncFolderItems.get(section));
  237. });
  238. setSyncButtonActiveIcon(holder.syncStatusButton, filteredSyncFolderItems.get(section).isEnabled());
  239. if (light) {
  240. holder.menuButton.setVisibility(View.GONE);
  241. } else {
  242. holder.menuButton.setVisibility(View.VISIBLE);
  243. holder.menuButton.setTag(section);
  244. holder.menuButton.setOnClickListener(v -> onOverflowIconClicked(section,
  245. filteredSyncFolderItems.get(section),
  246. v));
  247. }
  248. }
  249. }
  250. private void onOverflowIconClicked(int section, SyncedFolderDisplayItem item, View view) {
  251. PopupMenu popup = new PopupMenu(context, view);
  252. popup.inflate(R.menu.synced_folders_adapter);
  253. popup.setOnMenuItemClickListener(i -> optionsItemSelected(i, section, item));
  254. popup.getMenu()
  255. .findItem(R.id.action_auto_upload_folder_toggle_visibility)
  256. .setChecked(item.isHidden());
  257. popup.show();
  258. }
  259. private boolean optionsItemSelected(MenuItem menuItem, int section, SyncedFolderDisplayItem item) {
  260. if (menuItem.getItemId() == R.id.action_auto_upload_folder_toggle_visibility) {
  261. clickListener.onVisibilityToggleClick(section, item);
  262. } else {
  263. // default: R.id.action_create_custom_folder
  264. clickListener.onSyncFolderSettingsClick(section, item);
  265. }
  266. return true;
  267. }
  268. @Override
  269. public void onBindFooterViewHolder(SectionedViewHolder holder, int section) {
  270. if (isLastSection(section) && showFooter()) {
  271. FooterViewHolder footerHolder = (FooterViewHolder) holder;
  272. footerHolder.title.setOnClickListener(v -> toggleHiddenItemsVisibility());
  273. footerHolder.title.setText(
  274. context.getResources().getQuantityString(
  275. R.plurals.synced_folders_show_hidden_folders,
  276. getHiddenFolderCount(),
  277. getHiddenFolderCount()
  278. )
  279. );
  280. }
  281. }
  282. @Override
  283. public void onBindViewHolder(SectionedViewHolder commonHolder, int section, int relativePosition,
  284. int absolutePosition) {
  285. if (section < filteredSyncFolderItems.size() && filteredSyncFolderItems.get(section).getFilePaths() != null) {
  286. MainViewHolder holder = (MainViewHolder) commonHolder;
  287. File file = new File(filteredSyncFolderItems.get(section).getFilePaths().get(relativePosition));
  288. ThumbnailsCacheManager.MediaThumbnailGenerationTask task =
  289. new ThumbnailsCacheManager.MediaThumbnailGenerationTask(holder.image, context);
  290. ThumbnailsCacheManager.AsyncMediaThumbnailDrawable asyncDrawable =
  291. new ThumbnailsCacheManager.AsyncMediaThumbnailDrawable(
  292. context.getResources(),
  293. ThumbnailsCacheManager.mDefaultImg,
  294. task
  295. );
  296. holder.image.setImageDrawable(asyncDrawable);
  297. task.execute(file);
  298. // set proper tag
  299. holder.image.setTag(file.hashCode());
  300. holder.itemView.setTag(relativePosition % gridWidth);
  301. if (filteredSyncFolderItems.get(section).getNumberOfFiles() > gridTotal && relativePosition >= gridTotal - 1) {
  302. holder.counterValue.setText(String.format(Locale.US, "%d",
  303. filteredSyncFolderItems.get(section).getNumberOfFiles() - gridTotal));
  304. holder.counterBar.setVisibility(View.VISIBLE);
  305. holder.thumbnailDarkener.setVisibility(View.VISIBLE);
  306. } else {
  307. holder.counterBar.setVisibility(View.GONE);
  308. holder.thumbnailDarkener.setVisibility(View.GONE);
  309. }
  310. }
  311. }
  312. @NonNull
  313. @Override
  314. public SectionedViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
  315. if (viewType == VIEW_TYPE_HEADER) {
  316. View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.synced_folders_item_header, parent, false);
  317. return new HeaderViewHolder(v);
  318. } else if (viewType == VIEW_TYPE_FOOTER) {
  319. View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.synced_folders_footer, parent, false);
  320. return new FooterViewHolder(v);
  321. } else if (viewType == VIEW_TYPE_EMPTY) {
  322. View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.synced_folders_empty, parent, false);
  323. return new EmptyViewHolder(v);
  324. } else {
  325. View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.grid_sync_item, parent, false);
  326. return new MainViewHolder(v);
  327. }
  328. }
  329. private boolean isLastSection(int section) {
  330. return section >= getSectionCount() - 1;
  331. }
  332. public int getHiddenFolderCount() {
  333. if (syncFolderItems != null && filteredSyncFolderItems != null) {
  334. return syncFolderItems.size() - filteredSyncFolderItems.size();
  335. } else {
  336. return 0;
  337. }
  338. }
  339. public interface ClickListener {
  340. void onSyncStatusToggleClick(int section, SyncedFolderDisplayItem syncedFolderDisplayItem);
  341. void onSyncFolderSettingsClick(int section, SyncedFolderDisplayItem syncedFolderDisplayItem);
  342. void onVisibilityToggleClick(int section, SyncedFolderDisplayItem item);
  343. }
  344. static class HeaderViewHolder extends SectionedViewHolder {
  345. @BindView(R.id.header_container)
  346. public RelativeLayout mainHeaderContainer;
  347. @BindView(R.id.type)
  348. public ImageView type;
  349. @BindView(R.id.title)
  350. public TextView title;
  351. @BindView(R.id.syncStatusButton)
  352. public ImageButton syncStatusButton;
  353. @BindView(R.id.settingsButton)
  354. public ImageButton menuButton;
  355. private HeaderViewHolder(View itemView) {
  356. super(itemView);
  357. ButterKnife.bind(this, itemView);
  358. }
  359. }
  360. static class FooterViewHolder extends SectionedViewHolder {
  361. @BindView(R.id.footer_container)
  362. public LinearLayout mainFooterContainer;
  363. @BindView(R.id.footer_text)
  364. public TextView title;
  365. private FooterViewHolder(View itemView) {
  366. super(itemView);
  367. ButterKnife.bind(this, itemView);
  368. }
  369. }
  370. static class EmptyViewHolder extends SectionedViewHolder {
  371. private EmptyViewHolder(View itemView) {
  372. super(itemView);
  373. }
  374. }
  375. static class MainViewHolder extends SectionedViewHolder {
  376. @BindView(R.id.grid_item_container)
  377. public FrameLayout item_container;
  378. @BindView(R.id.thumbnail)
  379. public ImageView image;
  380. @BindView(R.id.counterLayout)
  381. public LinearLayout counterBar;
  382. @BindView(R.id.counter)
  383. public TextView counterValue;
  384. @BindView(R.id.thumbnailDarkener)
  385. public ImageView thumbnailDarkener;
  386. private MainViewHolder(View itemView) {
  387. super(itemView);
  388. ButterKnife.bind(this, itemView);
  389. }
  390. }
  391. private void setSyncButtonActiveIcon(ImageButton syncStatusButton, boolean enabled) {
  392. if (enabled) {
  393. syncStatusButton.setImageDrawable(ThemeUtils.tintDrawable(R.drawable.ic_cloud_sync_on,
  394. ThemeUtils.primaryColor(context)));
  395. } else {
  396. syncStatusButton.setImageResource(R.drawable.ic_cloud_sync_off);
  397. }
  398. }
  399. }