diff popups.module @ 0:76f9b43738f2

Popups 2.0-alpha5
author Franck Deroche <franck@defr.org>
date Fri, 31 Dec 2010 13:41:08 +0100
parents
children 4215c43e74eb c076d54409cb
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/popups.module	Fri Dec 31 13:41:08 2010 +0100
@@ -0,0 +1,451 @@
+<?php
+// $Id: popups.module,v 1.11.8.10 2009/03/21 00:57:15 starbow Exp $
+
+/**
+ * @file
+ * This module provides a hook_popups for links to be openned in an Ajax Popup Modal Dialog. 
+ */
+
+
+// **************************************************************************
+// CORE HOOK FUNCTIONS   ****************************************************
+// **************************************************************************
+
+/**
+ * Implementation of hook_menu().
+ *
+ * @return array of new menu items.
+ */
+function popups_menu() { 
+  
+  // Admin Settings.
+  $items['admin/settings/popups'] = array(
+    'page callback' => 'drupal_get_form',
+    'page arguments' => array('popups_admin_settings'),
+    'title' => 'Popups',
+    'access arguments' => array('administer site configuration'),
+    'description' => 'Configure the page-in-a-dialog behavior.',
+  ); 
+  
+  return $items;
+}
+
+/**
+ * Implementation of hook_init().
+ * 
+ * Look at the page path and see if popup behavior has been requested for any links in this page.
+ */
+function popups_init() {  
+  $popups = popups_get_popups();
+
+  if (variable_get('popups_always_scan', 0)) {
+    popups_add_popups();
+  }
+
+  foreach ($popups as $path => $popup_config) {
+    if ($path == $_GET['q']) {
+      popups_add_popups($popup_config);
+    }
+    elseif (strpos($path, '*') !== FALSE && drupal_match_path($_GET['q'], $path)) {
+      popups_add_popups($popup_config);
+    }
+  }  
+  
+  $render_mode = '';
+  if (isset($_SERVER['HTTP_X_DRUPAL_RENDER_MODE'])) {
+    $render_mode = $_SERVER['HTTP_X_DRUPAL_RENDER_MODE'];
+  }
+  
+  // Check and see if the page_override param is in the URL.
+  // Note - the magic happens here.
+  // Need to cache the page_override flag in the session, so it will effect
+  // the results page that follows a form submission.
+  if ($render_mode == 'json/popups') {
+    $_SESSION['page_override'] = TRUE;
+  }
+
+  // Move the page_override flag back out of the session.
+  if (isset($_SESSION['page_override'])) {
+    // This call will not return on form submission.
+    $content = menu_execute_active_handler();
+    
+    // The call did return, so it wasn't a form request, 
+    // so we are returning a result, so clear the session flag.
+    $override = $_SESSION['page_override'];
+    unset($_SESSION['page_override']);
+       
+    // Menu status constants are integers; page content is a string.
+    if (isset($content) && !is_int($content) && isset($override)) {
+      print popups_render_as_json($content); 
+      exit;  // Do not continue processing request in index.html.
+    }    
+  }
+  
+}
+
+/**
+ * Implementation of hook_form_alter().
+ * 
+ * Look at the form_id and see if popup behavior has been requested for any links in this form.
+ *
+ * @param form_array $form
+ * @param array $form_state
+ * @param str $form_id: 
+ */
+function popups_form_alter(&$form, $form_state, $form_id) {
+  // Add popup behavior to the form if requested.
+//  dsm($form_id);
+  $popups = popups_get_popups();
+  if (isset($popups[$form_id])) {
+    popups_add_popups($popups[$form_id]);
+  } 
+
+  // Alter the theme configuration pages, to add a per-theme-content selector. 
+  $theme = arg(4);
+  if ($form_id == 'system_theme_settings' && $theme) {
+    $form['popups'] = array(
+      '#type' => 'fieldset',
+      '#title' => t('Popup Settings'),
+      '#weight' => -2,
+    );
+    $form['popups']['popups_content_selector'] = array(
+      '#type' => 'textfield',
+      '#title' => t('Content Selector'),
+      '#default_value' => variable_get('popups_'. $theme .'_content_selector', _popups_default_content_selector()),
+      '#description' => t("jQuery selector to define the page's content area on this theme."),
+    ); 
+    $form['popups']['popups_theme'] = array(
+      '#type' => 'hidden',
+      '#value' => $theme,
+    ); 
+    $form['#submit'][] = 'popups_theme_settings_form_submit';
+  }
+}
+
+// **************************************************************************
+// UTILITY FUNCTIONS   ******************************************************
+// **************************************************************************
+
+/**
+ * Render the page contents in a custom json wrapper.
+ *
+ * @param $content: themed html.
+ * @return $content in a json wrapper with metadata.
+ */
+function popups_render_as_json($content) { 
+  $path = $_GET['q']; // Get current path from params.
+  return drupal_json(array(
+    'title' => drupal_get_title(),
+    'messages' => theme('status_messages'),
+    'path' => $path,
+    'content' => $content,
+    'js' => popups_get_js(),
+    'css' => popups_get_css(), 
+  ));
+}
+
+/**
+ * Get the added JS in a format that is readable by popups.js.
+ */
+function popups_get_js() {
+  $js = array_merge_recursive(drupal_add_js(), drupal_add_js(NULL, NULL, 'footer'));
+  $query_string = '?'. substr(variable_get('css_js_query_string', '0'), 0, 1);
+  
+  $popup_js = array();
+
+  foreach ($js as $type => $data) {
+    if (!$data) continue;
+    switch ($type) {
+      case 'setting':
+        // Why not just array_merge_recursive($data);
+        $popup_js['setting'] = call_user_func_array('array_merge_recursive', $data);
+        break;
+      case 'inline':
+        foreach ($data as $info) {
+          $popup_js['inline'][] = '<script type="text/javascript"' . ($info['defer'] ? ' defer="defer"' : '') . '>' . $info['code'] . "</script>\n";
+        }
+        break;
+      default:
+        foreach ($data as $path => $info) {
+          $popup_js[$type][$path] = '<script type="text/javascript"'. ($info['defer'] ? ' defer="defer"' : '') .' src="'. base_path() . $path . $query_string ."\"></script>\n";
+        }
+        break;
+    }
+  }
+
+  // A special exception, never include the popups settings in the JS array.
+  // ???
+//  unset($popup_js['setting']['popups']);
+
+  return $popup_js;
+}
+
+/**
+ * Get the added CSSS in a format that is readable by popups.js.
+ */
+function popups_get_css() {
+  $css = drupal_add_css();
+  $popup_css = array();
+
+  $query_string = '?'. substr(variable_get('css_js_query_string', '0'), 0, 1);
+
+  // Only process styles added to "all".
+  $media = 'all';
+  foreach ($css[$media] as $type => $files) {
+    if ($type == 'module') {
+      // Setup theme overrides for module styles.
+      $theme_styles = array();
+      foreach (array_keys($css[$media]['theme']) as $theme_style) {
+        $theme_styles[] = basename($theme_style);
+      }
+    }
+    foreach($css[$media][$type] as $file => $preprocess) {
+      // If the theme supplies its own style using the name of the module style, skip its inclusion.
+      // This includes any RTL styles associated with its main LTR counterpart.
+      if ($type == 'module' && in_array(str_replace('-rtl.css', '.css', basename($file)), $theme_styles)) {
+        // Unset the file to prevent its inclusion when CSS aggregation is enabled.
+        unset($css[$media][$type][$file]);
+        continue;
+      }
+      // Only include the stylesheet if it exists.
+      if (file_exists($file)) {
+        // If a CSS file is not to be preprocessed and it's a module CSS file, it needs to *always* appear at the *top*,
+        // regardless of whether preprocessing is on or off.
+        if ($type == 'module') {
+          $popup_css['module'][$file] = '<link type="text/css" rel="stylesheet" media="'. $media .'" href="'. base_path() . $file . $query_string .'" />'."\n";
+        }
+        // If a CSS file is not to be preprocessed and it's a theme CSS file, it needs to *always* appear at the *bottom*,
+        // regardless of whether preprocessing is on or off.
+        else if ($type == 'theme') {
+          $popup_css['theme'][$file] = '<link type="text/css" rel="stylesheet" media="'. $media .'" href="'. base_path() . $file . $query_string .'" />'."\n";
+        }
+        else {
+          $popup_css['unknown'][$file] = '<link type="text/css" rel="stylesheet" media="'. $media .'" href="'. base_path() . $file . $query_string .'" />'."\n";
+        }
+      }
+    }
+  }
+
+  return $popup_css;
+}
+
+
+/**
+ * Define hook_popups().
+ * Build the list of popup rules from all modules that implement hook_popups.
+ * 
+ * Retrieves the list of popup rules from all modules that implement hook_popups.
+ *
+ * @param $reset
+ *   (optional) If set to TRUE, forces the popup rule cache to reset.
+ * 
+ */
+function popups_get_popups($reset = FALSE) {
+  static $popups = NULL;
+  if (!isset($popups) || $reset) {
+    // Get popups hash out of cache.
+    if (!$reset && ($cache = cache_get('popups:popups')) && !empty($cache->data)) {
+      $popups = $cache->data;
+    }
+    else { 
+      // Call all hook_popups and cache results.
+      $popups = module_invoke_all('popups');
+      
+      // Invoke hook_popups_alter() to allow altering the default popups registry.
+      drupal_alter('popups', $popups);
+
+      // Save the popup registry in the cache.      
+      cache_set('popups:popups', $popups);
+    }  
+  }
+  return $popups;
+}
+
+/**
+ * Attach the popup behavior to the page.
+ * 
+ * The default behavoir of a popup is to open a form that will modify the original page.  
+ * The popup submits the form and reloads the original page with the resulting new content. 
+ * The popup then replaces the original page's content area with the new copy of that content area.
+ *
+ * @param array $rules: Array of rules to apply to the page or form, keyed by jQuery link selector.
+ *   See README.txt for a listing of the options, and popups_admin.module for examples. 
+ */
+function popups_add_popups($rules=NULL) { 
+  static $added = FALSE;
+  $settings = array('popups' => array());
+  
+  if (is_array($rules)) {
+    $settings['popups']['links'] = array();
+    foreach ($rules as $popup_selector => $options) { 
+      if (is_array($options)) {
+        $settings['popups']['links'][$popup_selector] = $options;
+      }
+      else {
+        $settings['popups']['links'][$options] = array();
+      }
+    }
+    if($added) {
+      drupal_add_js( $settings, 'setting' );
+    }
+  }
+  if (!$added) {
+    // Determing if we are showing the default theme or a custom theme.
+    global $custom_theme;
+    $theme = $custom_theme;
+    if (!$theme) {
+      $theme = variable_get('theme_default','none');
+    }
+    
+    drupal_add_js('misc/jquery.form.js');
+    drupal_add_css(drupal_get_path('module', 'popups') .'/popups.css');
+    drupal_add_js(drupal_get_path('module', 'popups') .'/popups.js');
+
+    // Allow skinning of the popup.
+    $skin = variable_get('popups_skin', 'Basic');
+    $skins = popups_skins();
+    if (!$skins[$skin]['css']) { // $skin == 'Unskinned'
+      // Look in the current theme for popups-skin.js
+      drupal_add_js(drupal_get_path('theme', $theme) . '/popups-skin.js');
+    }
+    else { // Get css and js from selected skin.
+      drupal_add_css($skins[$skin]['css']);
+      if (isset($skins[$skin]['js'])) {
+        drupal_add_js($skins[$skin]['js']);
+      }   
+    }
+        
+    $default_target_selector = variable_get('popups_'. $theme .'_content_selector', _popups_default_content_selector());
+    
+    $settings['popups']['originalPath'] = $_GET['q'];
+    $settings['popups']['defaultTargetSelector'] = $default_target_selector;
+    $settings['popups']['modulePath'] = drupal_get_path('module', 'popups');
+//    $settings['popups']['popupFinalMessage'] = variable_get('popups_popup_final_message', 1);
+    $settings['popups']['autoCloseFinalMessage'] = variable_get('popups_autoclose_final_message', 1);
+    drupal_add_js( $settings, 'setting' );
+    $added = TRUE;
+  }
+}
+
+ /**
+ * Retrieve all information from the popup skin registry.
+ *
+ * @param $reset
+ *   (optional) If TRUE, will force the the skin registry to reset.
+ * @see popups_popups_skins
+ */
+function popups_skins($reset = FALSE) {
+  static $skins = array();
+  if (empty($skins) || $reset) {
+    if (!$reset && ($cache = cache_get('popups:skins')) && !empty($cache->data)) {
+      $skins = $cache->data;
+    }
+    else {
+      // Create the popup skin registry (hook_popups_skins) and cache it.
+      $skins = module_invoke_all('popups_skins');
+      ksort($skins);  // Sort them alphabetically
+      cache_set('popups:skins', $skins, 'cache', CACHE_PERMANENT);
+    }
+  }
+  return $skins;
+}
+
+/**
+ * Implementation of hook_popups_skins.
+ * 
+ * This hook allows other modules to create additional custom skins for the
+ * popups module.
+ * 
+ * @return array
+ *   An array of key => value pairs suitable for inclusion as the #options in a
+ *   select or radios form element. Each key must be the location of at least a 
+ *   css file for a popups skin. Optionally can have a js index as well. Each 
+ *   value should be the name of the skin.
+ */
+function popups_popups_skins() {
+  $skins = array();
+  $skins_directory = drupal_get_path('module', 'popups') .'/skins';
+  $files = file_scan_directory($skins_directory, '\.css$');
+
+  foreach ($files as $file) {
+    $name = drupal_ucfirst($file->name);
+    $skins[$name]['css'] = $file->filename;
+    $js = substr_replace($file->filename, '.js', -4);
+    if (file_exists($js)) {
+      $skins[$name]['js'] = $js;
+    }
+  }
+  return $skins;
+}
+
+
+/**
+ * Returns the default jQuery content selector as a string.
+ * Currently uses the selector for Garland.  
+ * Sometime in the future I will change this to '#content' or '#content-content'.
+ */
+function _popups_default_content_selector() {
+  return 'div.left-corner > div.clear-block:last'; // Garland in Drupal 6.
+}
+
+// **************************************************************************
+// ADMIN SETTINGS   *********************************************************
+// **************************************************************************
+
+/**
+ * Form for the Popups Settings page.
+ *
+ */
+function popups_admin_settings() {
+  popups_add_popups();
+//  drupal_add_css(drupal_get_path('module', 'popups'). '/skins/blue/blue.css'); // temp
+  drupal_set_title("Popups Settings");
+  $form = array();
+
+  $form['popups_always_scan'] = array(
+    '#type' => 'checkbox',
+    '#title' => t('Scan all pages for popup links.'),
+    '#default_value' => variable_get('popups_always_scan', 0),
+  );
+  $form['popups_autoclose_final_message'] = array(
+    '#type' => 'checkbox',
+    '#title' => t('Automatically close final confirmation messages.'),
+    '#default_value' => variable_get('popups_autoclose_final_message', 1),
+  );
+
+  // Retrieve all available skins, forcing the registry to refresh.
+  $skins['Unskinned'] = array();
+  $skins += popups_skins(TRUE);
+  
+  $skin_options = drupal_map_assoc(array_keys($skins));
+  $form['popups_skins'] = array(
+    '#type' => 'fieldset',
+    '#title' => t('Skins'),
+    '#description' => t('Choose a skin from the list. After you save, click !here to test it out.', array('!here' => l('here', 'user', array('attributes' => array('class' => 'popups'))))),
+    '#collapsible' => TRUE,
+    '#collapsed' => FALSE,
+  );
+  $form['popups_skins']['popups_skin'] = array(
+    '#type' => 'radios',
+    '#title' => t('Available skins'),
+    '#default_value' => variable_get('popups_skin', 'Basic'),
+    '#options' => $skin_options,
+  );
+
+  
+  return system_settings_form($form);
+}
+
+/**
+ * popups_form_alter adds this submit handler to the theme pages.
+ * Set a per-theme jQuery content selector that gets passed into the js settings. 
+ *
+ * @param $form
+ * @param $form_state
+ */
+function popups_theme_settings_form_submit($form, &$form_state) {
+  $theme = $form_state['values']['popups_theme'];
+  $content_selector = $form_state['values']['popups_content_selector'];
+  variable_set('popups_'. $theme .'_content_selector', $content_selector);
+}
+