ControlledChaos
7/18/2014 - 2:56 PM

Plugin to allow for Advanced Custom Fields to be used in widgets within the Customizer

Plugin to allow for Advanced Custom Fields to be used in widgets within the Customizer

<?php
/*
Plugin Name: ACF Customizer Patch
Plugin URI: https://gist.github.com/fabrizim/9c0f36365f20705f7f73
Description: A class to allow acf widget fields to be stored with normal widget settings and allow for use in customizer.
Author: Mark Fabrizio
Version: 1.0
Author URI: http://owlwatch.com/
*/
class acf_customizer_patch
{
  
  protected $_capture_options = false;
  protected $_captured_options = array();
  protected $_instance;
  
  public function __construct()
  {
    add_filter('widget_display_callback', array($this, 'before_widget_display'), 20, 3);
    add_filter('in_widget_form',          array($this, 'before_edit_form'),       9, 3);
    add_filter('widget_update_callback',  array($this, 'before_save_acf_fields'), 9, 3);
    add_filter('pre_update_option',       array($this, 'pre_update_option'),      8, 3);
    add_filter('widget_update_callback',  array($this, 'after_save_acf_fields'), 11, 3);
    
    // need to include the acf input scripts
    // This could be included in the 'acf_input_listener' class
    add_action('customize_controls_print_footer_scripts', array( $this, 'admin_footer'), 20 );
  }
  
  public function admin_footer() {
    do_action('acf/input/admin_footer');
  }
  
  /**
   * There may be a better way to find out if we are in the customizer,
   * but this works for now. If someone has a better way, let me know.
   */
  protected function in_customizer()
  {
    return @$_REQUEST['wp_customize'] == 'on' || basename($_SERVER['SCRIPT_NAME']) == 'customize.php';
  }
  
  /**
   * @wp.filter       widget_display_callback
   * @wp.priority     20
   */
  public function before_widget_display( $instance, $widget, $args )
  {
    $this->prepare_widget_options( $widget, $instance );
    return $instance;
  }
  
  /**
   * @wp.filter       in_widget_form
   * @wp.priority     9
   */
  public function before_edit_form( $widget, $return, $instance )
  {
    $this->prepare_widget_options( $widget, $instance );
    return $widget;
  }
  
  public function prepare_widget_options( $widget, $instance )
  {
    $this->_instance = $instance;
    $field_groups = acf_get_field_groups(array(
      'widget'  => $widget->id_base
    ));
    
    if( count( $field_groups ) ) foreach( $field_groups as $group  ){
      $fields = acf_get_fields( $group );
      if( count($fields) ) foreach( $fields as $field ){
        $name = $field['name'];
        add_filter("pre_option_widget_{$widget->id}_{$name}", array($this, 'pre_get_option') );
      }
    }
    return $widget;
  }
  
  public function pre_get_option( $value )
  {
    $filter = current_filter();
    $name = str_replace('pre_option_widget_', '', $filter);
    preg_match('/(.+\-\d+?)_(.+)/', $name, $matches );
    
    list( $full, $widget_id, $var ) = $matches;
    
    if( $this->_instance && isset( $this->_instance[$var] ) ) return $this->_instance[$var];
    return $value;
  }
  
  /**
   * @wp.filter       widget_update_callback
   * @wp.priority     9
   */
  public function before_save_acf_fields( $instance, $new_instance, $old_instance )
  {
    global $wp_customize;
    $this->_capture_options = true;
    if( $this->in_customizer() )
      remove_filter( 'pre_update_option', array( $wp_customize->widgets, 'capture_filter_pre_update_option' ), 10);
    return $instance;
  }
  
  /**
   * @wp.filter       pre_update_option
   * @wp.priority     8
   */
  public function pre_update_option( $value, $option, $old_value )
  {
    global $wp_customize;
    
    if( !$this->_capture_options ) return $value;
    
    if( preg_match('/^([^_].+\-\d+?)_(.+)/', $option, $matches ) ){
      $this->_captured_options[$matches[2]] = $value;
    }
    // if this is not an acf field, this should be the actual
    // widget option (which needs to be captured by the customizer)
    if( !preg_match('/^(.+\-\d+?)_(.+)/', $option ) && $this->in_customizer() ){
      $wp_customize->widgets->capture_filter_pre_update_option( $value, $option, $old_value );
    }
    
     return $this->in_customizer() ? $old_value : $value;
  }
  
  /**
   * @wp.filter       widget_update_callback
   * @wp.priority     11
   */
  public function after_save_acf_fields( $instance, $new_instance, $old_instance )
  {
    $instance = array_merge( $instance, $this->_captured_options );
    $this->_capture_options = false;
    return $instance;
  }
  
}

$acf_customizer_patch = new acf_customizer_patch();