mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
NEW UploadField displays a warning before overwriting files (only relevant if config.yml: Upload:replaceFile = true).
This commit is contained in:
parent
5bbe48b799
commit
22c7bbfcd4
@ -13,4 +13,5 @@ UploadField:
|
|||||||
downloadTemplateName: 'ss-uploadfield-downloadtemplate'
|
downloadTemplateName: 'ss-uploadfield-downloadtemplate'
|
||||||
fileEditFields:
|
fileEditFields:
|
||||||
fileEditActions:
|
fileEditActions:
|
||||||
fileEditValidator:
|
fileEditValidator:
|
||||||
|
overwriteWarning: true # Warning before overwriting existing file (only relevant when Upload: replaceFile is true)
|
@ -52,9 +52,9 @@ body.cms.ss-uploadfield-edit-iframe .fieldholder-small label, .composite.ss-asse
|
|||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions { position: absolute; top: 0; right: 0; left: 0; z-index: 0; color: #f00; font-size: 14px; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions { position: absolute; top: 0; right: 0; left: 0; z-index: 0; color: #f00; font-size: 14px; }
|
||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button { background: none; border: 0; -webkit-box-shadow: none; -moz-box-shadow: none; box-shadow: none; text-shadow: none; color: white; float: right; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button { background: none; border: 0; -webkit-box-shadow: none; -moz-box-shadow: none; box-shadow: none; text-shadow: none; color: white; float: right; }
|
||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-delete { display: none; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-delete { display: none; }
|
||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-cancel { -webkit-border-radius: 0; -moz-border-radius: 0; -ms-border-radius: 0; -o-border-radius: 0; border-radius: 0; border-left: 1px solid rgba(255, 255, 255, 0.2); margin-top: 0px; cursor: pointer; opacity: 0.9; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-cancel, .ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-overwrite-warning { -webkit-border-radius: 0; -moz-border-radius: 0; -ms-border-radius: 0; -o-border-radius: 0; border-radius: 0; border-left: 1px solid rgba(255, 255, 255, 0.2); margin-top: 0px; cursor: pointer; opacity: 0.9; }
|
||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-cancel:hover { opacity: 1; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-cancel:hover, .ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-overwrite-warning:hover { opacity: 1; }
|
||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-cancel .ui-icon { display: block; margin: 0; position: realtive; top: 8px; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-cancel .ui-icon, .ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-overwrite-warning .ui-icon { display: block; margin: 0; position: realtive; top: 8px; }
|
||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-edit { opacity: 0.9; padding-top: 1px; padding-bottom: 0; height: 100%; -webkit-border-radius: 0; -moz-border-radius: 0; -ms-border-radius: 0; -o-border-radius: 0; border-radius: 0; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-edit { opacity: 0.9; padding-top: 1px; padding-bottom: 0; height: 100%; -webkit-border-radius: 0; -moz-border-radius: 0; -ms-border-radius: 0; -o-border-radius: 0; border-radius: 0; }
|
||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-edit.ui-state-hover { background: none; opacity: 1; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-edit.ui-state-hover { background: none; opacity: 1; }
|
||||||
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-edit.ui-state-hover span.toggle-details { opacity: 1; }
|
.ss-assetuploadfield .ss-uploadfield-files .ss-uploadfield-item-actions .ss-ui-button.ss-uploadfield-item-edit.ui-state-hover span.toggle-details { opacity: 1; }
|
||||||
|
@ -225,9 +225,9 @@ You can also configure the underlying `[api:Upload]` class, by using the YAML co
|
|||||||
|
|
||||||
:::yaml
|
:::yaml
|
||||||
Upload:
|
Upload:
|
||||||
# Globally disables automatic renaming of files
|
# Globally disables automatic renaming of files and displays a warning before overwriting an existing file
|
||||||
replaceFile: true
|
replaceFile: true
|
||||||
|
|
||||||
## TODO: Using the UploadField in a frontend form
|
## TODO: Using the UploadField in a frontend form
|
||||||
|
|
||||||
*At this moment the UploadField not yet fully supports being used on a frontend
|
*At this moment the UploadField not yet fully supports being used on a frontend
|
||||||
|
@ -153,24 +153,27 @@ class Upload extends Controller {
|
|||||||
|
|
||||||
// if filename already exists, version the filename (e.g. test.gif to test1.gif)
|
// if filename already exists, version the filename (e.g. test.gif to test1.gif)
|
||||||
if(!$this->replaceFile) {
|
if(!$this->replaceFile) {
|
||||||
while(file_exists("$base/$relativeFilePath")) {
|
while(file_exists("$base/$relativeFilePath")) {
|
||||||
$i = isset($i) ? ($i+1) : 2;
|
$i = isset($i) ? ($i+1) : 2;
|
||||||
$oldFilePath = $relativeFilePath;
|
$oldFilePath = $relativeFilePath;
|
||||||
// make sure archives retain valid extensions
|
// make sure archives retain valid extensions
|
||||||
if(substr($relativeFilePath, strlen($relativeFilePath) - strlen('.tar.gz')) == '.tar.gz' ||
|
if(substr($relativeFilePath, strlen($relativeFilePath) - strlen('.tar.gz')) == '.tar.gz' ||
|
||||||
substr($relativeFilePath, strlen($relativeFilePath) - strlen('.tar.bz2')) == '.tar.bz2') {
|
substr($relativeFilePath, strlen($relativeFilePath) - strlen('.tar.bz2')) == '.tar.bz2') {
|
||||||
$relativeFilePath = preg_replace('/[0-9]*(\.tar\.[^.]+$)/', $i . '\\1', $relativeFilePath);
|
$relativeFilePath = preg_replace('/[0-9]*(\.tar\.[^.]+$)/', $i . '\\1', $relativeFilePath);
|
||||||
} else if (strpos($relativeFilePath, '.') !== false) {
|
} else if (strpos($relativeFilePath, '.') !== false) {
|
||||||
$relativeFilePath = preg_replace('/[0-9]*(\.[^.]+$)/', $i . '\\1', $relativeFilePath);
|
$relativeFilePath = preg_replace('/[0-9]*(\.[^.]+$)/', $i . '\\1', $relativeFilePath);
|
||||||
} else if (strpos($relativeFilePath, '_') !== false) {
|
} else if (strpos($relativeFilePath, '_') !== false) {
|
||||||
$relativeFilePath = preg_replace('/_([^_]+$)/', '_'.$i, $relativeFilePath);
|
$relativeFilePath = preg_replace('/_([^_]+$)/', '_'.$i, $relativeFilePath);
|
||||||
} else {
|
} else {
|
||||||
$relativeFilePath .= '_'.$i;
|
$relativeFilePath .= '_'.$i;
|
||||||
}
|
}
|
||||||
if($oldFilePath == $relativeFilePath && $i > 2) {
|
if($oldFilePath == $relativeFilePath && $i > 2) {
|
||||||
user_error("Couldn't fix $relativeFilePath with $i tries", E_USER_ERROR);
|
user_error("Couldn't fix $relativeFilePath with $i tries", E_USER_ERROR);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
//reset the ownerID to the current member when replacing files
|
||||||
|
$this->file->OwnerID = (Member::currentUser() ? Member::currentUser()->ID : 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(file_exists($tmpFile['tmp_name']) && copy($tmpFile['tmp_name'], "$base/$relativeFilePath")) {
|
if(file_exists($tmpFile['tmp_name']) && copy($tmpFile['tmp_name'], "$base/$relativeFilePath")) {
|
||||||
@ -178,6 +181,7 @@ class Upload extends Controller {
|
|||||||
// This is to prevent it from trying to rename the file
|
// This is to prevent it from trying to rename the file
|
||||||
$this->file->Name = basename($relativeFilePath);
|
$this->file->Name = basename($relativeFilePath);
|
||||||
$this->file->write();
|
$this->file->write();
|
||||||
|
$this->extend('onAfterLoad', $this->file); //to allow extensions to e.g. create a version after an upload
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
$this->errors[] = _t('File.NOFILESIZE', 'Filesize is zero bytes.');
|
$this->errors[] = _t('File.NOFILESIZE', 'Filesize is zero bytes.');
|
||||||
|
@ -141,7 +141,11 @@ class UploadField extends FileField {
|
|||||||
* @example 'getCMSValidator'
|
* @example 'getCMSValidator'
|
||||||
* @var string
|
* @var string
|
||||||
*/
|
*/
|
||||||
'fileEditValidator' => null
|
'fileEditValidator' => null,
|
||||||
|
/**
|
||||||
|
* Show a warning when overwriting a file
|
||||||
|
*/
|
||||||
|
'overwriteWarning' => true,
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -416,6 +420,17 @@ class UploadField extends FileField {
|
|||||||
if (is_numeric($config['maxNumberOfFiles']) && $this->getItems()->count()) {
|
if (is_numeric($config['maxNumberOfFiles']) && $this->getItems()->count()) {
|
||||||
$configOverwrite['maxNumberOfFiles'] = $config['maxNumberOfFiles'] - $this->getItems()->count();
|
$configOverwrite['maxNumberOfFiles'] = $config['maxNumberOfFiles'] - $this->getItems()->count();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//get all the existing files in the current folder
|
||||||
|
if ($this->getConfig('overwriteWarning')) {
|
||||||
|
$folder = Folder::find_or_make($this->getFolderName());
|
||||||
|
$files = glob( $folder->getFullPath() . '/*' );
|
||||||
|
$config['existingFiles'] = array_map("basename", $files);;
|
||||||
|
|
||||||
|
//add overwrite warning error message to the config object sent to Javascript
|
||||||
|
$config['errorMessages']['overwriteWarning'] =
|
||||||
|
_t('UploadField.OVERWRITEWARNING','File with the same name already exists');
|
||||||
|
}
|
||||||
|
|
||||||
$config = array_merge($config, $this->ufConfig, $configOverwrite);
|
$config = array_merge($config, $this->ufConfig, $configOverwrite);
|
||||||
|
|
||||||
|
@ -48,6 +48,39 @@
|
|||||||
|
|
||||||
return result;
|
return result;
|
||||||
},
|
},
|
||||||
|
_onSend: function (e, data) {
|
||||||
|
//check the array of existing files to see if we are trying to upload a file that already exists
|
||||||
|
var that = this;
|
||||||
|
var config = $('div.ss-upload').entwine('ss').getConfig();
|
||||||
|
var existingFiles = [];
|
||||||
|
if (typeof (config.existingFiles) !== "undefined") {
|
||||||
|
existingFiles = config.existingFiles;
|
||||||
|
}
|
||||||
|
|
||||||
|
var fileExists = false;
|
||||||
|
jQuery.each(existingFiles,function(){
|
||||||
|
if ($(this)[0].toLowerCase() === data.files[0].name.toLowerCase()) {
|
||||||
|
fileExists = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (fileExists) {
|
||||||
|
//display the dialogs with the question to overwrite or not
|
||||||
|
data.context.find('.ss-uploadfield-item-status').text(config.errorMessages.overwriteWarning).css('max-width','75%');
|
||||||
|
data.context.find('.ss-uploadfield-item-progress').hide();
|
||||||
|
data.context.find('.ss-uploadfield-item-overwrite').show();
|
||||||
|
data.context.find('.ss-uploadfield-item-overwrite-warning').on('click', function(){
|
||||||
|
data.context.find('.ss-uploadfield-item-progress').show();
|
||||||
|
data.context.find('.ss-uploadfield-item-overwrite').hide();
|
||||||
|
|
||||||
|
//upload only if the "overwrite" button is clicked
|
||||||
|
$.blueimpUI.fileupload.prototype._onSend.call(that, e, data);
|
||||||
|
});
|
||||||
|
} else { //regular file upload
|
||||||
|
return $.blueimpUI.fileupload.prototype._onSend.call(that, e, data);
|
||||||
|
}
|
||||||
|
},
|
||||||
_onAlways: function (jqXHRorResult, textStatus, jqXHRorError, options) {
|
_onAlways: function (jqXHRorResult, textStatus, jqXHRorError, options) {
|
||||||
$.blueimpUI.fileupload.prototype._onAlways.call(this, jqXHRorResult, textStatus, jqXHRorError, options);
|
$.blueimpUI.fileupload.prototype._onAlways.call(this, jqXHRorResult, textStatus, jqXHRorError, options);
|
||||||
if (this._active === 0) {
|
if (this._active === 0) {
|
||||||
|
@ -21,6 +21,8 @@ window.tmpl.cache['ss-uploadfield-uploadtemplate'] = tmpl(
|
|||||||
'{% } %}' +
|
'{% } %}' +
|
||||||
'{% } %}' +
|
'{% } %}' +
|
||||||
'<div class="ss-uploadfield-item-cancel cancel"><button data-icon="deleteLight" class="ss-uploadfield-item-cancel" title="' + ss.i18n._t('UploadField.CANCELREMOVE', 'Cancel/Remove') + '">' + ss.i18n._t('UploadField.CANCEL', 'Cancel') + '</button></div>' +
|
'<div class="ss-uploadfield-item-cancel cancel"><button data-icon="deleteLight" class="ss-uploadfield-item-cancel" title="' + ss.i18n._t('UploadField.CANCELREMOVE', 'Cancel/Remove') + '">' + ss.i18n._t('UploadField.CANCEL', 'Cancel') + '</button></div>' +
|
||||||
|
'<div class="ss-uploadfield-item-overwrite hide ">'+
|
||||||
|
'<button data-icon="drive-upload" class="ss-uploadfield-item-overwrite-warning" title="' + ss.i18n._t('UploadField.OVERWRITE', 'Overwrite') + '">' + ss.i18n._t('UploadField.OVERWRITE', 'Overwrite') +
|
||||||
'</div>' +
|
'</div>' +
|
||||||
'</div>' +
|
'</div>' +
|
||||||
'</li>' +
|
'</li>' +
|
||||||
|
@ -34,6 +34,7 @@ if(typeof(ss) == 'undefined' || typeof(ss.i18n) == 'undefined') {
|
|||||||
'UploadField.EMPTYRESULT': 'Leere Datei erhalten',
|
'UploadField.EMPTYRESULT': 'Leere Datei erhalten',
|
||||||
'UploadField.LOADING': 'Lädt ...',
|
'UploadField.LOADING': 'Lädt ...',
|
||||||
'UploadField.Editing': 'Bearbeite ...',
|
'UploadField.Editing': 'Bearbeite ...',
|
||||||
'UploadField.Uploaded': 'Hochgeladen'
|
'UploadField.Uploaded': 'Hochgeladen',
|
||||||
|
'UploadField.OVERWRITEWARNING': 'Datei mit diesem Namen existiert bereits'
|
||||||
});
|
});
|
||||||
}
|
}
|
@ -35,6 +35,7 @@ if(typeof(ss) == 'undefined' || typeof(ss.i18n) == 'undefined') {
|
|||||||
'UploadField.EMPTYRESULT': 'Empty file upload result',
|
'UploadField.EMPTYRESULT': 'Empty file upload result',
|
||||||
'UploadField.LOADING': 'Loading ...',
|
'UploadField.LOADING': 'Loading ...',
|
||||||
'UploadField.Editing': 'Editing ...',
|
'UploadField.Editing': 'Editing ...',
|
||||||
'UploadField.Uploaded': 'Uploaded'
|
'UploadField.Uploaded': 'Uploaded',
|
||||||
|
'UploadField.OVERWRITEWARNING': 'File with the same name already exists'
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
// TODO tmp hack until we have permissions and can disable delete
|
// TODO tmp hack until we have permissions and can disable delete
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
&.ss-uploadfield-item-cancel{
|
&.ss-uploadfield-item-cancel, &.ss-uploadfield-item-overwrite-warning {
|
||||||
@include border-radius(0);
|
@include border-radius(0);
|
||||||
border-left:1px solid rgba(#fff, 0.2);
|
border-left:1px solid rgba(#fff, 0.2);
|
||||||
margin-top:0px;
|
margin-top:0px;
|
||||||
|
Loading…
Reference in New Issue
Block a user