1 /* ownCloud Android client application
2 * Copyright (C) 2012-2013 ownCloud Inc.
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2,
6 * as published by the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License
14 * along with this program. If not, see <http://www.gnu.org/licenses/>.
17 package com
.owncloud
.android
.ui
.preview
;
19 import java
.util
.HashMap
;
20 import java
.util
.HashSet
;
23 import java
.util
.Vector
;
25 import android
.accounts
.Account
;
26 import android
.support
.v4
.app
.Fragment
;
27 import android
.support
.v4
.app
.FragmentManager
;
28 import android
.support
.v4
.app
.FragmentStatePagerAdapter
;
29 import android
.view
.ViewGroup
;
31 import com
.owncloud
.android
.datamodel
.FileDataStorageManager
;
32 import com
.owncloud
.android
.datamodel
.OCFile
;
33 import com
.owncloud
.android
.ui
.fragment
.FileFragment
;
36 * Adapter class that provides Fragment instances
38 * @author David A. Velasco
40 //public class PreviewImagePagerAdapter extends PagerAdapter {
41 public class PreviewImagePagerAdapter
extends FragmentStatePagerAdapter
{
43 private Vector
<OCFile
> mImageFiles
;
44 private Account mAccount
;
45 private Set
<Object
> mObsoleteFragments
;
46 private Set
<Integer
> mObsoletePositions
;
47 private Set
<Integer
> mDownloadErrors
;
48 private FileDataStorageManager mStorageManager
;
50 private Map
<Integer
, FileFragment
> mCachedFragments
;
55 * @param fragmentManager {@link FragmentManager} instance that will handle the {@link Fragment}s provided by the adapter.
56 * @param parentFolder Folder where images will be searched for.
57 * @param storageManager Bridge to database.
59 public PreviewImagePagerAdapter(FragmentManager fragmentManager
, OCFile parentFolder
, Account account
, FileDataStorageManager storageManager
) {
60 super(fragmentManager
);
62 if (fragmentManager
== null
) {
63 throw new IllegalArgumentException("NULL FragmentManager instance");
65 if (parentFolder
== null
) {
66 throw new IllegalArgumentException("NULL parent folder");
68 if (storageManager
== null
) {
69 throw new IllegalArgumentException("NULL storage manager");
73 mStorageManager
= storageManager
;
74 mImageFiles
= mStorageManager
.getFolderImages(parentFolder
);
75 mObsoleteFragments
= new HashSet
<Object
>();
76 mObsoletePositions
= new HashSet
<Integer
>();
77 mDownloadErrors
= new HashSet
<Integer
>();
78 //mFragmentManager = fragmentManager;
79 mCachedFragments
= new HashMap
<Integer
, FileFragment
>();
84 * Returns the image files handled by the adapter.
86 * @return A vector with the image files handled by the adapter.
88 protected OCFile
getFileAt(int position
) {
89 return mImageFiles
.get(position
);
93 public Fragment
getItem(int i
) {
94 OCFile file
= mImageFiles
.get(i
);
95 Fragment fragment
= null
;
97 fragment
= new PreviewImageFragment(file
, mAccount
, mObsoletePositions
.contains(Integer
.valueOf(i
)));
99 } else if (mDownloadErrors
.contains(Integer
.valueOf(i
))) {
100 fragment
= new FileDownloadFragment(file
, mAccount
, true
);
101 ((FileDownloadFragment
)fragment
).setError(true
);
102 mDownloadErrors
.remove(Integer
.valueOf(i
));
105 fragment
= new FileDownloadFragment(file
, mAccount
, mObsoletePositions
.contains(Integer
.valueOf(i
)));
107 mObsoletePositions
.remove(Integer
.valueOf(i
));
111 public int getFilePosition(OCFile file
) {
112 return mImageFiles
.indexOf(file
);
116 public int getCount() {
117 return mImageFiles
.size();
121 public CharSequence
getPageTitle(int position
) {
122 return mImageFiles
.get(position
).getFileName();
126 public void updateFile(int position
, OCFile file
) {
127 FileFragment fragmentToUpdate
= mCachedFragments
.get(Integer
.valueOf(position
));
128 if (fragmentToUpdate
!= null
) {
129 mObsoleteFragments
.add(fragmentToUpdate
);
131 mObsoletePositions
.add(Integer
.valueOf(position
));
132 mImageFiles
.set(position
, file
);
136 public void updateWithDownloadError(int position
) {
137 FileFragment fragmentToUpdate
= mCachedFragments
.get(Integer
.valueOf(position
));
138 if (fragmentToUpdate
!= null
) {
139 mObsoleteFragments
.add(fragmentToUpdate
);
141 mDownloadErrors
.add(Integer
.valueOf(position
));
144 public void clearErrorAt(int position
) {
145 FileFragment fragmentToUpdate
= mCachedFragments
.get(Integer
.valueOf(position
));
146 if (fragmentToUpdate
!= null
) {
147 mObsoleteFragments
.add(fragmentToUpdate
);
149 mDownloadErrors
.remove(Integer
.valueOf(position
));
154 public int getItemPosition(Object object
) {
155 if (mObsoleteFragments
.contains(object
)) {
156 mObsoleteFragments
.remove(object
);
157 return POSITION_NONE
;
159 return super.getItemPosition(object
);
164 public Object
instantiateItem(ViewGroup container
, int position
) {
165 Object fragment
= super.instantiateItem(container
, position
);
166 mCachedFragments
.put(Integer
.valueOf(position
), (FileFragment
)fragment
);
171 public void destroyItem(ViewGroup container
, int position
, Object object
) {
172 mCachedFragments
.remove(Integer
.valueOf(position
));
173 super.destroyItem(container
, position
, object
);
177 public boolean pendingErrorAt(int position
) {
178 return mDownloadErrors
.contains(Integer
.valueOf(position
));
182 * Called when a change in the shown pages is going to start being made.
184 * @param container The containing View which is displaying this adapter's page views.
187 public void startUpdate(ViewGroup container) {
188 Log.e(TAG, "** startUpdate");
192 public Object instantiateItem(ViewGroup container, int position) {
193 Log.e(TAG, "** instantiateItem " + position);
195 if (mFragments.size() > position) {
196 Fragment fragment = mFragments.get(position);
197 if (fragment != null) {
198 Log.e(TAG, "** \t returning cached item");
203 if (mCurTransaction == null) {
204 mCurTransaction = mFragmentManager.beginTransaction();
207 Fragment fragment = getItem(position);
208 if (mSavedState.size() > position) {
209 Fragment.SavedState savedState = mSavedState.get(position);
210 if (savedState != null) {
212 // * The Fragment must currently be attached to the FragmentManager.
213 // * A new Fragment created using this saved state must be the same class type as the Fragment it was created from.
214 // * The saved state can not contain dependencies on other fragments -- that is it can't use putFragment(Bundle, String, Fragment)
215 // to store a fragment reference
216 fragment.setInitialSavedState(savedState);
219 while (mFragments.size() <= position) {
220 mFragments.add(null);
222 fragment.setMenuVisibility(false);
223 mFragments.set(position, fragment);
224 //Log.e(TAG, "** \t adding fragment at position " + position + ", containerId " + container.getId());
225 mCurTransaction.add(container.getId(), fragment);
231 public void destroyItem(ViewGroup container, int position, Object object) {
232 Log.e(TAG, "** destroyItem " + position);
233 Fragment fragment = (Fragment)object;
235 if (mCurTransaction == null) {
236 mCurTransaction = mFragmentManager.beginTransaction();
238 Log.e(TAG, "** \t removing fragment at position " + position);
239 while (mSavedState.size() <= position) {
240 mSavedState.add(null);
242 mSavedState.set(position, mFragmentManager.saveFragmentInstanceState(fragment));
243 mFragments.set(position, null);
245 mCurTransaction.remove(fragment);
249 public void setPrimaryItem(ViewGroup container, int position, Object object) {
250 Fragment fragment = (Fragment)object;
251 if (fragment != mCurrentPrimaryItem) {
252 if (mCurrentPrimaryItem != null) {
253 mCurrentPrimaryItem.setMenuVisibility(false);
255 if (fragment != null) {
256 fragment.setMenuVisibility(true);
258 mCurrentPrimaryItem = fragment;
263 public void finishUpdate(ViewGroup container) {
264 Log.e(TAG, "** finishUpdate (start)");
265 if (mCurTransaction != null) {
266 mCurTransaction.commitAllowingStateLoss();
267 mCurTransaction = null;
268 mFragmentManager.executePendingTransactions();
270 Log.e(TAG, "** finishUpdate (end)");
274 public boolean isViewFromObject(View view, Object object) {
275 return ((Fragment)object).getView() == view;
279 public Parcelable saveState() {
281 if (mSavedState.size() > 0) {
282 state = new Bundle();
283 Fragment.SavedState[] savedStates = new Fragment.SavedState[mSavedState.size()];
284 mSavedState.toArray(savedStates);
285 state.putParcelableArray("states", savedStates);
287 for (int i=0; i<mFragments.size(); i++) {
288 Fragment fragment = mFragments.get(i);
289 if (fragment != null) {
291 state = new Bundle();
293 String key = "f" + i;
294 mFragmentManager.putFragment(state, key, fragment);
301 public void restoreState(Parcelable state, ClassLoader loader) {
303 Bundle bundle = (Bundle)state;
304 bundle.setClassLoader(loader);
305 Parcelable[] states = bundle.getParcelableArray("states");
308 if (states != null) {
309 for (int i=0; i<states.length; i++) {
310 mSavedState.add((Fragment.SavedState)states[i]);
313 Iterable<String> keys = bundle.keySet();
314 for (String key: keys) {
315 if (key.startsWith("f")) {
316 int index = Integer.parseInt(key.substring(1));
317 Fragment f = mFragmentManager.getFragment(bundle, key);
319 while (mFragments.size() <= index) {
320 mFragments.add(null);
322 f.setMenuVisibility(false);
323 mFragments.set(index, f);
325 Log.w(TAG, "Bad fragment at key " + key);