1 /* ownCloud Android client application 
   2  *   Copyright (C) 2011  Bartek Przybylski 
   3  *   Copyright (C) 2012-2013 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/>. 
  19 package com
.owncloud
.android
.ui
.activity
; 
  23 import android
.accounts
.Account
; 
  24 import android
.app
.AlertDialog
; 
  25 import android
.app
.Dialog
; 
  26 import android
.app
.ProgressDialog
; 
  27 import android
.content
.BroadcastReceiver
; 
  28 import android
.content
.ComponentName
; 
  29 import android
.content
.ContentResolver
; 
  30 import android
.content
.Context
; 
  31 import android
.content
.DialogInterface
; 
  32 import android
.content
.Intent
; 
  33 import android
.content
.IntentFilter
; 
  34 import android
.content
.ServiceConnection
; 
  35 import android
.content
.SharedPreferences
; 
  36 import android
.content
.SyncRequest
; 
  37 import android
.content
.res
.Resources
.NotFoundException
; 
  38 import android
.database
.Cursor
; 
  39 import android
.net
.Uri
; 
  40 import android
.os
.Bundle
; 
  41 import android
.os
.Handler
; 
  42 import android
.os
.IBinder
; 
  43 import android
.preference
.PreferenceManager
; 
  44 import android
.provider
.MediaStore
; 
  45 import android
.support
.v4
.app
.Fragment
; 
  46 import android
.support
.v4
.app
.FragmentManager
; 
  47 import android
.support
.v4
.app
.FragmentTransaction
; 
  48 import android
.support
.v4
.content
.LocalBroadcastManager
; 
  49 import android
.util
.Log
; 
  50 import android
.view
.View
; 
  51 import android
.view
.ViewGroup
; 
  52 import android
.widget
.ArrayAdapter
; 
  53 import android
.widget
.TextView
; 
  54 import android
.widget
.Toast
; 
  56 import com
.actionbarsherlock
.app
.ActionBar
; 
  57 import com
.actionbarsherlock
.app
.ActionBar
.OnNavigationListener
; 
  58 import com
.actionbarsherlock
.view
.Menu
; 
  59 import com
.actionbarsherlock
.view
.MenuInflater
; 
  60 import com
.actionbarsherlock
.view
.MenuItem
; 
  61 import com
.actionbarsherlock
.view
.Window
; 
  62 import com
.owncloud
.android
.MainApp
; 
  63 import com
.owncloud
.android
.R
; 
  64 import com
.owncloud
.android
.datamodel
.FileDataStorageManager
; 
  65 import com
.owncloud
.android
.datamodel
.OCFile
; 
  66 import com
.owncloud
.android
.files
.services
.FileDownloader
; 
  67 import com
.owncloud
.android
.files
.services
.FileObserverService
; 
  68 import com
.owncloud
.android
.files
.services
.FileUploader
; 
  69 import com
.owncloud
.android
.files
.services
.FileDownloader
.FileDownloaderBinder
; 
  70 import com
.owncloud
.android
.files
.services
.FileUploader
.FileUploaderBinder
; 
  71 import com
.owncloud
.android
.operations
.CreateFolderOperation
; 
  73 import com
.owncloud
.android
.lib
.operations
.common
.OnRemoteOperationListener
; 
  74 import com
.owncloud
.android
.lib
.operations
.common
.RemoteOperation
; 
  75 import com
.owncloud
.android
.lib
.operations
.common
.RemoteOperationResult
; 
  76 import com
.owncloud
.android
.lib
.operations
.common
.RemoteOperationResult
.ResultCode
; 
  78 import com
.owncloud
.android
.operations
.RemoveFileOperation
; 
  79 import com
.owncloud
.android
.operations
.RenameFileOperation
; 
  80 import com
.owncloud
.android
.operations
.SynchronizeFileOperation
; 
  81 import com
.owncloud
.android
.operations
.SynchronizeFolderOperation
; 
  82 import com
.owncloud
.android
.services
.OperationsService
; 
  83 import com
.owncloud
.android
.syncadapter
.FileSyncAdapter
; 
  84 import com
.owncloud
.android
.ui
.dialog
.EditNameDialog
; 
  85 import com
.owncloud
.android
.ui
.dialog
.EditNameDialog
.EditNameDialogListener
; 
  86 import com
.owncloud
.android
.ui
.dialog
.LoadingDialog
; 
  87 import com
.owncloud
.android
.ui
.dialog
.SslValidatorDialog
; 
  88 import com
.owncloud
.android
.ui
.dialog
.SslValidatorDialog
.OnSslValidatorListener
; 
  89 import com
.owncloud
.android
.ui
.fragment
.FileDetailFragment
; 
  90 import com
.owncloud
.android
.ui
.fragment
.FileFragment
; 
  91 import com
.owncloud
.android
.ui
.fragment
.OCFileListFragment
; 
  92 import com
.owncloud
.android
.ui
.preview
.PreviewImageActivity
; 
  93 import com
.owncloud
.android
.ui
.preview
.PreviewMediaFragment
; 
  94 import com
.owncloud
.android
.ui
.preview
.PreviewVideoActivity
; 
  95 import com
.owncloud
.android
.utils
.DisplayUtils
; 
  96 import com
.owncloud
.android
.utils
.Log_OC
; 
 100  * Displays, what files the user has available in his ownCloud. 
 102  * @author Bartek Przybylski 
 103  * @author David A. Velasco 
 106 public class FileDisplayActivity 
extends FileActivity 
implements 
 107 OCFileListFragment
.ContainerActivity
, FileDetailFragment
.ContainerActivity
, OnNavigationListener
, OnSslValidatorListener
, OnRemoteOperationListener
, EditNameDialogListener 
{ 
 109     private ArrayAdapter
<String
> mDirectories
; 
 111     /** Access point to the cached database for the current ownCloud {@link Account} */ 
 112     private FileDataStorageManager mStorageManager 
= null
; 
 114     private SyncBroadcastReceiver mSyncBroadcastReceiver
; 
 115     private UploadFinishReceiver mUploadFinishReceiver
; 
 116     private DownloadFinishReceiver mDownloadFinishReceiver
; 
 117     private OperationsServiceReceiver mOperationsServiceReceiver
; 
 118     private FileDownloaderBinder mDownloaderBinder 
= null
; 
 119     private FileUploaderBinder mUploaderBinder 
= null
; 
 120     private ServiceConnection mDownloadConnection 
= null
, mUploadConnection 
= null
; 
 121     private RemoteOperationResult mLastSslUntrustedServerResult 
= null
; 
 123     private boolean mDualPane
; 
 124     private View mLeftFragmentContainer
; 
 125     private View mRightFragmentContainer
; 
 127     private static final String KEY_WAITING_TO_PREVIEW 
= "WAITING_TO_PREVIEW"; 
 128     private static final String KEY_SYNC_IN_PROGRESS 
= "SYNC_IN_PROGRESS"; 
 129     private static final String KEY_REFRESH_SHARES_IN_PROGRESS 
= "SHARES_IN_PROGRESS"; 
 131     public static final int DIALOG_SHORT_WAIT 
= 0; 
 132     private static final int DIALOG_CHOOSE_UPLOAD_SOURCE 
= 1; 
 133     private static final int DIALOG_SSL_VALIDATOR 
= 2; 
 134     private static final int DIALOG_CERT_NOT_SAVED 
= 3; 
 136     private static final String DIALOG_WAIT_TAG 
= "DIALOG_WAIT"; 
 138     public static final String ACTION_DETAILS 
= "com.owncloud.android.ui.activity.action.DETAILS"; 
 140     private static final int ACTION_SELECT_CONTENT_FROM_APPS 
= 1; 
 141     private static final int ACTION_SELECT_MULTIPLE_FILES 
= 2; 
 143     private static final String TAG 
= FileDisplayActivity
.class.getSimpleName(); 
 145     private static final String TAG_LIST_OF_FILES 
= "LIST_OF_FILES"; 
 146     private static final String TAG_SECOND_FRAGMENT 
= "SECOND_FRAGMENT"; 
 148     private OCFile mWaitingToPreview
; 
 149     private Handler mHandler
; 
 151     private boolean mSyncInProgress 
= false
; 
 152     private boolean mRefreshSharesInProgress 
= false
; 
 155     protected void onCreate(Bundle savedInstanceState
) { 
 156         Log_OC
.d(TAG
, "onCreate() start"); 
 157         requestWindowFeature(Window
.FEATURE_INDETERMINATE_PROGRESS
); 
 159         super.onCreate(savedInstanceState
); // this calls onAccountChanged() when ownCloud Account is valid 
 161         mHandler 
= new Handler(); 
 163         /// bindings to transference services 
 164         mUploadConnection 
= new ListServiceConnection();  
 165         mDownloadConnection 
= new ListServiceConnection(); 
 166         bindService(new Intent(this, FileUploader
.class), mUploadConnection
, Context
.BIND_AUTO_CREATE
); 
 167         bindService(new Intent(this, FileDownloader
.class), mDownloadConnection
, Context
.BIND_AUTO_CREATE
); 
 169         // PIN CODE request ;  best location is to decide, let's try this first 
 170         if (getIntent().getAction() != null 
&& getIntent().getAction().equals(Intent
.ACTION_MAIN
) && savedInstanceState 
== null
) { 
 172         } else if (getIntent().getAction() == null 
&& savedInstanceState 
== null
) { 
 177         Intent observer_intent 
= new Intent(this, FileObserverService
.class); 
 178         observer_intent
.putExtra(FileObserverService
.KEY_FILE_CMD
, FileObserverService
.CMD_INIT_OBSERVED_LIST
); 
 179         startService(observer_intent
); 
 181         /// Load of saved instance state 
 182         if(savedInstanceState 
!= null
) { 
 183             mWaitingToPreview 
= (OCFile
) savedInstanceState
.getParcelable(FileDisplayActivity
.KEY_WAITING_TO_PREVIEW
); 
 184             mSyncInProgress 
= savedInstanceState
.getBoolean(KEY_SYNC_IN_PROGRESS
); 
 185             mRefreshSharesInProgress 
= savedInstanceState
.getBoolean(KEY_REFRESH_SHARES_IN_PROGRESS
); 
 188             mWaitingToPreview 
= null
; 
 189             mSyncInProgress 
= false
; 
 190             mRefreshSharesInProgress 
= false
; 
 195         // Inflate and set the layout view 
 196         setContentView(R
.layout
.files
);     
 197         mDualPane 
= getResources().getBoolean(R
.bool
.large_land_layout
); 
 198         mLeftFragmentContainer 
= findViewById(R
.id
.left_fragment_container
); 
 199         mRightFragmentContainer 
= findViewById(R
.id
.right_fragment_container
); 
 200         if (savedInstanceState 
== null
) { 
 201             createMinFragments(); 
 205         mDirectories 
= new CustomArrayAdapter
<String
>(this, R
.layout
.sherlock_spinner_dropdown_item
); 
 206         getSupportActionBar().setHomeButtonEnabled(true
);       // mandatory since Android ICS, according to the official documentation 
 207         setSupportProgressBarIndeterminateVisibility(mSyncInProgress 
|| mRefreshSharesInProgress
);    // always AFTER setContentView(...) ; to work around bug in its implementation 
 209         Log_OC
.d(TAG
, "onCreate() end"); 
 213     protected void onStart() { 
 215         getSupportActionBar().setIcon(DisplayUtils
.getSeasonalIconId()); 
 219     protected void onDestroy() { 
 221         if (mDownloadConnection 
!= null
) 
 222             unbindService(mDownloadConnection
); 
 223         if (mUploadConnection 
!= null
) 
 224             unbindService(mUploadConnection
); 
 229      *  Called when the ownCloud {@link Account} associated to the Activity was just updated. 
 232     protected void onAccountSet(boolean stateWasRecovered
) { 
 233         if (getAccount() != null
) { 
 234             mStorageManager 
= new FileDataStorageManager(getAccount(), getContentResolver()); 
 236             /// Check whether the 'main' OCFile handled by the Activity is contained in the current Account 
 237             OCFile file 
= getFile(); 
 238             // get parent from path 
 239             String parentPath 
= ""; 
 241                 if (file
.isDown() && file
.getLastSyncDateForProperties() == 0) { 
 242                     // upload in progress - right now, files are not inserted in the local cache until the upload is successful 
 243                     // get parent from path 
 244                     parentPath 
= file
.getRemotePath().substring(0, file
.getRemotePath().lastIndexOf(file
.getFileName())); 
 245                     if (mStorageManager
.getFileByPath(parentPath
) ==  null
) 
 246                         file 
= null
; // not able to know the directory where the file is uploading 
 248                     file 
= mStorageManager
.getFileByPath(file
.getRemotePath());   // currentDir = null if not in the current Account 
 252                 // fall back to root folder 
 253                 file 
= mStorageManager
.getFileByPath(OCFile
.ROOT_PATH
);  // never returns null 
 256             setNavigationListWithFolder(file
); 
 258             if (!stateWasRecovered
) { 
 259                 Log_OC
.e(TAG
, "Initializing Fragments in onAccountChanged.."); 
 260                 initFragmentsWithFile(); 
 261                 if (file
.isFolder()) { 
 262                     startSyncFolderOperation(file
); 
 266                 updateFragmentsVisibility(!file
.isFolder()); 
 267                 updateNavigationElementsInActionBar(file
.isFolder() ? null 
: file
); 
 272             Log_OC
.wtf(TAG
, "onAccountChanged was called with NULL account associated!"); 
 277     private void setNavigationListWithFolder(OCFile file
) { 
 278         mDirectories
.clear(); 
 279         OCFile fileIt 
= file
; 
 281         while(fileIt 
!= null 
&& fileIt
.getFileName() != OCFile
.ROOT_PATH
) { 
 282             if (fileIt
.isFolder()) { 
 283                 mDirectories
.add(fileIt
.getFileName()); 
 285             //fileIt = mStorageManager.getFileById(fileIt.getParentId()); 
 286             // get parent from path 
 287             parentPath 
= fileIt
.getRemotePath().substring(0, fileIt
.getRemotePath().lastIndexOf(fileIt
.getFileName())); 
 288             fileIt 
= mStorageManager
.getFileByPath(parentPath
); 
 290         mDirectories
.add(OCFile
.PATH_SEPARATOR
); 
 294     private void createMinFragments() { 
 295         OCFileListFragment listOfFiles 
= new OCFileListFragment(); 
 296         FragmentTransaction transaction 
= getSupportFragmentManager().beginTransaction(); 
 297         transaction
.add(R
.id
.left_fragment_container
, listOfFiles
, TAG_LIST_OF_FILES
); 
 298         transaction
.commit(); 
 301     private void initFragmentsWithFile() { 
 302         if (getAccount() != null 
&& getFile() != null
) { 
 304             OCFileListFragment listOfFiles 
= getListOfFilesFragment();  
 305             if (listOfFiles 
!= null
) { 
 306                 listOfFiles
.listDirectory(getCurrentDir());    
 308                 Log
.e(TAG
, "Still have a chance to lose the initializacion of list fragment >("); 
 312             OCFile file 
= getFile();  
 313             Fragment secondFragment 
= chooseInitialSecondFragment(file
); 
 314             if (secondFragment 
!= null
) { 
 315                 setSecondFragment(secondFragment
); 
 316                 updateFragmentsVisibility(true
); 
 317                 updateNavigationElementsInActionBar(file
); 
 320                 cleanSecondFragment(); 
 324             Log
.wtf(TAG
, "initFragments() called with invalid NULLs!"); 
 325             if (getAccount() == null
) { 
 326                 Log
.wtf(TAG
, "\t account is NULL"); 
 328             if (getFile() == null
) { 
 329                 Log
.wtf(TAG
, "\t file is NULL"); 
 334     private Fragment 
chooseInitialSecondFragment(OCFile file
) { 
 335         Fragment secondFragment 
= null
; 
 336         if (file 
!= null 
&& !file
.isFolder()) { 
 337             if (file
.isDown() && PreviewMediaFragment
.canBePreviewed(file
)  
 338                     && file
.getLastSyncDateForProperties() > 0  // temporal fix 
 340                 int startPlaybackPosition 
= getIntent().getIntExtra(PreviewVideoActivity
.EXTRA_START_POSITION
, 0); 
 341                 boolean autoplay 
= getIntent().getBooleanExtra(PreviewVideoActivity
.EXTRA_AUTOPLAY
, true
); 
 342                 secondFragment 
= new PreviewMediaFragment(file
, getAccount(), startPlaybackPosition
, autoplay
); 
 345                 secondFragment 
= new FileDetailFragment(file
, getAccount()); 
 348         return secondFragment
; 
 353      * Replaces the second fragment managed by the activity with the received as 
 356      * Assumes never will be more than two fragments managed at the same time.  
 358      * @param fragment      New second Fragment to set. 
 360     private void setSecondFragment(Fragment fragment
) { 
 361         FragmentTransaction transaction 
= getSupportFragmentManager().beginTransaction(); 
 362         transaction
.replace(R
.id
.right_fragment_container
, fragment
, TAG_SECOND_FRAGMENT
); 
 363         transaction
.commit(); 
 367     private void updateFragmentsVisibility(boolean existsSecondFragment
) { 
 369             if (mLeftFragmentContainer
.getVisibility() != View
.VISIBLE
) { 
 370                 mLeftFragmentContainer
.setVisibility(View
.VISIBLE
); 
 372             if (mRightFragmentContainer
.getVisibility() != View
.VISIBLE
) { 
 373                 mRightFragmentContainer
.setVisibility(View
.VISIBLE
); 
 376         } else if (existsSecondFragment
) { 
 377             if (mLeftFragmentContainer
.getVisibility() != View
.GONE
) { 
 378                 mLeftFragmentContainer
.setVisibility(View
.GONE
); 
 380             if (mRightFragmentContainer
.getVisibility() != View
.VISIBLE
) { 
 381                 mRightFragmentContainer
.setVisibility(View
.VISIBLE
); 
 385             if (mLeftFragmentContainer
.getVisibility() != View
.VISIBLE
) { 
 386                 mLeftFragmentContainer
.setVisibility(View
.VISIBLE
); 
 388             if (mRightFragmentContainer
.getVisibility() != View
.GONE
) { 
 389                 mRightFragmentContainer
.setVisibility(View
.GONE
); 
 395     private OCFileListFragment 
getListOfFilesFragment() { 
 396         Fragment listOfFiles 
= getSupportFragmentManager().findFragmentByTag(FileDisplayActivity
.TAG_LIST_OF_FILES
); 
 397         if (listOfFiles 
!= null
) { 
 398             return (OCFileListFragment
)listOfFiles
; 
 400         Log_OC
.wtf(TAG
, "Access to unexisting list of files fragment!!"); 
 404     protected FileFragment 
getSecondFragment() { 
 405         Fragment second 
= getSupportFragmentManager().findFragmentByTag(FileDisplayActivity
.TAG_SECOND_FRAGMENT
); 
 406         if (second 
!= null
) { 
 407             return (FileFragment
)second
; 
 412     public void cleanSecondFragment() { 
 413         Fragment second 
= getSecondFragment(); 
 414         if (second 
!= null
) { 
 415             FragmentTransaction tr 
= getSupportFragmentManager().beginTransaction(); 
 419         updateFragmentsVisibility(false
); 
 420         updateNavigationElementsInActionBar(null
); 
 423     protected void refeshListOfFilesFragment() { 
 424         OCFileListFragment fileListFragment 
= getListOfFilesFragment(); 
 425         if (fileListFragment 
!= null
) {  
 426             fileListFragment
.listDirectory(); 
 430     protected void refreshSecondFragment(String downloadEvent
, String downloadedRemotePath
, boolean success
) { 
 431         FileFragment secondFragment 
= getSecondFragment(); 
 432         boolean waitedPreview 
= (mWaitingToPreview 
!= null 
&& mWaitingToPreview
.getRemotePath().equals(downloadedRemotePath
)); 
 433         if (secondFragment 
!= null 
&& secondFragment 
instanceof FileDetailFragment
) { 
 434             FileDetailFragment detailsFragment 
= (FileDetailFragment
) secondFragment
; 
 435             OCFile fileInFragment 
= detailsFragment
.getFile(); 
 436             if (fileInFragment 
!= null 
&& !downloadedRemotePath
.equals(fileInFragment
.getRemotePath())) { 
 437                 // the user browsed to other file ; forget the automatic preview  
 438                 mWaitingToPreview 
= null
; 
 440             } else if (downloadEvent
.equals(FileDownloader
.getDownloadAddedMessage())) { 
 441                 // grant that the right panel updates the progress bar 
 442                 detailsFragment
.listenForTransferProgress(); 
 443                 detailsFragment
.updateFileDetails(true
, false
); 
 445             } else if (downloadEvent
.equals(FileDownloader
.getDownloadFinishMessage())) { 
 446                 //  update the right panel 
 447                 boolean detailsFragmentChanged 
= false
; 
 450                         mWaitingToPreview 
= mStorageManager
.getFileById(mWaitingToPreview
.getFileId());   // update the file from database, for the local storage path 
 451                         if (PreviewMediaFragment
.canBePreviewed(mWaitingToPreview
)) { 
 452                             startMediaPreview(mWaitingToPreview
, 0, true
); 
 453                             detailsFragmentChanged 
= true
; 
 455                             openFile(mWaitingToPreview
); 
 458                     mWaitingToPreview 
= null
; 
 460                 if (!detailsFragmentChanged
) { 
 461                     detailsFragment
.updateFileDetails(false
, (success
)); 
 469     public boolean onCreateOptionsMenu(Menu menu
) { 
 470         MenuInflater inflater 
= getSherlock().getMenuInflater(); 
 471         inflater
.inflate(R
.menu
.main_menu
, menu
); 
 476     public boolean onOptionsItemSelected(MenuItem item
) { 
 477         boolean retval 
= true
; 
 478         switch (item
.getItemId()) { 
 479         case R
.id
.action_create_dir
: { 
 480             EditNameDialog dialog 
= EditNameDialog
.newInstance(getString(R
.string
.uploader_info_dirname
), "", -1, -1, this); 
 481             dialog
.show(getSupportFragmentManager(), "createdirdialog"); 
 484         case R
.id
.action_sync_account
: { 
 485             startSynchronization(); 
 488         case R
.id
.action_upload
: { 
 489             showDialog(DIALOG_CHOOSE_UPLOAD_SOURCE
); 
 492         case R
.id
.action_settings
: { 
 493             Intent settingsIntent 
= new Intent(this, Preferences
.class); 
 494             startActivity(settingsIntent
); 
 497         case android
.R
.id
.home
: { 
 498             FileFragment second 
= getSecondFragment(); 
 499             OCFile currentDir 
= getCurrentDir(); 
 500             if((currentDir 
!= null 
&& currentDir
.getParentId() != 0) ||  
 501                     (second 
!= null 
&& second
.getFile() != null
)) {                 
 508             retval 
= super.onOptionsItemSelected(item
); 
 513     private void startSynchronization() { 
 514         Log_OC
.e(TAG
, "Got to start sync"); 
 515         if (android
.os
.Build
.VERSION
.SDK_INT 
< android
.os
.Build
.VERSION_CODES
.KITKAT
) { 
 516             Log_OC
.e(TAG
, "Canceling all syncs for " + MainApp
.getAuthority()); 
 517             ContentResolver
.cancelSync(null
, MainApp
.getAuthority());   // cancel the current synchronizations of any ownCloud account 
 518             Bundle bundle 
= new Bundle(); 
 519             bundle
.putBoolean(ContentResolver
.SYNC_EXTRAS_MANUAL
, true
); 
 520             bundle
.putBoolean(ContentResolver
.SYNC_EXTRAS_EXPEDITED
, true
); 
 521             Log_OC
.e(TAG
, "Requesting sync for " + getAccount().name 
+ " at " + MainApp
.getAuthority()); 
 522             ContentResolver
.requestSync( 
 524                     MainApp
.getAuthority(), bundle
); 
 526             Log_OC
.e(TAG
, "Requesting sync for " + getAccount().name 
+ " at " + MainApp
.getAuthority() + " with new API"); 
 527             SyncRequest
.Builder builder 
= new SyncRequest
.Builder(); 
 528             builder
.setSyncAdapter(getAccount(), MainApp
.getAuthority()); 
 529             builder
.setExpedited(true
); 
 530             builder
.setManual(true
); 
 532             SyncRequest request 
= builder
.build(); 
 533             ContentResolver
.requestSync(request
); 
 539     public boolean onNavigationItemSelected(int itemPosition
, long itemId
) { 
 540         if (itemPosition 
!= 0) { 
 541             String targetPath 
= ""; 
 542             for (int i
=itemPosition
; i 
< mDirectories
.getCount() - 1; i
++) { 
 543                 targetPath 
= mDirectories
.getItem(i
) + OCFile
.PATH_SEPARATOR 
+ targetPath
;  
 545             targetPath 
= OCFile
.PATH_SEPARATOR 
+ targetPath
; 
 546             OCFile targetFolder 
= mStorageManager
.getFileByPath(targetPath
); 
 547             if (targetFolder 
!= null
) { 
 548                 browseTo(targetFolder
); 
 551             // the next operation triggers a new call to this method, but it's necessary to  
 552             // ensure that the name exposed in the action bar is the current directory when the  
 553             // user selected it in the navigation list 
 554             if (getSupportActionBar().getNavigationMode() == ActionBar
.NAVIGATION_MODE_LIST  
&& itemPosition 
!= 0)  
 555                 getSupportActionBar().setSelectedNavigationItem(0); 
 561      * Called, when the user selected something for uploading 
 563     protected void onActivityResult(int requestCode
, int resultCode
, Intent data
) { 
 564         super.onActivityResult(requestCode
, resultCode
, data
); 
 566         if (requestCode 
== ACTION_SELECT_CONTENT_FROM_APPS 
&& (resultCode 
== RESULT_OK 
|| resultCode 
== UploadFilesActivity
.RESULT_OK_AND_MOVE
)) { 
 567             requestSimpleUpload(data
, resultCode
); 
 569         } else if (requestCode 
== ACTION_SELECT_MULTIPLE_FILES 
&& (resultCode 
== RESULT_OK 
|| resultCode 
== UploadFilesActivity
.RESULT_OK_AND_MOVE
)) { 
 570             requestMultipleUpload(data
, resultCode
); 
 575     private void requestMultipleUpload(Intent data
, int resultCode
) { 
 576         String
[] filePaths 
= data
.getStringArrayExtra(UploadFilesActivity
.EXTRA_CHOSEN_FILES
); 
 577         if (filePaths 
!= null
) { 
 578             String
[] remotePaths 
= new String
[filePaths
.length
]; 
 579             String remotePathBase 
= ""; 
 580             for (int j 
= mDirectories
.getCount() - 2; j 
>= 0; --j
) { 
 581                 remotePathBase 
+= OCFile
.PATH_SEPARATOR 
+ mDirectories
.getItem(j
); 
 583             if (!remotePathBase
.endsWith(OCFile
.PATH_SEPARATOR
)) 
 584                 remotePathBase 
+= OCFile
.PATH_SEPARATOR
; 
 585             for (int j 
= 0; j
< remotePaths
.length
; j
++) { 
 586                 remotePaths
[j
] = remotePathBase 
+ (new File(filePaths
[j
])).getName(); 
 589             Intent i 
= new Intent(this, FileUploader
.class); 
 590             i
.putExtra(FileUploader
.KEY_ACCOUNT
, getAccount()); 
 591             i
.putExtra(FileUploader
.KEY_LOCAL_FILE
, filePaths
); 
 592             i
.putExtra(FileUploader
.KEY_REMOTE_FILE
, remotePaths
); 
 593             i
.putExtra(FileUploader
.KEY_UPLOAD_TYPE
, FileUploader
.UPLOAD_MULTIPLE_FILES
); 
 594             if (resultCode 
== UploadFilesActivity
.RESULT_OK_AND_MOVE
) 
 595                 i
.putExtra(FileUploader
.KEY_LOCAL_BEHAVIOUR
, FileUploader
.LOCAL_BEHAVIOUR_MOVE
); 
 599             Log_OC
.d(TAG
, "User clicked on 'Update' with no selection"); 
 600             Toast t 
= Toast
.makeText(this, getString(R
.string
.filedisplay_no_file_selected
), Toast
.LENGTH_LONG
); 
 607     private void requestSimpleUpload(Intent data
, int resultCode
) { 
 608         String filepath 
= null
; 
 610             Uri selectedImageUri 
= data
.getData(); 
 612             String filemanagerstring 
= selectedImageUri
.getPath(); 
 613             String selectedImagePath 
= getPath(selectedImageUri
); 
 615             if (selectedImagePath 
!= null
) 
 616                 filepath 
= selectedImagePath
; 
 618                 filepath 
= filemanagerstring
; 
 620         } catch (Exception e
) { 
 621             Log_OC
.e(TAG
, "Unexpected exception when trying to read the result of Intent.ACTION_GET_CONTENT", e
); 
 625             if (filepath 
== null
) { 
 626                 Log_OC
.e(TAG
, "Couldnt resolve path to file"); 
 627                 Toast t 
= Toast
.makeText(this, getString(R
.string
.filedisplay_unexpected_bad_get_content
), Toast
.LENGTH_LONG
); 
 633         Intent i 
= new Intent(this, FileUploader
.class); 
 634         i
.putExtra(FileUploader
.KEY_ACCOUNT
, 
 636         String remotepath 
= new String(); 
 637         for (int j 
= mDirectories
.getCount() - 2; j 
>= 0; --j
) { 
 638             remotepath 
+= OCFile
.PATH_SEPARATOR 
+ mDirectories
.getItem(j
); 
 640         if (!remotepath
.endsWith(OCFile
.PATH_SEPARATOR
)) 
 641             remotepath 
+= OCFile
.PATH_SEPARATOR
; 
 642         remotepath 
+= new File(filepath
).getName(); 
 644         i
.putExtra(FileUploader
.KEY_LOCAL_FILE
, filepath
); 
 645         i
.putExtra(FileUploader
.KEY_REMOTE_FILE
, remotepath
); 
 646         i
.putExtra(FileUploader
.KEY_UPLOAD_TYPE
, FileUploader
.UPLOAD_SINGLE_FILE
); 
 647         if (resultCode 
== UploadFilesActivity
.RESULT_OK_AND_MOVE
) 
 648             i
.putExtra(FileUploader
.KEY_LOCAL_BEHAVIOUR
, FileUploader
.LOCAL_BEHAVIOUR_MOVE
); 
 653     public void onBackPressed() { 
 654         OCFileListFragment listOfFiles 
= getListOfFilesFragment();  
 655         if (mDualPane 
|| getSecondFragment() == null
) { 
 656             if (listOfFiles 
!= null
) {  // should never be null, indeed 
 657                 if (mDirectories
.getCount() <= 1) { 
 661                 int levelsUp 
= listOfFiles
.onBrowseUp(); 
 662                 for (int i
=0; i 
< levelsUp 
&& mDirectories
.getCount() > 1 ; i
++) { 
 667         if (listOfFiles 
!= null
) {  // should never be null, indeed 
 668             setFile(listOfFiles
.getCurrentFile()); 
 670         cleanSecondFragment(); 
 675     protected void onSaveInstanceState(Bundle outState
) { 
 676         // responsibility of restore is preferred in onCreate() before than in onRestoreInstanceState when there are Fragments involved 
 677         Log_OC
.e(TAG
, "onSaveInstanceState() start"); 
 678         super.onSaveInstanceState(outState
); 
 679         outState
.putParcelable(FileDisplayActivity
.KEY_WAITING_TO_PREVIEW
, mWaitingToPreview
); 
 680         outState
.putBoolean(FileDisplayActivity
.KEY_SYNC_IN_PROGRESS
, mSyncInProgress
); 
 681         outState
.putBoolean(FileDisplayActivity
.KEY_REFRESH_SHARES_IN_PROGRESS
, mRefreshSharesInProgress
); 
 683         Log_OC
.d(TAG
, "onSaveInstanceState() end"); 
 689     protected void onResume() { 
 691         Log_OC
.e(TAG
, "onResume() start"); 
 693         // Listen for sync messages 
 694         IntentFilter syncIntentFilter 
= new IntentFilter(FileSyncAdapter
.EVENT_FULL_SYNC_START
); 
 695         syncIntentFilter
.addAction(FileSyncAdapter
.EVENT_FULL_SYNC_END
); 
 696         syncIntentFilter
.addAction(FileSyncAdapter
.EVENT_FOLDER_SIZE_SYNCED
); 
 697         syncIntentFilter
.addAction(FileSyncAdapter
.EVENT_FOLDER_CONTENTS_SYNCED
); 
 698         syncIntentFilter
.addAction(SynchronizeFolderOperation
.EVENT_SINGLE_FOLDER_SYNCED
); 
 699         mSyncBroadcastReceiver 
= new SyncBroadcastReceiver(); 
 700         //registerReceiver(mSyncBroadcastReceiver, syncIntentFilter); 
 701         LocalBroadcastManager
.getInstance(this).registerReceiver(mSyncBroadcastReceiver
, syncIntentFilter
); 
 703         // Listen for upload messages 
 704         IntentFilter uploadIntentFilter 
= new IntentFilter(FileUploader
.getUploadFinishMessage()); 
 705         mUploadFinishReceiver 
= new UploadFinishReceiver(); 
 706         registerReceiver(mUploadFinishReceiver
, uploadIntentFilter
); 
 708         // Listen for download messages 
 709         IntentFilter downloadIntentFilter 
= new IntentFilter(FileDownloader
.getDownloadAddedMessage()); 
 710         downloadIntentFilter
.addAction(FileDownloader
.getDownloadFinishMessage()); 
 711         mDownloadFinishReceiver 
= new DownloadFinishReceiver(); 
 712         registerReceiver(mDownloadFinishReceiver
, downloadIntentFilter
); 
 714         // Listen for messages from the OperationsService 
 715         IntentFilter operationsIntentFilter 
= new IntentFilter(OperationsService
.ACTION_OPERATION_ADDED
); 
 716         operationsIntentFilter
.addAction(OperationsService
.ACTION_OPERATION_FINISHED
); 
 717         mOperationsServiceReceiver 
= new OperationsServiceReceiver(); 
 718         LocalBroadcastManager
.getInstance(this).registerReceiver(mOperationsServiceReceiver
, operationsIntentFilter
); 
 720         Log_OC
.d(TAG
, "onResume() end"); 
 725     protected void onPause() { 
 727         Log_OC
.e(TAG
, "onPause() start"); 
 728         if (mSyncBroadcastReceiver 
!= null
) { 
 729             //unregisterReceiver(mSyncBroadcastReceiver); 
 730             LocalBroadcastManager
.getInstance(this).unregisterReceiver(mSyncBroadcastReceiver
); 
 731             mSyncBroadcastReceiver 
= null
; 
 733         if (mUploadFinishReceiver 
!= null
) { 
 734             unregisterReceiver(mUploadFinishReceiver
); 
 735             mUploadFinishReceiver 
= null
; 
 737         if (mDownloadFinishReceiver 
!= null
) { 
 738             unregisterReceiver(mDownloadFinishReceiver
); 
 739             mDownloadFinishReceiver 
= null
; 
 741         if (mOperationsServiceReceiver 
!= null
) { 
 742             LocalBroadcastManager
.getInstance(this).unregisterReceiver(mOperationsServiceReceiver
); 
 743             mOperationsServiceReceiver 
= null
; 
 745         Log_OC
.d(TAG
, "onPause() end"); 
 750     protected void onPrepareDialog(int id
, Dialog dialog
, Bundle args
) { 
 751         if (id 
== DIALOG_SSL_VALIDATOR 
&& mLastSslUntrustedServerResult 
!= null
) { 
 752             ((SslValidatorDialog
)dialog
).updateResult(mLastSslUntrustedServerResult
); 
 758     protected Dialog 
onCreateDialog(int id
) { 
 759         Dialog dialog 
= null
; 
 760         AlertDialog
.Builder builder
; 
 762         case DIALOG_SHORT_WAIT
: { 
 763             ProgressDialog working_dialog 
= new ProgressDialog(this); 
 764             working_dialog
.setMessage(getResources().getString( 
 765                     R
.string
.wait_a_moment
)); 
 766             working_dialog
.setIndeterminate(true
); 
 767             working_dialog
.setCancelable(false
); 
 768             dialog 
= working_dialog
; 
 771         case DIALOG_CHOOSE_UPLOAD_SOURCE
: { 
 773             String
[] items 
= null
; 
 775             String
[] allTheItems 
= { getString(R
.string
.actionbar_upload_files
), 
 776                     getString(R
.string
.actionbar_upload_from_apps
), 
 777                     getString(R
.string
.actionbar_failed_instant_upload
) }; 
 779             String
[] commonItems 
= { getString(R
.string
.actionbar_upload_files
), 
 780                     getString(R
.string
.actionbar_upload_from_apps
) }; 
 782             if (InstantUploadActivity
.IS_ENABLED
) 
 787             builder 
= new AlertDialog
.Builder(this); 
 788             builder
.setTitle(R
.string
.actionbar_upload
); 
 789             builder
.setItems(items
, new DialogInterface
.OnClickListener() { 
 790                 public void onClick(DialogInterface dialog
, int item
) { 
 793                             Intent action 
= new Intent(FileDisplayActivity
.this, UploadFilesActivity
.class); 
 794                             action
.putExtra(UploadFilesActivity
.EXTRA_ACCOUNT
, FileDisplayActivity
.this.getAccount()); 
 795                             startActivityForResult(action
, ACTION_SELECT_MULTIPLE_FILES
); 
 797                             // TODO create and handle new fragment 
 798                             // LocalFileListFragment 
 800                     } else if (item 
== 1) { 
 801                         Intent action 
= new Intent(Intent
.ACTION_GET_CONTENT
); 
 802                         action 
= action
.setType("*/*").addCategory(Intent
.CATEGORY_OPENABLE
); 
 803                         startActivityForResult(Intent
.createChooser(action
, getString(R
.string
.upload_chooser_title
)), 
 804                                 ACTION_SELECT_CONTENT_FROM_APPS
); 
 805                     } else if (item 
== 2 && InstantUploadActivity
.IS_ENABLED
) { 
 806                         Intent action 
= new Intent(FileDisplayActivity
.this, InstantUploadActivity
.class); 
 807                         action
.putExtra(FileUploader
.KEY_ACCOUNT
, FileDisplayActivity
.this.getAccount()); 
 808                         startActivity(action
); 
 812             dialog 
= builder
.create(); 
 815         case DIALOG_SSL_VALIDATOR
: { 
 816             dialog 
= SslValidatorDialog
.newInstance(this, mLastSslUntrustedServerResult
, this); 
 819         case DIALOG_CERT_NOT_SAVED
: { 
 820             builder 
= new AlertDialog
.Builder(this); 
 821             builder
.setMessage(getResources().getString(R
.string
.ssl_validator_not_saved
)); 
 822             builder
.setCancelable(false
); 
 823             builder
.setPositiveButton(R
.string
.common_ok
, new DialogInterface
.OnClickListener() { 
 825                 public void onClick(DialogInterface dialog
, int which
) { 
 829             dialog 
= builder
.create(); 
 841      * Show loading dialog  
 843     public void showLoadingDialog() { 
 845         LoadingDialog loading 
= new LoadingDialog(getResources().getString(R
.string
.wait_a_moment
)); 
 846         FragmentManager fm 
= getSupportFragmentManager(); 
 847         FragmentTransaction ft 
= fm
.beginTransaction(); 
 848         loading
.show(ft
, DIALOG_WAIT_TAG
); 
 853      * Dismiss loading dialog 
 855     public void dismissLoadingDialog(){ 
 856         Fragment frag 
= getSupportFragmentManager().findFragmentByTag(DIALOG_WAIT_TAG
); 
 858           LoadingDialog loading 
= (LoadingDialog
) frag
; 
 865      * Translates a content URI of an image to a physical path 
 867      * @param uri The URI to resolve 
 868      * @return The path to the image or null if it could not be found 
 870     public String 
getPath(Uri uri
) { 
 871         String
[] projection 
= { MediaStore
.Images
.Media
.DATA 
}; 
 872         Cursor cursor 
= managedQuery(uri
, projection
, null
, null
, null
); 
 873         if (cursor 
!= null
) { 
 874             int column_index 
= cursor
 
 875                     .getColumnIndexOrThrow(MediaStore
.Images
.Media
.DATA
); 
 876             cursor
.moveToFirst(); 
 877             return cursor
.getString(column_index
); 
 883      * Pushes a directory to the drop down list 
 884      * @param directory to push 
 885      * @throws IllegalArgumentException If the {@link OCFile#isFolder()} returns false. 
 887     public void pushDirname(OCFile directory
) { 
 888         if(!directory
.isFolder()){ 
 889             throw new IllegalArgumentException("Only directories may be pushed!"); 
 891         mDirectories
.insert(directory
.getFileName(), 0); 
 896      * Pops a directory name from the drop down list 
 897      * @return True, unless the stack is empty 
 899     public boolean popDirname() { 
 900         mDirectories
.remove(mDirectories
.getItem(0)); 
 901         return !mDirectories
.isEmpty(); 
 904     // Custom array adapter to override text colors 
 905     private class CustomArrayAdapter
<T
> extends ArrayAdapter
<T
> { 
 907         public CustomArrayAdapter(FileDisplayActivity ctx
, int view
) { 
 911         public View 
getView(int position
, View convertView
, ViewGroup parent
) { 
 912             View v 
= super.getView(position
, convertView
, parent
); 
 914             ((TextView
) v
).setTextColor(getResources().getColorStateList( 
 915                     android
.R
.color
.white
)); 
 917             fixRoot((TextView
) v 
); 
 921         public View 
getDropDownView(int position
, View convertView
, 
 923             View v 
= super.getDropDownView(position
, convertView
, parent
); 
 925             ((TextView
) v
).setTextColor(getResources().getColorStateList( 
 926                     android
.R
.color
.white
)); 
 928             fixRoot((TextView
) v 
); 
 932         private void fixRoot(TextView v
) { 
 933             if (v
.getText().equals(OCFile
.PATH_SEPARATOR
)) { 
 934                 v
.setText(R
.string
.default_display_name_for_root_folder
); 
 940     private class SyncBroadcastReceiver 
extends BroadcastReceiver 
{ 
 943          * {@link BroadcastReceiver} to enable syncing feedback in UI 
 946         public void onReceive(Context context
, Intent intent
) { 
 947             String event 
= intent
.getAction(); 
 948             String accountName 
= intent
.getStringExtra(FileSyncAdapter
.EXTRA_ACCOUNT_NAME
); 
 949             String synchFolderRemotePath 
= intent
.getStringExtra(FileSyncAdapter
.EXTRA_FOLDER_PATH
);  
 950             RemoteOperationResult synchResult 
= (RemoteOperationResult
)intent
.getSerializableExtra(FileSyncAdapter
.EXTRA_RESULT
); 
 951             boolean sameAccount 
= (getAccount() != null 
&& accountName
.equals(getAccount().name
) && mStorageManager 
!= null
);  
 955                 if (!FileSyncAdapter
.EVENT_FULL_SYNC_START
.equals(event
)) { 
 957                     OCFile currentFile 
= (getFile() == null
) ? null 
: mStorageManager
.getFileByPath(getFile().getRemotePath()); 
 958                     OCFile currentDir 
= (getCurrentDir() == null
) ? null 
: mStorageManager
.getFileByPath(getCurrentDir().getRemotePath()); 
 960                     if (currentDir 
== null
) { 
 961                         // current folder was removed from the server  
 962                         Toast
.makeText( FileDisplayActivity
.this,  
 963                                         String
.format(getString(R
.string
.sync_current_folder_was_removed
), mDirectories
.getItem(0)),  
 969                         if (currentFile 
== null 
&& !getFile().isFolder()) { 
 970                             // currently selected file was removed in the server, and now we know it 
 971                             cleanSecondFragment(); 
 972                             currentFile 
= currentDir
; 
 975                         if (synchFolderRemotePath 
!= null 
&& currentDir
.getRemotePath().equals(synchFolderRemotePath
)) { 
 976                             OCFileListFragment fileListFragment 
= getListOfFilesFragment(); 
 977                             if (fileListFragment 
!= null
) { 
 978                                 fileListFragment
.listDirectory(currentDir
); 
 981                         setFile(currentFile
); 
 984                     mSyncInProgress 
= (!FileSyncAdapter
.EVENT_FULL_SYNC_END
.equals(event
) &&  
 985                                         !SynchronizeFolderOperation
.EVENT_SINGLE_FOLDER_SYNCED
.equals(event
) && 
 986                                         (synchResult 
== null 
|| synchResult
.isSuccess())) ; 
 988                     if (synchResult 
!= null 
&&  
 989                         synchResult
.isSuccess() && 
 990                             (SynchronizeFolderOperation
.EVENT_SINGLE_FOLDER_SYNCED
.equals(event
) ||  
 991                                 FileSyncAdapter
.EVENT_FOLDER_CONTENTS_SYNCED
.equals(event
) 
 998                 //removeStickyBroadcast(intent); 
 999                 setSupportProgressBarIndeterminateVisibility(mSyncInProgress 
|| mRefreshSharesInProgress
); 
1002             if (synchResult 
!= null
) { 
1003                 if (synchResult
.getCode().equals(RemoteOperationResult
.ResultCode
.SSL_RECOVERABLE_PEER_UNVERIFIED
)) { 
1004                     mLastSslUntrustedServerResult 
= synchResult
; 
1005                     showDialog(DIALOG_SSL_VALIDATOR
);  
1012     private class UploadFinishReceiver 
extends BroadcastReceiver 
{ 
1014          * Once the file upload has finished -> update view 
1015          *  @author David A. Velasco 
1016          * {@link BroadcastReceiver} to enable upload feedback in UI 
1019         public void onReceive(Context context
, Intent intent
) { 
1020             String uploadedRemotePath 
= intent
.getStringExtra(FileDownloader
.EXTRA_REMOTE_PATH
); 
1021             String accountName 
= intent
.getStringExtra(FileUploader
.ACCOUNT_NAME
); 
1022             boolean sameAccount 
= getAccount() != null 
&& accountName
.equals(getAccount().name
); 
1023             OCFile currentDir 
= getCurrentDir(); 
1024             boolean isDescendant 
= (currentDir 
!= null
) && (uploadedRemotePath 
!= null
) && (uploadedRemotePath
.startsWith(currentDir
.getRemotePath())); 
1025             if (sameAccount 
&& isDescendant
) { 
1026                 refeshListOfFilesFragment(); 
1034      * Class waiting for broadcast events from the {@link FielDownloader} service. 
1036      * Updates the UI when a download is started or finished, provided that it is relevant for the 
1039     private class DownloadFinishReceiver 
extends BroadcastReceiver 
{ 
1041         public void onReceive(Context context
, Intent intent
) { 
1042             boolean sameAccount 
= isSameAccount(context
, intent
); 
1043             String downloadedRemotePath 
= intent
.getStringExtra(FileDownloader
.EXTRA_REMOTE_PATH
); 
1044             boolean isDescendant 
= isDescendant(downloadedRemotePath
); 
1046             if (sameAccount 
&& isDescendant
) { 
1047                 refeshListOfFilesFragment(); 
1048                 refreshSecondFragment(intent
.getAction(), downloadedRemotePath
, intent
.getBooleanExtra(FileDownloader
.EXTRA_DOWNLOAD_RESULT
, false
)); 
1051             removeStickyBroadcast(intent
); 
1054         private boolean isDescendant(String downloadedRemotePath
) { 
1055             OCFile currentDir 
= getCurrentDir(); 
1056             return (currentDir 
!= null 
&& downloadedRemotePath 
!= null 
&& downloadedRemotePath
.startsWith(currentDir
.getRemotePath())); 
1059         private boolean isSameAccount(Context context
, Intent intent
) { 
1060             String accountName 
= intent
.getStringExtra(FileDownloader
.ACCOUNT_NAME
); 
1061             return (accountName 
!= null 
&& getAccount() != null 
&& accountName
.equals(getAccount().name
)); 
1067      * Class waiting for broadcast events from the {@link OperationsService}. 
1069      * Updates the list of files when a get for shares is finished; at this moment the refresh of shares is the only 
1070      * operation performed in {@link OperationsService}. 
1072      * In the future will handle the progress or finalization of all the operations performed in {@link OperationsService},  
1073      * probably all the operations associated to the app model.  
1075     private class OperationsServiceReceiver 
extends BroadcastReceiver 
{ 
1078         public void onReceive(Context context
, Intent intent
) { 
1079             if (OperationsService
.ACTION_OPERATION_ADDED
.equals(intent
.getAction())) { 
1081             } else if (OperationsService
.ACTION_OPERATION_FINISHED
.equals(intent
.getAction())) { 
1082                 mRefreshSharesInProgress 
= false
; 
1084                 Account account 
= intent
.getParcelableExtra(OperationsService
.EXTRA_ACCOUNT
); 
1085                 RemoteOperationResult getSharesResult 
= (RemoteOperationResult
)intent
.getSerializableExtra(OperationsService
.EXTRA_RESULT
); 
1086                 if (getAccount() != null 
&& account
.name
.equals(getAccount().name
) 
1087                         && mStorageManager 
!= null
 
1089                     refeshListOfFilesFragment(); 
1091                 if ((getSharesResult 
!= null
) && 
1092                         RemoteOperationResult
.ResultCode
.SSL_RECOVERABLE_PEER_UNVERIFIED
.equals(getSharesResult
.getCode())) { 
1093                     mLastSslUntrustedServerResult 
= getSharesResult
; 
1094                     showDialog(DIALOG_SSL_VALIDATOR
);  
1097                 setSupportProgressBarIndeterminateVisibility(mRefreshSharesInProgress 
|| mSyncInProgress
); 
1109     public FileDataStorageManager 
getStorageManager() { 
1110         return mStorageManager
; 
1114     public void browseToRoot() { 
1115         OCFileListFragment listOfFiles 
= getListOfFilesFragment();  
1116         if (listOfFiles 
!= null
) {  // should never be null, indeed 
1117             while (mDirectories
.getCount() > 1) { 
1120             OCFile root 
= mStorageManager
.getFileByPath(OCFile
.ROOT_PATH
); 
1121             listOfFiles
.listDirectory(root
); 
1122             setFile(listOfFiles
.getCurrentFile()); 
1123             startSyncFolderOperation(root
); 
1125         cleanSecondFragment(); 
1129     public void browseTo(OCFile folder
) { 
1130         if (folder 
== null 
|| !folder
.isFolder()) { 
1131             throw new IllegalArgumentException("Trying to browse to invalid folder " + folder
); 
1133         OCFileListFragment listOfFiles 
= getListOfFilesFragment();  
1134         if (listOfFiles 
!= null
) { 
1135             setNavigationListWithFolder(folder
); 
1136             listOfFiles
.listDirectory(folder
); 
1137             setFile(listOfFiles
.getCurrentFile()); 
1138             startSyncFolderOperation(folder
); 
1140             Log_OC
.e(TAG
, "Unexpected null when accessing list fragment"); 
1142         cleanSecondFragment(); 
1149      * Updates action bar and second fragment, if in dual pane mode. 
1152     public void onBrowsedDownTo(OCFile directory
) { 
1153         pushDirname(directory
); 
1154         cleanSecondFragment(); 
1157         startSyncFolderOperation(directory
); 
1162      * Opens the image gallery showing the image {@link OCFile} received as parameter. 
1164      * @param file                      Image {@link OCFile} to show. 
1167     public void startImagePreview(OCFile file
) { 
1168         Intent showDetailsIntent 
= new Intent(this, PreviewImageActivity
.class); 
1169         showDetailsIntent
.putExtra(EXTRA_FILE
, file
); 
1170         showDetailsIntent
.putExtra(EXTRA_ACCOUNT
, getAccount()); 
1171         startActivity(showDetailsIntent
); 
1175      * Stars the preview of an already down media {@link OCFile}. 
1177      * @param file                      Media {@link OCFile} to preview. 
1178      * @param startPlaybackPosition     Media position where the playback will be started, in milliseconds. 
1179      * @param autoplay                  When 'true', the playback will start without user interactions. 
1182     public void startMediaPreview(OCFile file
, int startPlaybackPosition
, boolean autoplay
) { 
1183         Fragment mediaFragment 
= new PreviewMediaFragment(file
, getAccount(), startPlaybackPosition
, autoplay
); 
1184         setSecondFragment(mediaFragment
); 
1185         updateFragmentsVisibility(true
); 
1186         updateNavigationElementsInActionBar(file
); 
1191      * Requests the download of the received {@link OCFile} , updates the UI 
1192      * to monitor the download progress and prepares the activity to preview 
1193      * or open the file when the download finishes. 
1195      * @param file          {@link OCFile} to download and preview. 
1198     public void startDownloadForPreview(OCFile file
) { 
1199         Fragment detailFragment 
= new FileDetailFragment(file
, getAccount()); 
1200         setSecondFragment(detailFragment
); 
1201         mWaitingToPreview 
= file
; 
1202         requestForDownload(); 
1203         updateFragmentsVisibility(true
); 
1204         updateNavigationElementsInActionBar(file
); 
1210      * Shows the information of the {@link OCFile} received as a  
1211      * parameter in the second fragment. 
1213      * @param file          {@link OCFile} whose details will be shown 
1216     public void showDetails(OCFile file
) { 
1217         Fragment detailFragment 
= new FileDetailFragment(file
, getAccount()); 
1218         setSecondFragment(detailFragment
); 
1219         updateFragmentsVisibility(true
); 
1220         updateNavigationElementsInActionBar(file
); 
1228     private void updateNavigationElementsInActionBar(OCFile chosenFile
) { 
1229         ActionBar actionBar 
= getSupportActionBar();  
1230         if (chosenFile 
== null 
|| mDualPane
) { 
1231             // only list of files - set for browsing through folders 
1232             OCFile currentDir 
= getCurrentDir(); 
1233             boolean noRoot 
= (currentDir 
!= null 
&& currentDir
.getParentId() != 0); 
1234             actionBar
.setDisplayHomeAsUpEnabled(noRoot
); 
1235             actionBar
.setDisplayShowTitleEnabled(!noRoot
);  
1237                 actionBar
.setTitle(getString(R
.string
.default_display_name_for_root_folder
)); 
1239             actionBar
.setNavigationMode(!noRoot ? ActionBar
.NAVIGATION_MODE_STANDARD 
: ActionBar
.NAVIGATION_MODE_LIST
); 
1240             actionBar
.setListNavigationCallbacks(mDirectories
, this);   // assuming mDirectories is updated 
1243             actionBar
.setDisplayHomeAsUpEnabled(true
); 
1244             actionBar
.setDisplayShowTitleEnabled(true
); 
1245             actionBar
.setTitle(chosenFile
.getFileName()); 
1246             actionBar
.setNavigationMode(ActionBar
.NAVIGATION_MODE_STANDARD
); 
1255     public void onFileStateChanged() { 
1256         refeshListOfFilesFragment(); 
1257         updateNavigationElementsInActionBar(getSecondFragment().getFile()); 
1265     public FileDownloaderBinder 
getFileDownloaderBinder() { 
1266         return mDownloaderBinder
; 
1274     public FileUploaderBinder 
getFileUploaderBinder() { 
1275         return mUploaderBinder
; 
1279     /** Defines callbacks for service binding, passed to bindService() */ 
1280     private class ListServiceConnection 
implements ServiceConnection 
{ 
1283         public void onServiceConnected(ComponentName component
, IBinder service
) { 
1284             if (component
.equals(new ComponentName(FileDisplayActivity
.this, FileDownloader
.class))) { 
1285                 Log_OC
.d(TAG
, "Download service connected"); 
1286                 mDownloaderBinder 
= (FileDownloaderBinder
) service
; 
1287                 if (mWaitingToPreview 
!= null
) { 
1288                     requestForDownload(); 
1291             } else if (component
.equals(new ComponentName(FileDisplayActivity
.this, FileUploader
.class))) { 
1292                 Log_OC
.d(TAG
, "Upload service connected"); 
1293                 mUploaderBinder 
= (FileUploaderBinder
) service
; 
1297             // a new chance to get the mDownloadBinder through getFileDownloadBinder() - THIS IS A MESS 
1298             OCFileListFragment listOfFiles 
= getListOfFilesFragment();  
1299             if (listOfFiles 
!= null
) { 
1300                 listOfFiles
.listDirectory(); 
1302             FileFragment secondFragment 
= getSecondFragment(); 
1303             if (secondFragment 
!= null 
&& secondFragment 
instanceof FileDetailFragment
) { 
1304                 FileDetailFragment detailFragment 
= (FileDetailFragment
)secondFragment
; 
1305                 detailFragment
.listenForTransferProgress(); 
1306                 detailFragment
.updateFileDetails(false
, false
); 
1311         public void onServiceDisconnected(ComponentName component
) { 
1312             if (component
.equals(new ComponentName(FileDisplayActivity
.this, FileDownloader
.class))) { 
1313                 Log_OC
.d(TAG
, "Download service disconnected"); 
1314                 mDownloaderBinder 
= null
; 
1315             } else if (component
.equals(new ComponentName(FileDisplayActivity
.this, FileUploader
.class))) { 
1316                 Log_OC
.d(TAG
, "Upload service disconnected"); 
1317                 mUploaderBinder 
= null
; 
1325      * Launch an intent to request the PIN code to the user before letting him use the app 
1327     private void requestPinCode() { 
1328         boolean pinStart 
= false
; 
1329         SharedPreferences appPrefs 
= PreferenceManager
.getDefaultSharedPreferences(getApplicationContext()); 
1330         pinStart 
= appPrefs
.getBoolean("set_pincode", false
); 
1332             Intent i 
= new Intent(getApplicationContext(), PinCodeActivity
.class); 
1333             i
.putExtra(PinCodeActivity
.EXTRA_ACTIVITY
, "FileDisplayActivity"); 
1340     public void onSavedCertificate() { 
1341         startSyncFolderOperation(getCurrentDir());                 
1346     public void onFailedSavingCertificate() { 
1347         showDialog(DIALOG_CERT_NOT_SAVED
); 
1352      * Updates the view associated to the activity after the finish of some operation over files 
1353      * in the current account. 
1355      * @param operation     Removal operation performed. 
1356      * @param result        Result of the removal. 
1359     public void onRemoteOperationFinish(RemoteOperation operation
, RemoteOperationResult result
) { 
1360         if (operation 
instanceof RemoveFileOperation
) { 
1361             onRemoveFileOperationFinish((RemoveFileOperation
)operation
, result
); 
1363         } else if (operation 
instanceof RenameFileOperation
) { 
1364             onRenameFileOperationFinish((RenameFileOperation
)operation
, result
); 
1366         } else if (operation 
instanceof SynchronizeFileOperation
) { 
1367             onSynchronizeFileOperationFinish((SynchronizeFileOperation
)operation
, result
); 
1369         } else if (operation 
instanceof CreateFolderOperation
) { 
1370             onCreateFolderOperationFinish((CreateFolderOperation
)operation
, result
); 
1376      * Updates the view associated to the activity after the finish of an operation trying to remove a  
1379      * @param operation     Removal operation performed. 
1380      * @param result        Result of the removal. 
1382     private void onRemoveFileOperationFinish(RemoveFileOperation operation
, RemoteOperationResult result
) { 
1383         dismissLoadingDialog(); 
1384         if (result
.isSuccess()) { 
1385             Toast msg 
= Toast
.makeText(this, R
.string
.remove_success_msg
, Toast
.LENGTH_LONG
); 
1387             OCFile removedFile 
= operation
.getFile(); 
1388             getSecondFragment(); 
1389             FileFragment second 
= getSecondFragment(); 
1390             if (second 
!= null 
&& removedFile
.equals(second
.getFile())) { 
1391                 cleanSecondFragment(); 
1393             if (mStorageManager
.getFileById(removedFile
.getParentId()).equals(getCurrentDir())) { 
1394                 refeshListOfFilesFragment(); 
1398             Toast msg 
= Toast
.makeText(this, R
.string
.remove_fail_msg
, Toast
.LENGTH_LONG
);  
1400             if (result
.isSslRecoverableException()) { 
1401                 mLastSslUntrustedServerResult 
= result
; 
1402                 showDialog(DIALOG_SSL_VALIDATOR
);  
1408      * Updates the view associated to the activity after the finish of an operation trying create a new folder 
1410      * @param operation     Creation operation performed. 
1411      * @param result        Result of the creation. 
1413     private void onCreateFolderOperationFinish(CreateFolderOperation operation
, RemoteOperationResult result
) { 
1414         if (result
.isSuccess()) { 
1415             dismissLoadingDialog(); 
1416             refeshListOfFilesFragment(); 
1419             dismissLoadingDialog(); 
1420             if (result
.getCode() == ResultCode
.INVALID_CHARACTER_IN_NAME
) { 
1421                 Toast
.makeText(FileDisplayActivity
.this, R
.string
.filename_forbidden_characters
, Toast
.LENGTH_LONG
).show(); 
1424                 Toast msg 
= Toast
.makeText(FileDisplayActivity
.this, R
.string
.create_dir_fail_msg
, Toast
.LENGTH_LONG
);  
1427             } catch (NotFoundException e
) { 
1428                 Log_OC
.e(TAG
, "Error while trying to show fail message " , e
); 
1436      * Updates the view associated to the activity after the finish of an operation trying to rename a  
1439      * @param operation     Renaming operation performed. 
1440      * @param result        Result of the renaming. 
1442     private void onRenameFileOperationFinish(RenameFileOperation operation
, RemoteOperationResult result
) { 
1443         dismissLoadingDialog(); 
1444         OCFile renamedFile 
= operation
.getFile(); 
1445         if (result
.isSuccess()) { 
1447                 FileFragment details 
= getSecondFragment(); 
1448                 if (details 
!= null 
&& details 
instanceof FileDetailFragment 
&& renamedFile
.equals(details
.getFile()) ) { 
1449                     ((FileDetailFragment
) details
).updateFileDetails(renamedFile
, getAccount()); 
1452             if (mStorageManager
.getFileById(renamedFile
.getParentId()).equals(getCurrentDir())) { 
1453                 refeshListOfFilesFragment(); 
1457             if (result
.getCode().equals(ResultCode
.INVALID_LOCAL_FILE_NAME
)) { 
1458                 Toast msg 
= Toast
.makeText(this, R
.string
.rename_local_fail_msg
, Toast
.LENGTH_LONG
);  
1460                 // TODO throw again the new rename dialog 
1461             } if (result
.getCode().equals(ResultCode
.INVALID_CHARACTER_IN_NAME
)) { 
1462                 Toast msg 
= Toast
.makeText(this, R
.string
.filename_forbidden_characters
, Toast
.LENGTH_LONG
);  
1465                 Toast msg 
= Toast
.makeText(this, R
.string
.rename_server_fail_msg
, Toast
.LENGTH_LONG
);  
1467                 if (result
.isSslRecoverableException()) { 
1468                     mLastSslUntrustedServerResult 
= result
; 
1469                     showDialog(DIALOG_SSL_VALIDATOR
);  
1476     private void onSynchronizeFileOperationFinish(SynchronizeFileOperation operation
, RemoteOperationResult result
) { 
1477         dismissLoadingDialog(); 
1478         OCFile syncedFile 
= operation
.getLocalFile(); 
1479         if (!result
.isSuccess()) { 
1480             if (result
.getCode() == ResultCode
.SYNC_CONFLICT
) { 
1481                 Intent i 
= new Intent(this, ConflictsResolveActivity
.class); 
1482                 i
.putExtra(ConflictsResolveActivity
.EXTRA_FILE
, syncedFile
); 
1483                 i
.putExtra(ConflictsResolveActivity
.EXTRA_ACCOUNT
, getAccount()); 
1489             if (operation
.transferWasRequested()) { 
1490                 refeshListOfFilesFragment(); 
1491                 onTransferStateChanged(syncedFile
, true
, true
); 
1494                 Toast msg 
= Toast
.makeText(this, R
.string
.sync_file_nothing_to_do_msg
, Toast
.LENGTH_LONG
);  
1505     public void onTransferStateChanged(OCFile file
, boolean downloading
, boolean uploading
) { 
1507             FileFragment details 
= getSecondFragment(); 
1508             if (details 
!= null 
&& details 
instanceof FileDetailFragment 
&& file
.equals(details
.getFile()) ) { 
1509                 if (downloading 
|| uploading
) { 
1510                     ((FileDetailFragment
)details
).updateFileDetails(file
, getAccount()); 
1512                     ((FileDetailFragment
)details
).updateFileDetails(false
, true
); 
1519     public void onDismiss(EditNameDialog dialog
) { 
1520         if (dialog
.getResult()) { 
1521             String newDirectoryName 
= dialog
.getNewFilename().trim(); 
1522             Log_OC
.d(TAG
, "'create directory' dialog dismissed with new name " + newDirectoryName
); 
1523             if (newDirectoryName
.length() > 0) { 
1524                 String path 
= getCurrentDir().getRemotePath(); 
1527                 path 
+= newDirectoryName 
+ OCFile
.PATH_SEPARATOR
; 
1528                 RemoteOperation operation 
= new CreateFolderOperation(path
, false
, mStorageManager
); 
1529                 operation
.execute(  getAccount(),  
1530                         FileDisplayActivity
.this,  
1531                         FileDisplayActivity
.this,  
1533                         FileDisplayActivity
.this); 
1535                 showLoadingDialog(); 
1541     private void requestForDownload() { 
1542         Account account 
= getAccount(); 
1543         if (!mDownloaderBinder
.isDownloading(account
, mWaitingToPreview
)) { 
1544             Intent i 
= new Intent(this, FileDownloader
.class); 
1545             i
.putExtra(FileDownloader
.EXTRA_ACCOUNT
, account
); 
1546             i
.putExtra(FileDownloader
.EXTRA_FILE
, mWaitingToPreview
); 
1552     private OCFile 
getCurrentDir() { 
1553         OCFile file 
= getFile(); 
1555             if (file
.isFolder()) { 
1557             } else if (mStorageManager 
!= null
) { 
1558                 String parentPath 
= file
.getRemotePath().substring(0, file
.getRemotePath().lastIndexOf(file
.getFileName())); 
1559                 return mStorageManager
.getFileByPath(parentPath
); 
1565     public void startSyncFolderOperation(OCFile folder
) { 
1566         long currentSyncTime 
= System
.currentTimeMillis();  
1568         mSyncInProgress 
= true
; 
1570         // perform folder synchronization 
1571         RemoteOperation synchFolderOp 
= new SynchronizeFolderOperation( folder
,   
1574                                                                         getStorageManager(),  
1576                                                                         getApplicationContext() 
1578         synchFolderOp
.execute(getAccount(), this, null
, null
, this); 
1580         setSupportProgressBarIndeterminateVisibility(true
); 
1584     private void startGetShares() { 
1585         // Get shared files/folders 
1586         Intent intent 
= new Intent(this, OperationsService
.class); 
1587         intent
.putExtra(OperationsService
.EXTRA_ACCOUNT
, getAccount()); 
1588         startService(intent
); 
1590         mRefreshSharesInProgress 
= true
;