diff --git a/docs/features/field/nav_menu/index.md b/docs/features/field/nav_menu/index.md new file mode 100644 index 00000000..b1ac75b6 --- /dev/null +++ b/docs/features/field/nav_menu/index.md @@ -0,0 +1,17 @@ +# Nav Menu Field + +The Nav Menu Field field provides a way to select nav menus and output them. + +## Key Features + +- Different Return Value (Object, HTML, ID) +- Different Menu Container (nav, div) +- Ability to select no value + +## Settings + +- Return Value +- Menu Container +- Allow Null? + + diff --git a/docs/features/field/nav_menu/tutorial.md b/docs/features/field/nav_menu/tutorial.md new file mode 100644 index 00000000..bc21954d --- /dev/null +++ b/docs/features/field/nav_menu/tutorial.md @@ -0,0 +1,71 @@ +# **Using the Nav Menu Field** + +## **Basic Setup** + +To get started with the **Nav Menu Field** in SCF, follow these steps to configure and display a WordPress navigation menu: + +### **Step 1: Create a New Field Group** + +First, create a new field group in **Secure Custom Fields (SCF)**. This field group will hold your custom fields, including the **Nav Menu** field. + +1. Go to **Custom Fields > Add New** in your WordPress admin panel. +2. Title your field group and choose the location rules for where this field group should be displayed (e.g., on a specific page or post type). + +### **Step 2: Add a Nav Menu Field** + +Once you've created a new field group, add a new field of type **Nav Menu**. This field will allow users to select a navigation menu from the available menus on your site. + +1. Click **Add Field** within the field group. +2. Set the **Field Type** to **Nav Menu**. +3. Configure the necessary field options based on your needs. + +### **Step 3: Configure Options** + +Configure the field options for the **Nav Menu** field to tailor it to your needs. + +- **Return Value:** Choose how you want the selected menu to be returned: + - **Menu ID:** The ID of the selected menu. + - **Nav Menu HTML:** The raw HTML of the selected menu. + - **Nav Menu Object:** The complete menu object, including properties like the menu name and term ID. + +- **Menu Container:** This option determines the wrapper tag for the menu when the **Nav Menu HTML** return format is selected. Common options include `div`, `nav`, or `none`. You can also add additional container tags using the `wp_nav_menu_container_allowed_tags` filter. + +- **Allow Null:** Set this option to **true** or **false** to allow or disallow a **null** value (i.e., no menu selected). When **true**, an empty option will be available for the user to select. + +### **Code Example: Basic Setup** + +Here's how you can implement the **Nav Menu Field** in a template file: + +```php +// Check if the Nav Menu Field has a value +$menu_id = get_field('your_nav_menu_field_name'); + +if ($menu_id) { + // Get the menu object or HTML based on your configuration + $menu_html = wp_nav_menu(array( + 'menu' => $menu_id, // Use the menu ID + 'container' => 'nav', // Use a 'nav' container + 'echo' => false, // Don't output directly; return the HTML + )); + + echo $menu_html; // Output the menu HTML +} +``` + +## **Common Use Cases** + +1. **Use to Output WordPress Navigation Menu Anywhere:** + The **Nav Menu Field** allows you to output a WordPress navigation menu in custom locations across your site, such as in custom templates, widgets, or shortcodes. + +## **Tips** + +- **To Add More Menu Containers:** + Use the `wp_nav_menu_container_allowed_tags` hook to add additional allowed container tags for the Nav Menu field. This will enable more flexibility in the menu's wrapper tag. + + Example: + ```php + function my_custom_menu_container_tags($tags) { + $tags[] = 'section'; // Adds 'section' as an allowed container tag + return $tags; + } + add_filter('wp_nav_menu_container_allowed_tags', 'my_custom_menu_container_tags'); diff --git a/includes/fields/class-acf-field-nav-menu.php b/includes/fields/class-acf-field-nav-menu.php new file mode 100644 index 00000000..4e612e5b --- /dev/null +++ b/includes/fields/class-acf-field-nav-menu.php @@ -0,0 +1,243 @@ +name = 'nav_menu'; + $this->label = _x( 'Nav Menu', 'noun', 'secure-custom-fields' ); + $this->category = 'choice'; + $this->description = __( 'A dropdown list with a selection of menu choices that you can chose.', 'secure-custom-fields' ); + $this->preview_image = acf_get_url() . '/assets/images/field-type-previews/field-preview-select.png'; + $this->doc_url = 'https://developer.wordpress.org/secure-custom-fields/features/fields/nav_menu/'; + $this->tutorial_url = 'https://developer.wordpress.org/secure-custom-fields/features/fields/select/nav_menu-tutorial/'; + $this->defaults = array( + 'save_format' => 'id', + 'allow_null' => 0, + 'container' => 'div', + ); + + add_filter( 'acf/field_wrapper_attributes', array( $this, 'nav_menu_field_wrapper_attributes' ), 10, 2 ); + } + + /** + * Renders the Nav Menu Field options seen when editing a Nav Menu Field. + * + * @param array $field The array representation of the current Nav Menu Field. + */ + public function render_field_settings( $field ) { + $allow_null = $field['allow_null']; + $nav_menus = wp_get_nav_menus( $allow_null ); + if ( current_theme_supports( 'menus' ) ) { + if ( empty( $nav_menus ) ) { + ?> +
+
+

+ ' . esc_html__( 'the menu settings page', 'secure-custom-fields' ) . '' + ); + ?> +

+
+
+ +
+
+

+ +

+
+
+ __( 'Return Value', 'secure-custom-fields' ), + 'instructions' => __( 'Specify the returned value on front end', 'secure-custom-fields' ), + 'type' => 'radio', + 'name' => 'save_format', + 'layout' => 'horizontal', + 'choices' => array( + 'object' => __( 'Nav Menu Object', 'secure-custom-fields' ), + 'menu' => __( 'Nav Menu HTML', 'secure-custom-fields' ), + 'id' => __( 'Nav Menu ID', 'secure-custom-fields' ), + ), + ) + ); + + // Register the Menu Container setting + acf_render_field_setting( + $field, + array( + 'label' => __( 'Menu Container', 'secure-custom-fields' ), + 'instructions' => __( "What to wrap the Menu's ul with (when returning HTML only)", 'secure-custom-fields' ), + 'type' => 'select', + 'name' => 'container', + 'choices' => $this->get_allowed_nav_container_tags(), + ) + ); + + // Register the Allow Null setting + acf_render_field_setting( + $field, + array( + 'label' => __( 'Allow Null?', 'secure-custom-fields' ), + 'type' => 'radio', + 'name' => 'allow_null', + 'layout' => 'horizontal', + 'choices' => array( + 1 => __( 'Yes', 'secure-custom-fields' ), + 0 => __( 'No', 'secure-custom-fields' ), + ), + ) + ); + } + /** + * Get the allowed wrapper tags for use with wp_nav_menu(). + * + * @return array An array of allowed wrapper tags. + */ + private function get_allowed_nav_container_tags() { + $tags = apply_filters( 'wp_nav_menu_container_allowed_tags', array( 'div', 'nav' ) ); + $formatted_tags = array( + '0' => 'None', + ); + + foreach ( $tags as $tag ) { + $formatted_tags[ $tag ] = $tag; + } + + return $formatted_tags; + } + /** + * Renders the Nav Menu Field. + * + * @param array $field The array representation of the current Nav Menu Field. + */ + public function render_field( $field ) { + $allow_null = $field['allow_null']; + $nav_menus = wp_get_nav_menus( $allow_null ); + if ( ! current_theme_supports( 'menus' ) || empty( $nav_menus ) ) { + return; // Don't render the field + } + + ?> + + ID = $wp_menu_object->term_id; + $menu_object->name = $wp_menu_object->name; + $menu_object->slug = $wp_menu_object->slug; + $menu_object->count = $wp_menu_object->count; + + return $menu_object; + } elseif ( 'menu' === $field['save_format'] ) { + ob_start(); + + wp_nav_menu( + array( + 'menu' => $value, + 'container' => $field['container'], + ) + ); + + return ob_get_clean(); + } + + // Just return the Nav Menu ID + return $value; + } + /** + * Hide Field if no support + * + * @param array $wrapper Wrapper array that contains all field main wrapper attributes. + * @param array $field main field array will all field data. + */ + public function nav_menu_field_wrapper_attributes( $wrapper, $field ) { + // Check if it's the nav menu field (or any other specific field type) + if ( isset( $field['type'] ) && 'nav_menu' === $field['type'] ) { + // Check if menus are available and the theme supports them + if ( ! current_theme_supports( 'menus' ) ) { + // Add inline CSS to hide the field if no menus are available + $wrapper['style'] = 'display: none;'; // You can also add additional styles + } + } + + return $wrapper; + } + } + + + // initialize + acf_register_field_type( 'Acf_Field_Nav_Menu' ); +endif; // class_exists check diff --git a/secure-custom-fields.php b/secure-custom-fields.php index 4f86fa79..dc494ade 100644 --- a/secure-custom-fields.php +++ b/secure-custom-fields.php @@ -328,6 +328,7 @@ public function init() { acf_include( 'includes/fields/class-acf-field-flexible-content.php' ); acf_include( 'includes/fields/class-acf-field-gallery.php' ); acf_include( 'includes/fields/class-acf-field-clone.php' ); + acf_include( 'includes/fields/class-acf-field-nav-menu.php' ); /** * Fires after field types have been included.