398 lines
8.0 KiB
PHP
398 lines
8.0 KiB
PHP
|
<?php
|
||
|
|
||
|
if ( ! defined( 'ABSPATH' ) ) {
|
||
|
exit; // Exit if accessed directly
|
||
|
}
|
||
|
|
||
|
if ( ! class_exists( 'acf_form_nav_menu' ) ) :
|
||
|
|
||
|
class acf_form_nav_menu {
|
||
|
|
||
|
/*
|
||
|
* __construct
|
||
|
*
|
||
|
* This function will setup the class functionality
|
||
|
*
|
||
|
* @type function
|
||
|
* @date 5/03/2014
|
||
|
* @since 5.0.0
|
||
|
*
|
||
|
* @param n/a
|
||
|
* @return n/a
|
||
|
*/
|
||
|
|
||
|
function __construct() {
|
||
|
|
||
|
// actions
|
||
|
add_action( 'admin_enqueue_scripts', array( $this, 'admin_enqueue_scripts' ) );
|
||
|
add_action( 'wp_update_nav_menu', array( $this, 'update_nav_menu' ) );
|
||
|
add_action( 'acf/validate_save_post', array( $this, 'acf_validate_save_post' ), 5 );
|
||
|
add_action( 'wp_nav_menu_item_custom_fields', array( $this, 'wp_nav_menu_item_custom_fields' ), 10, 5 );
|
||
|
|
||
|
// filters
|
||
|
add_filter( 'wp_get_nav_menu_items', array( $this, 'wp_get_nav_menu_items' ), 10, 3 );
|
||
|
add_filter( 'wp_edit_nav_menu_walker', array( $this, 'wp_edit_nav_menu_walker' ), 10, 2 );
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* admin_enqueue_scripts
|
||
|
*
|
||
|
* This action is run after post query but before any admin script / head actions.
|
||
|
* It is a good place to register all actions.
|
||
|
*
|
||
|
* @type action (admin_enqueue_scripts)
|
||
|
* @date 26/01/13
|
||
|
* @since 3.6.0
|
||
|
*
|
||
|
* @param N/A
|
||
|
* @return N/A
|
||
|
*/
|
||
|
|
||
|
function admin_enqueue_scripts() {
|
||
|
|
||
|
// validate screen
|
||
|
if ( ! acf_is_screen( 'nav-menus' ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// load acf scripts
|
||
|
acf_enqueue_scripts();
|
||
|
|
||
|
// actions
|
||
|
add_action( 'admin_footer', array( $this, 'admin_footer' ), 1 );
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* wp_nav_menu_item_custom_fields
|
||
|
*
|
||
|
* description
|
||
|
*
|
||
|
* @date 30/7/18
|
||
|
* @since 5.6.9
|
||
|
*
|
||
|
* @param type $var Description. Default.
|
||
|
* @return type Description.
|
||
|
*/
|
||
|
|
||
|
function wp_nav_menu_item_custom_fields( $item_id, $item, $depth, $args, $id = '' ) {
|
||
|
|
||
|
// vars
|
||
|
$prefix = "menu-item-acf[$item_id]";
|
||
|
|
||
|
// get field groups
|
||
|
$field_groups = acf_get_field_groups(
|
||
|
array(
|
||
|
'nav_menu_item' => $item->type,
|
||
|
'nav_menu_item_id' => $item_id,
|
||
|
'nav_menu_item_depth' => $depth,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
// render
|
||
|
if ( ! empty( $field_groups ) ) {
|
||
|
|
||
|
// open
|
||
|
echo '<div class="acf-menu-item-fields acf-fields -clear">';
|
||
|
|
||
|
// loop
|
||
|
foreach ( $field_groups as $field_group ) {
|
||
|
|
||
|
// load fields
|
||
|
$fields = acf_get_fields( $field_group );
|
||
|
|
||
|
// bail if not fields
|
||
|
if ( empty( $fields ) ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// change prefix
|
||
|
acf_prefix_fields( $fields, $prefix );
|
||
|
|
||
|
// render
|
||
|
acf_render_fields( $fields, $item_id, 'div', $field_group['instruction_placement'] );
|
||
|
}
|
||
|
|
||
|
// close
|
||
|
echo '</div>';
|
||
|
|
||
|
// Trigger append for newly created menu item (via AJAX)
|
||
|
if ( acf_is_ajax( 'add-menu-item' ) ) : ?>
|
||
|
<script type="text/javascript">
|
||
|
(function($) {
|
||
|
acf.doAction('append', $('#menu-item-settings-<?php echo $item_id; ?>') );
|
||
|
})(jQuery);
|
||
|
</script>
|
||
|
<?php
|
||
|
endif;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* update_nav_menu
|
||
|
*
|
||
|
* description
|
||
|
*
|
||
|
* @type function
|
||
|
* @date 26/5/17
|
||
|
* @since 5.6.0
|
||
|
*
|
||
|
* @param $post_id (int)
|
||
|
* @return $post_id (int)
|
||
|
*/
|
||
|
|
||
|
function update_nav_menu( $menu_id ) {
|
||
|
|
||
|
// vars
|
||
|
$post_id = 'term_' . $menu_id;
|
||
|
|
||
|
// verify and remove nonce
|
||
|
if ( ! acf_verify_nonce( 'nav_menu' ) ) {
|
||
|
return $menu_id;
|
||
|
}
|
||
|
|
||
|
// validate and show errors
|
||
|
acf_validate_save_post( true );
|
||
|
|
||
|
// save
|
||
|
acf_save_post( $post_id );
|
||
|
|
||
|
// save nav menu items
|
||
|
$this->update_nav_menu_items( $menu_id );
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* update_nav_menu_items
|
||
|
*
|
||
|
* description
|
||
|
*
|
||
|
* @type function
|
||
|
* @date 26/5/17
|
||
|
* @since 5.6.0
|
||
|
*
|
||
|
* @param $post_id (int)
|
||
|
* @return $post_id (int)
|
||
|
*/
|
||
|
|
||
|
function update_nav_menu_items( $menu_id ) {
|
||
|
|
||
|
// phpcs:disable WordPress.Security.NonceVerification.Missing -- Verified elsewhere.
|
||
|
if ( empty( $_POST['menu-item-acf'] ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$posted_values = acf_sanitize_request_args( $_POST['menu-item-acf'] );
|
||
|
|
||
|
foreach ( $posted_values as $post_id => $values ) {
|
||
|
|
||
|
acf_save_post( $post_id, $values );
|
||
|
|
||
|
}
|
||
|
// phpcs:enable WordPress.Security.NonceVerification.Missing
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* wp_get_nav_menu_items
|
||
|
*
|
||
|
* WordPress does not provide an easy way to find the current menu being edited.
|
||
|
* This function listens to when a menu's items are loaded and stores the menu.
|
||
|
* Needed on nav-menus.php page for new menu with no items
|
||
|
*
|
||
|
* @date 23/2/18
|
||
|
* @since 5.6.9
|
||
|
*
|
||
|
* @param type $var Description. Default.
|
||
|
* @return type Description.
|
||
|
*/
|
||
|
|
||
|
function wp_get_nav_menu_items( $items, $menu, $args ) {
|
||
|
acf_set_data( 'nav_menu_id', $menu->term_id );
|
||
|
return $items;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Called when WP renders a menu edit form.
|
||
|
* Used to set global data and customize the Walker class.
|
||
|
*
|
||
|
* @date 26/5/17
|
||
|
* @since 5.6.0
|
||
|
*
|
||
|
* @param string $class The walker class to use. Default 'Walker_Nav_Menu_Edit'.
|
||
|
* @param int $menu_id ID of the menu being rendered.
|
||
|
* @return string
|
||
|
*/
|
||
|
function wp_edit_nav_menu_walker( $class, $menu_id = 0 ) {
|
||
|
|
||
|
// update data (needed for ajax location rules to work)
|
||
|
acf_set_data( 'nav_menu_id', $menu_id );
|
||
|
|
||
|
// Return class.
|
||
|
return $class;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* acf_validate_save_post
|
||
|
*
|
||
|
* This function will loop over $_POST data and validate
|
||
|
*
|
||
|
* @type action 'acf/validate_save_post' 5
|
||
|
* @date 7/09/2016
|
||
|
* @since 5.4.0
|
||
|
*
|
||
|
* @param n/a
|
||
|
* @return n/a
|
||
|
*/
|
||
|
|
||
|
function acf_validate_save_post() {
|
||
|
|
||
|
// phpcs:disable WordPress.Security.NonceVerification.Missing -- Verified elsewhere.
|
||
|
if ( empty( $_POST['menu-item-acf'] ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$posted_values = acf_sanitize_request_args( $_POST['menu-item-acf'] );
|
||
|
|
||
|
foreach ( $posted_values as $post_id => $values ) {
|
||
|
|
||
|
// vars
|
||
|
$prefix = 'menu-item-acf[' . $post_id . ']';
|
||
|
|
||
|
// validate
|
||
|
acf_validate_values( $values, $prefix );
|
||
|
|
||
|
}
|
||
|
// phpcs:enable // phpcs:disable WordPress.Security.NonceVerification.Missing
|
||
|
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* admin_footer
|
||
|
*
|
||
|
* This function will add some custom HTML to the footer of the edit page
|
||
|
*
|
||
|
* @type function
|
||
|
* @date 11/06/2014
|
||
|
* @since 5.0.0
|
||
|
*
|
||
|
* @param n/a
|
||
|
* @return n/a
|
||
|
*/
|
||
|
|
||
|
function admin_footer() {
|
||
|
|
||
|
// vars
|
||
|
$nav_menu_id = acf_get_data( 'nav_menu_id' );
|
||
|
$post_id = 'term_' . $nav_menu_id;
|
||
|
|
||
|
// get field groups
|
||
|
$field_groups = acf_get_field_groups(
|
||
|
array(
|
||
|
'nav_menu' => $nav_menu_id,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
?>
|
||
|
<div id="tmpl-acf-menu-settings" style="display: none;">
|
||
|
<?php
|
||
|
|
||
|
// data (always needed to save nav menu items)
|
||
|
acf_form_data(
|
||
|
array(
|
||
|
'screen' => 'nav_menu',
|
||
|
'post_id' => $post_id,
|
||
|
'ajax' => 1,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
// render
|
||
|
if ( ! empty( $field_groups ) ) {
|
||
|
|
||
|
// loop
|
||
|
foreach ( $field_groups as $field_group ) {
|
||
|
|
||
|
$fields = acf_get_fields( $field_group );
|
||
|
|
||
|
echo '<div class="acf-menu-settings -' . $field_group['style'] . '">';
|
||
|
|
||
|
echo '<h2>' . $field_group['title'] . '</h2>';
|
||
|
|
||
|
echo '<div class="acf-fields -left -clear">';
|
||
|
|
||
|
acf_render_fields( $fields, $post_id, 'div', $field_group['instruction_placement'] );
|
||
|
|
||
|
echo '</div>';
|
||
|
|
||
|
echo '</div>';
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
?>
|
||
|
</div>
|
||
|
<script type="text/javascript">
|
||
|
(function($) {
|
||
|
|
||
|
// append html
|
||
|
var html = $('#tmpl-acf-menu-settings').html();
|
||
|
$('#tmpl-acf-menu-settings').remove();
|
||
|
$('#post-body-content').append( html );
|
||
|
|
||
|
|
||
|
// avoid WP over-writing $_POST data
|
||
|
// - https://core.trac.wordpress.org/ticket/41502#ticket
|
||
|
$(document).on('submit', '#update-nav-menu', function() {
|
||
|
|
||
|
// vars
|
||
|
var $form = $(this);
|
||
|
var $input = $('input[name="nav-menu-data"]');
|
||
|
|
||
|
|
||
|
// decode json
|
||
|
var json = $form.serializeArray();
|
||
|
var json2 = [];
|
||
|
|
||
|
|
||
|
// loop
|
||
|
$.each( json, function( i, pair ) {
|
||
|
|
||
|
// avoid nesting (unlike WP)
|
||
|
if( pair.name === 'nav-menu-data' ) return;
|
||
|
|
||
|
|
||
|
// bail early if is 'acf[' input
|
||
|
if( pair.name.indexOf('acf[') > -1 ) return;
|
||
|
|
||
|
|
||
|
// append
|
||
|
json2.push( pair );
|
||
|
|
||
|
});
|
||
|
|
||
|
|
||
|
// update
|
||
|
$input.val( JSON.stringify(json2) );
|
||
|
|
||
|
});
|
||
|
|
||
|
|
||
|
})(jQuery);
|
||
|
</script>
|
||
|
<?php
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
acf_new_instance( 'acf_form_nav_menu' );
|
||
|
|
||
|
endif;
|
||
|
?>
|