De Drupal 7 à Drupal 8

une (r)évolution à la portée de Tous

Qui suis-je ?

  • DrDam
  • @DrDam8584

 

  • 4-5 années de Drupal
  • Référent Technique Drupal
  •  
  • Spécialisé dans les architectures à fort trafic

 

  • Filière biologique, formation Ingénieur Agronome

Drupal : une évolution

Modèle MVC

  • ORM
  • Autoloader
  • Twig

Les apports de Drupal 8

Symfony 2

  • Fichier en Yaml
  • Routing

POO avancée

  • Héritage
  • Polymorphisme
  • Namespace
  • Annotation
  • Injection de dépendance

Drupal 7 VS Drupal 8

« Ceci est une Révolution »

« J'comprends rien à c'truc, j'peux pas l'faire à l'arrache ? »

« Il n'y a qu'une seule voie, celle de la vertu »

S. Jobs

M. Robespierre

Dev PHP du dimanche

Premières réactions

Anatomie Comparée

Premier constat

  • Le module
  • Les permissions
  • Les formulaires
  • Les blocs
  • Les menu_items
  • Le thèming
  • Le templating
  • ...
  • Les altérations
  • le hook_theme

Ce qui change

Ce qui ne change "presque" pas

Déclarer un module

Drupal 7

Drupal 8

mon_module

  • mon_module.info
  • mon_module.module

mon_module

  • mon_module.info.yml
  • mon_module.module
  • src/
name = "Mon module"
description = "Blabla"
package = "Mon package"


core = "7.x"
version = "7.x-1.0"

dependencies[] = dependance_1
dependencies[] = dependance_2
name: Mon module
type: module
description: Blabla
package: mon package

core: '8.x'
version: '8.x-1.0'

dependencies:
  - dependance_1
  - dependance_2

Les permissions

Déclarer une permission

mon_module_permission () {
  return array(
     'ma_permission' => array(
           'title' => t('Ma permission'),
           'description' => t('STOP'),
      ),
   );
}
'ma permission':
  title: 'Ma permission'
  description: 'STOP'

Drupal 7

Drupal 8

mon_module.module

 mon_module.permissions.yml

Utiliser les permissions

Drupal 7

Drupal 8

// Utiliser l'utilisateur courant
global $user;
// Utiliser l'utilisateur courant
$user = \Drupal::currentUser();
// Charger un utilisateur
$user2 = user_load(2) ; 

// L'utilisateur est-il authentifié ?
if(user_is_logged_in()) {
 /* ... */
}
// L'utilisateur a-t-il la permission ?
if(user_access('ma_permission')){
  /* ... */
}
// Charger un utilisateur
$user2 = \Drupal\user\Entity\User::load(2);
// L'utilisateur a-t-il la permission ?
if($user->hasPermission('ma_permission')) {
  /* ... */
} 
// L'utilisateur est-il authentifié ?
if($user->isAuthenticated()) {
  /* ... */
}

Les formulaires

Petits rappel de Drupal 7

Déclaration :

/* Utilisation du hook_form(). */
function mon_module_formtest_form() {
  /* ... */
}

/* paramètre de drupal_get_form() */
function mon_module_form_deux() {
  /* ... */
}

Validateur(s) :

  // Dans une déclaration de formulaire
  // ou un hook_form_alter 
  $form['#validate'][] = "form_validator";
}

function form_validator($form, &form_state) {
  /* ... */
}

// ou un hook_form_validate

Contrôle(s) de soumission :

  // Dans une déclaration de formulaire ou un hook_form_alter 
  $form['#submit'][] = "control_submit";
}

function control_submit($form, &form_state) {
  /* ... */
}

/* Utilisation du hook_form_submit(). */
function mon_module_formtest_form_submit($form, &form_state) {
  /* ... */
}

Déclaration d'un formulaire

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

class FormTest extends FormBase {

 

 
 
 

 

 

}

dans le fichier formtest.php
situé : mon_module/src/Form

  public function getFormId() { return 'Formulaire_de_test'; }

  public function buildForm(array $form, FormStateInterface $form_state)    {
  
  // Utilisation de la formAPI
  /* ... */
  
  return $form;
  }

Validation et soumission

méthodes de la classe " FormTest "

class FormTest extends FormBase {

  
  public function getFormId() { return 'id_du_formulaire'; }

  public function buildForm(array $form, FormStateInterface $form_state) {
     /* ... */
  }

 
  
 
           
 
 







}
  public function validateForm(array &$form, FormStateInterface $form_state) {
    // accès aux values
    $values = $form_state->getValues();
            
    // marquer une erreur
    $form_state->setErrorByName($field_name, $this->t('Error on field'));
  }


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

Utilisation d'un formulaire


$form = \Drupal::formBuilder()->getForm('Drupal\mon_module\Form\FormTest');

Ex-drupal_get_form

mon_module.testform:
  path: '/mon_module/testform'
  defaults:
    _title: 'Formulaire de test'
    _form: 'Drupal\mon_module\Form\FormTest'

Page de formulaire

Altération d'un formulaire


hook_form_alter(&$form, \Drupal\Core\Form\FormStateInterface $form_state, $form_id)

hook_form_FORM_ID_alter(&$form, \Drupal\Core\Form\FormStateInterface $form_state, $form_id)

hook_form_BASE_FORM_ID_alter(&$form, \Drupal\Core\Form\FormStateInterface $form_state, $form_id)

Par des fonctions

Par un objet

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

class AlterFormTest extends FormTest {

  public function getFormId() { return 'Formulaire_de_test_modif'; }

  public function buildForm(array $form, FormStateInterface $form_state) {
  
    $form = parent::buildForm($form, $form_state);
    unset($form['field_mail']);
  }

}

Les Blocs

Une longue évolution

Drupal 6

Drupal 7

function mon_module_block ( $op, $delta, $edit) {
  
  switch($op){



function mon_module_block_view($delta = '') {
  if($delta == 'mon_block') {
    // fabrique mon block
    $blocks['content'] = $outHTML;
  }
  /* ... */
  return $blocks;
}
function mon_module_block_info() {
  $blocks['mon_block'] = array(
    'info' => t('mon block'), 
  );
  /* ... */
  return $blocks;
}
   case 'view' :
    // fabrique les blocs
    /* ... */
    break;
   case 'list' :
    // declare les blocs
    /* ... */
    break;






  }
}

Déclarer un bloc

Drupal 7

Drupal 8

namespace '...';
use '...';








class blockOne extends BlockBase {
  





} 
function mon_module_block_info() {
  $blocks['mon_block'] = array(
    'info' => t('mon block'), 
  );
  /* ... */
  return $blocks;
}

mon_module.module

dans le fichier BlockOne.php
situé : mon_module/src/Plugin/Block

function mon_module_block_view($delta = '') {
  if($delta == 'mon_block') {
    // fabrique mon block
    $blocks['content'] = $outHTML;
  }
  /* ... */
  return $blocks;
}
/**
* @Block(
*   id = "block_one",
*   admin_label = "Block One",
* )
*/
  public function build() {
    // fabrique mon block
    return $themable_array;
  }

Le routing

La Théorie

Nouveau paradigme

Drupal 7

Drupal 8

Un hook pour les trouver tous.

Un hook pour les gouverner tous.

Un hook pour les agréger tous

et dans les menus les lier

La prophétie du hook

Le monde décentralisé 

Chacun son chemin

Chacun sa nature

Chacun son ascendance

Chacun son type

Un hook unique utilisé partout

Une multitude de fichiers

Différences théoriques

Drupal 7

Drupal 8

Type par défaut

menu_item

menu_callback

Callback de page

Fonction

Objet

Paramètre de requête

"path/%/%" et "page_argument"

auto-chargement possible

"path/{one}/{two}"

Prédéclaration des variables

Le routing

Les routes

La route unitaire

Route simple

Drupal 7

Drupal 8

mon_module

mon_module.routing.yml

mon_module_menu () {  
  $items['go'] = array(
    'page callback' => 'runGo',
    'access arguments' =>
	   array('ma_permission'),
    'title' => 'le titre de ma page'
    'type' => MENU_CALLBACK
  );
  /* ... */
  return $items;
}
monmodule.go:
  path: 'go'
  defaults:
    _content: 
'\Drupal\mon_module\Controller\GoController::runGo'
    _title: 'le titre de ma page'
  requirements:
    _permission: 'ma_permission'

La route unitaire

Route avec paramètres

Drupal 7

Drupal 8

mon_module

mon_module.routing.yml

mon_module_menu () {  
  $items['go/%/%'] = array(
    'page arguments' => array(1,2),
    'page callback' => 'run_go',
    'access arguments' =>
	   array('ma_permission'),
    'title' => 'le titre de ma page'
    'type' => MENU_CALLBACK
  );
  /* ... */
  return $items;
}
monmodule.go:
  path: 'go/{one}/{two}'
  defaults:
    _content: 
'\Drupal\mon_module\Controller\GoController::runGo'
    _title: 'le titre de ma page'
  requirements:
    _permission: 'ma_permission'

La route unitaire

Déclarer une page

Dans le fichier GoController.php
situé : mon_module/src/Controller

namespace Drupal\mon_module\Controller;
use Drupal\Core\Controller\ControllerBase;

class GoController extends ControllerBase {
	
    public function runGo($one,$two) {
        // je fabrique ma page

        return $theming_array;  
    }
} 

Le routing

Les arborescences

Arborescence

Classique

mon_module.links.menu.yml

menu.go_results:
  title: 'Resultat'
  parent: mon_module.go
  description: 'Resultats'
  route_name: mon_module.go_results
menu.go_results_more:
  title: 'Plus de Resultat'
  parent: mon_module.go_results
  description: 'Plus de Resultats'
  route_name: mon_module.go_results_more

Arborescence

Local task

mon_module.links.task.yml

mon_module.root:
    title: 'page de configuration'
    route_name: mon_module.conf
    base_route : mon_module.conf
mon_module.advanced:
    title: 'Configuration avancée'
    route_name: mon_module.more
    base_route : mon_module.conf

Et après ? 

Le thèming

Nouvelle conception

  • Remplacement de PHPTemplate par Twig
  • 1 entrée hook_theme déclare 1 template Twig
  • Aucune fonction ne retourne directement du HTML
  • "drupal_render" est au cœur de l'affichage

Preprocessing

  • hook_preprocess / hook_preprocess_HOOK
  • 4 niveaux d'interception
    • module / template / engin / theme
  • Altérations des theme_suggestions

Et après ? 

Entity_API

Côté code PHP : 

  • POO Avancée !
  • Annotations
  • Classes parentes & Interfaces
  • Héritage multiple ?

Changement de fonctionnement

  • Ctools + EntityAPI + Rules + DS
  • Objets techniques
  • Déclaration Yaml

Et après ? 

La gestion des configurations

Dans la théorie :

  • Récupération de Feature & Strongarm
  • Export en format Yaml

Dans la pratique :

  • Gestion d'état (A jour / surchargé)
  • Gestion de version (activ / stagging)
  • Workflow de montée de version

Conclusions

« Ceci est une Révolution »

« J'comprends rien à c'truc, j'peux pas l'faire à l'arrache ? »

« Il n'y aura qu'une seule voie, celle de la vertu »

S. Jobs

M. Drupespierre

Dev PHP du dimanche

Merci

De Drupal 7 à Drupal 8

By Damien Robert (DrDam)

De Drupal 7 à Drupal 8

Une introduction aux évolutions apportées par Drupal 8

  • 2,769