2007-07-19 12:40:28 +02:00
< ? php
2016-05-27 03:09:03 +02:00
use Embed\Adapters\AdapterInterface ;
use Embed\Embed ;
2016-06-15 06:03:16 +02:00
use SilverStripe\ORM\DataObjectInterface ;
use SilverStripe\ORM\DataObject ;
2016-07-14 04:43:53 +02:00
use SilverStripe\ORM\FieldType\DBField ;
2016-06-15 06:03:16 +02:00
2016-05-27 03:09:03 +02:00
2007-07-19 12:40:28 +02:00
/**
2009-10-11 02:07:13 +02:00
* A TinyMCE - powered WYSIWYG HTML editor field with image and link insertion and tracking capabilities . Editor fields
* are created from < textarea > tags , which are then converted with JavaScript .
*
2008-01-09 05:18:36 +01:00
* @ package forms
* @ subpackage fields - formattedinput
2007-07-19 12:40:28 +02:00
*/
2016-05-06 01:43:47 +02:00
class HTMLEditorField extends TextareaField {
2012-01-06 12:01:33 +01:00
2016-06-03 10:51:02 +02:00
private static $casting = [
'Value' => 'HTMLText' ,
];
2016-04-01 06:09:24 +02:00
/**
* Use TinyMCE ' s GZIP compressor
*
* @ config
* @ var bool
*/
private static $use_gzip = true ;
2012-01-06 12:01:33 +01:00
/**
2016-05-06 01:43:47 +02:00
* Should we check the valid_elements ( & extended_valid_elements ) rules from HTMLEditorConfig server side ?
2015-09-15 04:52:02 +02:00
*
2013-03-21 19:48:54 +01:00
* @ config
2015-09-15 04:52:02 +02:00
* @ var bool
2012-01-06 12:01:33 +01:00
*/
2016-02-12 04:00:15 +01:00
private static $sanitise_server_side = false ;
2012-02-10 12:15:21 +01:00
2013-07-03 10:06:49 +02:00
/**
2016-02-12 04:00:15 +01:00
* Number of rows
*
2013-07-03 10:06:49 +02:00
* @ config
2016-02-12 04:00:15 +01:00
* @ var int
2013-07-03 10:06:49 +02:00
*/
2016-02-12 04:00:15 +01:00
private static $default_rows = 30 ;
2013-07-03 10:06:49 +02:00
2015-09-15 04:52:02 +02:00
/**
2016-02-12 04:00:15 +01:00
* ID or instance of editorconfig
2015-09-15 04:52:02 +02:00
*
2016-05-06 01:43:47 +02:00
* @ var string | HTMLEditorConfig
2015-09-15 04:52:02 +02:00
*/
2016-02-12 04:00:15 +01:00
protected $editorConfig = null ;
2015-08-14 02:39:33 +02:00
2009-05-18 02:19:47 +02:00
/**
2016-05-06 01:43:47 +02:00
* Gets the HTMLEditorConfig instance
2016-02-12 04:00:15 +01:00
*
2016-05-06 01:43:47 +02:00
* @ return HTMLEditorConfig
2009-05-18 02:19:47 +02:00
*/
2016-02-12 04:00:15 +01:00
public function getEditorConfig () {
// Instance override
2016-05-06 01:43:47 +02:00
if ( $this -> editorConfig instanceof HTMLEditorConfig ) {
2016-02-12 04:00:15 +01:00
return $this -> editorConfig ;
}
2013-06-21 00:32:08 +02:00
2016-02-12 04:00:15 +01:00
// Get named / active config
2016-05-06 01:43:47 +02:00
return HTMLEditorConfig :: get ( $this -> editorConfig );
2016-02-12 04:00:15 +01:00
}
2015-06-09 02:01:21 +02:00
2016-02-12 04:00:15 +01:00
/**
* Assign a new configuration instance or identifier
*
2016-05-06 01:43:47 +02:00
* @ param string | HTMLEditorConfig $config
2016-02-12 04:00:15 +01:00
* @ return $this
*/
public function setEditorConfig ( $config ) {
$this -> editorConfig = $config ;
return $this ;
}
2015-06-09 02:01:21 +02:00
2007-07-19 12:40:28 +02:00
/**
2015-06-09 02:01:21 +02:00
* Creates a new HTMLEditorField .
2009-10-11 02:07:13 +02:00
* @ see TextareaField :: __construct ()
2015-06-09 02:01:21 +02:00
*
* @ param string $name The internal field name , passed to forms .
* @ param string $title The human - readable field label .
* @ param mixed $value The value of the field .
2016-05-06 01:43:47 +02:00
* @ param string $config HTMLEditorConfig identifier to be used . Default to the active one .
2015-08-14 02:39:33 +02:00
*/
2015-06-09 02:01:21 +02:00
public function __construct ( $name , $title = null , $value = '' , $config = null ) {
2012-01-02 17:44:23 +01:00
parent :: __construct ( $name , $title , $value );
2013-06-21 00:32:08 +02:00
2016-02-12 04:00:15 +01:00
if ( $config ) {
$this -> setEditorConfig ( $config );
}
$this -> setRows ( $this -> config () -> default_rows );
2007-07-19 12:40:28 +02:00
}
2013-06-21 00:32:08 +02:00
2012-09-19 12:07:39 +02:00
public function getAttributes () {
2011-12-22 13:10:57 +01:00
return array_merge (
parent :: getAttributes (),
2016-02-12 04:00:15 +01:00
$this -> getEditorConfig () -> getAttributes ()
2011-12-22 13:10:57 +01:00
);
}
2013-06-21 00:32:08 +02:00
2012-04-11 07:00:57 +02:00
public function saveInto ( DataObjectInterface $record ) {
2013-04-29 09:32:05 +02:00
if ( $record -> hasField ( $this -> name ) && $record -> escapeTypeForField ( $this -> name ) != 'xml' ) {
2009-10-11 02:07:13 +02:00
throw new Exception (
2016-05-06 01:43:47 +02:00
'HTMLEditorField->saveInto(): This field should save into a HTMLText or HTMLVarchar field.'
2008-01-17 05:22:13 +01:00
);
}
2013-06-21 00:32:08 +02:00
2013-08-07 01:02:11 +02:00
// Sanitise if requested
2016-03-08 12:20:51 +01:00
$htmlValue = Injector :: inst () -> create ( 'HTMLValue' , $this -> Value ());
2013-07-03 10:06:49 +02:00
if ( $this -> config () -> sanitise_server_side ) {
2016-05-06 01:43:47 +02:00
$santiser = Injector :: inst () -> create ( 'HTMLEditorSanitiser' , HTMLEditorConfig :: get_active ());
2013-07-03 10:06:49 +02:00
$santiser -> sanitise ( $htmlValue );
}
2014-12-10 08:36:45 +01:00
// optionally manipulate the HTML after a TinyMCE edit and prior to a save
$this -> extend ( 'processHTML' , $htmlValue );
2013-08-07 01:02:11 +02:00
// Store into record
2009-10-11 02:07:27 +02:00
$record -> { $this -> name } = $htmlValue -> getContent ();
2007-07-19 12:40:28 +02:00
}
2009-11-26 23:18:59 +01:00
2016-03-08 12:20:51 +01:00
public function setValue ( $value ) {
// Regenerate links prior to preview, so that the editor can see them.
$value = Image :: regenerate_html_links ( $value );
return parent :: setValue ( $value );
}
2009-10-11 02:07:13 +02:00
/**
2016-05-06 01:43:47 +02:00
* @ return HTMLEditorField_Readonly
2009-10-11 02:07:13 +02:00
*/
public function performReadonlyTransformation () {
2016-06-03 10:51:02 +02:00
return $this -> castedCopy ( 'HTMLEditorField_Readonly' );
2007-07-19 12:40:28 +02:00
}
2013-06-21 00:32:08 +02:00
2012-01-02 16:19:22 +01:00
public function performDisabledTransformation () {
return $this -> performReadonlyTransformation ();
}
2016-02-12 04:00:15 +01:00
public function Field ( $properties = array ()) {
// Include requirements
$this -> getEditorConfig () -> init ();
return parent :: Field ( $properties );
}
2007-07-19 12:40:28 +02:00
}
2008-01-10 01:34:18 +01:00
/**
* Readonly version of an { @ link HTMLEditorField } .
* @ package forms
* @ subpackage fields - formattedinput
*/
2016-06-03 10:51:02 +02:00
class HTMLEditorField_Readonly extends HTMLReadonlyField {
private static $casting = [
2016-07-14 04:43:53 +02:00
'Value' => 'HTMLText' ,
2016-06-03 10:51:02 +02:00
];
2012-09-19 12:07:39 +02:00
public function Type () {
2007-11-06 22:01:37 +01:00
return 'htmleditorfield readonly' ;
2007-07-19 12:40:28 +02:00
}
}
2008-01-10 01:34:18 +01:00
/**
2012-02-09 17:17:39 +01:00
* Toolbar shared by all instances of { @ link HTMLEditorField }, to avoid too much markup duplication .
* Needs to be inserted manually into the template in order to function - see { @ link LeftAndMain -> EditorToolbar ()} .
2013-06-21 00:32:08 +02:00
*
2008-01-10 01:34:18 +01:00
* @ package forms
* @ subpackage fields - formattedinput
*/
2016-05-06 01:43:47 +02:00
class HTMLEditorField_Toolbar extends RequestHandler {
2012-02-08 21:32:25 +01:00
2013-03-21 19:48:54 +01:00
private static $allowed_actions = array (
2012-02-08 21:32:25 +01:00
'LinkForm' ,
'MediaForm' ,
2014-07-30 00:59:13 +02:00
'viewfile' ,
'getanchors'
2012-02-08 21:32:25 +01:00
);
2012-02-09 17:17:39 +01:00
/**
* @ var string
*/
2016-07-13 14:36:52 +02:00
protected $templateViewFile = 'Includes/HTMLEditorField_viewfile' ;
2012-02-09 17:17:39 +01:00
2007-07-19 12:40:28 +02:00
protected $controller , $name ;
2013-06-21 00:32:08 +02:00
2012-09-19 12:07:39 +02:00
public function __construct ( $controller , $name ) {
2008-11-05 02:59:27 +01:00
parent :: __construct ();
2012-02-08 21:32:25 +01:00
2007-07-19 12:40:28 +02:00
$this -> controller = $controller ;
$this -> name = $name ;
}
2010-04-13 01:34:08 +02:00
2012-04-12 22:28:00 +02:00
public function forTemplate () {
return sprintf (
'<div id="cms-editor-dialogs" data-url-linkform="%s" data-url-mediaform="%s"></div>' ,
2012-11-09 04:44:48 +01:00
Controller :: join_links ( $this -> controller -> Link (), $this -> name , 'LinkForm' , 'forTemplate' ),
Controller :: join_links ( $this -> controller -> Link (), $this -> name , 'MediaForm' , 'forTemplate' )
2012-04-12 22:28:00 +02:00
);
}
2010-04-13 01:34:08 +02:00
/**
* Searches the SiteTree for display in the dropdown
2013-06-21 00:32:08 +02:00
*
2010-04-13 01:34:08 +02:00
* @ return callback
2013-06-21 00:32:08 +02:00
*/
2012-09-19 12:07:39 +02:00
public function siteTreeSearchCallback ( $sourceObject , $labelField , $search ) {
2013-06-21 00:32:08 +02:00
return DataObject :: get ( $sourceObject ) -> filterAny ( array (
'MenuTitle:PartialMatch' => $search ,
'Title:PartialMatch' => $search
));
2010-04-13 01:34:08 +02:00
}
2013-06-21 00:32:08 +02:00
2007-07-19 12:40:28 +02:00
/**
2008-11-04 05:12:54 +01:00
* Return a { @ link Form } instance allowing a user to
* add links in the TinyMCE content editor .
2013-06-21 00:32:08 +02:00
*
2008-11-04 05:12:54 +01:00
* @ return Form
2007-07-19 12:40:28 +02:00
*/
2012-09-19 12:07:39 +02:00
public function LinkForm () {
2016-05-06 01:43:47 +02:00
$siteTree = TreeDropdownField :: create ( 'internal' , _t ( 'HTMLEditorField.PAGE' , " Page " ),
2016-08-05 00:32:13 +02:00
'SilverStripe\\CMS\\Model\\SiteTree' , 'ID' , 'MenuTitle' , true );
2010-04-13 01:34:08 +02:00
// mimic the SiteTree::getMenuTitle(), which is bypassed when the search is performed
$siteTree -> setSearchFunction ( array ( $this , 'siteTreeSearchCallback' ));
2013-06-21 00:32:08 +02:00
2016-09-02 06:50:13 +02:00
$numericLabelTmpl = '<span class="step-label"><span class="flyout">Step %d.</span>'
. '<span class="title">%s</span></span>' ;
2007-07-19 12:40:28 +02:00
$form = new Form (
$this -> controller ,
2013-06-21 00:32:08 +02:00
" { $this -> name } /LinkForm " ,
2011-05-11 09:51:54 +02:00
new FieldList (
2012-06-25 05:31:39 +02:00
$headerWrap = new CompositeField (
new LiteralField (
2013-06-21 00:32:08 +02:00
'Heading' ,
2012-09-26 23:34:00 +02:00
sprintf ( '<h3 class="htmleditorfield-mediaform-heading insert">%s</h3>' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.LINK' , 'Insert Link' ))
2012-06-25 05:31:39 +02:00
)
2007-07-19 12:40:28 +02:00
),
2009-11-21 03:22:44 +01:00
$contentComposite = new CompositeField (
2013-10-25 21:49:41 +02:00
OptionsetField :: create (
2009-11-21 03:22:44 +01:00
'LinkType' ,
2016-07-14 04:43:53 +02:00
DBField :: create_field (
'HTMLFragment' ,
2016-09-02 06:50:13 +02:00
sprintf ( $numericLabelTmpl , '1' , _t ( 'HTMLEditorField.LINKTYPE' , 'Link type' ))
2016-07-14 04:43:53 +02:00
),
2009-11-21 03:22:44 +01:00
array (
2016-09-02 06:50:13 +02:00
'internal' => _t ( 'HTMLEditorField.LINKINTERNAL' , 'Link to a page on this site' ),
'external' => _t ( 'HTMLEditorField.LINKEXTERNAL' , 'Link to another website' ),
'anchor' => _t ( 'HTMLEditorField.LINKANCHOR' , 'Link to an anchor on this page' ),
'email' => _t ( 'HTMLEditorField.LINKEMAIL' , 'Link to an email address' ),
'file' => _t ( 'HTMLEditorField.LINKFILE' , 'Link to download a file' ),
2012-05-09 07:11:59 +02:00
),
'internal'
2009-11-21 03:22:44 +01:00
),
2013-10-25 21:49:41 +02:00
LiteralField :: create ( 'Step2' ,
2012-09-26 23:34:00 +02:00
'<div class="step2">'
2016-09-02 06:50:13 +02:00
. sprintf ( $numericLabelTmpl , '2' , _t ( 'HTMLEditorField.LINKDETAILS' , 'Link details' )) . '</div>'
2012-01-03 22:30:24 +01:00
),
2010-04-13 01:34:08 +02:00
$siteTree ,
2016-05-06 01:43:47 +02:00
TextField :: create ( 'external' , _t ( 'HTMLEditorField.URL' , 'URL' ), 'http://' ),
EmailField :: create ( 'email' , _t ( 'HTMLEditorField.EMAIL' , 'Email address' )),
$fileField = UploadField :: create ( 'file' , _t ( 'HTMLEditorField.FILE' , 'File' )),
TextField :: create ( 'Anchor' , _t ( 'HTMLEditorField.ANCHORVALUE' , 'Anchor' )),
TextField :: create ( 'Subject' , _t ( 'HTMLEditorField.SUBJECT' , 'Email subject' )),
TextField :: create ( 'Description' , _t ( 'HTMLEditorField.LINKDESCR' , 'Link description' )),
2013-10-25 21:49:41 +02:00
CheckboxField :: create ( 'TargetBlank' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.LINKOPENNEWWIN' , 'Open link in a new window?' )),
2013-10-25 21:49:41 +02:00
HiddenField :: create ( 'Locale' , null , $this -> controller -> Locale )
2009-11-21 03:22:44 +01:00
)
2007-07-19 12:40:28 +02:00
),
2015-10-01 03:16:07 +02:00
new FieldList ()
2007-07-19 12:40:28 +02:00
);
2012-06-25 05:31:39 +02:00
2016-07-28 07:03:46 +02:00
$headerWrap -> setName ( 'HeaderWrap' );
2016-07-25 07:24:26 +02:00
$headerWrap -> addExtraClass ( 'CompositeField composite cms-content-header form-group--no-label ' );
2016-07-28 07:03:46 +02:00
$contentComposite -> setName ( 'ContentBody' );
2012-06-25 05:31:39 +02:00
$contentComposite -> addExtraClass ( 'ss-insert-link content' );
2015-06-03 01:20:29 +02:00
$fileField -> setAllowedMaxFileNumber ( 1 );
2013-06-21 00:32:08 +02:00
2009-11-21 03:31:11 +01:00
$form -> unsetValidator ();
2007-07-19 12:40:28 +02:00
$form -> loadDataFrom ( $this );
2015-09-21 00:00:05 +02:00
$form -> addExtraClass ( 'htmleditorfield-form htmleditorfield-linkform cms-mediaform-content' );
2013-06-21 00:32:08 +02:00
2010-05-25 06:56:02 +02:00
$this -> extend ( 'updateLinkForm' , $form );
2013-06-21 00:32:08 +02:00
2007-07-19 12:40:28 +02:00
return $form ;
}
2014-07-21 05:52:10 +02:00
/**
* Get the folder ID to filter files by for the " from cms " tab
*
* @ return int
*/
protected function getAttachParentID () {
$parentID = $this -> controller -> getRequest () -> requestVar ( 'ParentID' );
$this -> extend ( 'updateAttachParentID' , $parentID );
return $parentID ;
}
2008-11-04 05:12:54 +01:00
/**
* Return a { @ link Form } instance allowing a user to
2012-02-08 21:32:25 +01:00
* add images and flash objects to the TinyMCE content editor .
2013-06-21 00:32:08 +02:00
*
2008-11-04 05:12:54 +01:00
* @ return Form
*/
2012-09-19 12:07:39 +02:00
public function MediaForm () {
2012-09-26 23:34:00 +02:00
// TODO Handle through GridState within field - currently this state set too late to be useful here (during
// request handling)
2014-07-21 05:52:10 +02:00
$parentID = $this -> getAttachParentID ();
2012-02-08 21:32:25 +01:00
2012-07-10 23:19:50 +02:00
$fileFieldConfig = GridFieldConfig :: create () -> addComponents (
new GridFieldFilterHeader (),
new GridFieldSortableHeader (),
new GridFieldDataColumns (),
2015-07-28 00:08:25 +02:00
new GridFieldPaginator ( 7 ),
2013-06-14 13:53:09 +02:00
// TODO Shouldn't allow delete here, its too confusing with a "remove from editor view" action.
// Remove once we can fit the search button in the last actual title column
2012-07-10 23:19:50 +02:00
new GridFieldDeleteAction (),
new GridFieldDetailForm ()
);
2015-09-01 02:38:58 +02:00
$fileField = GridField :: create ( 'Files' , false , null , $fileFieldConfig );
2012-02-08 21:32:25 +01:00
$fileField -> setList ( $this -> getFiles ( $parentID ));
$fileField -> setAttribute ( 'data-selectable' , true );
$fileField -> setAttribute ( 'data-multiselect' , true );
2012-04-19 01:25:37 +02:00
$columns = $fileField -> getConfig () -> getComponentByType ( 'GridFieldDataColumns' );
$columns -> setDisplayFields ( array (
2015-07-28 00:08:25 +02:00
'StripThumbnail' => false ,
'Title' => _t ( 'File.Title' ),
'Created' => singleton ( 'File' ) -> fieldLabel ( 'Created' ),
));
$columns -> setFieldCasting ( array (
2016-06-15 06:03:16 +02:00
'Created' => 'DBDatetime->Nice'
2012-02-09 17:17:39 +01:00
));
2013-06-21 00:32:08 +02:00
2012-03-29 02:34:55 +02:00
$fromCMS = new CompositeField (
2014-07-21 05:52:10 +02:00
$select = TreeDropdownField :: create ( 'ParentID' , " " , 'Folder' )
-> addExtraClass ( 'noborder' )
-> setValue ( $parentID ),
2012-09-26 23:34:00 +02:00
$fileField
2012-03-29 02:34:55 +02:00
);
2013-06-21 00:32:08 +02:00
2015-09-21 00:00:05 +02:00
$fromCMS -> addExtraClass ( 'content ss-uploadfield htmleditorfield-from-cms' );
2012-06-22 08:04:13 +02:00
$select -> addExtraClass ( 'content-select' );
2012-03-29 02:34:55 +02:00
2016-05-06 01:43:47 +02:00
$URLDescription = _t ( 'HTMLEditorField.URLDESCRIPTION' , 'Insert videos and images from the web into your page simply by entering the URL of the file. Make sure you have the rights or permissions before sharing media directly from the web.<br /><br />Please note that files are not added to the file store of the CMS but embeds the file from its original location, if for some reason the file is no longer available in its original location it will no longer be viewable on this page.' );
2012-05-22 06:45:36 +02:00
$fromWeb = new CompositeField (
2015-10-27 01:15:33 +01:00
$description = new LiteralField ( 'URLDescription' , '<div class="url-description">' . $URLDescription . '</div>' ),
2012-06-21 04:19:41 +02:00
$remoteURL = new TextField ( 'RemoteURL' , 'http://' ),
2012-09-26 23:34:00 +02:00
new LiteralField ( 'addURLImage' ,
2015-11-03 02:23:16 +01:00
'<button type="button" class="action ui-action-constructive ui-button field font-icon-plus add-url">' .
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.BUTTONADDURL' , 'Add url' ) . '</button>' )
2012-05-22 06:45:36 +02:00
);
2012-06-22 08:04:13 +02:00
2012-05-22 06:45:36 +02:00
$remoteURL -> addExtraClass ( 'remoteurl' );
2015-09-21 00:00:05 +02:00
$fromWeb -> addExtraClass ( 'content ss-uploadfield htmleditorfield-from-web' );
2012-05-22 06:45:36 +02:00
2016-04-19 11:04:43 +02:00
Requirements :: css ( FRAMEWORK_DIR . '/client/dist/styles/AssetUploadField.css' );
2012-03-29 02:34:55 +02:00
$computerUploadField = Object :: create ( 'UploadField' , 'AssetUploadField' , '' );
$computerUploadField -> setConfig ( 'previewMaxWidth' , 40 );
$computerUploadField -> setConfig ( 'previewMaxHeight' , 30 );
2015-09-21 00:00:05 +02:00
$computerUploadField -> addExtraClass ( 'ss-assetuploadfield htmleditorfield-from-computer' );
2012-03-29 02:34:55 +02:00
$computerUploadField -> removeExtraClass ( 'ss-uploadfield' );
2016-05-06 01:43:47 +02:00
$computerUploadField -> setTemplate ( 'HTMLEditorField_UploadField' );
2013-03-21 19:48:54 +01:00
$computerUploadField -> setFolderName ( Config :: inst () -> get ( 'Upload' , 'uploads_folder' ));
2016-03-08 21:50:18 +01:00
2015-10-27 01:15:33 +01:00
$defaultPanel = new CompositeField (
$computerUploadField ,
$fromCMS
);
2016-03-08 21:50:18 +01:00
2015-10-27 01:15:33 +01:00
$fromWebPanel = new CompositeField (
$fromWeb
);
2016-03-08 21:50:18 +01:00
2015-10-27 01:15:33 +01:00
$defaultPanel -> addExtraClass ( 'htmleditorfield-default-panel' );
$fromWebPanel -> addExtraClass ( 'htmleditorfield-web-panel' );
2012-03-29 02:34:55 +02:00
$allFields = new CompositeField (
2015-10-27 01:15:33 +01:00
$defaultPanel ,
$fromWebPanel ,
2012-03-29 02:34:55 +02:00
$editComposite = new CompositeField (
2012-05-22 08:18:31 +02:00
new LiteralField ( 'contentEdit' , '<div class="content-edit ss-uploadfield-files files"></div>' )
2012-05-28 12:27:20 +02:00
)
2007-07-19 12:40:28 +02:00
);
2012-03-29 02:34:55 +02:00
2012-05-28 12:27:20 +02:00
$allFields -> addExtraClass ( 'ss-insert-media' );
2012-05-22 08:18:31 +02:00
$headings = new CompositeField (
2012-03-29 02:34:55 +02:00
new LiteralField (
'Heading' ,
2012-09-26 23:34:00 +02:00
sprintf ( '<h3 class="htmleditorfield-mediaform-heading insert">%s</h3>' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.INSERTMEDIA' , 'Insert media from' )) .
2012-09-26 23:34:00 +02:00
sprintf ( '<h3 class="htmleditorfield-mediaform-heading update">%s</h3>' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.UpdateMEDIA' , 'Update media' ))
2012-05-28 12:27:20 +02:00
)
2012-05-22 08:18:31 +02:00
);
$headings -> addExtraClass ( 'cms-content-header' );
$editComposite -> addExtraClass ( 'ss-assetuploadfield' );
$fields = new FieldList (
$headings ,
2012-03-29 02:34:55 +02:00
$allFields
);
2013-06-21 00:32:08 +02:00
2010-05-25 06:56:02 +02:00
$form = new Form (
$this -> controller ,
2012-02-08 21:32:25 +01:00
" { $this -> name } /MediaForm " ,
2010-05-25 06:56:02 +02:00
$fields ,
2015-10-01 03:16:07 +02:00
new FieldList ()
2010-05-25 06:56:02 +02:00
);
2013-06-21 00:32:08 +02:00
2012-03-29 02:34:55 +02:00
2009-11-21 03:31:11 +01:00
$form -> unsetValidator ();
2009-06-28 03:17:20 +02:00
$form -> disableSecurityToken ();
2007-07-19 12:40:28 +02:00
$form -> loadDataFrom ( $this );
2012-02-08 21:32:25 +01:00
$form -> addExtraClass ( 'htmleditorfield-form htmleditorfield-mediaform cms-dialog-content' );
2012-02-09 17:17:39 +01:00
2013-06-21 00:32:08 +02:00
// Allow other people to extend the fields being added to the imageform
2012-02-09 17:17:39 +01:00
$this -> extend ( 'updateMediaForm' , $form );
2013-06-21 00:32:08 +02:00
2007-07-19 12:40:28 +02:00
return $form ;
}
2016-05-03 04:16:05 +02:00
/**
* List of allowed schemes ( no wildcard , all lower case ) or empty to allow all schemes
*
* @ config
* @ var array
*/
private static $fileurl_scheme_whitelist = array ( 'http' , 'https' );
/**
* List of allowed domains ( no wildcard , all lower case ) or empty to allow all domains
*
* @ config
* @ var array
*/
private static $fileurl_domain_whitelist = array ();
/**
* Find local File dataobject given ID
*
* @ param int $id
* @ return array
*/
protected function viewfile_getLocalFileByID ( $id ) {
/** @var File $file */
$file = DataObject :: get_by_id ( 'File' , $id );
if ( $file && $file -> canView ()) {
return array ( $file , $file -> getURL ());
}
return [ null , null ];
}
/**
* Get remote File given url
*
* @ param string $fileUrl Absolute URL
* @ return array
* @ throws SS_HTTPResponse_Exception
*/
protected function viewfile_getRemoteFileByURL ( $fileUrl ) {
if ( ! Director :: is_absolute_url ( $fileUrl )) {
throw $this -> getErrorFor ( _t (
2016-05-06 01:43:47 +02:00
" HTMLEditorField_Toolbar.ERROR_ABSOLUTE " ,
2016-05-03 04:16:05 +02:00
" Only absolute urls can be embedded "
));
}
$scheme = strtolower ( parse_url ( $fileUrl , PHP_URL_SCHEME ));
$allowed_schemes = self :: config () -> fileurl_scheme_whitelist ;
if ( ! $scheme || ( $allowed_schemes && ! in_array ( $scheme , $allowed_schemes ))) {
throw $this -> getErrorFor ( _t (
2016-05-06 01:43:47 +02:00
" HTMLEditorField_Toolbar.ERROR_SCHEME " ,
2016-05-03 04:16:05 +02:00
" This file scheme is not included in the whitelist "
));
}
$domain = strtolower ( parse_url ( $fileUrl , PHP_URL_HOST ));
$allowed_domains = self :: config () -> fileurl_domain_whitelist ;
if ( ! $domain || ( $allowed_domains && ! in_array ( $domain , $allowed_domains ))) {
throw $this -> getErrorFor ( _t (
2016-05-06 01:43:47 +02:00
" HTMLEditorField_Toolbar.ERROR_HOSTNAME " ,
2016-05-03 04:16:05 +02:00
" This file hostname is not included in the whitelist "
));
}
return [ null , $fileUrl ];
}
/**
* Prepare error for the front end
*
* @ param string $message
* @ param int $code
* @ return SS_HTTPResponse_Exception
*/
protected function getErrorFor ( $message , $code = 400 ) {
$exception = new SS_HTTPResponse_Exception ( $message , $code );
$exception -> getResponse () -> addHeader ( 'X-Status' , $message );
return $exception ;
}
2012-02-09 17:17:39 +01:00
/**
* View of a single file , either on the filesystem or on the web .
2015-09-15 04:52:02 +02:00
*
2016-05-03 04:16:05 +02:00
* @ throws SS_HTTPResponse_Exception
2015-09-15 04:52:02 +02:00
* @ param SS_HTTPRequest $request
* @ return string
2012-02-09 17:17:39 +01:00
*/
public function viewfile ( $request ) {
2015-09-15 04:52:02 +02:00
$file = null ;
2016-05-03 04:16:05 +02:00
$url = null ;
// Get file and url by request method
if ( $fileUrl = $request -> getVar ( 'FileURL' )) {
// Get remote url
list ( $file , $url ) = $this -> viewfile_getRemoteFileByURL ( $fileUrl );
2012-02-09 17:17:39 +01:00
} elseif ( $id = $request -> getVar ( 'ID' )) {
2016-05-03 04:16:05 +02:00
// Or we could have been passed an ID directly
list ( $file , $url ) = $this -> viewfile_getLocalFileByID ( $id );
2012-02-09 17:17:39 +01:00
} else {
2016-05-03 04:16:05 +02:00
// Or we could have been passed nothing, in which case panic
throw $this -> getErrorFor ( _t (
2016-05-06 01:43:47 +02:00
" HTMLEditorField_Toolbar.ERROR_ID " ,
2016-05-03 04:16:05 +02:00
'Need either "ID" or "FileURL" parameter to identify the file'
));
}
// Validate file exists
if ( ! $url ) {
throw $this -> getErrorFor ( _t (
2016-05-06 01:43:47 +02:00
" HTMLEditorField_Toolbar.ERROR_NOTFOUND " ,
2016-05-03 04:16:05 +02:00
'Unable to find file to view'
));
2012-02-09 17:17:39 +01:00
}
2016-05-27 03:09:03 +02:00
// Instantiate file wrapper and get fields based on its type
// Check if appCategory is an image and exists on the local system, otherwise use Embed to reference a
2012-09-26 23:34:00 +02:00
// remote image
2016-05-03 04:16:05 +02:00
$fileCategory = $this -> getFileCategory ( $url , $file );
2015-09-15 04:52:02 +02:00
switch ( $fileCategory ) {
case 'image' :
case 'image/supported' :
2016-05-06 01:43:47 +02:00
$fileWrapper = new HTMLEditorField_Image ( $url , $file );
2015-09-15 04:52:02 +02:00
break ;
case 'flash' :
2016-05-06 01:43:47 +02:00
$fileWrapper = new HTMLEditorField_Flash ( $url , $file );
2015-09-15 04:52:02 +02:00
break ;
default :
// Only remote files can be linked via o-embed
2016-05-06 01:43:47 +02:00
// {@see HTMLEditorField_Toolbar::getAllowedExtensions())
2015-09-15 04:52:02 +02:00
if ( $file ) {
2016-05-03 04:16:05 +02:00
throw $this -> getErrorFor ( _t (
2016-05-06 01:43:47 +02:00
" HTMLEditorField_Toolbar.ERROR_OEMBED_REMOTE " ,
2016-05-27 03:09:03 +02:00
" Embed is only compatible with remote files "
2016-05-03 04:16:05 +02:00
));
2015-09-15 04:52:02 +02:00
}
2016-05-03 04:16:05 +02:00
2016-05-27 03:09:03 +02:00
// Other files should fallback to embed
2016-05-06 01:43:47 +02:00
$fileWrapper = new HTMLEditorField_Embed ( $url , $file );
2015-09-15 04:52:02 +02:00
break ;
2012-02-09 17:17:39 +01:00
}
2012-03-12 16:31:32 +01:00
2015-09-15 04:52:02 +02:00
// Render fields and return
2012-02-09 17:17:39 +01:00
$fields = $this -> getFieldsForFile ( $url , $fileWrapper );
return $fileWrapper -> customise ( array (
'Fields' => $fields ,
)) -> renderWith ( $this -> templateViewFile );
}
2016-05-03 04:16:05 +02:00
/**
* Guess file category from either a file or url
*
* @ param string $url
* @ param File $file
* @ return string
*/
protected function getFileCategory ( $url , $file ) {
if ( $file ) {
return $file -> appCategory ();
}
if ( $url ) {
return File :: get_app_category ( File :: get_file_extension ( $url ));
}
return null ;
}
2014-07-30 00:59:13 +02:00
/**
* Find all anchors available on the given page .
*
* @ return array
2016-05-03 04:16:05 +02:00
* @ throws SS_HTTPResponse_Exception
2014-07-30 00:59:13 +02:00
*/
public function getanchors () {
2015-04-30 01:04:08 +02:00
$id = ( int ) $this -> getRequest () -> getVar ( 'PageID' );
2014-07-30 00:59:13 +02:00
$anchors = array ();
if (( $page = Page :: get () -> byID ( $id )) && ! empty ( $page )) {
if ( ! $page -> canView ()) {
throw new SS_HTTPResponse_Exception (
_t (
2016-05-06 01:43:47 +02:00
'HTMLEditorField.ANCHORSCANNOTACCESSPAGE' ,
2014-07-30 00:59:13 +02:00
'You are not permitted to access the content of the target page.'
),
403
);
}
2016-06-17 06:28:58 +02:00
// Parse the shortcodes so [img id=x] doesn't end up as anchor x
$htmlValue = $page -> obj ( 'Content' ) -> forTemplate ();
2016-05-06 01:43:47 +02:00
// Similar to the regex found in HTMLEditorField.js / getAnchors method.
2015-11-18 17:26:26 +01:00
if ( preg_match_all (
2016-01-19 05:08:40 +01:00
" / \\ s+(name|id) \\ s*= \\ s*([ \" '])([^ \\ 2 \\ s>]*?) \\ 2| \\ s+(name|id) \\ s*= \\ s*([^ \" ']+)[ \\ s +>]/im " ,
2016-06-17 06:28:58 +02:00
$htmlValue ,
2015-11-18 17:26:26 +01:00
$matches
)) {
2016-01-19 05:08:40 +01:00
$anchors = array_values ( array_unique ( array_filter (
array_merge ( $matches [ 3 ], $matches [ 5 ]))
));
2014-07-30 00:59:13 +02:00
}
} else {
throw new SS_HTTPResponse_Exception (
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.ANCHORSPAGENOTFOUND' , 'Target page not found.' ),
2014-07-30 00:59:13 +02:00
404
);
}
return json_encode ( $anchors );
}
2012-02-09 17:17:39 +01:00
/**
* Similar to { @ link File -> getCMSFields ()}, but only returns fields
* for manipulating the instance of the file as inserted into the HTML content ,
* not the " master record " in the database - hence there ' s no form or saving logic .
2013-06-21 00:32:08 +02:00
*
2015-09-15 04:52:02 +02:00
* @ param string $url Abolute URL to asset
2016-05-06 01:43:47 +02:00
* @ param HTMLEditorField_File $file Asset wrapper
2012-02-09 17:17:39 +01:00
* @ return FieldList
*/
2016-05-06 01:43:47 +02:00
protected function getFieldsForFile ( $url , HTMLEditorField_File $file ) {
2012-02-09 17:17:39 +01:00
$fields = $this -> extend ( 'getFieldsForFile' , $url , $file );
if ( ! $fields ) {
2015-09-15 04:52:02 +02:00
$fields = $file -> getFields ();
$file -> extend ( 'updateFields' , $fields );
2012-02-09 17:17:39 +01:00
}
$this -> extend ( 'updateFieldsForFile' , $fields , $url , $file );
return $fields ;
}
/**
2015-09-15 04:52:02 +02:00
* Gets files filtered by a given parent with the allowed extensions
*
* @ param int $parentID
2012-02-08 21:32:25 +01:00
* @ return DataList
*/
protected function getFiles ( $parentID = null ) {
2012-02-09 17:17:39 +01:00
$exts = $this -> getAllowedExtensions ();
2016-03-08 21:50:18 +01:00
$dotExts = array_map ( function ( $ext ) {
2015-09-15 04:52:02 +02:00
return " . { $ext } " ;
}, $exts );
$files = File :: get () -> filter ( 'Name:EndsWith' , $dotExts );
2012-02-08 21:32:25 +01:00
// Limit by folder (if required)
2012-06-15 05:49:22 +02:00
if ( $parentID ) {
$files = $files -> filter ( 'ParentID' , $parentID );
}
2012-02-08 21:32:25 +01:00
return $files ;
2007-07-19 12:40:28 +02:00
}
2012-02-09 17:17:39 +01:00
/**
* @ return Array All extensions which can be handled by the different views .
*/
protected function getAllowedExtensions () {
2015-09-15 04:52:02 +02:00
$exts = array ( 'jpg' , 'gif' , 'png' , 'swf' , 'jpeg' );
2012-02-09 17:17:39 +01:00
$this -> extend ( 'updateAllowedExtensions' , $exts );
return $exts ;
}
2012-04-12 22:28:00 +02:00
2012-02-09 17:17:39 +01:00
}
/**
* Encapsulation of a file which can either be a remote URL
* or a { @ link File } on the local filesystem , exhibiting common properties
* such as file name or the URL .
*
* @ todo Remove once core has support for remote files
2013-11-29 05:12:47 +01:00
* @ package forms
* @ subpackage fields - formattedinput
2012-02-09 17:17:39 +01:00
*/
2016-05-06 01:43:47 +02:00
abstract class HTMLEditorField_File extends ViewableData {
2015-09-15 04:52:02 +02:00
/**
* Default insertion width for Images and Media
*
* @ config
* @ var int
*/
private static $insert_width = 600 ;
/**
* Default insert height for images and media
*
* @ config
* @ var int
*/
private static $insert_height = 360 ;
/**
* Max width for insert - media preview .
*
* Matches CSS rule for . cms - file - info - preview
*
* @ var int
*/
private static $media_preview_width = 176 ;
/**
* Max height for insert - media preview .
*
* Matches CSS rule for . cms - file - info - preview
*
* @ var int
*/
private static $media_preview_height = 128 ;
2012-02-09 17:17:39 +01:00
2014-06-11 01:34:31 +02:00
private static $casting = array (
'URL' => 'Varchar' ,
'Name' => 'Varchar'
);
2015-09-15 04:52:02 +02:00
/**
* Absolute URL to asset
*
* @ var string
*/
2012-02-09 17:17:39 +01:00
protected $url ;
2015-09-15 04:52:02 +02:00
/**
* File dataobject ( if available )
*
* @ var File
*/
2012-02-09 17:17:39 +01:00
protected $file ;
/**
2015-09-15 04:52:02 +02:00
* @ param string $url
* @ param File $file
2012-02-09 17:17:39 +01:00
*/
2015-09-15 04:52:02 +02:00
public function __construct ( $url , File $file = null ) {
2012-02-09 17:17:39 +01:00
$this -> url = $url ;
$this -> file = $file ;
2012-02-28 18:07:55 +01:00
$this -> failover = $file ;
2012-02-09 17:17:39 +01:00
parent :: __construct ();
}
/**
2015-09-15 04:52:02 +02:00
* @ return FieldList
*/
public function getFields () {
$fields = new FieldList (
CompositeField :: create (
CompositeField :: create ( LiteralField :: create ( " ImageFull " , $this -> getPreview ()))
-> setName ( " FilePreviewImage " )
-> addExtraClass ( 'cms-file-info-preview' ),
CompositeField :: create ( $this -> getDetailFields ())
-> setName ( " FilePreviewData " )
-> addExtraClass ( 'cms-file-info-data' )
)
-> setName ( " FilePreview " )
-> addExtraClass ( 'cms-file-info' ),
2016-05-06 01:43:47 +02:00
TextField :: create ( 'CaptionText' , _t ( 'HTMLEditorField.CAPTIONTEXT' , 'Caption text' )),
2015-09-15 04:52:02 +02:00
DropdownField :: create (
'CSSClass' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.CSSCLASS' , 'Alignment / style' ),
2015-09-15 04:52:02 +02:00
array (
2016-05-06 01:43:47 +02:00
'leftAlone' => _t ( 'HTMLEditorField.CSSCLASSLEFTALONE' , 'On the left, on its own.' ),
'center' => _t ( 'HTMLEditorField.CSSCLASSCENTER' , 'Centered, on its own.' ),
'left' => _t ( 'HTMLEditorField.CSSCLASSLEFT' , 'On the left, with text wrapping around.' ),
'right' => _t ( 'HTMLEditorField.CSSCLASSRIGHT' , 'On the right, with text wrapping around.' )
2015-09-15 04:52:02 +02:00
)
),
2016-05-06 01:43:47 +02:00
FieldGroup :: create ( _t ( 'HTMLEditorField.IMAGEDIMENSIONS' , 'Dimensions' ),
2015-09-15 04:52:02 +02:00
TextField :: create (
'Width' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.IMAGEWIDTHPX' , 'Width' ),
2015-09-15 04:52:02 +02:00
$this -> getInsertWidth ()
) -> setMaxLength ( 5 ),
TextField :: create (
'Height' ,
2016-05-06 01:43:47 +02:00
" x " . _t ( 'HTMLEditorField.IMAGEHEIGHTPX' , 'Height' ),
2015-09-15 04:52:02 +02:00
$this -> getInsertHeight ()
) -> setMaxLength ( 5 )
) -> addExtraClass ( 'dimensions last' ),
HiddenField :: create ( 'URL' , false , $this -> getURL ()),
HiddenField :: create ( 'FileID' , false , $this -> getFileID ())
);
return $fields ;
}
/**
* Get list of fields for previewing this records details
*
* @ return FieldList
*/
protected function getDetailFields () {
$fields = new FieldList (
ReadonlyField :: create ( " FileType " , _t ( 'AssetTableField.TYPE' , 'File type' ), $this -> getFileType ()),
2016-07-13 09:08:09 +02:00
HTMLReadonlyField :: create (
2015-09-15 04:52:02 +02:00
'ClickableURL' , _t ( 'AssetTableField.URL' , 'URL' ), $this -> getExternalLink ()
2016-07-13 09:08:09 +02:00
)
2015-09-15 04:52:02 +02:00
);
// Get file size
if ( $this -> getSize ()) {
$fields -> insertAfter (
'FileType' ,
ReadonlyField :: create ( " Size " , _t ( 'AssetTableField.SIZE' , 'File size' ), $this -> getSize ())
);
}
// Get modified details of local record
if ( $this -> getFile ()) {
$fields -> push ( new DateField_Disabled (
" Created " ,
_t ( 'AssetTableField.CREATED' , 'First uploaded' ),
$this -> getFile () -> Created
));
$fields -> push ( new DateField_Disabled (
" LastEdited " ,
_t ( 'AssetTableField.LASTEDIT' , 'Last changed' ),
$this -> getFile () -> LastEdited
));
}
return $fields ;
2016-03-08 21:50:18 +01:00
2015-09-15 04:52:02 +02:00
}
/**
* Get file DataObject
*
* Might not be set ( for remote files )
*
* @ return File
2012-02-09 17:17:39 +01:00
*/
2012-09-19 12:07:39 +02:00
public function getFile () {
2012-02-09 17:17:39 +01:00
return $this -> file ;
}
2015-09-15 04:52:02 +02:00
/**
* Get file ID
2016-03-08 21:50:18 +01:00
*
2015-09-15 04:52:02 +02:00
* @ return int
*/
public function getFileID () {
if ( $file = $this -> getFile ()) {
return $file -> ID ;
}
}
/**
* Get absolute URL
*
* @ return string
*/
2012-09-19 12:07:39 +02:00
public function getURL () {
2012-02-09 17:17:39 +01:00
return $this -> url ;
}
2015-09-15 04:52:02 +02:00
/**
* Get basename
*
* @ return string
*/
2012-09-19 12:07:39 +02:00
public function getName () {
2015-09-15 04:52:02 +02:00
return $this -> file
? $this -> file -> Name
: preg_replace ( '/\?.*/' , '' , basename ( $this -> url ));
}
/**
* Get descriptive file type
*
* @ return string
*/
public function getFileType () {
return File :: get_file_type ( $this -> getName ());
}
/**
* Get file size ( if known ) as string
*
* @ return string | false String value , or false if doesn ' t exist
*/
public function getSize () {
if ( $this -> file ) {
return $this -> file -> getSize ();
}
return false ;
2012-02-09 17:17:39 +01:00
}
/**
2015-09-15 04:52:02 +02:00
* HTML content for preview
*
* @ return string HTML
2012-02-09 17:17:39 +01:00
*/
2012-09-19 12:07:39 +02:00
public function getPreview () {
2012-02-09 17:17:39 +01:00
$preview = $this -> extend ( 'getPreview' );
2015-09-15 04:52:02 +02:00
if ( $preview ) {
return $preview ;
}
2016-03-08 21:50:18 +01:00
2015-09-15 04:52:02 +02:00
// Generate tag from preview
$thumbnailURL = Convert :: raw2att (
Controller :: join_links ( $this -> getPreviewURL (), " ?r= " . rand ( 1 , 100000 ))
);
$fileName = Convert :: raw2att ( $this -> Name );
return sprintf (
" <img id='thumbnailImage' class='thumbnail-preview' src='%s' alt='%s' /> \n " ,
$thumbnailURL ,
$fileName
);
}
2012-02-09 17:17:39 +01:00
2015-09-15 04:52:02 +02:00
/**
* HTML Content for external link
*
* @ return string
*/
public function getExternalLink () {
$title = $this -> file
? $this -> file -> getTitle ()
: $this -> getName ();
return sprintf (
'<a href="%1$s" title="%2$s" target="_blank" rel="external" class="file-url">%1$s</a>' ,
Convert :: raw2att ( $this -> url ),
Convert :: raw2att ( $title )
);
}
/**
* Generate thumbnail url
*
* @ return string
*/
public function getPreviewURL () {
// Get preview from file
2012-02-09 17:17:39 +01:00
if ( $this -> file ) {
2015-09-15 04:52:02 +02:00
return $this -> getFilePreviewURL ();
2012-02-09 17:17:39 +01:00
}
2015-09-15 04:52:02 +02:00
// Generate default icon html
return File :: get_icon_for_extension ( $this -> getExtension ());
2012-02-09 17:17:39 +01:00
}
2015-09-15 04:52:02 +02:00
/**
* Generate thumbnail URL from file dataobject ( if available )
*
* @ return string
*/
protected function getFilePreviewURL () {
// Get preview from file
if ( $this -> file ) {
$width = $this -> config () -> media_preview_width ;
$height = $this -> config () -> media_preview_height ;
return $this -> file -> ThumbnailURL ( $width , $height );
}
}
/**
* Get file extension
*
* @ return string
*/
2012-09-19 12:07:39 +02:00
public function getExtension () {
2015-09-15 04:52:02 +02:00
$extension = File :: get_file_extension ( $this -> getName ());
return strtolower ( $extension );
2012-02-09 17:17:39 +01:00
}
2015-09-15 04:52:02 +02:00
/**
* Category name
*
* @ return string
*/
2012-09-19 12:07:39 +02:00
public function appCategory () {
2012-02-09 17:17:39 +01:00
if ( $this -> file ) {
return $this -> file -> appCategory ();
} else {
2015-09-15 04:52:02 +02:00
return File :: get_app_category ( $this -> getExtension ());
2012-02-09 17:17:39 +01:00
}
}
2015-09-15 04:52:02 +02:00
/**
* Get height of this item
*/
public function getHeight () {
if ( $this -> file ) {
$height = $this -> file -> getHeight ();
if ( $height ) {
return $height ;
}
}
return $this -> config () -> insert_height ;
}
/**
* Get width of this item
*
* @ return type
*/
public function getWidth () {
if ( $this -> file ) {
$width = $this -> file -> getWidth ();
if ( $width ) {
return $width ;
}
}
return $this -> config () -> insert_width ;
}
/**
* Provide an initial width for inserted media , restricted based on $embed_width
*
* @ return int
*/
public function getInsertWidth () {
$width = $this -> getWidth ();
$maxWidth = $this -> config () -> insert_width ;
return ( $width <= $maxWidth ) ? $width : $maxWidth ;
}
/**
* Provide an initial height for inserted media , scaled proportionally to the initial width
*
* @ return int
*/
public function getInsertHeight () {
$width = $this -> getWidth ();
$height = $this -> getHeight ();
$maxWidth = $this -> config () -> insert_width ;
return ( $width <= $maxWidth ) ? $height : round ( $height * ( $maxWidth / $width ));
}
2012-02-09 17:17:39 +01:00
}
2013-11-29 05:12:47 +01:00
/**
2016-05-27 03:09:03 +02:00
* Encapsulation of an embed tag , linking to an external media source .
2013-11-29 05:12:47 +01:00
*
2016-05-27 03:09:03 +02:00
* @ see Embed
2013-11-29 05:12:47 +01:00
* @ package forms
* @ subpackage fields - formattedinput
*/
2016-05-06 01:43:47 +02:00
class HTMLEditorField_Embed extends HTMLEditorField_File {
2014-06-11 01:34:31 +02:00
private static $casting = array (
'Type' => 'Varchar' ,
'Info' => 'Varchar'
);
2015-09-15 04:52:02 +02:00
/**
2016-05-27 03:09:03 +02:00
* Embed result
2015-09-15 04:52:02 +02:00
*
2016-05-27 03:09:03 +02:00
* @ var Embed
2015-09-15 04:52:02 +02:00
*/
2016-05-27 03:09:03 +02:00
protected $embed ;
2012-05-22 06:45:36 +02:00
2015-09-15 04:52:02 +02:00
public function __construct ( $url , File $file = null ) {
2012-05-22 06:45:36 +02:00
parent :: __construct ( $url , $file );
2016-05-27 03:09:03 +02:00
$this -> embed = Embed :: create ( $url );
if ( ! $this -> embed ) {
2012-05-24 04:28:47 +02:00
$controller = Controller :: curr ();
2015-08-14 02:39:33 +02:00
$response = $controller -> getResponse ();
$response -> addHeader ( 'X-Status' ,
2012-05-24 04:28:47 +02:00
rawurlencode ( _t (
2016-05-06 01:43:47 +02:00
'HTMLEditorField.URLNOTANOEMBEDRESOURCE' ,
2012-05-24 04:28:47 +02:00
" The URL ' { url}' could not be turned into a media resource. " ,
2012-09-26 23:34:00 +02:00
" The given URL is not a valid Oembed resource; the embed element couldn't be created. " ,
2012-05-24 04:28:47 +02:00
array ( 'url' => $url )
)));
2015-08-14 02:39:33 +02:00
$response -> setStatusCode ( 404 );
2012-05-24 04:28:47 +02:00
2015-08-14 02:39:33 +02:00
throw new SS_HTTPResponse_Exception ( $response );
2012-05-22 06:45:36 +02:00
}
}
2015-09-15 04:52:02 +02:00
/**
* Get file - edit fields for this filed
*
* @ return FieldList
*/
public function getFields () {
$fields = parent :: getFields ();
if ( $this -> Type === 'photo' ) {
$fields -> insertBefore ( 'CaptionText' , new TextField (
'AltText' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.IMAGEALTTEXT' , 'Alternative text (alt) - shown if image can\'t be displayed' ),
2015-09-15 04:52:02 +02:00
$this -> Title ,
80
));
$fields -> insertBefore ( 'CaptionText' , new TextField (
'Title' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.IMAGETITLE' , 'Title text (tooltip) - for additional information about the image' )
2015-09-15 04:52:02 +02:00
));
}
return $fields ;
2012-05-22 06:45:36 +02:00
}
2013-06-24 22:25:16 +02:00
/**
2016-05-27 03:09:03 +02:00
* Get width of this Embed
2014-06-11 01:34:31 +02:00
*
2013-06-24 22:25:16 +02:00
* @ return int
*/
2015-09-15 04:52:02 +02:00
public function getWidth () {
2016-05-27 03:09:03 +02:00
return $this -> embed -> width ? : 100 ;
2013-06-24 22:25:16 +02:00
}
/**
2016-05-27 03:09:03 +02:00
* Get height of this Embed
2014-06-11 01:34:31 +02:00
*
2013-06-24 22:25:16 +02:00
* @ return int
*/
2015-09-15 04:52:02 +02:00
public function getHeight () {
2016-05-27 03:09:03 +02:00
return $this -> embed -> height ? : 100 ;
2013-06-24 22:25:16 +02:00
}
2015-09-15 04:52:02 +02:00
public function getPreviewURL () {
// Use thumbnail url
2016-05-27 03:09:03 +02:00
if ( $this -> embed -> image ) {
return $this -> embed -> image ;
2012-05-22 06:45:36 +02:00
}
2015-09-15 04:52:02 +02:00
// Use direct image type
2016-05-27 03:09:03 +02:00
if ( $this -> getType () == 'photo' && ! empty ( $this -> embed -> url )) {
return $this -> embed -> url ;
2015-09-15 04:52:02 +02:00
}
// Default media
return FRAMEWORK_DIR . '/images/default_media.png' ;
2012-05-22 06:45:36 +02:00
}
public function getName () {
2016-05-27 03:09:03 +02:00
if ( $this -> embed -> title ) {
return $this -> embed -> title ;
2012-05-22 06:45:36 +02:00
} else {
return parent :: getName ();
}
}
2015-09-15 04:52:02 +02:00
/**
2016-05-27 03:09:03 +02:00
* Get Embed type
2015-09-15 04:52:02 +02:00
*
* @ return string
*/
2012-05-22 06:45:36 +02:00
public function getType () {
2016-05-27 03:09:03 +02:00
return $this -> embed -> type ;
2012-05-22 06:45:36 +02:00
}
2016-05-27 03:09:03 +02:00
/**
* Get filetype
*
* @ return string
*/
2015-09-15 04:52:02 +02:00
public function getFileType () {
return $this -> getType ()
? : parent :: getFileType ();
}
/**
2016-05-27 03:09:03 +02:00
* @ return AdapterInterface
2015-09-15 04:52:02 +02:00
*/
2016-05-27 03:09:03 +02:00
public function getEmbed () {
return $this -> embed ;
2012-05-22 06:45:36 +02:00
}
public function appCategory () {
return 'embed' ;
}
2013-06-21 00:32:08 +02:00
2015-09-15 04:52:02 +02:00
/**
2016-05-27 03:09:03 +02:00
* Info for this Embed
2015-09-15 04:52:02 +02:00
*
* @ return string
*/
2012-06-25 17:23:26 +02:00
public function getInfo () {
2016-05-27 03:09:03 +02:00
return $this -> embed -> info ;
2012-06-25 17:23:26 +02:00
}
2012-05-22 06:45:36 +02:00
}
2013-11-29 05:12:47 +01:00
/**
* Encapsulation of an image tag , linking to an image either internal or external to the site .
*
* @ package forms
* @ subpackage fields - formattedinput
*/
2016-05-06 01:43:47 +02:00
class HTMLEditorField_Image extends HTMLEditorField_File {
2012-02-09 17:17:39 +01:00
2015-09-15 04:52:02 +02:00
/**
* @ var int
*/
2012-02-09 17:17:39 +01:00
protected $width ;
2015-09-15 04:52:02 +02:00
/**
* @ var int
*/
2012-02-09 17:17:39 +01:00
protected $height ;
2015-09-15 04:52:02 +02:00
/**
* File size details
*
* @ var string
*/
protected $size ;
public function __construct ( $url , File $file = null ) {
2012-02-09 17:17:39 +01:00
parent :: __construct ( $url , $file );
2015-09-15 04:52:02 +02:00
if ( $file ) {
return ;
}
// Get size of remote file
$size = @ filesize ( $url );
if ( $size ) {
$this -> size = $size ;
}
// Get dimensions of remote file
2012-02-09 17:17:39 +01:00
$info = @ getimagesize ( $url );
if ( $info ) {
$this -> width = $info [ 0 ];
$this -> height = $info [ 1 ];
}
}
2015-09-15 04:52:02 +02:00
public function getFields () {
$fields = parent :: getFields ();
// Alt text
$fields -> insertBefore (
'CaptionText' ,
TextField :: create (
'AltText' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.IMAGEALT' , 'Alternative text (alt)' ),
2015-09-15 04:52:02 +02:00
$this -> Title ,
80
) -> setDescription (
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.IMAGEALTTEXTDESC' , 'Shown to screen readers or if image can\'t be displayed' )
2015-09-15 04:52:02 +02:00
)
);
// Tooltip
$fields -> insertAfter (
'AltText' ,
TextField :: create (
'Title' ,
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.IMAGETITLETEXT' , 'Title text (tooltip)' )
2015-09-15 04:52:02 +02:00
) -> setDescription (
2016-05-06 01:43:47 +02:00
_t ( 'HTMLEditorField.IMAGETITLETEXTDESC' , 'For additional information about the image' )
2015-09-15 04:52:02 +02:00
)
);
return $fields ;
}
protected function getDetailFields () {
$fields = parent :: getDetailFields ();
$width = $this -> getOriginalWidth ();
$height = $this -> getOriginalHeight ();
// Show dimensions of original
if ( $width && $height ) {
$fields -> insertAfter (
'ClickableURL' ,
ReadonlyField :: create (
" OriginalWidth " ,
_t ( 'AssetTableField.WIDTH' , 'Width' ),
$width
)
);
$fields -> insertAfter (
'OriginalWidth' ,
ReadonlyField :: create (
" OriginalHeight " ,
_t ( 'AssetTableField.HEIGHT' , 'Height' ),
$height
)
);
}
return $fields ;
}
/**
* Get width of original , if known
*
* @ return int
*/
public function getOriginalWidth () {
if ( $this -> width ) {
return $this -> width ;
}
if ( $this -> file ) {
$width = $this -> file -> getWidth ();
if ( $width ) {
return $width ;
}
}
}
/**
* Get height of original , if known
*
* @ return int
*/
public function getOriginalHeight () {
if ( $this -> height ) {
return $this -> height ;
}
if ( $this -> file ) {
$height = $this -> file -> getHeight ();
if ( $height ) {
return $height ;
}
}
}
2012-09-19 12:07:39 +02:00
public function getWidth () {
2015-09-15 04:52:02 +02:00
if ( $this -> width ) {
return $this -> width ;
}
return parent :: getWidth ();
2012-02-09 17:17:39 +01:00
}
2012-09-19 12:07:39 +02:00
public function getHeight () {
2015-09-15 04:52:02 +02:00
if ( $this -> height ) {
return $this -> height ;
}
return parent :: getHeight ();
}
public function getSize () {
if ( $this -> size ) {
return File :: format_size ( $this -> size );
}
parent :: getSize ();
2012-02-09 17:17:39 +01:00
}
2013-06-24 22:25:16 +02:00
/**
* Provide an initial width for inserted image , restricted based on $embed_width
2013-06-21 00:32:08 +02:00
*
2013-06-24 22:25:16 +02:00
* @ return int
*/
public function getInsertWidth () {
$width = $this -> getWidth ();
2015-09-15 04:52:02 +02:00
$maxWidth = $this -> config () -> insert_width ;
return $width <= $maxWidth
? $width
: $maxWidth ;
2013-06-24 22:25:16 +02:00
}
/**
* Provide an initial height for inserted image , scaled proportionally to the initial width
2013-06-21 00:32:08 +02:00
*
2013-06-24 22:25:16 +02:00
* @ return int
*/
public function getInsertHeight () {
$width = $this -> getWidth ();
$height = $this -> getHeight ();
2015-09-15 04:52:02 +02:00
$maxWidth = $this -> config () -> insert_width ;
2013-06-24 22:25:16 +02:00
return ( $width <= $maxWidth ) ? $height : round ( $height * ( $maxWidth / $width ));
}
2015-09-15 04:52:02 +02:00
public function getPreviewURL () {
// Get preview from file
if ( $this -> file ) {
return $this -> getFilePreviewURL ();
}
// Embed image directly
return $this -> url ;
2012-02-09 17:17:39 +01:00
}
2015-09-15 04:52:02 +02:00
}
/**
* Generate flash file embed
*/
2016-05-06 01:43:47 +02:00
class HTMLEditorField_Flash extends HTMLEditorField_File {
2012-02-09 17:17:39 +01:00
2015-09-15 04:52:02 +02:00
public function getFields () {
$fields = parent :: getFields ();
$fields -> removeByName ( 'CaptionText' , true );
return $fields ;
}
2012-03-01 09:13:42 +01:00
}