You are here

blockgroup.module in Block Group 7

Same filename and directory in other branches
  1. 8 blockgroup.module
  2. 7.2 blockgroup.module

Add block groups to block configuration page

File

blockgroup.module
View source
<?php

/**
 * @file
 * Add block groups to block configuration page
 */
define('BLOCKGROUP_MAX_BLOCKGROP_NAME_LENGTH_UI', 27);

/**
 * Implements hook_boot().
 *
 * Blockgroup works by injecting regions into themes at runtime using
 * hook_system_info_alter. If this hook is invoked when the blockgroup module
 * is not (yet) loaded, the regions defined by block groups are not injected
 * into the theme. This may happen when either system_rebuild_theme_data or
 * list_themes is called before drupal is fully bootstrapped. In order to avoid
 * this problem we implement hook_boot. As a result blockgroup gets loaded very
 * early in the bootstrapping phase and the all important
 * blockgroup_system_info_alter is present in any case.
 *
 * Probably hook_system_info_alter should also become a bootstrap-hook in
 * core.
 *
 * @see:
 * - bootstrap_hooks().
 * - hook_system_info_alter().
 * - list_themes().
 * - system_rebuild_theme_data().
 * - http://drupal.org/node/534594
 */
function blockgroup_boot() {

  // Intentionally left empty (see above).
}

/**
 * Implements hook_menu().
 */
function blockgroup_menu() {
  $default_theme = variable_get('theme_default', 'bartik');
  $items['admin/structure/block/manage/blockgroup/%blockgroup/delete'] = array(
    'title' => 'Delete block group',
    'page callback' => 'drupal_get_form',
    'page arguments' => array(
      'blockgroup_delete_confirm',
      5,
    ),
    'access arguments' => array(
      'administer block groups',
    ),
    'type' => MENU_CALLBACK,
    'context' => MENU_CONTEXT_NONE,
    'file' => 'blockgroup.admin.inc',
  );
  $items['admin/structure/block/groupadd'] = array(
    'title' => 'Add group',
    'page callback' => 'drupal_get_form',
    'page arguments' => array(
      'blockgroup_add_form',
    ),
    'access arguments' => array(
      'administer block groups',
    ),
    'type' => MENU_LOCAL_ACTION,
    'file' => 'blockgroup.admin.inc',
  );
  foreach (list_themes() as $key => $theme) {
    if ($key != $default_theme) {
      $items['admin/structure/block/list/' . $key . '/groupadd'] = array(
        'title' => 'Add group',
        'page callback' => 'drupal_get_form',
        'page arguments' => array(
          'blockgroup_add_form',
        ),
        'access arguments' => array(
          'administer block groups',
        ),
        'type' => MENU_LOCAL_ACTION,
        'file' => 'blockgroup.admin.inc',
      );
    }
  }
  return $items;
}

/**
 * Implements hook_permission().
 */
function blockgroup_permission() {
  return array(
    'administer block groups' => array(
      'title' => t('Administer block groups'),
    ),
  );
}

/**
 * Implements hook_features_api().
 */
function blockgroup_features_api() {
  return array(
    'blockgroup' => array(
      'name' => 'Block Group',
      'file' => drupal_get_path('module', 'blockgroup') . '/blockgroup.features.inc',
      'default_hook' => 'default_blockgroups',
      'feature_source' => TRUE,
    ),
  );
}

/**
 * Implements hook_element_info().
 */
function blockgroup_element_info() {
  $types['blockgroup_pullup'] = array(
    '#pre_render' => array(
      'blockgroup_pullup',
    ),
    '#base' => NULL,
    '#key' => NULL,
  );
  return $types;
}

/**
 * A pre_render callback for pulling in elements from another part of the page.
 */
function blockgroup_pullup($build) {
  $key = $build['#key'];
  return !empty($build['#base'][$key]) ? $build['#base'][$key] : array();
}

/**
 * Implements hook_block_info().
 */
function blockgroup_block_info() {
  $blocks = array();

  // Build up a mapping region_key => title.
  $blockgroups = blockgroup_list();
  foreach ($blockgroups as $delta => $title) {
    $blocks[$delta] = array(
      'info' => t('Block group: @title', array(
        '@title' => $title,
      )),
      'cache' => DRUPAL_NO_CACHE,
    );
  }
  return $blocks;
}

/**
 * Implements hook_block_configure().
 */
function blockgroup_block_configure($delta = '') {
  $blockgroup = blockgroup_load($delta);
  $form['title'] = array(
    '#type' => 'textfield',
    '#title' => t('Group title'),
    '#maxlength' => 64,
    '#description' => t('A brief description of your block group. Used on the <a href="@overview">Blocks administration page</a>.', array(
      '@overview' => url('admin/structure/block'),
    )),
    '#weight' => -19,
  );
  $form['machine_name'] = array(
    '#type' => 'machine_name',
    '#title' => t('Machine name'),
    '#maxlength' => BLOCKGROUP_MAX_BLOCKGROP_NAME_LENGTH_UI,
    '#description' => t('A unique name. It must only contain lowercase letters, numbers and hyphens.'),
    '#machine_name' => array(
      'exists' => 'blockgroup_edit_blockgroup_name_exists',
      'source' => array(
        'settings',
        'title',
      ),
      'replace_pattern' => '[^a-z0-9-]+',
      'replace' => '-',
    ),
  );
  if (!empty($blockgroup->delta)) {
    $form['title']['#default_value'] = $blockgroup->title;
    $form['machine_name']['#default_value'] = $blockgroup->delta;
    $form['machine_name']['#disabled'] = TRUE;
  }
  return $form;
}

/**
 * Returns whether a block group already exists.
 *
 * @see blockgroup_block_configure()
 */
function blockgroup_edit_blockgroup_name_exists($value) {
  $blockgroups = blockgroup_list();
  return array_key_exists($value, $blockgroups);
}

/**
 * Given a block id and a forest structure, return the root.
 */
function _blockgroup_tree_get_top($bid, $tree, $seen = array()) {
  if (isset($seen[$bid])) {

    // Return NULL when a cycle was detected.
    return NULL;
  }
  if (!isset($tree[$bid])) {

    // Return bid if we reached the top.
    return $bid;
  }

  // Apply function to children.
  $seen[$bid] = $bid;
  return _blockgroup_tree_get_top($tree[$bid], $tree, $seen);
}

/**
 * Return an array of block ids which are descentants to the given bid.
 */
function _blockgroup_branch_flatten($bid, $branches) {
  $result = array();
  $result[] = $bid;
  if (isset($branches[$bid])) {
    foreach ($branches[$bid] as $branch) {
      $result = array_merge($result, _blockgroup_branch_flatten($branch, $branches));
    }
  }
  return $result;
}

/**
 * Implements hook_module_implements_alter().
 */
function blockgroup_module_implements_alter(&$implementations, $hook) {
  if ($hook == 'block_list_alter' || $hook == 'page_alter') {
    $implementation = $implementations['blockgroup'];
    unset($implementations['blockgroup']);
    $implementations['blockgroup'] = $implementation;
  }
}

/**
 * Implements hook_block_list_alter().
 *
 * Remove all blocks contained in invisible block groups.
 */
function blockgroup_block_list_alter(&$blocks) {
  global $theme_key;

  // Assumption 1: Regions which are not injected by blockgroup are considered
  // system regions. Each system region is assigned a negative fake block id.
  $all_regions = system_region_list($theme_key);
  $system_regions = array_diff_key($all_regions, blockgroup_region_list());
  $system_regions = array_combine(array_keys($system_regions), range(-1, -count($system_regions)));

  // Assumption 2: recursion only occures through block groups. There are no
  // other modules doing the same thing than we do.
  $group_regions = array();
  foreach ($blocks as $bid => $block) {
    if ($block->module === 'blockgroup') {
      $group_regions[blockgroup_get_region($blocks[$bid]->delta)] = $bid;
    }
  }

  // Construct tree (mapping bid -> parent bid).
  $tree = array();
  foreach ($blocks as $bid => $block) {
    if (isset($system_regions[$block->region])) {
      $tree[$bid] = $system_regions[$block->region];
    }
    elseif (isset($group_regions[$block->region])) {
      $tree[$bid] = $group_regions[$block->region];
    }
  }

  // Remove all entries from parent list where top > 0 (i.e. not attached to a
  // system region).
  foreach ($tree as $bid => $pbid) {
    $top = _blockgroup_tree_get_top($bid, $tree);
    if (_blockgroup_tree_get_top($bid, $tree) > 0) {
      unset($tree[$bid]);
    }
  }

  // Construct branch list (mapping base -> children).
  $branches = array();
  foreach ($tree as $bid => $pbid) {
    $branches[$pbid][] = $bid;
  }

  // Purge all branches containing only block groups and no content blocks.
  foreach (array_keys($branches) as $bid) {
    $bids = _blockgroup_branch_flatten($bid, $branches);
    $content_blocks = array_diff($bids, $group_regions);
    $content_blocks = array_filter($content_blocks, function ($bid) {
      return $bid > 0;
    });
    if (empty($content_blocks)) {
      $tree = array_diff_key($tree, drupal_map_assoc($bids));
    }
  }

  // Unset all blocks which are not in the tree.
  $blocks = array_intersect_key($blocks, $tree);
}

/**
 * Implements hook_block_view().
 *
 * Only supply a placeholder block. Use a pre_render callback which will
 * populate the block with the content of the proper block-group region.
 */
function blockgroup_block_view($delta) {
  $block['content'] = array(
    '#type' => 'blockgroup_pullup',
    '#key' => blockgroup_get_region($delta),
  );
  return $block;
}

/**
 * Implements hook_page_alter().
 *
 * Loop through every region and all blocks on the page. Whenever a block group
 * is encountered, set the #base parameter of the blockgroup_pullup element to
 * the page.
 */
function blockgroup_page_alter(&$page) {
  $page['#blockgroups'] = array();
  foreach ($page as $region => &$blocks) {
    if (is_array($blocks)) {
      foreach ($blocks as $key => &$build) {

        // If we encounter a blockgroup-block, set its pullup base to
        // $page['#blockgroups'].
        if (isset($build['#type']) && $build['#type'] == 'blockgroup_pullup') {
          $build['#base'] =& $page['#blockgroups'];
        }
      }
    }
  }

  // Move all blockgroup regions into $page['#blockgroups'] in order to prevent
  // certain themes from messing with them.
  $blockgroup_regions = array_intersect_key($page, blockgroup_region_list());
  foreach (array_keys($blockgroup_regions) as $region) {
    $page['#blockgroups'][$region] =& $page[$region];
    unset($page[$region]);
  }
}

/**
 * Implements hook_theme().
 */
function blockgroup_theme() {
  return array(
    'block__blockgroup__default' => array(
      'render element' => 'elements',
      'template' => 'block-blockgroup-default',
    ),
  );
}

/**
 * Implements hook_preprocess_HOOK().
 *
 * Prepend theme_hook_suggestions with block__blockgroup__default such that the
 * default blockgroup template is picked up only if there is no more specific
 * template for this particular block group.
 */
function blockgroup_preprocess_block(&$variables) {
  if ($variables['block']->module === 'blockgroup') {
    array_unshift($variables['theme_hook_suggestions'], 'block__blockgroup__default');
    $variables['region'] = blockgroup_get_region($variables['block']->delta);

    // Allow the use of a region--blockgroup.tpl.php template that will affect
    // all block group regions.
    $variables['theme_hook_suggestions'][] = 'region__blockgroup';
  }
}

/**
 * Implements hook_preprocess_HOOK().
 *
 * Insert 'blockgroup' into classes array for regions defined by this module.
 */
function blockgroup_preprocess_region(&$variables) {
  $blockgroup_regions = blockgroup_region_list();
  if (isset($blockgroup_regions[$variables['region']])) {
    $variables['classes_array'][] = 'blockgroup';
  }
}

/**
 * Implements hook_system_info_alter().
 *
 * Inject defined block groups as regions into all enabled themes.
 */
function blockgroup_system_info_alter(&$info, $file, $type) {
  if ($type == 'theme') {
    $info['regions'] += blockgroup_region_list();
  }
}

/**
 * Implements hook_FORM_ID_alter().
 *
 * Add delete-link to the main block administration form for blockgroups.
 */
function blockgroup_form_block_admin_display_form_alter(&$form, &$form_state) {
  $blockgroups = blockgroup_list();
  foreach ($blockgroups as $delta => $title) {
    $region_key = blockgroup_get_region($delta);
    $region_anchor = 'region-' . $region_key;
    $block_key = 'blockgroup_' . $delta;
    $block_anchor = 'block-' . $region_key;

    // This is not a bug, I want to use region_key here (because of block_key uses an underscore).
    // Add go-to-block link
    $rtb_options = array(
      'fragment' => $block_anchor,
      'attributes' => array(
        'name' => $region_anchor,
      ),
    );
    $rtb_link = l(t('Go to block'), $_GET['q'], $rtb_options);
    $form['block_regions']['#value'][$region_key] .= ' ' . $rtb_link;

    // Add go-to-region link
    $btr_options = array(
      'fragment' => $region_anchor,
      'attributes' => array(
        'name' => $block_anchor,
      ),
    );
    $btr_link = l(t('Go to region'), $_GET['q'], $btr_options);
    $form['blocks'][$block_key]['info']['#markup'] .= ' ' . $btr_link;

    // Add delete link to block
    $form['blocks'][$block_key]['delete'] = array(
      '#type' => 'link',
      '#title' => t('delete'),
      '#href' => 'admin/structure/block/manage/blockgroup/' . $delta . '/delete',
    );
  }
}

/**
 * Return a list of block groups.
 */
function blockgroup_list() {
  $blockgroups =& drupal_static(__FUNCTION__);
  if (!isset($blockgroups)) {
    $blockgroups = db_select('block', 'b')
      ->fields('b', array(
      'delta',
      'title',
    ))
      ->condition('module', 'blockgroup')
      ->execute()
      ->fetchAllKeyed();
  }
  return $blockgroups;
}

/**
 * Return a list of regions created by block groups.
 */
function blockgroup_region_list() {
  $regions =& drupal_static(__FUNCTION__);
  if (!isset($regions)) {
    $regions = array();
    foreach (blockgroup_list() as $delta => $title) {
      $regions[blockgroup_get_region($delta)] = t('Block group: @title', array(
        '@title' => $title,
      ));
    }
  }
  return $regions;
}

/**
 * Return the block group given its machine_name.
 */
function blockgroup_load($delta) {
  return block_load('blockgroup', $delta);
}

/**
 * Insert new block group into database.
 */
function blockgroup_add($blockgroup) {

  // Insert block-records for all themes with default values.
  $query = db_insert('block')
    ->fields(array(
    'pages',
    'title',
    'module',
    'theme',
    'delta',
  ));
  foreach (list_themes() as $key => $theme) {
    if ($theme->status) {
      $query
        ->values(array(
        'pages' => '',
        'title' => $blockgroup->title,
        'module' => 'blockgroup',
        'theme' => $theme->name,
        'delta' => $blockgroup->delta,
      ));
    }
  }
  $query
    ->execute();
  blockgroup_rebuild_data();
}

/**
 * Remove the given block group form the database.
 */
function blockgroup_delete($blockgroup) {
  db_delete('block')
    ->condition('module', 'blockgroup')
    ->condition('delta', $blockgroup->delta)
    ->execute();
  db_delete('block_role')
    ->condition('module', 'blockgroup')
    ->condition('delta', $blockgroup->delta)
    ->execute();
  blockgroup_rebuild_data();
}

/**
 * Clear caches and rebuild blockgroup and theme data.
 */
function blockgroup_rebuild_data() {
  drupal_static_reset('blockgroup_list');
  drupal_static_reset('blockgroup_region_list');
  cache_clear_all();
  system_rebuild_theme_data();
  drupal_theme_rebuild();
}

/**
 * Returns the region name defined by the given blockgroup.
 *
 * @param string $delta
 *   The machine name of the blockgroup.
 *
 * @returns string
 *   The machine name of the region.
 */
function blockgroup_get_region($delta) {
  return 'blockgroup-' . $delta;
}

Functions

Namesort descending Description
blockgroup_add Insert new block group into database.
blockgroup_block_configure Implements hook_block_configure().
blockgroup_block_info Implements hook_block_info().
blockgroup_block_list_alter Implements hook_block_list_alter().
blockgroup_block_view Implements hook_block_view().
blockgroup_boot Implements hook_boot().
blockgroup_delete Remove the given block group form the database.
blockgroup_edit_blockgroup_name_exists Returns whether a block group already exists.
blockgroup_element_info Implements hook_element_info().
blockgroup_features_api Implements hook_features_api().
blockgroup_form_block_admin_display_form_alter Implements hook_FORM_ID_alter().
blockgroup_get_region Returns the region name defined by the given blockgroup.
blockgroup_list Return a list of block groups.
blockgroup_load Return the block group given its machine_name.
blockgroup_menu Implements hook_menu().
blockgroup_module_implements_alter Implements hook_module_implements_alter().
blockgroup_page_alter Implements hook_page_alter().
blockgroup_permission Implements hook_permission().
blockgroup_preprocess_block Implements hook_preprocess_HOOK().
blockgroup_preprocess_region Implements hook_preprocess_HOOK().
blockgroup_pullup A pre_render callback for pulling in elements from another part of the page.
blockgroup_rebuild_data Clear caches and rebuild blockgroup and theme data.
blockgroup_region_list Return a list of regions created by block groups.
blockgroup_system_info_alter Implements hook_system_info_alter().
blockgroup_theme Implements hook_theme().
_blockgroup_branch_flatten Return an array of block ids which are descentants to the given bid.
_blockgroup_tree_get_top Given a block id and a forest structure, return the root.

Constants

Namesort descending Description
BLOCKGROUP_MAX_BLOCKGROP_NAME_LENGTH_UI @file Add block groups to block configuration page