/wp-content/plugins/ocean-extra/includes/metabox/butterbean/js/butterbean.js

https://bitbucket.org/zshi0001/fit5120-kic · JavaScript · 875 lines · 325 code · 169 blank · 381 comment · 24 complexity · 4c06532c479ccd8ac8b1f9933a3b29fe MD5 · raw file

  1. window.butterbean = window.butterbean || {};
  2. ( function() {
  3. // Bail if we don't have the JSON, which is passed in via `wp_localize_script()`.
  4. if ( _.isUndefined( butterbean_data ) ) {
  5. return;
  6. }
  7. /**
  8. * Our global object. The `butterbean` object is just a wrapper to house everything
  9. * in a single namespace.
  10. *
  11. * @since 1.0.0
  12. * @access public
  13. * @var object
  14. */
  15. var api = butterbean = {
  16. /**
  17. * Houses the manager, section, and control views based on the `type`.
  18. *
  19. * @since 1.0.0
  20. * @access public
  21. * @var object
  22. */
  23. views : { managers : {}, sections : {}, controls : {} },
  24. /**
  25. * Houses the manager, section, and control templates based on the `type`.
  26. *
  27. * @since 1.0.0
  28. * @access public
  29. * @var object
  30. */
  31. templates : { managers : {}, sections : {}, controls : {} }
  32. };
  33. /**
  34. * Creates a new manager view.
  35. *
  36. * @since 1.0.0
  37. * @access public
  38. * @param string $type
  39. * @param object $args
  40. * @return void
  41. */
  42. api.views.register_manager = function( type, args ) {
  43. if ( 'default' !== type )
  44. this.managers[ type ] = this.managers.default.extend( args );
  45. };
  46. /**
  47. * Returns a manager view.
  48. *
  49. * @since 1.0.0
  50. * @access public
  51. * @param string $type
  52. * @return object
  53. */
  54. api.views.get_manager = function( type ) {
  55. if ( this.manager_exists( type ) )
  56. return this.managers[ type ];
  57. return this.managers.default;
  58. };
  59. /**
  60. * Removes a manager view.
  61. *
  62. * @since 1.0.0
  63. * @access public
  64. * @param string $type
  65. * @return void
  66. */
  67. api.views.unregister_manager = function( type ) {
  68. if ( 'default' !== type && this.manager_exists( type ) )
  69. delete this.managers[ type ];
  70. };
  71. /**
  72. * Checks if a manager view exists.
  73. *
  74. * @since 1.0.0
  75. * @access public
  76. * @param string $type
  77. * @return bool
  78. */
  79. api.views.manager_exists = function( type ) {
  80. return this.managers.hasOwnProperty( type );
  81. };
  82. /**
  83. * Creates a new section view.
  84. *
  85. * @since 1.0.0
  86. * @access public
  87. * @param string $type
  88. * @param object $args
  89. * @return void
  90. */
  91. api.views.register_section = function( type, args ) {
  92. if ( 'default' !== type )
  93. this.sections[ type ] = this.sections.default.extend( args );
  94. };
  95. /**
  96. * Returns a section view.
  97. *
  98. * @since 1.0.0
  99. * @access public
  100. * @param string $type
  101. * @return object
  102. */
  103. api.views.get_section = function( type ) {
  104. if ( this.section_exists( type ) )
  105. return this.sections[ type ];
  106. return this.sections.default;
  107. };
  108. /**
  109. * Removes a section view.
  110. *
  111. * @since 1.0.0
  112. * @access public
  113. * @param string $type
  114. * @return void
  115. */
  116. api.views.unregister_section = function( type ) {
  117. if ( 'default' !== type && this.section_exists( type ) )
  118. delete this.sections[ type ];
  119. };
  120. /**
  121. * Checks if a section view exists.
  122. *
  123. * @since 1.0.0
  124. * @access public
  125. * @param string $type
  126. * @return bool
  127. */
  128. api.views.section_exists = function( type ) {
  129. return this.sections.hasOwnProperty( type );
  130. };
  131. /**
  132. * Creates a new control view.
  133. *
  134. * @since 1.0.0
  135. * @access public
  136. * @param string $type
  137. * @param object $args
  138. * @return void
  139. */
  140. api.views.register_control = function( type, args ) {
  141. if ( 'default' !== type )
  142. this.controls[ type ] = this.controls.default.extend( args );
  143. };
  144. /**
  145. * Returns a control view.
  146. *
  147. * @since 1.0.0
  148. * @access public
  149. * @param string $type
  150. * @return object
  151. */
  152. api.views.get_control = function( type ) {
  153. if ( this.control_exists( type ) )
  154. return this.controls[ type ];
  155. return this.controls.default;
  156. };
  157. /**
  158. * Removes a control view.
  159. *
  160. * @since 1.0.0
  161. * @access public
  162. * @param string $type
  163. * @return void
  164. */
  165. api.views.unregister_control = function( type ) {
  166. if ( 'default' !== type && this.control_exists( type ) )
  167. delete this.controls[ type ];
  168. };
  169. /**
  170. * Checks if a control view exists.
  171. *
  172. * @since 1.0.0
  173. * @access public
  174. * @param string $type
  175. * @return bool
  176. */
  177. api.views.control_exists = function( type ) {
  178. return this.controls.hasOwnProperty( type );
  179. };
  180. /**
  181. * Creates a new manager template.
  182. *
  183. * @since 1.0.0
  184. * @access public
  185. * @param string $type
  186. * @param object $args
  187. * @return void
  188. */
  189. api.templates.register_manager = function( type ) {
  190. this.managers[ type ] = wp.template( 'butterbean-manager-' + type );
  191. };
  192. /**
  193. * Returns a manager template.
  194. *
  195. * @since 1.0.0
  196. * @access public
  197. * @param string $type
  198. * @return function
  199. */
  200. api.templates.get_manager = function( type ) {
  201. return this.manager_exists( type ) ? this.managers[ type ] : false;
  202. };
  203. /**
  204. * Removes a manager template.
  205. *
  206. * @since 1.0.0
  207. * @access public
  208. * @param string $type
  209. * @return void
  210. */
  211. api.templates.unregister_manager = function( type ) {
  212. if ( this.manager_exists( type ) )
  213. delete this.managers[ type ];
  214. };
  215. /**
  216. * Checks if a manager template exists.
  217. *
  218. * @since 1.0.0
  219. * @access public
  220. * @param string $type
  221. * @return bool
  222. */
  223. api.templates.manager_exists = function( type ) {
  224. return this.managers.hasOwnProperty( type );
  225. };
  226. /**
  227. * Creates a new section template.
  228. *
  229. * @since 1.0.0
  230. * @access public
  231. * @param string $type
  232. * @param object $args
  233. * @return void
  234. */
  235. api.templates.register_section = function( type ) {
  236. this.sections[ type ] = wp.template( 'butterbean-section-' + type );
  237. };
  238. /**
  239. * Returns a section template.
  240. *
  241. * @since 1.0.0
  242. * @access public
  243. * @param string $type
  244. * @return function
  245. */
  246. api.templates.get_section = function( type ) {
  247. return this.section_exists( type ) ? this.sections[ type ] : false;
  248. };
  249. /**
  250. * Removes a section template.
  251. *
  252. * @since 1.0.0
  253. * @access public
  254. * @param string $type
  255. * @return void
  256. */
  257. api.templates.unregister_section = function( type ) {
  258. if ( this.section_exists( type ) )
  259. delete this.sections[ type ];
  260. };
  261. /**
  262. * Checks if a section template exists.
  263. *
  264. * @since 1.0.0
  265. * @access public
  266. * @param string $type
  267. * @return bool
  268. */
  269. api.templates.section_exists = function( type ) {
  270. return this.sections.hasOwnProperty( type );
  271. };
  272. /**
  273. * Creates a new control template.
  274. *
  275. * @since 1.0.0
  276. * @access public
  277. * @param string $type
  278. * @param object $args
  279. * @return void
  280. */
  281. api.templates.register_control = function( type ) {
  282. this.controls[ type ] = wp.template( 'butterbean-control-' + type );
  283. };
  284. /**
  285. * Returns a control template.
  286. *
  287. * @since 1.0.0
  288. * @access public
  289. * @param string $type
  290. * @return function
  291. */
  292. api.templates.get_control = function( type ) {
  293. return this.control_exists( type ) ? this.controls[ type ] : false;
  294. };
  295. /**
  296. * Removes a control template.
  297. *
  298. * @since 1.0.0
  299. * @access public
  300. * @param string $type
  301. * @return void
  302. */
  303. api.templates.unregister_control = function( type ) {
  304. if ( this.control_exists( type ) )
  305. delete this.controls[ type ];
  306. };
  307. /**
  308. * Checks if a control template exists.
  309. *
  310. * @since 1.0.0
  311. * @access public
  312. * @param string $type
  313. * @return bool
  314. */
  315. api.templates.control_exists = function( type ) {
  316. return this.controls.hasOwnProperty( type );
  317. };
  318. /**
  319. * Renders our managers, sections, and controls.
  320. *
  321. * @since 1.0.0
  322. * @access private
  323. * @return void
  324. */
  325. api.render = function() {
  326. // Loop through each of the managers and render their api.views.
  327. _.each( butterbean_data.managers, function( data ) {
  328. // Create a new manager model with the JSON data for the manager.
  329. var manager = new Manager( data );
  330. // Get the manager view callback.
  331. var callback = api.views.get_manager( data.type );
  332. // Create a new manager view.
  333. var view = new callback( { model : manager } );
  334. // Get the meta box element.
  335. var metabox = document.getElementById( 'butterbean-ui-' + manager.get( 'name' ) );
  336. // Add the `.butterbean-ui` class to the meta box.
  337. metabox.className += ' butterbean-ui';
  338. // Render the manager view.
  339. metabox.querySelector( '.inside' ).appendChild( view.render().el );
  340. // Render the manager subviews.
  341. view.subview_render();
  342. // Call the view's ready method.
  343. view.ready();
  344. } );
  345. };
  346. /* === Templates === */
  347. // Nav template.
  348. var nav_template = wp.template( 'butterbean-nav' );
  349. /* === Models === */
  350. // Manager model (each manager is housed within a meta box).
  351. var Manager = Backbone.Model.extend( {
  352. defaults : {
  353. name : '',
  354. type : '',
  355. sections : {},
  356. controls : {}
  357. }
  358. } );
  359. // Section model (each section belongs to a manager).
  360. var Section = Backbone.Model.extend( {
  361. defaults : {
  362. name : '',
  363. type : '',
  364. label : '',
  365. description : '',
  366. icon : '',
  367. manager : '',
  368. active : '',
  369. selected : false
  370. }
  371. } );
  372. // Control model (each control belongs to a manager and section).
  373. var Control = Backbone.Model.extend( {
  374. defaults : {
  375. name : '',
  376. type : '',
  377. label : '',
  378. description : '',
  379. icon : '',
  380. value : '',
  381. choices : {},
  382. attr : '',
  383. active : '',
  384. manager : '',
  385. section : '',
  386. setting : ''
  387. }
  388. } );
  389. /* === Collections === */
  390. /**
  391. * Stores our collection of section models.
  392. *
  393. * @since 1.0.0
  394. * @access private
  395. * @var object
  396. */
  397. var Sections = Backbone.Collection.extend( {
  398. model : Section
  399. } );
  400. /* === Views === */
  401. /**
  402. * The default manager view. Other views can extend this using the
  403. * `butterbean.views.register_manager()` function.
  404. *
  405. * @since 1.0.0
  406. * @access public
  407. * @var object
  408. */
  409. api.views.managers[ 'default' ] = Backbone.View.extend( {
  410. // Wrapper element for the manager view.
  411. tagName : 'div',
  412. // Adds some custom attributes to the wrapper.
  413. attributes : function() {
  414. return {
  415. 'id' : 'butterbean-manager-' + this.model.get( 'name' ),
  416. 'class' : 'butterbean-manager butterbean-manager-' + this.model.get( 'type' )
  417. };
  418. },
  419. // Initializes the view.
  420. initialize : function() {
  421. var type = this.model.get( 'type' );
  422. // If there's not yet a template for this manager type, create it.
  423. if ( ! api.templates.manager_exists( type ) )
  424. api.templates.register_manager( type );
  425. // Get the manager template.
  426. this.template = api.templates.get_manager( type );
  427. },
  428. // Renders the manager.
  429. render : function() {
  430. this.el.innerHTML = this.template( this.model.toJSON() );
  431. return this;
  432. },
  433. // Renders the manager's sections and controls.
  434. // Important! This may change drastically in the future, possibly even
  435. // taken out of the manager view altogether. It's for this reason that
  436. // it's not recommended to create custom views for managers right now.
  437. subview_render : function() {
  438. // Create a new section collection.
  439. var sections = new Sections();
  440. // Loop through each section and add it to the collection.
  441. _.each( this.model.get( 'sections' ), function( data ) {
  442. sections.add( new Section( data ) );
  443. } );
  444. // Loop through each section in the collection and render its view.
  445. sections.forEach( function( section, i ) {
  446. // Create a new nav item view for the section.
  447. var nav_view = new Nav_View( { model : section } );
  448. // Render the nav item view.
  449. document.querySelector( '#butterbean-ui-' + section.get( 'manager' ) + ' .butterbean-nav' ).appendChild( nav_view.render().el );
  450. // Get the section view callback.
  451. var callback = api.views.get_section( section.attributes.type );
  452. // Create a new section view.
  453. var view = new callback( { model : section } );
  454. // Render the section view.
  455. document.querySelector( '#butterbean-ui-' + section.get( 'manager' ) + ' .butterbean-content' ).appendChild( view.render().el );
  456. // Call the section view's ready method.
  457. view.ready();
  458. // If the first model, set it to selected.
  459. section.set( 'selected', 0 === i );
  460. }, this );
  461. // Loop through each control for the manager and render its view.
  462. _.each( this.model.get( 'controls' ), function( data ) {
  463. // Create a new control model.
  464. var control = new Control( data );
  465. // Get the control view callback.
  466. var callback = api.views.get_control( data.type );
  467. // Create a new control view.
  468. var view = new callback( { model : control } );
  469. // Render the view.
  470. document.getElementById( 'butterbean-' + control.get( 'manager' ) + '-section-' + control.get( 'section' ) ).appendChild( view.render().el );
  471. // Call the view's ready method.
  472. view.ready();
  473. } );
  474. return this;
  475. },
  476. // Function that is executed *after* the view has been rendered.
  477. // This is meant to be overwritten in sub-views.
  478. ready : function() {}
  479. } );
  480. /**
  481. * The default section view. Other views can extend this using the
  482. * `butterbean.views.register_section()` function.
  483. *
  484. * @since 1.0.0
  485. * @access public
  486. * @var object
  487. */
  488. api.views.sections[ 'default' ] = Backbone.View.extend( {
  489. // Wrapper element for the section.
  490. tagName : 'div',
  491. // Adds custom attributes for the section wrapper.
  492. attributes : function() {
  493. return {
  494. 'id' : 'butterbean-' + this.model.get( 'manager' ) + '-section-' + this.model.get( 'name' ),
  495. 'class' : 'butterbean-section butterbean-section-' + this.model.get( 'type' ),
  496. 'aria-hidden' : ! this.model.get( 'selected' )
  497. };
  498. },
  499. // Initializes the view.
  500. initialize : function() {
  501. // Add an event for when the model changes.
  502. this.model.on( 'change', this.onchange, this );
  503. // Get the section type.
  504. var type = this.model.get( 'type' );
  505. // If there's no template for this section type, create it.
  506. if ( ! api.templates.section_exists( type ) )
  507. api.templates.register_section( type );
  508. // Gets the section template.
  509. this.template = api.templates.get_section( type );
  510. },
  511. // Renders the section.
  512. render : function() {
  513. // Only render template if model is active.
  514. if ( this.model.get( 'active' ) )
  515. this.el.innerHTML = this.template( this.model.toJSON() );
  516. return this;
  517. },
  518. // Executed when the model changes.
  519. onchange : function() {
  520. // Set the view's `aria-hidden` attribute based on whether the model is selected.
  521. this.el.setAttribute( 'aria-hidden', ! this.model.get( 'selected' ) );
  522. },
  523. // Function that is executed *after* the view has been rendered.
  524. // This is meant to be overwritten in sub-views.
  525. ready : function() {}
  526. } );
  527. /**
  528. * The nav item view for each section.
  529. *
  530. * @since 1.0.0
  531. * @access public
  532. * @var object
  533. */
  534. var Nav_View = Backbone.View.extend( {
  535. // Sets the template used.
  536. template : nav_template,
  537. // Wrapper element for the nav item.
  538. tagName : 'li',
  539. // Sets some custom attributes for the nav item wrapper.
  540. attributes : function() {
  541. return {
  542. 'aria-selected' : this.model.get( 'selected' )
  543. };
  544. },
  545. // Initializes the nav item view.
  546. initialize : function() {
  547. this.model.on( 'change', this.render, this );
  548. this.model.on( 'change', this.onchange, this );
  549. },
  550. // Renders the nav item.
  551. render : function() {
  552. // Only render template if model is active.
  553. if ( this.model.get( 'active' ) )
  554. this.el.innerHTML = this.template( this.model.toJSON() );
  555. return this;
  556. },
  557. // Custom events.
  558. events : {
  559. 'click a' : 'onselect'
  560. },
  561. // Executed when the section model changes.
  562. onchange : function() {
  563. // Set the `aria-selected` attibute based on the model selected state.
  564. this.el.setAttribute( 'aria-selected', this.model.get( 'selected' ) );
  565. },
  566. // Executed when the link for the nav item is clicked.
  567. onselect : function( event ) {
  568. event.preventDefault();
  569. // Loop through each of the models in the collection and set them to inactive.
  570. _.each( this.model.collection.models, function( m ) {
  571. m.set( 'selected', false );
  572. }, this );
  573. // Set this view's model to selected.
  574. this.model.set( 'selected', true );
  575. }
  576. } );
  577. /**
  578. * The default control view. Other views can extend this using the
  579. * `butterbean.views.register_control()` function.
  580. *
  581. * @since 1.0.0
  582. * @access public
  583. * @var object
  584. */
  585. api.views.controls[ 'default' ] = Backbone.View.extend( {
  586. // Wrapper element for the control.
  587. tagName : 'div',
  588. // Custom attributes for the control wrapper.
  589. attributes : function() {
  590. return {
  591. 'id' : 'butterbean-control-' + this.model.get( 'name' ),
  592. 'class' : 'butterbean-control butterbean-control-' + this.model.get( 'type' )
  593. };
  594. },
  595. // Initiazlies the control view.
  596. initialize : function() {
  597. var type = this.model.get( 'type' );
  598. // Only add a new control template if we have a different control type.
  599. if ( ! api.templates.control_exists( type ) )
  600. api.templates.register_control( type );
  601. // Get the control template.
  602. this.template = api.templates.get_control( type );
  603. // Bind changes so that the view is re-rendered when the model changes.
  604. _.bindAll( this, 'render' );
  605. this.model.bind( 'change', this.render );
  606. },
  607. // Renders the control template.
  608. render : function() {
  609. // Only render template if model is active.
  610. if ( this.model.get( 'active' ) )
  611. this.el.innerHTML = this.template( this.model.toJSON() );
  612. return this;
  613. },
  614. // Function that is executed *after* the view has been rendered.
  615. // This is meant to be overwritten in sub-views.
  616. ready : function() {}
  617. } );
  618. /**
  619. * Adds the color control view.
  620. *
  621. * @since 1.0.0
  622. */
  623. api.views.register_control( 'color', {
  624. // Calls the core WP color picker for the control's input.
  625. ready : function() {
  626. var options = this.model.attributes.options;
  627. jQuery( this.$el ).find( '.butterbean-color-picker' ).wpColorPicker( options );
  628. }
  629. } );
  630. /**
  631. * Adds the color palette view.
  632. *
  633. * @since 1.0.0
  634. */
  635. api.views.register_control( 'palette', {
  636. // Adds custom events.
  637. events : {
  638. 'change input' : 'onselect'
  639. },
  640. // Executed when one of the color palette's value has changed.
  641. // These are radio inputs.
  642. onselect : function() {
  643. // Get the value of the input.
  644. var value = document.querySelector( '#' + this.el.id + ' input:checked' ).getAttribute( 'value' );
  645. // Get all choices.
  646. var choices = this.model.get( 'choices' );
  647. // Loop through choices and change the selected value.
  648. _.each( choices, function( choice, key ) {
  649. choice.selected = key === value;
  650. } );
  651. // Because `choices` is an array, it's not recognized as a change. So, we
  652. // have to manually trigger a change here so that the view gets re-rendered.
  653. this.model.set( 'choices', choices ).trigger( 'change', this.model );
  654. }
  655. } );
  656. /**
  657. * Adds the image control view.
  658. *
  659. * @since 1.0.0
  660. */
  661. api.views.register_control( 'image', {
  662. // Adds custom events.
  663. events : {
  664. 'click .butterbean-add-media' : 'showmodal',
  665. 'click .butterbean-change-media' : 'showmodal',
  666. 'click .butterbean-remove-media' : 'removemedia'
  667. },
  668. // Executed when the show modal button is clicked.
  669. showmodal : function() {
  670. // If we already have a media modal, open it.
  671. if ( ! _.isUndefined( this.media_modal ) ) {
  672. this.media_modal.open();
  673. return;
  674. }
  675. // Create a new media modal.
  676. this.media_modal = wp.media( {
  677. frame : 'select',
  678. multiple : false,
  679. editing : true,
  680. title : this.model.get( 'l10n' ).choose,
  681. library : { type : 'image' },
  682. button : { text: this.model.get( 'l10n' ).set }
  683. } );
  684. // Runs when an image is selected in the media modal.
  685. this.media_modal.on( 'select', function() {
  686. // Gets the JSON data for the first selection.
  687. var media = this.media_modal.state().get( 'selection' ).first().toJSON();
  688. // Size of image to display.
  689. var size = this.model.attributes.size;
  690. // Updates the model for the view.
  691. this.model.set( {
  692. src : media.sizes[ size ] ? media.sizes[ size ]['url'] : media.url,
  693. alt : media.alt,
  694. value : media.id
  695. } );
  696. }, this );
  697. // Opens the media modal.
  698. this.media_modal.open();
  699. },
  700. // Executed when the remove media button is clicked.
  701. removemedia : function() {
  702. // Updates the model for the view.
  703. this.model.set( { src : '', alt : '', value : '' } );
  704. }
  705. } );
  706. }() );