1 /* ownCloud Android client application 
   2  *   Copyright (C) 2012-2014 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/>. 
  18 package com
.owncloud
.android
.ui
.activity
; 
  20 import java
.io
.IOException
; 
  22 import android
.accounts
.Account
; 
  23 import android
.accounts
.AccountManager
; 
  24 import android
.accounts
.AuthenticatorException
; 
  25 import android
.accounts
.OperationCanceledException
; 
  26 import android
.content
.BroadcastReceiver
; 
  27 import android
.content
.Context
; 
  28 import android
.content
.Intent
; 
  29 import android
.content
.IntentFilter
; 
  30 import android
.content
.res
.Resources
.NotFoundException
; 
  31 import android
.os
.Bundle
; 
  32 import android
.support
.v4
.app
.Fragment
; 
  33 import android
.support
.v4
.app
.FragmentTransaction
; 
  34 import android
.support
.v4
.widget
.SwipeRefreshLayout
; 
  35 import android
.util
.Log
; 
  36 import android
.view
.View
; 
  37 import android
.view
.View
.OnClickListener
; 
  38 import android
.widget
.Button
; 
  39 import android
.widget
.Toast
; 
  41 import com
.actionbarsherlock
.app
.ActionBar
; 
  42 import com
.actionbarsherlock
.view
.Menu
; 
  43 import com
.actionbarsherlock
.view
.MenuInflater
; 
  44 import com
.actionbarsherlock
.view
.MenuItem
; 
  45 import com
.actionbarsherlock
.view
.Window
; 
  46 import com
.owncloud
.android
.R
; 
  47 import com
.owncloud
.android
.datamodel
.OCFile
; 
  48 import com
.owncloud
.android
.lib
.common
.OwnCloudAccount
; 
  49 import com
.owncloud
.android
.lib
.common
.OwnCloudClient
; 
  50 import com
.owncloud
.android
.lib
.common
.OwnCloudClientManagerFactory
; 
  51 import com
.owncloud
.android
.lib
.common
.OwnCloudCredentials
; 
  52 import com
.owncloud
.android
.lib
.common
.accounts
.AccountUtils
.AccountNotFoundException
; 
  53 import com
.owncloud
.android
.lib
.common
.operations
.RemoteOperation
; 
  54 import com
.owncloud
.android
.lib
.common
.operations
.RemoteOperationResult
; 
  55 import com
.owncloud
.android
.lib
.common
.operations
.RemoteOperationResult
.ResultCode
; 
  56 import com
.owncloud
.android
.operations
.CreateFolderOperation
; 
  57 import com
.owncloud
.android
.operations
.SynchronizeFolderOperation
; 
  58 import com
.owncloud
.android
.syncadapter
.FileSyncAdapter
; 
  59 import com
.owncloud
.android
.ui
.dialog
.CreateFolderDialogFragment
; 
  60 import com
.owncloud
.android
.ui
.fragment
.FileFragment
; 
  61 import com
.owncloud
.android
.ui
.fragment
.OCFileListFragment
; 
  62 import com
.owncloud
.android
.utils
.DisplayUtils
; 
  63 import com
.owncloud
.android
.utils
.ErrorMessageAdapter
; 
  64 import com
.owncloud
.android
.utils
.Log_OC
; 
  66 public class MoveActivity 
extends HookActivity 
implements FileFragment
.ContainerActivity
,  
  67     OnClickListener
, SwipeRefreshLayout
.OnRefreshListener 
{ 
  69     public static final String EXTRA_CURRENT_FOLDER 
= UploadFilesActivity
.class.getCanonicalName() + ".EXTRA_CURRENT_FOLDER"; 
  70     public static final String EXTRA_TARGET_FILE 
= UploadFilesActivity
.class.getCanonicalName() + "EXTRA_TARGET_FILE"; 
  72     public static final int RESULT_OK_AND_MOVE 
= 1; 
  74     private SyncBroadcastReceiver mSyncBroadcastReceiver
; 
  76     private static final String TAG 
= MoveActivity
.class.getSimpleName(); 
  78     private static final String TAG_LIST_OF_FOLDERS 
= "LIST_OF_FOLDERS"; 
  80     private boolean mSyncInProgress 
= false
; 
  82     private Button mCancelBtn
; 
  83     private Button mChooseBtn
; 
  87     protected void onCreate(Bundle savedInstanceState
) { 
  88         Log_OC
.d(TAG
, "onCreate() start"); 
  89         requestWindowFeature(Window
.FEATURE_INDETERMINATE_PROGRESS
); 
  91         super.onCreate(savedInstanceState
);  
  93         setContentView(R
.layout
.files_move
); 
  95         if (savedInstanceState 
== null
) { 
  99         // sets callback listeners for UI elements 
 103         ActionBar actionBar 
= getSupportActionBar(); 
 104         actionBar
.setDisplayShowTitleEnabled(true
); 
 105         actionBar
.setNavigationMode(ActionBar
.NAVIGATION_MODE_STANDARD
); 
 106         setSupportProgressBarIndeterminateVisibility(mSyncInProgress
); 
 107             // always AFTER setContentView(...) ; to work around bug in its implementation 
 109         // sets message for empty list of folders 
 112         Log_OC
.d(TAG
, "onCreate() end"); 
 117     protected void onStart() { 
 119         getSupportActionBar().setIcon(DisplayUtils
.getSeasonalIconId()); 
 123     protected void onDestroy() { 
 128      *  Called when the ownCloud {@link Account} associated to the Activity was just updated. 
 131     protected void onAccountSet(boolean stateWasRecovered
) { 
 132         super.onAccountSet(stateWasRecovered
); 
 133         if (getAccount() != null
) { 
 137             OCFile folder 
= getFile(); 
 138             if (folder 
== null 
|| !folder
.isFolder()) { 
 139                 // fall back to root folder 
 140                 setFile(getStorageManager().getFileByPath(OCFile
.ROOT_PATH
)); 
 144             if (!stateWasRecovered
) { 
 145                 OCFileListFragment listOfFolders 
= getListOfFilesFragment();  
 146                 listOfFolders
.listDirectory(folder
);    
 148                 startSyncFolderOperation(folder
, false
); 
 151             updateNavigationElementsInActionBar(); 
 155     private void createFragments() { 
 156         OCFileListFragment listOfFiles 
= new OCFileListFragment(); 
 157         Bundle args 
= new Bundle(); 
 158         args
.putBoolean(OCFileListFragment
.ARG_JUST_FOLDERS
, true
); 
 159         args
.putBoolean(OCFileListFragment
.ARG_ALLOW_CONTEXTUAL_ACTIONS
, false
); 
 160         listOfFiles
.setArguments(args
); 
 161         FragmentTransaction transaction 
= getSupportFragmentManager().beginTransaction(); 
 162         transaction
.add(R
.id
.fragment_container
, listOfFiles
, TAG_LIST_OF_FOLDERS
); 
 163         transaction
.commit(); 
 167      * Show a text message on screen view for notifying user if content is 
 168      * loading or folder is empty 
 170     private void setBackgroundText() { 
 171         OCFileListFragment listFragment 
= getListOfFilesFragment(); 
 172         if (listFragment 
!= null
) { 
 173             int message 
= R
.string
.file_list_loading
; 
 174             if (!mSyncInProgress
) { 
 175                 // In case folder list is empty 
 176                 message 
= R
.string
.file_list_empty_moving
; 
 178             listFragment
.setMessageForEmptyList(getString(message
)); 
 180             Log
.e(TAG
, "OCFileListFragment is null"); 
 184     private OCFileListFragment 
getListOfFilesFragment() { 
 185         Fragment listOfFiles 
= getSupportFragmentManager().findFragmentByTag(MoveActivity
.TAG_LIST_OF_FOLDERS
); 
 186         if (listOfFiles 
!= null
) { 
 187             return (OCFileListFragment
)listOfFiles
; 
 189         Log_OC
.wtf(TAG
, "Access to unexisting list of files fragment!!"); 
 197      * Updates action bar and second fragment, if in dual pane mode. 
 200     public void onBrowsedDownTo(OCFile directory
) { 
 202         updateNavigationElementsInActionBar(); 
 204         startSyncFolderOperation(directory
, false
); 
 209     public void startSyncFolderOperation(OCFile folder
, boolean ignoreTag
) { 
 210         long currentSyncTime 
= System
.currentTimeMillis();  
 212         mSyncInProgress 
= true
; 
 214         // perform folder synchronization 
 215         RemoteOperation synchFolderOp 
= new SynchronizeFolderOperation( folder
,   
 218                                                                         getFileOperationsHelper().isSharedSupported(), 
 222                                                                         getApplicationContext() 
 224         synchFolderOp
.execute(getAccount(), this, null
, null
); 
 226         setSupportProgressBarIndeterminateVisibility(true
); 
 232     protected void onResume() { 
 234         Log_OC
.e(TAG
, "onResume() start"); 
 236         // refresh list of files 
 237         refreshListOfFilesFragment(); 
 239         // Listen for sync messages 
 240         IntentFilter syncIntentFilter 
= new IntentFilter(FileSyncAdapter
.EVENT_FULL_SYNC_START
); 
 241         syncIntentFilter
.addAction(FileSyncAdapter
.EVENT_FULL_SYNC_END
); 
 242         syncIntentFilter
.addAction(FileSyncAdapter
.EVENT_FULL_SYNC_FOLDER_CONTENTS_SYNCED
); 
 243         syncIntentFilter
.addAction(SynchronizeFolderOperation
.EVENT_SINGLE_FOLDER_CONTENTS_SYNCED
); 
 244         syncIntentFilter
.addAction(SynchronizeFolderOperation
.EVENT_SINGLE_FOLDER_SHARES_SYNCED
); 
 245         mSyncBroadcastReceiver 
= new SyncBroadcastReceiver(); 
 246         registerReceiver(mSyncBroadcastReceiver
, syncIntentFilter
); 
 248         Log_OC
.d(TAG
, "onResume() end"); 
 252     protected void onPause() { 
 253         Log_OC
.e(TAG
, "onPause() start"); 
 254         if (mSyncBroadcastReceiver 
!= null
) { 
 255             unregisterReceiver(mSyncBroadcastReceiver
); 
 256             //LocalBroadcastManager.getInstance(this).unregisterReceiver(mSyncBroadcastReceiver); 
 257             mSyncBroadcastReceiver 
= null
; 
 260         Log_OC
.d(TAG
, "onPause() end"); 
 265     public boolean onCreateOptionsMenu(Menu menu
) { 
 266         MenuInflater inflater 
= getSherlock().getMenuInflater(); 
 267         inflater
.inflate(R
.menu
.main_menu
, menu
); 
 268         menu
.findItem(R
.id
.action_upload
).setVisible(false
); 
 269         menu
.findItem(R
.id
.action_settings
).setVisible(false
); 
 270         menu
.findItem(R
.id
.action_sync_account
).setVisible(false
); 
 275     public boolean onOptionsItemSelected(MenuItem item
) { 
 276         boolean retval 
= true
; 
 277         switch (item
.getItemId()) { 
 278         case R
.id
.action_create_dir
: { 
 279             CreateFolderDialogFragment dialog 
=  
 280                     CreateFolderDialogFragment
.newInstance(getCurrentFolder()); 
 282                     getSupportFragmentManager(),  
 283                     CreateFolderDialogFragment
.CREATE_FOLDER_FRAGMENT
 
 287         case android
.R
.id
.home
: { 
 288             OCFile currentDir 
= getCurrentFolder(); 
 289             if(currentDir 
!= null 
&& currentDir
.getParentId() != 0) { 
 295             retval 
= super.onOptionsItemSelected(item
); 
 300     private OCFile 
getCurrentFolder() { 
 301         OCFile file 
= getFile(); 
 303             if (file
.isFolder()) { 
 305             } else if (getStorageManager() != null
) { 
 306                 String parentPath 
= file
.getRemotePath().substring(0, file
.getRemotePath().lastIndexOf(file
.getFileName())); 
 307                 return getStorageManager().getFileByPath(parentPath
); 
 313     protected void refreshListOfFilesFragment() { 
 314         OCFileListFragment fileListFragment 
= getListOfFilesFragment(); 
 315         if (fileListFragment 
!= null
) {  
 316             fileListFragment
.listDirectory(); 
 320     public void browseToRoot() { 
 321         OCFileListFragment listOfFiles 
= getListOfFilesFragment();  
 322         if (listOfFiles 
!= null
) {  // should never be null, indeed 
 323             OCFile root 
= getStorageManager().getFileByPath(OCFile
.ROOT_PATH
); 
 324             listOfFiles
.listDirectory(root
); 
 325             setFile(listOfFiles
.getCurrentFile()); 
 326             updateNavigationElementsInActionBar(); 
 327             startSyncFolderOperation(root
, false
); 
 332     public void onBackPressed() { 
 333         OCFileListFragment listOfFiles 
= getListOfFilesFragment(); 
 334         if (listOfFiles 
!= null
) {  // should never be null, indeed 
 335             int levelsUp 
= listOfFiles
.onBrowseUp(); 
 340             setFile(listOfFiles
.getCurrentFile()); 
 341             updateNavigationElementsInActionBar(); 
 345     private void updateNavigationElementsInActionBar() { 
 346         ActionBar actionBar 
= getSupportActionBar(); 
 347         OCFile currentDir 
= getCurrentFolder(); 
 348         boolean atRoot 
= (currentDir 
== null 
|| currentDir
.getParentId() == 0); 
 349         actionBar
.setDisplayHomeAsUpEnabled(!atRoot
); 
 350         actionBar
.setHomeButtonEnabled(!atRoot
); 
 353                 ? 
getString(R
.string
.default_display_name_for_root_folder
)  
 354                 : currentDir
.getFileName() 
 359      * Set per-view controllers 
 361     private void initControls(){ 
 362         mCancelBtn 
= (Button
) findViewById(R
.id
.move_files_btn_cancel
); 
 363         mCancelBtn
.setOnClickListener(this); 
 364         mChooseBtn 
= (Button
) findViewById(R
.id
.move_files_btn_choose
); 
 365         mChooseBtn
.setOnClickListener(this); 
 369     public void onClick(View v
) { 
 370         if (v 
== mCancelBtn
) { 
 372         } else if (v 
== mChooseBtn
) { 
 373             Intent i 
= getIntent(); 
 374             OCFile targetFile 
= (OCFile
) i
.getParcelableExtra(MoveActivity
.EXTRA_TARGET_FILE
); 
 376             Intent data 
= new Intent(); 
 377             data
.putExtra(EXTRA_CURRENT_FOLDER
, getCurrentFolder()); 
 378             data
.putExtra(EXTRA_TARGET_FILE
, targetFile
); 
 379             setResult(RESULT_OK_AND_MOVE
, data
); 
 386     public void onRemoteOperationFinish(RemoteOperation operation
, RemoteOperationResult result
) { 
 387         super.onRemoteOperationFinish(operation
, result
); 
 389         if (operation 
instanceof CreateFolderOperation
) { 
 390             onCreateFolderOperationFinish((CreateFolderOperation
)operation
, result
); 
 397      * Updates the view associated to the activity after the finish of an operation trying  
 398      * to create a new folder. 
 400      * @param operation     Creation operation performed. 
 401      * @param result        Result of the creation. 
 403     private void onCreateFolderOperationFinish( 
 404             CreateFolderOperation operation
, RemoteOperationResult result
 
 407         if (result
.isSuccess()) { 
 408             dismissLoadingDialog(); 
 409             refreshListOfFilesFragment(); 
 411             dismissLoadingDialog(); 
 413                 Toast msg 
= Toast
.makeText(MoveActivity
.this,  
 414                         ErrorMessageAdapter
.getErrorCauseMessage(result
, operation
, getResources()),  
 418             } catch (NotFoundException e
) { 
 419                 Log_OC
.e(TAG
, "Error while trying to show fail message " , e
); 
 426     private class SyncBroadcastReceiver 
extends BroadcastReceiver 
{ 
 429          * {@link BroadcastReceiver} to enable syncing feedback in UI 
 432         public void onReceive(Context context
, Intent intent
) { 
 434                 String event 
= intent
.getAction(); 
 435                 Log_OC
.d(TAG
, "Received broadcast " + event
); 
 436                 String accountName 
= intent
.getStringExtra(FileSyncAdapter
.EXTRA_ACCOUNT_NAME
); 
 437                 String synchFolderRemotePath 
= intent
.getStringExtra(FileSyncAdapter
.EXTRA_FOLDER_PATH
);  
 438                 RemoteOperationResult synchResult 
= (RemoteOperationResult
)intent
.getSerializableExtra(FileSyncAdapter
.EXTRA_RESULT
); 
 439                 boolean sameAccount 
= (getAccount() != null 
&& accountName
.equals(getAccount().name
) && getStorageManager() != null
);  
 443                     if (FileSyncAdapter
.EVENT_FULL_SYNC_START
.equals(event
)) { 
 444                         mSyncInProgress 
= true
; 
 447                         OCFile currentFile 
= (getFile() == null
) ? null 
: getStorageManager().getFileByPath(getFile().getRemotePath()); 
 448                         OCFile currentDir 
= (getCurrentFolder() == null
) ? null 
: getStorageManager().getFileByPath(getCurrentFolder().getRemotePath()); 
 450                         if (currentDir 
== null
) { 
 451                             // current folder was removed from the server  
 452                             Toast
.makeText( MoveActivity
.this,  
 453                                             String
.format(getString(R
.string
.sync_current_folder_was_removed
), getCurrentFolder().getFileName()),  
 459                             if (currentFile 
== null 
&& !getFile().isFolder()) { 
 460                                 // currently selected file was removed in the server, and now we know it 
 461                                 currentFile 
= currentDir
; 
 464                             if (synchFolderRemotePath 
!= null 
&& currentDir
.getRemotePath().equals(synchFolderRemotePath
)) { 
 465                                 OCFileListFragment fileListFragment 
= getListOfFilesFragment(); 
 466                                 if (fileListFragment 
!= null
) { 
 467                                     fileListFragment
.listDirectory(currentDir
); 
 470                             setFile(currentFile
); 
 473                         mSyncInProgress 
= (!FileSyncAdapter
.EVENT_FULL_SYNC_END
.equals(event
) && !SynchronizeFolderOperation
.EVENT_SINGLE_FOLDER_SHARES_SYNCED
.equals(event
)); 
 475                         if (SynchronizeFolderOperation
.EVENT_SINGLE_FOLDER_CONTENTS_SYNCED
. 
 477                                 /// TODO refactor and make common 
 478                                 synchResult 
!= null 
&& !synchResult
.isSuccess() &&   
 479                                 (synchResult
.getCode() == ResultCode
.UNAUTHORIZED   
||  
 480                                     synchResult
.isIdPRedirection()                  || 
 481                                     (synchResult
.isException() && synchResult
.getException()  
 482                                             instanceof AuthenticatorException
))) { 
 484                             OwnCloudClient client 
= null
; 
 486                                 OwnCloudAccount ocAccount 
=  
 487                                         new OwnCloudAccount(getAccount(), context
); 
 488                                 client 
= (OwnCloudClientManagerFactory
.getDefaultSingleton(). 
 489                                         removeClientFor(ocAccount
)); 
 490                                 // TODO get rid of these exceptions 
 491                             } catch (AccountNotFoundException e
) { 
 493                             } catch (AuthenticatorException e
) { 
 495                             } catch (OperationCanceledException e
) { 
 497                             } catch (IOException e
) { 
 501                             if (client 
!= null
) { 
 502                                 OwnCloudCredentials cred 
= client
.getCredentials(); 
 504                                     AccountManager am 
= AccountManager
.get(context
); 
 505                                     if (cred
.authTokenExpires()) { 
 506                                         am
.invalidateAuthToken( 
 511                                         am
.clearPassword(getAccount()); 
 516                             requestCredentialsUpdate(); 
 520                     removeStickyBroadcast(intent
); 
 521                     Log_OC
.d(TAG
, "Setting progress visibility to " + mSyncInProgress
); 
 522                     setSupportProgressBarIndeterminateVisibility(mSyncInProgress 
/*|| mRefreshSharesInProgress*/); 
 528             } catch (RuntimeException e
) { 
 529                 // avoid app crashes after changing the serial id of RemoteOperationResult  
 530                 // in owncloud library with broadcast notifications pending to process 
 531                 removeStickyBroadcast(intent
); 
 539      * Shows the information of the {@link OCFile} received as a  
 540      * parameter in the second fragment. 
 542      * @param file          {@link OCFile} whose details will be shown 
 545     public void showDetails(OCFile file
) { 
 553     public void onTransferStateChanged(OCFile file
, boolean downloading
, boolean uploading
) { 
 559     public void onRefresh() { 
 560         OCFileListFragment listOfFiles 
= getListOfFilesFragment(); 
 561         if (listOfFiles 
!= null
) { 
 562             OCFile folder 
= listOfFiles
.getCurrentFile(); 
 563             if (folder 
!= null
) { 
 564                 startSyncFolderOperation(folder
, true
);