Como adiciono uma guia personalizada na página de perfil do usuário?


11

Preciso adicionar uma guia personalizada na página de perfil do usuário. Eu defini minha rota como abaixo:

mymodule.routing.yml

mymodule.account:
path: '/user/{user}/custom'
defaults: 
  _form: '\Drupal\mymodule\Form\MyModuleUserSettingsForm'
  _title: 'Custom Settings'
  user: \d+
requirements:
  _permission: 'access content'

mymodule.links.task.yml

mymodule.account:
  title: Mymodule Settings
  route_name: mymodule.account
  base_route: entity.user.canonical

mymodule.links.menu.yml

mymodule.account:
  title: My module Settings
  parent: entity.user.canonical
  route_name: mymodule.account

Depois que a guia Limpar cache aparecer na página de perfil. Mas quando abro a URL / user / 1 / custom , vejo a mensagem Página não encontrada.


1
usuário: \ d + deve obedecer aos requisitos - faça essa alteração, limpe todo o cache e veja se a página ainda não foi encontrada.
21716 Kevin

Respostas:


5

Seu problema está no mymodule.routing.ymlarquivo, o grande problema é o local de user: \d+, esta linha deve estar na requirements:seção, o outro problema é o recuo. Portanto, o código final deve ser:

mymodule.account:
  path: '/user/{user}/custom'
  defaults: 
    _form: '\Drupal\mymodule\Form\MyModuleUserSettingsForm'
    _title: 'Custom Settings'
  requirements:
    _permission: 'access content'
    user: \d+

E é claro que você precisa da definição da classe de formulário em src/Form/MyModuleUserSettingsForm.php

<?php

namespace Drupal\mymodule\Form;

use Drupal\Core\Form\FormBase;
use Drupal\Core\Form\FormStateInterface;

/**
 * Class MyModuleUserSettingsForm.
 *
 * @package Drupal\mymodule\Form
 */
class MyModuleUserSettingsForm extends FormBase {

  /**
   * {@inheritdoc}
   */
  public function getFormId() {
    return 'simple_form';
  }

  /**
   * {@inheritdoc}
   */
  public function buildForm(array $form, FormStateInterface $form_state) {
    $form['title'] = [
      '#type' => 'textfield',
      '#title' => $this->t('Title'),
      '#maxlength' => 64,
      '#size' => 64,
      '#required' => TRUE,
    ];
    $form['submit'] = [
      '#type' => 'submit',
      '#value' => t('Submit'),
    ];

    return $form;
  }

  public function validateForm(array &$form, FormStateInterface $form_state) {  }

  /**
   * {@inheritdoc}
   */
  public function submitForm(array &$form, FormStateInterface $form_state) {  }

}

E um mymodule.info.ymlarquivo para concluir seu módulo (neste caso, chamado mymodule)

name: My Module
type: module
description: 'My module'
core: 8.x
package: Custom

Referência: Estrutura de rotas


Interessante que a estrutura das rotas não mencione a chave user:nem para onde deve ir ... Talvez os documentos precisem ser aprimorados / editados? ou eu perdi alguma coisa?
No Sssweat 23/02

NVV, eu vejo, está sob _entity_access: e eles usaram o nó como um exemplo lá, portanto, por que meu CTRL + F "usuário:" não encontrou nada.
No Sssweat

3

Existem duas maneiras de carregar um formulário usando uma rota. Você pode carregar um retorno de chamada que carrega um formulário e o retorna como parte da matriz de compilação ou pode carregar o formulário diretamente, definindo o parâmetro _form nos padrões.

Você pode pesquisar na base de código para encontrar exemplos de trabalho, copiá-los em seu mymodule.routing.yml, editá-los de acordo com suas necessidades e depois reconstruir o cache.

Carregando formulário a partir do retorno de chamada:

Há um exemplo de trabalho no módulo de contato:

/core/modules/contact/contact.routing.yml

entity.user.contact_form:
  path: '/user/{user}/contact'
  defaults:
    _title: 'Contact'
    _controller: '\Drupal\contact\Controller\ContactController::contactPersonalPage'
  requirements:
    _access_contact_personal_tab: 'TRUE'
    user: \d+

Em seguida, em /core/modules/contact/src/Controller/ContactController.php

você pode ver um exemplo de como carregar um formulário no retorno de chamada:

  public function contactPersonalPage(UserInterface $user) {
    // Do not continue if the user does not have an email address configured.
    if (!$user->getEmail()) {
      throw new NotFoundHttpException();
    }

    $message = $this->entityManager()->getStorage('contact_message')->create(array(
      'contact_form' => 'personal',
      'recipient' => $user->id(),
    ));

    $form = $this->entityFormBuilder()->getForm($message);
    $form['#title'] = $this->t('Contact @username', array('@username' => $user->getDisplayName()));
    $form['#cache']['contexts'][] = 'user.permissions';
    return $form;
  }

Carregando formulário diretamente da rota:

Se você deseja carregar o formulário diretamente usando o padrão _form, há um exemplo no módulo de atalho em /core/modules/shortcut/shortcut.routing.yml

shortcut.set_switch:
  path: '/user/{user}/shortcuts'
  defaults:
    _form: 'Drupal\shortcut\Form\SwitchShortcutSet'
    _title: 'Shortcuts'
  requirements:
    _custom_access: 'Drupal\shortcut\Form\SwitchShortcutSet::checkAccess'
  options:
    _admin_route: TRUE
    user: \d+

Nesse caso, o usuário é passado como parâmetro para o formulário, consulte /core/modules/shortcut/src/Form/SwitchShortcutSet.php

  public function buildForm(array $form, FormStateInterface $form_state, UserInterface $user = NULL) {

1

Algumas coisas que posso ver de imediato ...

Você tem TUDO chamado mymodule.account. Eu diversificaria isso um pouco. Considere ir com isso para a página de tarefas:

mymodule.account_tab:
  title: Mymodule Settings
  route_name: mymodule.account
  base_route: entity.user.canonical

Também não acredito que você precise de algo no menu para isso.

Entre esses dois, você deve estar pronto! Sinta-se à vontade para entrar em contato diretamente comigo se você não conseguir descobrir isso porque eu - APENAS-- fiz isso funcionar na minha porta D8 do Apply For Role!


1

Em modulename.routing.yml, você deve passar o argumento do usuário como abaixo

profile.user_information:
  path: '/user/{user}/profile'
  defaults:
    _form: '\Drupal\profile\Form\UserInformation'
    _title: 'UserInformation'
  requirements:
    _permission: 'access content'
  options:
    user: \d+

e em modulename.links.task.yml você tem o código como abaixo

profile.user_information:
  title: User profile
  route_name: profile.user_information
  base_route: entity.user.canonical
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.