THERE IS A SNIPPET FOR THAT!
10 handverlesene Drupal snippets
Was ist ein Snippet?
Ein kleines wiederverwendbares Code-Stück
Kleines Programmierbeispiel
Mini-Modul
Beispiel
"REDIRECT USER AFTER LOGIN"
Text
/**
* Implements hook_user_login().
*/
function MY_MODULE_user_login(&$edit, $account) {
// Don't redirect on password reset.
$current_menu_item = menu_get_item();
if ($current_menu_item['path'] == 'user/reset/%/%/%') {
return;
}
// Redirect user to "node/35" after the login.
$_GET['destination'] = 'node/35';
}
Snippet online: http://dropbucket.org/node/746
Aber da gibts doch auch ein Modul?!
-
https://www.drupal.org/project/login_destination
- > 500 Lines of Code
- 1 Datenbank Tabelle
- 5 Seiten mit Konfiguration
- Vorteile des Moduls
- Keine Programmierskills nötig
- Anpassung über UI
- Maintained
Modul oder Snippet?
It depends.
Vorteile von Snippets
-
Nimm dir nur den Teil raus den du benötigst
-
Exakt anpassbar an die jeweiligen Anforderungen
-
“Everything in code”
-
Am Ende schlankere Drupal Instanz mit weniger Modulen.
Wo findet man Snippets?
-
Drupal core, Contrib Module, “Examples” Modul
Warum Snippets online teilen?
- Feedback von anderen
- Alternative Lösungswege
- Probleme von diesem Lösungsweg
- "Wartung" / Maintainence der Snippets
- Einfacher Weg etwas zurückzugeben
10 SnippetS
Wie erstelle ich ein eigenes Token?
Problem
[term:root] funktioniert nur wenn der Term nicht selbst der Root-Term ist.
Lösung
Neues Token [term:root-or-current] selbst implementieren.
(oder konkret: Ein "besseres" [term:root] token implementieren.
HOOK_TOKEN_inFO
/**
* Implements hook_token_info().
*/
function MY_MODULE_token_info() {
$info['tokens']['term']['root-or-current'] = array(
'type' => 'text',
'name' => t('Root or current term'),
'description' => t('Root of term or current term if it is already the root'),
);
return $info;
}
hook_tokens
/**
* Implements hook_tokens().
*/
function MY_MODULE_tokens($type, $tokens, array $data = array(), array $options = array()) {
$sanitize = !empty($options['sanitize']);
$replacements = array();
// Our token acts on "term" entities
if ($type == 'term' && !empty($data['term'])) {
$term = $data['term'];
foreach ($tokens as $name => $original) {
switch ($name) {
// The name matches our token!
case 'root-or-current':
$parents = taxonomy_get_parents_all($term->tid);
$root_term = end($parents);
$replacements[$original] = $sanitize ?
check_plain($root_term->name) : $root_term->name;
break;
}
}
}
return $replacements;
}
Wie definiere ich einen Zusätzlichen "View-mode"?
Problem
Zusätzlich zu "Full content" und "Teaser" will ich einen weiteren "View Mode" verwenden.
Lösung
hook_entity_info_alter implementieren
HOOK_entity_inFO_alter
/**
* Implements hook_entity_info_alter().
*/
function YOUR_MODULE_NAME_entity_info_alter(&$entity_info) {
$entity_info['node']['view modes']['custom_teaser'] = array(
'label' => t('Custom teaser'),
// should custom settings for this view mode enabled by default
'custom settings' => FALSE,
);
}
BONUS: Eigenes .tpl FÜR View-mode
/**
* Implements hook_preprocess_node().
*/
function YOUR_MODULE_NAME_preprocess_node(&$variables) {
if ($variables['view_mode'] == 'custom_teaser') {
// eg. "node--article--custom-teaser.tpl.php" will be used if available.
$variables['theme_hook_suggestions'][] = 'node__' . $variables['type'] . '__custom_teaser';
}
}
Wie definiere ich neue date-formats in Code?
Problem
Ich will ein neues Datums Format verwenden (Beispiel: Datum soll im Format "1. Dezember" ausgegeben werden.)
Lösung
- hook_date_format_types - "Format Type"
- hook_date_formats - "Format String"
- "Format Type" soll "Format String" verwenden
hook_date_format_types
/**
* Implements hook_date_format_types().
*/
function hook_date_format_types() {
return array(
'my_format' => t('My format'),
);
}
Drupal Beischeid geben, dass wir einen neuen Format-Type wollen.
hook_date_formats
/**
* Implements hook_date_formats().
*/
function hook_date_formats() {
$formats = array();
$formats[] = array(
'type' => 'my_format',
'format' => 'j. F', // eg "1. Dezember", http://php.net/manual/de/function.date.php
'locales' => array(),
);
return $formats;
}
Neuen Format-String definieren.
Date Type mit Format Verbinden
// in hook_install die variable setzen
variable_set('date_format_my_format', 'j. F');
// oder in der settings.php hardcoden
$conf['date_format_my_format'] = 'j. F';
// oder in einem Feature exportieren
Beides verbinden durch setzen einer Variable.
Bei "access denied" auf die login Seite Leiten
Problem
Bei "Access denied" soll auf die Login Seite weitergeleitet werden
Lösung
Eigenes menu-item implementieren, das bei Bedarf weiterleitet.
HOOK_MENU
/**
* Implements hook_menu().
*/
function MYMODULE_menu() {
$items = array();
// custom access denied menu item (redirects to login)
$items['access-denied'] = array(
'page callback' => 'MYMODULE_access_denied',
'access callback' => TRUE,
);
return $items;
}
Page Callback
function MYMODULE_access_denied() {
global $user;
if ($user->uid) {
return MENU_ACCESS_DENIED;
}
// The code in drupal_get_destination() doesn't preserve any query string
// on 403 pages, so reproduce the part we want here.
$path = $_GET['destination'];
$query = drupal_http_build_query(
drupal_get_query_parameters(NULL, array('q', 'destination'))
);
if ($query != '') {
$path .= '?' . $query;
}
$destination = array('destination' => $path);
$url = url('user/login', array('query' => $destination, 'absolute' => TRUE));
header('Location: ' . $url , TRUE, 302);
drupal_exit();
}
Use the new menu-item
// Tell Drupal to use the new menu-item as a 403 page.
// For example like this:
$conf['site_403'] = 'access-denied';
Wie definiere ich ein Pseudo Field (Inkl. Bonus)?
Problem
Bei jeden Node sollen zusätzliche Daten dargestellt werden (ohne dafür ein Feld anzulegen)
Lösung
5 hooks und 1 callback
First, the Callback
/**
* Getter to return the phone number.
*/
function MY_MODULE_phone($node) {
return '089 / 12345' . $node->nid;
}
HOOK_NODE_Load
/**
* Implements hook_node_load().
*/
function MY_MODULE_node_load($nodes, $types) {
if (!in_array('article', $types)) {
return;
}
foreach ($nodes as $node) {
// add the project number to each node
$node->phone = MY_MODULE_phone($node);
}
}
Add the property to the node object on load.
hook_field_extra_fields
/**
* Implements hook_field_extra_fields().
*/
function MY_MODULE_field_extra_fields() {
$extra = array();
$extra['node']['article']['display']['phone'] = array(
'label' => t('Phone number'),
'description' => t('Phone number'),
'weight' => 10,
);
return $extra;
}
Make it available in the display settings of the node type.
HOOK_NODE_VIEW
/**
* Implements hook_node_view
*/
function MY_MODULE_node_view($node, $view_mode, $langcode) {
if ($node->type != 'article') {
return;
}
$node->content['phone'] = array(
'#type' => 'item',
'#title' => t('Phone number: '),
'#markup' => $node->phone,
);
}
Add it to the content array when the node is viewed.
hook_entity_property_info_alter
/**
* Implements hook_entity_property_info_alter().
*
* Note: This hook is provided by Entity API.
*/
function MY_MODULE_entity_property_info_alter(&$info) {
$info['node']['bundles']['article']['properties']['phone'] = array(
'type' => 'text',
'label' => t('Phone number'),
'sanitized' => TRUE,
'getter callback' => 'MY_MODULE_phone',
);
}
BONUS: Make it available for Entity API (Rules / Search API).
HOOK_VIEWS_DATA_ALTER
/**
* Implements hook_views_data_alter().
*
* Note: This hook is provided by Views.
*/
function MY_MODULE_views_data_alter(&$data){
// Entity API seems to add a handler, but at a different place.
$data['node']['phone'] = $data['entity_node']['phone'];
}
BONUS: Make it available for Views.
Wie Definiere einen eigenen "Daten Filter" für DIE SEARCH API?
Problem
Daten sollen vor dem Indizieren verändert werden, zB bestimmte Nodes sollen nicht indiziert werden.
Lösung
1 hook und 1 Plugin
hook_search_api_alter_callback
/**
* Implements hook_search_api_alter_callback_info().
*/
function MYMODULE_search_api_alter_callback_info() {
$callbacks['callback_nid_filter'] = array(
'name' => t('NID filter'),
'description' => t('Exclude NIDs from being indexed.'),
'class' => 'SearchApiAlterNIDFilter',
'weight' => 100,
);
return $callbacks;
}
Definiere das neue Callback.
The Callback Class
class SearchApiAlterNIDFilter extends SearchApiAbstractAlterCallback {
public function supportsIndex(SearchApiIndex $index) {
return $index->getEntityType() === 'node';
}
// Implement alterItems
public function alterItems(array &$items) {
$nids = array(41,52,102);
foreach($nids as $nid) {
if (isset($items[$nid])) {
unset($items[$nid]);
}
}
}
}
(version with config form on dropbucket.org)
files[] = includes/callback_nid_filter.inc
In MYMODULE.info
erlauben DIe eigenen Comments zu löschen
Problem
Menschen sollen ihre eigenen Kommentare löschen dürfen.
Lösung
Custom access callback für comment/%/delete
hook_menu_alter
/**
* Implements hook_menu_alter().
*/
function MY_MODULE_menu_alter(&$items) {
$items['comment/%/delete']['access callback'] = 'MY_MODULE_comment_delete_access';
$items['comment/%/delete']['access arguments'] = array(1);
}
Change the access callback for the "delete comment" menu item.
function MY_MODULE_comment_delete_access($cid) {
global $user;
if (!user_is_logged_in()) {
return FALSE;
}
// Allow admin to delete any comments.
if (user_access('administer comments')){
return TRUE;
}
// Load the comment.
$comment = comment_load($cid);
// If user is comment author ..
if ($comment->uid == $user->uid) {
// .. check if comment mode for this node type is flat.
$node = node_load($comment->nid);
$comment_mode = variable_get('comment_default_mode_' . $node->type, COMMENT_MODE_THREADED);
if ($comment_mode == COMMENT_MODE_FLAT) {
// Comment mode is flat and user is author of the comment.
return TRUE;
}
}
return FALSE;
}
The new access callback for the "delete comment" menu item.
Und AUs.
- Diese Snippets → http://dropbucket.org/list/1561
- Alle meine Snipptes → http://dropbucket.org/valderama
- Alle anderen Snippets finden sie auf → http://google.de
Kontakt: jenner@formundcode.de