$info) {
$items['spaces_'. $type] = array(
'name' => $info['title'],
'description' => t('Sets a spaces context.'),
'callback' => 'spaces_init_context',
'callback arguments' => array($type),
'example' => 'my-space',
);
}
return $items;
}
/**
* Context prefix provider callback.
*/
function spaces_init_context($type, $sid) {
static $once;
if (empty($once)) {
$space = spaces_load($type, $sid, TRUE);
spaces_set_space($space);
$once = TRUE;
}
}
/**
* Implementation of hook_menu().
*/
function spaces_menu() {
$items = array();
$items['admin/build/spaces'] = array(
'title' => 'Spaces presets',
'description' => 'Create and configure spaces for your site.',
'page callback' => 'drupal_get_form',
'page arguments' => array('spaces_preset_default_form'),
'type' => MENU_NORMAL_ITEM,
);
$items['admin/build/spaces/presets'] = array(
'title' => 'Presets',
'page callback' => 'drupal_get_form',
'page arguments' => array('spaces_preset_default_form'),
'type' => MENU_DEFAULT_LOCAL_TASK,
'weight' => -1,
);
$items['admin/build/spaces/presets/add'] = array(
'title' => 'Add',
'page callback' => 'drupal_get_form',
'page arguments' => array('spaces_preset_form', 'add'),
);
$items['admin/build/spaces/presets/edit'] = array(
'page callback' => 'drupal_get_form',
'page arguments' => array('spaces_preset_form', 'edit'),
);
$items['admin/build/spaces/presets/export'] = array(
'title' => 'Export preset',
'page callback' => 'drupal_get_form',
'page arguments' => array('spaces_preset_export'),
);
$items['admin/build/spaces/presets/delete'] = array(
'page callback' => 'drupal_get_form',
'page arguments' => array('spaces_preset_delete_form', 5, 6),
);
$items['admin/build/spaces/presets/disable'] = array(
'page callback' => '_spaces_preset_disable_page',
);
$items['admin/build/spaces/presets/enable'] = array(
'page callback' => '_spaces_preset_enable_page',
);
foreach ($items as $path => $item) {
$items[$path]['access callback'] = 'user_access';
$items[$path]['access arguments'] = array('administer spaces');
$items[$path]['file'] = 'spaces_admin.inc';
if (!isset($item['type'])) {
$items[$path]['type'] = MENU_CALLBACK;
}
}
$items['spaces-access-denied'] = array(
'title' => 'Access denied',
'description' => 'Page callback for early-stack spaces router access denied.',
'page callback' => 'spaces_access_denied',
'page arguments' => array(),
'access callback' => FALSE,
'type' => MENU_CALLBACK,
);
return $items;
}
/**
* Implementation of hook_menu_alter().
*/
function spaces_menu_alter(&$items) {
$router_items = array(
'node/%node',
'node/%node/edit',
'user/%user/view',
'user/%user_uid_optional',
'user/%user_category/edit',
);
foreach (node_get_types('types', NULL, TRUE) as $type) {
$type_url_str = str_replace('_', '-', $type->type);
$router_items[] = 'node/add/'. $type_url_str;
}
foreach ($router_items as $path) {
if (isset($items[$path])) {
$arguments = isset($items[$path]['access arguments']) ? $items[$path]['access arguments'] : array();
$arguments[] = isset($items[$path]['access callback']) ? $items[$path]['access callback'] : NULL;
$items[$path]['access callback'] = 'spaces_menu_access';
$items[$path]['access arguments'] = $arguments;
}
}
}
/**
* Spaces menu access callback. Allows space types to manage menu
* access as related to their space workflow. See hook_menu_alter()
* for how the original menu access callback / argument gets passed
* to an altered item.
*/
function spaces_menu_access() {
$args = func_get_args();
$op = 'menu';
$object = NULL;
if (!empty($args)) {
$access_callback = array_pop($args);
if ($access_callback == 'node_access' && $args[0] == 'create') {
$object = new StdClass();
$object->type = $args[1];
$op = 'node';
}
else {
foreach ($args as $arg) {
if (is_object($arg)) {
$object = $arg;
if (isset($object->nid)) {
$op = 'node';
}
else if (isset($object->uid)) {
$op = 'user';
}
break;
}
}
}
}
$access = true;
$types = spaces_types();
// Check menu access for the active space type
if ($space = spaces_get_space()) {
$access = $access && $space->menu_access($op, $object);
unset($types[$space->type]);
}
// Run each non-active space type's menu access check
foreach ($types as $type => $info) {
$access = $access && call_user_func(array($info['class'], 'menu_access'), $op, $object, FALSE);
}
$standard_access = !empty($access_callback) ? call_user_func_array($access_callback, $args) : TRUE;
return $access && $standard_access;
}
/**
* Menu callback for early access_denied calls. Going through the a menu
* callback allows Drupal bootstrap to complete.
*/
function spaces_access_denied() {
drupal_access_denied();
exit;
}
/**
* Implementation of hook_help().
*/
function spaces_help($path, $arg) {
switch ($path) {
case 'admin/build/spaces/presets/export':
return "
". t('You can use exported presets in your modules by returning an array of presets in hook_spaces_presets()
.') ."
";
}
}
/**
* Implementation of hook_theme().
*/
function spaces_theme() {
$items = array();
$items['spaces_features_form'] =
$items['spaces_customize_item'] =
$items['spaces_preset_default_form'] =
$items['spaces_form_presets'] =
$items['spaces_block_customizer_settings_form'] = array(
'file' => 'spaces.theme.inc',
);
return $items;
}
/**
* Implementation of hook_user().
*/
function spaces_user($op, &$edit, &$account, $category = NULL) {
switch ($op) {
case 'view':
case 'form':
spaces_router('user', $account);
break;
}
}
/**
* Implementation of hook_nodeapi().
*/
function spaces_nodeapi(&$node, $op, $teaser = NULL, $page = NULL) {
if ($op == 'view' && $page && menu_get_object() === $node) {
spaces_router('node', $node);
}
if ($op == 'prepare') {
spaces_router('node', $node);
}
}
/**
* Implementation of hook_flush_caches().
*/
function spaces_flush_caches() {
spaces_features_map(NULL, TRUE);
return array();
}
/**
* Implementation of hook_spaces_settings().
*/
function spaces_spaces_settings() {
return array(
'home' => array(
'class' => 'space_setting_home',
'file' => drupal_get_path('module', 'spaces') .'/spaces.spaces.inc',
),
);
}
/**
* Implementation of hook_spaces_customizers().
*/
function spaces_spaces_customizers() {
return array(
'menu' => array(
'class' => 'space_customizer_menu',
'file' => drupal_get_path('module', 'spaces') .'/spaces.spaces.inc',
),
'block' => array(
'class' => 'space_customizer_block',
'file' => drupal_get_path('module', 'spaces') .'/spaces.spaces.inc',
),
);
}
/**
* Implementation of hook_views_api().
*/
function spaces_views_api() {
return array(
'api' => 2,
'path' => drupal_get_path('module', 'spaces') .'/includes',
);
}
/**
* Implementation of hook_features_api().
*/
function spaces_features_api() {
return array(
'spaces' => array(
'default_hook' => 'spaces_presets',
'default_file' => FEATURES_DEFAULTS_INCLUDED_COMMON,
'file' => drupal_get_path('module', 'spaces') .'/spaces.features.inc',
),
);
}
/**
* Implementation of hook_context_conditions().
*/
function spaces_context_conditions() {
$type_options = array();
foreach (spaces_types() as $type => $info) {
$type_options[$type] = $info['title'];
}
$items = array();
$items['spaces_type'] = array(
'#title' => t('Spaces type'),
'#type' => 'checkboxes',
'#options' => $type_options,
'#description' => t('Set this context when the selected space types are active.'),
);
return $items;
}
/**
* SPACES API =========================================================
*/
/**
* Interface for space objects.
*/
interface space {
/**
* Core API-related functions
*/
function __construct($type, $sid = NULL, $is_active = FALSE);
function save();
function delete();
/**
* Method that defines the feature options available for this space
* type.
*
* @return array
* A FormAPI suitable options array of feature options.
*/
function feature_options();
/**
* Method that allows the space type to add user links provided to
* visitors in a space.
*
* @return array()
* Array of links for use with theme_links.
*/
function user_links();
/**
* Method that allows the space type to provide to administrative links
* for users who pass the $space->admin_access() check. It is the
* responsibility of the caller to perform this access check.
*
* @return array()
* Array of links for use with theme_links.
*/
function admin_links();
/**
* Method that provides a space-type specific check for whether the
* provided feature is accessible by the current user.
*
* @param $feature
* The feature identifier string.
*
* @return bool
* TRUE if the user has access to this feature. FALSE if not.
*/
function feature_access($feature = NULL);
/**
* Method that provides a space-type specific check for whether the
* the space can be administered by the current user.
*
* @return bool
* TRUE if the user has admin access to this space. FALSE if not.
*/
function admin_access();
/**
* Master router method that allows the space type to define routing
* workflow rules. Currently called at the following hooks:
*
* hook_menu(), where $may_cache == FALSE
* hook_nodeapi(), where $op == 'view'
* hook_form_alter(), when editing a node
* hook_user(), where $op == 'view'
*
* @param $op
* The current hook from which the router is being called.
* Can be one of the following: 'menu', 'node view', 'node form',
* 'user view'
* @param $object
* The object relevant to the current $op called. e.g. when
* $op == 'node view', $object is the node object.
* @param $is_active
* Boolean for whether this router has been called as part of a
* fully instantiated space object. If FALSE, the router should
* not assume the space has been fully constructed and should take
* the appropriate actions as necessary.
*
* @return bool
* TRUE to allow the user to pass through. FALSE to return a
* drupal_access_denied() page to the user.
*/
function router($op, $object = NULL, $is_active = TRUE);
/**
* Redirect handler that abstracts redirect logic and allows space
* types to define their own definitions of various spaces concepts.
*
* @param $op
* The page being redirected to. Currently only $op value is 'home'
*/
function redirect($op = 'home');
/**
* Allows the space type to implement custom form options on the
* feature/preset form.
*
* @return array
* A FormAPI element array. It will be included as a child element
* of the master space feature form.
*/
function form();
/**
* Validate handler for the space type's custom form.
*
* @param $values
* The submitted values.
*/
function validate($values);
/**
* Custom submit handler for the space type's custom form. For
* example, allows the space type to process values in preparation
* for spaces_save().
*
* @param $values
* The submitted values.
*
* @return array
* An array of values.
*/
function submit($values);
/**
* Views filter callback that allows the space type to filter views
* when it is the current active space.
*
* @param $query
* The views query object.
*/
function views_filter(&$query, $base_table = '', $relationship = '');
/**
* Allows the space type to take additional action when enforcing a
* preset against the current space.
*
* @param $preset
* A spaces preset definition.
*/
function preset_enforce($preset);
}
/**
* Interface for space settings.
*/
interface space_setting {
function __construct($id = NULL);
function form($space, $value = array());
function validate($space, $value);
function submit($space, $value);
}
/**
* Interface for space customizers.
*/
interface space_customizer {
function form($space, $feature);
function validate($space, $feature, $value);
function submit($space, $feature, $value);
}
/**
* Implementation of hook_context_active_contexts_alter().
*/
function spaces_context_active_contexts_alter(&$contexts) {
$space = spaces_get_space();
foreach ($contexts as $identifier => $context) {
if (!empty($context->block)) {
spaces_customizers();
space_customizer_block::customize($space, $identifier, $context->block);
}
}
}
/**
* Load a space.
*
* @param $type
* The type of the space to be loaded. Must be one of the keys in the
* array returned by spaces_types().
* @param $sid
* The id of the space to be loaded. If omitted, a "prototype" space
* will be constructed.
* @param $is_active
* Optional boolean flag for whether this space is active or not.
* Defaults to FALSE.
*
* @return
* The requested space object or FALSE if something went wrong.
*/
function spaces_load($type, $sid = NULL, $is_active = FALSE) {
$types = spaces_types();
if (isset($types[$type])) {
if (!empty($types[$type]['file']) && is_file($types[$type]['file'])) {
require_once $types[$type]['file'];
}
$class = $types[$type]['class'];
// Create a new space object
$space = new $class($type, $sid, $is_active);
// Initialize various space variables
$space->type = $type;
$space->features = array();
$space->settings = array();
$space->customizer = array();
// Initialize space specific settings if $sid is provided
if ($sid) {
$space->sid = $sid;
// Load the PURL modifier
if ($modifier = purl_load(array('provider' => "spaces_{$type}", 'id' => $sid))) {
$space->purl = $modifier['value'];
}
// Load features
$result = db_query("SELECT id, value FROM {spaces_features} WHERE sid = %d AND type = '%s' ORDER BY weight ASC", $sid, $type);
while ($row = db_fetch_object($result)) {
$space->features[$row->id] = $row->value;
}
// Load settings
$result = db_query("SELECT id, value FROM {spaces_settings} WHERE sid = %d AND type = '%s'", $sid, $type);
while ($row = db_fetch_object($result)) {
$space->settings[$row->id] = unserialize($row->value);
}
// Load customizer & preset
$row = db_fetch_object(db_query("SELECT customizer, preset FROM {spaces} WHERE sid = %d AND type = '%s'", $space->sid, $space->type));
$space->customizer = $row ? unserialize($row->customizer) : array();
// Enforce preset or use default if not found
$valid_presets = spaces_presets($type);
$default_presets = variable_get('spaces_default_presets', array());
if ($row && isset($valid_presets[$row->preset])) {
$space->preset = $row->preset;
}
else if (isset($default_presets[$type])) {
$space->preset = $default_presets[$type];
}
if (!empty($space->preset)) {
spaces_preset_enforce($space);
}
}
return $space;
}
return false;
}
/**
* Saves a space object's feature/setting values.
*
* @param $space
* The space object to save.
*
* @return
* Returns TRUE on success, FALSE on failure.
*/
function spaces_save($space) {
if (!empty($space->sid) && !empty($space->type)) {
// Force the preset if it is has changed
$existing = spaces_load($space->type, $space->sid);
if ($space->preset != $existing->preset) {
spaces_preset_enforce($space, TRUE);
}
else {
spaces_preset_enforce($space);
}
// Update features
db_query("DELETE FROM {spaces_features} WHERE sid = %d AND type = '%s'", $space->sid, $space->type);
$valid_features = spaces_features($space->type);
$weight = -10;
foreach ($space->features as $feature => $value) {
if (isset($valid_features[$feature])) {
$values = array($space->sid, $space->type, $feature, $value, $weight);
db_query("INSERT INTO {spaces_features} (sid, type, id, value, weight) VALUES (%d, '%s', '%s', '%s', %d)", $values);
$weight++;
}
}
// Update settings
db_query("DELETE FROM {spaces_settings} WHERE sid = %d AND type = '%s'", $space->sid, $space->type);
// Build list of valid settings including those associated with features.
$valid_settings = spaces_settings($space->type);
foreach (spaces_feature_settings() as $feature => $settings) {
foreach ($settings as $id => $class) {
$valid_settings[$id] = $class;
}
}
foreach ($space->settings as $setting => $value) {
if (isset($valid_settings[$setting]) && !empty($value)) {
$value = serialize($value);
$values = array($space->sid, $space->type, $setting, $value);
db_query("INSERT INTO {spaces_settings} (sid, type, id, value) VALUES (%d, '%s', '%s', '%s')", $values);
}
}
// Update preset & customizer
$exists = db_result(db_query("SELECT count(sid) FROM {spaces} WHERE sid = %d AND type = '%s'", $space->sid, $space->type));
if ($exists) {
db_query("UPDATE {spaces} SET preset = '%s', customizer = '%s' WHERE sid = %d AND type = '%s'", $space->preset, serialize($space->customizer), $space->sid, $space->type);
}
else {
db_query("INSERT INTO {spaces} (sid, type, preset, customizer) VALUES(%d, '%s', '%s', '%s')", $space->sid, $space->type, $space->preset, serialize($space->customizer));
}
// Save context prefix if space type allows prefix customization
$types = spaces_types();
$save_purl = isset($types[$space->type]['custom purl']) && $types[$space->type]['custom purl'];
if ($space->purl && $save_purl) {
// We need to concatenate the type/sid so that collisions between space types do not occur.
$modifier = array(
'provider' => 'spaces_'. $space->type,
'id' => $space->sid,
'value' => $space->purl,
);
purl_save($modifier);
}
// Allow space type to do its own saving
$space->save();
return true;
}
return false;
}
/**
* Deletes a space object's records in the database.
*
* @param $space
* The space object to delete.
*
* @return
* Returns TRUE for now.
*/
function spaces_delete($space) {
// Remove all features and settings
db_query("DELETE FROM {spaces} WHERE sid = %d AND type = '%s'", $space->sid, $space->type);
db_query("DELETE FROM {spaces_features} WHERE sid = %d AND type = '%s'", $space->sid, $space->type);
db_query("DELETE FROM {spaces_settings} WHERE sid = %d AND type = '%s'", $space->sid, $space->type);
// Delete the purl modifier
$modifier = array('provider' => 'spaces_'. $space->type, 'id' => $space->sid);
purl_delete($modifier);
// Allow space type to do its own deleting
$space->delete();
return true;
}
/**
* Enforces a spaces preset.
*/
function spaces_preset_enforce(&$space, $force = FALSE) {
$presets = spaces_presets($space->type);
if (isset($space->preset) && isset($presets[$space->preset])) {
$preset = $presets[$space->preset]['preset'];
// Enforce features, settings, customizer
$keys = array('features', 'settings', 'customizer');
foreach ($keys as $key) {
if (isset($preset[$key]) && is_array($preset[$key])) {
// If forced, completely replace any existing settings.
if ($force) {
$space->{$key} = $preset[$key];
}
// Otherwise, attempt to preserve user-chosen settings.
else {
foreach ($preset[$key] as $k => $v) {
if (!empty($preset['locked'][$key][$k]) || !isset($space->{$key}[$k])) {
$space->{$key}[$k] = $v;
}
}
}
}
}
// Weights are a bit trickier
// Sorting features in a preset according to their weights
if (isset($preset['weights']) && is_array($preset['weights'])) {
// Exclude any weight values that don't correspond to entries in the features array.
$preset['weights'] = array_intersect_key($preset['weights'], $space->features);
// Fill in excessively high weight entries for any features that don't exist in
// the weights array.
$i = 0;
foreach (array_diff_key($space->features, $preset['weights']) as $feature => $value) {
$preset['weights'][$feature] = 1000 + $i;
$i++;
}
// Ensure the keys line up to begin with
ksort($preset['weights']);
ksort($space->features);
array_multisort($preset['weights'], $space->features, SORT_NUMERIC);
}
// Type-specific presets
$space->preset_enforce($preset);
}
}
/**
* Invokes hook_spaces_types() to gather an array of space types and
* associated classes.
*
* Implementing modules should provide an associate array in
* hook_spaces_types() of the following format:
*
* return array(
* $id => array(
* 'class' => $space_class,
* 'title' => $space_type_name,
* 'purl modifier' => $bool,
* 'file' => $direct_path_to_file
* ),
* );
*
* Where:
*
* $id: A unique string identifier for this space type. e.g. "og"
* $space_class: The PHP class to construct for this space. e.g. "spaces_og"
* $space_type_name: The human-readable name of your space type. e.g. "Group space"
* $bool: TRUE or FALSE for whether spaces should provide a UI for
* users to provide a custom URL modifier for each space. If FALSE,
* the implementing module should implement
* hook_purl_modifiers() in order to provide PURL modifiers programatically.
* $direct_path_to_file: the relative path from the Drupal root to the file that
* defines the PHP class. (Optional)
*
* @param $reset
* Optional reset flag for clearing the static cache.
*
* @return
* An array of space types where $key => $value corresponds to the space type => space class.
*/
function spaces_types($reset = false) {
static $spaces_types;
if (!isset($spaces_types) || $reset) {
$spaces_types = module_invoke_all('spaces_types');
}
return $spaces_types;
}
/**
* Gather an array of spaces presets from the DB.
*
* Spaces presets allow for easier space instantiation by the user in the UI.
* A spaces preset lets you define which features are enabled and diabled and
* pre-configure settings for a space instance, all of which a user would
* normally have to take care of in the UI. The result of a spaces_preset
* is a radio button on the UI that the user can select in order to use
* a defined preset for the instance of a space they are about to create.
* Spaces presets can be built in the UI and then exported to code by
* implementing hook_spaces_presets which should define and return the
* array exported from the UI on admin/build/space
*
* @param $type
* Optional space type to filter results by.
* @param $include_disabled
* Optional flag to return all presets, including disabled ones.
* @param $reset
* Optional reset flag for clearing the static cache.
*
* @return
* An array of space types where $key => $value corresponds to the space type => space class.
*/
function spaces_presets($type = NULL, $include_disabled = FALSE, $reset = FALSE) {
static $presets;
if (!isset($presets)) {
$presets = array();
$disabled = variable_get('spaces_disabled_presets', array());
// Collect presets provided by modules in code
foreach (module_implements('spaces_presets') as $module) {
$items = module_invoke($module, 'spaces_presets');
foreach ($items as $id => $preset) {
$preset['disabled'] = isset($disabled[$preset['type']][$id]);
$preset['storage'] = SPACES_PRESET_DEFAULT;
$presets[$preset['type']][$id] = $preset;
}
}
// Allow modules to alter presets.
drupal_alter('spaces_presets', $presets);
$result = db_query("SELECT * FROM {spaces_presets}");
while ($row = db_fetch_object($result)) {
$preset = array(
'name' => check_plain($row->name),
'description' => check_plain($row->description),
'type' => check_plain($row->type),
'preset' => unserialize($row->value),
'disabled' => isset($disabled[$row->type][$row->id]),
'storage' => isset($presets[$row->type][$row->id]) ? SPACES_PRESET_OVERRIDDEN : SPACES_PRESET_NORMAL,
);
$presets[$row->type][$row->id] = $preset;
}
}
$return = $presets;
if (!$include_disabled) {
foreach (array_keys($return) as $preset_type) {
foreach ($return[$preset_type] as $id => $preset) {
if ($preset['disabled']) {
unset($return[$preset_type][$id]);
}
}
}
}
if ($type) {
return isset($return[$type]) ? $return[$type] : array();
}
return $return;
}
/**
* Wrapper function around spaces_set_space(). Retrieves the current
* active space.
*/
function spaces_get_space() {
return spaces_set_space();
}
/**
* Sets the specified space as the current active space. Returns the
* active space if no space is provided.
*
* @param $space
* The space object to set as the active space. Optional.
* @param $reset
* Optional flag to reset the static cache.
*
* @return
* The active space object or FALSE if there is no active space.
*/
function spaces_set_space($space = NULL, $reset = FALSE) {
static $current_space;
if (!isset($current_space) || $reset) {
$current_space = $space;
if ($space) {
// Context integration with spaces_type setter.
context_set_by_condition('spaces_type', $space->type);
}
}
return is_object($current_space) ? $current_space : FALSE;
}
/**
* Wrapper around implementations of $space->router. Provides
* additional intelligence, including a global killswitch and routing
* when no spaces are active.
*
* @param $op
* The current "hook" or "hook op" identifier for the $space->router
* to act on.
* @param $object
* Optional object to pass to the $space->router.
*/
function spaces_router($op, $object = NULL) {
// Check global killswitch
if (spaces_router_get()) {
$types = spaces_types();
// Run the router for the active space
if ($space = spaces_get_space()) {
$space->router($op, $object);
unset($types[$space->type]);
}
// Run each non-active space type's router
foreach ($types as $type => $info) {
call_user_func(array($info['class'], 'router'), $op, $object, FALSE);
}
}
}
/**
* Wrapper around spaces_router_get().
*/
function spaces_router_set($status) {
return spaces_router_get($status, TRUE);
}
/**
* Sets a static variable that is used to disable spaces routing
* altogether -- e.g. for install/update scripts, migrations, etc.
*
* @param $enabled
* Optional boolean for enabling/disabling spaces routing.
* @param $reset
* Optional boolean for resetting the static cache.
*
* @return
* Returns a boolean for whether routing is enabled/disabled.
*/
function spaces_router_get($enabled = 1, $reset = FALSE) {
static $status;
if (!isset($status) || $reset) {
$status = $enabled;
}
return $status;
}
/**
* Retrieve all available features.
*
* @param $type
* Optional type flag by which to filter available features.
* @param $reset
* Optional boolean flag for resetting the static cache.
*
* @return
* Keyed array of potential features.
*/
function spaces_features($type = NULL, $reset = FALSE) {
static $spaces_features;
if (!isset($spaces_features) || $reset) {
$spaces_features = array('all' => array(), 'common' => array());
$features = features_get_features();
foreach ($features as $feature) {
if (module_exists($feature->name) && !empty($feature->info['spaces'])) {
if (!empty($feature->info['spaces']['types']) && is_array($feature->info['spaces']['types'])) {
foreach ($feature->info['spaces']['types'] as $t) {
$spaces_features[$t][$feature->name] = $feature;
}
}
else {
$spaces_features['common'][$feature->name] = $feature;
}
$spaces_features['all'][$feature->name] = $feature;
}
}
foreach (array_keys($spaces_features) as $t) {
if ($t != 'all' && $t != 'common') {
$spaces_features[$t] = array_merge($spaces_features[$t], $spaces_features['common']);
}
}
}
if ($type) {
return !empty($spaces_features[$type]) ? $spaces_features[$type] : $spaces_features['common'];
}
return $spaces_features['all'];
}
/**
* Retrieve settings provided by a given feature.
*
* @param $feature
* Optional feature name whose settings should be returned.
* @param $reset
* Boolean flag to reset static cache.
*
* @return
* An array of instantiated setting classes.
*/
function spaces_feature_settings($feature = NULL, $reset = FALSE) {
static $feature_settings;
if (!isset($feature_settings) || $reset) {
$feature_settings = array();
// Exclude feature-specific settings here
$features = spaces_features();
foreach (module_implements('spaces_settings') as $module) {
if (isset($features[$module])) {
$settings = module_invoke($module, 'spaces_settings');
foreach ($settings as $id => $info) {
// Load any setting includes before instantiating its class.
if (is_array($info)) {
if (isset($info['file']) && is_file($info['file'])) {
require_once($info['file']);
}
if (isset($info['class']) && class_exists($info['class'])) {
$class = $info['class'];
$setting = new $class();
}
}
// This format is deprecated -- included for backwards compatibility.
else if (is_string($info) && class_exists($info)) {
$setting = new $info();
}
else if (is_object($info)) {
$setting = $info;
}
$feature_settings[$module][$id] = $setting;
}
}
}
}
if ($feature) {
return isset($feature_settings[$feature]) ? $feature_settings[$feature] : array();
}
return $feature_settings;
}
/**
* Retrieve all available settings.
*
* @param $type
* Optional space type to return a subset of settings that only apply
* to the given space type.
* @param $reset
* Optional boolean flag for resetting the static cache.
*
* @return
* Keyed array of potential settings.
*/
function spaces_settings($type = NULL, $reset = FALSE) {
static $spaces_settings;
if (!isset($spaces_settings) || $reset) {
$spaces_settings = array('all' => array(), 'common' => array());
// Exclude feature-specific settings.
// Use spaces_feature_settings() to retrieve these.
$settings = array();
$features = spaces_features();
foreach (module_implements('spaces_settings') as $module) {
if (!isset($features[$module])) {
$settings = array_merge($settings, module_invoke($module, 'spaces_settings'));
}
}
foreach ($settings as $setting_name => $info) {
// Load any setting includes before instantiating its class.
if (is_array($info)) {
if (isset($info['file']) && is_file($info['file'])) {
require_once($info['file']);
}
if (isset($info['class']) && class_exists($info['class'])) {
$class = $info['class'];
$setting = new $class();
}
}
// This format is deprecated -- included for backwards compatibility.
else if (is_string($info) && class_exists($info)) {
$setting = new $info();
}
else if (is_object($info)) {
$setting = $info;
}
if (!empty($setting->types)) {
foreach ($setting->types as $t) {
if (!isset($spaces_settings[$t])) {
$spaces_settings[$t] = array();
}
$spaces_settings[$t][$setting_name] = $setting;
}
}
else {
$spaces_settings['common'][$setting_name] = $setting;
}
$spaces_settings['all'][$setting_name] = $setting;
}
foreach (array_keys($spaces_settings) as $t) {
if ($t != 'all' && $t != 'common') {
$spaces_settings[$t] = array_merge($spaces_settings[$t], $spaces_settings['common']);
}
}
}
if ($type) {
return !empty($spaces_settings[$type]) ? $spaces_settings[$type] : $spaces_settings['common'];
}
return $spaces_settings['all'];
}
/**
* Retrieve all available customizers.
*
* @param $reset
* Optional boolean flag for resetting the static cache.
*
* @return
* Keyed array of customizers.
*/
function spaces_customizers($reset = FALSE) {
static $spaces_customizers;
if (!isset($spaces_customizers) || $reset) {
$customizers = module_invoke_all('spaces_customizers');
foreach ($customizers as $customizer_name => $info) {
// Load any includes before instantiating its class.
if (is_array($info)) {
if (isset($info['file']) && is_file($info['file'])) {
require_once($info['file']);
}
if (isset($info['class']) && class_exists($info['class'])) {
$class = $info['class'];
$customizer = new $class();
}
}
// This format is deprecated -- included for backwards compatibility.
else if (is_string($info) && class_exists($info)) {
$customizer = new $info();
}
else if (is_object($info)) {
$customizer = $info;
}
$spaces_customizers[$customizer_name] = $customizer;
}
}
return $spaces_customizers;
}
/**
* Caches a map of feature component => feature name
*/
function spaces_features_map($type = NULL, $reset = FALSE) {
static $map;
if (!isset($map) || $reset) {
$map = array();
$cache = cache_get('spaces_map', 'cache');
if ($cache && !$reset) {
$map = $cache->data;
}
else {
// Generate component to feature map based on feature info files
$features = spaces_features();
foreach ($features as $feature_name => $feature) {
foreach ($feature->info['features'] as $component => $items) {
if (is_array($items)) {
foreach ($items as $item) {
if (!isset($map[$component][$item])) {
$map[$component][$item] = $feature_name;
}
}
}
}
}
// Go through contexts and add relevant context components in as well
$contexts = context_enabled_contexts();
$components = array('node', 'views');
if (!empty($map['context'])) {
foreach ($map['context'] as $identifier => $feature_name) {
$context = $contexts[$identifier];
if (!empty($context)) {
foreach ($components as $component) {
if (!empty($context->{$component})) {
foreach ($context->{$component} as $item) {
if (!isset($map[$component][$item])) {
$map[$component][$item] = $feature_name;
}
}
}
}
}
}
}
cache_set('spaces_map', $map, 'cache');
}
}
if (!empty($type)) {
return isset($map[$type]) ? $map[$type] : array();
}
return $map;
}
/**
* Returns an array of items of a given type that belong to a given feature.
*/
function spaces_features_items($type, $feature) {
$features = spaces_features();
if (!empty($features[$feature]) && !empty($features[$feature]->info['features'][$type])) {
return $features[$feature]->info['features'][$type];
}
return array();
}
/**
* Implementation of hook_features_menu_links_alter().
* This function is deprecated and is no longer a real drupal_alter() callback.
* It is invoked directly from spaces_preprocess_page().
*/
function spaces_features_menu_links_alter(&$links) {
$space = spaces_get_space();
if ($space) {
// Load up the spaces menu links
spaces_customizers();
space_customizer_menu::customize($space, $links);
// Retrieve the menu cache for a path to feature mapping
$map = spaces_features_map('menu');
// Sort the menu by feature weight & hide any items for features
// that are not available (this **should**) be handled by access
// control on the links' respective menu callbacks, but awkward
// permissioning (e.g. user 1) makes it nice for these to be
// hidden manually.
$weights = array_flip(array_keys($space->features));
$weighted = array();
foreach ($links as $k => $item) {
$feature = '';
if (!empty($map[$item['href']])) {
$feature = $map[$item['href']];
}
if (!empty($feature) && $space->feature_access($feature)) {
// Retrieve path > feature > weight
$links[$k]['#weight'] = $weights[$feature];
}
else {
unset($links[$k]);
}
}
uasort($links, 'element_sort');
}
}
/**
* Preset options form that can be reused by implementing modules.
*
* @param $space
* A space object.
*
* @return
* A FormAPI array structure.
*/
function spaces_form_presets($space) {
$presets = spaces_presets($space->type);
if ($presets) {
// If there's one preset, then select it and don't show the form.
if (count($presets) == 1) {
$form['preset'] = array('#type' => 'value', '#value' => array_shift(array_keys($presets)));
}
else {
$default_presets = variable_get('spaces_default_presets', array());
if (isset($space->preset) && !empty($presets[$space->preset])) {
$default_preset = $space->preset;
}
else if (isset($default_presets[$space->type])) {
$default_preset = $default_presets[$space->type];
}
else {
$default_preset = NULL;
}
// Radios for presets
$form = array(
'#tree' => false,
'#theme' => 'spaces_form_presets',
);
$form['preset'] = array(
'#title' => t('Preset'),
'#type' => 'radios',
'#required' => true,
'#options' => array(),
'#default_value' => $default_preset,
);
$form['info'] = array();
foreach ($presets as $id => $preset) {
$form['preset']['#options'][$id] = $preset['name'];
$form['info'][$id] = array(
'#type' => 'item',
'#title' => $preset['name'],
'#description' => $preset['description'],
);
}
}
return $form;
}
return array();
}
/**
* Menu admin access wrapper.
*/
function spaces_admin_access($type = NULL, $op = NULL) {
$return = false;
$space = spaces_get_space();
if ($space && $type == $space->type) {
$return = $space->admin_access();
}
if ($op == 'features') {
return $return && (user_access('administer spaces') || user_access('configure spaces features'));
}
return $return;
}
/**
* Menu feature access wrapper.
*/
function spaces_feature_access($feature = NULL) {
if ($space = spaces_get_space()) {
return user_access('access content') && $space->feature_access($feature);
}
return user_access('access content');
}
/**
* A mild abstraction of hook_menu() items that can be used by
* implementing modules to embed/graft relevant spaces items into the
* menu tree.
*
* @param $space
* A space object.
* @param $local_tasks
* Optional boolean flag for whether the items are fully rendered as
* local tasks.
* @param $path_prefix
* A path to prefix the menu item paths by.
*
* @return
* An array of menu items.
*/
function spaces_active_space_menu($type, $local_tasks = FALSE, $path_prefix = '') {
$types = spaces_types();
$arg_count = !empty($path_prefix) ? count(explode('/', $path_prefix)) : 0;
$path_prefix = !empty($path_prefix) ? $path_prefix .'/' : '';
$spaces_path = drupal_get_path('module', 'spaces');
$items[$path_prefix. 'spaces/features'] = array(
'title' => 'Features',
'page arguments' => array('spaces_features_form'),
'access arguments' => array($type, 'features'),
'type' => MENU_LOCAL_TASK,
'weight' => 1,
);
$items[$path_prefix. 'spaces/features/%'] = array(
'title' => 'Features',
'page arguments' => array('spaces_customize_form', NULL, 2 + $arg_count),
'access arguments' => array($type, 'features'),
'type' => MENU_LOCAL_TASK,
'weight' => 1,
);
foreach ($items as $path => $item) {
$items[$path]['page callback'] = 'drupal_get_form';
$items[$path]['access callback'] = 'spaces_admin_access';
$items[$path]['file'] = 'spaces_admin.inc';
$items[$path]['file path'] = $spaces_path;
}
return $items;
}
/**
* THEME FUNCTIONS ====================================================
*/
/**
* Generates an array of admin links for the current space suitable
* for use in theme_links().
*/
function spaces_admin_links($space = NULL) {
$space = !isset($space) ? spaces_get_space() : $space;
if ($space && $space->admin_access()) {
$links = $space->admin_links();
drupal_alter('spaces_admin_links', $links, $space);
return $links;
}
return array();
}
/**
* Generates an array of user links for the current space suitable
* for use in theme_links().
*/
function spaces_user_links($space = NULL) {
$space = !isset($space) ? spaces_get_space() : $space;
if ($space) {
$links = $space->user_links();
drupal_alter('spaces_user_links', $links, $space);
return $links;
}
return array();
}
/**
* theme_block() preprocessor.
*/
function spaces_preprocess_block(&$vars) {
$space = spaces_get_space();
if (!empty($vars['block'])) {
spaces_customizers();
space_customizer_block::customize_subject($space, $vars['block']);
}
}
/**
* theme_page() preprocessor.
*/
function spaces_preprocess_page(&$vars) {
if (variable_get('menu_primary_links_source', 'primary-links') === 'features' && !empty($vars['primary_links'])) {
spaces_features_menu_links_alter($vars['primary_links']);
}
$space = spaces_get_space();
if ($space) {
if ($links = spaces_admin_links()) {
$vars['space_admin_links'] = theme('links', $links);
}
if ($links = spaces_user_links()) {
$vars['space_user_links'] = theme('links', $links);
}
if (!empty($space->title)) {
$vars['space_title'] = l($space->title, '');
}
$vars['body_classes'] .= " spaces-{$space->type}";
}
}