ExtendedListFragment.java 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  1. /**
  2. * ownCloud Android client application
  3. *
  4. * Copyright (C) 2012 Bartek Przybylski
  5. * Copyright (C) 2012-2015 ownCloud Inc.
  6. *
  7. * This program is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License version 2,
  9. * as published by the Free Software Foundation.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. *
  19. */
  20. package com.owncloud.android.ui.fragment;
  21. import java.util.ArrayList;
  22. import android.os.Build;
  23. import android.os.Bundle;
  24. import android.support.v4.widget.SwipeRefreshLayout;
  25. import android.view.LayoutInflater;
  26. import android.view.View;
  27. import android.view.ViewGroup;
  28. import android.widget.AbsListView;
  29. import android.widget.AdapterView;
  30. import android.widget.AdapterView.OnItemClickListener;
  31. import android.widget.GridView;
  32. import android.widget.ListAdapter;
  33. import android.widget.ListView;
  34. import android.widget.TextView;
  35. import com.actionbarsherlock.app.SherlockFragment;
  36. import com.owncloud.android.R;
  37. import com.owncloud.android.lib.common.utils.Log_OC;
  38. import com.owncloud.android.ui.ExtendedListView;
  39. import com.owncloud.android.ui.activity.OnEnforceableRefreshListener;
  40. import com.owncloud.android.ui.adapter.FileListListAdapter;
  41. import third_parties.in.srain.cube.GridViewWithHeaderAndFooter;
  42. /**
  43. * TODO extending SherlockListFragment instead of SherlockFragment
  44. */
  45. public class ExtendedListFragment extends SherlockFragment
  46. implements OnItemClickListener, OnEnforceableRefreshListener {
  47. private static final String TAG = ExtendedListFragment.class.getSimpleName();
  48. private static final String KEY_SAVED_LIST_POSITION = "SAVED_LIST_POSITION";
  49. private static final String KEY_INDEXES = "INDEXES";
  50. private static final String KEY_FIRST_POSITIONS= "FIRST_POSITIONS";
  51. private static final String KEY_TOPS = "TOPS";
  52. private static final String KEY_HEIGHT_CELL = "HEIGHT_CELL";
  53. private static final String KEY_EMPTY_LIST_MESSAGE = "EMPTY_LIST_MESSAGE";
  54. private SwipeRefreshLayout mRefreshListLayout;
  55. private SwipeRefreshLayout mRefreshGridLayout;
  56. private SwipeRefreshLayout mRefreshEmptyLayout;
  57. private TextView mEmptyListMessage;
  58. // Save the state of the scroll in browsing
  59. private ArrayList<Integer> mIndexes;
  60. private ArrayList<Integer> mFirstPositions;
  61. private ArrayList<Integer> mTops;
  62. private int mHeightCell = 0;
  63. private OnEnforceableRefreshListener mOnRefreshListener = null;
  64. protected AbsListView mCurrentListView;
  65. private ExtendedListView mListView;
  66. private View mListFooterView;
  67. private GridViewWithHeaderAndFooter mGridView;
  68. private View mGridFooterView;
  69. private ListAdapter mAdapter;
  70. protected void setListAdapter(ListAdapter listAdapter) {
  71. mAdapter = listAdapter;
  72. if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
  73. mCurrentListView.setAdapter(listAdapter);
  74. } else {
  75. ((ListView)mCurrentListView).setAdapter(listAdapter);
  76. }
  77. mCurrentListView.invalidate();
  78. }
  79. protected AbsListView getListView() {
  80. return mCurrentListView;
  81. }
  82. protected void switchToGridView() {
  83. if ((mCurrentListView == mListView)) {
  84. mListView.setAdapter(null);
  85. mRefreshListLayout.setVisibility(View.GONE);
  86. if (mAdapter instanceof FileListListAdapter) {
  87. ((FileListListAdapter) mAdapter).setGridMode(true);
  88. }
  89. mGridView.setAdapter(mAdapter);
  90. mRefreshGridLayout.setVisibility(View.VISIBLE);
  91. mCurrentListView = mGridView;
  92. }
  93. }
  94. protected void switchToListView() {
  95. if (mCurrentListView == mGridView) {
  96. mGridView.setAdapter(null);
  97. mRefreshGridLayout.setVisibility(View.GONE);
  98. if (mAdapter instanceof FileListListAdapter) {
  99. ((FileListListAdapter) mAdapter).setGridMode(false);
  100. }
  101. mListView.setAdapter(mAdapter);
  102. mRefreshListLayout.setVisibility(View.VISIBLE);
  103. mCurrentListView = mListView;
  104. }
  105. }
  106. @Override
  107. public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
  108. Log_OC.d(TAG, "onCreateView");
  109. View v = inflater.inflate(R.layout.list_fragment, null);
  110. mListView = (ExtendedListView)(v.findViewById(R.id.list_root));
  111. mListView.setOnItemClickListener(this);
  112. mListFooterView = inflater.inflate(R.layout.list_footer, null, false);
  113. mGridView = (GridViewWithHeaderAndFooter) (v.findViewById(R.id.grid_root));
  114. mGridView.setNumColumns(GridView.AUTO_FIT);
  115. mGridView.setOnItemClickListener(this);
  116. mGridFooterView = inflater.inflate(R.layout.list_footer, null, false);
  117. if (savedInstanceState != null) {
  118. int referencePosition = savedInstanceState.getInt(KEY_SAVED_LIST_POSITION);
  119. if (mCurrentListView == mListView) {
  120. Log_OC.v(TAG, "Setting and centering around list position " + referencePosition);
  121. mListView.setAndCenterSelection(referencePosition);
  122. } else {
  123. Log_OC.v(TAG, "Setting grid position " + referencePosition);
  124. mGridView.setSelection(referencePosition);
  125. }
  126. }
  127. // Pull-down to refresh layout
  128. mRefreshListLayout = (SwipeRefreshLayout) v.findViewById(R.id.swipe_containing_list);
  129. mRefreshGridLayout = (SwipeRefreshLayout) v.findViewById(R.id.swipe_containing_grid);
  130. mRefreshEmptyLayout = (SwipeRefreshLayout) v.findViewById(R.id.swipe_containing_empty);
  131. mEmptyListMessage = (TextView) v.findViewById(R.id.empty_list_view);
  132. onCreateSwipeToRefresh(mRefreshListLayout);
  133. onCreateSwipeToRefresh(mRefreshGridLayout);
  134. onCreateSwipeToRefresh(mRefreshEmptyLayout);
  135. mListView.setEmptyView(mRefreshEmptyLayout);
  136. mGridView.setEmptyView(mRefreshEmptyLayout);
  137. mCurrentListView = mListView; // list as default
  138. return v;
  139. }
  140. /**
  141. * {@inheritDoc}
  142. */
  143. @Override
  144. public void onActivityCreated(Bundle savedInstanceState) {
  145. super.onActivityCreated(savedInstanceState);
  146. if (savedInstanceState != null) {
  147. mIndexes = savedInstanceState.getIntegerArrayList(KEY_INDEXES);
  148. mFirstPositions = savedInstanceState.getIntegerArrayList(KEY_FIRST_POSITIONS);
  149. mTops = savedInstanceState.getIntegerArrayList(KEY_TOPS);
  150. mHeightCell = savedInstanceState.getInt(KEY_HEIGHT_CELL);
  151. setMessageForEmptyList(savedInstanceState.getString(KEY_EMPTY_LIST_MESSAGE));
  152. } else {
  153. mIndexes = new ArrayList<Integer>();
  154. mFirstPositions = new ArrayList<Integer>();
  155. mTops = new ArrayList<Integer>();
  156. mHeightCell = 0;
  157. }
  158. }
  159. @Override
  160. public void onSaveInstanceState(Bundle savedInstanceState) {
  161. super.onSaveInstanceState(savedInstanceState);
  162. Log_OC.d(TAG, "onSaveInstanceState()");
  163. savedInstanceState.putInt(KEY_SAVED_LIST_POSITION, getReferencePosition());
  164. savedInstanceState.putIntegerArrayList(KEY_INDEXES, mIndexes);
  165. savedInstanceState.putIntegerArrayList(KEY_FIRST_POSITIONS, mFirstPositions);
  166. savedInstanceState.putIntegerArrayList(KEY_TOPS, mTops);
  167. savedInstanceState.putInt(KEY_HEIGHT_CELL, mHeightCell);
  168. savedInstanceState.putString(KEY_EMPTY_LIST_MESSAGE, getEmptyViewText());
  169. }
  170. /**
  171. * Calculates the position of the item that will be used as a reference to
  172. * reposition the visible items in the list when the device is turned to
  173. * other position.
  174. *
  175. * The current policy is take as a reference the visible item in the center
  176. * of the screen.
  177. *
  178. * @return The position in the list of the visible item in the center of the
  179. * screen.
  180. */
  181. protected int getReferencePosition() {
  182. if (mCurrentListView != null) {
  183. return (mCurrentListView.getFirstVisiblePosition() +
  184. mCurrentListView.getLastVisiblePosition()) / 2;
  185. } else {
  186. return 0;
  187. }
  188. }
  189. /*
  190. * Restore index and position
  191. */
  192. protected void restoreIndexAndTopPosition() {
  193. if (mIndexes.size() > 0) {
  194. // needs to be checked; not every browse-up had a browse-down before
  195. int index = mIndexes.remove(mIndexes.size() - 1);
  196. final int firstPosition = mFirstPositions.remove(mFirstPositions.size() -1);
  197. int top = mTops.remove(mTops.size() - 1);
  198. Log_OC.v(TAG, "Setting selection to position: " + firstPosition + "; top: "
  199. + top + "; index: " + index);
  200. if (mCurrentListView == mListView) {
  201. if (mHeightCell*index <= mListView.getHeight()) {
  202. mListView.setSelectionFromTop(firstPosition, top);
  203. } else {
  204. mListView.setSelectionFromTop(index, 0);
  205. }
  206. } else {
  207. if (mHeightCell*index <= mGridView.getHeight()) {
  208. mGridView.setSelection(firstPosition);
  209. //mGridView.smoothScrollToPosition(firstPosition);
  210. } else {
  211. mGridView.setSelection(index);
  212. //mGridView.smoothScrollToPosition(index);
  213. }
  214. }
  215. }
  216. }
  217. /*
  218. * Save index and top position
  219. */
  220. protected void saveIndexAndTopPosition(int index) {
  221. mIndexes.add(index);
  222. int firstPosition = mCurrentListView.getFirstVisiblePosition();
  223. mFirstPositions.add(firstPosition);
  224. View view = mCurrentListView.getChildAt(0);
  225. int top = (view == null) ? 0 : view.getTop() ;
  226. mTops.add(top);
  227. // Save the height of a cell
  228. mHeightCell = (view == null || mHeightCell != 0) ? mHeightCell : view.getHeight();
  229. }
  230. @Override
  231. public void onItemClick (AdapterView<?> parent, View view, int position, long id) {
  232. // to be @overriden
  233. }
  234. @Override
  235. public void onRefresh() {
  236. mRefreshListLayout.setRefreshing(false);
  237. mRefreshGridLayout.setRefreshing(false);
  238. mRefreshEmptyLayout.setRefreshing(false);
  239. if (mOnRefreshListener != null) {
  240. mOnRefreshListener.onRefresh();
  241. }
  242. }
  243. public void setOnRefreshListener(OnEnforceableRefreshListener listener) {
  244. mOnRefreshListener = listener;
  245. }
  246. /**
  247. * Disables swipe gesture.
  248. *
  249. * Sets the 'enabled' state of the refresh layouts contained in the fragment.
  250. *
  251. * When 'false' is set, prevents user gestures but keeps the option to refresh programatically,
  252. *
  253. * @param enabled Desired state for capturing swipe gesture.
  254. */
  255. public void setSwipeEnabled(boolean enabled) {
  256. mRefreshListLayout.setEnabled(enabled);
  257. mRefreshGridLayout.setEnabled(enabled);
  258. mRefreshEmptyLayout.setEnabled(enabled);
  259. }
  260. /**
  261. * Set message for empty list view
  262. */
  263. public void setMessageForEmptyList(String message) {
  264. if (mEmptyListMessage != null) {
  265. mEmptyListMessage.setText(message);
  266. }
  267. }
  268. /**
  269. * Get the text of EmptyListMessage TextView
  270. *
  271. * @return String
  272. */
  273. public String getEmptyViewText() {
  274. return (mEmptyListMessage != null) ? mEmptyListMessage.getText().toString() : "";
  275. }
  276. private void onCreateSwipeToRefresh(SwipeRefreshLayout refreshLayout) {
  277. // Colors in animations: background
  278. refreshLayout.setColorScheme(R.color.background_color, R.color.background_color,
  279. R.color.background_color, R.color.background_color);
  280. refreshLayout.setOnRefreshListener(this);
  281. }
  282. @Override
  283. public void onRefresh(boolean ignoreETag) {
  284. mRefreshListLayout.setRefreshing(false);
  285. mRefreshGridLayout.setRefreshing(false);
  286. mRefreshEmptyLayout.setRefreshing(false);
  287. if (mOnRefreshListener != null) {
  288. mOnRefreshListener.onRefresh(ignoreETag);
  289. }
  290. }
  291. protected void setChoiceMode(int choiceMode) {
  292. if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
  293. mListView.setChoiceMode(choiceMode);
  294. mGridView.setChoiceMode(choiceMode);
  295. } else {
  296. ((ListView)mListView).setChoiceMode(choiceMode);
  297. }
  298. }
  299. protected void registerForContextMenu() {
  300. registerForContextMenu(mListView);
  301. registerForContextMenu(mGridView);
  302. mListView.setOnCreateContextMenuListener(this);
  303. mGridView.setOnCreateContextMenuListener(this);
  304. }
  305. /**
  306. * TODO doc
  307. * To be called before setAdapter, or GridViewWithHeaderAndFooter will throw an exception
  308. *
  309. * @param enabled
  310. */
  311. protected void setFooterEnabled(boolean enabled) {
  312. if (enabled) {
  313. if (mGridView.getFooterViewCount() == 0) {
  314. if (mGridFooterView.getParent() != null ) {
  315. ((ViewGroup) mGridFooterView.getParent()).removeView(mGridFooterView);
  316. }
  317. mGridView.addFooterView(mGridFooterView, null, false);
  318. }
  319. mGridFooterView.invalidate();
  320. if (mListView.getFooterViewsCount() == 0) {
  321. if (mListFooterView.getParent() != null ) {
  322. ((ViewGroup) mListFooterView.getParent()).removeView(mListFooterView);
  323. }
  324. mListView.addFooterView(mListFooterView, null, false);
  325. }
  326. mListFooterView.invalidate();
  327. } else {
  328. // mGridView.removeFooterView(mGridFooterView);
  329. // mListView.removeFooterView(mListFooterView);
  330. }
  331. }
  332. /**
  333. * TODO doc
  334. * @param text
  335. */
  336. protected void setFooterText(String text) {
  337. if (text != null && text.length() > 0) {
  338. ((TextView)mListFooterView.findViewById(R.id.footerText)).setText(text);
  339. ((TextView)mGridFooterView.findViewById(R.id.footerText)).setText(text);
  340. setFooterEnabled(true);
  341. } else {
  342. setFooterEnabled(false);
  343. }
  344. }
  345. }