説明なし

nav-menu.php 46KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303
  1. <?php
  2. /**
  3. * Core Navigation Menu API
  4. *
  5. * @package WordPress
  6. * @subpackage Nav_Menus
  7. * @since 3.0.0
  8. */
  9. /** Walker_Nav_Menu_Edit class */
  10. require_once ABSPATH . 'wp-admin/includes/class-walker-nav-menu-edit.php';
  11. /** Walker_Nav_Menu_Checklist class */
  12. require_once ABSPATH . 'wp-admin/includes/class-walker-nav-menu-checklist.php';
  13. /**
  14. * Prints the appropriate response to a menu quick search.
  15. *
  16. * @since 3.0.0
  17. *
  18. * @param array $request The unsanitized request values.
  19. */
  20. function _wp_ajax_menu_quick_search( $request = array() ) {
  21. $args = array();
  22. $type = isset( $request['type'] ) ? $request['type'] : '';
  23. $object_type = isset( $request['object_type'] ) ? $request['object_type'] : '';
  24. $query = isset( $request['q'] ) ? $request['q'] : '';
  25. $response_format = isset( $request['response-format'] ) ? $request['response-format'] : '';
  26. if ( ! $response_format || ! in_array( $response_format, array( 'json', 'markup' ), true ) ) {
  27. $response_format = 'json';
  28. }
  29. if ( 'markup' === $response_format ) {
  30. $args['walker'] = new Walker_Nav_Menu_Checklist;
  31. }
  32. if ( 'get-post-item' === $type ) {
  33. if ( post_type_exists( $object_type ) ) {
  34. if ( isset( $request['ID'] ) ) {
  35. $object_id = (int) $request['ID'];
  36. if ( 'markup' === $response_format ) {
  37. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', array( get_post( $object_id ) ) ), 0, (object) $args );
  38. } elseif ( 'json' === $response_format ) {
  39. echo wp_json_encode(
  40. array(
  41. 'ID' => $object_id,
  42. 'post_title' => get_the_title( $object_id ),
  43. 'post_type' => get_post_type( $object_id ),
  44. )
  45. );
  46. echo "\n";
  47. }
  48. }
  49. } elseif ( taxonomy_exists( $object_type ) ) {
  50. if ( isset( $request['ID'] ) ) {
  51. $object_id = (int) $request['ID'];
  52. if ( 'markup' === $response_format ) {
  53. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', array( get_term( $object_id, $object_type ) ) ), 0, (object) $args );
  54. } elseif ( 'json' === $response_format ) {
  55. $post_obj = get_term( $object_id, $object_type );
  56. echo wp_json_encode(
  57. array(
  58. 'ID' => $object_id,
  59. 'post_title' => $post_obj->name,
  60. 'post_type' => $object_type,
  61. )
  62. );
  63. echo "\n";
  64. }
  65. }
  66. }
  67. } elseif ( preg_match( '/quick-search-(posttype|taxonomy)-([a-zA-Z_-]*\b)/', $type, $matches ) ) {
  68. if ( 'posttype' === $matches[1] && get_post_type_object( $matches[2] ) ) {
  69. $post_type_obj = _wp_nav_menu_meta_box_object( get_post_type_object( $matches[2] ) );
  70. $args = array_merge(
  71. $args,
  72. array(
  73. 'no_found_rows' => true,
  74. 'update_post_meta_cache' => false,
  75. 'update_post_term_cache' => false,
  76. 'posts_per_page' => 10,
  77. 'post_type' => $matches[2],
  78. 's' => $query,
  79. )
  80. );
  81. if ( isset( $post_type_obj->_default_query ) ) {
  82. $args = array_merge( $args, (array) $post_type_obj->_default_query );
  83. }
  84. $search_results_query = new WP_Query( $args );
  85. if ( ! $search_results_query->have_posts() ) {
  86. return;
  87. }
  88. while ( $search_results_query->have_posts() ) {
  89. $post = $search_results_query->next_post();
  90. if ( 'markup' === $response_format ) {
  91. $var_by_ref = $post->ID;
  92. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', array( get_post( $var_by_ref ) ) ), 0, (object) $args );
  93. } elseif ( 'json' === $response_format ) {
  94. echo wp_json_encode(
  95. array(
  96. 'ID' => $post->ID,
  97. 'post_title' => get_the_title( $post->ID ),
  98. 'post_type' => $matches[2],
  99. )
  100. );
  101. echo "\n";
  102. }
  103. }
  104. } elseif ( 'taxonomy' === $matches[1] ) {
  105. $terms = get_terms(
  106. array(
  107. 'taxonomy' => $matches[2],
  108. 'name__like' => $query,
  109. 'number' => 10,
  110. 'hide_empty' => false,
  111. )
  112. );
  113. if ( empty( $terms ) || is_wp_error( $terms ) ) {
  114. return;
  115. }
  116. foreach ( (array) $terms as $term ) {
  117. if ( 'markup' === $response_format ) {
  118. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', array( $term ) ), 0, (object) $args );
  119. } elseif ( 'json' === $response_format ) {
  120. echo wp_json_encode(
  121. array(
  122. 'ID' => $term->term_id,
  123. 'post_title' => $term->name,
  124. 'post_type' => $matches[2],
  125. )
  126. );
  127. echo "\n";
  128. }
  129. }
  130. }
  131. }
  132. }
  133. /**
  134. * Register nav menu meta boxes and advanced menu items.
  135. *
  136. * @since 3.0.0
  137. */
  138. function wp_nav_menu_setup() {
  139. // Register meta boxes.
  140. wp_nav_menu_post_type_meta_boxes();
  141. add_meta_box( 'add-custom-links', __( 'Custom Links' ), 'wp_nav_menu_item_link_meta_box', 'nav-menus', 'side', 'default' );
  142. wp_nav_menu_taxonomy_meta_boxes();
  143. // Register advanced menu items (columns).
  144. add_filter( 'manage_nav-menus_columns', 'wp_nav_menu_manage_columns' );
  145. // If first time editing, disable advanced items by default.
  146. if ( false === get_user_option( 'managenav-menuscolumnshidden' ) ) {
  147. $user = wp_get_current_user();
  148. update_user_meta(
  149. $user->ID,
  150. 'managenav-menuscolumnshidden',
  151. array(
  152. 0 => 'link-target',
  153. 1 => 'css-classes',
  154. 2 => 'xfn',
  155. 3 => 'description',
  156. 4 => 'title-attribute',
  157. )
  158. );
  159. }
  160. }
  161. /**
  162. * Limit the amount of meta boxes to pages, posts, links, and categories for first time users.
  163. *
  164. * @since 3.0.0
  165. *
  166. * @global array $wp_meta_boxes
  167. */
  168. function wp_initial_nav_menu_meta_boxes() {
  169. global $wp_meta_boxes;
  170. if ( get_user_option( 'metaboxhidden_nav-menus' ) !== false || ! is_array( $wp_meta_boxes ) ) {
  171. return;
  172. }
  173. $initial_meta_boxes = array( 'add-post-type-page', 'add-post-type-post', 'add-custom-links', 'add-category' );
  174. $hidden_meta_boxes = array();
  175. foreach ( array_keys( $wp_meta_boxes['nav-menus'] ) as $context ) {
  176. foreach ( array_keys( $wp_meta_boxes['nav-menus'][ $context ] ) as $priority ) {
  177. foreach ( $wp_meta_boxes['nav-menus'][ $context ][ $priority ] as $box ) {
  178. if ( in_array( $box['id'], $initial_meta_boxes, true ) ) {
  179. unset( $box['id'] );
  180. } else {
  181. $hidden_meta_boxes[] = $box['id'];
  182. }
  183. }
  184. }
  185. }
  186. $user = wp_get_current_user();
  187. update_user_meta( $user->ID, 'metaboxhidden_nav-menus', $hidden_meta_boxes );
  188. }
  189. /**
  190. * Creates meta boxes for any post type menu item..
  191. *
  192. * @since 3.0.0
  193. */
  194. function wp_nav_menu_post_type_meta_boxes() {
  195. $post_types = get_post_types( array( 'show_in_nav_menus' => true ), 'object' );
  196. if ( ! $post_types ) {
  197. return;
  198. }
  199. foreach ( $post_types as $post_type ) {
  200. /**
  201. * Filters whether a menu items meta box will be added for the current
  202. * object type.
  203. *
  204. * If a falsey value is returned instead of an object, the menu items
  205. * meta box for the current meta box object will not be added.
  206. *
  207. * @since 3.0.0
  208. *
  209. * @param WP_Post_Type|false $post_type The current object to add a menu items
  210. * meta box for.
  211. */
  212. $post_type = apply_filters( 'nav_menu_meta_box_object', $post_type );
  213. if ( $post_type ) {
  214. $id = $post_type->name;
  215. // Give pages a higher priority.
  216. $priority = ( 'page' === $post_type->name ? 'core' : 'default' );
  217. add_meta_box( "add-post-type-{$id}", $post_type->labels->name, 'wp_nav_menu_item_post_type_meta_box', 'nav-menus', 'side', $priority, $post_type );
  218. }
  219. }
  220. }
  221. /**
  222. * Creates meta boxes for any taxonomy menu item.
  223. *
  224. * @since 3.0.0
  225. */
  226. function wp_nav_menu_taxonomy_meta_boxes() {
  227. $taxonomies = get_taxonomies( array( 'show_in_nav_menus' => true ), 'object' );
  228. if ( ! $taxonomies ) {
  229. return;
  230. }
  231. foreach ( $taxonomies as $tax ) {
  232. /** This filter is documented in wp-admin/includes/nav-menu.php */
  233. $tax = apply_filters( 'nav_menu_meta_box_object', $tax );
  234. if ( $tax ) {
  235. $id = $tax->name;
  236. add_meta_box( "add-{$id}", $tax->labels->name, 'wp_nav_menu_item_taxonomy_meta_box', 'nav-menus', 'side', 'default', $tax );
  237. }
  238. }
  239. }
  240. /**
  241. * Check whether to disable the Menu Locations meta box submit button and inputs.
  242. *
  243. * @since 3.6.0
  244. * @since 5.3.1 The `$echo` parameter was added.
  245. *
  246. * @global bool $one_theme_location_no_menus to determine if no menus exist
  247. *
  248. * @param int|string $nav_menu_selected_id ID, name, or slug of the currently selected menu.
  249. * @param bool $echo Whether to echo or just return the string.
  250. * @return string|false Disabled attribute if at least one menu exists, false if not.
  251. */
  252. function wp_nav_menu_disabled_check( $nav_menu_selected_id, $echo = true ) {
  253. global $one_theme_location_no_menus;
  254. if ( $one_theme_location_no_menus ) {
  255. return false;
  256. }
  257. return disabled( $nav_menu_selected_id, 0, $echo );
  258. }
  259. /**
  260. * Displays a meta box for the custom links menu item.
  261. *
  262. * @since 3.0.0
  263. *
  264. * @global int $_nav_menu_placeholder
  265. * @global int|string $nav_menu_selected_id
  266. */
  267. function wp_nav_menu_item_link_meta_box() {
  268. global $_nav_menu_placeholder, $nav_menu_selected_id;
  269. $_nav_menu_placeholder = 0 > $_nav_menu_placeholder ? $_nav_menu_placeholder - 1 : -1;
  270. ?>
  271. <div class="customlinkdiv" id="customlinkdiv">
  272. <input type="hidden" value="custom" name="menu-item[<?php echo $_nav_menu_placeholder; ?>][menu-item-type]" />
  273. <p id="menu-item-url-wrap" class="wp-clearfix">
  274. <label class="howto" for="custom-menu-item-url"><?php _e( 'URL' ); ?></label>
  275. <input id="custom-menu-item-url" name="menu-item[<?php echo $_nav_menu_placeholder; ?>][menu-item-url]" type="text"<?php wp_nav_menu_disabled_check( $nav_menu_selected_id ); ?> class="code menu-item-textbox form-required" placeholder="https://" />
  276. </p>
  277. <p id="menu-item-name-wrap" class="wp-clearfix">
  278. <label class="howto" for="custom-menu-item-name"><?php _e( 'Link Text' ); ?></label>
  279. <input id="custom-menu-item-name" name="menu-item[<?php echo $_nav_menu_placeholder; ?>][menu-item-title]" type="text"<?php wp_nav_menu_disabled_check( $nav_menu_selected_id ); ?> class="regular-text menu-item-textbox" />
  280. </p>
  281. <p class="button-controls wp-clearfix">
  282. <span class="add-to-menu">
  283. <input type="submit"<?php wp_nav_menu_disabled_check( $nav_menu_selected_id ); ?> class="button submit-add-to-menu right" value="<?php esc_attr_e( 'Add to Menu' ); ?>" name="add-custom-menu-item" id="submit-customlinkdiv" />
  284. <span class="spinner"></span>
  285. </span>
  286. </p>
  287. </div><!-- /.customlinkdiv -->
  288. <?php
  289. }
  290. /**
  291. * Displays a meta box for a post type menu item.
  292. *
  293. * @since 3.0.0
  294. *
  295. * @global int $_nav_menu_placeholder
  296. * @global int|string $nav_menu_selected_id
  297. *
  298. * @param string $object Not used.
  299. * @param array $box {
  300. * Post type menu item meta box arguments.
  301. *
  302. * @type string $id Meta box 'id' attribute.
  303. * @type string $title Meta box title.
  304. * @type callable $callback Meta box display callback.
  305. * @type WP_Post_Type $args Extra meta box arguments (the post type object for this meta box).
  306. * }
  307. */
  308. function wp_nav_menu_item_post_type_meta_box( $object, $box ) {
  309. global $_nav_menu_placeholder, $nav_menu_selected_id;
  310. $post_type_name = $box['args']->name;
  311. $post_type = get_post_type_object( $post_type_name );
  312. $tab_name = $post_type_name . '-tab';
  313. // Paginate browsing for large numbers of post objects.
  314. $per_page = 50;
  315. $pagenum = isset( $_REQUEST[ $tab_name ] ) && isset( $_REQUEST['paged'] ) ? absint( $_REQUEST['paged'] ) : 1;
  316. $offset = 0 < $pagenum ? $per_page * ( $pagenum - 1 ) : 0;
  317. $args = array(
  318. 'offset' => $offset,
  319. 'order' => 'ASC',
  320. 'orderby' => 'title',
  321. 'posts_per_page' => $per_page,
  322. 'post_type' => $post_type_name,
  323. 'suppress_filters' => true,
  324. 'update_post_term_cache' => false,
  325. 'update_post_meta_cache' => false,
  326. );
  327. if ( isset( $box['args']->_default_query ) ) {
  328. $args = array_merge( $args, (array) $box['args']->_default_query );
  329. }
  330. /*
  331. * If we're dealing with pages, let's prioritize the Front Page,
  332. * Posts Page and Privacy Policy Page at the top of the list.
  333. */
  334. $important_pages = array();
  335. if ( 'page' === $post_type_name ) {
  336. $suppress_page_ids = array();
  337. // Insert Front Page or custom Home link.
  338. $front_page = 'page' === get_option( 'show_on_front' ) ? (int) get_option( 'page_on_front' ) : 0;
  339. $front_page_obj = null;
  340. if ( ! empty( $front_page ) ) {
  341. $front_page_obj = get_post( $front_page );
  342. $front_page_obj->front_or_home = true;
  343. $important_pages[] = $front_page_obj;
  344. $suppress_page_ids[] = $front_page_obj->ID;
  345. } else {
  346. $_nav_menu_placeholder = ( 0 > $_nav_menu_placeholder ) ? (int) $_nav_menu_placeholder - 1 : -1;
  347. $front_page_obj = (object) array(
  348. 'front_or_home' => true,
  349. 'ID' => 0,
  350. 'object_id' => $_nav_menu_placeholder,
  351. 'post_content' => '',
  352. 'post_excerpt' => '',
  353. 'post_parent' => '',
  354. 'post_title' => _x( 'Home', 'nav menu home label' ),
  355. 'post_type' => 'nav_menu_item',
  356. 'type' => 'custom',
  357. 'url' => home_url( '/' ),
  358. );
  359. $important_pages[] = $front_page_obj;
  360. }
  361. // Insert Posts Page.
  362. $posts_page = 'page' === get_option( 'show_on_front' ) ? (int) get_option( 'page_for_posts' ) : 0;
  363. if ( ! empty( $posts_page ) ) {
  364. $posts_page_obj = get_post( $posts_page );
  365. $posts_page_obj->posts_page = true;
  366. $important_pages[] = $posts_page_obj;
  367. $suppress_page_ids[] = $posts_page_obj->ID;
  368. }
  369. // Insert Privacy Policy Page.
  370. $privacy_policy_page_id = (int) get_option( 'wp_page_for_privacy_policy' );
  371. if ( ! empty( $privacy_policy_page_id ) ) {
  372. $privacy_policy_page = get_post( $privacy_policy_page_id );
  373. if ( $privacy_policy_page instanceof WP_Post && 'publish' === $privacy_policy_page->post_status ) {
  374. $privacy_policy_page->privacy_policy_page = true;
  375. $important_pages[] = $privacy_policy_page;
  376. $suppress_page_ids[] = $privacy_policy_page->ID;
  377. }
  378. }
  379. // Add suppression array to arguments for WP_Query.
  380. if ( ! empty( $suppress_page_ids ) ) {
  381. $args['post__not_in'] = $suppress_page_ids;
  382. }
  383. }
  384. // @todo Transient caching of these results with proper invalidation on updating of a post of this type.
  385. $get_posts = new WP_Query;
  386. $posts = $get_posts->query( $args );
  387. // Only suppress and insert when more than just suppression pages available.
  388. if ( ! $get_posts->post_count ) {
  389. if ( ! empty( $suppress_page_ids ) ) {
  390. unset( $args['post__not_in'] );
  391. $get_posts = new WP_Query;
  392. $posts = $get_posts->query( $args );
  393. } else {
  394. echo '<p>' . __( 'No items.' ) . '</p>';
  395. return;
  396. }
  397. } elseif ( ! empty( $important_pages ) ) {
  398. $posts = array_merge( $important_pages, $posts );
  399. }
  400. $num_pages = $get_posts->max_num_pages;
  401. $page_links = paginate_links(
  402. array(
  403. 'base' => add_query_arg(
  404. array(
  405. $tab_name => 'all',
  406. 'paged' => '%#%',
  407. 'item-type' => 'post_type',
  408. 'item-object' => $post_type_name,
  409. )
  410. ),
  411. 'format' => '',
  412. 'prev_text' => '<span aria-label="' . esc_attr__( 'Previous page' ) . '">' . __( '&laquo;' ) . '</span>',
  413. 'next_text' => '<span aria-label="' . esc_attr__( 'Next page' ) . '">' . __( '&raquo;' ) . '</span>',
  414. 'before_page_number' => '<span class="screen-reader-text">' . __( 'Page' ) . '</span> ',
  415. 'total' => $num_pages,
  416. 'current' => $pagenum,
  417. )
  418. );
  419. $db_fields = false;
  420. if ( is_post_type_hierarchical( $post_type_name ) ) {
  421. $db_fields = array(
  422. 'parent' => 'post_parent',
  423. 'id' => 'ID',
  424. );
  425. }
  426. $walker = new Walker_Nav_Menu_Checklist( $db_fields );
  427. $current_tab = 'most-recent';
  428. if ( isset( $_REQUEST[ $tab_name ] ) && in_array( $_REQUEST[ $tab_name ], array( 'all', 'search' ), true ) ) {
  429. $current_tab = $_REQUEST[ $tab_name ];
  430. }
  431. if ( ! empty( $_REQUEST[ 'quick-search-posttype-' . $post_type_name ] ) ) {
  432. $current_tab = 'search';
  433. }
  434. $removed_args = array(
  435. 'action',
  436. 'customlink-tab',
  437. 'edit-menu-item',
  438. 'menu-item',
  439. 'page-tab',
  440. '_wpnonce',
  441. );
  442. $most_recent_url = '';
  443. $view_all_url = '';
  444. $search_url = '';
  445. if ( $nav_menu_selected_id ) {
  446. $most_recent_url = esc_url( add_query_arg( $tab_name, 'most-recent', remove_query_arg( $removed_args ) ) );
  447. $view_all_url = esc_url( add_query_arg( $tab_name, 'all', remove_query_arg( $removed_args ) ) );
  448. $search_url = esc_url( add_query_arg( $tab_name, 'search', remove_query_arg( $removed_args ) ) );
  449. }
  450. ?>
  451. <div id="posttype-<?php echo $post_type_name; ?>" class="posttypediv">
  452. <ul id="posttype-<?php echo $post_type_name; ?>-tabs" class="posttype-tabs add-menu-item-tabs">
  453. <li <?php echo ( 'most-recent' === $current_tab ? ' class="tabs"' : '' ); ?>>
  454. <a class="nav-tab-link" data-type="tabs-panel-posttype-<?php echo esc_attr( $post_type_name ); ?>-most-recent" href="<?php echo $most_recent_url; ?>#tabs-panel-posttype-<?php echo $post_type_name; ?>-most-recent">
  455. <?php _e( 'Most Recent' ); ?>
  456. </a>
  457. </li>
  458. <li <?php echo ( 'all' === $current_tab ? ' class="tabs"' : '' ); ?>>
  459. <a class="nav-tab-link" data-type="<?php echo esc_attr( $post_type_name ); ?>-all" href="<?php echo $view_all_url; ?>#<?php echo $post_type_name; ?>-all">
  460. <?php _e( 'View All' ); ?>
  461. </a>
  462. </li>
  463. <li <?php echo ( 'search' === $current_tab ? ' class="tabs"' : '' ); ?>>
  464. <a class="nav-tab-link" data-type="tabs-panel-posttype-<?php echo esc_attr( $post_type_name ); ?>-search" href="<?php echo $search_url; ?>#tabs-panel-posttype-<?php echo $post_type_name; ?>-search">
  465. <?php _e( 'Search' ); ?>
  466. </a>
  467. </li>
  468. </ul><!-- .posttype-tabs -->
  469. <div id="tabs-panel-posttype-<?php echo $post_type_name; ?>-most-recent" class="tabs-panel <?php echo ( 'most-recent' === $current_tab ? 'tabs-panel-active' : 'tabs-panel-inactive' ); ?>" role="region" aria-label="<?php _e( 'Most Recent' ); ?>" tabindex="0">
  470. <ul id="<?php echo $post_type_name; ?>checklist-most-recent" class="categorychecklist form-no-clear">
  471. <?php
  472. $recent_args = array_merge(
  473. $args,
  474. array(
  475. 'orderby' => 'post_date',
  476. 'order' => 'DESC',
  477. 'posts_per_page' => 15,
  478. )
  479. );
  480. $most_recent = $get_posts->query( $recent_args );
  481. $args['walker'] = $walker;
  482. /**
  483. * Filters the posts displayed in the 'Most Recent' tab of the current
  484. * post type's menu items meta box.
  485. *
  486. * The dynamic portion of the hook name, `$post_type_name`, refers to the post type name.
  487. *
  488. * Possible hook names include:
  489. *
  490. * - `nav_menu_items_post_recent`
  491. * - `nav_menu_items_page_recent`
  492. *
  493. * @since 4.3.0
  494. * @since 4.9.0 Added the `$recent_args` parameter.
  495. *
  496. * @param WP_Post[] $most_recent An array of post objects being listed.
  497. * @param array $args An array of `WP_Query` arguments for the meta box.
  498. * @param array $box Arguments passed to `wp_nav_menu_item_post_type_meta_box()`.
  499. * @param array $recent_args An array of `WP_Query` arguments for 'Most Recent' tab.
  500. */
  501. $most_recent = apply_filters( "nav_menu_items_{$post_type_name}_recent", $most_recent, $args, $box, $recent_args );
  502. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', $most_recent ), 0, (object) $args );
  503. ?>
  504. </ul>
  505. </div><!-- /.tabs-panel -->
  506. <div class="tabs-panel <?php echo ( 'search' === $current_tab ? 'tabs-panel-active' : 'tabs-panel-inactive' ); ?>" id="tabs-panel-posttype-<?php echo $post_type_name; ?>-search" role="region" aria-label="<?php echo $post_type->labels->search_items; ?>" tabindex="0">
  507. <?php
  508. if ( isset( $_REQUEST[ 'quick-search-posttype-' . $post_type_name ] ) ) {
  509. $searched = esc_attr( $_REQUEST[ 'quick-search-posttype-' . $post_type_name ] );
  510. $search_results = get_posts(
  511. array(
  512. 's' => $searched,
  513. 'post_type' => $post_type_name,
  514. 'fields' => 'all',
  515. 'order' => 'DESC',
  516. )
  517. );
  518. } else {
  519. $searched = '';
  520. $search_results = array();
  521. }
  522. ?>
  523. <p class="quick-search-wrap">
  524. <label for="quick-search-posttype-<?php echo $post_type_name; ?>" class="screen-reader-text"><?php _e( 'Search' ); ?></label>
  525. <input type="search"<?php wp_nav_menu_disabled_check( $nav_menu_selected_id ); ?> class="quick-search" value="<?php echo $searched; ?>" name="quick-search-posttype-<?php echo $post_type_name; ?>" id="quick-search-posttype-<?php echo $post_type_name; ?>" />
  526. <span class="spinner"></span>
  527. <?php submit_button( __( 'Search' ), 'small quick-search-submit hide-if-js', 'submit', false, array( 'id' => 'submit-quick-search-posttype-' . $post_type_name ) ); ?>
  528. </p>
  529. <ul id="<?php echo $post_type_name; ?>-search-checklist" data-wp-lists="list:<?php echo $post_type_name; ?>" class="categorychecklist form-no-clear">
  530. <?php if ( ! empty( $search_results ) && ! is_wp_error( $search_results ) ) : ?>
  531. <?php
  532. $args['walker'] = $walker;
  533. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', $search_results ), 0, (object) $args );
  534. ?>
  535. <?php elseif ( is_wp_error( $search_results ) ) : ?>
  536. <li><?php echo $search_results->get_error_message(); ?></li>
  537. <?php elseif ( ! empty( $searched ) ) : ?>
  538. <li><?php _e( 'No results found.' ); ?></li>
  539. <?php endif; ?>
  540. </ul>
  541. </div><!-- /.tabs-panel -->
  542. <div id="<?php echo $post_type_name; ?>-all" class="tabs-panel tabs-panel-view-all <?php echo ( 'all' === $current_tab ? 'tabs-panel-active' : 'tabs-panel-inactive' ); ?>" role="region" aria-label="<?php echo $post_type->labels->all_items; ?>" tabindex="0">
  543. <?php if ( ! empty( $page_links ) ) : ?>
  544. <div class="add-menu-item-pagelinks">
  545. <?php echo $page_links; ?>
  546. </div>
  547. <?php endif; ?>
  548. <ul id="<?php echo $post_type_name; ?>checklist" data-wp-lists="list:<?php echo $post_type_name; ?>" class="categorychecklist form-no-clear">
  549. <?php
  550. $args['walker'] = $walker;
  551. if ( $post_type->has_archive ) {
  552. $_nav_menu_placeholder = ( 0 > $_nav_menu_placeholder ) ? (int) $_nav_menu_placeholder - 1 : -1;
  553. array_unshift(
  554. $posts,
  555. (object) array(
  556. 'ID' => 0,
  557. 'object_id' => $_nav_menu_placeholder,
  558. 'object' => $post_type_name,
  559. 'post_content' => '',
  560. 'post_excerpt' => '',
  561. 'post_title' => $post_type->labels->archives,
  562. 'post_type' => 'nav_menu_item',
  563. 'type' => 'post_type_archive',
  564. 'url' => get_post_type_archive_link( $post_type_name ),
  565. )
  566. );
  567. }
  568. /**
  569. * Filters the posts displayed in the 'View All' tab of the current
  570. * post type's menu items meta box.
  571. *
  572. * The dynamic portion of the hook name, `$post_type_name`, refers
  573. * to the slug of the current post type.
  574. *
  575. * Possible hook names include:
  576. *
  577. * - `nav_menu_items_post`
  578. * - `nav_menu_items_page`
  579. *
  580. * @since 3.2.0
  581. * @since 4.6.0 Converted the `$post_type` parameter to accept a WP_Post_Type object.
  582. *
  583. * @see WP_Query::query()
  584. *
  585. * @param object[] $posts The posts for the current post type. Mostly `WP_Post` objects, but
  586. * can also contain "fake" post objects to represent other menu items.
  587. * @param array $args An array of `WP_Query` arguments.
  588. * @param WP_Post_Type $post_type The current post type object for this menu item meta box.
  589. */
  590. $posts = apply_filters( "nav_menu_items_{$post_type_name}", $posts, $args, $post_type );
  591. $checkbox_items = walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', $posts ), 0, (object) $args );
  592. echo $checkbox_items;
  593. ?>
  594. </ul>
  595. <?php if ( ! empty( $page_links ) ) : ?>
  596. <div class="add-menu-item-pagelinks">
  597. <?php echo $page_links; ?>
  598. </div>
  599. <?php endif; ?>
  600. </div><!-- /.tabs-panel -->
  601. <p class="button-controls wp-clearfix" data-items-type="posttype-<?php echo esc_attr( $post_type_name ); ?>">
  602. <span class="list-controls hide-if-no-js">
  603. <input type="checkbox"<?php wp_nav_menu_disabled_check( $nav_menu_selected_id ); ?> id="<?php echo esc_attr( $tab_name ); ?>" class="select-all" />
  604. <label for="<?php echo esc_attr( $tab_name ); ?>"><?php _e( 'Select All' ); ?></label>
  605. </span>
  606. <span class="add-to-menu">
  607. <input type="submit"<?php wp_nav_menu_disabled_check( $nav_menu_selected_id ); ?> class="button submit-add-to-menu right" value="<?php esc_attr_e( 'Add to Menu' ); ?>" name="add-post-type-menu-item" id="<?php echo esc_attr( 'submit-posttype-' . $post_type_name ); ?>" />
  608. <span class="spinner"></span>
  609. </span>
  610. </p>
  611. </div><!-- /.posttypediv -->
  612. <?php
  613. }
  614. /**
  615. * Displays a meta box for a taxonomy menu item.
  616. *
  617. * @since 3.0.0
  618. *
  619. * @global int|string $nav_menu_selected_id
  620. *
  621. * @param string $object Not used.
  622. * @param array $box {
  623. * Taxonomy menu item meta box arguments.
  624. *
  625. * @type string $id Meta box 'id' attribute.
  626. * @type string $title Meta box title.
  627. * @type callable $callback Meta box display callback.
  628. * @type object $args Extra meta box arguments (the taxonomy object for this meta box).
  629. * }
  630. */
  631. function wp_nav_menu_item_taxonomy_meta_box( $object, $box ) {
  632. global $nav_menu_selected_id;
  633. $taxonomy_name = $box['args']->name;
  634. $taxonomy = get_taxonomy( $taxonomy_name );
  635. $tab_name = $taxonomy_name . '-tab';
  636. // Paginate browsing for large numbers of objects.
  637. $per_page = 50;
  638. $pagenum = isset( $_REQUEST[ $tab_name ] ) && isset( $_REQUEST['paged'] ) ? absint( $_REQUEST['paged'] ) : 1;
  639. $offset = 0 < $pagenum ? $per_page * ( $pagenum - 1 ) : 0;
  640. $args = array(
  641. 'taxonomy' => $taxonomy_name,
  642. 'child_of' => 0,
  643. 'exclude' => '',
  644. 'hide_empty' => false,
  645. 'hierarchical' => 1,
  646. 'include' => '',
  647. 'number' => $per_page,
  648. 'offset' => $offset,
  649. 'order' => 'ASC',
  650. 'orderby' => 'name',
  651. 'pad_counts' => false,
  652. );
  653. $terms = get_terms( $args );
  654. if ( ! $terms || is_wp_error( $terms ) ) {
  655. echo '<p>' . __( 'No items.' ) . '</p>';
  656. return;
  657. }
  658. $num_pages = ceil(
  659. wp_count_terms(
  660. array_merge(
  661. $args,
  662. array(
  663. 'number' => '',
  664. 'offset' => '',
  665. )
  666. )
  667. ) / $per_page
  668. );
  669. $page_links = paginate_links(
  670. array(
  671. 'base' => add_query_arg(
  672. array(
  673. $tab_name => 'all',
  674. 'paged' => '%#%',
  675. 'item-type' => 'taxonomy',
  676. 'item-object' => $taxonomy_name,
  677. )
  678. ),
  679. 'format' => '',
  680. 'prev_text' => '<span aria-label="' . esc_attr__( 'Previous page' ) . '">' . __( '&laquo;' ) . '</span>',
  681. 'next_text' => '<span aria-label="' . esc_attr__( 'Next page' ) . '">' . __( '&raquo;' ) . '</span>',
  682. 'before_page_number' => '<span class="screen-reader-text">' . __( 'Page' ) . '</span> ',
  683. 'total' => $num_pages,
  684. 'current' => $pagenum,
  685. )
  686. );
  687. $db_fields = false;
  688. if ( is_taxonomy_hierarchical( $taxonomy_name ) ) {
  689. $db_fields = array(
  690. 'parent' => 'parent',
  691. 'id' => 'term_id',
  692. );
  693. }
  694. $walker = new Walker_Nav_Menu_Checklist( $db_fields );
  695. $current_tab = 'most-used';
  696. if ( isset( $_REQUEST[ $tab_name ] ) && in_array( $_REQUEST[ $tab_name ], array( 'all', 'most-used', 'search' ), true ) ) {
  697. $current_tab = $_REQUEST[ $tab_name ];
  698. }
  699. if ( ! empty( $_REQUEST[ 'quick-search-taxonomy-' . $taxonomy_name ] ) ) {
  700. $current_tab = 'search';
  701. }
  702. $removed_args = array(
  703. 'action',
  704. 'customlink-tab',
  705. 'edit-menu-item',
  706. 'menu-item',
  707. 'page-tab',
  708. '_wpnonce',
  709. );
  710. $most_used_url = '';
  711. $view_all_url = '';
  712. $search_url = '';
  713. if ( $nav_menu_selected_id ) {
  714. $most_used_url = esc_url( add_query_arg( $tab_name, 'most-used', remove_query_arg( $removed_args ) ) );
  715. $view_all_url = esc_url( add_query_arg( $tab_name, 'all', remove_query_arg( $removed_args ) ) );
  716. $search_url = esc_url( add_query_arg( $tab_name, 'search', remove_query_arg( $removed_args ) ) );
  717. }
  718. ?>
  719. <div id="taxonomy-<?php echo $taxonomy_name; ?>" class="taxonomydiv">
  720. <ul id="taxonomy-<?php echo $taxonomy_name; ?>-tabs" class="taxonomy-tabs add-menu-item-tabs">
  721. <li <?php echo ( 'most-used' === $current_tab ? ' class="tabs"' : '' ); ?>>
  722. <a class="nav-tab-link" data-type="tabs-panel-<?php echo esc_attr( $taxonomy_name ); ?>-pop" href="<?php echo $most_used_url; ?>#tabs-panel-<?php echo $taxonomy_name; ?>-pop">
  723. <?php echo esc_html( $taxonomy->labels->most_used ); ?>
  724. </a>
  725. </li>
  726. <li <?php echo ( 'all' === $current_tab ? ' class="tabs"' : '' ); ?>>
  727. <a class="nav-tab-link" data-type="tabs-panel-<?php echo esc_attr( $taxonomy_name ); ?>-all" href="<?php echo $view_all_url; ?>#tabs-panel-<?php echo $taxonomy_name; ?>-all">
  728. <?php _e( 'View All' ); ?>
  729. </a>
  730. </li>
  731. <li <?php echo ( 'search' === $current_tab ? ' class="tabs"' : '' ); ?>>
  732. <a class="nav-tab-link" data-type="tabs-panel-search-taxonomy-<?php echo esc_attr( $taxonomy_name ); ?>" href="<?php echo $search_url; ?>#tabs-panel-search-taxonomy-<?php echo $taxonomy_name; ?>">
  733. <?php _e( 'Search' ); ?>
  734. </a>
  735. </li>
  736. </ul><!-- .taxonomy-tabs -->
  737. <div id="tabs-panel-<?php echo $taxonomy_name; ?>-pop" class="tabs-panel <?php echo ( 'most-used' === $current_tab ? 'tabs-panel-active' : 'tabs-panel-inactive' ); ?>" role="region" aria-label="<?php echo $taxonomy->labels->most_used; ?>" tabindex="0">
  738. <ul id="<?php echo $taxonomy_name; ?>checklist-pop" class="categorychecklist form-no-clear" >
  739. <?php
  740. $popular_terms = get_terms(
  741. array(
  742. 'taxonomy' => $taxonomy_name,
  743. 'orderby' => 'count',
  744. 'order' => 'DESC',
  745. 'number' => 10,
  746. 'hierarchical' => false,
  747. )
  748. );
  749. $args['walker'] = $walker;
  750. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', $popular_terms ), 0, (object) $args );
  751. ?>
  752. </ul>
  753. </div><!-- /.tabs-panel -->
  754. <div id="tabs-panel-<?php echo $taxonomy_name; ?>-all" class="tabs-panel tabs-panel-view-all <?php echo ( 'all' === $current_tab ? 'tabs-panel-active' : 'tabs-panel-inactive' ); ?>" role="region" aria-label="<?php echo $taxonomy->labels->all_items; ?>" tabindex="0">
  755. <?php if ( ! empty( $page_links ) ) : ?>
  756. <div class="add-menu-item-pagelinks">
  757. <?php echo $page_links; ?>
  758. </div>
  759. <?php endif; ?>
  760. <ul id="<?php echo $taxonomy_name; ?>checklist" data-wp-lists="list:<?php echo $taxonomy_name; ?>" class="categorychecklist form-no-clear">
  761. <?php
  762. $args['walker'] = $walker;
  763. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', $terms ), 0, (object) $args );
  764. ?>
  765. </ul>
  766. <?php if ( ! empty( $page_links ) ) : ?>
  767. <div class="add-menu-item-pagelinks">
  768. <?php echo $page_links; ?>
  769. </div>
  770. <?php endif; ?>
  771. </div><!-- /.tabs-panel -->
  772. <div class="tabs-panel <?php echo ( 'search' === $current_tab ? 'tabs-panel-active' : 'tabs-panel-inactive' ); ?>" id="tabs-panel-search-taxonomy-<?php echo $taxonomy_name; ?>" role="region" aria-label="<?php echo $taxonomy->labels->search_items; ?>" tabindex="0">
  773. <?php
  774. if ( isset( $_REQUEST[ 'quick-search-taxonomy-' . $taxonomy_name ] ) ) {
  775. $searched = esc_attr( $_REQUEST[ 'quick-search-taxonomy-' . $taxonomy_name ] );
  776. $search_results = get_terms(
  777. array(
  778. 'taxonomy' => $taxonomy_name,
  779. 'name__like' => $searched,
  780. 'fields' => 'all',
  781. 'orderby' => 'count',
  782. 'order' => 'DESC',
  783. 'hierarchical' => false,
  784. )
  785. );
  786. } else {
  787. $searched = '';
  788. $search_results = array();
  789. }
  790. ?>
  791. <p class="quick-search-wrap">
  792. <label for="quick-search-taxonomy-<?php echo $taxonomy_name; ?>" class="screen-reader-text"><?php _e( 'Search' ); ?></label>
  793. <input type="search" class="quick-search" value="<?php echo $searched; ?>" name="quick-search-taxonomy-<?php echo $taxonomy_name; ?>" id="quick-search-taxonomy-<?php echo $taxonomy_name; ?>" />
  794. <span class="spinner"></span>
  795. <?php submit_button( __( 'Search' ), 'small quick-search-submit hide-if-js', 'submit', false, array( 'id' => 'submit-quick-search-taxonomy-' . $taxonomy_name ) ); ?>
  796. </p>
  797. <ul id="<?php echo $taxonomy_name; ?>-search-checklist" data-wp-lists="list:<?php echo $taxonomy_name; ?>" class="categorychecklist form-no-clear">
  798. <?php if ( ! empty( $search_results ) && ! is_wp_error( $search_results ) ) : ?>
  799. <?php
  800. $args['walker'] = $walker;
  801. echo walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', $search_results ), 0, (object) $args );
  802. ?>
  803. <?php elseif ( is_wp_error( $search_results ) ) : ?>
  804. <li><?php echo $search_results->get_error_message(); ?></li>
  805. <?php elseif ( ! empty( $searched ) ) : ?>
  806. <li><?php _e( 'No results found.' ); ?></li>
  807. <?php endif; ?>
  808. </ul>
  809. </div><!-- /.tabs-panel -->
  810. <p class="button-controls wp-clearfix" data-items-type="taxonomy-<?php echo esc_attr( $taxonomy_name ); ?>">
  811. <span class="list-controls hide-if-no-js">
  812. <input type="checkbox"<?php wp_nav_menu_disabled_check( $nav_menu_selected_id ); ?> id="<?php echo esc_attr( $tab_name ); ?>" class="select-all" />
  813. <label for="<?php echo esc_attr( $tab_name ); ?>"><?php _e( 'Select All' ); ?></label>
  814. </span>
  815. <span class="add-to-menu">
  816. <input type="submit"<?php wp_nav_menu_disabled_check( $nav_menu_selected_id ); ?> class="button submit-add-to-menu right" value="<?php esc_attr_e( 'Add to Menu' ); ?>" name="add-taxonomy-menu-item" id="<?php echo esc_attr( 'submit-taxonomy-' . $taxonomy_name ); ?>" />
  817. <span class="spinner"></span>
  818. </span>
  819. </p>
  820. </div><!-- /.taxonomydiv -->
  821. <?php
  822. }
  823. /**
  824. * Save posted nav menu item data.
  825. *
  826. * @since 3.0.0
  827. *
  828. * @param int $menu_id The menu ID for which to save this item. Value of 0 makes a draft, orphaned menu item. Default 0.
  829. * @param array[] $menu_data The unsanitized POSTed menu item data.
  830. * @return int[] The database IDs of the items saved
  831. */
  832. function wp_save_nav_menu_items( $menu_id = 0, $menu_data = array() ) {
  833. $menu_id = (int) $menu_id;
  834. $items_saved = array();
  835. if ( 0 == $menu_id || is_nav_menu( $menu_id ) ) {
  836. // Loop through all the menu items' POST values.
  837. foreach ( (array) $menu_data as $_possible_db_id => $_item_object_data ) {
  838. if (
  839. // Checkbox is not checked.
  840. empty( $_item_object_data['menu-item-object-id'] ) &&
  841. (
  842. // And item type either isn't set.
  843. ! isset( $_item_object_data['menu-item-type'] ) ||
  844. // Or URL is the default.
  845. in_array( $_item_object_data['menu-item-url'], array( 'https://', 'http://', '' ), true ) ||
  846. // Or it's not a custom menu item (but not the custom home page).
  847. ! ( 'custom' === $_item_object_data['menu-item-type'] && ! isset( $_item_object_data['menu-item-db-id'] ) ) ||
  848. // Or it *is* a custom menu item that already exists.
  849. ! empty( $_item_object_data['menu-item-db-id'] )
  850. )
  851. ) {
  852. // Then this potential menu item is not getting added to this menu.
  853. continue;
  854. }
  855. // If this possible menu item doesn't actually have a menu database ID yet.
  856. if (
  857. empty( $_item_object_data['menu-item-db-id'] ) ||
  858. ( 0 > $_possible_db_id ) ||
  859. $_possible_db_id != $_item_object_data['menu-item-db-id']
  860. ) {
  861. $_actual_db_id = 0;
  862. } else {
  863. $_actual_db_id = (int) $_item_object_data['menu-item-db-id'];
  864. }
  865. $args = array(
  866. 'menu-item-db-id' => ( isset( $_item_object_data['menu-item-db-id'] ) ? $_item_object_data['menu-item-db-id'] : '' ),
  867. 'menu-item-object-id' => ( isset( $_item_object_data['menu-item-object-id'] ) ? $_item_object_data['menu-item-object-id'] : '' ),
  868. 'menu-item-object' => ( isset( $_item_object_data['menu-item-object'] ) ? $_item_object_data['menu-item-object'] : '' ),
  869. 'menu-item-parent-id' => ( isset( $_item_object_data['menu-item-parent-id'] ) ? $_item_object_data['menu-item-parent-id'] : '' ),
  870. 'menu-item-position' => ( isset( $_item_object_data['menu-item-position'] ) ? $_item_object_data['menu-item-position'] : '' ),
  871. 'menu-item-type' => ( isset( $_item_object_data['menu-item-type'] ) ? $_item_object_data['menu-item-type'] : '' ),
  872. 'menu-item-title' => ( isset( $_item_object_data['menu-item-title'] ) ? $_item_object_data['menu-item-title'] : '' ),
  873. 'menu-item-url' => ( isset( $_item_object_data['menu-item-url'] ) ? $_item_object_data['menu-item-url'] : '' ),
  874. 'menu-item-description' => ( isset( $_item_object_data['menu-item-description'] ) ? $_item_object_data['menu-item-description'] : '' ),
  875. 'menu-item-attr-title' => ( isset( $_item_object_data['menu-item-attr-title'] ) ? $_item_object_data['menu-item-attr-title'] : '' ),
  876. 'menu-item-target' => ( isset( $_item_object_data['menu-item-target'] ) ? $_item_object_data['menu-item-target'] : '' ),
  877. 'menu-item-classes' => ( isset( $_item_object_data['menu-item-classes'] ) ? $_item_object_data['menu-item-classes'] : '' ),
  878. 'menu-item-xfn' => ( isset( $_item_object_data['menu-item-xfn'] ) ? $_item_object_data['menu-item-xfn'] : '' ),
  879. );
  880. $items_saved[] = wp_update_nav_menu_item( $menu_id, $_actual_db_id, $args );
  881. }
  882. }
  883. return $items_saved;
  884. }
  885. /**
  886. * Adds custom arguments to some of the meta box object types.
  887. *
  888. * @since 3.0.0
  889. *
  890. * @access private
  891. *
  892. * @param object $object The post type or taxonomy meta-object.
  893. * @return object The post type or taxonomy object.
  894. */
  895. function _wp_nav_menu_meta_box_object( $object = null ) {
  896. if ( isset( $object->name ) ) {
  897. if ( 'page' === $object->name ) {
  898. $object->_default_query = array(
  899. 'orderby' => 'menu_order title',
  900. 'post_status' => 'publish',
  901. );
  902. // Posts should show only published items.
  903. } elseif ( 'post' === $object->name ) {
  904. $object->_default_query = array(
  905. 'post_status' => 'publish',
  906. );
  907. // Categories should be in reverse chronological order.
  908. } elseif ( 'category' === $object->name ) {
  909. $object->_default_query = array(
  910. 'orderby' => 'id',
  911. 'order' => 'DESC',
  912. );
  913. // Custom post types should show only published items.
  914. } else {
  915. $object->_default_query = array(
  916. 'post_status' => 'publish',
  917. );
  918. }
  919. }
  920. return $object;
  921. }
  922. /**
  923. * Returns the menu formatted to edit.
  924. *
  925. * @since 3.0.0
  926. *
  927. * @param int $menu_id Optional. The ID of the menu to format. Default 0.
  928. * @return string|WP_Error The menu formatted to edit or error object on failure.
  929. */
  930. function wp_get_nav_menu_to_edit( $menu_id = 0 ) {
  931. $menu = wp_get_nav_menu_object( $menu_id );
  932. // If the menu exists, get its items.
  933. if ( is_nav_menu( $menu ) ) {
  934. $menu_items = wp_get_nav_menu_items( $menu->term_id, array( 'post_status' => 'any' ) );
  935. $result = '<div id="menu-instructions" class="post-body-plain';
  936. $result .= ( ! empty( $menu_items ) ) ? ' menu-instructions-inactive">' : '">';
  937. $result .= '<p>' . __( 'Add menu items from the column on the left.' ) . '</p>';
  938. $result .= '</div>';
  939. if ( empty( $menu_items ) ) {
  940. return $result . ' <ul class="menu" id="menu-to-edit"> </ul>';
  941. }
  942. /**
  943. * Filters the Walker class used when adding nav menu items.
  944. *
  945. * @since 3.0.0
  946. *
  947. * @param string $class The walker class to use. Default 'Walker_Nav_Menu_Edit'.
  948. * @param int $menu_id ID of the menu being rendered.
  949. */
  950. $walker_class_name = apply_filters( 'wp_edit_nav_menu_walker', 'Walker_Nav_Menu_Edit', $menu_id );
  951. if ( class_exists( $walker_class_name ) ) {
  952. $walker = new $walker_class_name;
  953. } else {
  954. return new WP_Error(
  955. 'menu_walker_not_exist',
  956. sprintf(
  957. /* translators: %s: Walker class name. */
  958. __( 'The Walker class named %s does not exist.' ),
  959. '<strong>' . $walker_class_name . '</strong>'
  960. )
  961. );
  962. }
  963. $some_pending_menu_items = false;
  964. $some_invalid_menu_items = false;
  965. foreach ( (array) $menu_items as $menu_item ) {
  966. if ( isset( $menu_item->post_status ) && 'draft' === $menu_item->post_status ) {
  967. $some_pending_menu_items = true;
  968. }
  969. if ( ! empty( $menu_item->_invalid ) ) {
  970. $some_invalid_menu_items = true;
  971. }
  972. }
  973. if ( $some_pending_menu_items ) {
  974. $result .= '<div class="notice notice-info notice-alt inline"><p>' . __( 'Click Save Menu to make pending menu items public.' ) . '</p></div>';
  975. }
  976. if ( $some_invalid_menu_items ) {
  977. $result .= '<div class="notice notice-error notice-alt inline"><p>' . __( 'There are some invalid menu items. Please check or delete them.' ) . '</p></div>';
  978. }
  979. $result .= '<ul class="menu" id="menu-to-edit"> ';
  980. $result .= walk_nav_menu_tree( array_map( 'wp_setup_nav_menu_item', $menu_items ), 0, (object) array( 'walker' => $walker ) );
  981. $result .= ' </ul> ';
  982. return $result;
  983. } elseif ( is_wp_error( $menu ) ) {
  984. return $menu;
  985. }
  986. }
  987. /**
  988. * Returns the columns for the nav menus page.
  989. *
  990. * @since 3.0.0
  991. *
  992. * @return string[] Array of column titles keyed by their column name.
  993. */
  994. function wp_nav_menu_manage_columns() {
  995. return array(
  996. '_title' => __( 'Show advanced menu properties' ),
  997. 'cb' => '<input type="checkbox" />',
  998. 'link-target' => __( 'Link Target' ),
  999. 'title-attribute' => __( 'Title Attribute' ),
  1000. 'css-classes' => __( 'CSS Classes' ),
  1001. 'xfn' => __( 'Link Relationship (XFN)' ),
  1002. 'description' => __( 'Description' ),
  1003. );
  1004. }
  1005. /**
  1006. * Deletes orphaned draft menu items
  1007. *
  1008. * @access private
  1009. * @since 3.0.0
  1010. *
  1011. * @global wpdb $wpdb WordPress database abstraction object.
  1012. */
  1013. function _wp_delete_orphaned_draft_menu_items() {
  1014. global $wpdb;
  1015. $delete_timestamp = time() - ( DAY_IN_SECONDS * EMPTY_TRASH_DAYS );
  1016. // Delete orphaned draft menu items.
  1017. $menu_items_to_delete = $wpdb->get_col( $wpdb->prepare( "SELECT ID FROM $wpdb->posts AS p LEFT JOIN $wpdb->postmeta AS m ON p.ID = m.post_id WHERE post_type = 'nav_menu_item' AND post_status = 'draft' AND meta_key = '_menu_item_orphaned' AND meta_value < %d", $delete_timestamp ) );
  1018. foreach ( (array) $menu_items_to_delete as $menu_item_id ) {
  1019. wp_delete_post( $menu_item_id, true );
  1020. }
  1021. }
  1022. /**
  1023. * Saves nav menu items
  1024. *
  1025. * @since 3.6.0
  1026. *
  1027. * @param int|string $nav_menu_selected_id ID, slug, or name of the currently-selected menu.
  1028. * @param string $nav_menu_selected_title Title of the currently-selected menu.
  1029. * @return array The menu updated message
  1030. */
  1031. function wp_nav_menu_update_menu_items( $nav_menu_selected_id, $nav_menu_selected_title ) {
  1032. $unsorted_menu_items = wp_get_nav_menu_items(
  1033. $nav_menu_selected_id,
  1034. array(
  1035. 'orderby' => 'ID',
  1036. 'output' => ARRAY_A,
  1037. 'output_key' => 'ID',
  1038. 'post_status' => 'draft,publish',
  1039. )
  1040. );
  1041. $messages = array();
  1042. $menu_items = array();
  1043. // Index menu items by DB ID.
  1044. foreach ( $unsorted_menu_items as $_item ) {
  1045. $menu_items[ $_item->db_id ] = $_item;
  1046. }
  1047. $post_fields = array(
  1048. 'menu-item-db-id',
  1049. 'menu-item-object-id',
  1050. 'menu-item-object',
  1051. 'menu-item-parent-id',
  1052. 'menu-item-position',
  1053. 'menu-item-type',
  1054. 'menu-item-title',
  1055. 'menu-item-url',
  1056. 'menu-item-description',
  1057. 'menu-item-attr-title',
  1058. 'menu-item-target',
  1059. 'menu-item-classes',
  1060. 'menu-item-xfn',
  1061. );
  1062. wp_defer_term_counting( true );
  1063. // Loop through all the menu items' POST variables.
  1064. if ( ! empty( $_POST['menu-item-db-id'] ) ) {
  1065. foreach ( (array) $_POST['menu-item-db-id'] as $_key => $k ) {
  1066. // Menu item title can't be blank.
  1067. if ( ! isset( $_POST['menu-item-title'][ $_key ] ) || '' === $_POST['menu-item-title'][ $_key ] ) {
  1068. continue;
  1069. }
  1070. $args = array();
  1071. foreach ( $post_fields as $field ) {
  1072. $args[ $field ] = isset( $_POST[ $field ][ $_key ] ) ? $_POST[ $field ][ $_key ] : '';
  1073. }
  1074. $menu_item_db_id = wp_update_nav_menu_item( $nav_menu_selected_id, ( $_POST['menu-item-db-id'][ $_key ] != $_key ? 0 : $_key ), $args );
  1075. if ( is_wp_error( $menu_item_db_id ) ) {
  1076. $messages[] = '<div id="message" class="error"><p>' . $menu_item_db_id->get_error_message() . '</p></div>';
  1077. } else {
  1078. unset( $menu_items[ $menu_item_db_id ] );
  1079. }
  1080. }
  1081. }
  1082. // Remove menu items from the menu that weren't in $_POST.
  1083. if ( ! empty( $menu_items ) ) {
  1084. foreach ( array_keys( $menu_items ) as $menu_item_id ) {
  1085. if ( is_nav_menu_item( $menu_item_id ) ) {
  1086. wp_delete_post( $menu_item_id );
  1087. }
  1088. }
  1089. }
  1090. // Store 'auto-add' pages.
  1091. $auto_add = ! empty( $_POST['auto-add-pages'] );
  1092. $nav_menu_option = (array) get_option( 'nav_menu_options' );
  1093. if ( ! isset( $nav_menu_option['auto_add'] ) ) {
  1094. $nav_menu_option['auto_add'] = array();
  1095. }
  1096. if ( $auto_add ) {
  1097. if ( ! in_array( $nav_menu_selected_id, $nav_menu_option['auto_add'], true ) ) {
  1098. $nav_menu_option['auto_add'][] = $nav_menu_selected_id;
  1099. }
  1100. } else {
  1101. $key = array_search( $nav_menu_selected_id, $nav_menu_option['auto_add'], true );
  1102. if ( false !== $key ) {
  1103. unset( $nav_menu_option['auto_add'][ $key ] );
  1104. }
  1105. }
  1106. // Remove non-existent/deleted menus.
  1107. $nav_menu_option['auto_add'] = array_intersect( $nav_menu_option['auto_add'], wp_get_nav_menus( array( 'fields' => 'ids' ) ) );
  1108. update_option( 'nav_menu_options', $nav_menu_option );
  1109. wp_defer_term_counting( false );
  1110. /** This action is documented in wp-includes/nav-menu.php */
  1111. do_action( 'wp_update_nav_menu', $nav_menu_selected_id );
  1112. $messages[] = '<div id="message" class="updated notice is-dismissible"><p>' .
  1113. sprintf(
  1114. /* translators: %s: Nav menu title. */
  1115. __( '%s has been updated.' ),
  1116. '<strong>' . $nav_menu_selected_title . '</strong>'
  1117. ) . '</p></div>';
  1118. unset( $menu_items, $unsorted_menu_items );
  1119. return $messages;
  1120. }
  1121. /**
  1122. * If a JSON blob of navigation menu data is in POST data, expand it and inject
  1123. * it into `$_POST` to avoid PHP `max_input_vars` limitations. See #14134.
  1124. *
  1125. * @ignore
  1126. * @since 4.5.3
  1127. * @access private
  1128. */
  1129. function _wp_expand_nav_menu_post_data() {
  1130. if ( ! isset( $_POST['nav-menu-data'] ) ) {
  1131. return;
  1132. }
  1133. $data = json_decode( stripslashes( $_POST['nav-menu-data'] ) );
  1134. if ( ! is_null( $data ) && $data ) {
  1135. foreach ( $data as $post_input_data ) {
  1136. // For input names that are arrays (e.g. `menu-item-db-id[3][4][5]`),
  1137. // derive the array path keys via regex and set the value in $_POST.
  1138. preg_match( '#([^\[]*)(\[(.+)\])?#', $post_input_data->name, $matches );
  1139. $array_bits = array( $matches[1] );
  1140. if ( isset( $matches[3] ) ) {
  1141. $array_bits = array_merge( $array_bits, explode( '][', $matches[3] ) );
  1142. }
  1143. $new_post_data = array();
  1144. // Build the new array value from leaf to trunk.
  1145. for ( $i = count( $array_bits ) - 1; $i >= 0; $i-- ) {
  1146. if ( count( $array_bits ) - 1 == $i ) {
  1147. $new_post_data[ $array_bits[ $i ] ] = wp_slash( $post_input_data->value );
  1148. } else {
  1149. $new_post_data = array( $array_bits[ $i ] => $new_post_data );
  1150. }
  1151. }
  1152. $_POST = array_replace_recursive( $_POST, $new_post_data );
  1153. }
  1154. }
  1155. }