author | ymh <ymh.work@gmail.com> |
Tue, 15 Oct 2019 15:48:13 +0200 | |
changeset 13 | d255fe9cd479 |
parent 9 | 177826044cd9 |
child 16 | a86126ab1dd4 |
permissions | -rw-r--r-- |
0 | 1 |
<?php |
2 |
/** |
|
3 |
* Displays Administration Menu. |
|
4 |
* |
|
5 |
* @package WordPress |
|
6 |
* @subpackage Administration |
|
7 |
*/ |
|
8 |
||
9 |
/** |
|
10 |
* The current page. |
|
11 |
* |
|
12 |
* @global string $self |
|
13 |
*/ |
|
9 | 14 |
$self = preg_replace( '|^.*/wp-admin/network/|i', '', $_SERVER['PHP_SELF'] ); |
15 |
$self = preg_replace( '|^.*/wp-admin/|i', '', $self ); |
|
16 |
$self = preg_replace( '|^.*/plugins/|i', '', $self ); |
|
17 |
$self = preg_replace( '|^.*/mu-plugins/|i', '', $self ); |
|
0 | 18 |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
19 |
/** |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
20 |
* For when admin-header is included from within a function. |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
21 |
* |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
22 |
* @global array $menu |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
23 |
* @global array $submenu |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
24 |
* @global string $parent_file |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
25 |
* @global string $submenu_file |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
26 |
*/ |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
27 |
global $menu, $submenu, $parent_file, $submenu_file; |
5 | 28 |
|
29 |
/** |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
30 |
* Filters the parent file of an admin menu sub-menu item. |
5 | 31 |
* |
32 |
* Allows plugins to move sub-menu items around. |
|
33 |
* |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
34 |
* @since MU (3.0.0) |
5 | 35 |
* |
36 |
* @param string $parent_file The parent file. |
|
37 |
*/ |
|
38 |
$parent_file = apply_filters( 'parent_file', $parent_file ); |
|
0 | 39 |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
40 |
/** |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
41 |
* Filters the file of an admin menu sub-menu item. |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
42 |
* |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
43 |
* @since 4.4.0 |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
44 |
* |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
45 |
* @param string $submenu_file The submenu file. |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
46 |
* @param string $parent_file The submenu item's parent file. |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
47 |
*/ |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
48 |
$submenu_file = apply_filters( 'submenu_file', $submenu_file, $parent_file ); |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
49 |
|
0 | 50 |
get_admin_page_parent(); |
51 |
||
52 |
/** |
|
53 |
* Display menu. |
|
54 |
* |
|
55 |
* @access private |
|
56 |
* @since 2.7.0 |
|
57 |
* |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
58 |
* @global string $self |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
59 |
* @global string $parent_file |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
60 |
* @global string $submenu_file |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
61 |
* @global string $plugin_page |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
62 |
* @global string $typenow |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
63 |
* |
0 | 64 |
* @param array $menu |
65 |
* @param array $submenu |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
66 |
* @param bool $submenu_as_parent |
0 | 67 |
*/ |
68 |
function _wp_menu_output( $menu, $submenu, $submenu_as_parent = true ) { |
|
69 |
global $self, $parent_file, $submenu_file, $plugin_page, $typenow; |
|
70 |
||
71 |
$first = true; |
|
5 | 72 |
// 0 = menu_title, 1 = capability, 2 = menu_slug, 3 = page_title, 4 = classes, 5 = hookname, 6 = icon_url |
0 | 73 |
foreach ( $menu as $key => $item ) { |
74 |
$admin_is_parent = false; |
|
9 | 75 |
$class = array(); |
0 | 76 |
$aria_attributes = ''; |
9 | 77 |
$aria_hidden = ''; |
78 |
$is_separator = false; |
|
0 | 79 |
|
80 |
if ( $first ) { |
|
81 |
$class[] = 'wp-first-item'; |
|
9 | 82 |
$first = false; |
0 | 83 |
} |
84 |
||
5 | 85 |
$submenu_items = array(); |
9 | 86 |
if ( ! empty( $submenu[ $item[2] ] ) ) { |
87 |
$class[] = 'wp-has-submenu'; |
|
88 |
$submenu_items = $submenu[ $item[2] ]; |
|
0 | 89 |
} |
90 |
||
9 | 91 |
if ( ( $parent_file && $item[2] == $parent_file ) || ( empty( $typenow ) && $self == $item[2] ) ) { |
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
92 |
if ( ! empty( $submenu_items ) ) { |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
93 |
$class[] = 'wp-has-current-submenu wp-menu-open'; |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
94 |
} else { |
9 | 95 |
$class[] = 'current'; |
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
96 |
$aria_attributes .= 'aria-current="page"'; |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
97 |
} |
0 | 98 |
} else { |
99 |
$class[] = 'wp-not-current-submenu'; |
|
9 | 100 |
if ( ! empty( $submenu_items ) ) { |
0 | 101 |
$aria_attributes .= 'aria-haspopup="true"'; |
9 | 102 |
} |
0 | 103 |
} |
104 |
||
9 | 105 |
if ( ! empty( $item[4] ) ) { |
5 | 106 |
$class[] = esc_attr( $item[4] ); |
9 | 107 |
} |
0 | 108 |
|
9 | 109 |
$class = $class ? ' class="' . join( ' ', $class ) . '"' : ''; |
110 |
$id = ! empty( $item[5] ) ? ' id="' . preg_replace( '|[^a-zA-Z0-9_:.]|', '-', $item[5] ) . '"' : ''; |
|
111 |
$img = $img_style = ''; |
|
5 | 112 |
$img_class = ' dashicons-before'; |
113 |
||
114 |
if ( false !== strpos( $class, 'wp-menu-separator' ) ) { |
|
115 |
$is_separator = true; |
|
116 |
} |
|
117 |
||
118 |
/* |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
119 |
* If the string 'none' (previously 'div') is passed instead of a URL, don't output |
5 | 120 |
* the default menu image so an icon can be added to div.wp-menu-image as background |
121 |
* with CSS. Dashicons and base64-encoded data:image/svg_xml URIs are also handled |
|
122 |
* as special cases. |
|
123 |
*/ |
|
124 |
if ( ! empty( $item[6] ) ) { |
|
125 |
$img = '<img src="' . $item[6] . '" alt="" />'; |
|
126 |
||
127 |
if ( 'none' === $item[6] || 'div' === $item[6] ) { |
|
128 |
$img = '<br />'; |
|
129 |
} elseif ( 0 === strpos( $item[6], 'data:image/svg+xml;base64,' ) ) { |
|
9 | 130 |
$img = '<br />'; |
5 | 131 |
$img_style = ' style="background-image:url(\'' . esc_attr( $item[6] ) . '\')"'; |
132 |
$img_class = ' svg'; |
|
133 |
} elseif ( 0 === strpos( $item[6], 'dashicons-' ) ) { |
|
9 | 134 |
$img = '<br />'; |
5 | 135 |
$img_class = ' dashicons-before ' . sanitize_html_class( $item[6] ); |
136 |
} |
|
137 |
} |
|
0 | 138 |
$arrow = '<div class="wp-menu-arrow"><div></div></div>'; |
139 |
||
140 |
$title = wptexturize( $item[0] ); |
|
141 |
||
9 | 142 |
// Hide separators from screen readers. |
5 | 143 |
if ( $is_separator ) { |
144 |
$aria_hidden = ' aria-hidden="true"'; |
|
145 |
} |
|
0 | 146 |
|
5 | 147 |
echo "\n\t<li$class$id$aria_hidden>"; |
148 |
||
149 |
if ( $is_separator ) { |
|
0 | 150 |
echo '<div class="separator"></div>'; |
151 |
} elseif ( $submenu_as_parent && ! empty( $submenu_items ) ) { |
|
152 |
$submenu_items = array_values( $submenu_items ); // Re-index. |
|
9 | 153 |
$menu_hook = get_plugin_page_hook( $submenu_items[0][2], $item[2] ); |
154 |
$menu_file = $submenu_items[0][2]; |
|
155 |
if ( false !== ( $pos = strpos( $menu_file, '?' ) ) ) { |
|
0 | 156 |
$menu_file = substr( $menu_file, 0, $pos ); |
9 | 157 |
} |
0 | 158 |
if ( ! empty( $menu_hook ) || ( ( 'index.php' != $submenu_items[0][2] ) && file_exists( WP_PLUGIN_DIR . "/$menu_file" ) && ! file_exists( ABSPATH . "/wp-admin/$menu_file" ) ) ) { |
159 |
$admin_is_parent = true; |
|
5 | 160 |
echo "<a href='admin.php?page={$submenu_items[0][2]}'$class $aria_attributes>$arrow<div class='wp-menu-image$img_class'$img_style>$img</div><div class='wp-menu-name'>$title</div></a>"; |
0 | 161 |
} else { |
5 | 162 |
echo "\n\t<a href='{$submenu_items[0][2]}'$class $aria_attributes>$arrow<div class='wp-menu-image$img_class'$img_style>$img</div><div class='wp-menu-name'>$title</div></a>"; |
0 | 163 |
} |
164 |
} elseif ( ! empty( $item[2] ) && current_user_can( $item[1] ) ) { |
|
165 |
$menu_hook = get_plugin_page_hook( $item[2], 'admin.php' ); |
|
166 |
$menu_file = $item[2]; |
|
9 | 167 |
if ( false !== ( $pos = strpos( $menu_file, '?' ) ) ) { |
0 | 168 |
$menu_file = substr( $menu_file, 0, $pos ); |
9 | 169 |
} |
0 | 170 |
if ( ! empty( $menu_hook ) || ( ( 'index.php' != $item[2] ) && file_exists( WP_PLUGIN_DIR . "/$menu_file" ) && ! file_exists( ABSPATH . "/wp-admin/$menu_file" ) ) ) { |
171 |
$admin_is_parent = true; |
|
5 | 172 |
echo "\n\t<a href='admin.php?page={$item[2]}'$class $aria_attributes>$arrow<div class='wp-menu-image$img_class'$img_style>$img</div><div class='wp-menu-name'>{$item[0]}</div></a>"; |
0 | 173 |
} else { |
5 | 174 |
echo "\n\t<a href='{$item[2]}'$class $aria_attributes>$arrow<div class='wp-menu-image$img_class'$img_style>$img</div><div class='wp-menu-name'>{$item[0]}</div></a>"; |
0 | 175 |
} |
176 |
} |
|
177 |
||
178 |
if ( ! empty( $submenu_items ) ) { |
|
179 |
echo "\n\t<ul class='wp-submenu wp-submenu-wrap'>"; |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
180 |
echo "<li class='wp-submenu-head' aria-hidden='true'>{$item[0]}</li>"; |
0 | 181 |
|
182 |
$first = true; |
|
5 | 183 |
|
184 |
// 0 = menu_title, 1 = capability, 2 = menu_slug, 3 = page_title, 4 = classes |
|
0 | 185 |
foreach ( $submenu_items as $sub_key => $sub_item ) { |
9 | 186 |
if ( ! current_user_can( $sub_item[1] ) ) { |
0 | 187 |
continue; |
9 | 188 |
} |
0 | 189 |
|
9 | 190 |
$class = array(); |
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
191 |
$aria_attributes = ''; |
0 | 192 |
if ( $first ) { |
193 |
$class[] = 'wp-first-item'; |
|
9 | 194 |
$first = false; |
0 | 195 |
} |
196 |
||
197 |
$menu_file = $item[2]; |
|
198 |
||
9 | 199 |
if ( false !== ( $pos = strpos( $menu_file, '?' ) ) ) { |
0 | 200 |
$menu_file = substr( $menu_file, 0, $pos ); |
9 | 201 |
} |
0 | 202 |
|
203 |
// Handle current for post_type=post|page|foo pages, which won't match $self. |
|
204 |
$self_type = ! empty( $typenow ) ? $self . '?post_type=' . $typenow : 'nothing'; |
|
205 |
||
206 |
if ( isset( $submenu_file ) ) { |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
207 |
if ( $submenu_file == $sub_item[2] ) { |
9 | 208 |
$class[] = 'current'; |
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
209 |
$aria_attributes .= ' aria-current="page"'; |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
210 |
} |
9 | 211 |
// If plugin_page is set the parent must either match the current page or not physically exist. |
212 |
// This allows plugin pages with the same hook to exist under different parents. |
|
5 | 213 |
} elseif ( |
0 | 214 |
( ! isset( $plugin_page ) && $self == $sub_item[2] ) || |
9 | 215 |
( isset( $plugin_page ) && $plugin_page == $sub_item[2] && ( $item[2] == $self_type || $item[2] == $self || file_exists( $menu_file ) === false ) ) |
0 | 216 |
) { |
9 | 217 |
$class[] = 'current'; |
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
218 |
$aria_attributes .= ' aria-current="page"'; |
0 | 219 |
} |
220 |
||
5 | 221 |
if ( ! empty( $sub_item[4] ) ) { |
222 |
$class[] = esc_attr( $sub_item[4] ); |
|
223 |
} |
|
224 |
||
0 | 225 |
$class = $class ? ' class="' . join( ' ', $class ) . '"' : ''; |
226 |
||
9 | 227 |
$menu_hook = get_plugin_page_hook( $sub_item[2], $item[2] ); |
228 |
$sub_file = $sub_item[2]; |
|
229 |
if ( false !== ( $pos = strpos( $sub_file, '?' ) ) ) { |
|
230 |
$sub_file = substr( $sub_file, 0, $pos ); |
|
231 |
} |
|
0 | 232 |
|
9 | 233 |
$title = wptexturize( $sub_item[0] ); |
0 | 234 |
|
235 |
if ( ! empty( $menu_hook ) || ( ( 'index.php' != $sub_item[2] ) && file_exists( WP_PLUGIN_DIR . "/$sub_file" ) && ! file_exists( ABSPATH . "/wp-admin/$sub_file" ) ) ) { |
|
236 |
// If admin.php is the current page or if the parent exists as a file in the plugins or admin dir |
|
9 | 237 |
if ( ( ! $admin_is_parent && file_exists( WP_PLUGIN_DIR . "/$menu_file" ) && ! is_dir( WP_PLUGIN_DIR . "/{$item[2]}" ) ) || file_exists( $menu_file ) ) { |
0 | 238 |
$sub_item_url = add_query_arg( array( 'page' => $sub_item[2] ), $item[2] ); |
9 | 239 |
} else { |
0 | 240 |
$sub_item_url = add_query_arg( array( 'page' => $sub_item[2] ), 'admin.php' ); |
9 | 241 |
} |
0 | 242 |
|
243 |
$sub_item_url = esc_url( $sub_item_url ); |
|
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
244 |
echo "<li$class><a href='$sub_item_url'$class$aria_attributes>$title</a></li>"; |
0 | 245 |
} else { |
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
246 |
echo "<li$class><a href='{$sub_item[2]}'$class$aria_attributes>$title</a></li>"; |
0 | 247 |
} |
248 |
} |
|
9 | 249 |
echo '</ul>'; |
0 | 250 |
} |
9 | 251 |
echo '</li>'; |
0 | 252 |
} |
253 |
||
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
254 |
echo '<li id="collapse-menu" class="hide-if-no-js">' . |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
255 |
'<button type="button" id="collapse-button" aria-label="' . esc_attr__( 'Collapse Main menu' ) . '" aria-expanded="true">' . |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
256 |
'<span class="collapse-button-icon" aria-hidden="true"></span>' . |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
257 |
'<span class="collapse-button-label">' . __( 'Collapse menu' ) . '</span>' . |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
258 |
'</button></li>'; |
0 | 259 |
} |
260 |
||
261 |
?> |
|
262 |
||
5 | 263 |
<div id="adminmenumain" role="navigation" aria-label="<?php esc_attr_e( 'Main menu' ); ?>"> |
264 |
<a href="#wpbody-content" class="screen-reader-shortcut"><?php _e( 'Skip to main content' ); ?></a> |
|
265 |
<a href="#wp-toolbar" class="screen-reader-shortcut"><?php _e( 'Skip to toolbar' ); ?></a> |
|
0 | 266 |
<div id="adminmenuback"></div> |
267 |
<div id="adminmenuwrap"> |
|
5 | 268 |
<ul id="adminmenu"> |
0 | 269 |
|
270 |
<?php |
|
271 |
||
272 |
_wp_menu_output( $menu, $submenu ); |
|
5 | 273 |
/** |
274 |
* Fires after the admin menu has been output. |
|
275 |
* |
|
276 |
* @since 2.5.0 |
|
277 |
*/ |
|
0 | 278 |
do_action( 'adminmenu' ); |
279 |
||
280 |
?> |
|
281 |
</ul> |
|
282 |
</div> |
|
5 | 283 |
</div> |