Also available at https://wordpress.org/plugins/super-admin-all-sites-menu/
Use | Prerequisite | Install | Filters | Demo | Behind the scenes | Changelog
For the super admin, replace WP Admin Bar My Sites menu with an All Sites menu.
- Doesn't use
switch_to_blog()
, i.e. Super Admin All Sites Menu is faster and uses less resources than the WP Admin Bar My Sites menu.get_blog_details('blogname')
useswitch_to_blog()
,$site->__get( 'blogname' )
doesn't.get_site_url()
useswitch_to_blog()
,$site->__get( 'siteurl' )
doesn't.
- Subsite menu data are stored locally in IndexedDB (did I say it's fast?). The local storage is updated when;
- the plugin is activated.
- a site is added or deleted.
- you change a blog name.
- IndexedDB is out of sync with site changes.
- Restricted Site Access is activated or deactivated.
- When subsite menu data is updated, REST is used and it's done in increments (100 sites per increment).
- List all subsites. WP Admin Bar My Sites only list sites you're a local admin on.
- Mark sites that has Restricted Site Access activated with a red icon.
- Sites menu is sorted alphabetically.
- Search filter.
- Add more menu choices:
- Under "Network Admin"
- Add New Site
- Per subsite.
- 'New Page'
- 'Users'
- 'Plugins'
- 'Settings'
- Under "Network Admin"
Inspired by WP REST API vs admin-ajax.php vs Must-Use Plugin: Handling AJAX Requests in WordPress, I wrote a must-use plugin that makes the plugin go from fast to faster (ca 60% faster). Save the file in
mu-plugins
.
Menu data are stored locally in IndexedDB.
- WordPress Multisite
- A modern browser, IE 11 isn't supported.
Also available at https://wordpress.org/plugins/super-admin-all-sites-menu/
Also available via composer:
composer require soderlind/super-admin-all-sites-menu
You can use the following filters to override the defaults:
all_sites_menu_order_by
- Sort menu by. Default value is
name
, acceptsid
,url
orname
add_filter( 'all_sites_menu_order_by', function( string $order_by ) : string { return 'url'; } );
- Sort menu by. Default value is
all_sites_menu_load_increments
- REST load increments. Default value is 100.
add_filter( 'all_sites_menu_load_increments', function( int $increments ) : int { return 300; } );
all_sites_menu_plugin_trigger
- Trigger an update of local storage (IndexedDB) when a plugin is (de)activated. Default is
[ 'restricted-site-access/restricted_site_access.php' ]
.Note: Must be an array and each element in the array must point to the main plugin file. Syntax
'plugin-dir/plugin-file.php'
add_filter( 'all_sites_menu_plugin_trigger', function( array $plugins ) : array { return [ 'restricted-site-access/restricted_site_access.php', 'myplugin/myplugin.php', ]; } );
- Trigger an update of local storage (IndexedDB) when a plugin is (de)activated. Default is
all_sites_menu_search_threshold
- Don't display search field if there's less than N subsites. Default value is 20.
add_filter( 'all_sites_menu_search_threshold', function( int $increments ) : int { return 40; } );
all_sites_menu_force_refresh_expiration
- How often a forced refresh should be taken. Default value is
3600
. Set the value to0
to disable forced refresh.add_filter( 'all_sites_menu_force_refresh_expiration', function( int $seconds ) : int { return 3600; } );
- How often a forced refresh should be taken. Default value is
A demo is available in WordPress Playground. It's a bit slow loading, 50 subsites are added.
-
If you disable Super Admin All Sites Menu in the Main Site plugins menu, you'll see the WP Admin Bar My Sites menu doesn't allow you to scroll and see all sites. This is a 14-year-old (!) bug on WordPress.
-
If you activate the Restricted Site Access plugin (included), you'll see a red icon next to the site name. ATM, this only works on the main site due to issues with WordPress Playground.
Let me break down the dataflow related to the timestamp in the super-admin-all-sites-menu plugin:
-
Initial Timestamp Creation:
- In super-admin-all-sites-menu.php, a timestamp is generated when the plugin initializes
- This is exposed via the
get_timestamp()
method of the main plugin class
-
PHP to JavaScript Transfer:
- The timestamp is passed to JavaScript via
wp_add_inline_script()
in super-admin-all-sites-menu.php around line 409:
wp_add_inline_script( 'super-admin-all-sites-menu', "const pluginAllSitesMenu = {$data};", 'before' );
- Where
$data
includes the timestamp:
$data = wp_json_encode([ 'timestamp' => $this->get_timestamp(), // other data... ]);
- The timestamp is passed to JavaScript via
-
JavaScript Storage Check:
- In index.js, the
populateDB()
function compares timestamps:
if ( typeof data !== 'undefined' && typeof data.timestamp !== 'undefined' && pluginAllSitesMenu.timestamp > data.timestamp ) { await db.delete(); }
- In index.js, the
-
Database Population:
- If the IndexedDB is empty or was deleted due to timestamp mismatch:
if ((await db.count()) === 0) { loadSites(db, { offset: 0, delayMs: 200, }); }
-
Timestamp Update Triggers:
- The timestamp is updated when:
- A site is added/deleted from the network
- A plugin that affects the menu is activated/deactivated
- A blog name is changed
- The Restricted Site Access plugin status changes
- The timestamp is updated when:
This creates a caching mechanism where:
- The PHP timestamp acts as a "version" of the site data
- The JS code compares this against the stored timestamp
- Mismatches trigger a refresh of the cached data
- Matches allow using the existing cached data
See CHANGELOG.md
- Dexie.js, which has an Apache License Version 2.0
- Submmenu offset adjustment: https://qiita.com/zephyr7501/items/dd0967fddabd888b28c4
- CSS for search field from https://github.com/trepmal/my-sites-search
Super Admin All Sites Menu is copyright 2021 Per Soderlind
Super Admin All Sites Menu is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version.
Super Admin All Sites Menu is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU Lesser General Public License along with the Extension. If not, see http://www.gnu.org/licenses/.