The czr_base_fmk has already been loaded' );
}
return;
}
// Set the namsepace as a global so we can use it when fired from another theme/plugin using the fmk
global $czr_base_fmk_namespace;
$czr_base_fmk_namespace = __NAMESPACE__ . '\\';
do_action( 'nimble_base_fmk_loaded' );
////////////////////////////////////////////////////////////////
// CZR_Fmk_Base
if ( ! class_exists( 'CZR_Fmk_Base_Construct' ) ) :
class CZR_Fmk_Base_Construct {
static $instance;
public $registered_modules = array();//stores the collection of dynamic modules registration candidates
public $registered_settings = array();//stores the collection of dynamic settings registration candidates
public $default_dynamic_setting_params = array();
public $default_dynamic_module_params = array();
public $czr_css_attr = array();
public $current_module_params_when_ajaxing;// store the params when ajaxing and allows us to access the currently requested module params at any point of the ajax action
public static function czr_fmk_get_instance( $params ) {
if ( ! isset( self::$instance ) && ! ( self::$instance instanceof CZR_Fmk_Base ) ) {
self::$instance = new CZR_Fmk_Base( $params );
}
return self::$instance;
}
//@param $params = array(
// 'base_url' => '' <= path to root class folder
//)
function __construct( $params = array() ) {
if ( ! is_array( $params ) || empty( $params ) ) {
error_log( 'CZR_Fmk_Base => constructor => missing params');
return;
}
if ( empty( $params['base_url'] ) ) {
error_log( 'CZR_Fmk_Base => constructor => wrong params');
return;
}
// DEFINITIONS
if ( ! defined( 'NIMBLE_FMK_BASE_URL' ) ) { define( 'NIMBLE_FMK_BASE_URL' , $params['base_url'] ); }
if ( ! defined( 'NIMBLE_FMK_BASE_VERSION' ) ) { define( 'NIMBLE_FMK_BASE_VERSION' , isset( $params['version'] ) ? $params['version'] : '1.0.0' ); }
// Cache the css attr used in the tmpl builder and in the localized params
$this -> czr_css_attr = $this -> czr_fmk_get_customizer_controls_css_attr();
// Cache the default dynamic params
$this -> default_dynamic_setting_params = $this -> czr_fmk_get_default_dynamic_setting_params();
$this -> default_dynamic_module_params = $this -> czr_fmk_get_default_dynamic_module_params();
// Enqueue the fmk control js + a module tmpl
$this -> czr_enqueue_fmk_resources();
// ajax filters + template generator
$this -> czr_setup_ajax_tmpl();
// Dynamic Module Registration
$this -> czr_setup_dynamic_settings_registration();
$this -> czr_setup_dynamic_modules_registration();
// Content picker
$this -> czr_setup_content_picker_ajax_actions();
}//__construct
// fired in the constructor to cache the params in a property
private function czr_fmk_get_default_dynamic_setting_params() {
return array(
'setting_id' => '',
'dynamic_registration' => true,
'module_type' => '',
'option_value' => array(),
'setting' => array(
'type' => 'option',
'default' => array(),
'transport' => 'refresh',
'setting_class' => '',//array( 'path' => '', 'name' => '' )
'sanitize_callback' => '',
'validate_callback' => '',
),
'section' => array(
'id' => '',
'title' => '',
'panel' => '',
'priority' => 10
),
'control' => array(
'label' => '',
'type' => 'czr_module',
'priority' => 10,
'control_class' => ''//array( 'path' => '', 'name' => '' )
)
);
}
// fired in the constructor to cache the params in a property
private function czr_fmk_get_default_dynamic_module_params() {
return array(
'dynamic_registration' => true,
'module_type' => '',
'sanitize_callback' => '', //<= used when dynamically registering a setting
'validate_callback' => '', //<= used when dynamically registering a setting
'customizer_assets' => array(
'control_js' => array(),
'localized_control_js' => array()
),
'tmpl' => array()
);
}
// Copy of czr_fn_get_controls_css_attr() and the equivalent in Hueman Pro
public function czr_fmk_get_customizer_controls_css_attr() {
return apply_filters('czr_fmk_controls_css_attr',
array(
'multi_input_wrapper' => 'czr-multi-input-wrapper',
'sub_set_wrapper' => 'czr-sub-set',
'sub_set_input' => 'czr-input',
'img_upload_container' => 'czr-imgup-container',
'edit_modopt_icon' => 'czr-toggle-modopt',
'close_modopt_icon' => 'czr-close-modopt',
'mod_opt_wrapper' => 'czr-mod-opt-wrapper',
'items_wrapper' => 'czr-items-wrapper',
'single_item' => 'czr-single-item',
'item_content' => 'czr-item-content',
'item_header' => 'czr-item-header',
'item_title' => 'czr-item-title',
'item_btns' => 'czr-item-btns',
'item_sort_handle' => 'czr-item-sort-handle',
//remove dialog
'display_alert_btn' => 'czr-display-alert',
'remove_alert_wrapper' => 'czr-remove-alert-wrapper',
'cancel_alert_btn' => 'czr-cancel-button',
'remove_view_btn' => 'czr-remove-button',
'edit_view_btn' => 'czr-edit-view',
//pre add dialog
'open_pre_add_btn' => 'czr-open-pre-add-new',
'adding_new' => 'czr-adding-new',
'pre_add_wrapper' => 'czr-pre-add-wrapper',
'pre_add_item_content' => 'czr-pre-add-view-content',
'cancel_pre_add_btn' => 'czr-cancel-add-new',
'add_new_btn' => 'czr-add-new',
'pre_add_success' => 'czr-add-success'
)
);
}
}//class
endif;
?> $this -> czr_css_attr,
'isDevMode' => ( defined('WP_DEBUG') && true === WP_DEBUG ) || ( defined('CZR_DEV') && true === CZR_DEV ),
'docURL' => esc_url('docs.presscustomizr.com/'),
'i18n' => array(
'edit' => __('Edit', 'customizr'),
'close' => __('Close', 'customizr'),
'notset' => __('Not set', 'customizr'),
'successMessage' => __('Done !', 'customizr'),
'readDocumentation' => __('Learn more about this in the documentation', 'customizr'),
'Settings' => __('Settings', 'customizr'),
'Options for' => __('Options for', 'customizr'),
// img upload translation
'select_image' => __( 'Select Image', 'customizr' ),
'change_image' => __( 'Change Image', 'customizr' ),
'remove_image' => __( 'Remove', 'customizr' ),
'default_image' => __( 'Default', 'customizr' ),
'placeholder_image' => __( 'No image selected', 'customizr' ),
'frame_title_image' => __( 'Select Image', 'customizr' ),
'frame_button_image' => __( 'Choose Image', 'customizr' ),
'Customizing' => __('Customizing', 'customizr'),
),
'paramsForDynamicRegistration' => apply_filters( 'czr_fmk_dynamic_setting_js_params', array() )
)
)
);
}
// Enqueue the fmk css when standalone plugin
// hook : 'customize_controls_enqueue_scripts'
function ac_load_additional_controls_css() {
wp_enqueue_style(
'czr-fmk-controls-style',
sprintf('%1$s/assets/css/czr-ccat-control-base%2$s.css', NIMBLE_FMK_BASE_URL, ( defined('WP_DEBUG') && true === WP_DEBUG ) ? '' : '.min'),
array( 'customize-controls' ),
( defined('WP_DEBUG') && true === WP_DEBUG ) ? time() : NIMBLE_FMK_BASE_VERSION,
$media = 'all'
);
//select2 stylesheet
//overriden by some specific style in czr-control-base.css
wp_enqueue_style(
'czr-select2-css',
sprintf('%1$s/assets/css/lib/czrSelect2.min.css', NIMBLE_FMK_BASE_URL, ( defined('WP_DEBUG') && true === WP_DEBUG ) ? '' : '.min'),
array( 'customize-controls' ),
( defined('WP_DEBUG') && true === WP_DEBUG ) ? time() : NIMBLE_FMK_BASE_VERSION,
$media = 'all'
);
wp_enqueue_style(
'czr-font-awesome',
sprintf('%1$s/assets/fonts/css/fontawesome-all.min.css', NIMBLE_FMK_BASE_URL, ( defined('WP_DEBUG') && true === WP_DEBUG ) ? '' : '.min'),
array(),
( defined('WP_DEBUG') && true === WP_DEBUG ) ? time() : NIMBLE_FMK_BASE_VERSION,
$media = 'all'
);
}
//hook : customize_preview_init
function ac_customize_load_preview_js() {
global $wp_version;
wp_enqueue_script(
'czr-customizer-preview' ,
sprintf(
'%1$s/assets/js/%2$s',
NIMBLE_FMK_BASE_URL,
defined('CZR_DEV') && true === CZR_DEV ? 'czr-preview-base.js' : 'czr-preview-base.min.js'
),
array( 'customize-preview', 'underscore'),
( defined('WP_DEBUG') && true === WP_DEBUG ) ? time() : NIMBLE_FMK_BASE_VERSION,
true
);
//localizes
wp_localize_script(
'czr-customizer-preview',
'serverPreviewParams',
apply_filters('czr_base_fmk_customizer_preview_params' ,
array(
'themeFolder' => get_template_directory_uri(),
//patch for old wp versions which don't trigger preview-ready signal => since WP 4.1
'preview_ready_event_exists' => version_compare( $wp_version, '4.1' , '>=' ),
'blogname' => get_bloginfo('name'),
'isRTL' => is_rtl()
)
)
);
}
// DO WE STILL NEED TO PRINT THIS TMPL ?
/////////////////////////////////////////////////////
/// WHEN EMBEDDED IN A CONTROL //////////////////////
/////////////////////////////////////////////////////
//add specific js templates for the czr_module control
//this is usually called in the manager for "registered" controls that need to be rendered with js
//for this control, we'll do it another way because we need several js templates
//=> that's why this control has not been "registered" and js templates are printed with the following action
function ac_print_module_control_templates() {
//Render the control wrapper for the CRUD types modules
?>
unauthenticated' );
}
if ( ! current_user_can( 'edit_theme_options' ) ) {
wp_send_json_error('ac_set_ajax_czr_tmpl => user_cant_edit_theme_options');
}
if ( ! current_user_can( 'customize' ) ) {
status_header( 403 );
wp_send_json_error( 'ac_set_ajax_czr_tmpl => customize_not_allowed' );
} else if ( ! isset( $_SERVER['REQUEST_METHOD'] ) || 'POST' !== $_SERVER['REQUEST_METHOD'] ) {
status_header( 405 );
wp_send_json_error( 'ac_set_ajax_czr_tmpl => bad_method' );
}
$action = 'save-customize_' . get_stylesheet();
if ( ! check_ajax_referer( $action, 'nonce', false ) ) {
wp_send_json_error( array(
'code' => 'invalid_nonce',
'message' => __( 'ac_set_ajax_czr_tmpl => Security check failed.', 'customizr' ),
) );
}
if ( ! isset( $_POST['module_type'] ) || empty( $_POST['module_type'] ) ) {
wp_send_json_error( 'ac_set_ajax_czr_tmpl => missing module_type property in posted data' );
}
if ( ! isset( $_POST['tmpl'] ) || empty( $_POST['tmpl'] ) ) {
wp_send_json_error( 'ac_set_ajax_czr_tmpl => missing tmpl property in posted data' );
}
$tmpl = $_POST['tmpl'];
$module_type = $_POST['module_type'];
///////////////////////////////////////////////////////////////////////
// @param $tmpl = 'item-inputs'
//
// @param $_POST = {
// [tmpl] => item-inputs
// [module_type] => czr_heading_child
// [module_id] => __nimble__51b2f35191b3__main_settings_czr_module
// [cache] => true
// [nonce] => b4b0aea848
// [control_id] => __nimble__51b2f35191b3__main_settings
// [item_model] => Array
// (
// [id] => czr_heading_child_0
// [title] =>
// [heading_text] => This is a heading.
// [heading_tag] => h1
// [h_alignment_css] => Array
// (
// [desktop] => center
// )
// [heading_title] =>
// [link-to] =>
// [link-custom-url] =>
// [link-target] =>
// )
// [action] => ac_get_template
// }
$html = apply_filters( "ac_set_ajax_czr_tmpl___{$module_type}", '', $tmpl, $_POST );
///////////////////////////////////////////////////////////////////////////
if ( empty( $html ) ) {
wp_send_json_error( 'ac_set_ajax_czr_tmpl => module ' . $module_type . ' => template empty for requested tmpl : ' . $tmpl );
} else {
wp_send_json_success( apply_filters( 'tmpl_results', $html, $tmpl ) );
}
}
// hook : ac_set_ajax_czr_tmpl___all_modules
// this dynamic filter is declared on wp_ajax_ac_get_template
// It allows us to populate the server response with the relevant module html template
// $html = apply_filters( "ac_set_ajax_czr_tmpl___{$module_type}", '', $tmpl );
//
// For all modules, there are 3 types of templates :
// 1) the pre-item, rendered when adding an item
// 2) the module meta options, or mod-opt
// 3) the item input options
function ac_get_all_modules_tmpl( $html, $requested_tmpl = '', $posted_params = array() ) {
$css_attr = $this -> czr_css_attr;
if ( empty( $requested_tmpl ) ) {
wp_send_json_error( 'ac_get_all_modules_tmpl => the requested tmpl is empty' );
}
ob_start();
switch ( $requested_tmpl ) {
case 'crud-module-part' :
?>
{{ data.title }} ?', 'customizr'); ?>
<# if ( ( data.attachment && data.attachment.id ) ) { #>
<# } else if ( ! _.isEmpty( data.fromUrl ) ) { #>
placeholder ?>
<# } else { #>
<# } #>
no template was found for tmpl => ' . $requested_tmpl );
}
return $html;//will be sent by wp_send_json_success() in ::ac_set_ajax_czr_tmpl()
}
}//class
endif;
?> 'text',
'title' => '',
'default' => '',
'html_before' => '',
'notice_before_title' => '',
'notice_before' => '',
'notice_after' => '',
'placeholder' => '',
'html_after' => '',
// typically used for the number and range inputs
'step' => '',
'min' => '',
'max' => '',
'orientation' => '',//vertical / horizontal
'unit' => '',//% or px for example
'transport' => '',//<= can be set as a data property of the input wrapper, and used when instanciating the input
'input_template' => '',//<= a static html template can be provided to render the input, in this case it will be used in priority
'tmpl_callback' => '',//<= a callback function to be used to print the entire input template, including the wrapper
'width-100' => false,//<= to force a width of 100%
'title_width' => '',//width-80
'input_width' => '',//width-20
'code_type' => '',//<= used for to specify the language type of the codemirror editor (if not specified full a html editor will be instantiated)
'refresh_markup' => null,
'refresh_stylesheet' => null,
'refresh_fonts' => null,
'refresh_preview' => null,
'sanitize_cb' => '',
'validate_cb' => '',
'css_selectors' => array(), //<= used to specify css selectors on which we will apply the dynamically generated css for a given input id @see \Nimble\sek_add_css_rules_for_generic_css_input_types'
'css_identifier' => '',//<= the identifier allowing us to map a css generation rule. @see \Nimble\sek_add_css_rules_for_css_sniffed_input_id
'important_input_list' => array(),//<= the list of input_id that an important input can flag !important @see \Nimble\sek_add_css_rules_for_css_sniffed_input_id
'choices' => array(), // <= used to declare the option list of a select input
'has_device_switcher' => false, // <= indicates if the input value shall be saved by device or not
'scope' => 'local',// <= used when resetting the sections
// introduced for https://github.com/presscustomizr/nimble-builder/issues/403
'editor_params' => array(),
// introduced for https://github.com/presscustomizr/nimble-builder/issues/431
'section_collection' => array()
);
foreach( $tmpl_map as $input_id => $input_data ) {
if ( ! is_string( $input_id ) || empty( $input_id ) ) {
wp_send_json_error( __FUNCTION__ . ' => wrong input id' );
break;
}
if ( ! is_array( $input_data ) ) {
wp_send_json_error( __FUNCTION__ . ' => wrong var type for the input_data of input id : ' . $input_id );
break;
}
// check that we have no unknown entries in the provided input_data
$maybe_diff = array_diff_key( $input_data, $default_input_entries );
if ( ! empty( $maybe_diff ) ) {
error_log('<' . __FUNCTION__ . '>');
error_log( '=> at least one unknown param in the registered input params for input id : ' . $input_id );
error_log( print_r( $maybe_diff, true ) );
error_log('' . __FUNCTION__ . '>');
//wp_send_json_error( 'ac_generate_czr_tmpl_from_map => at least one unknow entry in the input data for input id : ' . $input_id );
break;
}
// we're clear, let's go
$input_data = wp_parse_args( $input_data, $default_input_entries );
// Do we have a specific template provided ?
if ( ! empty( $input_data[ 'tmpl_callback' ] ) && function_exists( $input_data[ 'tmpl_callback' ] ) ) {
$html .= call_user_func_array( $input_data[ 'tmpl_callback' ], array( $input_data ) );
} else {
$html .= $this -> ac_get_default_input_tmpl( $input_id, $input_data );
}
}
return $html;////will be sent by wp_send_json_success() in ::ac_set_ajax_czr_tmpl()
}
// Fired in ac_generate_czr_tmpl_from_map
function ac_get_default_input_tmpl( $input_id, $input_data ) {
if ( ! array_key_exists( 'input_type', $input_data ) || empty( $input_data[ 'input_type' ] ) ) {
wp_send_json_error( 'ac_get_input_tmpl => missing input type for input id : ' . $input_id );
}
$input_type = $input_data[ 'input_type' ];
// some inputs have a width of 100% even if not specified in the input_data
$is_width_100 = true === $input_data[ 'width-100' ];
if ( in_array( $input_type, array( 'color', 'radio', 'textarea' ) ) ) {
$is_width_100 = true;
}
$css_attr = $this -> czr_css_attr;
ob_start();
//
printf( '',
$css_attr['sub_set_wrapper'],
$is_width_100 ? 'width-100' : '',
'hidden' === $input_type ? 'hidden' : '',
$input_type,
! empty( $input_data['transport'] ) ? 'data-transport="'. $input_data['transport'] .'"' : ''
);
?>
%2$s
', ! empty( $input_data['title_width'] ) ? $input_data['title_width'] : '', $input_data['title'] );
}
?>
', ! empty( $input_data['input_width'] ) ? $input_data['input_width'] : '' ); ?>
ac_set_input_tmpl_content( $input_type, $input_id, $input_data );
}
?>
$tmpl_html = apply_filters( "czr_set_input_tmpl___{$input_type}", ob_get_clean(), $input_id, $input_data );
//error_log( print_r($tmpl_html, true ) );
if ( empty( $tmpl_html ) ) {
wp_send_json_error( 'ac_get_input_tmpl => no html returned for input ' . $input_id );
}
return $tmpl_html;
}//ac_get_input_tmpl()
// fired in ::ac_get_default_input_tmpl();
private function ac_set_input_tmpl_content( $input_type, $input_id, $input_data ) {
$css_attr = $this -> czr_css_attr;
$input_tmpl_content = null;
// First fires a hook to allow the input content to be remotely set
// For example the module_picker, the spacing, h_text_alignment... are printed this way
ob_start();
do_action( 'czr_set_input_tmpl_content', $input_type, $input_id, $input_data );
$input_tmpl_content = ob_get_clean();
if ( ! empty( $input_tmpl_content ) ) {
echo $input_tmpl_content;
} else {
// Then, if we have no content yet, let's go thought the default input cases
switch ( $input_type ) {
/* ------------------------------------------------------------------------- *
* HIDDEN
/* ------------------------------------------------------------------------- */
case 'hidden':
?>
',
! empty( $input_data['step'] ) ? 'step="'. $input_data['step'] .'"' : '',
! empty( $input_data['min'] ) ? 'min="'. $input_data['min'] .'"' : '',
! empty( $input_data['max'] ) ? 'max="'. $input_data['max'] .'"' : '',
$input_id
);
?>
<#
var _checked = ( false != data[''] ) ? "checked=checked" : '';
#>
<#
var _checked = ( false != data[''] ) ? "checked=checked" : '';
#>
<#
var _checked = ( false != data[''] ) ? "checked=checked" : '';
#>
We need a unique ID here so that input and label are tied by a unique link
// @see https://www.w3.org/TR/html401/interact/forms.html#h-17.9.1
// @see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/Input/checkbox
$unique_id = sprintf('%04X%04X', mt_rand(0, 65535), mt_rand(0, 65535));
?>
Switch
data range_slide => ', data ); #> ?>
',
! empty( $input_data['orientation'] ) ? 'data-orientation="'. $input_data['orientation'] .'"' : '',
! empty( $input_data['unit'] ) ? 'data-unit="'. $input_data['unit'] .'"' : '',
! empty( $input_data['min'] ) ? 'min="'. $input_data['min'] .'"' : '',
! empty( $input_data['max'] ) ? 'max="'. $input_data['max'] .'"' : '',
$input_id
);
?>
', $input_id );
?>
' . __CLASS__ . '::' . __FUNCTION__ . ' this input type has no template : ' . $input_type );
break;
}//switch ( $input_type ) {
}//if ( empty( $input_tmpl_content ) )()
}//function()
}//class
endif;
?> '',
// 'dynamic_registration' => true,
// 'module_type' => '',
// 'option_value' => array(),
// 'setting' => array(
// 'type' => 'option',
// 'default' => array(),
// 'transport' => 'refresh',
// 'setting_class' => '',//array( 'path' => '', 'name' => '' )
// 'sanitize_callback' => '',
// 'validate_callback' => '',
// ),
// 'section' => array(
// 'id' => '',
// 'title' => '',
// 'panel' => '',
// 'priority' => 10
// ),
// 'control' => array(
// 'label' => '',
// 'type' => 'czr_module',
// 'priority' => 10,
// 'control_class' => ''//array( 'path' => '', 'name' => '' )
// )
// )
function czr_pre_register_dynamic_setting( $setting_params ) {
if ( ! is_array( $setting_params ) || empty( $setting_params ) ) {
error_log( 'czr_pre_register_dynamic_setting => empty $setting_params submitted' );
return;
}
if ( ! array_key_exists( 'setting_id', $setting_params ) || empty( $setting_params['setting_id'] ) ) {
error_log( 'czr_pre_register_dynamic_setting => missing setting id' );
return;
}
// normalize
$setting_params = wp_parse_args( $setting_params, $this -> default_dynamic_setting_params );
$registered = $this->registered_settings;
$setting_id_candidate = $setting_params['setting_id'];
// A setting id can be registered only once.
// Already registered ?
if ( array_key_exists( $setting_id_candidate, $registered ) ) {
error_log( 'czr_pre_register_dynamic_setting => setting id already registered => ' . $setting_id_candidate );
return;
}
$registered[ $setting_id_candidate ] = $setting_params;
$this->registered_settings = $registered;
}
////////////////////////////////////////////////////////////////
// FILTER DYNAMIC SETTING AND CLASS ARGS
// hook : customize_register
// Those filters are declared by WP core in class-wp-customize-manager.php
// in => add_action( 'customize_register', array( $this, 'register_dynamic_settings' ), 11 );
function czr_setup_dynamic_setting_registration( $wp_customize ) {
add_filter( 'customize_dynamic_setting_args', array( $this, 'czr_setup_customizer_dynamic_setting_args' ), 10, 2 );
add_filter( 'customize_dynamic_setting_class', array( $this, 'czr_setup_customizer_dynamic_setting_class' ), 10, 3 );
}
// hook : 'customize_dynamic_setting_args'
function czr_setup_customizer_dynamic_setting_args( $setting_args, $setting_id ) {
//sek_error_log( __CLASS__ . '::' . __FUNCTION__ , $this->registered_settings );
if ( ! is_array( $this->registered_settings ) || empty( $this->registered_settings ) )
return $setting_args;
// let's initialize the args to the provided param
$registered_setting_args = $setting_args;
// loop on each registered modules
foreach ( $this->registered_settings as $registerered_setting_id => $params ) {
if ( array_key_exists('dynamic_registration', $params) && true !== $params['dynamic_registration'] ) {
continue;
}
$params = wp_parse_args( $params, $this -> default_dynamic_setting_params );
if ( $registerered_setting_id != $setting_id || empty( $registerered_setting_id ) )
continue;
$setting_args = is_array( $params['setting'] ) ? $params['setting'] : array();
$setting_args = wp_parse_args( $setting_args, array(
'type' => 'option',
'default' => array(),
'transport' => 'refresh',
'sanitize_callback' => '',
'validate_callback' => ''
) );
// Provide new setting args
$registered_setting_args = array(
'type' => empty( $setting_args[ 'type' ] ) ? 'option' : $setting_args[ 'type' ],
'default' => array(),
'transport' => $setting_args[ 'transport' ],
'sanitize_callback' => ( ! empty( $setting_args[ 'sanitize_callback' ] ) && function_exists( $setting_args[ 'sanitize_callback' ] ) ) ? $setting_args[ 'sanitize_callback' ] : '',
'validate_callback' => ( ! empty( $setting_args[ 'validate_callback' ] ) && function_exists( $setting_args[ 'validate_callback' ] ) ) ? $setting_args[ 'validate_callback' ] : ''
);
// if this is a module setting, it can have specific sanitize and validate callback set for the module
// Let's check if the module_type is registered, and if there are any callback set.
// If a match is found, we'll use those callback
$module_params = $this -> czr_get_registered_dynamic_module( $params[ 'module_type' ] );
if ( false !== $module_params && is_array( $module_params ) ) {
if ( array_key_exists( 'validate_callback', $module_params ) && function_exists( $module_params[ 'validate_callback' ] ) ) {
$registered_setting_args[ 'validate_callback' ] = $module_params[ 'validate_callback' ];
}
if ( array_key_exists( 'sanitize_callback', $module_params ) && function_exists( $module_params[ 'sanitize_callback' ] ) ) {
$registered_setting_args[ 'sanitize_callback' ] = $module_params[ 'sanitize_callback' ];
}
}
//error_log( 'REGISTERING DYNAMICALLY for setting =>'. $setting_id );
}
return $registered_setting_args;
}
// hook : 'customize_dynamic_setting_class'
function czr_setup_customizer_dynamic_setting_class( $class, $setting_id, $args ) {
if ( ! is_array( $this->registered_settings ) || empty( $this->registered_settings ) )
return $class;
// let's initialize the args to the provided class
$registered_setting_class = $class;//'WP_Customize_Setting' by default
// loop on each registered modules
foreach ( $this->registered_settings as $registerered_setting_id => $params ) {
$params = wp_parse_args( $params, $this -> default_dynamic_setting_params );
if ( true !== $params['dynamic_registration'] ) {
continue;
}
if ( $registerered_setting_id != $setting_id || empty( $registerered_setting_id ) )
continue;
$setting_args = $params['setting'];
if ( is_array( $setting_args ) && array_key_exists( 'setting_class', $setting_args ) ) {
// provide new setting class if exists and not yet loaded
if ( is_array( $setting_args[ 'setting_class' ] ) && array_key_exists( 'name', $setting_args[ 'setting_class' ] ) && array_key_exists( 'path', $setting_args[ 'setting_class' ] ) ) {
if ( ! class_exists( $setting_args[ 'setting_class' ][ 'name' ] ) && file_exists( $setting_args[ 'setting_class' ]['path'] ) ) {
require_once( $setting_args[ 'setting_class' ]['path'] );
}
if ( class_exists( $setting_args[ 'setting_class' ][ 'name' ] ) ) {
$registered_setting_class = $setting_args[ 'setting_class' ][ 'name' ];
}
}
}
//error_log( 'REGISTERING CLASS DYNAMICALLY for setting =>'. $setting_id );
}
return $registered_setting_class;
}
////////////////////////////////////////////////////////////////
// Print js params for dynamic control and section registration
// hook : czr_fmk_dynamic_setting_js_params'
// @param js_param = array()
function czr_setup_localized_params_for_dynamic_js_registration( $js_params ) {
// error_log( '' );
// error_log( print_r( $this->registered_settings, true ) );
// error_log( ' ' );
if ( ! is_array( $this->registered_settings ) || empty( $this->registered_settings ) )
return $js_params;
$js_params = ! is_array( $js_params ) ? array() : $js_params;
// 'localized_control_js' => array(
// 'deps' => 'czr-customizer-fmk',
// 'global_var_name' => 'socialLocalized',
// 'params' => array(
// //Social Module
// 'defaultSocialColor' => 'rgb(90,90,90)',
// 'defaultSocialSize' => 14,
// //option value for dynamic registration
//
// )
// 'dynamic_setting_registration' => array(
// 'values' => $args['option_value'],
// 'section' => $args['section']
// )
// )
// loop on each registered modules
foreach ( $this->registered_settings as $registerered_setting_id => $params ) {
$params = wp_parse_args( $params, $this -> default_dynamic_setting_params );
// We need the 'option_value' entry, even if empty
if ( ! array_key_exists( 'option_value', $params ) || ! is_array( $params['option_value'] ) )
continue;
// Check if not already setup
if ( array_key_exists( $registerered_setting_id, $params ) ) {
error_log( 'czr_setup_localized_params_for_dynamic_js_registration => js_params already setup for setting : ' . $registerered_setting_id );
}
$js_params[ $registerered_setting_id ] = array(
'setting_id' => $registerered_setting_id,
'module_type' => $params[ 'module_type' ],
'module_registration_params' => $this -> czr_get_registered_dynamic_module( $params[ 'module_type' ] ),
'option_value' => $params['option_value'],
// 'setting' => array(
// 'type' => 'option',
// 'default' => array(),
// 'transport' => 'refresh',
// 'setting_class' => '',//array( 'path' => '', 'name' => '' )
// 'sanitize_callback' => '',
// 'validate_callback' => '',
// ),
'setting' => array_key_exists( 'setting', $params ) ? $params[ 'setting' ] : array(),
// 'section' => array(
// 'id' => '',
// 'title' => '',
// 'panel' => '',
// 'priority' => 10
// ),
'section' => array_key_exists( 'section', $params ) ? $params[ 'section' ] : array(),
// 'control' => array(
// 'label' => '',
// 'type' => 'czr_module',
// 'priority' => 10,
// 'control_class' => ''//array( 'path' => '', 'name' => '' )
// ),
'control' => array_key_exists( 'control', $params ) ? $params[ 'control' ] : array(),
);
}
return $js_params;
}
////////////////////////////////////////////////////////////////
// TO DEPRECATE ?
// REGISTER IF NOT DYNAMIC
// hook : customize_register
function czr_register_not_dynamic_settings( $wp_customize ) {
// error_log('');
// error_log(print_r( $this->registered_settings, true ));
// error_log(' ');
if ( ! is_array( $this->registered_settings ) || empty( $this->registered_settings ) )
return;
// loop on each registered modules
foreach ( $this->registered_settings as $setting_id => $params ) {
$params = wp_parse_args( $params, $this -> default_dynamic_setting_params );
if ( true === $params['dynamic_registration'] )
continue;
// SETTING
$setting_args = $params['setting'];
$registered_setting_class = 'WP_Customize_Setting';
if ( is_array( $setting_args ) && array_key_exists( 'setting_class', $setting_args ) ) {
// provide new setting class if exists and not yet loaded
if ( is_array( $setting_args[ 'setting_class' ] ) && array_key_exists( 'name', $setting_args[ 'setting_class' ] ) && array_key_exists( 'path', $setting_args[ 'setting_class' ] ) ) {
if ( ! class_exists( $setting_args[ 'setting_class' ][ 'name' ] ) && file_exists( $setting_args[ 'setting_class' ]['path'] ) ) {
require_once( $setting_args[ 'setting_class' ]['path'] );
}
if ( class_exists( $setting_args[ 'setting_class' ][ 'name' ] ) ) {
$registered_setting_class = $setting_args[ 'setting_class' ][ 'name' ];
}
}
}
$wp_customize->add_setting( new $registered_setting_class( $wp_customize, $setting_id, array(
'default' => $setting_args[ 'default' ],
'type' => $setting_args[ 'type' ],
'sanitize_callback' => isset( $settings_args[ 'sanitize_callback' ] ) ? $settings_args[ 'sanitize_callback' ] : ''
) ) );
// CONTROL
$control_args = $params['control'];
$registered_control_class = 'WP_Customize_Control';
if ( is_array( $control_args ) && array_key_exists( 'control_class', $control_args ) ) {
// provide new setting class if exists and not yet loaded
if ( is_array( $control_args[ 'control_class' ] ) && array_key_exists( 'name', $control_args[ 'control_class' ] ) && array_key_exists( 'path', $control_args[ 'control_class' ] ) ) {
if ( ! class_exists( $control_args[ 'control_class' ][ 'name' ] ) && file_exists( $control_args[ 'control_class' ]['path'] ) ) {
require_once( $control_args[ 'control_class' ]['path'] );
}
if ( class_exists( $control_args[ 'control_class' ][ 'name' ] ) ) {
$registered_control_class = $control_args[ 'control_class' ][ 'name' ];
}
}
}
$wp_customize -> add_control( new $registered_control_class( $wp_customize, $setting_id, array(
'type' => $control_args[ 'type' ],
'label' => $control_args[ 'label' ],
'section' => $params[ 'section' ]['id'],
'module_type' => $params[ 'module_type' ]
) ) );
}//foreach
}//czr_register_not_dynamic_settings
}//class
endif;
?> '',
// 'customizer_assets' => array(
// 'control_js' => array(),
// 'localized_control_js' => array()
// ),
// 'tmpl' => array()
// )
function czr_pre_register_dynamic_module( $module_params ) {
// error_log( '' );
// error_log( print_r( $module_params, true ) );
// error_log( ' ' );
if ( ! is_array( $module_params ) || empty( $module_params ) ) {
error_log( 'czr_pre_register_dynamic_module => empty $module_params submitted' );
return;
}
if ( ! array_key_exists( 'module_type', $module_params ) || empty( $module_params['module_type'] ) ) {
error_log( 'czr_pre_register_dynamic_module => missing module_type' );
return;
}
// normalize
$module_params = wp_parse_args( $module_params, $this -> default_dynamic_module_params );
$registered = $this->registered_modules;
$module_type_candidate = $module_params['module_type'];
// A module type can be registered only once.
// Already registered ?
if ( array_key_exists( $module_type_candidate, $registered ) ) {
error_log( 'czr_pre_register_dynamic_module => module type already registered => ' . $module_type_candidate );
return;
}
$registered[ $module_type_candidate ] = $module_params;
$this->registered_modules = $registered;
}
// HELPER
// @return boolean or array of module params
function czr_get_registered_dynamic_module( $module_type = '' ) {
$registered = $this->registered_modules;
if ( empty( $module_type ) || ! is_array( $registered ) || empty( $registered ) )
return;
return array_key_exists( $module_type , $registered ) ? $registered[ $module_type ] : false;
}
// @return bool
function czr_is_module_registered( $module_type = '' ) {
$registered = $this->registered_modules;
if ( empty( $module_type ) || ! is_array( $registered ) || empty( $registered ) )
return;
return array_key_exists( $module_type , $registered );
}
////////////////////////////////////////////////////////////////
// ENQUEUE ASSETS
// hook : customize_controls_enqueue_scripts
//
// 'customizer_assets' => array(
// 'control_js' => array(
// // handle + params for wp_enqueue_script()
// // @see https://developer.wordpress.org/reference/functions/wp_enqueue_script/
// 'czr-social-links-module' => array(
// 'src' => sprintf(
// '%1$s/assets/js/%2$s',
// $args['base_url_path'],
// '_2_7_socials_module.js'
// ),
// 'deps' => array('customize-controls' , 'jquery', 'underscore'),
// 'ver' => ( defined('WP_DEBUG') && true === WP_DEBUG ) ? time() : $args['version'],
// 'in_footer' => true
// )
// ),
// 'localized_control_js' => array(
// 'deps' => 'czr-customizer-fmk',
// 'global_var_name' => 'socialLocalized',
// 'params' => array(
// //Social Module
// 'defaultSocialColor' => 'rgb(90,90,90)',
// 'defaultSocialSize' => 14,
// //option value for dynamic registration
// )
// )
// ),
function czr_register_dynamic_modules_customizer_control_assets() {
if ( ! is_array( $this->registered_modules ) || empty( $this->registered_modules ) )
return;
$wp_scripts = wp_scripts();
// loop on each registered modules
foreach ( $this->registered_modules as $module_type => $params ) {
$params = wp_parse_args( $params, $this -> default_dynamic_module_params );
//error_log( print_r( $params, true ) );
$control_js_params = $params[ 'customizer_assets' ][ 'control_js' ];
// Enqueue the list of registered scripts
if ( ! empty( $control_js_params ) ) {
foreach ( $control_js_params as $handle => $script_args ) {
if ( ! isset( $wp_scripts->registered[$handle] ) ) {
wp_enqueue_script(
$handle,
array_key_exists( 'src', $script_args ) ? $script_args['src'] : null,
array_key_exists( 'deps', $script_args ) ? $script_args['deps'] : null,
array_key_exists( 'ver', $script_args ) ? $script_args['ver'] : null,
array_key_exists( 'in_footer', $script_args ) ? $script_args['in_footer'] : false
);
} else {
error_log( __CLASS__ . '::' . __FUNCTION__ . " => handle already registered : " . $handle . " , this asset won't be enqueued => " . $script_args['src'] );
}
}
}
// 'localized_control_js' => array(
// 'deps' => 'czr-customizer-fmk',
// 'global_var_name' => 'socialLocalized',
// 'params' => array(
// //Social Module
// 'defaultSocialColor' => 'rgb(90,90,90)',
// 'defaultSocialSize' => 14,
// //option value for dynamic registration
// )
// )
// Print localized params if any
if ( array_key_exists( 'localized_control_js', $params[ 'customizer_assets' ] ) ) {
$localized_control_js_params = is_array( $params[ 'customizer_assets' ][ 'localized_control_js' ] ) ? $params[ 'customizer_assets' ][ 'localized_control_js' ] : array();
if ( is_array( $localized_control_js_params ) && ! empty( $localized_control_js_params ) ) {
wp_localize_script(
array_key_exists( 'deps', $localized_control_js_params ) ? $localized_control_js_params['deps'] : '',
array_key_exists( 'global_var_name', $localized_control_js_params ) ? $localized_control_js_params['global_var_name'] : '',
array_key_exists( 'params', $localized_control_js_params ) ? $localized_control_js_params['params'] : array()
);
}
}
}//foreach
}
////////////////////////////////////////////////////////////////
// AJAX TEMPLATE FILTERS
// hook : init
function czr_schedule_ajax_tmpl() {
if ( ! is_array( $this->registered_modules ) || empty( $this->registered_modules ) )
return;
foreach ( $this->registered_modules as $module_type => $params ) {
$params = wp_parse_args( $params, $this -> default_dynamic_module_params );
if ( ! empty( $params['tmpl'] ) ) {
$module_type = $params['module_type'];
// filter declared with $html = apply_filters( "ac_set_ajax_czr_tmpl___{$module_type}", '', $tmpl, $_POST );
add_filter( "ac_set_ajax_czr_tmpl___{$module_type}", array( $this, 'ac_get_ajax_module_tmpl'), 10, 3 );
}
}//foreach
}
// AJAX TMPL FILTERS
// this dynamic filter is declared on wp_ajax_ac_get_template
// It allows us to populate the server response with the relevant module html template
// $html = apply_filters( "ac_set_ajax_czr_tmpl___{$module_type}", '', $tmpl );
//
// Each template is built from a map, each input type having its own unique piece of tmpl
//
// 3 types of templates :
// 1) the pre-item, rendered when adding an item
// 2) the module meta options, or mod-opt
// 3) the item input options
// @param $posted_params is the $_POST
// hook : ac_set_ajax_czr_tmpl___{$module_type}
function ac_get_ajax_module_tmpl( $html, $requested_tmpl = '', $posted_params = array() ) {
// error_log( '' );
// error_log( print_r( $this->registered_modules, true ) );
// error_log( ' ' );
// error_log( '' );
// error_log( print_r( $posted_params, true ) );
// error_log( ' ' );
// the module type is sent in the $posted_params
if ( ! is_array( $posted_params ) || empty( $posted_params ) ) {
wp_send_json_error( 'ac_get_ajax_module_tmpl => empty posted_params' );
}
if ( ! array_key_exists( 'module_type', $posted_params ) || empty( $posted_params['module_type'] ) ) {
wp_send_json_error( 'ac_get_ajax_module_tmpl => missing module_type' );
}
// if ( ! array_key_exists( 'control_id', $posted_params ) || empty( $posted_params['control_id'] ) ) {
// wp_send_json_error( 'ac_get_ajax_module_tmpl => missing control_id' );
// }
// find the requested module_id in the list of registered modules
$registered_modules = $this->registered_modules;
$module_type = $posted_params['module_type'];
if ( ! array_key_exists( $module_type, $registered_modules ) || empty( $registered_modules[ $module_type ] ) ) {
return;
}
$module_params = $registered_modules[ $module_type ];
// Store the params now, so we can access them when rendering the input templates
$this->current_module_params_when_ajaxing = $module_params;
$tmpl_params = $module_params[ 'tmpl' ];
// Enqueue the list of registered scripts
if ( empty( $tmpl_params ) ) {
return;
}
// the requested_tmpl can be pre-item, mod-opt or item-inputs
$tmpl_map = array_key_exists( $requested_tmpl, $tmpl_params ) ? $tmpl_params[ $requested_tmpl ] : array();
if ( empty( $tmpl_map ) ) {
return;
}
// Do we have tabs ?
// With tabs
// 'tabs' => array(
// array(
// 'title' => __('Spacing', 'text_doma'),
// 'inputs' => array(
// 'padding' => array(
// 'input_type' => 'number',
// 'title' => __('Padding', 'text_doma')
// ),
// 'margin' => array(
// 'input_type' => 'number',
// 'title' => __('Margin', 'text_doma')
// )
// )
// ),
// array( ... )
//
//
// Without tabs :
// 'padding' => array(
// 'input_type' => 'number',
// 'title' => __('Padding', 'text_doma')
// ),
// 'margin' => array(
// 'input_type' => 'number',
// 'title' => __('Margin', 'text_doma')
// )
if ( array_key_exists( 'tabs', $tmpl_map ) ) {
ob_start();
?>
$tab ) {
printf( '%3$s ',
$_key + 1,
array_key_exists('attributes', $tab) ? $tab['attributes'] : '',
$tab['title']
);
}//foreach
?>
$tab ) {
printf( '',
$_key + 1,
$this -> ac_generate_czr_tmpl_from_map( $tab['inputs'] )
);
}//foreach
?>
ac_generate_czr_tmpl_from_map( $tmpl_map );
}
}
}//class
endif;
?> sprintf( '%1$s ', __('Set a custom url', 'customizr') ),
'type' => '',
'type_label' => '',
'object' => '',
'id' => '_custom_',
'url' => ''
)
),
$items
);
} else {
return $items;
}
}
//hook : 'pre_post_link'
function dont_use_fancy_permalinks() {
return '';
}
/* ------------------------------------------------------------------------- *
* LOAD
/* ------------------------------------------------------------------------- */
/**
* Ajax handler for loading available content items.
* hook : wp_ajax_load-available-content-items-customizer
*/
public function ajax_load_available_items() {
$action = 'save-customize_' . get_stylesheet();
if ( ! check_ajax_referer( $action, 'nonce', false ) ) {
wp_send_json_error( array(
'code' => 'invalid_nonce',
'message' => __( 'ajax_load_available_items => Security check failed.', 'customizr' ),
) );
}
if ( ! current_user_can( 'edit_theme_options' ) ) {
wp_send_json_error('ajax_load_available_items => user_cant_edit_theme_options');
}
if ( ! isset( $_POST['wp_object_types'] ) || empty( $_POST['wp_object_types'] ) ) {
wp_send_json_error( 'czr_ajax_content_picker_missing_type_or_object_parameter' );
}
if ( ! isset( $_POST['page'] ) ) {
wp_send_json_error( 'czr_ajax_content_picker_missing_pagination_param' );
}
//error_log( print_r($_POST, true ) );
$wp_object_types = json_decode( wp_unslash( $_POST['wp_object_types'] ), true );
//$wp_object_types should look like :
//array(
// post : '',//<= all post types
// taxonomy : ''//<= all taxonomy types
//) OR
//array(
// post : [ 'page', 'cpt1', ...]
// taxonomy : [ 'category', 'tag', 'Custom_Tax_1', ... ]
//) OR
//array(
// post : [ 'page', 'cpt1', ...]
// taxonomy : '_none_'//<= don't load or search taxonomies
//)
if ( ! is_array( $wp_object_types ) || empty( $wp_object_types ) ) {
wp_send_json_error( 'czr_ajax_content_picker_missing_object_types' );
}
$page = empty( $_POST['page'] ) ? 0 : absint( $_POST['page'] );
//do we need that ?
// if ( $page < 1 ) {
// $page = 1;
// }
$items = array();
add_filter( 'pre_post_link', array( $this, 'dont_use_fancy_permalinks' ), 999 );
foreach ( $wp_object_types as $_type => $_obj_types ) {
if ( '_none_' == $_obj_types )
continue;
$item_candidates = $this -> load_available_items_query(
array(
'type' => $_type, //<= post or taxonomy
'object_types' => $_obj_types,//<= '' or array( type1, type2, ... )
'page' => $page
)
);
if ( is_array( $item_candidates ) ) {
$items = array_merge(
$items,
$item_candidates
);
}
}
remove_filter( 'pre_post_link', array( $this, 'dont_use_fancy_permalinks' ), 999 );
if ( is_wp_error( $items ) ) {
wp_send_json_error( $items->get_error_code() );
} else {
wp_send_json_success( array(
'items' => apply_filters( 'content_picker_ajax_items', $items, $page, 'ajax_load_available_items' )
) );
}
}
/**
* Performs the post_type and taxonomy queries for loading available items.
*
* @since
* @access public
*
* @param string $type Optional. Accepts any custom object type and has built-in support for
* 'post_type' and 'taxonomy'. Default is 'post_type'.
* @param string $object Optional. Accepts any registered taxonomy or post type name. Default is 'page'.
* @param int $page Optional. The page number used to generate the query offset. Default is '0'.
* @return WP_Error|array Returns either a WP_Error object or an array of menu items.
*/
public function load_available_items_query( $args ) {
//normalize args
$args = wp_parse_args( $args, array(
'type' => 'post',
'object_types' => '_all_',//could be page, post, or any CPT
'page' => 0
) );
$type = $args['type'];
$object_types = $args['object_types'];
$page = $args['page'];
$items = array();
if ( 'post' === $type ) {
//What are the post types we need to fetch ?
if ( '_all_' == $object_types || ! is_array( $object_types ) || ( is_array( $object_types ) && empty( $object_types ) ) ) {
$post_types = get_post_types( array( 'public' => true ) );
} else {
$post_types = $object_types;
}
if ( ! $post_types || ! is_array( $post_types ) || empty( $post_types ) ) {
return new \WP_Error( 'czr_contents_invalid_post_type' );
}
$posts = get_posts( array(
'numberposts' => 5,
'offset' => 5 * $page,
'orderby' => 'date',
'order' => 'DESC',
'post_type' => $post_types,
) );
foreach ( $posts as $post ) {
$post_title = $post->post_title;
if ( '' === $post_title ) {
// translators: %d: ID of a post
$post_title = sprintf( __( '#%d (no title)', 'customizr' ), $post->ID );
}
$items[] = array(
'title' => html_entity_decode( $post_title, ENT_QUOTES, get_bloginfo( 'charset' ) ),
'type' => 'post',
'type_label' => get_post_type_object( $post->post_type )->labels->singular_name,
'object' => $post->post_type,
'id' => intval( $post->ID ),
'url' => get_permalink( intval( $post->ID ) ),
);
}
} elseif ( 'taxonomy' === $type ) {
//What are the taxonomy types we need to fetch ?
if ( '_all_' == $object_types || ! is_array( $object_types ) || ( is_array( $object_types ) && empty( $object_types ) ) ) {
$taxonomies = get_taxonomies( array( 'show_in_nav_menus' => true ), 'names' );
} else {
$taxonomies = $object_types;
}
if ( ! $taxonomies || ! is_array( $taxonomies ) || empty( $taxonomies ) ) {
return new \WP_Error( 'czr_contents_invalid_post_type' );
}
$terms = get_terms( $taxonomies, array(
'child_of' => 0,
'exclude' => '',
'hide_empty' => false,
'hierarchical' => 1,
'include' => '',
'number' => 5,
'offset' => 5 * $page,
'order' => 'DESC',
'orderby' => 'count',
'pad_counts' => false,
) );
if ( is_wp_error( $terms ) ) {
return $terms;
}
foreach ( $terms as $term ) {
$items[] = array(
'title' => html_entity_decode( $term->name, ENT_QUOTES, get_bloginfo( 'charset' ) ),
'type' => 'taxonomy',
'type_label' => get_taxonomy( $term->taxonomy )->labels->singular_name,
'object' => $term->taxonomy,
'id' => intval( $term->term_id ),
'url' => get_term_link( intval( $term->term_id ), $term->taxonomy ),
);
}
}
/**
* Filters the available items.
*
* @since
*
* @param array $items The array of menu items.
* @param string $type The object type.
* @param string $object The object name.
* @param int $page The current page number.
*/
$items = apply_filters( 'czr_customize_content_picker_available_items', $items, $type, $object_types, $page );
return $items;
}
/* ------------------------------------------------------------------------- *
* SEARCH
/* ------------------------------------------------------------------------- */
/**
* Ajax handler for searching available menu items.
* hook : wp_ajax_search-available-content-items-customizer
*/
public function ajax_search_available_items() {
$action = 'save-customize_' . get_stylesheet();
if ( ! check_ajax_referer( $action, 'nonce', false ) ) {
wp_send_json_error( array(
'code' => 'invalid_nonce',
'message' => __( 'ajax_load_available_items => Security check failed.', 'customizr' ),
) );
}
if ( ! current_user_can( 'edit_theme_options' ) ) {
wp_send_json_error('ajax_load_available_items => user_cant_edit_theme_options');
}
if ( ! isset( $_POST['wp_object_types'] ) || empty( $_POST['wp_object_types'] ) ) {
wp_send_json_error( 'czr_ajax_content_picker_missing_type_or_object_parameter' );
}
if ( empty( $_POST['search'] ) ) {
wp_send_json_error( 'czr_contents_missing_search_parameter' );
}
$p = isset( $_POST['page'] ) ? absint( $_POST['page'] ) : 0;
if ( $p < 1 ) {
$p = 1;
}
$s = sanitize_text_field( wp_unslash( $_POST['search'] ) );
$wp_object_types = json_decode( wp_unslash( $_POST['wp_object_types'] ), true );
//$wp_object_types should look like :
//array(
// post : '',//<= all post types
// taxonomy : ''//<= all taxonomy types
//) OR
//array(
// post : [ 'page', 'cpt1', ...]
// taxonomy : [ 'category', 'tag', 'Custom_Tax_1', ... ]
//) OR
//array(
// post : [ 'page', 'cpt1', ...]
// taxonomy : '_none_'//<= don't load or search taxonomies
//)
if ( ! is_array( $wp_object_types ) || empty( $wp_object_types ) ) {
wp_send_json_error( 'czr_ajax_content_picker_missing_object_types' );
}
$items = array();
add_filter( 'pre_post_link', array( $this, 'dont_use_fancy_permalinks' ), 999 );
foreach ( $wp_object_types as $_type => $_obj_types ) {
if ( '_none_' == $_obj_types )
continue;
$item_candidates = $this -> search_available_items_query(
array(
'type' => $_type, //<= post or taxonomy
'object_types' => $_obj_types,//<= '' or array( type1, type2, ... )
'pagenum' => $p,
's' => $s
)
);
if ( is_array( $item_candidates ) ) {
$items = array_merge(
$items,
$item_candidates
);
}
}
remove_filter( 'pre_post_link', array( $this, 'dont_use_fancy_permalinks' ), 999 );
if ( empty( $items ) ) {
wp_send_json_success( array( 'message' => __( 'No results found.', 'customizr') ) );
} else {
wp_send_json_success( array(
'items' => apply_filters( 'content_picker_ajax_items', $items, $p, 'ajax_search_available_items' )
) );
}
}
/**
* Performs post queries for available-item searching.
*
* Based on WP_Editor::wp_link_query().
*
* @since 4.3.0
* @access public
*
* @param array $args Optional. Accepts 'pagenum' and 's' (search) arguments.
* @return array Menu items.
*/
public function search_available_items_query( $args = array() ) {
//normalize args
$args = wp_parse_args( $args, array(
'pagenum' => 1,
's' => '',
'type' => 'post',
'object_types' => '_all_'//could be page, post, or any CPT
) );
$object_types = $args['object_types'];
//TODO: need a search only on the allowed types
$items = array();
if ( 'post' === $args['type'] ) {
//What are the post types we need to fetch ?
if ( '_all_' == $object_types || ! is_array( $object_types ) || ( is_array( $object_types ) && empty( $object_types ) ) ) {
$post_types = get_post_types( array( 'public' => true ) );
} else {
$post_types = $object_types;
}
if ( ! $post_types || empty( $post_types ) ) {
return new \WP_Error( 'czr_contents_invalid_post_type' );
}
$query = array(
'suppress_filters' => true,
'update_post_term_cache' => false,
'update_post_meta_cache' => false,
'post_status' => 'publish',
'posts_per_page' => 10,
);
$args['pagenum'] = isset( $args['pagenum'] ) ? absint( $args['pagenum'] ) : 1;
$query['offset'] = $args['pagenum'] > 1 ? $query['posts_per_page'] * ( $args['pagenum'] - 1 ) : 0;
$query['post_type'] = $post_types;
if ( isset( $args['s'] ) ) {
$query['s'] = $args['s'];
}
// Query posts.
$get_posts = new \WP_Query( $query );
// Check if any posts were found.
if ( $get_posts->post_count ) {
foreach ( $get_posts->posts as $post ) {
$post_title = $post->post_title;
if ( '' === $post_title ) {
/* translators: %d: ID of a post */
$post_title = sprintf( __( '#%d (no title)', 'customizr' ), $post->ID );
}
$items[] = array(
'title' => html_entity_decode( $post_title, ENT_QUOTES, get_bloginfo( 'charset' ) ),
'type' => 'post',
'type_label' => get_post_type_object( $post->post_type )->labels->singular_name,
'object' => $post->post_type,
'id' => intval( $post->ID ),
'url' => get_permalink( intval( $post->ID ) ),
);
}
}
} elseif ( 'taxonomy' === $args['type'] ) {
//What are the taxonomy types we need to fetch ?
if ( '_all_' == $object_types || ! is_array( $object_types ) || ( is_array( $object_types ) && empty( $object_types ) ) ) {
$taxonomies = get_taxonomies( array( 'show_in_nav_menus' => true ), 'names' );
} else {
$taxonomies = $object_types;
}
if ( ! $taxonomies || ! is_array( $taxonomies ) || empty( $taxonomies ) ) {
return new \WP_Error( 'czr_contents_invalid_post_type' );
}
$terms = get_terms( $taxonomies, array(
'name__like' => $args['s'],
'number' => 10,
'offset' => 10 * ( $args['pagenum'] - 1 )
) );
// Check if any taxonomies were found.
if ( ! empty( $terms ) ) {
foreach ( $terms as $term ) {
$items[] = array(
'title' => html_entity_decode( $term->name, ENT_QUOTES, get_bloginfo( 'charset' ) ),
'type' => 'taxonomy',
'type_label' => get_taxonomy( $term->taxonomy )->labels->singular_name,
'object' => $term->taxonomy,
'id' => intval( $term->term_id ),
'url' => get_term_link( intval( $term->term_id ), $term->taxonomy ),
);
}
}
/**
* Filters the available menu items during a search request.
*
* @since 4.5.0
*
* @param array $items The array of menu items.
* @param array $args Includes 'pagenum' and 's' (search) arguments.
*/
$items = apply_filters( 'czr_customize_content_picker_searched_items', $items, $args );
}
return $items;
}
}
endif;
?> parent() ) {
$theme_slug = $theme_data -> parent() -> Name;
}
return sanitize_file_name( strtolower( $theme_slug ) );
}
//@return boolean
function czr_is_multi_item_module( $module_type ) {
$is_multi_item = false;
$module_params = CZR_Fmk_Base() -> czr_get_registered_dynamic_module( $module_type );
if ( is_array( $module_params ) ) {
if ( array_key_exists( 'is_crud', $module_params ) ) {
$is_multi_item = (bool)$module_params['is_crud'];
}
if ( array_key_exists( 'is_multi_item', $module_params ) ) {
$is_multi_item = (bool)$module_params['is_multi_item'];
}
}
return $is_multi_item;
}
//Creates a new instance
//@params ex :
//array(
// 'base_url' => NIMBLE_BASE_URL . '/inc/czr-base-fmk'
// )
function CZR_Fmk_Base( $params = array() ) {
return CZR_Fmk_Base::czr_fmk_get_instance( $params );
}
?>