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:
Ingo Schommer 2013-02-04 00:44:50 +01:00
parent 931b726589
commit 00097a5d5d
6 changed files with 128 additions and 182 deletions

View File

@ -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());
}
} }

View File

@ -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 />' : '';

View File

@ -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; }

View File

@ -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);}
);
},
/**
* Function: _addActions
*
* Utility to add edit buttons and actions
*
*/
_addActions: function() {
var self = this,
field = this.find(':text'),
preview,
editAction,
updateAction,
cancelAction;
// element to display non-editable text
preview = $('<span />', {
'class': 'preview'
}); });
// edit button $('.field.urlsegment .edit').entwine({
editAction = $('<button />', { onclick: function(e) {
'class': 'ss-ui-button ss-ui-button-small edit',
'text': ss.i18n._t('URLSEGMENT.Edit', 'Edit'),
'click': function(e) {
e.preventDefault(); e.preventDefault();
self.edit(); this.closest('.field').toggleEdit(true);
self.find(':text').focus();
} }
}); });
// update button $('.field.urlsegment .update').entwine({
updateAction = $('<button />', { onclick: function(e) {
'class': 'update ss-ui-button-small',
'text': ss.i18n._t('URLSEGMENT.OK', 'OK'),
'click': function(e) {
e.preventDefault(); e.preventDefault();
self.update(); this.closest('.field').update();
} }
}); });
// cancel button $('.field.urlsegment .cancel').entwine({
cancelAction = $('<button />', { onclick: function(e) {
'class': 'cancel ss-ui-action-minor ss-ui-button-small',
'href': '#',
'text': ss.i18n._t('URLSEGMENT.Cancel', 'Cancel'),
'click': function(e) {
e.preventDefault(); e.preventDefault();
self.cancel(); this.closest('.field').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));

View File

@ -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 {

View File

@ -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>