summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/jetpack/modules/calypsoify/class.jetpack-calypsoify.php')
-rw-r--r--plugins/jetpack/modules/calypsoify/class.jetpack-calypsoify.php462
1 files changed, 462 insertions, 0 deletions
diff --git a/plugins/jetpack/modules/calypsoify/class.jetpack-calypsoify.php b/plugins/jetpack/modules/calypsoify/class.jetpack-calypsoify.php
new file mode 100644
index 00000000..49c44f42
--- /dev/null
+++ b/plugins/jetpack/modules/calypsoify/class.jetpack-calypsoify.php
@@ -0,0 +1,462 @@
+<?php
+/**
+ * This is Calypso skin of the wp-admin interface that is conditionally triggered via the ?calypsoify=1 param.
+ * Ported from an internal Automattic plugin.
+ */
+class Jetpack_Calypsoify {
+
+ /**
+ * Singleton instance of `Jetpack_Calypsoify`.
+ *
+ * @var object
+ */
+ public static $instance = false;
+
+ /**
+ * Is Calypsoify enabled, based on any value of `calypsoify` user meta.
+ *
+ * @var bool
+ */
+ public $is_calypsoify_enabled = false;
+
+ private function __construct() {
+ add_action( 'wp_loaded', array( $this, 'setup' ) );
+ }
+
+ public static function getInstance() {
+ if ( ! self::$instance ) {
+ self::$instance = new self();
+ }
+
+ return self::$instance;
+ }
+
+ public function setup() {
+ $this->is_calypsoify_enabled = 1 == (int) get_user_meta( get_current_user_id(), 'calypsoify', true );
+ add_action( 'admin_init', array( $this, 'check_param' ), 4 );
+
+ if ( $this->is_calypsoify_enabled ) {
+ add_action( 'admin_init', array( $this, 'setup_admin' ), 6 );
+ }
+
+ // Make this always available -- in case calypsoify gets toggled off.
+ add_action( 'wp_ajax_jetpack_toggle_autoupdate', array( $this, 'jetpack_toggle_autoupdate' ) );
+ add_filter( 'handle_bulk_actions-plugins', array( $this, 'handle_bulk_actions_plugins' ), 10, 3 );
+ }
+
+ public function setup_admin() {
+ // Masterbar is currently required for this to work properly. Mock the instance of it
+ if ( ! Jetpack::is_module_active( 'masterbar' ) ) {
+ $this->mock_masterbar_activation();
+ }
+
+ if ( $this->is_page_gutenberg() ) {
+ add_action( 'admin_enqueue_scripts', array( $this, 'enqueue_for_gutenberg' ), 100 );
+ return;
+ }
+
+ add_action( 'admin_init', array( $this, 'check_page' ) );
+ add_action( 'admin_menu', array( $this, 'remove_core_menus' ), 100 );
+ add_action( 'admin_menu', array( $this, 'add_plugin_menus' ), 101 );
+ add_action( 'admin_enqueue_scripts', array( $this, 'enqueue' ), 100 );
+ add_action( 'in_admin_header', array( $this, 'insert_sidebar_html' ) );
+ add_action( 'wp_before_admin_bar_render', array( $this, 'modify_masterbar' ), 100000 );
+
+ add_filter( 'get_user_option_admin_color', array( $this, 'admin_color_override' ) );
+
+ add_action( 'manage_plugins_columns', array( $this, 'manage_plugins_columns_header' ) );
+ add_action( 'manage_plugins_custom_column', array( $this, 'manage_plugins_custom_column' ), 10, 2 );
+ add_filter( 'bulk_actions-plugins', array( $this, 'bulk_actions_plugins' ) );
+
+ if ( 'plugins.php' === basename( $_SERVER['PHP_SELF'] ) ) {
+ add_action( 'admin_notices', array( $this, 'plugins_admin_notices' ) );
+ }
+ }
+
+ public function manage_plugins_columns_header( $columns ) {
+ if ( current_user_can( 'jetpack_manage_autoupdates' ) ) {
+ $columns['autoupdate'] = __( 'Automatic Update', 'jetpack' );
+ }
+ return $columns;
+ }
+
+ public function manage_plugins_custom_column( $column_name, $slug ) {
+ static $repo_plugins = array();
+
+ if ( ! current_user_can( 'jetpack_manage_autoupdates' ) ) {
+ return;
+ }
+
+ if ( empty( $repo_plugins ) ) {
+ $repo_plugins = self::get_dotorg_repo_plugins();
+ }
+
+ $autoupdating_plugins = Jetpack_Options::get_option( 'autoupdate_plugins', array() );
+ // $autoupdating_plugins_translations = Jetpack_Options::get_option( 'autoupdate_plugins_translations', array() );
+ if ( 'autoupdate' === $column_name ) {
+ if ( ! in_array( $slug, $repo_plugins ) ) {
+ return;
+ }
+ // Shamelessly swiped from https://github.com/Automattic/wp-calypso/blob/59bdfeeb97eda4266ad39410cb0a074d2c88dbc8/client/components/forms/form-toggle
+ ?>
+
+ <span class="form-toggle__wrapper">
+ <input
+ id="autoupdate_plugin-toggle-<?php echo esc_attr( $slug ) ?>"
+ name="autoupdate_plugins[<?php echo esc_attr( $slug ) ?>]"
+ value="autoupdate"
+ class="form-toggle autoupdate-toggle"
+ type="checkbox"
+ <?php checked( in_array( $slug, $autoupdating_plugins ) ); ?>
+ readonly
+ data-slug="<?php echo esc_attr( $slug ); ?>"
+ />
+ <label class="form-toggle__label" for="autoupdate_plugin-toggle-<?php echo esc_attr( $slug ) ?>">
+ <span class="form-toggle__switch" role="checkbox"></span>
+ <span class="form-toggle__label-content"><?php /* */ ?></span>
+ </label>
+ </span>
+
+ <?php
+ }
+ }
+
+ public static function get_dotorg_repo_plugins() {
+ $plugins = get_site_transient( 'update_plugins' );
+ return array_merge( array_keys( $plugins->response ), array_keys( $plugins->no_update ) );
+ }
+
+ public function bulk_actions_plugins( $bulk_actions ) {
+ $bulk_actions['jetpack_enable_plugin_autoupdates'] = __( 'Enable Automatic Updates', 'jetpack' );
+ $bulk_actions['jetpack_disable_plugin_autoupdates'] = __( 'Disable Automatic Updates', 'jetpack' );
+ return $bulk_actions;
+ }
+
+ public function handle_bulk_actions_plugins( $redirect_to, $action, $slugs ) {
+ $redirect_to = remove_query_arg( array( 'jetpack_enable_plugin_autoupdates', 'jetpack_disable_plugin_autoupdates' ), $redirect_to );
+ if ( in_array( $action, array( 'jetpack_enable_plugin_autoupdates', 'jetpack_disable_plugin_autoupdates' ) ) ) {
+ $list = Jetpack_Options::get_option( 'autoupdate_plugins', array() );
+ $initial_qty = sizeof( $list );
+
+ if ( 'jetpack_enable_plugin_autoupdates' === $action ) {
+ $list = array_unique( array_merge( $list, $slugs ) );
+ } elseif ( 'jetpack_disable_plugin_autoupdates' === $action ) {
+ $list = array_diff( $list, $slugs );
+ }
+
+ Jetpack_Options::update_option( 'autoupdate_plugins', $list );
+ $redirect_to = add_query_arg( $action, absint( sizeof( $list ) - $initial_qty ), $redirect_to );
+ }
+ return $redirect_to;
+ }
+
+ public function plugins_admin_notices() {
+ if ( ! empty( $_GET['jetpack_enable_plugin_autoupdates'] ) ) {
+ $qty = (int) $_GET['jetpack_enable_plugin_autoupdates'];
+ printf( '<div id="message" class="updated fade"><p>' . _n( 'Enabled automatic updates on %d plugin.', 'Enabled automatic updates on %d plugins.', $qty, 'jetpack' ) . '</p></div>', $qty );
+ } elseif ( ! empty( $_GET['jetpack_disable_plugin_autoupdates'] ) ) {
+ $qty = (int) $_GET['jetpack_disable_plugin_autoupdates'];
+ printf( '<div id="message" class="updated fade"><p>' . _n( 'Disabled automatic updates on %d plugin.', 'Disabled automatic updates on %d plugins.', $qty, 'jetpack' ) . '</p></div>', $qty );
+ }
+ }
+
+ public function jetpack_toggle_autoupdate() {
+ if ( ! current_user_can( 'jetpack_manage_autoupdates' ) ) {
+ wp_send_json_error();
+ return;
+ }
+
+ $type = $_POST['type'];
+ $slug = $_POST['slug'];
+ $active = 'false' !== $_POST['active'];
+
+ check_ajax_referer( "jetpack_toggle_autoupdate-{$type}" );
+
+ if ( ! in_array( $type, array( 'plugins', 'plugins_translations' ) ) ) {
+ wp_send_json_error();
+ return;
+ }
+
+ $jetpack_option_name = "autoupdate_{$type}";
+
+ $list = Jetpack_Options::get_option( $jetpack_option_name, array() );
+
+ if ( $active ) {
+ $list = array_unique( array_merge( $list, (array) $slug ) );
+ } else {
+ $list = array_diff( $list, (array) $slug );
+ }
+
+ Jetpack_Options::update_option( $jetpack_option_name, $list );
+
+ wp_send_json_success( $list );
+ }
+
+ public function admin_color_override( $color ) {
+ return 'fresh';
+ }
+
+ public function mock_masterbar_activation() {
+ include_once JETPACK__PLUGIN_DIR . 'modules/masterbar/masterbar.php';
+ new A8C_WPCOM_Masterbar;
+ }
+
+ public function remove_core_menus() {
+ remove_menu_page( 'index.php' );
+ remove_menu_page( 'jetpack' );
+ remove_menu_page( 'edit.php' );
+ remove_menu_page( 'edit.php?post_type=feedback' );
+ remove_menu_page( 'upload.php' );
+ remove_menu_page( 'edit.php?post_type=page' );
+ remove_menu_page( 'edit-comments.php' );
+ remove_menu_page( 'themes.php' );
+ remove_menu_page( 'plugins.php' );
+ remove_menu_page( 'users.php' );
+ remove_menu_page( 'tools.php' );
+ remove_menu_page( 'link-manager.php' );
+
+ // Core settings pages
+ remove_submenu_page( 'options-general.php', 'options-general.php' );
+ remove_submenu_page( 'options-general.php', 'options-writing.php' );
+ remove_submenu_page( 'options-general.php', 'options-reading.php' );
+ remove_submenu_page( 'options-general.php', 'options-discussion.php' );
+ remove_submenu_page( 'options-general.php', 'options-media.php' );
+ remove_submenu_page( 'options-general.php', 'options-permalink.php' );
+ remove_submenu_page( 'options-general.php', 'privacy.php' );
+ remove_submenu_page( 'options-general.php', 'sharing' );
+ }
+
+ public function add_plugin_menus() {
+ global $menu, $submenu;
+
+ add_menu_page( __( 'Manage Plugins', 'jetpack' ), __( 'Manage Plugins', 'jetpack' ), 'activate_plugins', 'plugins.php', '', $this->installed_plugins_icon(), 1 );
+
+ // // Count the settings page submenus, if it's zero then don't show this.
+ if ( empty( $submenu['options-general.php'] ) ) {
+ remove_menu_page( 'options-general.php' );
+ } else {
+ // Rename and make sure the plugin settings menu is always last.
+ // Sneaky plugins seem to override this otherwise.
+ // Settings is always key 80.
+ $menu[80][0] = __( 'Plugin Settings', 'jetpack' );
+ $menu[ max( array_keys( $menu ) ) + 1 ] = $menu[80];
+ unset( $menu[80] );
+ }
+ }
+
+ public function enqueue() {
+ wp_enqueue_style( 'calypsoify_wpadminmods_css', plugin_dir_url( __FILE__ ) . 'style.min.css', false, JETPACK__VERSION );
+ wp_style_add_data( 'calypsoify_wpadminmods_css', 'rtl', 'replace' );
+ wp_style_add_data( 'calypsoify_wpadminmods_css', 'suffix', '.min' );
+
+ wp_enqueue_script( 'calypsoify_wpadminmods_js', plugin_dir_url( __FILE__ ) . 'mods.js', false, JETPACK__VERSION );
+ wp_localize_script( 'calypsoify_wpadminmods_js', 'CalypsoifyOpts', array(
+ 'nonces' => array(
+ 'autoupdate_plugins' => wp_create_nonce( 'jetpack_toggle_autoupdate-plugins' ),
+ 'autoupdate_plugins_translations' => wp_create_nonce( 'jetpack_toggle_autoupdate-plugins_translations' ),
+ )
+ ) );
+ }
+
+ public function enqueue_for_gutenberg() {
+ wp_enqueue_style( 'calypsoify_wpadminmods_css', plugin_dir_url( __FILE__ ) . 'style-gutenberg.min.css', false, JETPACK__VERSION );
+ wp_style_add_data( 'calypsoify_wpadminmods_css', 'rtl', 'replace' );
+ wp_style_add_data( 'calypsoify_wpadminmods_css', 'suffix', '.min' );
+
+ wp_enqueue_script( 'calypsoify_wpadminmods_js', plugin_dir_url( __FILE__ ) . 'mods-gutenberg.js', false, JETPACK__VERSION );
+ wp_localize_script(
+ 'calypsoify_wpadminmods_js',
+ 'calypsoifyGutenberg',
+ array(
+ 'closeUrl' => $this->get_close_gutenberg_url(),
+ )
+ );
+ }
+
+ public function insert_sidebar_html() { ?>
+ <a href="<?php echo esc_url( 'https://wordpress.com/stats/day/' . Jetpack::build_raw_urls( home_url() ) ); ?>" id="calypso-sidebar-header">
+ <svg class="gridicon gridicons-chevron-left" height="24" width="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><g><path d="M14 20l-8-8 8-8 1.414 1.414L8.828 12l6.586 6.586"></path></g></svg>
+
+ <ul>
+ <li id="calypso-sitename"><?php bloginfo( 'name' ); ?></li>
+ <li id="calypso-plugins"><?php esc_html_e( 'Plugins' ); ?></li>
+ </ul>
+ </a>
+ <?php
+ }
+
+ public function modify_masterbar() {
+ global $wp_admin_bar;
+
+ // Add proper links to masterbar top sections.
+ $my_sites_node = (object) $wp_admin_bar->get_node( 'blog' );
+ $my_sites_node->href = 'https://wordpress.com/stats/day/' . Jetpack::build_raw_urls( home_url() );
+ $wp_admin_bar->add_node( $my_sites_node );
+
+ $reader_node = (object) $wp_admin_bar->get_node( 'newdash' );
+ $reader_node->href = 'https://wordpress.com';
+ $wp_admin_bar->add_node( $reader_node );
+
+ $me_node = (object) $wp_admin_bar->get_node( 'my-account' );
+ $me_node->href = 'https://wordpress.com/me';
+ $wp_admin_bar->add_node( $me_node );
+ }
+
+ private function installed_plugins_icon() {
+ $svg = '<svg class="gridicon gridicons-plugins" height="24" width="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 24"><g><path d="M16 8V3c0-.552-.448-1-1-1s-1 .448-1 1v5h-4V3c0-.552-.448-1-1-1s-1 .448-1 1v5H5v4c0 2.79 1.637 5.193 4 6.317V22h6v-3.683c2.363-1.124 4-3.527 4-6.317V8h-3z" fill="black"></path></g></svg>';
+
+ return 'data:image/svg+xml;base64,' . base64_encode( $svg );
+ }
+
+ /**
+ * Returns the Calypso domain that originated the current request.
+ *
+ * @return string
+ */
+ private function get_calypso_origin() {
+ $origin = ! empty( $_GET['origin'] ) ? $_GET['origin'] : 'https://wordpress.com';
+ $whitelist = array(
+ 'http://calypso.localhost:3000',
+ 'http://127.0.0.1:41050', // Desktop App
+ 'https://wpcalypso.wordpress.com',
+ 'https://horizon.wordpress.com',
+ 'https://wordpress.com',
+ );
+ return in_array( $origin, $whitelist ) ? $origin : 'https://wordpress.com';
+ }
+
+ /**
+ * Returns the site slug suffix to be used as part of the Calypso URLs. It already
+ * includes the slash separator at the beginning.
+ *
+ * @example "https://wordpress.com/block-editor" . $this->get_site_suffix()
+ *
+ * @return string
+ */
+ private function get_site_suffix() {
+ if ( class_exists( 'Jetpack' ) && method_exists( 'Jetpack', 'build_raw_urls' ) ) {
+ $site_suffix = Jetpack::build_raw_urls( home_url() );
+ } elseif ( class_exists( 'WPCOM_Masterbar' ) && method_exists( 'WPCOM_Masterbar', 'get_calypso_site_slug' ) ) {
+ $site_suffix = WPCOM_Masterbar::get_calypso_site_slug( get_current_blog_id() );
+ }
+
+ if ( $site_suffix ) {
+ return "/${site_suffix}";
+ }
+ return '';
+ }
+
+ /**
+ * Returns the Calypso URL that displays either the current post type list (if no args
+ * are supplied) or the classic editor for the current post (if a post ID is supplied).
+ *
+ * @param int|null $post_id
+ * @return string
+ */
+ public function get_calypso_url( $post_id = null ) {
+ $screen = get_current_screen();
+ $post_type = $screen->post_type;
+ if ( is_null( $post_id ) ) {
+ // E.g. `posts`, `pages`, or `types/some_custom_post_type`
+ $post_type_suffix = ( 'post' === $post_type || 'page' === $post_type )
+ ? "/${post_type}s"
+ : "/types/${post_type}";
+ $post_suffix = '';
+ } else {
+ $post_type_suffix = ( 'post' === $post_type || 'page' === $post_type )
+ ? "/${post_type}"
+ : "/edit/${post_type}";
+ $post_suffix = "/${post_id}";
+ }
+
+ return $this->get_calypso_origin() . $post_type_suffix . $this->get_site_suffix() . $post_suffix;
+ }
+
+ /**
+ * Returns the URL to be used on the block editor close button for going back to the
+ * Calypso post list.
+ *
+ * @return string
+ */
+ public function get_close_gutenberg_url() {
+ return $this->get_calypso_url();
+ }
+
+ /**
+ * Returns the URL for switching the user's editor to the Calypso (WordPress.com Classic) editor.
+ *
+ * @return string
+ */
+ public function get_switch_to_classic_editor_url() {
+ return add_query_arg(
+ 'set-editor',
+ 'classic',
+ $this->is_calypsoify_enabled ? $this->get_calypso_url( get_the_ID() ) : false
+ );
+ }
+
+ public function check_param() {
+ if ( isset( $_GET['calypsoify'] ) ) {
+ if ( 1 == (int) $_GET['calypsoify'] ) {
+ update_user_meta( get_current_user_id(), 'calypsoify', 1 );
+ } else {
+ update_user_meta( get_current_user_id(), 'calypsoify', 0 );
+ }
+
+ $page = remove_query_arg( 'calypsoify', wp_basename( $_SERVER['REQUEST_URI'] ) );
+
+ wp_safe_redirect( admin_url( $page ) );
+ }
+ }
+
+ public function check_page() {
+ // If the user hits plain /wp-admin/ then disable Calypso styles.
+ $page = wp_basename( esc_url( $_SERVER['REQUEST_URI'] ) );
+
+ if ( false !== strpos( 'index.php', $page ) || false !== strpos( 'wp-admin', $page ) ) {
+ update_user_meta( get_current_user_id(), 'calypsoify', 0 );
+ wp_safe_redirect( admin_url() );
+ die;
+ }
+ }
+
+ /**
+ * Return whether a post type should display the Gutenberg/block editor.
+ *
+ * @since 6.7.0
+ */
+ public function is_post_type_gutenberg( $post_type ) {
+ return use_block_editor_for_post_type( $post_type );
+ }
+
+ public function is_page_gutenberg() {
+ $page = wp_basename( esc_url( $_SERVER['REQUEST_URI'] ) );
+
+ if ( false !== strpos( $page, 'post-new.php' ) && empty ( $_GET['post_type'] ) ) {
+ return true;
+ }
+
+ if ( false !== strpos( $page, 'post-new.php' ) && isset( $_GET['post_type'] ) && $this->is_post_type_gutenberg( $_GET['post_type'] ) ) {
+ return true;
+ }
+
+ if ( false !== strpos( $page, 'post.php' ) ) {
+ $post = get_post( $_GET['post'] );
+ if ( isset( $post ) && isset( $post->post_type ) && $this->is_post_type_gutenberg( $post->post_type ) ) {
+ return true;
+ }
+ }
+
+ if ( false !== strpos( $page, 'revision.php' ) ) {
+ $post = get_post( $_GET['revision'] );
+ $parent = get_post( $post->post_parent );
+ if ( isset( $parent ) && isset( $parent->post_type ) && $this->is_post_type_gutenberg( $parent->post_type ) ) {
+ return true;
+ }
+ }
+
+ return false;
+ }
+}
+
+$Jetpack_Calypsoify = Jetpack_Calypsoify::getInstance();