1 /* ownCloud Android client application 
   2  *   Copyright (C) 2011  Bartek Przybylski 
   3  *   Copyright (C) 2012-2014 ownCloud Inc. 
   5  *   This program is free software: you can redistribute it and/or modify 
   6  *   it under the terms of the GNU General Public License version 2, 
   7  *   as published by the Free Software Foundation. 
   9  *   This program is distributed in the hope that it will be useful, 
  10  *   but WITHOUT ANY WARRANTY; without even the implied warranty of 
  11  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the 
  12  *   GNU General Public License for more details. 
  14  *   You should have received a copy of the GNU General Public License 
  15  *   along with this program.  If not, see <http://www.gnu.org/licenses/>. 
  18 package com
.owncloud
.android
.ui
.adapter
; 
  22 import java
.util
.Collections
; 
  23 import java
.util
.Comparator
; 
  24 import java
.util
.Vector
; 
  26 import third_parties
.daveKoeller
.AlphanumComparator
; 
  27 import android
.accounts
.Account
; 
  28 import android
.content
.Context
; 
  29 import android
.content
.SharedPreferences
; 
  30 import android
.graphics
.Bitmap
; 
  31 import android
.preference
.PreferenceManager
; 
  32 import android
.text
.format
.DateUtils
; 
  33 import android
.view
.LayoutInflater
; 
  34 import android
.view
.View
; 
  35 import android
.view
.ViewGroup
; 
  36 import android
.widget
.BaseAdapter
; 
  37 import android
.widget
.ImageView
; 
  38 import android
.widget
.ListAdapter
; 
  39 import android
.widget
.ListView
; 
  40 import android
.widget
.TextView
; 
  42 import com
.owncloud
.android
.R
; 
  43 import com
.owncloud
.android
.authentication
.AccountUtils
; 
  44 import com
.owncloud
.android
.datamodel
.FileDataStorageManager
; 
  45 import com
.owncloud
.android
.datamodel
.OCFile
; 
  46 import com
.owncloud
.android
.datamodel
.ThumbnailsCacheManager
; 
  47 import com
.owncloud
.android
.datamodel
.ThumbnailsCacheManager
.AsyncDrawable
; 
  48 import com
.owncloud
.android
.files
.services
.FileDownloader
.FileDownloaderBinder
; 
  49 import com
.owncloud
.android
.files
.services
.FileUploader
.FileUploaderBinder
; 
  50 import com
.owncloud
.android
.services
.OperationsService
.OperationsServiceBinder
; 
  51 import com
.owncloud
.android
.ui
.activity
.ComponentsGetter
; 
  52 import com
.owncloud
.android
.utils
.DisplayUtils
; 
  53 import com
.owncloud
.android
.utils
.FileStorageUtils
; 
  57  * This Adapter populates a ListView with all files and folders in an ownCloud 
  60  * @author Bartek Przybylski 
  61  * @author Tobias Kaminsky 
  62  * @author David A. Velasco 
  64 public class FileListListAdapter 
extends BaseAdapter 
implements ListAdapter 
{ 
  65     private final static String PERMISSION_SHARED_WITH_ME 
= "S"; 
  67     private Context mContext
; 
  68     private OCFile mFile 
= null
; 
  69     private Vector
<OCFile
> mFiles 
= null
; 
  70     private boolean mJustFolders
; 
  72     private FileDataStorageManager mStorageManager
; 
  73     private Account mAccount
; 
  74     private ComponentsGetter mTransferServiceGetter
; 
  75     private Integer mSortOrder
; 
  76     public static final Integer SORT_NAME 
= 0; 
  77     public static final Integer SORT_DATE 
= 1; 
  78     public static final Integer SORT_SIZE 
= 2; 
  79     private Boolean mSortAscending
; 
  80     private SharedPreferences mAppPreferences
; 
  82     public FileListListAdapter( 
  85             ComponentsGetter transferServiceGetter
 
  88         mJustFolders 
= justFolders
; 
  90         mAccount 
= AccountUtils
.getCurrentOwnCloudAccount(mContext
); 
  92         mTransferServiceGetter 
= transferServiceGetter
; 
  94         mAppPreferences 
= PreferenceManager
 
  95                 .getDefaultSharedPreferences(mContext
); 
  97         // Read sorting order, default to sort by name ascending 
  98         mSortOrder 
= mAppPreferences
 
  99                 .getInt("sortOrder", 0); 
 100         mSortAscending 
= mAppPreferences
.getBoolean("sortAscending", true
); 
 102         // initialise thumbnails cache on background thread 
 103         new ThumbnailsCacheManager
.InitDiskCacheTask().execute(); 
 108     public boolean areAllItemsEnabled() { 
 113     public boolean isEnabled(int position
) { 
 118     public int getCount() { 
 119         return mFiles 
!= null ? mFiles
.size() : 0; 
 123     public Object 
getItem(int position
) { 
 124         if (mFiles 
== null 
|| mFiles
.size() <= position
) 
 126         return mFiles
.get(position
); 
 130     public long getItemId(int position
) { 
 131         if (mFiles 
== null 
|| mFiles
.size() <= position
) 
 133         return mFiles
.get(position
).getFileId(); 
 137     public int getItemViewType(int position
) { 
 142     public View 
getView(int position
, View convertView
, ViewGroup parent
) { 
 143         View view 
= convertView
; 
 145             LayoutInflater inflator 
= (LayoutInflater
) mContext
 
 146                     .getSystemService(Context
.LAYOUT_INFLATER_SERVICE
); 
 147             view 
= inflator
.inflate(R
.layout
.list_item
, null
); 
 150         if (mFiles 
!= null 
&& mFiles
.size() > position
) { 
 151             OCFile file 
= mFiles
.get(position
); 
 152             TextView fileName 
= (TextView
) view
.findViewById(R
.id
.Filename
);            
 153             String name 
= file
.getFileName(); 
 155             fileName
.setText(name
); 
 156             ImageView fileIcon 
= (ImageView
) view
.findViewById(R
.id
.imageView1
); 
 157             fileIcon
.setTag(file
.getFileId()); 
 158             ImageView sharedIconV 
= (ImageView
) view
.findViewById(R
.id
.sharedIcon
); 
 159             ImageView sharedWithMeIconV 
= (ImageView
) view
.findViewById(R
.id
.sharedWithMeIcon
); 
 160             sharedWithMeIconV
.setVisibility(View
.GONE
); 
 162             ImageView localStateView 
= (ImageView
) view
.findViewById(R
.id
.imageView2
); 
 163             localStateView
.bringToFront(); 
 164             FileDownloaderBinder downloaderBinder 
=  
 165                     mTransferServiceGetter
.getFileDownloaderBinder(); 
 166             FileUploaderBinder uploaderBinder 
= mTransferServiceGetter
.getFileUploaderBinder(); 
 167             OperationsServiceBinder opsBinder 
= mTransferServiceGetter
.getOperationsServiceBinder(); 
 168             if ((downloaderBinder 
!= null 
&& downloaderBinder
.isDownloading(mAccount
, file
)) || 
 169                  (opsBinder 
!= null 
&& opsBinder
.isSynchronizing(mAccount
, file
.getRemotePath()))) { 
 170                 localStateView
.setImageResource(R
.drawable
.downloading_file_indicator
); 
 171                 localStateView
.setVisibility(View
.VISIBLE
); 
 172             } else if (uploaderBinder 
!= null 
&& uploaderBinder
.isUploading(mAccount
, file
)) { 
 173                 localStateView
.setImageResource(R
.drawable
.uploading_file_indicator
); 
 174                 localStateView
.setVisibility(View
.VISIBLE
); 
 175             } else if (file
.isDown()) { 
 176                 localStateView
.setImageResource(R
.drawable
.local_file_indicator
); 
 177                 localStateView
.setVisibility(View
.VISIBLE
); 
 179                 localStateView
.setVisibility(View
.INVISIBLE
); 
 182             TextView fileSizeV 
= (TextView
) view
.findViewById(R
.id
.file_size
); 
 183             TextView lastModV 
= (TextView
) view
.findViewById(R
.id
.last_mod
); 
 184             ImageView checkBoxV 
= (ImageView
) view
.findViewById(R
.id
.custom_checkbox
); 
 186             if (!file
.isFolder()) { 
 187                 fileSizeV
.setVisibility(View
.VISIBLE
); 
 188                 fileSizeV
.setText(DisplayUtils
.bytesToHumanReadable(file
.getFileLength())); 
 189                 lastModV
.setVisibility(View
.VISIBLE
); 
 190                 lastModV
.setText(showRelativeTimestamp(file
)); 
 191                 // this if-else is needed even thoe fav icon is visible by default 
 192                 // because android reuses views in listview 
 193                 if (!file
.keepInSync()) { 
 194                     view
.findViewById(R
.id
.imageView3
).setVisibility(View
.GONE
); 
 196                     view
.findViewById(R
.id
.imageView3
).setVisibility(View
.VISIBLE
); 
 199                 ListView parentList 
= (ListView
)parent
; 
 200                 if (parentList
.getChoiceMode() == ListView
.CHOICE_MODE_NONE
) {  
 201                     checkBoxV
.setVisibility(View
.GONE
); 
 203                     if (parentList
.isItemChecked(position
)) { 
 204                         checkBoxV
.setImageResource(android
.R
.drawable
.checkbox_on_background
); 
 206                         checkBoxV
.setImageResource(android
.R
.drawable
.checkbox_off_background
); 
 208                     checkBoxV
.setVisibility(View
.VISIBLE
); 
 211                 // get Thumbnail if file is image 
 212                 if (file
.isImage() && file
.getRemoteId() != null
){ 
 213                      // Thumbnail in Cache? 
 214                     Bitmap thumbnail 
= ThumbnailsCacheManager
.getBitmapFromDiskCache( 
 215                             String
.valueOf(file
.getRemoteId()) 
 217                     if (thumbnail 
!= null 
&& !file
.needsUpdateThumbnail()){ 
 218                         fileIcon
.setImageBitmap(thumbnail
); 
 220                         // generate new Thumbnail 
 221                         if (ThumbnailsCacheManager
.cancelPotentialWork(file
, fileIcon
)) { 
 222                             final ThumbnailsCacheManager
.ThumbnailGenerationTask task 
=  
 223                                     new ThumbnailsCacheManager
.ThumbnailGenerationTask( 
 224                                             fileIcon
, mStorageManager
, mAccount
 
 226                             if (thumbnail 
== null
) { 
 227                                 thumbnail 
= ThumbnailsCacheManager
.mDefaultImg
; 
 229                             final AsyncDrawable asyncDrawable 
= new AsyncDrawable( 
 230                                     mContext
.getResources(),  
 234                             fileIcon
.setImageDrawable(asyncDrawable
); 
 239                     fileIcon
.setImageResource( 
 240                             DisplayUtils
.getResourceId(file
.getMimetype(), file
.getFileName()) 
 244                 if (checkIfFileIsSharedWithMe(file
)) { 
 245                     sharedWithMeIconV
.setVisibility(View
.VISIBLE
); 
 249                   // TODO Re-enable when server supports folder-size calculation 
 250 //                if (FileStorageUtils.getDefaultSavePathFor(mAccount.name, file) != null){ 
 251 //                    fileSizeV.setVisibility(View.VISIBLE); 
 252 //                    fileSizeV.setText(getFolderSizeHuman(FileStorageUtils.getDefaultSavePathFor(mAccount.name, file))); 
 254                     fileSizeV
.setVisibility(View
.INVISIBLE
); 
 257                 lastModV
.setVisibility(View
.VISIBLE
); 
 258                 lastModV
.setText(showRelativeTimestamp(file
)); 
 259                 checkBoxV
.setVisibility(View
.GONE
); 
 260                 view
.findViewById(R
.id
.imageView3
).setVisibility(View
.GONE
); 
 262                 if (checkIfFileIsSharedWithMe(file
)) { 
 263                     fileIcon
.setImageResource(R
.drawable
.shared_with_me_folder
); 
 264                     sharedWithMeIconV
.setVisibility(View
.VISIBLE
); 
 266                     fileIcon
.setImageResource( 
 267                             DisplayUtils
.getResourceId(file
.getMimetype(), file
.getFileName()) 
 271                 // If folder is sharedByLink, icon folder must be changed to 
 273                 if (file
.isShareByLink()) { 
 274                     fileIcon
.setImageResource(R
.drawable
.folder_public
); 
 278             if (file
.isShareByLink()) { 
 279                 sharedIconV
.setVisibility(View
.VISIBLE
); 
 281                 sharedIconV
.setVisibility(View
.GONE
); 
 289      * Local Folder size in human readable format 
 293      * @return Size in human readable format 
 295     private String 
getFolderSizeHuman(String path
) { 
 297         File dir 
= new File(path
); 
 300             long bytes 
= getFolderSize(dir
); 
 301             return DisplayUtils
.bytesToHumanReadable(bytes
); 
 310      * @return Size in bytes 
 312     private long getFolderSize(File dir
) { 
 315             File
[] fileList 
= dir
.listFiles(); 
 316             for(int i 
= 0; i 
< fileList
.length
; i
++) { 
 317                 if(fileList
[i
].isDirectory()) { 
 318                     result 
+= getFolderSize(fileList
[i
]); 
 320                     result 
+= fileList
[i
].length(); 
 329     public int getViewTypeCount() { 
 334     public boolean hasStableIds() { 
 339     public boolean isEmpty() { 
 340         return (mFiles 
== null 
|| mFiles
.isEmpty()); 
 344      * Change the adapted directory for a new one 
 345      * @param directory                 New file to adapt. Can be NULL, meaning  
 346      *                                  "no content to adapt". 
 347      * @param updatedStorageManager     Optional updated storage manager; used to replace  
 348      *                                  mStorageManager if is different (and not NULL) 
 350     public void swapDirectory(OCFile directory
, FileDataStorageManager updatedStorageManager
) { 
 352         if (updatedStorageManager 
!= null 
&& updatedStorageManager 
!= mStorageManager
) { 
 353             mStorageManager 
= updatedStorageManager
; 
 354             mAccount 
= AccountUtils
.getCurrentOwnCloudAccount(mContext
); 
 356         if (mStorageManager 
!= null
) { 
 357             mFiles 
= mStorageManager
.getFolderContent(mFile
); 
 359                 mFiles 
= getFolders(mFiles
); 
 369      * Sorts all filenames, regarding last user decision  
 371     private void sortDirectory(){ 
 374             sortByName(mSortAscending
); 
 377             sortByDate(mSortAscending
); 
 380             sortBySize(mSortAscending
); 
 384         notifyDataSetChanged(); 
 389      * Filter for getting only the folders 
 391      * @return Vector<OCFile> 
 393     public Vector
<OCFile
> getFolders(Vector
<OCFile
> files
) { 
 394         Vector
<OCFile
> ret 
= new Vector
<OCFile
>();  
 395         OCFile current 
= null
;  
 396         for (int i
=0; i
<files
.size(); i
++) { 
 397             current 
= files
.get(i
); 
 398             if (current
.isFolder()) { 
 407      * Check if parent folder does not include 'S' permission and if file/folder 
 410      * @param file: OCFile 
 411      * @return boolean: True if it is shared with me and false if it is not 
 413     private boolean checkIfFileIsSharedWithMe(OCFile file
) { 
 414         return (mFile
.getPermissions() != null 
 
 415                 && !mFile
.getPermissions().contains(PERMISSION_SHARED_WITH_ME
) 
 416                 && file
.getPermissions() != null 
 
 417                 && file
.getPermissions().contains(PERMISSION_SHARED_WITH_ME
)); 
 422      * @param sortAscending true: ascending, false: descending 
 424     private void sortByDate(boolean sortAscending
){ 
 432         Collections
.sort(mFiles
, new Comparator
<OCFile
>() { 
 433             public int compare(OCFile o1
, OCFile o2
) { 
 434                 if (o1
.isFolder() && o2
.isFolder()) { 
 435                     Long obj1 
= o1
.getModificationTimestamp(); 
 436                     return val 
* obj1
.compareTo(o2
.getModificationTimestamp()); 
 438                 else if (o1
.isFolder()) { 
 440                 } else if (o2
.isFolder()) { 
 442                 } else if (o1
.getModificationTimestamp() == 0 || o2
.getModificationTimestamp() == 0){ 
 445                     Long obj1 
= o1
.getModificationTimestamp(); 
 446                     return val 
* obj1
.compareTo(o2
.getModificationTimestamp()); 
 454      * @param sortAscending true: ascending, false: descending 
 456     private void sortBySize(boolean sortAscending
){ 
 464         Collections
.sort(mFiles
, new Comparator
<OCFile
>() { 
 465             public int compare(OCFile o1
, OCFile o2
) { 
 466                 if (o1
.isFolder() && o2
.isFolder()) { 
 467                     Long obj1 
= getFolderSize(new File(FileStorageUtils
.getDefaultSavePathFor(mAccount
.name
, o1
))); 
 468                     return val 
* obj1
.compareTo(getFolderSize(new File(FileStorageUtils
.getDefaultSavePathFor(mAccount
.name
, o2
)))); 
 470                 else if (o1
.isFolder()) { 
 472                 } else if (o2
.isFolder()) { 
 474                 } else if (o1
.getFileLength() == 0 || o2
.getFileLength() == 0){ 
 477                     Long obj1 
= o1
.getFileLength(); 
 478                     return val 
* obj1
.compareTo(o2
.getFileLength()); 
 486      * @param sortAscending true: ascending, false: descending 
 488     private void sortByName(boolean sortAscending
){ 
 496         Collections
.sort(mFiles
, new Comparator
<OCFile
>() { 
 497             public int compare(OCFile o1
, OCFile o2
) { 
 498                 if (o1
.isFolder() && o2
.isFolder()) { 
 499                     return val 
* o1
.getRemotePath().toLowerCase().compareTo(o2
.getRemotePath().toLowerCase()); 
 500                 } else if (o1
.isFolder()) { 
 502                 } else if (o2
.isFolder()) { 
 505                 return val 
* new AlphanumComparator().compare(o1
, o2
); 
 510     public void setSortOrder(Integer order
, boolean ascending
) { 
 511         SharedPreferences
.Editor editor 
= mAppPreferences
.edit(); 
 512         editor
.putInt("sortOrder", order
); 
 513         editor
.putBoolean("sortAscending", ascending
); 
 517         mSortAscending 
= ascending
; 
 522     private CharSequence 
showRelativeTimestamp(OCFile file
){ 
 523         return DisplayUtils
.getRelativeDateTimeString(mContext
, file
.getModificationTimestamp(), 
 524                 DateUtils
.SECOND_IN_MILLIS
, DateUtils
.WEEK_IN_MILLIS
, 0);