042db255c8c700ee1968c18c6bdaae24f32848ac
[moodle.git] / lib / editor / atto / plugins / media / yui / src / button / js / button.js
1 // This file is part of Moodle - http://moodle.org/
2 //
3 // Moodle is free software: you can redistribute it and/or modify
4 // it under the terms of the GNU General Public License as published by
5 // the Free Software Foundation, either version 3 of the License, or
6 // (at your option) any later version.
7 //
8 // Moodle is distributed in the hope that it will be useful,
9 // but WITHOUT ANY WARRANTY; without even the implied warranty of
10 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11 // GNU General Public License for more details.
12 //
13 // You should have received a copy of the GNU General Public License
14 // along with Moodle.  If not, see <http://www.gnu.org/licenses/>.
16 /*
17  * @package    atto_media
18  * @copyright  2013 Damyon Wiese  <damyon@moodle.com>
19  * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
20  */
22 /**
23  * @module moodle-atto_media-button
24  */
26 /**
27  * Atto media selection tool.
28  *
29  * @namespace M.atto_media
30  * @class Button
31  * @extends M.editor_atto.EditorPlugin
32  */
34 var COMPONENTNAME = 'atto_media',
35     MEDIA_TYPES = {LINK: 'LINK', VIDEO: 'VIDEO', AUDIO: 'AUDIO'},
36     TRACK_KINDS = {
37         SUBTITLES: 'SUBTITLES',
38         CAPTIONS: 'CAPTIONS',
39         DESCRIPTIONS: 'DESCRIPTIONS',
40         CHAPTERS: 'CHAPTERS',
41         METADATA: 'METADATA'
42     },
43     CSS = {
44         SOURCE: 'atto_media_source',
45         TRACK: 'atto_media_track',
46         MEDIA_SOURCE: 'atto_media_media_source',
47         LINK_SOURCE: 'atto_media_link_source',
48         POSTER_SOURCE: 'atto_media_poster_source',
49         TRACK_SOURCE: 'atto_media_track_source',
50         DISPLAY_OPTIONS: 'atto_media_display_options',
51         NAME_INPUT: 'atto_media_name_entry',
52         TITLE_INPUT: 'atto_media_title_entry',
53         URL_INPUT: 'atto_media_url_entry',
54         POSTER_SIZE: 'atto_media_poster_size',
55         LINK_SIZE: 'atto_media_link_size',
56         WIDTH_INPUT: 'atto_media_width_entry',
57         HEIGHT_INPUT: 'atto_media_height_entry',
58         TRACK_KIND_INPUT: 'atto_media_track_kind_entry',
59         TRACK_LABEL_INPUT: 'atto_media_track_label_entry',
60         TRACK_LANG_INPUT: 'atto_media_track_lang_entry',
61         TRACK_DEFAULT_SELECT: 'atto_media_track_default',
62         MEDIA_CONTROLS_TOGGLE: 'atto_media_controls',
63         MEDIA_AUTOPLAY_TOGGLE: 'atto_media_autoplay',
64         MEDIA_MUTE_TOGGLE: 'atto_media_mute',
65         MEDIA_LOOP_TOGGLE: 'atto_media_loop',
66         ADVANCED_SETTINGS: 'atto_media_advancedsettings',
67         LINK: MEDIA_TYPES.LINK.toLowerCase(),
68         VIDEO: MEDIA_TYPES.VIDEO.toLowerCase(),
69         AUDIO: MEDIA_TYPES.AUDIO.toLowerCase(),
70         TRACK_SUBTITLES: TRACK_KINDS.SUBTITLES.toLowerCase(),
71         TRACK_CAPTIONS: TRACK_KINDS.CAPTIONS.toLowerCase(),
72         TRACK_DESCRIPTIONS: TRACK_KINDS.DESCRIPTIONS.toLowerCase(),
73         TRACK_CHAPTERS: TRACK_KINDS.CHAPTERS.toLowerCase(),
74         TRACK_METADATA: TRACK_KINDS.METADATA.toLowerCase()
75     },
76     SELECTORS = {
77         SOURCE: '.' + CSS.SOURCE,
78         TRACK: '.' + CSS.TRACK,
79         MEDIA_SOURCE: '.' + CSS.MEDIA_SOURCE,
80         POSTER_SOURCE: '.' + CSS.POSTER_SOURCE,
81         TRACK_SOURCE: '.' + CSS.TRACK_SOURCE,
82         DISPLAY_OPTIONS: '.' + CSS.DISPLAY_OPTIONS,
83         NAME_INPUT: '.' + CSS.NAME_INPUT,
84         TITLE_INPUT: '.' + CSS.TITLE_INPUT,
85         URL_INPUT: '.' + CSS.URL_INPUT,
86         POSTER_SIZE: '.' + CSS.POSTER_SIZE,
87         LINK_SIZE: '.' + CSS.LINK_SIZE,
88         WIDTH_INPUT: '.' + CSS.WIDTH_INPUT,
89         HEIGHT_INPUT: '.' + CSS.HEIGHT_INPUT,
90         TRACK_KIND_INPUT: '.' + CSS.TRACK_KIND_INPUT,
91         TRACK_LABEL_INPUT: '.' + CSS.TRACK_LABEL_INPUT,
92         TRACK_LANG_INPUT: '.' + CSS.TRACK_LANG_INPUT,
93         TRACK_DEFAULT_SELECT: '.' + CSS.TRACK_DEFAULT_SELECT,
94         MEDIA_CONTROLS_TOGGLE: '.' + CSS.MEDIA_CONTROLS_TOGGLE,
95         MEDIA_AUTOPLAY_TOGGLE: '.' + CSS.MEDIA_AUTOPLAY_TOGGLE,
96         MEDIA_MUTE_TOGGLE: '.' + CSS.MEDIA_MUTE_TOGGLE,
97         MEDIA_LOOP_TOGGLE: '.' + CSS.MEDIA_LOOP_TOGGLE,
98         ADVANCED_SETTINGS: '.' + CSS.ADVANCED_SETTINGS,
99         LINK_TAB: 'li[data-medium-type="' + CSS.LINK + '"]',
100         LINK_PANE: '.tab-pane[data-medium-type="' + CSS.LINK + '"]',
101         VIDEO_TAB: 'li[data-medium-type="' + CSS.VIDEO + '"]',
102         VIDEO_PANE: '.tab-pane[data-medium-type="' + CSS.VIDEO + '"]',
103         AUDIO_TAB: 'li[data-medium-type="' + CSS.AUDIO + '"]',
104         AUDIO_PANE: '.tab-pane[data-medium-type="' + CSS.AUDIO + '"]',
105         TRACK_SUBTITLES_TAB: 'li[data-track-kind="' + CSS.TRACK_SUBTITLES + '"]',
106         TRACK_SUBTITLES_PANE: '.tab-pane[data-track-kind="' + CSS.TRACK_SUBTITLES + '"]',
107         TRACK_CAPTIONS_TAB: 'li[data-track-kind="' + CSS.TRACK_CAPTIONS + '"]',
108         TRACK_CAPTIONS_PANE: '.tab-pane[data-track-kind="' + CSS.TRACK_CAPTIONS + '"]',
109         TRACK_DESCRIPTIONS_TAB: 'li[data-track-kind="' + CSS.TRACK_DESCRIPTIONS + '"]',
110         TRACK_DESCRIPTIONS_PANE: '.tab-pane[data-track-kind="' + CSS.TRACK_DESCRIPTIONS + '"]',
111         TRACK_CHAPTERS_TAB: 'li[data-track-kind="' + CSS.TRACK_CHAPTERS + '"]',
112         TRACK_CHAPTERS_PANE: '.tab-pane[data-track-kind="' + CSS.TRACK_CHAPTERS + '"]',
113         TRACK_METADATA_TAB: 'li[data-track-kind="' + CSS.TRACK_METADATA + '"]',
114         TRACK_METADATA_PANE: '.tab-pane[data-track-kind="' + CSS.TRACK_METADATA + '"]'
115     },
116     TEMPLATES = {
117         ROOT: '' +
118             '<form class="mform atto_form atto_media" id="{{elementid}}_atto_media_form">' +
119                 '<ul class="root nav nav-tabs m-b-1" role="tablist">' +
120                     '<li data-medium-type="{{CSS.LINK}}" class="nav-item">' +
121                         '<a class="nav-link active" href="#{{elementid}}_{{CSS.LINK}}" role="tab" data-toggle="tab">' +
122                             '{{get_string "link" component}}' +
123                         '</a>' +
124                     '</li>' +
125                     '<li data-medium-type="{{CSS.VIDEO}}" class="nav-item">' +
126                         '<a class="nav-link" href="#{{elementid}}_{{CSS.VIDEO}}" role="tab" data-toggle="tab">' +
127                             '{{get_string "video" component}}' +
128                         '</a>' +
129                     '</li>' +
130                     '<li data-medium-type="{{CSS.AUDIO}}" class="nav-item">' +
131                         '<a class="nav-link" href="#{{elementid}}_{{CSS.AUDIO}}" role="tab" data-toggle="tab">' +
132                             '{{get_string "audio" component}}' +
133                         '</a>' +
134                     '</li>' +
135                 '</ul>' +
136                 '<div class="root tab-content">' +
137                     '<div data-medium-type="{{CSS.LINK}}" class="tab-pane active" id="{{elementid}}_{{CSS.LINK}}">' +
138                         '{{> tab_panes.link}}' +
139                     '</div>' +
140                     '<div data-medium-type="{{CSS.VIDEO}}" class="tab-pane" id="{{elementid}}_{{CSS.VIDEO}}">' +
141                         '{{> tab_panes.video}}' +
142                     '</div>' +
143                     '<div data-medium-type="{{CSS.AUDIO}}" class="tab-pane" id="{{elementid}}_{{CSS.AUDIO}}">' +
144                         '{{> tab_panes.audio}}' +
145                     '</div>' +
146                 '</div>' +
147                 '<div class="mdl-align">' +
148                     '<br/>' +
149                     '<button class="btn btn-default submit" type="submit">{{get_string "createmedia" component}}</button>' +
150                 '</div>' +
151             '</form>',
152         TAB_PANES: {
153             LINK: '' +
154                 '{{renderPartial "form_components.source" context=this id=CSS.LINK_SOURCE}}' +
155                 '<label for="{{elementid}}_link_nameentry">{{get_string "entername" component}}</label>' +
156                 '<input class="form-control fullwidth {{CSS.NAME_INPUT}}" type="text" id="{{elementid}}_link_nameentry"' +
157                         'size="32" required="true"/>',
158             VIDEO: '' +
159                 '{{renderPartial "form_components.source" context=this id=CSS.MEDIA_SOURCE entersourcelabel="videosourcelabel"' +
160                     ' addcomponentlabel="addsource" multisource="true" addsourcehelp=helpStrings.addsource}}' +
161                 '<fieldset class="collapsible collapsed" id="{{elementid}}_video-display-options">' +
162                     '<input name="mform_isexpanded_{{elementid}}_video-display-options" type="hidden">' +
163                     '<legend class="ftoggler">{{get_string "displayoptions" component}}</legend>' +
164                     '<div class="fcontainer">' +
165                         '{{renderPartial "form_components.display_options" context=this id=CSS.VIDEO mediatype_video=true}}' +
166                     '</div>' +
167                 '</fieldset>' +
168                 '<fieldset class="collapsible collapsed" id="{{elementid}}_video-advanced-settings">' +
169                     '<input name="mform_isexpanded_{{elementid}}_video-advanced-settings" type="hidden">' +
170                     '<legend class="ftoggler">{{get_string "advancedsettings" component}}</legend>' +
171                     '<div class="fcontainer">' +
172                         '{{renderPartial "form_components.advanced_settings" context=this id=CSS.VIDEO}}' +
173                     '</div>' +
174                 '</fieldset>' +
175                 '<fieldset class="collapsible collapsed" id="{{elementid}}_video-tracks">' +
176                     '<input name="mform_isexpanded_{{elementid}}_video-tracks" type="hidden">' +
177                     '<legend class="ftoggler">{{get_string "tracks" component}} {{{helpStrings.tracks}}}</legend>' +
178                     '<div class="fcontainer">' +
179                         '{{renderPartial "form_components.track_tabs" context=this id=CSS.VIDEO}}' +
180                     '</div>' +
181                 '</fieldset>',
182             AUDIO: '' +
183                 '{{renderPartial "form_components.source" context=this id=CSS.MEDIA_SOURCE entersourcelabel="audiosourcelabel"' +
184                     ' addcomponentlabel="addsource" multisource="true" addsourcehelp=helpStrings.addsource}}' +
185                 '<fieldset class="collapsible collapsed" id="{{elementid}}_audio-display-options">' +
186                     '<input name="mform_isexpanded_{{elementid}}_audio-display-options" type="hidden">' +
187                     '<legend class="ftoggler">{{get_string "displayoptions" component}}</legend>' +
188                     '<div class="fcontainer">' +
189                         '{{renderPartial "form_components.display_options" context=this id=CSS.AUDIO}}' +
190                     '</div>' +
191                 '</fieldset>' +
192                 '<fieldset class="collapsible collapsed" id="{{elementid}}_audio-advanced-settings">' +
193                     '<input name="mform_isexpanded_{{elementid}}_audio-advanced-settings" type="hidden">' +
194                     '<legend class="ftoggler">{{get_string "advancedsettings" component}}</legend>' +
195                     '<div class="fcontainer">' +
196                         '{{renderPartial "form_components.advanced_settings" context=this id=CSS.AUDIO}}' +
197                     '</div>' +
198                 '</fieldset>' +
199                 '<fieldset class="collapsible collapsed" id="{{elementid}}_audio-tracks">' +
200                     '<input name="mform_isexpanded_{{elementid}}_audio-tracks" type="hidden">' +
201                     '<legend class="ftoggler">{{get_string "tracks" component}} {{{helpStrings.tracks}}}</legend>' +
202                     '<div class="fcontainer">' +
203                         '{{renderPartial "form_components.track_tabs" context=this id=CSS.AUDIO}}' +
204                     '</div>' +
205                 '</fieldset>'
206         },
207         FORM_COMPONENTS: {
208             SOURCE: '' +
209                 '<div class="{{CSS.SOURCE}} {{id}}">' +
210                     '<div class="m-b-1">' +
211                         '<label for="url-input">' +
212                         '{{#entersourcelabel}}{{get_string ../entersourcelabel ../component}}{{/entersourcelabel}}' +
213                         '{{^entersourcelabel}}{{get_string "entersource" ../component}}{{/entersourcelabel}}' +
214                         '</label>' +
215                         '<div class="input-group input-append w-100">' +
216                             '<input id="url-input" class="form-control {{CSS.URL_INPUT}}" type="url" size="32"/>' +
217                             '<span class="input-group-append">' +
218                                 '<button class="btn btn-default openmediabrowser" type="button">' +
219                                 '{{get_string "browserepositories" component}}</button>' +
220                             '</span>' +
221                         '</div>' +
222                     '</div>' +
223                     '{{#multisource}}' +
224                         '{{renderPartial "form_components.add_component" context=../this label=../addcomponentlabel ' +
225                             ' help=../addsourcehelp}}' +
226                     '{{/multisource}}' +
227                 '</div>',
228             ADD_COMPONENT: '' +
229                 '<div>' +
230                     '<a href="#" class="addcomponent">' +
231                         '{{#label}}{{get_string ../label ../component}}{{/label}}' +
232                         '{{^label}}{{get_string "add" ../component}}{{/label}}' +
233                     '</a>' +
234                     '{{#help}}{{{../help}}}{{/help}}' +
235                 '</div>',
236             REMOVE_COMPONENT: '' +
237                 '<div>' +
238                     '<a href="#" class="removecomponent">' +
239                         '{{#label}}{{get_string ../label ../component}}{{/label}}' +
240                         '{{^label}}{{get_string "remove" ../component}}{{/label}}' +
241                     '</a>' +
242                 '</div>',
243             DISPLAY_OPTIONS: '' +
244                 '<div class="{{CSS.DISPLAY_OPTIONS}}">' +
245                     '<div class="m-b-1">' +
246                         '<label for="{{id}}_media-title-entry">{{get_string "entertitle" component}}</label>' +
247                         '<input class="form-control fullwidth {{CSS.TITLE_INPUT}}" type="text" id="{{id}}_media-title-entry"' +
248                             'size="32"/>' +
249                     '</div>' +
250                     '<div class="clearfix"></div>' +
251                     '{{#mediatype_video}}' +
252                     '<div class="m-b-1">' +
253                         '<label>{{get_string "size" component}}</label>' +
254                         '<div class="form-inline {{CSS.POSTER_SIZE}}">' +
255                             '<label class="accesshide">{{get_string "videowidth" component}}</label>' +
256                             '<input type="text" class="form-control m-r-1 {{CSS.WIDTH_INPUT}} input-mini" size="4"/>' +
257                             ' x ' +
258                             '<label class="accesshide">{{get_string "videoheight" component}}</label>' +
259                             '<input type="text" class="form-control m-l-1 {{CSS.HEIGHT_INPUT}} input-mini" size="4"/>' +
260                         '</div>' +
261                     '</div>' +
262                     '<div class="clearfix"></div>' +
263                     '{{renderPartial "form_components.source" context=this id=CSS.POSTER_SOURCE entersourcelabel="poster"}}' +
264                     '{{/mediatype_video}}' +
265                 '<div>',
266             ADVANCED_SETTINGS: '' +
267                 '<div class="{{CSS.ADVANCED_SETTINGS}}">' +
268                     '<div class="form-check">' +
269                         '<input type="checkbox" checked="true" class="form-check-input {{CSS.MEDIA_CONTROLS_TOGGLE}}"' +
270                         'id="{{id}}_media-controls-toggle"/>' +
271                         '<label class="form-check-label" for="{{id}}_media-controls-toggle">' +
272                         '{{get_string "controls" component}}' +
273                         '</label>' +
274                     '</div>' +
275                     '<div class="form-check">' +
276                         '<input type="checkbox" class="form-check-input {{CSS.MEDIA_AUTOPLAY_TOGGLE}}"' +
277                         'id="{{id}}_media-autoplay-toggle"/>' +
278                         '<label class="form-check-label" for="{{id}}_media-autoplay-toggle">' +
279                         '{{get_string "autoplay" component}}' +
280                         '</label>' +
281                     '</div>' +
282                     '<div class="form-check">' +
283                         '<input type="checkbox" class="form-check-input {{CSS.MEDIA_MUTE_TOGGLE}}" ' +
284                             'id="{{id}}_media-mute-toggle"/>' +
285                         '<label class="form-check-label" for="{{id}}_media-mute-toggle">' +
286                         '{{get_string "mute" component}}' +
287                         '</label>' +
288                     '</div>' +
289                     '<div class="form-check">' +
290                         '<input type="checkbox" class="form-check-input {{CSS.MEDIA_LOOP_TOGGLE}}" ' +
291                             'id="{{id}}_media-loop-toggle"/>' +
292                         '<label class="form-check-label" for="{{id}}_media-loop-toggle">' +
293                         '{{get_string "loop" component}}' +
294                         '</label>' +
295                     '</div>' +
296                 '</div>',
297             TRACK_TABS: '' +
298                 '<ul class="nav nav-tabs mb-3">' +
299                     '<li data-track-kind="{{CSS.TRACK_SUBTITLES}}" class="nav-item">' +
300                         '<a class="nav-link active" href="#{{elementid}}_{{id}}_{{CSS.TRACK_SUBTITLES}}"' +
301                             ' role="tab" data-toggle="tab">' +
302                             '{{get_string "subtitles" component}}' +
303                         '</a>' +
304                     '</li>' +
305                     '<li data-track-kind="{{CSS.TRACK_CAPTIONS}}" class="nav-item">' +
306                         '<a class="nav-link" href="#{{elementid}}_{{id}}_{{CSS.TRACK_CAPTIONS}}" role="tab" data-toggle="tab">' +
307                             '{{get_string "captions" component}}' +
308                         '</a>' +
309                     '</li>' +
310                     '<li data-track-kind="{{CSS.TRACK_DESCRIPTIONS}}"  class="nav-item">' +
311                         '<a class="nav-link" href="#{{elementid}}_{{id}}_{{CSS.TRACK_DESCRIPTIONS}}"' +
312                             ' role="tab" data-toggle="tab">' +
313                             '{{get_string "descriptions" component}}' +
314                         '</a>' +
315                     '</li>' +
316                     '<li data-track-kind="{{CSS.TRACK_CHAPTERS}}" class="nav-item">' +
317                         '<a class="nav-link" href="#{{elementid}}_{{id}}_{{CSS.TRACK_CHAPTERS}}" role="tab" data-toggle="tab">' +
318                             '{{get_string "chapters" component}}' +
319                         '</a>' +
320                     '</li>' +
321                     '<li data-track-kind="{{CSS.TRACK_METADATA}}" class="nav-item">' +
322                         '<a class="nav-link" href="#{{elementid}}_{{id}}_{{CSS.TRACK_METADATA}}" role="tab" data-toggle="tab">' +
323                             '{{get_string "metadata" component}}' +
324                         '</a>' +
325                     '</li>' +
326                 '</ul>' +
327                 '<div class="tab-content">' +
328                     '<div data-track-kind="{{CSS.TRACK_SUBTITLES}}" class="tab-pane active"' +
329                         ' id="{{elementid}}_{{id}}_{{CSS.TRACK_SUBTITLES}}">' +
330                         '<div class="trackhelp">{{{helpStrings.subtitles}}}</div>' +
331                         '{{renderPartial "form_components.track" context=this sourcelabel="subtitlessourcelabel"' +
332                             ' addcomponentlabel="addsubtitlestrack"}}' +
333                     '</div>' +
334                     '<div data-track-kind="{{CSS.TRACK_CAPTIONS}}" class="tab-pane"' +
335                         ' id="{{elementid}}_{{id}}_{{CSS.TRACK_CAPTIONS}}">' +
336                         '<div class="trackhelp">{{{helpStrings.captions}}}</div>' +
337                         '{{renderPartial "form_components.track" context=this sourcelabel="captionssourcelabel"' +
338                             ' addcomponentlabel="addcaptionstrack"}}' +
339                     '</div>' +
340                     '<div data-track-kind="{{CSS.TRACK_DESCRIPTIONS}}" class="tab-pane"' +
341                         ' id="{{elementid}}_{{id}}_{{CSS.TRACK_DESCRIPTIONS}}">' +
342                         '<div class="trackhelp">{{{helpStrings.descriptions}}}</div>' +
343                         '{{renderPartial "form_components.track" context=this sourcelabel="descriptionssourcelabel"' +
344                             ' addcomponentlabel="adddescriptionstrack"}}' +
345                     '</div>' +
346                     '<div data-track-kind="{{CSS.TRACK_CHAPTERS}}" class="tab-pane"' +
347                         ' id="{{elementid}}_{{id}}_{{CSS.TRACK_CHAPTERS}}">' +
348                         '<div class="trackhelp">{{{helpStrings.chapters}}}</div>' +
349                         '{{renderPartial "form_components.track" context=this sourcelabel="chapterssourcelabel"' +
350                             ' addcomponentlabel="addchapterstrack"}}' +
351                     '</div>' +
352                     '<div data-track-kind="{{CSS.TRACK_METADATA}}" class="tab-pane"' +
353                         ' id="{{elementid}}_{{id}}_{{CSS.TRACK_METADATA}}">' +
354                         '<div class="trackhelp">{{{helpStrings.metadata}}}</div>' +
355                         '{{renderPartial "form_components.track" context=this sourcelabel="metadatasourcelabel"' +
356                             ' addcomponentlabel="addmetadatatrack"}}' +
357                     '</div>' +
358                 '</div>',
359             TRACK: '' +
360                 '<div class="m-b-1 {{CSS.TRACK}}">' +
361                     '{{renderPartial "form_components.source" context=this id=CSS.TRACK_SOURCE entersourcelabel=sourcelabel}}' +
362                     '<div class="form-group">' +
363                         '<label class="w-100" for="lang-input">{{get_string "srclang" component}}</label>' +
364                         '<select id="lang-input" class="custom-select {{CSS.TRACK_LANG_INPUT}}">' +
365                             '<optgroup label="{{get_string "languagesinstalled" component}}">' +
366                                 '{{#langsinstalled}}' +
367                                     '<option value="{{code}}" {{#default}}selected="selected"{{/default}}>{{lang}}</option>' +
368                                 '{{/langsinstalled}}' +
369                             '</optgroup>' +
370                             '<optgroup label="{{get_string "languagesavailable" component}} ">' +
371                                 '{{#langsavailable}}<option value="{{code}}">{{lang}}</option>{{/langsavailable}}' +
372                             '</optgroup>' +
373                         '</select>' +
374                     '</div>' +
375                     '<div class="form-group">' +
376                         '<label class="w-100" for="track-input">{{get_string "label" component}}</label>' +
377                         '<input id="track-input" class="form-control {{CSS.TRACK_LABEL_INPUT}}" type="text"/>' +
378                     '</div>' +
379                     '<div class="form-check">' +
380                         '<input type="checkbox" class="form-check-input {{CSS.TRACK_DEFAULT_SELECT}}"/>' +
381                         '<label class="form-check-label">{{get_string "default" component}}</label>' +
382                     '</div>' +
383                     '{{renderPartial "form_components.add_component" context=this label=addcomponentlabel}}' +
384                 '</div>'
385         },
386         HTML_MEDIA: {
387             VIDEO: '' +
388                 '&nbsp;<video ' +
389                     '{{#width}}width="{{../width}}" {{/width}}' +
390                     '{{#height}}height="{{../height}}" {{/height}}' +
391                     '{{#poster}}poster="{{../poster}}" {{/poster}}' +
392                     '{{#showControls}}controls="true" {{/showControls}}' +
393                     '{{#loop}}loop="true" {{/loop}}' +
394                     '{{#muted}}muted="true" {{/muted}}' +
395                     '{{#autoplay}}autoplay="true" {{/autoplay}}' +
396                     '{{#title}}title="{{../title}}" {{/title}}' +
397                 '>' +
398                     '{{#sources}}<source src="{{source}}">{{/sources}}' +
399                     '{{#tracks}}' +
400                         '<track src="{{track}}" kind="{{kind}}" srclang="{{srclang}}" label="{{label}}"' +
401                             ' {{#defaultTrack}}default="true"{{/defaultTrack}}>' +
402                     '{{/tracks}}' +
403                     '{{#description}}{{../description}}{{/description}}' +
404                 '</video>&nbsp',
405             AUDIO: '' +
406                 '&nbsp;<audio ' +
407                     '{{#showControls}}controls="true" {{/showControls}}' +
408                     '{{#loop}}loop="true" {{/loop}}' +
409                     '{{#muted}}muted="true" {{/muted}}' +
410                     '{{#autoplay}}autoplay="true" {{/autoplay}}' +
411                     '{{#title}}title="{{../title}}" {{/title}}' +
412                 '>' +
413                     '{{#sources}}<source src="{{source}}">{{/sources}}' +
414                     '{{#tracks}}' +
415                         '<track src="{{track}}" kind="{{kind}}" srclang="{{srclang}}" label="{{label}}"' +
416                             ' {{#defaultTrack}}default="true"{{/defaultTrack}}>' +
417                     '{{/tracks}}' +
418                     '{{#description}}{{../description}}{{/description}}' +
419                 '</audio>&nbsp',
420             LINK: '' +
421                 '<a href="{{url}}" ' +
422                     '{{#width}}data-width="{{../width}}" {{/width}}' +
423                     '{{#height}}data-height="{{../height}}"{{/height}}' +
424                 '>{{#name}}{{../name}}{{/name}}{{^name}}{{../url}}{{/name}}</a>'
425          }
426     };
428 Y.namespace('M.atto_media').Button = Y.Base.create('button', Y.M.editor_atto.EditorPlugin, [], {
430     initializer: function() {
431         if (this.get('host').canShowFilepicker('media')) {
432             this.editor.delegate('dblclick', this._displayDialogue, 'video', this);
433             this.editor.delegate('click', this._handleClick, 'video', this);
435             this.addButton({
436                 icon: 'e/insert_edit_video',
437                 callback: this._displayDialogue,
438                 tags: 'video, audio',
439                 tagMatchRequiresAll: false
440             });
441         }
442     },
444     /**
445      * Gets the root context for all templates, with extra supplied context.
446      *
447      * @method _getContext
448      * @param  {Object} extra The extra context to add
449      * @return {Object}
450      * @private
451      */
452     _getContext: function(extra) {
453         return Y.merge({
454             elementid: this.get('host').get('elementid'),
455             component: COMPONENTNAME,
456             langsinstalled: this.get('langs').installed,
457             langsavailable: this.get('langs').available,
458             helpStrings: this.get('help'),
459             CSS: CSS
460         }, extra);
461     },
463     /**
464      * Handles a click on a media element.
465      *
466      * @method _handleClick
467      * @param  {EventFacade} e
468      * @private
469      */
470     _handleClick: function(e) {
471         var medium = e.target;
473         var selection = this.get('host').getSelectionFromNode(medium);
474         if (this.get('host').getSelection() !== selection) {
475             this.get('host').setSelection(selection);
476         }
477     },
479     /**
480      * Display the media editing tool.
481      *
482      * @method _displayDialogue
483      * @private
484      */
485     _displayDialogue: function() {
486         if (this.get('host').getSelection() === false) {
487             return;
488         }
490         if (!('renderPartial' in Y.Handlebars.helpers)) {
491             (function smashPartials(chain, obj) {
492                 Y.each(obj, function(value, index) {
493                     chain.push(index);
494                     if (typeof value !== "object") {
495                         Y.Handlebars.registerPartial(chain.join('.').toLowerCase(), value);
496                     } else {
497                         smashPartials(chain, value);
498                     }
499                     chain.pop();
500                 });
501             })([], TEMPLATES);
503             Y.Handlebars.registerHelper('renderPartial', function(partialName, options) {
504                 if (!partialName) {
505                     return '';
506                 }
508                 var partial = Y.Handlebars.partials[partialName];
509                 var parentContext = options.hash.context ? Y.clone(options.hash.context) : {};
510                 var context = Y.merge(parentContext, options.hash);
511                 delete context.context;
513                 if (!partial) {
514                     return '';
515                 }
516                 return new Y.Handlebars.SafeString(Y.Handlebars.compile(partial)(context));
517             });
518         }
520         var dialogue = this.getDialogue({
521             headerContent: M.util.get_string('createmedia', COMPONENTNAME),
522             focusAfterHide: true,
523             width: 660,
524             focusOnShowSelector: SELECTORS.URL_INPUT
525         });
527         // Set the dialogue content, and then show the dialogue.
528         dialogue.set('bodyContent', this._getDialogueContent(this.get('host').getSelection())).show();
529         M.form.shortforms({formid: this.get('host').get('elementid') + '_atto_media_form'});
530     },
532     /**
533      * Returns the dialogue content for the tool.
534      *
535      * @method _getDialogueContent
536      * @param  {WrappedRange[]} selection Current editor selection
537      * @return {Y.Node}
538      * @private
539      */
540     _getDialogueContent: function(selection) {
541         var content = Y.Node.create(
542             Y.Handlebars.compile(TEMPLATES.ROOT)(this._getContext())
543         );
545         var medium = this.get('host').getSelectedNodes().filter('video,audio').shift();
546         var mediumProperties = medium ? this._getMediumProperties(medium) : false;
547         return this._attachEvents(this._applyMediumProperties(content, mediumProperties), selection);
548     },
550     /**
551      * Attaches required events to the content node.
552      *
553      * @method _attachEvents
554      * @param  {Y.Node}         content The content to which events will be attached
555      * @param  {WrappedRange[]} selection Current editor selection
556      * @return {Y.Node}
557      * @private
558      */
559     _attachEvents: function(content, selection) {
560         // Delegate add component link for media source fields.
561         content.delegate('click', function(e) {
562             e.preventDefault();
563             this._addMediaSourceComponent(e.currentTarget);
564         }, SELECTORS.MEDIA_SOURCE + ' .addcomponent', this);
566         // Delegate add component link for track fields.
567         content.delegate('click', function(e) {
568             e.preventDefault();
569             this._addTrackComponent(e.currentTarget);
570         }, SELECTORS.TRACK + ' .addcomponent', this);
572         // Only allow one track per tab to be selected as "default".
573         content.delegate('click', function(e) {
574             var element = e.currentTarget;
575             if (element.get('checked')) {
576                 var getKind = function(el) {
577                     return this._getTrackTypeFromTabPane(el.ancestor('.tab-pane'));
578                 }.bind(this);
580                 element.ancestor('.root.tab-content').all(SELECTORS.TRACK_DEFAULT_SELECT).each(function(select) {
581                     if (select !== element && getKind(element) === getKind(select)) {
582                         select.set('checked', false);
583                     }
584                 });
585             }
586         }, SELECTORS.TRACK_DEFAULT_SELECT, this);
588         // Set up filepicker click event.
589         content.delegate('click', function(e) {
590             var element = e.currentTarget;
591             var fptype = (element.ancestor(SELECTORS.POSTER_SOURCE) && 'image') ||
592                     (element.ancestor(SELECTORS.TRACK_SOURCE) && 'subtitle') ||
593                     'media';
594             e.preventDefault();
595             this.get('host').showFilepicker(fptype, this._getFilepickerCallback(element, fptype), this);
596         }, '.openmediabrowser', this);
598         // This is a nasty hack. Basically we are using BS4 markup for the tabs
599         // but it isn't completely backwards compatible with BS2. The main problem is
600         // that the "active" class goes on a different node. So the idea is to put it
601         // the node for BS4, and then use CSS to make it look right in BS2. However,
602         // once another tab is clicked, everything sorts itself out, more or less. Except
603         // that the original "active" tab hasn't had the BS4 "active" class removed
604         // (so the styles will still apply to it). So we need to remove the "active"
605         // class on the BS4 node so that BS2 is happy.
606         //
607         // This doesn't upset BS4 since it removes this class anyway when clicking on
608         // another tab.
609         content.all('.nav-item').on('click', function(elem) {
610             elem.currentTarget.get('parentNode').all('.active').removeClass('active');
611         });
613         content.one('.submit').on('click', function(e) {
614             e.preventDefault();
615             var mediaHTML = this._getMediaHTML(e.currentTarget.ancestor('.atto_form')),
616                 host = this.get('host');
617             this.getDialogue({
618                 focusAfterHide: null
619             }).hide();
620             if (mediaHTML) {
621                 host.setSelection(selection);
622                 host.insertContentAtFocusPoint(mediaHTML);
623                 this.markUpdated();
624             }
625         }, this);
627         return content;
628     },
630     /**
631      * Applies medium properties to the content node.
632      *
633      * @method _applyMediumProperties
634      * @param  {Y.Node} content The content to apply the properties to
635      * @param  {object} properties The medium properties to apply
636      * @return {Y.Node}
637      * @private
638      */
639     _applyMediumProperties: function(content, properties) {
640         if (!properties) {
641             return content;
642         }
644         var applyTrackProperties = function(track, properties) {
645             track.one(SELECTORS.TRACK_SOURCE + ' ' + SELECTORS.URL_INPUT).set('value', properties.src);
646             track.one(SELECTORS.TRACK_LANG_INPUT).set('value', properties.srclang);
647             track.one(SELECTORS.TRACK_LABEL_INPUT).set('value', properties.label);
648             track.one(SELECTORS.TRACK_DEFAULT_SELECT).set('checked', properties.defaultTrack);
649         };
651         var tabPane = content.one('.root.tab-content > .tab-pane#' + this.get('host').get('elementid') +
652                               '_' + properties.type.toLowerCase());
654         // Populate sources.
655         tabPane.one(SELECTORS.MEDIA_SOURCE + ' ' + SELECTORS.URL_INPUT).set('value', properties.sources[0]);
656         Y.Array.each(properties.sources.slice(1), function(source) {
657             this._addMediaSourceComponent(tabPane.one(SELECTORS.MEDIA_SOURCE + ' .addcomponent'), function(newComponent) {
658                 newComponent.one(SELECTORS.URL_INPUT).set('value', source);
659             });
660         }, this);
662         // Populate tracks.
663         Y.Object.each(properties.tracks, function(value, key) {
664             var trackData = value.length ? value : [{src: '', srclang: '', label: '', defaultTrack: false}];
665             var paneSelector = SELECTORS['TRACK_' + key.toUpperCase() + '_PANE'];
667             applyTrackProperties(tabPane.one(paneSelector + ' ' + SELECTORS.TRACK), trackData[0]);
668             Y.Array.each(trackData.slice(1), function(track) {
669                 this._addTrackComponent(
670                     tabPane.one(paneSelector + ' ' + SELECTORS.TRACK + ' .addcomponent'), function(newComponent) {
671                     applyTrackProperties(newComponent, track);
672                 });
673             }, this);
674         }, this);
676         // Populate values.
677         tabPane.one(SELECTORS.TITLE_INPUT).set('value', properties.title);
678         tabPane.one(SELECTORS.MEDIA_CONTROLS_TOGGLE).set('checked', properties.controls);
679         tabPane.one(SELECTORS.MEDIA_AUTOPLAY_TOGGLE).set('checked', properties.autoplay);
680         tabPane.one(SELECTORS.MEDIA_MUTE_TOGGLE).set('checked', properties.muted);
681         tabPane.one(SELECTORS.MEDIA_LOOP_TOGGLE).set('checked', properties.loop);
683         // Determine medium type.
684         var mediumType = this._getMediumTypeFromTabPane(tabPane);
686         if (mediumType === 'video') {
687             // Populate values unique for video.
688             tabPane.one(SELECTORS.POSTER_SOURCE + ' ' + SELECTORS.URL_INPUT).setAttribute('value', properties.poster);
689             tabPane.one(SELECTORS.WIDTH_INPUT).set('value', properties.width);
690             tabPane.one(SELECTORS.HEIGHT_INPUT).set('value', properties.height);
691         }
693         // Switch to the correct tab.
694         // Remove active class from all tabs + tab panes.
695         tabPane.siblings('.active').removeClass('active');
696         content.all('.root.nav-tabs .nav-item a').removeClass('active');
698         // Add active class to the desired tab and tab pane.
699         tabPane.addClass('active');
700         content.one(SELECTORS[mediumType.toUpperCase() + '_TAB'] + ' a').addClass('active');
702         return content;
703     },
705     /**
706      * Extracts medium properties.
707      *
708      * @method _getMediumProperties
709      * @param  {Y.Node} medium The medium node from which to extract
710      * @return {Object}
711      * @private
712      */
713     _getMediumProperties: function(medium) {
714         var boolAttr = function(elem, attr) {
715             return elem.getAttribute(attr) ? true : false;
716         };
718         var tracks = {
719             subtitles: [],
720             captions: [],
721             descriptions: [],
722             chapters: [],
723             metadata: []
724         };
726         medium.all('track').each(function(track) {
727             tracks[track.getAttribute('kind')].push({
728                 src: track.getAttribute('src'),
729                 srclang: track.getAttribute('srclang'),
730                 label: track.getAttribute('label'),
731                 defaultTrack: boolAttr(track, 'default')
732             });
733         });
735         return {
736             type: medium.test('video') ? MEDIA_TYPES.VIDEO : MEDIA_TYPES.AUDIO,
737             sources: medium.all('source').get('src'),
738             poster: medium.getAttribute('poster'),
739             title: medium.getAttribute('title'),
740             width: medium.getAttribute('width'),
741             height: medium.getAttribute('height'),
742             autoplay: boolAttr(medium, 'autoplay'),
743             loop: boolAttr(medium, 'loop'),
744             muted: boolAttr(medium, 'muted'),
745             controls: boolAttr(medium, 'controls'),
746             tracks: tracks
747         };
748     },
750     /**
751      * Adds a track form component.
752      *
753      * @method _addTrackComponent
754      * @param  {Y.Node}   element    The element which was used to trigger this function
755      * @param  {Function} [callback] Function to be called when the new component is added
756      *     @param {Y.Node}    callback.newComponent The compiled component
757      * @private
758      */
759     _addTrackComponent: function(element, callback) {
760         var trackType = this._getTrackTypeFromTabPane(element.ancestor('.tab-pane'));
761         var context = this._getContext({
762             sourcelabel: trackType + 'sourcelabel',
763             addcomponentlabel: 'add' + trackType + 'track'
764         });
766         this._addComponent(element, TEMPLATES.FORM_COMPONENTS.TRACK, SELECTORS.TRACK, context, callback);
767     },
769     /**
770      * Adds a media source form component.
771      *
772      * @method _addMediaSourceComponent
773      * @param  {Y.Node}   element    The element which was used to trigger this function
774      * @param  {Function} [callback] Function to be called when the new component is added
775      *     @param {Y.Node}    callback.newComponent The compiled component
776      * @private
777      */
778     _addMediaSourceComponent: function(element, callback) {
779         var mediumType = this._getMediumTypeFromTabPane(element.ancestor('.tab-pane'));
780         var context = this._getContext({
781             multisource: true,
782             id: CSS.MEDIA_SOURCE,
783             entersourcelabel: mediumType + 'sourcelabel',
784             addcomponentlabel: 'addsource',
785             addsourcehelp: this.get('help').addsource
786         });
787         this._addComponent(element, TEMPLATES.FORM_COMPONENTS.SOURCE, SELECTORS.MEDIA_SOURCE, context, callback);
788     },
790     /**
791      * Adds an arbitrary form component.
792      *
793      * This function Compiles and adds the provided component in the supplied 'ancestor' container.
794      * It will also add links to add/remove the relevant components, attaching the
795      * necessary events.
796      *
797      * @method _addComponent
798      * @param  {Y.Node}   element    The element which was used to trigger this function
799      * @param  {String}   component  The component to compile and add
800      * @param  {String}   ancestor   A selector used to find an ancestor of 'component', to which
801      *                               the compiled component will be appended
802      * @param  {Object}   context    The context with which to render the component
803      * @param  {Function} [callback] Function to be called when the new component is added
804      *     @param {Y.Node}    callback.newComponent The compiled component
805      * @private
806      */
807     _addComponent: function(element, component, ancestor, context, callback) {
808         var currentComponent = element.ancestor(ancestor),
809             newComponent = Y.Node.create(Y.Handlebars.compile(component)(context)),
810             removeNodeContext = this._getContext(context);
812         removeNodeContext.label = "remove";
813         var removeNode = Y.Node.create(Y.Handlebars.compile(TEMPLATES.FORM_COMPONENTS.REMOVE_COMPONENT)(removeNodeContext));
815         removeNode.one('.removecomponent').on('click', function(e) {
816             e.preventDefault();
817             currentComponent.remove(true);
818         });
820         currentComponent.insert(newComponent, 'after');
821         element.ancestor().insert(removeNode, 'after');
822         element.ancestor().remove(true);
824         if (callback) {
825             callback.call(this, newComponent);
826         }
827     },
829     /**
830      * Returns the callback for the file picker to call after a file has been selected.
831      *
832      * @method _getFilepickerCallback
833      * @param  {Y.Node} element The element which triggered the callback
834      * @param  {String} fptype  The file pickertype (as would be passed to `showFilePicker`)
835      * @return {Function} The function to be used as a callback when the file picker returns the file
836      * @private
837      */
838     _getFilepickerCallback: function(element, fptype) {
839         return function(params) {
840             if (params.url !== '') {
841                 var tabPane = element.ancestor('.tab-pane');
842                 element.ancestor(SELECTORS.SOURCE).one(SELECTORS.URL_INPUT).set('value', params.url);
844                 // Links (and only links) have a name field.
845                 if (tabPane.get('id') === this.get('host').get('elementid') + '_' + CSS.LINK) {
846                     tabPane.one(SELECTORS.NAME_INPUT).set('value', params.file);
847                 }
849                 if (fptype === 'subtitle') {
850                     var subtitleLang = params.file.split('.vtt')[0].split('-').slice(-1)[0];
851                     var langObj = this.get('langs').available.reduce(function(carry, lang) {
852                         return lang.code === subtitleLang ? lang : carry;
853                     }, false);
854                     if (langObj) {
855                         element.ancestor(SELECTORS.TRACK).one(SELECTORS.TRACK_LABEL_INPUT).set('value',
856                                 langObj.lang.substr(0, langObj.lang.lastIndexOf(' ')));
857                         element.ancestor(SELECTORS.TRACK).one(SELECTORS.TRACK_LANG_INPUT).set('value', langObj.code);
858                     }
859                 }
860             }
861         };
862     },
864     /**
865      * Given a "medium" tab pane, returns what kind of medium it contains.
866      *
867      * @method _getMediumTypeFromTabPane
868      * @param  {Y.Node} tabPane The tab pane
869      * @return {String} The type of medium in the pane
870      */
871     _getMediumTypeFromTabPane: function(tabPane) {
872         return tabPane.getAttribute('data-medium-type');
873     },
875     /**
876      * Given a "track" tab pane, returns what kind of track it contains.
877      *
878      * @method _getTrackTypeFromTabPane
879      * @param  {Y.Node} tabPane The tab pane
880      * @return {String} The type of track in the pane
881      */
882     _getTrackTypeFromTabPane: function(tabPane) {
883         return tabPane.getAttribute('data-track-kind');
884     },
886     /**
887      * Returns the HTML to be inserted to the text area.
888      *
889      * @method _getMediaHTML
890      * @param  {Y.Node} form The form from which to extract data
891      * @return {String} The compiled markup
892      * @private
893      */
894     _getMediaHTML: function(form) {
895         var mediumType = this._getMediumTypeFromTabPane(form.one('.root.tab-content > .tab-pane.active'));
896         var tabContent = form.one(SELECTORS[mediumType.toUpperCase() + '_PANE']);
898         return this['_getMediaHTML' + mediumType[0].toUpperCase() + mediumType.substr(1)](tabContent);
899     },
901     /**
902      * Returns the HTML to be inserted to the text area for the link tab.
903      *
904      * @method _getMediaHTMLLink
905      * @param  {Y.Node} tab The tab from which to extract data
906      * @return {String} The compiled markup
907      * @private
908      */
909     _getMediaHTMLLink: function(tab) {
910         var context = {
911             url: tab.one(SELECTORS.URL_INPUT).get('value'),
912             name: tab.one(SELECTORS.NAME_INPUT).get('value') || false
913         };
915         return context.url ? Y.Handlebars.compile(TEMPLATES.HTML_MEDIA.LINK)(context) : '';
916     },
918     /**
919      * Returns the HTML to be inserted to the text area for the video tab.
920      *
921      * @method _getMediaHTMLVideo
922      * @param  {Y.Node} tab The tab from which to extract data
923      * @return {String} The compiled markup
924      * @private
925      */
926     _getMediaHTMLVideo: function(tab) {
927         var context = this._getContextForMediaHTML(tab);
928         context.width = tab.one(SELECTORS.WIDTH_INPUT).get('value') || false;
929         context.height = tab.one(SELECTORS.HEIGHT_INPUT).get('value') || false;
930         context.poster = tab.one(SELECTORS.POSTER_SOURCE + ' ' + SELECTORS.URL_INPUT).get('value') || false;
932         return context.sources.length ? Y.Handlebars.compile(TEMPLATES.HTML_MEDIA.VIDEO)(context) : '';
933     },
935     /**
936      * Returns the HTML to be inserted to the text area for the audio tab.
937      *
938      * @method _getMediaHTMLAudio
939      * @param  {Y.Node} tab The tab from which to extract data
940      * @return {String} The compiled markup
941      * @private
942      */
943     _getMediaHTMLAudio: function(tab) {
944         var context = this._getContextForMediaHTML(tab);
946         return context.sources.length ? Y.Handlebars.compile(TEMPLATES.HTML_MEDIA.AUDIO)(context) : '';
947     },
949     /**
950      * Returns the context with which to render a media template.
951      *
952      * @method _getContextForMediaHTML
953      * @param  {Y.Node} tab The tab from which to extract data
954      * @return {Object}
955      * @private
956      */
957     _getContextForMediaHTML: function(tab) {
958         var tracks = [];
960         tab.all(SELECTORS.TRACK).each(function(track) {
961             tracks.push({
962                 track: track.one(SELECTORS.TRACK_SOURCE + ' ' + SELECTORS.URL_INPUT).get('value'),
963                 kind: this._getTrackTypeFromTabPane(track.ancestor('.tab-pane')),
964                 label: track.one(SELECTORS.TRACK_LABEL_INPUT).get('value') ||
965                     track.one(SELECTORS.TRACK_LANG_INPUT).get('value'),
966                 srclang: track.one(SELECTORS.TRACK_LANG_INPUT).get('value'),
967                 defaultTrack: track.one(SELECTORS.TRACK_DEFAULT_SELECT).get('checked') ? "true" : null
968             });
969         }, this);
971         return {
972             sources: tab.all(SELECTORS.MEDIA_SOURCE + ' ' + SELECTORS.URL_INPUT).get('value').filter(function(source) {
973                 return !!source;
974             }).map(function(source) {
975                 return {source: source};
976             }),
977             description: tab.one(SELECTORS.MEDIA_SOURCE + ' ' + SELECTORS.URL_INPUT).get('value') || false,
978             tracks: tracks.filter(function(track) {
979                 return !!track.track;
980             }),
981             showControls: tab.one(SELECTORS.MEDIA_CONTROLS_TOGGLE).get('checked'),
982             autoplay: tab.one(SELECTORS.MEDIA_AUTOPLAY_TOGGLE).get('checked'),
983             muted: tab.one(SELECTORS.MEDIA_MUTE_TOGGLE).get('checked'),
984             loop: tab.one(SELECTORS.MEDIA_LOOP_TOGGLE).get('checked'),
985             title: tab.one(SELECTORS.TITLE_INPUT).get('value') || false
986         };
987     }
988 }, {
989     ATTRS: {
990         langs: {},
991         help: {}
992     }
993 });