2012-09-04 23:27:04 +04:00
var FileList = {
2012-04-15 17:48:02 +04:00
useUndo : true ,
2013-08-17 15:07:18 +04:00
postProcessList : function ( ) {
$ ( '#fileList tr' ) . each ( function ( ) {
//little hack to set unescape filenames in attribute
$ ( this ) . attr ( 'data-file' , decodeURIComponent ( $ ( this ) . attr ( 'data-file' ) ) ) ;
} ) ;
} ,
2011-06-04 20:44:14 +04:00
update : function ( fileListHtml ) {
2013-08-30 01:45:02 +04:00
var $fileList = $ ( '#fileList' ) ,
permissions = $ ( '#permissions' ) . val ( ) ,
isCreatable = ( permissions & OC . PERMISSION _CREATE ) !== 0 ;
2013-08-17 15:07:18 +04:00
$fileList . empty ( ) . html ( fileListHtml ) ;
2013-08-30 01:45:02 +04:00
$ ( '#emptycontent' ) . toggleClass ( 'hidden' , ! isCreatable || $fileList . find ( 'tr' ) . length > 0 ) ;
2013-08-17 15:07:18 +04:00
$fileList . find ( 'tr' ) . each ( function ( ) {
FileActions . display ( $ ( this ) . children ( 'td.filename' ) ) ;
} ) ;
$fileList . trigger ( jQuery . Event ( "fileActionsReady" ) ) ;
FileList . postProcessList ( ) ;
// "Files" might not be loaded in extending apps
if ( window . Files ) {
Files . setupDragAndDrop ( ) ;
}
2013-08-29 03:17:04 +04:00
FileList . updateFileSummary ( ) ;
2013-08-17 15:07:18 +04:00
$fileList . trigger ( jQuery . Event ( "updated" ) ) ;
2011-06-04 20:44:14 +04:00
} ,
2013-02-09 17:42:03 +04:00
createRow : function ( type , name , iconurl , linktarget , size , lastModified , permissions ) {
var td , simpleSize , basename , extension ;
//containing tr
var tr = $ ( '<tr></tr>' ) . attr ( {
"data-type" : type ,
"data-size" : size ,
"data-file" : name ,
"data-permissions" : permissions
} ) ;
// filename td
td = $ ( '<td></td>' ) . attr ( {
"class" : "filename" ,
2013-08-26 16:36:18 +04:00
"style" : 'background-image:url(' + iconurl + '); background-size: 32px;'
2013-02-09 17:42:03 +04:00
} ) ;
2013-08-26 18:33:51 +04:00
var rand = Math . random ( ) . toString ( 16 ) . slice ( 2 ) ;
td . append ( '<input id="select-' + rand + '" type="checkbox" /><label for="select-' + rand + '"></label>' ) ;
2013-02-09 17:42:03 +04:00
var link _elem = $ ( '<a></a>' ) . attr ( {
"class" : "name" ,
"href" : linktarget
} ) ;
//split extension from filename for non dirs
if ( type != 'dir' && name . indexOf ( '.' ) != - 1 ) {
2012-09-23 05:16:52 +04:00
basename = name . substr ( 0 , name . lastIndexOf ( '.' ) ) ;
extension = name . substr ( name . lastIndexOf ( '.' ) ) ;
2013-02-09 17:42:03 +04:00
} else {
2012-09-23 05:16:52 +04:00
basename = name ;
extension = false ;
2011-07-29 01:52:49 +04:00
}
2013-02-09 17:42:03 +04:00
var name _span = $ ( '<span></span>' ) . addClass ( 'nametext' ) . text ( basename ) ;
link _elem . append ( name _span ) ;
2012-04-15 15:32:45 +04:00
if ( extension ) {
2013-02-09 17:42:03 +04:00
name _span . append ( $ ( '<span></span>' ) . addClass ( 'extension' ) . text ( extension ) ) ;
}
2013-02-22 20:21:57 +04:00
//dirs can show the number of uploaded files
2013-02-09 17:42:03 +04:00
if ( type == 'dir' ) {
link _elem . append ( $ ( '<span></span>' ) . attr ( {
'class' : 'uploadtext' ,
'currentUploads' : 0
} ) ) ;
2011-07-29 01:52:49 +04:00
}
2013-02-09 17:42:03 +04:00
td . append ( link _elem ) ;
tr . append ( td ) ;
2013-02-22 20:21:57 +04:00
2013-02-09 17:42:03 +04:00
//size column
if ( size != t ( 'files' , 'Pending' ) ) {
2013-07-19 00:15:26 +04:00
simpleSize = humanFileSize ( size ) ;
2011-07-29 01:52:49 +04:00
} else {
2013-02-09 17:42:03 +04:00
simpleSize = t ( 'files' , 'Pending' ) ;
2011-07-29 01:52:49 +04:00
}
2013-06-11 16:30:13 +04:00
var sizeColor = Math . round ( 160 - Math . pow ( ( size / ( 1024 * 1024 ) ) , 2 ) ) ;
2013-02-09 17:42:03 +04:00
var lastModifiedTime = Math . round ( lastModified . getTime ( ) / 1000 ) ;
td = $ ( '<td></td>' ) . attr ( {
"class" : "filesize" ,
"style" : 'color:rgb(' + sizeColor + ',' + sizeColor + ',' + sizeColor + ')'
} ) . text ( simpleSize ) ;
tr . append ( td ) ;
2013-02-22 20:21:57 +04:00
2013-02-09 17:42:03 +04:00
// date column
var modifiedColor = Math . round ( ( Math . round ( ( new Date ( ) ) . getTime ( ) / 1000 ) - lastModifiedTime ) / 60 / 60 / 24 * 5 ) ;
td = $ ( '<td></td>' ) . attr ( { "class" : "date" } ) ;
td . append ( $ ( '<span></span>' ) . attr ( {
"class" : "modified" ,
"title" : formatDate ( lastModified ) ,
"style" : 'color:rgb(' + modifiedColor + ',' + modifiedColor + ',' + modifiedColor + ')'
} ) . text ( relative _modified _date ( lastModified . getTime ( ) / 1000 ) ) ) ;
tr . append ( td ) ;
return tr ;
} ,
2013-06-25 14:24:14 +04:00
addFile : function ( name , size , lastModified , loading , hidden , param ) {
2013-02-09 17:42:03 +04:00
var imgurl ;
2013-06-25 14:24:14 +04:00
if ( ! param ) {
param = { } ;
}
var download _url = null ;
if ( ! param . download _url ) {
download _url = OC . Router . generate ( 'download' , { file : $ ( '#dir' ) . val ( ) + '/' + name } ) ;
} else {
download _url = param . download _url ;
}
2013-02-09 17:42:03 +04:00
if ( loading ) {
imgurl = OC . imagePath ( 'core' , 'loading.gif' ) ;
} else {
imgurl = OC . imagePath ( 'core' , 'filetypes/file.png' ) ;
}
var tr = this . createRow (
'file' ,
name ,
imgurl ,
2013-06-25 14:24:14 +04:00
download _url ,
2013-02-09 17:42:03 +04:00
size ,
lastModified ,
$ ( '#permissions' ) . val ( )
) ;
2013-02-22 20:21:57 +04:00
2013-03-13 20:26:37 +04:00
FileList . insertElement ( name , 'file' , tr ) ;
2011-07-19 22:57:40 +04:00
if ( loading ) {
2013-03-13 20:26:37 +04:00
tr . data ( 'loading' , true ) ;
2011-07-19 22:57:40 +04:00
} else {
2013-03-13 20:26:37 +04:00
tr . find ( 'td.filename' ) . draggable ( dragOptions ) ;
2011-07-19 22:57:40 +04:00
}
2012-09-05 08:12:11 +04:00
if ( hidden ) {
2013-03-13 20:26:37 +04:00
tr . hide ( ) ;
2012-09-05 08:12:11 +04:00
}
2013-03-13 20:26:37 +04:00
FileActions . display ( tr . find ( 'td.filename' ) ) ;
return tr ;
2011-06-04 20:44:14 +04:00
} ,
2012-09-05 08:12:11 +04:00
addDir : function ( name , size , lastModified , hidden ) {
2013-02-22 20:21:57 +04:00
2013-02-09 17:42:03 +04:00
var tr = this . createRow (
'dir' ,
name ,
OC . imagePath ( 'core' , 'filetypes/folder.png' ) ,
OC . linkTo ( 'files' , 'index.php' ) + "?dir=" + encodeURIComponent ( $ ( '#dir' ) . val ( ) + '/' + name ) . replace ( /%2F/g , '/' ) ,
size ,
lastModified ,
$ ( '#permissions' ) . val ( )
) ;
2013-02-22 20:21:57 +04:00
2013-02-09 17:42:03 +04:00
FileList . insertElement ( name , 'dir' , tr ) ;
2013-03-13 20:26:37 +04:00
var td = tr . find ( 'td.filename' ) ;
td . draggable ( dragOptions ) ;
td . droppable ( folderDropOptions ) ;
2012-09-05 08:12:11 +04:00
if ( hidden ) {
2013-03-13 20:26:37 +04:00
tr . hide ( ) ;
2012-09-05 08:12:11 +04:00
}
2013-03-13 20:26:37 +04:00
FileActions . display ( tr . find ( 'td.filename' ) ) ;
return tr ;
2011-06-04 20:44:14 +04:00
} ,
2013-08-17 15:07:18 +04:00
/ * *
* @ brief Changes the current directory and reload the file list .
* @ param targetDir target directory ( non URL encoded )
* @ param changeUrl false if the URL must not be changed ( defaults to true )
* /
2013-08-29 23:56:14 +04:00
changeDirectory : function ( targetDir , changeUrl , force ) {
2013-08-17 15:07:18 +04:00
var $dir = $ ( '#dir' ) ,
url ,
currentDir = $dir . val ( ) || '/' ;
targetDir = targetDir || '/' ;
2013-08-29 23:56:14 +04:00
if ( ! force && currentDir === targetDir ) {
2013-08-17 15:07:18 +04:00
return ;
}
FileList . setCurrentDir ( targetDir , changeUrl ) ;
FileList . reload ( ) ;
} ,
2013-08-29 23:56:14 +04:00
linkTo : function ( dir ) {
return OC . linkTo ( 'files' , 'index.php' ) + "?dir=" + encodeURIComponent ( dir ) . replace ( /%2F/g , '/' ) ;
} ,
2013-08-17 15:07:18 +04:00
setCurrentDir : function ( targetDir , changeUrl ) {
$ ( '#dir' ) . val ( targetDir ) ;
2013-08-29 23:56:14 +04:00
if ( changeUrl !== false ) {
if ( window . history . pushState && changeUrl !== false ) {
url = FileList . linkTo ( targetDir ) ;
window . history . pushState ( { dir : targetDir } , '' , url ) ;
}
// use URL hash for IE8
else {
window . location . hash = '?dir=' + encodeURIComponent ( targetDir ) . replace ( /%2F/g , '/' ) ;
}
2013-08-17 15:07:18 +04:00
}
} ,
/ * *
* @ brief Reloads the file list using ajax call
* /
reload : function ( ) {
FileList . showMask ( ) ;
if ( FileList . _reloadCall ) {
FileList . _reloadCall . abort ( ) ;
}
FileList . _reloadCall = $ . ajax ( {
url : OC . filePath ( 'files' , 'ajax' , 'list.php' ) ,
data : {
dir : $ ( '#dir' ) . val ( ) ,
breadcrumb : true
} ,
error : function ( result ) {
FileList . reloadCallback ( result ) ;
} ,
success : function ( result ) {
FileList . reloadCallback ( result ) ;
}
} ) ;
} ,
reloadCallback : function ( result ) {
var $controls = $ ( '#controls' ) ;
delete FileList . _reloadCall ;
FileList . hideMask ( ) ;
if ( ! result || result . status === 'error' ) {
OC . Notification . show ( result . data . message ) ;
return ;
}
if ( result . status === 404 ) {
// go back home
FileList . changeDirectory ( '/' ) ;
return ;
}
2013-08-30 01:45:02 +04:00
if ( result . data . permissions ) {
FileList . setDirectoryPermissions ( result . data . permissions ) ;
}
2011-06-04 20:44:14 +04:00
if ( typeof ( result . data . breadcrumb ) != 'undefined' ) {
2013-08-17 15:07:18 +04:00
$controls . find ( '.crumb' ) . remove ( ) ;
$controls . prepend ( result . data . breadcrumb ) ;
2013-08-27 15:13:00 +04:00
var width = $ ( window ) . width ( ) ;
Files . initBreadCrumbs ( ) ;
Files . resizeBreadcrumbs ( width , true ) ;
2013-08-27 13:18:59 +04:00
// in case svg is not supported by the browser we need to execute the fallback mechanism
if ( ! SVGSupport ( ) ) {
replaceSVG ( ) ;
}
2011-06-04 20:44:14 +04:00
}
2013-08-27 13:18:59 +04:00
2011-06-04 20:44:14 +04:00
FileList . update ( result . data . files ) ;
} ,
2013-08-30 01:45:02 +04:00
setDirectoryPermissions : function ( permissions ) {
var isCreatable = ( permissions & OC . PERMISSION _CREATE ) !== 0 ;
$ ( '#permissions' ) . val ( permissions ) ;
$ ( '.creatable' ) . toggleClass ( 'hidden' , ! isCreatable ) ;
$ ( '.notCreatable' ) . toggleClass ( 'hidden' , isCreatable ) ;
} ,
2011-06-04 20:44:14 +04:00
remove : function ( name ) {
2011-11-02 01:35:13 +04:00
$ ( 'tr' ) . filterAttr ( 'data-file' , name ) . find ( 'td.filename' ) . draggable ( 'destroy' ) ;
$ ( 'tr' ) . filterAttr ( 'data-file' , name ) . remove ( ) ;
2013-07-03 21:50:03 +04:00
FileList . updateFileSummary ( ) ;
2011-07-30 16:42:58 +04:00
if ( $ ( 'tr[data-file]' ) . length == 0 ) {
2013-08-17 15:07:18 +04:00
$ ( '#emptycontent' ) . removeClass ( 'hidden' ) ;
2011-07-30 16:42:58 +04:00
}
2011-06-04 20:44:14 +04:00
} ,
insertElement : function ( name , type , element ) {
//find the correct spot to insert the file or folder
2012-09-23 05:16:52 +04:00
var pos , fileElements = $ ( 'tr[data-file][data-type="' + type + '"]:visible' ) ;
2011-06-04 20:44:14 +04:00
if ( name . localeCompare ( $ ( fileElements [ 0 ] ) . attr ( 'data-file' ) ) < 0 ) {
2011-07-26 18:55:28 +04:00
pos = - 1 ;
2011-06-04 20:44:14 +04:00
} else if ( name . localeCompare ( $ ( fileElements [ fileElements . length - 1 ] ) . attr ( 'data-file' ) ) > 0 ) {
pos = fileElements . length - 1 ;
} else {
2012-09-23 05:16:52 +04:00
for ( pos = 0 ; pos < fileElements . length - 1 ; pos ++ ) {
2011-06-04 20:44:14 +04:00
if ( name . localeCompare ( $ ( fileElements [ pos ] ) . attr ( 'data-file' ) ) > 0 && name . localeCompare ( $ ( fileElements [ pos + 1 ] ) . attr ( 'data-file' ) ) < 0 ) {
break ;
}
}
}
2011-07-04 23:46:20 +04:00
if ( fileElements . length ) {
2011-07-26 18:55:28 +04:00
if ( pos == - 1 ) {
$ ( fileElements [ 0 ] ) . before ( element ) ;
} else {
$ ( fileElements [ pos ] ) . after ( element ) ;
}
} else if ( type == 'dir' && $ ( 'tr[data-file]' ) . length > 0 ) {
$ ( 'tr[data-file]' ) . first ( ) . before ( element ) ;
2013-07-12 12:40:24 +04:00
} else if ( type == 'file' && $ ( 'tr[data-file]' ) . length > 0 ) {
$ ( 'tr[data-file]' ) . last ( ) . before ( element ) ;
2011-07-04 23:46:20 +04:00
} else {
$ ( '#fileList' ) . append ( element ) ;
}
2013-08-17 15:07:18 +04:00
$ ( '#emptycontent' ) . addClass ( 'hidden' ) ;
2013-07-03 21:50:03 +04:00
FileList . updateFileSummary ( ) ;
2011-07-19 22:57:40 +04:00
} ,
2012-10-08 19:28:56 +04:00
loadingDone : function ( name , id ) {
2012-09-23 05:16:52 +04:00
var mime , tr = $ ( 'tr' ) . filterAttr ( 'data-file' , name ) ;
2011-10-08 23:18:47 +04:00
tr . data ( 'loading' , false ) ;
2012-09-23 05:16:52 +04:00
mime = tr . data ( 'mime' ) ;
2011-10-08 23:18:47 +04:00
tr . attr ( 'data-mime' , mime ) ;
2012-10-08 19:47:02 +04:00
if ( id != null ) {
tr . attr ( 'data-id' , id ) ;
}
2013-08-24 01:19:21 +04:00
var path = getPathForPreview ( name ) ;
2013-08-14 15:25:07 +04:00
lazyLoadPreview ( path , mime , function ( previewpath ) {
2013-07-02 13:13:22 +04:00
tr . find ( 'td.filename' ) . attr ( 'style' , 'background-image:url(' + previewpath + ')' ) ;
2011-10-08 23:18:47 +04:00
} ) ;
tr . find ( 'td.filename' ) . draggable ( dragOptions ) ;
2011-07-19 22:57:40 +04:00
} ,
isLoading : function ( name ) {
2011-11-02 01:35:13 +04:00
return $ ( 'tr' ) . filterAttr ( 'data-file' , name ) . data ( 'loading' ) ;
2011-07-29 01:04:34 +04:00
} ,
rename : function ( name ) {
2012-09-23 05:16:52 +04:00
var tr , td , input , form ;
tr = $ ( 'tr' ) . filterAttr ( 'data-file' , name ) ;
2011-07-29 04:26:20 +04:00
tr . data ( 'renaming' , true ) ;
2012-09-23 05:16:52 +04:00
td = tr . children ( 'td.filename' ) ;
2013-08-17 12:46:03 +04:00
input = $ ( '<input type="text" class="filename"/>' ) . val ( name ) ;
2012-09-23 05:16:52 +04:00
form = $ ( '<form></form>' ) ;
2011-07-29 01:04:34 +04:00
form . append ( input ) ;
2012-10-18 16:16:59 +04:00
td . children ( 'a.name' ) . hide ( ) ;
td . append ( form ) ;
2011-07-29 01:04:34 +04:00
input . focus ( ) ;
2013-05-13 17:54:45 +04:00
//preselect input
var len = input . val ( ) . lastIndexOf ( '.' ) ;
if ( len === - 1 ) {
len = input . val ( ) . length ;
}
input . selectRange ( 0 , len ) ;
2013-06-25 14:24:14 +04:00
2011-07-29 01:04:34 +04:00
form . submit ( function ( event ) {
event . stopPropagation ( ) ;
event . preventDefault ( ) ;
2011-07-29 19:51:17 +04:00
var newname = input . val ( ) ;
2013-01-06 15:52:00 +04:00
if ( ! Files . isFileNameValid ( newname ) ) {
2012-11-22 16:03:17 +04:00
return false ;
2013-01-07 13:39:35 +04:00
} else if ( newname != name ) {
2012-09-05 08:12:11 +04:00
if ( FileList . checkName ( name , newname , false ) ) {
2012-07-30 20:21:58 +04:00
newname = name ;
2012-10-14 23:04:08 +04:00
} else {
2013-05-03 02:15:28 +04:00
// save background image, because it's replaced by a spinner while async request
var oldBackgroundImage = td . css ( 'background-image' ) ;
// mark as loading
td . css ( 'background-image' , 'url(' + OC . imagePath ( 'core' , 'loading.gif' ) + ')' ) ;
$ . ajax ( {
url : OC . filePath ( 'files' , 'ajax' , 'rename.php' ) ,
data : {
dir : $ ( '#dir' ) . val ( ) ,
newname : newname ,
file : name
} ,
success : function ( result ) {
if ( ! result || result . status === 'error' ) {
OC . Notification . show ( result . data . message ) ;
newname = name ;
// revert changes
tr . attr ( 'data-file' , newname ) ;
var path = td . children ( 'a.name' ) . attr ( 'href' ) ;
td . children ( 'a.name' ) . attr ( 'href' , path . replace ( encodeURIComponent ( name ) , encodeURIComponent ( newname ) ) ) ;
if ( newname . indexOf ( '.' ) > 0 && tr . data ( 'type' ) !== 'dir' ) {
var basename = newname . substr ( 0 , newname . lastIndexOf ( '.' ) ) ;
} else {
var basename = newname ;
}
td . find ( 'a.name span.nametext' ) . text ( basename ) ;
if ( newname . indexOf ( '.' ) > 0 && tr . data ( 'type' ) !== 'dir' ) {
if ( td . find ( 'a.name span.extension' ) . length === 0 ) {
td . find ( 'a.name span.nametext' ) . append ( '<span class="extension"></span>' ) ;
}
td . find ( 'a.name span.extension' ) . text ( newname . substr ( newname . lastIndexOf ( '.' ) ) ) ;
}
tr . find ( '.fileactions' ) . effect ( 'highlight' , { } , 5000 ) ;
tr . effect ( 'highlight' , { } , 5000 ) ;
}
// remove loading mark and recover old image
td . css ( 'background-image' , oldBackgroundImage ) ;
2012-07-30 20:21:58 +04:00
}
} ) ;
2012-09-19 16:05:09 +04:00
}
2012-10-02 20:38:17 +04:00
}
2012-11-15 20:16:18 +04:00
tr . data ( 'renaming' , false ) ;
2012-10-02 20:38:17 +04:00
tr . attr ( 'data-file' , newname ) ;
var path = td . children ( 'a.name' ) . attr ( 'href' ) ;
td . children ( 'a.name' ) . attr ( 'href' , path . replace ( encodeURIComponent ( name ) , encodeURIComponent ( newname ) ) ) ;
if ( newname . indexOf ( '.' ) > 0 && tr . data ( 'type' ) != 'dir' ) {
var basename = newname . substr ( 0 , newname . lastIndexOf ( '.' ) ) ;
} else {
var basename = newname ;
}
2012-10-18 16:16:59 +04:00
td . find ( 'a.name span.nametext' ) . text ( basename ) ;
2012-10-02 20:38:17 +04:00
if ( newname . indexOf ( '.' ) > 0 && tr . data ( 'type' ) != 'dir' ) {
2012-10-18 16:16:59 +04:00
if ( td . find ( 'a.name span.extension' ) . length == 0 ) {
td . find ( 'a.name span.nametext' ) . append ( '<span class="extension"></span>' ) ;
}
td . find ( 'a.name span.extension' ) . text ( newname . substr ( newname . lastIndexOf ( '.' ) ) ) ;
2012-07-30 20:21:58 +04:00
}
2012-10-18 16:16:59 +04:00
form . remove ( ) ;
td . children ( 'a.name' ) . show ( ) ;
2011-12-06 02:51:44 +04:00
return false ;
2011-07-29 01:04:34 +04:00
} ) ;
2012-12-18 19:39:01 +04:00
input . keyup ( function ( event ) {
if ( event . keyCode == 27 ) {
tr . data ( 'renaming' , false ) ;
form . remove ( ) ;
td . children ( 'a.name' ) . show ( ) ;
}
} ) ;
2011-12-06 02:51:44 +04:00
input . click ( function ( event ) {
2011-07-29 01:04:34 +04:00
event . stopPropagation ( ) ;
event . preventDefault ( ) ;
} ) ;
input . blur ( function ( ) {
2011-07-29 19:51:17 +04:00
form . trigger ( 'submit' ) ;
2011-07-29 01:04:34 +04:00
} ) ;
2011-08-04 02:22:44 +04:00
} ,
2012-09-05 08:12:11 +04:00
checkName : function ( oldName , newName , isNewFile ) {
if ( isNewFile || $ ( 'tr' ) . filterAttr ( 'data-file' , newName ) . length > 0 ) {
2013-02-09 20:20:08 +04:00
var html ;
if ( isNewFile ) {
html = t ( 'files' , '{new_name} already exists' , { new _name : escapeHTML ( newName ) } ) + '<span class="replace">' + t ( 'files' , 'replace' ) + '</span><span class="suggest">' + t ( 'files' , 'suggest name' ) + '</span> <span class="cancel">' + t ( 'files' , 'cancel' ) + '</span>' ;
} else {
html = t ( 'files' , '{new_name} already exists' , { new _name : escapeHTML ( newName ) } ) + '<span class="replace">' + t ( 'files' , 'replace' ) + '</span><span class="cancel">' + t ( 'files' , 'cancel' ) + '</span>' ;
}
html = $ ( '<span>' + html + '</span>' ) ;
html . attr ( 'data-oldName' , oldName ) ;
html . attr ( 'data-newName' , newName ) ;
html . attr ( 'data-isNewFile' , isNewFile ) ;
OC . Notification . showHtml ( html ) ;
2012-09-05 08:12:11 +04:00
return true ;
} else {
return false ;
}
} ,
replace : function ( oldName , newName , isNewFile ) {
2012-07-30 20:21:58 +04:00
// Finish any existing actions
2012-09-05 08:12:11 +04:00
$ ( 'tr' ) . filterAttr ( 'data-file' , oldName ) . hide ( ) ;
$ ( 'tr' ) . filterAttr ( 'data-file' , newName ) . hide ( ) ;
var tr = $ ( 'tr' ) . filterAttr ( 'data-file' , oldName ) . clone ( ) ;
tr . attr ( 'data-replace' , 'true' ) ;
tr . attr ( 'data-file' , newName ) ;
var td = tr . children ( 'td.filename' ) ;
td . children ( 'a.name .span' ) . text ( newName ) ;
var path = td . children ( 'a.name' ) . attr ( 'href' ) ;
td . children ( 'a.name' ) . attr ( 'href' , path . replace ( encodeURIComponent ( oldName ) , encodeURIComponent ( newName ) ) ) ;
if ( newName . indexOf ( '.' ) > 0 ) {
var basename = newName . substr ( 0 , newName . lastIndexOf ( '.' ) ) ;
} else {
var basename = newName ;
}
td . children ( 'a.name' ) . empty ( ) ;
var span = $ ( '<span class="nametext"></span>' ) ;
span . text ( basename ) ;
td . children ( 'a.name' ) . append ( span ) ;
if ( newName . indexOf ( '.' ) > 0 ) {
span . append ( $ ( '<span class="extension">' + newName . substr ( newName . lastIndexOf ( '.' ) ) + '</span>' ) ) ;
}
FileList . insertElement ( newName , tr . data ( 'type' ) , tr ) ;
tr . show ( ) ;
2012-07-30 20:21:58 +04:00
FileList . replaceCanceled = false ;
FileList . replaceOldName = oldName ;
FileList . replaceNewName = newName ;
2012-09-05 08:12:11 +04:00
FileList . replaceIsNewFile = isNewFile ;
2012-08-20 19:24:10 +04:00
FileList . lastAction = function ( ) {
2012-07-30 20:21:58 +04:00
FileList . finishReplace ( ) ;
} ;
2013-02-09 20:20:08 +04:00
if ( ! isNewFile ) {
2013-01-05 02:34:09 +04:00
OC . Notification . showHtml ( t ( 'files' , 'replaced {new_name} with {old_name}' , { new _name : newName } , { old _name : oldName } ) + '<span class="undo">' + t ( 'files' , 'undo' ) + '</span>' ) ;
2012-09-05 08:12:11 +04:00
}
2012-07-30 20:21:58 +04:00
} ,
finishReplace : function ( ) {
if ( ! FileList . replaceCanceled && FileList . replaceOldName && FileList . replaceNewName ) {
2012-09-06 06:13:50 +04:00
$ . ajax ( { url : OC . filePath ( 'files' , 'ajax' , 'rename.php' ) , async : false , data : { dir : $ ( '#dir' ) . val ( ) , newname : FileList . replaceNewName , file : FileList . replaceOldName } , success : function ( result ) {
if ( result && result . status == 'success' ) {
$ ( 'tr' ) . filterAttr ( 'data-replace' , 'true' ) . removeAttr ( 'data-replace' ) ;
} else {
OC . dialogs . alert ( result . data . message , 'Error moving file' ) ;
}
FileList . replaceCanceled = true ;
FileList . replaceOldName = null ;
FileList . replaceNewName = null ;
FileList . lastAction = null ;
} } ) ;
2012-07-30 20:21:58 +04:00
}
} ,
2011-08-28 03:32:48 +04:00
do _delete : function ( files ) {
2013-01-30 17:32:20 +04:00
if ( files . substr ) {
files = [ files ] ;
2013-02-21 00:57:50 +04:00
}
for ( var i = 0 ; i < files . length ; i ++ ) {
2013-01-30 17:32:20 +04:00
var deleteAction = $ ( 'tr' ) . filterAttr ( 'data-file' , files [ i ] ) . children ( "td.date" ) . children ( ".action.delete" ) ;
2013-07-29 20:27:11 +04:00
deleteAction . removeClass ( 'delete-icon' ) . addClass ( 'progress-icon' ) ;
2013-01-30 17:32:20 +04:00
}
2012-07-30 20:21:58 +04:00
// Finish any existing actions
2012-09-19 13:56:31 +04:00
if ( FileList . lastAction ) {
2012-07-30 20:21:58 +04:00
FileList . lastAction ( ) ;
2011-08-04 02:22:44 +04:00
}
2012-10-14 23:04:08 +04:00
2013-01-22 20:43:46 +04:00
var fileNames = JSON . stringify ( files ) ;
$ . post ( OC . filePath ( 'files' , 'ajax' , 'delete.php' ) ,
{ dir : $ ( '#dir' ) . val ( ) , files : fileNames } ,
function ( result ) {
if ( result . status == 'success' ) {
$ . each ( files , function ( index , file ) {
var files = $ ( 'tr' ) . filterAttr ( 'data-file' , file ) ;
2013-02-25 19:22:06 +04:00
files . remove ( ) ;
2013-01-22 20:43:46 +04:00
files . find ( 'input[type="checkbox"]' ) . removeAttr ( 'checked' ) ;
files . removeClass ( 'selected' ) ;
2011-08-04 02:22:44 +04:00
} ) ;
2013-01-22 20:43:46 +04:00
procesSelection ( ) ;
2013-07-26 13:13:43 +04:00
checkTrashStatus ( ) ;
2013-07-03 21:50:03 +04:00
FileList . updateFileSummary ( ) ;
2013-01-30 17:32:20 +04:00
} else {
$ . each ( files , function ( index , file ) {
2013-07-29 20:27:11 +04:00
var deleteAction = $ ( 'tr' ) . filterAttr ( 'data-file' , files [ i ] ) . children ( "td.date" ) . children ( ".action.delete" ) ;
deleteAction . removeClass ( 'progress-icon' ) . addClass ( 'delete-icon' ) ;
2013-01-30 17:32:20 +04:00
} ) ;
2013-02-22 20:21:57 +04:00
}
2013-01-22 20:43:46 +04:00
} ) ;
2013-07-03 21:50:03 +04:00
} ,
createFileSummary : function ( ) {
if ( $ ( '#fileList tr' ) . length > 0 ) {
var totalDirs = 0 ;
var totalFiles = 0 ;
var totalSize = 0 ;
// Count types and filesize
$ . each ( $ ( 'tr[data-file]' ) , function ( index , value ) {
if ( $ ( value ) . data ( 'type' ) === 'dir' ) {
totalDirs ++ ;
} else if ( $ ( value ) . data ( 'type' ) === 'file' ) {
totalFiles ++ ;
}
totalSize += parseInt ( $ ( value ) . data ( 'size' ) ) ;
} ) ;
// Get translations
var directoryInfo = n ( 'files' , '%n folder' , '%n folders' , totalDirs ) ;
var fileInfo = n ( 'files' , '%n file' , '%n files' , totalFiles ) ;
var infoVars = {
dirs : '<span class="dirinfo">' + directoryInfo + '</span><span class="connector">' ,
files : '</span><span class="fileinfo">' + fileInfo + '</span>'
}
var info = t ( 'files' , '{dirs} and {files}' , infoVars ) ;
// don't show the filesize column, if filesize is NaN (e.g. in trashbin)
if ( isNaN ( totalSize ) ) {
var fileSize = '' ;
} else {
var fileSize = '<td class="filesize">' + humanFileSize ( totalSize ) + '</td>' ;
}
$ ( '#fileList' ) . append ( '<tr class="summary"><td><span class="info">' + info + '</span></td>' + fileSize + '<td></td></tr>' ) ;
var $dirInfo = $ ( '.summary .dirinfo' ) ;
var $fileInfo = $ ( '.summary .fileinfo' ) ;
var $connector = $ ( '.summary .connector' ) ;
// Show only what's necessary, e.g.: no files: don't show "0 files"
if ( $dirInfo . html ( ) . charAt ( 0 ) === "0" ) {
$dirInfo . hide ( ) ;
$connector . hide ( ) ;
}
if ( $fileInfo . html ( ) . charAt ( 0 ) === "0" ) {
$fileInfo . hide ( ) ;
$connector . hide ( ) ;
}
}
} ,
updateFileSummary : function ( ) {
var $summary = $ ( '.summary' ) ;
// Check if we should remove the summary to show "Upload something"
if ( $ ( '#fileList tr' ) . length === 1 && $summary . length === 1 ) {
$summary . remove ( ) ;
}
// If there's no summary create one (createFileSummary checks if there's data)
else if ( $summary . length === 0 ) {
FileList . createFileSummary ( ) ;
}
// There's a summary and data -> Update the summary
else if ( $ ( '#fileList tr' ) . length > 1 && $summary . length === 1 ) {
var totalDirs = 0 ;
var totalFiles = 0 ;
var totalSize = 0 ;
$ . each ( $ ( 'tr[data-file]' ) , function ( index , value ) {
if ( $ ( value ) . data ( 'type' ) === 'dir' ) {
totalDirs ++ ;
} else if ( $ ( value ) . data ( 'type' ) === 'file' ) {
totalFiles ++ ;
}
if ( $ ( value ) . data ( 'size' ) !== undefined ) {
totalSize += parseInt ( $ ( value ) . data ( 'size' ) ) ;
}
} ) ;
var $dirInfo = $ ( '.summary .dirinfo' ) ;
var $fileInfo = $ ( '.summary .fileinfo' ) ;
var $connector = $ ( '.summary .connector' ) ;
// Substitute old content with new translations
$dirInfo . html ( n ( 'files' , '%n folder' , '%n folders' , totalDirs ) ) ;
$fileInfo . html ( n ( 'files' , '%n file' , '%n files' , totalFiles ) ) ;
$ ( '.summary .filesize' ) . html ( humanFileSize ( totalSize ) ) ;
// Show only what's necessary (may be hidden)
if ( $dirInfo . html ( ) . charAt ( 0 ) === "0" ) {
$dirInfo . hide ( ) ;
$connector . hide ( ) ;
} else {
$dirInfo . show ( ) ;
}
if ( $fileInfo . html ( ) . charAt ( 0 ) === "0" ) {
$fileInfo . hide ( ) ;
$connector . hide ( ) ;
} else {
$fileInfo . show ( ) ;
}
if ( $dirInfo . html ( ) . charAt ( 0 ) !== "0" && $fileInfo . html ( ) . charAt ( 0 ) !== "0" ) {
$connector . show ( ) ;
}
}
2013-08-17 15:07:18 +04:00
} ,
showMask : function ( ) {
// in case one was shown before
var $mask = $ ( '#content .mask' ) ;
if ( $mask . length ) {
return ;
}
$mask = $ ( '<div class="mask transparent"></div>' ) ;
$mask . css ( 'background-image' , 'url(' + OC . imagePath ( 'core' , 'loading.gif' ) + ')' ) ;
2013-08-27 12:29:28 +04:00
$mask . css ( 'background-repeat' , 'no-repeat' ) ;
2013-08-17 15:07:18 +04:00
$ ( '#content' ) . append ( $mask ) ;
// block UI, but only make visible in case loading takes longer
FileList . _maskTimeout = window . setTimeout ( function ( ) {
// reset opacity
$mask . removeClass ( 'transparent' ) ;
} , 250 ) ;
} ,
hideMask : function ( ) {
var $mask = $ ( '#content .mask' ) . remove ( ) ;
if ( FileList . _maskTimeout ) {
window . clearTimeout ( FileList . _maskTimeout ) ;
}
2011-06-04 20:44:14 +04:00
}
2012-09-23 05:16:52 +04:00
} ;
2011-08-04 02:22:44 +04:00
$ ( document ) . ready ( function ( ) {
2013-03-13 20:26:37 +04:00
// handle upload events
2013-03-27 18:55:44 +04:00
var file _upload _start = $ ( '#file_upload_start' ) ;
file _upload _start . on ( 'fileuploaddrop' , function ( e , data ) {
2013-03-13 20:26:37 +04:00
// only handle drop to dir if fileList exists
if ( $ ( '#fileList' ) . length > 0 ) {
var dropTarget = $ ( e . originalEvent . target ) . closest ( 'tr' ) ;
if ( dropTarget && dropTarget . data ( 'type' ) === 'dir' ) { // drag&drop upload to folder
var dirName = dropTarget . data ( 'file' ) ;
// update folder in form
data . formData = function ( form ) {
var formArray = form . serializeArray ( ) ;
// array index 0 contains the max files size
// array index 1 contains the request token
// array index 2 contains the directory
var parentDir = formArray [ 2 ] [ 'value' ] ;
if ( parentDir === '/' ) {
formArray [ 2 ] [ 'value' ] += dirName ;
} else {
formArray [ 2 ] [ 'value' ] += '/' + dirName ;
}
return formArray ;
}
}
}
} ) ;
2013-03-27 18:55:44 +04:00
file _upload _start . on ( 'fileuploadadd' , function ( e , data ) {
2013-03-13 20:26:37 +04:00
// only add to fileList if it exists
if ( $ ( '#fileList' ) . length > 0 ) {
if ( FileList . deleteFiles && FileList . deleteFiles . indexOf ( data . files [ 0 ] . name ) != - 1 ) { //finish delete if we are uploading a deleted file
FileList . finishDelete ( null , true ) ; //delete file before continuing
}
// add ui visualization to existing folder or as new stand-alone file?
var dropTarget = $ ( e . originalEvent . target ) . closest ( 'tr' ) ;
if ( dropTarget && dropTarget . data ( 'type' ) === 'dir' ) {
// add to existing folder
var dirName = dropTarget . data ( 'file' ) ;
// set dir context
data . context = $ ( 'tr' ) . filterAttr ( 'data-type' , 'dir' ) . filterAttr ( 'data-file' , dirName ) ;
// update upload counter ui
var uploadtext = data . context . find ( '.uploadtext' ) ;
var currentUploads = parseInt ( uploadtext . attr ( 'currentUploads' ) ) ;
currentUploads += 1 ;
uploadtext . attr ( 'currentUploads' , currentUploads ) ;
2013-08-14 08:29:19 +04:00
var translatedText = n ( 'files' , 'Uploading %n file' , 'Uploading %n files' , currentUploads ) ;
2013-03-13 20:26:37 +04:00
if ( currentUploads === 1 ) {
var img = OC . imagePath ( 'core' , 'loading.gif' ) ;
data . context . find ( 'td.filename' ) . attr ( 'style' , 'background-image:url(' + img + ')' ) ;
2013-08-09 22:37:18 +04:00
uploadtext . text ( translatedText ) ;
2013-03-13 20:26:37 +04:00
uploadtext . show ( ) ;
} else {
2013-08-09 22:37:18 +04:00
uploadtext . text ( translatedText ) ;
2013-03-13 20:26:37 +04:00
}
} else {
// add as stand-alone row to filelist
var uniqueName = getUniqueName ( data . files [ 0 ] . name ) ;
var size = t ( 'files' , 'Pending' ) ;
if ( data . files [ 0 ] . size >= 0 ) {
size = data . files [ 0 ] . size ;
}
var date = new Date ( ) ;
2013-06-25 14:24:14 +04:00
var param = { } ;
2013-07-01 17:47:17 +04:00
if ( $ ( '#publicUploadRequestToken' ) . length ) {
2013-06-25 14:24:14 +04:00
param . download _url = document . location . href + '&download&path=/' + $ ( '#dir' ) . val ( ) + '/' + uniqueName ;
}
2013-03-13 20:26:37 +04:00
// create new file context
2013-06-25 14:24:14 +04:00
data . context = FileList . addFile ( uniqueName , size , date , true , false , param ) ;
2013-03-13 20:26:37 +04:00
}
}
} ) ;
2013-03-27 18:55:44 +04:00
file _upload _start . on ( 'fileuploaddone' , function ( e , data ) {
2013-03-13 20:26:37 +04:00
// only update the fileList if it exists
if ( $ ( '#fileList' ) . length > 0 ) {
var response ;
if ( typeof data . result === 'string' ) {
response = data . result ;
} else {
// fetch response from iframe
response = data . result [ 0 ] . body . innerText ;
}
var result = $ . parseJSON ( response ) ;
if ( typeof result [ 0 ] !== 'undefined' && result [ 0 ] . status === 'success' ) {
var file = result [ 0 ] ;
if ( data . context . data ( 'type' ) === 'file' ) {
// update file data
data . context . attr ( 'data-mime' , file . mime ) . attr ( 'data-id' , file . id ) ;
var size = data . context . data ( 'size' ) ;
if ( size != file . size ) {
data . context . attr ( 'data-size' , file . size ) ;
data . context . find ( 'td.filesize' ) . text ( humanFileSize ( file . size ) ) ;
}
if ( FileList . loadingDone ) {
FileList . loadingDone ( file . name , file . id ) ;
}
} else {
// update upload counter ui
var uploadtext = data . context . find ( '.uploadtext' ) ;
var currentUploads = parseInt ( uploadtext . attr ( 'currentUploads' ) ) ;
currentUploads -= 1 ;
uploadtext . attr ( 'currentUploads' , currentUploads ) ;
if ( currentUploads === 0 ) {
var img = OC . imagePath ( 'core' , 'filetypes/folder.png' ) ;
data . context . find ( 'td.filename' ) . attr ( 'style' , 'background-image:url(' + img + ')' ) ;
uploadtext . text ( '' ) ;
uploadtext . hide ( ) ;
} else {
uploadtext . text ( currentUploads + ' ' + t ( 'files' , 'files uploading' ) ) ;
}
// update folder size
2013-08-17 15:07:18 +04:00
var size = parseInt ( data . context . data ( 'size' ) ) ;
size += parseInt ( file . size ) ;
2013-03-13 20:26:37 +04:00
data . context . attr ( 'data-size' , size ) ;
data . context . find ( 'td.filesize' ) . text ( humanFileSize ( size ) ) ;
}
}
}
} ) ;
2013-03-27 18:55:44 +04:00
file _upload _start . on ( 'fileuploadfail' , function ( e , data ) {
2013-03-13 20:26:37 +04:00
// only update the fileList if it exists
// cleanup files, error notification has been shown by fileupload code
var tr = data . context ;
if ( typeof tr === 'undefined' ) {
tr = $ ( 'tr' ) . filterAttr ( 'data-file' , data . files [ 0 ] . name ) ;
}
if ( tr . attr ( 'data-type' ) === 'dir' ) {
//cleanup uploading to a dir
var uploadtext = tr . find ( '.uploadtext' ) ;
var img = OC . imagePath ( 'core' , 'filetypes/folder.png' ) ;
tr . find ( 'td.filename' ) . attr ( 'style' , 'background-image:url(' + img + ')' ) ;
uploadtext . text ( '' ) ;
uploadtext . hide ( ) ; //TODO really hide already
} else {
//remove file
tr . fadeOut ( ) ;
tr . remove ( ) ;
}
} ) ;
2011-08-05 09:37:08 +04:00
$ ( '#notification' ) . hide ( ) ;
2013-01-19 01:38:44 +04:00
$ ( '#notification' ) . on ( 'click' , '.undo' , function ( ) {
2012-07-30 20:21:58 +04:00
if ( FileList . deleteFiles ) {
2011-10-17 22:39:01 +04:00
$ . each ( FileList . deleteFiles , function ( index , file ) {
2011-11-02 01:35:13 +04:00
$ ( 'tr' ) . filterAttr ( 'data-file' , file ) . show ( ) ;
2011-10-17 22:39:01 +04:00
} ) ;
FileList . deleteCanceled = true ;
FileList . deleteFiles = null ;
2012-07-30 20:21:58 +04:00
} else if ( FileList . replaceOldName && FileList . replaceNewName ) {
2012-09-05 08:12:11 +04:00
if ( FileList . replaceIsNewFile ) {
// Delete the new uploaded file
FileList . deleteCanceled = false ;
FileList . deleteFiles = [ FileList . replaceOldName ] ;
} else {
$ ( 'tr' ) . filterAttr ( 'data-file' , FileList . replaceOldName ) . show ( ) ;
}
$ ( 'tr' ) . filterAttr ( 'data-replace' , 'true' ) . remove ( ) ;
$ ( 'tr' ) . filterAttr ( 'data-file' , FileList . replaceNewName ) . show ( ) ;
2012-07-30 20:21:58 +04:00
FileList . replaceCanceled = true ;
FileList . replaceOldName = null ;
FileList . replaceNewName = null ;
2012-09-05 08:12:11 +04:00
FileList . replaceIsNewFile = null ;
2011-10-17 22:39:01 +04:00
}
2012-08-23 01:08:10 +04:00
FileList . lastAction = null ;
2013-01-05 02:34:09 +04:00
OC . Notification . hide ( ) ;
2011-08-04 02:22:44 +04:00
} ) ;
2013-02-09 20:20:08 +04:00
$ ( '#notification:first-child' ) . on ( 'click' , '.replace' , function ( ) {
2013-01-05 02:34:09 +04:00
OC . Notification . hide ( function ( ) {
2013-02-09 20:20:08 +04:00
FileList . replace ( $ ( '#notification > span' ) . attr ( 'data-oldName' ) , $ ( '#notification > span' ) . attr ( 'data-newName' ) , $ ( '#notification > span' ) . attr ( 'data-isNewFile' ) ) ;
2013-01-02 18:09:40 +04:00
} ) ;
2012-07-30 20:21:58 +04:00
} ) ;
2013-02-09 20:20:08 +04:00
$ ( '#notification:first-child' ) . on ( 'click' , '.suggest' , function ( ) {
$ ( 'tr' ) . filterAttr ( 'data-file' , $ ( '#notification > span' ) . attr ( 'data-oldName' ) ) . show ( ) ;
2013-01-05 02:34:09 +04:00
OC . Notification . hide ( ) ;
2012-09-05 08:12:11 +04:00
} ) ;
2013-02-09 20:20:08 +04:00
$ ( '#notification:first-child' ) . on ( 'click' , '.cancel' , function ( ) {
if ( $ ( '#notification > span' ) . attr ( 'data-isNewFile' ) ) {
2012-09-05 08:12:11 +04:00
FileList . deleteCanceled = false ;
2013-02-09 20:20:08 +04:00
FileList . deleteFiles = [ $ ( '#notification > span' ) . attr ( 'data-oldName' ) ] ;
2012-09-05 08:12:11 +04:00
}
2012-07-30 20:21:58 +04:00
} ) ;
2012-09-23 05:16:52 +04:00
FileList . useUndo = ( window . onbeforeunload ) ? true : false ;
2011-08-04 02:22:44 +04:00
$ ( window ) . bind ( 'beforeunload' , function ( ) {
2012-07-30 20:21:58 +04:00
if ( FileList . lastAction ) {
FileList . lastAction ( ) ;
}
2011-08-04 02:22:44 +04:00
} ) ;
2012-11-05 21:42:44 +04:00
$ ( window ) . unload ( function ( ) {
$ ( window ) . trigger ( 'beforeunload' ) ;
} ) ;
2013-07-03 21:50:03 +04:00
2013-08-29 23:56:14 +04:00
function parseHashQuery ( ) {
var hash = window . location . hash ,
pos = hash . indexOf ( '?' ) ,
query ;
if ( pos >= 0 ) {
return hash . substr ( pos + 1 ) ;
}
return '' ;
}
function parseCurrentDirFromUrl ( ) {
var query = parseHashQuery ( ) ,
params ,
dir = '/' ;
// try and parse from URL hash first
if ( query ) {
params = OC . parseQueryString ( query ) ;
}
// else read from query attributes
if ( ! params ) {
params = OC . parseQueryString ( location . search ) ;
}
return ( params && params . dir ) || '/' ;
}
// fallback to hashchange when no history support
if ( ! window . history . pushState ) {
$ ( window ) . on ( 'hashchange' , function ( ) {
FileList . changeDirectory ( parseCurrentDirFromUrl ( ) , false ) ;
} ) ;
}
2013-08-17 15:07:18 +04:00
window . onpopstate = function ( e ) {
var targetDir ;
if ( e . state && e . state . dir ) {
targetDir = e . state . dir ;
}
else {
// read from URL
2013-08-29 23:56:14 +04:00
targetDir = parseCurrentDirFromUrl ( ) ;
2013-08-17 15:07:18 +04:00
}
if ( targetDir ) {
FileList . changeDirectory ( targetDir , false ) ;
}
}
2013-08-29 23:56:14 +04:00
if ( parseInt ( $ ( '#ajaxLoad' ) . val ( ) , 10 ) === 1 ) {
// need to initially switch the dir to the one from the hash (IE8)
FileList . changeDirectory ( parseCurrentDirFromUrl ( ) , false , true ) ;
}
2013-07-03 21:50:03 +04:00
FileList . createFileSummary ( ) ;
2011-08-04 02:22:44 +04:00
} ) ;