Великий виклик меню адміністратора WordPress січня 2011 року (також як вирішити деякі проблеми при зміні системи меню WordPress Admin?)


14

Це питання трохи унікальне.

Частково це "виклик", який я видаю команді WordPress ( або комусь іншому ), пов’язаному з проїзними квитками: # 16048 , # 16050 та # 16204 .

Мета

Мета - обійти три (3) проблеми, проілюстровані на скріншоті нижче, намагаючись змінити розділ меню WordPress Admin:

  1. Отримайте сторінку підменю "Мікросайт", яку слід виділити під час редагування адвоката (для цього нам потрібно якось бути в змозі застосувати "поточний" до пункту підменю, _ та деякі гачки в функції _wp_menu_output () забезпечить необхідне тут) ,

  2. Отримайте посилання на сторінку меню адвоката, на яке посилаєтесь під /wp-admin/edit.php?post_type=attorneyчас редагування адвоката (і ті самі потрібні гачки в функції _wp_menu_output () могли б це впоратися) , і

  3. Отримайте посилання "Мікросайт", щоб не викликати помилку "У вас недостатньо дозволів для доступу до цієї сторінки" помилка * (це найгучніше вирішити, і гачок на зворотному значенніuser_can_access_admin_page() міг би добре вирішити цю проблему.)

Скріншот екрана програми The Great WordPress Admin Menu Challenge січня 2011 року
(джерело: mikeschinkel.com )

Більше, ніж просто моє використання

Ці три (3) проблеми є моїм випадком використання, але вони є символом проблем, пов’язаних із налаштуванням адміністраторських меню в WordPress.

Кілька з команди WordPress сказали, що це легко, і, таким чином, мається на увазі, що мені щось не вистачає (що може бути правильним), але я не дивився на цю проблему тижнями і не з'ясував, як її обійти, тому я створив плагін, який ви бачите нижче ( також можна завантажити з Gist ) як найпростіший приклад використання проблем. Код в admin_menu2()досить хакітський, але це майже все, що потрібно для зміни меню адміністратора в WordPress.

Зауважте, що я не намагався використовувати нову remove_menu_page()чи нову remove_submenu_page()функції в 3.1, тому що для створення плагіна знадобилося б більше часу - у мене вже був код admin_menu2()з існуючого проекту - і я не вірю, що вони звертаються до адреси проблема все одно.

Що мені потрібно?

Мені потрібна одна з двох (2) речей:

  1. Вирішення проблем, які я виставляю з цим плагіном і пояснюю в цьому питанні та на скріншоті (BTW, я дискваліфікую ваше рішення, якщо ви використовуєте PHP Output Buffering для вирішення будь-якої частини цього) , або

  2. Щоб команда WordPress визнала, що справді є необхідність у цих гачках, а також змусити їх переглянути свою позицію в квитках.

Як ви приймаєте виклик?

  1. Завантажте та встановіть первозданну нову копію WordPress 3.1 (можливо, будь-яка редакція) ,

  2. Завантажте, встановіть та активуйте плагін "The Great WordPress Admin Menu Challenge of January 2011" нижче (або завантажте плагін з Gist ) , а потім

  3. Дотримуйтесь інструкцій, знайдених на сторінці плагіна для цього плагіна (див. Наступний знімок екрана), але в основному завантажте знімок екрана, який ви бачите вище, а потім спробуйте розібратися у трьох (3) описаних проблемах:

Скріншот інструкцій з плагіну "The Great WordPress Admin Menu Challenge of January 2011"
(джерело: mikeschinkel.com )

Один промінь надії

На щастя, Ендрю Начін із команди WordPress запропонував переглянути це, як тільки я його зашифрував, тому я в першу чергу публікую тут, щоб він переглянув та прокоментував, а також мав можливість коментувати інших. Я знаю, що він зайнятий, але я сподіваюся, що він (або навіть ви) може знайти час, щоб встановити цей плагін на незайману установку v3.1 і побачити, чи зможе він вирішити проблему.

Якщо ви згодні з тим, що виклик неможливий?

Якщо ви спробували цей виклик, ви прийшли до того самого висновку, що і я, і якщо ви хочете, щоб меню адміністраторів WordPress було більш налаштованим, будь ласка, прокоментуйте ці квитки на трек ( # 16048 - # 16050 - # 16204 ) і проголосуйте за це питання виявити підтримку.

Я з радістю визнаю, що щось пропустив, якби це зробив

Звичайно, можливо, я міг би бути повністю загиблим у цьому мозку, і хтось міг би вказати, як це зробити. Насправді, я дуже сподіваюся, що так і вийшло; Я б швидше помилявся і працював над цим, ніж навпаки.

І ось плагін

Ви також можете завантажити, якщо з Gist :

<?php
/*
Plugin Name: The Great WordPress Admin Menu Challenge of Jan 2011
Description: <em>"The Great WordPress Admin Menu Challenge of Jan 2011"</em> was inspired by the WordPress team's apparent lack of understanding of the problems addressed by trac tickets <a href="http://core.trac.wordpress.org/ticket/16048">#16048</a> and <a href="http://core.trac.wordpress.org/ticket/16050">#16050</a> <em>(See also: <a href="http://core.trac.wordpress.org/ticket/16204">#16204</a>)</em> and suggestion that the <a href="http://wordpress.org/extend/plugins/admin-menu-editor/>Admin Menu Editor</a> plugin handles the use-cases that the tickets address. Debate spilled over onto Twitter with participation from <a href="http://twitter.com/nacin">@nacin</a>, <a href="http://twitter.com/aaronjorbin">@aaronjorbin</a>, <a href="http://twitter.com/petemall">@petemall</a>, <a href="http://twitter.com/westi">@westi</a>, <a href="http://twitter.com/janeforshort">@janeforshort</a>, <a href="http://twitter.com/PatchesWelcome">@PatchesWelcome</a>; supportive comments from <a href="http://twitter.com/ramsey">@ramsey</a>, <a href="http://twitter.com/brianlayman">@brianlayman</a>, <a href="http://twitter.com/TheLeggett">@TheLeggett</a>, a retweeting of @nacin's simple yet <em>(AFAICT)</em> insufficient solution by <a href="http://twitter.com/vbakaitis">@vbakaitis</a>, <a href="http://twitter.com/Viper007Bond">@Viper007Bond</a>, <a href="http://twitter.com/nickopris">@nickopris</a>, <a href="http://twitter.com/Trademark">@Trademark</a>, <a href="http://twitter.com/favstar_pop">@favstar_pop</a>, <a href="http://twitter.com/designsimply">@designsimply</a>, <a href="http://twitter.com/darylkoop">@darylkoop</a>, <a href="http://twitter.com/iamjohnford">@iamjohnford</a>, <a href="http://twitter.com/markjaquith">@markjaquith</a>, <a href="http://twitter.com/JohnJamesJacoby">@JohnJamesJacoby</a> and <a href="http://twitter.com/dd32">@dd32</a>. Also see <a href="http://andrewnacin.com/2010/12/20/better-admin-menu-controls-custom-post-types-wordpress-3-1/#comment-6360">comments</a> on @nacin's blog post entitled "<em>Better admin menu handling for post types in WordPress 3.1</em>." <strong>The desired goal of the <em>"challenge"</em></strong> is to simply either to find a solution that has eluded me or, to get those who are dismissing it as solvable without added hooks in WordPress to have a tangible example to explore in hopes they will recognize that there is indeed a need for at least some of the requested hooks. <strong>There are three (3) steps to the challenge:</strong> 1.) Get the "Microsite" submenu page to be highlighted when editing an Attorney, 2.) Get the Attorney Menu Page link to link <a href="/wordpress//wp-admin/edit.php?post_type=attorney">here</a>  when editing an Attorney, and 3.) Get the "Microsite" link not to trigger a "You do not have sufficient permissions to access this page" error.  Here is <a href="https://mikeschinkel.com/websnaps/skitched-20110114-235302.png" target="_blank"><strong>a screenshot</strong> that attempts to illustrate the callenge</a>. The code can be found on gist <a href="https://gist.github.com/780709"><strong>here</strong></a>. Activate it as a plugin in a WordPress 3.1 install and go <a href="/wordpress//wp-admin/post.php?post=10&action=edit"><strong>here</strong></a> to see what the screenshot illustrates. <strong>Be sure to load the <a href="https://mikeschinkel.com/websnaps/skitched-20110114-235302.png" target="_blank">screenshot</a> in another browser tab or window first</strong>.
Author:      Mike Schinkel
Author URI:  http://about.me/mikeschinkel
Plugin URI:  https://gist.github.com/780709
*/
if (!class_exists('TheGreatWordPressAdminMenuChallenge')) {
  class TheGreatWordPressAdminMenuChallenge {
    static function on_load() {
      add_action('init',array(__CLASS__,'init'));
      add_action('admin_menu',array(__CLASS__,'admin_menu1'));      // Simulates generic "Microsite" plugin
      add_action('admin_menu',array(__CLASS__,'admin_menu2'),100);  // Simulates website-specific plugin
      add_action('post_row_actions',array(__CLASS__,'post_row_actions'),10,2);
    }
    static function post_row_actions($actions,$post) {
      $url = admin_url(self::this_microsite_url($post->ID));
      $actions = array_merge(array('microsite'=>"<a href=\"{$url}\" title=\"Manage this Microsite\">Microsite</a>"),$actions);
      return $actions;
    }
    static function the_microsite_editor() {
      echo "We are in the Microsite Editor for " . self::post_title();
    }
    static function admin_menu1() {
      if (self::this_post_id() && in_array(self::this_post_type(),array('attorney','practice_area'))) {
        add_submenu_page(
          self::this_parent_slug(),
          self::microsite_page_title(),
          self::microsite_page_title(),
          $capability = 'edit_posts',
          'microsite',
          array($microsite,'the_microsite_editor')
        );
        global $wp_post_types;
        $parent_type_meta = $wp_post_types[self::this_post_type()];
        global $menu;
        $slug = false;
        foreach($menu as $index => $menu_page)
          if ($menu_page[0]===$parent_type_meta->label) {
            $slug = $menu_page[2];
            break;
          }
        if ($slug) {
          global $pagenow;
          global $submenu;
          // Setting this makes gives the link to the microsite in the menu the highlight for "current" menu option
          global $submenu_file;
          $submenu_file = self::this_microsite_url();
          $index = end(array_keys($submenu[$slug]));
          $submenu[$slug][$index][12] = $submenu_file;
        }
      }
    }
    static function this_parent_slug() {
      return "edit.php?post_type=" . self::this_post_type();
    }
    static function post_title() {
      $post_id = self::this_post_id();
      return ($post_id ? get_post($post_id)->post_title : false);
    }
    static function microsite_page_title() {
      return 'Microsite for ' . self::post_title();
    }
    static function this_post_type($get_post=true) {
      $post_type = (isset($_GET['post_type']) ? $_GET['post_type'] : false);
      if (!$post_type && $get_post) {
        $post_id = self::this_post_id();
        $post_type = get_post($post_id)->post_type;
      }
      return $post_type;
    }
    static function this_post_id() {
      $post_id = false;
      $post_type = self::this_post_type(false);
      if (isset($_GET[$post_type]))
        $post_id = intval($_GET[$post_type]);
      else if (isset($_GET['post']))
        $post_id = intval($_GET['post']);
      return $post_id;
    }
    static function this_microsite_url($post_id=false) {
      $post_type = self::this_post_type();
      $post_id = $post_id ? intval($post_id) : self::this_post_id();
      return "edit.php?post_type={$post_type}&page=microsite&attorney={$post_id}";
    }
    static function admin_menu2() {
      // The code required for this is super, nasty, ugly and shouldn't be, but at least it *is* doable
      global $menu;
      global $submenu;
      global $microsite;

      $parent_type = self::this_post_type();
      foreach(array('attorney','practice_area') as $post_type) {
        $slug = "edit.php?post_type={$post_type}";
        if ($post_type==$parent_type) {  // If a microsite remove everything except the microsite editor
          $microsite_url = self::this_microsite_url();
          foreach($submenu[$slug] as $submenu_index => $submenu_page) {
            if ($submenu_page[2]!=$microsite_url) {
              unset($submenu[$slug][$submenu_index]);
            }
          }
        } else {
          $submenu[$slug] = array();
        }
      }

       // Remove the Submenus for each menu
      unset($submenu['index.php']);
      unset($submenu['edit.php?post_type=article']);
      unset($submenu['edit.php?post_type=event']);
      unset($submenu['edit.php?post_type=case_study']);
      unset($submenu['edit.php?post_type=news_item']);
      unset($submenu['edit.php?post_type=transaction']);
      unset($submenu['edit.php?post_type=page']);
      unset($submenu['upload.php']);

      unset($submenu['users.php'][13]); // Removed the "Add New"

      $remove = array_flip(array(
        'edit.php',
        'link-manager.php',
        'edit-comments.php',
        'edit.php?post_type=microsite-page',
      ));
      if (!current_user_can('manage_tools'))
        $remove['tools.php'] = count($remove);

      foreach($menu as $index => $menu_page) {
        if (isset($remove[$menu_page[2]])) {
          unset($submenu[$menu_page[2]]);
          unset($menu[$index]);
        }
      }

      $move = array(
        'edit.php?post_type=page' => array( 'move-to' => 35,  0 => 'Other Pages' ),
        'separator2' => array( 'move-to' => 40 ),
        'upload.php' => array( 'move-to' => 50, 0 => 'Media Library' ),
      );
      $add = array();
      foreach($menu as $index => $menu_page) {
        if (isset($move[$menu_page[2]])) {
          foreach($move[$menu_page[2]] as $value_index => $value) {
            if ($value_index==='move-to') {
              $move_to = $value;
            } else {
              $menu_page[$value_index] = $value;
            }
          }
          $add[$move_to] = $menu_page;
          unset($menu[$index]);
        }
      }
      foreach($add as $index => $value)
        $menu[$index] = $value;

      add_menu_page(
        'Attorney Positions',
        'Attorney Positions',
        'edit_posts',
        'edit-tags.php?taxonomy=attorney-position&amp;post_type=attorney',
        false,
        false,
        55);

      ksort($menu); // Need to sort or it doesn't come out right.
    }
    static function init() {
      register_post_type('attorney',array(
        'label'           => 'Attorneys',
        'public'          => true,
      ));
      register_post_type('practice_area',array(
        'label'           => 'Practice Areas',
        'public'          => true,
      ));
      register_taxonomy('attorney-position','attorney',array(
        'label'=>'Attorney Positions',
      ));
      register_post_type('article',array(
        'label'           => 'Articles & Presentations',
        'public'          => true,
      ));
      register_post_type('case_study',array(
        'label'           => 'Case Studies',
        'public'          => true,
      ));
      register_post_type('news_item',array(
        'label'           => 'Firm News',
        'public'          => true,
      ));
      register_post_type('event',array(
        'label'           => 'Events',
        'public'          => true,
      ));
      register_post_type('transaction',array(
        'label'           => 'Transactions',
        'public'          => true,
      ));

      // Install the test data
      $post_id = 10;
      $attorney = get_post($post_id);
      if (!$attorney) {
        global $wpdb;
        $wpdb->insert($wpdb->posts,array(
          'ID' => $post_id,
          'post_title' => 'John Smith',
          'post_type' => 'attorney',
          'post_content' => 'This is a post about the Attorney John Smith.',
          'post_status' => 'publish',
          'post_author' => 1,
        ));
      }
    }
  }
  TheGreatWordPressAdminMenuChallenge::on_load();
}

Всім, хто це читає, я дуже сподіваюся, що ви можете допомогти.

Заздалегідь спасибі.


Мені цікаво (і мені справді потрібно зрівняти свій досвід роботи з адміністратором), але, напевно, чекаю 3.1 остаточного випуску. Мій локальний тестовий стек не дуже підходить для декількох основних версій, тому я дотримуюся поточної стабільності.
Рарст

Я точно знаю проблему, про яку ви говорите про Майка, я не думаю, що я міг би описати її краще, ніж у вас там, але я бачив ті самі проблеми, коли писав спадне меню для адміністратора (для задоволення), просто додаючи мій +1.
t31о

@ t310s - Дякуємо, що додали +1. Мені знадобилося, ймовірно, два тижні досліджень, щоб можна було описати проблему, тому цей факт ви навіть можете визнати, що це актуально (і не витратили 2 тижні, які я маю) означає, що ви випереджаєте більшість усіх із цього, в тому числі я!
MikeSchinkel

Відповіді:


2

Майк, я ознайомився з кодом та вашим ідеальним випадком кінцевого використання ... а деякі з них, чесно кажучи, неможливі в поточній системі. Знову ваші вимоги:

  1. Отримайте сторінку підменю "Мікросайт", яку слід виділити під час редагування адвоката
  2. Отримайте посилання на сторінку меню адвоката, на яке посилаєтесь під /wp-admin/edit.php?post_type=attorneyчас редагування адвоката
  3. Отримайте посилання "Мікросайт", щоб не викликати помилку "У вас недостатньо дозволів для доступу до цієї сторінки"

І ключове питання тут - №2.

Що я спробував

Я спробував додати спеціальний тип публікації для адвокатів, і мені відразу було нагадано, що /wp-admin/edit.php?post_type=attorneyви отримаєте список адвокатів, а не фактичний екран редагування. Фактичне редагування відбувається з/wp-admin/post.php?post=10&action=edit . Тож якщо ви дійсно прив’язані до №2, два інші критерії не працюватимуть.

Ось чому номер 3 не спрацьовує у впровадженні ... і мені навіть не вдалося спробувати №1, тому що я не міг так далеко дістатись.


Вважайте, ваш аналіз правильний. Розміщення, вказане у питанні, - це те, що я мав більше одного запиту клієнта в своїх цілях, щоб спростити структури меню для їх використання. Я запропонував використовувати підменю, але це їм не сподобалось; вони вважали, що це буде занадто заплутано для їх користувачів. Одне, що я, можливо, не згадував - це те, що я розробляю продукт на базі WordPress для їх розповсюдження порівняно з веб-сайтом WordPress, де я можу просто навчити їх знати, як зробити так, щоб справи працювали. Інший їх варіант - відмовитися від WordPress; не те, що я хочу, щоб вони робили.
MikeSchinkel

2

Ей, Майку, твій випуск №3 пояснюється тим, що ти вказуєш ($microsite, 'the_microsite_editor'), де воно має бути (__CLASS__, 'the_microsite_editor').

Оновлення: Провівши занадто багато часу, намагаючись вирішити якусь подібну проблему для мого власного плагіна, ось що я знайшов, що може допомогти у вашому виклику (зауважте, що функції - це методи під вашим класом):

function add_posttype_submenu_page($mytype, $label, $cap, $slug) {  
    /* we add two submenu pages to work around the 
       edit.php?post_type=...&page=...problem and have 
       our page called as admin.php?page=... instead */
    //first create a 'blind' pseudo-entry to register our page callback
    add_submenu_page($mytype, $label, $label, $cap, $slug, 
                     array( &$this, 'admin_'.$mytype ));
    //then create a real entry that 'calls' our pseudo-entry
    add_submenu_page('edit.php?post_type='.$mytype, $label, 
                     $label, $cap, 'admin.php?page='.$slug);
    /* then lets fix/hack the highlighting */
    global $plugin_page;
    global $submenu_file;
    if ($plugin_page == $slug) {
        // this next line highlights the submenu entry
        $submenu_file = 'admin.php?page='.$slug; 
        add_filter('parent_file', 
                   array(&$this, 'evil_parent_file_hack'));
    }
} 

function evil_parent_file_hack() {
    //we do this to get the parent menu properly highlighted, too
    //it only gets called on the submenu menu page in question
    global $self;
    global $parent_file;
    $self = $parent_file;
    remove_filter('parent_file', array(&$this, 'evil_parent_file_hack'));
}

Тоді ви просто дзвоните add_posttype_submenu_page()за відповідними параметрами. Це має належним чином додати пункт підменю до меню, яке було створено автоматично під час register_post_type()дзвінка.


ой ... подвійні підкреслення навколо класу були перетворені на
жирне

Я полагодив це. :)
fuxia

О, це просто приголомшливо; Спасибі! Як я міг пропустити цей перший пункт?!? До!
MikeSchinkel

Спасибі, Майку. Повернення до початкової теми ... WP внутрішньо генерує ідентифікатор для деяких записів у меню і зберігає його у значенні 4 масивів меню. Наприклад, гачки для плагінів на сторінці. Однак для користувальницьких підтипів він зберігає ідентифікатор, що не відповідає формату плагін-сторінка-гачок. Я думаю, що багато чого могло б допомогти, якщо WP зробить це послідовно (тобто створити гачки сторінок для всього і використовувати їх для пов'язування записів меню один з одним, а не для приєднання підменю за допомогою "батьківського
слугу
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.