mirror of
https://github.com/silverstripe/silverstripe-reports
synced 2024-10-22 11:05:53 +02:00
NEW Clickable URL preview in CMS
- Refactored SiteTreeURLSegmentField to render controls in template rather than JS for better clientside performance, and cleaner behaviour. - Added dynamic ellipsis to start of URL, to retain most relevant part of the URL (the last bits) - Added "suffix" setting to field, which defaults to ?stage=Stage - Removed prefix from edit view to leave more room for URL Thanks to @sunnysideup for getting this started in https://github.com/silverstripe/silverstripe-cms/pull/269
This commit is contained in:
parent
931b726589
commit
00097a5d5d
@ -15,7 +15,7 @@ class SiteTreeURLSegmentField extends TextField {
|
|||||||
/**
|
/**
|
||||||
* @var string
|
* @var string
|
||||||
*/
|
*/
|
||||||
protected $helpText, $urlPrefix;
|
protected $helpText, $urlPrefix, $urlSuffix;
|
||||||
|
|
||||||
static $allowed_actions = array(
|
static $allowed_actions = array(
|
||||||
'suggest'
|
'suggest'
|
||||||
@ -25,6 +25,16 @@ class SiteTreeURLSegmentField extends TextField {
|
|||||||
return rawurldecode($this->value);
|
return rawurldecode($this->value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getAttributes() {
|
||||||
|
return array_merge(
|
||||||
|
parent::getAttributes(),
|
||||||
|
array(
|
||||||
|
'data-prefix' => $this->getURLPrefix(),
|
||||||
|
'data-suffix' => '?stage=Stage'
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
public function Field($properties = array()) {
|
public function Field($properties = array()) {
|
||||||
Requirements::javascript(CMS_DIR . '/javascript/SiteTreeURLSegmentField.js');
|
Requirements::javascript(CMS_DIR . '/javascript/SiteTreeURLSegmentField.js');
|
||||||
Requirements::add_i18n_javascript(CMS_DIR . '/javascript/lang', false, true);
|
Requirements::add_i18n_javascript(CMS_DIR . '/javascript/lang', false, true);
|
||||||
@ -85,9 +95,20 @@ class SiteTreeURLSegmentField extends TextField {
|
|||||||
return $this->urlPrefix;
|
return $this->urlPrefix;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getURLSuffix() {
|
||||||
|
return $this->urlSuffix;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setURLSuffix($suffix) {
|
||||||
|
$this->urlSuffix = $suffix;
|
||||||
|
}
|
||||||
|
|
||||||
public function Type() {
|
public function Type() {
|
||||||
return 'text urlsegment';
|
return 'text urlsegment';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getURL() {
|
||||||
|
return Controller::join_links($this->getURLPrefix(), $this->Value(), $this->getURLSuffix());
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1829,11 +1829,8 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
(self::nested_urls() && $this->ParentID ? $this->Parent()->RelativeLink(true) : null)
|
(self::nested_urls() && $this->ParentID ? $this->Parent()->RelativeLink(true) : null)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
$url = (strlen($baseLink) > 36) ? "..." .substr($baseLink, -32) : $baseLink;
|
|
||||||
$urlsegment = new SiteTreeURLSegmentField("URLSegment", $this->fieldLabel('URLSegment'));
|
$urlsegment = new SiteTreeURLSegmentField("URLSegment", $this->fieldLabel('URLSegment'));
|
||||||
$urlsegment->setURLPrefix($url);
|
$urlsegment->setURLPrefix($baseLink);
|
||||||
$helpText = (self::nested_urls() && count($this->Children())) ? $this->fieldLabel('LinkChangeNote') : '';
|
$helpText = (self::nested_urls() && count($this->Children())) ? $this->fieldLabel('LinkChangeNote') : '';
|
||||||
if(!URLSegmentFilter::$default_allow_multibyte) {
|
if(!URLSegmentFilter::$default_allow_multibyte) {
|
||||||
$helpText .= $helpText ? '<br />' : '';
|
$helpText .= $helpText ? '<br />' : '';
|
||||||
|
@ -22,10 +22,11 @@
|
|||||||
|
|
||||||
/** ------------------------------------------------------------------ URLSegment field ----------------------------------------------------------------- */
|
/** ------------------------------------------------------------------ URLSegment field ----------------------------------------------------------------- */
|
||||||
.field.urlsegment.loading { background: url(../images/loading.gif) no-repeat 162px 8px; }
|
.field.urlsegment.loading { background: url(../images/loading.gif) no-repeat 162px 8px; }
|
||||||
.field.urlsegment .prefix, .field.urlsegment .preview { padding-top: 8px; display: inline-block; }
|
.field.urlsegment .preview { padding-top: 8px; display: inline-block; }
|
||||||
.field.urlsegment .prefix { color: #777; }
|
.field.urlsegment input.text { width: 250px; }
|
||||||
.field.urlsegment .cancel, .field.urlsegment .update, .field.urlsegment .edit { margin-left: 7px; }
|
.field.urlsegment input.text, .field.urlsegment .cancel, .field.urlsegment .update, .field.urlsegment .edit { margin-right: 8px; }
|
||||||
.field.urlsegment .help { margin-left: 0; }
|
.field.urlsegment .help { margin-left: 0; }
|
||||||
|
.field.urlsegment .edit-holder { display: none; }
|
||||||
|
|
||||||
#Form_EditForm #Title .update { margin-left: 7px; }
|
#Form_EditForm #Title .update { margin-left: 7px; }
|
||||||
|
|
||||||
|
@ -3,214 +3,124 @@
|
|||||||
/**
|
/**
|
||||||
* Class: .field.urlsegment
|
* Class: .field.urlsegment
|
||||||
*
|
*
|
||||||
* Input validation on the URLSegment field
|
* Provides enhanced functionality (read-only/edit switch) and
|
||||||
|
* input validation on the URLSegment field
|
||||||
*/
|
*/
|
||||||
$('.field.urlsegment:not(.readonly)').entwine({
|
$('.field.urlsegment:not(.readonly)').entwine({
|
||||||
|
|
||||||
/**
|
// Roughly matches the field width including edit button
|
||||||
* Constructor: onmatch
|
MaxPreviewLength: 55,
|
||||||
*/
|
|
||||||
|
Ellipsis: '...',
|
||||||
|
|
||||||
onmatch : function() {
|
onmatch : function() {
|
||||||
// Only initialize the field if it contains an editable field.
|
// Only initialize the field if it contains an editable field.
|
||||||
// This ensures we don't get bogus previews on readonly fields.
|
// This ensures we don't get bogus previews on readonly fields.
|
||||||
if(this.find(':text').length) {
|
if(this.find(':text').length) this.toggleEdit(false);
|
||||||
this._addActions(); // add elements and actions for editing
|
this.redraw();
|
||||||
this.edit(); // toggle
|
|
||||||
this._autoInputWidth(); // set width of input field
|
|
||||||
}
|
|
||||||
|
|
||||||
this._super();
|
this._super();
|
||||||
},
|
},
|
||||||
onunmatch: function() {
|
|
||||||
this._super();
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Function: edit
|
|
||||||
*
|
|
||||||
* Toggles the edit state of the field
|
|
||||||
*
|
|
||||||
* Return URLSegemnt val()
|
|
||||||
*
|
|
||||||
* Parameters:
|
|
||||||
* (Bool) auto (optional, triggers a second toggle)
|
|
||||||
*/
|
|
||||||
edit: function(auto) {
|
|
||||||
|
|
||||||
|
redraw: function() {
|
||||||
var field = this.find(':text'),
|
var field = this.find(':text'),
|
||||||
holder = this.find('.preview'),
|
url = field.data('prefix') + field.val(),
|
||||||
edit = this.find('.edit'),
|
previewUrl = url;
|
||||||
update = this.find('.update'),
|
|
||||||
cancel = this.find('.cancel'),
|
|
||||||
help = this.find('.help');
|
|
||||||
|
|
||||||
// transfer current value to holder
|
// Truncate URL if required (ignoring the suffix, retaining the full value)
|
||||||
holder.text(field.val());
|
if(url.length > this.getMaxPreviewLength()) {
|
||||||
|
previewUrl = this.getEllipsis() + url.substr(url.length - this.getMaxPreviewLength(), url.length);
|
||||||
// toggle elements
|
|
||||||
if (field.is(':visible')) {
|
|
||||||
update.hide();
|
|
||||||
cancel.hide();
|
|
||||||
field.hide();
|
|
||||||
holder.show();
|
|
||||||
edit.show();
|
|
||||||
help.hide();
|
|
||||||
} else {
|
|
||||||
edit.hide();
|
|
||||||
holder.hide();
|
|
||||||
field.show();
|
|
||||||
update.show();
|
|
||||||
cancel.show();
|
|
||||||
help.show();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// field updated from another fields value
|
// Transfer current value to holder
|
||||||
// reset to original state
|
this.find('.preview').attr('href', url + field.data('suffix')).text(previewUrl);
|
||||||
if (auto) this.edit();
|
},
|
||||||
|
|
||||||
return field.val();
|
/**
|
||||||
|
* @param Boolean
|
||||||
|
*/
|
||||||
|
toggleEdit: function(toggle) {
|
||||||
|
var field = this.find(':text');
|
||||||
|
|
||||||
|
this.find('.preview-holder')[toggle ? 'hide' : 'show']();
|
||||||
|
this.find('.edit-holder')[toggle ? 'show' : 'hide']();
|
||||||
|
|
||||||
|
if(toggle) {
|
||||||
|
field.data("origval", field.val()); //retain current value for cancel
|
||||||
|
field.focus();
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function: update
|
|
||||||
*
|
|
||||||
* Commits the change of the URLSegment to the field
|
* Commits the change of the URLSegment to the field
|
||||||
* Optional: pass in (String)
|
* Optional: pass in (String) to update the URLSegment
|
||||||
* to update the URLSegment
|
|
||||||
*/
|
*/
|
||||||
update: function() {
|
update: function() {
|
||||||
|
|
||||||
var self = this,
|
var self = this,
|
||||||
field = this.find(':text'),
|
field = this.find(':text'),
|
||||||
holder = this.find('.preview'),
|
currentVal = field.data('origval'),
|
||||||
currentVal = holder.text(),
|
title = arguments[0],
|
||||||
updateVal,
|
updateVal = (title && title !== "") ? title : field.val();
|
||||||
title = arguments[0];
|
|
||||||
|
|
||||||
if (title && title !== "") {
|
|
||||||
updateVal = title;
|
|
||||||
} else {
|
|
||||||
updateVal = field.val();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (currentVal != updateVal) {
|
if (currentVal != updateVal) {
|
||||||
self.addClass('loading');
|
this.addClass('loading');
|
||||||
self.suggest(updateVal, function(data) {
|
this.suggest(updateVal, function(data) {
|
||||||
var newVal = decodeURIComponent(data.value);
|
field.val(decodeURIComponent(data.value));
|
||||||
field.val(newVal);
|
self.toggleEdit(false);
|
||||||
self.edit(title);
|
|
||||||
self.removeClass('loading');
|
self.removeClass('loading');
|
||||||
|
self.redraw();
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
self.edit();
|
this.toggleEdit(false);
|
||||||
|
this.redraw();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function: cancel
|
|
||||||
*
|
|
||||||
* Cancels any changes to the field
|
* Cancels any changes to the field
|
||||||
*
|
|
||||||
* Return URLSegemnt val()
|
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
cancel: function() {
|
cancel: function() {
|
||||||
var field = this.find(':text'),
|
var field = this.find(':text');
|
||||||
holder = this.find('.preview');
|
field.val(field.data("origval"));
|
||||||
field.val(holder.text());
|
this.toggleEdit(false);
|
||||||
this.edit();
|
|
||||||
|
|
||||||
return field.val();
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function: suggest
|
|
||||||
*
|
|
||||||
* Return a value matching the criteria.
|
* Return a value matching the criteria.
|
||||||
*
|
*
|
||||||
* Parameters:
|
* @param (String)
|
||||||
* (String) val
|
* @param (Function)
|
||||||
* (Function) callback
|
|
||||||
*/
|
*/
|
||||||
suggest: function(val, callback) {
|
suggest: function(val, callback) {
|
||||||
var field = this.find(':text'), urlParts = $.path.parseUrl(this.closest('form').attr('action')),
|
var field = this.find(':text'),
|
||||||
|
urlParts = $.path.parseUrl(this.closest('form').attr('action')),
|
||||||
url = urlParts.hrefNoSearch + '/field/' + field.attr('name') + '/suggest/?value=' + encodeURIComponent(val);
|
url = urlParts.hrefNoSearch + '/field/' + field.attr('name') + '/suggest/?value=' + encodeURIComponent(val);
|
||||||
if(urlParts.search) url += '&' + urlParts.search.replace(/^\?/, '');
|
if(urlParts.search) url += '&' + urlParts.search.replace(/^\?/, '');
|
||||||
|
|
||||||
$.get(
|
$.get(url, function(data) {callback.apply(this, arguments);});
|
||||||
url,
|
}
|
||||||
function(data) {callback.apply(this, arguments);}
|
});
|
||||||
);
|
|
||||||
|
|
||||||
},
|
$('.field.urlsegment .edit').entwine({
|
||||||
|
onclick: function(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
this.closest('.field').toggleEdit(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
/**
|
$('.field.urlsegment .update').entwine({
|
||||||
* Function: _addActions
|
onclick: function(e) {
|
||||||
*
|
e.preventDefault();
|
||||||
* Utility to add edit buttons and actions
|
this.closest('.field').update();
|
||||||
*
|
}
|
||||||
*/
|
});
|
||||||
_addActions: function() {
|
|
||||||
var self = this,
|
|
||||||
field = this.find(':text'),
|
|
||||||
preview,
|
|
||||||
editAction,
|
|
||||||
updateAction,
|
|
||||||
cancelAction;
|
|
||||||
|
|
||||||
// element to display non-editable text
|
$('.field.urlsegment .cancel').entwine({
|
||||||
preview = $('<span />', {
|
onclick: function(e) {
|
||||||
'class': 'preview'
|
e.preventDefault();
|
||||||
});
|
this.closest('.field').cancel();
|
||||||
|
|
||||||
// edit button
|
|
||||||
editAction = $('<button />', {
|
|
||||||
'class': 'ss-ui-button ss-ui-button-small edit',
|
|
||||||
'text': ss.i18n._t('URLSEGMENT.Edit', 'Edit'),
|
|
||||||
'click': function(e) {
|
|
||||||
e.preventDefault();
|
|
||||||
self.edit();
|
|
||||||
self.find(':text').focus();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// update button
|
|
||||||
updateAction = $('<button />', {
|
|
||||||
'class': 'update ss-ui-button-small',
|
|
||||||
'text': ss.i18n._t('URLSEGMENT.OK', 'OK'),
|
|
||||||
'click': function(e) {
|
|
||||||
e.preventDefault();
|
|
||||||
self.update();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// cancel button
|
|
||||||
cancelAction = $('<button />', {
|
|
||||||
'class': 'cancel ss-ui-action-minor ss-ui-button-small',
|
|
||||||
'href': '#',
|
|
||||||
'text': ss.i18n._t('URLSEGMENT.Cancel', 'Cancel'),
|
|
||||||
'click': function(e) {
|
|
||||||
e.preventDefault();
|
|
||||||
self.cancel();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// insert elements
|
|
||||||
preview.insertAfter('.prefix');
|
|
||||||
editAction.insertAfter(field);
|
|
||||||
cancelAction.insertAfter(field);
|
|
||||||
updateAction.insertAfter(field);
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Function: _autoInputWidth
|
|
||||||
*
|
|
||||||
* Sets the width of input so it lines up with the other fields
|
|
||||||
*/
|
|
||||||
_autoInputWidth: function() {
|
|
||||||
var field = this.find(':text');
|
|
||||||
field.width((field.width() + 15) - this.find('.prefix').width());
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
}(jQuery));
|
}(jQuery));
|
||||||
|
@ -98,23 +98,26 @@
|
|||||||
background: url(../images/loading.gif) no-repeat 162px 8px;
|
background: url(../images/loading.gif) no-repeat 162px 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.prefix,
|
|
||||||
.preview {
|
.preview {
|
||||||
padding-top: 8px;
|
padding-top: 8px;
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.prefix {
|
input.text {
|
||||||
color: #777;
|
width: 250px; // ensure there's enough room for buttons
|
||||||
}
|
}
|
||||||
|
|
||||||
.cancel, .update, .edit {
|
input.text, .cancel, .update, .edit {
|
||||||
margin-left: 7px;
|
margin-right: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.help {
|
.help {
|
||||||
margin-left: 0;
|
margin-left: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.edit-holder {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#Form_EditForm #Title .update {
|
#Form_EditForm #Title .update {
|
||||||
|
@ -1,4 +1,18 @@
|
|||||||
<span class="prefix">$URLPrefix</span><input $AttributesHTML />
|
<div class="preview-holder">
|
||||||
<% if HelpText %>
|
<a class="preview" href="$URL" target="_blank">
|
||||||
<p class="help">$HelpText</p>
|
$URL
|
||||||
<% end_if %>
|
</a>
|
||||||
|
<button class="ss-ui-button ss-ui-button-small edit">
|
||||||
|
<% _t('URLSegmentField.Edit', 'Edit') %>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="edit-holder">
|
||||||
|
<input $AttributesHTML />
|
||||||
|
<button class="update ss-ui-button-small">
|
||||||
|
<% _t('URLSegmentField.OK', 'OK') %>
|
||||||
|
</button>
|
||||||
|
<button class="cancel ss-ui-button-small ss-ui-action-minor">
|
||||||
|
<% _t('URLSegmentField.Cancel', 'Cancel') %>
|
||||||
|
</button>
|
||||||
|
<% if HelpText %><p class="help">$HelpText</p><% end_if %>
|
||||||
|
</div>
|
Loading…
Reference in New Issue
Block a user