|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file
|
|
|
|
* Defines the multi-page ingest form and any relevant hooks and functions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if the given configuration can be used to display the ingest form.
|
|
|
|
*
|
|
|
|
* @param array $configuration
|
|
|
|
* The list of key/value pairs of configuration.
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
* TRUE if the give configuration defines one or more form steps, FALSE
|
|
|
|
* otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_can_display_ingest_form(array $configuration) {
|
|
|
|
$form_state = array();
|
|
|
|
islandora_ingest_form_init_form_state_storage($form_state, $configuration);
|
|
|
|
$form_steps = islandora_ingest_form_get_form_steps($form_state);
|
|
|
|
// Forget the stubbed steps for the remainder of this request.
|
|
|
|
drupal_static_reset('islandora_ingest_form_get_steps');
|
|
|
|
return count($form_steps) > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Ingest form build function.
|
|
|
|
*
|
|
|
|
* Initializes the form state, and builds the initial list of steps, excutes
|
|
|
|
* the current step.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $configuration
|
|
|
|
* An associative array of configuration values that are used to build the
|
|
|
|
* list of steps to be executed, including:
|
|
|
|
* - id: The PID with which the object should be created.
|
|
|
|
* - namespace: The PID namespace in which the object should be created.
|
|
|
|
* (id is used first, if it is given).
|
|
|
|
* - label: The initial label for the object. Defaults to "New Object".
|
|
|
|
* - collections: An array of collection PIDs, to which the new object should
|
|
|
|
* be related.
|
|
|
|
* - models: An array of content model PIDs, to which the new object might
|
|
|
|
* subscribe
|
|
|
|
* - parent: The parent of the child to be ingested. This is needed for XACML
|
|
|
|
* to correctly apply the parent's POLICY to children.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The form definition of the current step.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form(array $form, array &$form_state, array $configuration) {
|
|
|
|
try {
|
|
|
|
islandora_ingest_form_init_form_state_storage($form_state, $configuration);
|
|
|
|
return islandora_ingest_form_execute_step($form, $form_state);
|
|
|
|
}
|
|
|
|
catch (Exception $e) {
|
|
|
|
watchdog(
|
|
|
|
'islandora',
|
|
|
|
'Exception during ingest form processing with Message: "@exception", and Trace: @trace',
|
|
|
|
array('@exception' => $e->getMessage(), '@trace' => $e->getTraceAsString()),
|
|
|
|
WATCHDOG_ERROR
|
|
|
|
);
|
|
|
|
drupal_set_message($e->getMessage(), 'error');
|
|
|
|
return array(array(
|
|
|
|
'#markup' => l(t('Back'), 'javascript:window.history.back();', array('external' => TRUE))));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initializes the form_state storage for use in the ingest multi-page forms.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $configuration
|
|
|
|
* A list of key value pairs that are used to build the list of steps to be
|
|
|
|
* executed.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_init_form_state_storage(array &$form_state, array $configuration) {
|
|
|
|
if (empty($form_state['islandora'])) {
|
|
|
|
$configuration['models'] = isset($configuration['models']) ? $configuration['models'] : array();
|
|
|
|
// Make sure the models actually exist.
|
|
|
|
foreach ($configuration['models'] as $key => $model) {
|
|
|
|
if (!islandora_object_load($model)) {
|
|
|
|
unset($configuration['models'][$key]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Required for step hooks.
|
|
|
|
$form_state['islandora'] = array(
|
|
|
|
'step_id' => NULL,
|
|
|
|
'objects' => $configuration['objects'],
|
|
|
|
'shared_storage' => $configuration,
|
|
|
|
'step_storage' => array(),
|
|
|
|
);
|
|
|
|
// No need to persist the 'objects' within the configuration.
|
|
|
|
unset($configuration['objects']);
|
|
|
|
// Must be called after $form_state['islandora'] is initialized, otherwise,
|
|
|
|
// the values in 'islandora' would not be availible to the step hooks.
|
|
|
|
$form_state['islandora']['step_id'] = islandora_ingest_form_get_first_step_id($form_state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the first step ID.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* The 'id' of the very first step.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_first_step_id(array &$form_state) {
|
|
|
|
$steps = islandora_ingest_form_get_steps($form_state);
|
|
|
|
$keys = array_keys($steps);
|
|
|
|
return array_shift($keys);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the last step ID.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* The 'id' of the very last step.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_last_step_id(array &$form_state) {
|
|
|
|
$steps = islandora_ingest_form_get_steps($form_state);
|
|
|
|
$keys = array_keys($steps);
|
|
|
|
return array_pop($keys);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the given/current step.
|
|
|
|
*
|
|
|
|
* If the current step is not defined it's assumed that all steps have executed.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param string $step_id
|
|
|
|
* The ID of the step. The current step is returned if no step ID is given.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The given/current step if found, NULL otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_step(array &$form_state, $step_id = NULL) {
|
|
|
|
$step_id = isset($step_id) ? $step_id : islandora_ingest_form_get_current_step_id($form_state);
|
|
|
|
$steps = islandora_ingest_form_get_steps($form_state);
|
|
|
|
if (isset($step_id) && isset($steps[$step_id])) {
|
|
|
|
return $steps[$step_id];
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the next step.
|
|
|
|
*
|
|
|
|
* If the current step is not defined it's assumed that all steps have executed.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step
|
|
|
|
* The step relative to the result, if not provided the current step is used.
|
|
|
|
*
|
|
|
|
* @return array|null
|
|
|
|
* The next step if found, NULL otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_next_step(array &$form_state, array $step = NULL) {
|
|
|
|
$step = isset($step) ? $step : islandora_ingest_form_get_step($form_state);
|
|
|
|
$next_step_id = islandora_ingest_form_get_next_step_id($form_state, $step['id']);
|
|
|
|
return isset($next_step_id) ? islandora_ingest_form_get_step($form_state, $next_step_id) : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the previous step.
|
|
|
|
*
|
|
|
|
* If the current step is not defined it's assumed that all steps have executed.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step
|
|
|
|
* The step relative to the result, if not provided the current step is used.
|
|
|
|
*
|
|
|
|
* @return array|null
|
|
|
|
* The next step if found, NULL otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_previous_step(array &$form_state, array $step = NULL) {
|
|
|
|
$step = isset($step) ? $step : islandora_ingest_form_get_step($form_state);
|
|
|
|
$previous_step_id = islandora_ingest_form_get_previous_step_id($form_state, $step['id']);
|
|
|
|
return isset($previous_step_id) ? islandora_ingest_form_get_step($form_state, $previous_step_id) : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the ID of the current step.
|
|
|
|
*
|
|
|
|
* If the current step is not defined it's assumed that all steps have executed.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* The step ID.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_current_step_id(array &$form_state) {
|
|
|
|
return $form_state['islandora']['step_id'];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the ID of the next step.
|
|
|
|
*
|
|
|
|
* If the current step is not defined it's assumed that all steps have executed.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param string $step_id
|
|
|
|
* The ID of the step relative to the result, if not provided the current
|
|
|
|
* step_id is used.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* The next step ID if found, NULL otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_next_step_id(array &$form_state, $step_id = NULL) {
|
|
|
|
$step_id = isset($step_id) ? $step_id : islandora_ingest_form_get_current_step_id($form_state);
|
|
|
|
$step_ids = array_keys(islandora_ingest_form_get_steps($form_state));
|
|
|
|
$index = array_search($step_id, $step_ids);
|
|
|
|
$count = count($step_ids);
|
|
|
|
if ($index !== FALSE && ++$index < $count) {
|
|
|
|
return $step_ids[$index];
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the ID of the previous step.
|
|
|
|
*
|
|
|
|
* If the current step is not defined it's assumed that all steps have executed.
|
|
|
|
* In such cases the last step will be returned.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param string $step_id
|
|
|
|
* The ID of the step relative to the result, if not provided the current
|
|
|
|
* step_id is used.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* The previous step ID if found, NULL otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_previous_step_id(array &$form_state, $step_id = NULL) {
|
|
|
|
$step_id = isset($step_id) ? $step_id : islandora_ingest_form_get_current_step_id($form_state);
|
|
|
|
// If the current step is not defined it's assumed that all steps have
|
|
|
|
// executed. So return the very last step.
|
|
|
|
if ($step_id == NULL) {
|
|
|
|
return islandora_ingest_form_get_last_step_id($form_state);
|
|
|
|
}
|
|
|
|
$step_ids = array_keys(islandora_ingest_form_get_steps($form_state));
|
|
|
|
$index = array_search($step_id, $step_ids);
|
|
|
|
if ($index !== FALSE && --$index >= 0) {
|
|
|
|
return $step_ids[$index];
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Increments the current step if possible.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_increment_step(array &$form_state) {
|
|
|
|
// When going to the next step rebuild the list of steps as the submit
|
|
|
|
// of the current step could have added/removed a step.
|
|
|
|
drupal_static_reset('islandora_ingest_form_get_steps');
|
|
|
|
$next_step_id = islandora_ingest_form_get_next_step_id($form_state);
|
|
|
|
islandora_ingest_form_stash_info($form_state);
|
|
|
|
$form_state['islandora']['step_id'] = $next_step_id;
|
|
|
|
islandora_ingest_form_grab_info($form_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decrement the current step if possible.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_decrement_step(array &$form_state) {
|
|
|
|
$previous_step_id = islandora_ingest_form_get_previous_step_id($form_state);
|
|
|
|
// Don't decrement passed the first step.
|
|
|
|
if (isset($previous_step_id)) {
|
|
|
|
islandora_ingest_form_stash_info($form_state);
|
|
|
|
$form_state['islandora']['step_id'] = $previous_step_id;
|
|
|
|
islandora_ingest_form_grab_info($form_state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Executes the current step.
|
|
|
|
*
|
|
|
|
* Builds the form definition and appends on any additonal elements required
|
|
|
|
* for the step to function.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param string $step_id
|
|
|
|
* The ID of the step relative to the result, if not provided the current
|
|
|
|
* step_id is used.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The form definition of the current step.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_execute_step(array $form, array &$form_state, $step_id = NULL) {
|
|
|
|
// Load any required files for the current step.
|
|
|
|
islandora_ingest_form_load_include($form_state);
|
|
|
|
$step = isset($step_id) ? islandora_ingest_form_get_step($form_state) : islandora_ingest_form_get_step($form_state, $step_id);
|
|
|
|
switch ($step['type']) {
|
|
|
|
case 'callback':
|
|
|
|
// Execute all the consecutive callbacks, and move then attempt to process
|
|
|
|
// the next step.
|
|
|
|
islandora_ingest_form_execute_consecutive_callback_steps($form, $form_state, $step);
|
|
|
|
return islandora_ingest_form_execute_step($form, $form_state);
|
|
|
|
|
|
|
|
case 'form':
|
|
|
|
return islandora_ingest_form_execute_form_step($form, $form_state, $step);
|
|
|
|
|
|
|
|
case 'batch':
|
|
|
|
// @todo Implement if possible.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute the given 'form' step.
|
|
|
|
*
|
|
|
|
* Assumes the given step is a 'form' step.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step
|
|
|
|
* The step we are executing.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The form definition of the given step.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_execute_form_step(array $form, array &$form_state, array $step) {
|
|
|
|
$args = array($form, &$form_state);
|
|
|
|
$args = isset($step['args']) ? array_merge($args, $step['args']) : $args;
|
|
|
|
$shared_storage = islandora_ingest_form_get_shared_storage($form_state);
|
|
|
|
// Build the form step.
|
|
|
|
$form = call_user_func_array($step['form_id'], $args);
|
|
|
|
// Since the list of steps depends on the shared storage we will rebuild the
|
|
|
|
// list of steps if the shared storage has changed. This must be done before
|
|
|
|
// stepifying, so the prev/next buttons get updated.
|
|
|
|
if ($shared_storage != islandora_ingest_form_get_shared_storage($form_state)) {
|
|
|
|
drupal_static_reset('islandora_ingest_form_get_steps');
|
|
|
|
}
|
|
|
|
return islandora_ingest_form_stepify($form, $form_state, $step);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute the given 'callback' step and any consecutive 'callback' steps.
|
|
|
|
*
|
|
|
|
* Assumes the given step is a 'callback' step.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step
|
|
|
|
* The step that execution begins from.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_execute_consecutive_callback_steps(array $form, array &$form_state, array $step) {
|
|
|
|
do {
|
|
|
|
islandora_ingest_form_execute_callback_step($form, $form_state, $step);
|
|
|
|
islandora_ingest_form_increment_step($form_state);
|
|
|
|
$step = islandora_ingest_form_get_step($form_state);
|
|
|
|
} while (isset($step) && $step['type'] == 'callback');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute the given 'callback' step.
|
|
|
|
*
|
|
|
|
* Assumes the given step is a 'callback' step.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step
|
|
|
|
* The step currently being executed.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_execute_callback_step(array $form, array &$form_state, array $step) {
|
|
|
|
$args = array(&$form_state);
|
|
|
|
$args = isset($step['do_function']['args']) ? array_merge($args, $step['do_function']['args']) : $args;
|
|
|
|
if (isset($step['do_function']['file'])) {
|
|
|
|
require_once drupal_get_path('module', $step['module']) . "/" . $step['do_function']['file'];
|
|
|
|
}
|
|
|
|
call_user_func_array($step['do_function']['function'], $args);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Undo the given 'callback' step and any consecutive 'callback' steps.
|
|
|
|
*
|
|
|
|
* Assumes the given $step is a 'callback' step.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step
|
|
|
|
* The step that execution begins from.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_undo_consecutive_callback_steps(array $form, array &$form_state, array $step) {
|
|
|
|
do {
|
|
|
|
islandora_ingest_form_undo_callback_step($form, $form_state, $step);
|
|
|
|
islandora_ingest_form_decrement_step($form_state);
|
|
|
|
$step = islandora_ingest_form_get_step($form_state);
|
|
|
|
} while (isset($step) && $step['type'] == 'callback');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Undo the given 'callback' step.
|
|
|
|
*
|
|
|
|
* Assumes the given $step is a 'callback' step.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step
|
|
|
|
* The step which the undo callback is being called on.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_undo_callback_step(array $form, array &$form_state, array $step) {
|
|
|
|
if (isset($step['undo_function'])) {
|
|
|
|
$args = array(&$form_state);
|
|
|
|
$args = isset($step['undo_function']['args']) ? array_merge($args, $step['undo_function']['args']) : $args;
|
|
|
|
if (isset($step['undo_function']['file'])) {
|
|
|
|
require_once drupal_get_path('module', $step['module']) . "/" . $step['undo_function']['file'];
|
|
|
|
}
|
|
|
|
call_user_func_array($step['undo_function']['function'], $args);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Append Prev/Next buttons submit/validation handlers etc.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step
|
|
|
|
* An array defining a ingest step.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The stepified Drupal form definition for the given step.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_stepify(array $form, array &$form_state, array $step) {
|
|
|
|
$first_form_step = islandora_ingest_form_on_first_form_step($form_state);
|
|
|
|
$last_form_step = islandora_ingest_form_on_last_form_step($form_state);
|
|
|
|
$form['form_step_id'] = array(
|
|
|
|
'#type' => 'hidden',
|
|
|
|
'#value' => islandora_ingest_form_get_current_step_id($form_state),
|
|
|
|
);
|
|
|
|
$form['prev'] = $first_form_step ? NULL : islandora_ingest_form_previous_button($form_state);
|
|
|
|
$form['next'] = $last_form_step ? islandora_ingest_form_ingest_button($form_state) : islandora_ingest_form_next_button($form_state);
|
|
|
|
// Duplicate next button and hide it at the top of the form so that the form
|
|
|
|
// always triggers "next" if the user submits the form with the enter key.
|
|
|
|
if (!is_null($form['prev'])) {
|
|
|
|
$form['hidden_next'] = $form['next'];
|
|
|
|
$form['hidden_next']['#weight'] = -99;
|
|
|
|
$form['hidden_next']['#prefix'] = '<div style="display:none;">';
|
|
|
|
$form['hidden_next']['#suffix'] = '</div>';
|
|
|
|
}
|
|
|
|
// Allow for a hook_form_FORM_ID_alter().
|
|
|
|
drupal_alter(array('form_' . $step['form_id'], 'form'), $form, $form_state, $step['form_id']);
|
|
|
|
return $form;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if we are on the first form step.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
* TRUE if we are currently on the first step, FALSE otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_on_first_form_step(array &$form_state) {
|
|
|
|
return !islandora_ingest_form_get_previous_form_step($form_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if we are on the last form step.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
* TRUE if we are currently on the last step, FALSE otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_on_last_form_step(array &$form_state) {
|
|
|
|
return !islandora_ingest_form_get_next_form_step($form_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the previous form step relative to the current step.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The previous form step if one exists, NULL otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_previous_form_step(array &$form_state) {
|
|
|
|
$step = NULL;
|
|
|
|
do {
|
|
|
|
$step = islandora_ingest_form_get_previous_step($form_state, $step);
|
|
|
|
} while (isset($step) && $step['type'] != 'form');
|
|
|
|
return $step;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the next form step relative to the current step.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The next form step if one exists, NULL otherwise.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_next_form_step(array &$form_state) {
|
|
|
|
$step = NULL;
|
|
|
|
do {
|
|
|
|
$step = islandora_ingest_form_get_next_step($form_state, $step);
|
|
|
|
} while (isset($step) && $step['type'] != 'form');
|
|
|
|
return $step;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Defines the previous button for the ingest form.
|
|
|
|
*
|
|
|
|
* Adds submit handlers for the form step if they exist.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The previous button for the ingest form.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_previous_button(array &$form_state) {
|
|
|
|
// Before we move back to the previous step we should tell the previous steps
|
|
|
|
// to undo whatever its submit handler did.
|
|
|
|
$prev_form_step = islandora_ingest_form_get_previous_form_step($form_state);
|
|
|
|
$form_id = $prev_form_step['form_id'];
|
|
|
|
$submit_callback = $form_id . '_undo_submit';
|
|
|
|
$submit = function_exists($submit_callback) ? array('islandora_ingest_form_previous_submit', $submit_callback) : array('islandora_ingest_form_previous_submit');
|
|
|
|
return array(
|
|
|
|
'#type' => 'submit',
|
|
|
|
'#value' => t('Previous'),
|
|
|
|
'#name' => 'prev',
|
|
|
|
'#submit' => $submit,
|
|
|
|
// #limit_validation_errors, is why when the previous button is pressed no
|
|
|
|
// values persisted in the form_state, but its also what allows us to go
|
|
|
|
// back when validation errors occur. To have a better solution going
|
|
|
|
// forward we can either limit validation only on required fields, or we can
|
|
|
|
// convert all required fields to use #element_validation functions, and
|
|
|
|
// Remove the need for #limit_validation_errors. Or maybe there is some
|
|
|
|
// other solution, regardless of what it is, it won't be standard.
|
|
|
|
'#limit_validation_errors' => array(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The submit handler for the ingest form previous button.
|
|
|
|
*
|
|
|
|
* Stores the current form steps values in the form storage.
|
|
|
|
* Moves the focus of the multi-page ingest form back one step.
|
|
|
|
* Restores the form values for the previous step.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_previous_submit(array $form, array &$form_state) {
|
|
|
|
islandora_ingest_form_decrement_step($form_state);
|
|
|
|
$step = islandora_ingest_form_get_step($form_state);
|
|
|
|
// Undo all callbacks that occured after the previous step.
|
|
|
|
if ($step['type'] == 'callback') {
|
|
|
|
islandora_ingest_form_undo_consecutive_callback_steps($form, $form_state, $step);
|
|
|
|
}
|
|
|
|
$form_state['rebuild'] = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Defines the next button for the ingest form.
|
|
|
|
*
|
|
|
|
* Adds submit/validate handlers for the form step if they exist.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The next button for the ingest form.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_next_button(array &$form_state) {
|
|
|
|
$step = islandora_ingest_form_get_step($form_state);
|
|
|
|
$form_id = $step['form_id'];
|
|
|
|
$validate_callback = $form_id . '_validate';
|
|
|
|
$validate = function_exists($validate_callback) ? array($validate_callback) : NULL;
|
|
|
|
$submit_callback = $form_id . '_submit';
|
|
|
|
$submit = function_exists($submit_callback) ? array($submit_callback, 'islandora_ingest_form_next_submit') : array('islandora_ingest_form_next_submit');
|
|
|
|
return array(
|
|
|
|
'#type' => 'submit',
|
|
|
|
'#value' => t('Next'),
|
|
|
|
'#name' => 'next',
|
|
|
|
'#validate' => $validate,
|
|
|
|
'#submit' => $submit,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The submit handler for the ingest form next button.
|
|
|
|
*
|
|
|
|
* Stores the current form steps values in the form storage.
|
|
|
|
* Moves the focus of the multi-page ingest form forward one step.
|
|
|
|
* Restores the form values for the next step if present.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_next_submit(array $form, array &$form_state) {
|
|
|
|
islandora_ingest_form_increment_step($form_state);
|
|
|
|
$form_state['rebuild'] = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Push current info into the current step's storage.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_stash_info(array &$form_state) {
|
|
|
|
$storage = &islandora_ingest_form_get_step_storage($form_state);
|
|
|
|
if ($storage && isset($form_state['values'])) {
|
|
|
|
$storage['values'] = $form_state['values'];
|
|
|
|
unset($form_state['values']);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pops the info for the given step from storage into the form_state.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_grab_info(array &$form_state) {
|
|
|
|
$storage = islandora_ingest_form_get_step_storage($form_state);
|
|
|
|
$form_state['values'] = isset($storage['values']) ? $storage['values'] : array();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Defines the ingest button for the ingest form.
|
|
|
|
*
|
|
|
|
* This button is only shown on the last page of the multi-page ingest form.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The ingest button for the ingest form.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_ingest_button(array &$form_state) {
|
|
|
|
$step = islandora_ingest_form_get_step($form_state);
|
|
|
|
$form_id = $step['form_id'];
|
|
|
|
$validate_callback = $form_id . '_validate';
|
|
|
|
$validate = function_exists($validate_callback) ? array($validate_callback) : NULL;
|
|
|
|
$submit_callback = $form_id . '_submit';
|
|
|
|
$submit = function_exists($submit_callback) ? array(
|
|
|
|
$submit_callback,
|
|
|
|
'islandora_ingest_form_pre_submit',
|
|
|
|
'islandora_ingest_form_submit',
|
|
|
|
) : array('islandora_ingest_form_pre_submit', 'islandora_ingest_form_submit');
|
|
|
|
return array(
|
|
|
|
'#type' => 'submit',
|
|
|
|
'#name' => 'ingest',
|
|
|
|
'#value' => t('Ingest'),
|
|
|
|
'#process' => array('islandora_ingest_form_ingest_button_process'),
|
|
|
|
'#validate' => $validate,
|
|
|
|
'#submit' => $submit,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Process hook for the ingest button, adds the please wait spinning icon.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_ingest_button_process(array $element) {
|
|
|
|
$settings['spinner'][$element['#id']] = array(
|
|
|
|
'message' => t('Please be patient while the page loads.'),
|
|
|
|
'options' => array(
|
|
|
|
'lines' => 10,
|
|
|
|
'length' => 20,
|
|
|
|
'width' => 10,
|
|
|
|
'radius' => 30,
|
|
|
|
'corners' => 1,
|
|
|
|
'rotate' => 0,
|
|
|
|
'direction' => 1,
|
|
|
|
'color' => '#000',
|
|
|
|
'speed' => 1,
|
|
|
|
'trail' => 60,
|
|
|
|
'shadow' => FALSE,
|
|
|
|
'hwaccel' => FALSE,
|
|
|
|
'className' => 'spinner',
|
|
|
|
'zIndex' => 2000000000,
|
|
|
|
'top' => 'auto',
|
|
|
|
'left' => 'auto',
|
|
|
|
),
|
|
|
|
);
|
|
|
|
drupal_add_js($settings, 'setting');
|
|
|
|
$islandora_path = drupal_get_path('module', 'islandora');
|
|
|
|
$element['#attached'] = array(
|
|
|
|
'js' => array(
|
|
|
|
"$islandora_path/js/spin/spin.min.js",
|
|
|
|
"$islandora_path/js/spinner.js",
|
|
|
|
),
|
|
|
|
);
|
|
|
|
$element['#attributes']['class'][] = 'islandora-spinner-submit';
|
|
|
|
return $element;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The submit handler for the ingest form.
|
|
|
|
*
|
|
|
|
* Attempts to ingest every object built by the previous steps.
|
|
|
|
*
|
|
|
|
* @param array $form
|
|
|
|
* The Drupal form.
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_submit(array $form, array &$form_state) {
|
|
|
|
// Execute any remaining callbacks.
|
|
|
|
islandora_ingest_form_increment_step($form_state);
|
|
|
|
$step = islandora_ingest_form_get_step($form_state);
|
|
|
|
if (isset($step) && $step['type'] == 'callback') {
|
|
|
|
islandora_ingest_form_execute_consecutive_callback_steps($form, $form_state, $step);
|
|
|
|
}
|
|
|
|
// Ingest the objects.
|
|
|
|
$set_redirect = isset($form_state['redirect']) ? FALSE : TRUE;
|
|
|
|
foreach ($form_state['islandora']['objects'] as &$object) {
|
|
|
|
try {
|
|
|
|
islandora_add_object($object);
|
|
|
|
// We want to redirect to the first object as it's considered to be the
|
|
|
|
// primary object.
|
|
|
|
if ($set_redirect) {
|
|
|
|
$form_state['redirect'] = "islandora/object/{$object->id}";
|
|
|
|
$set_redirect = FALSE;
|
|
|
|
}
|
|
|
|
drupal_set_message(
|
|
|
|
t('"@label" (ID: @pid) has been ingested.', array('@label' => $object->label, '@pid' => $object->id)),
|
|
|
|
'status');
|
|
|
|
}
|
|
|
|
catch (Exception $e) {
|
|
|
|
// If post hooks throws it may already exist at this point but may be
|
|
|
|
// invalid, so don't say failed.
|
|
|
|
watchdog(
|
|
|
|
'islandora',
|
|
|
|
'Exception during ingest with Message: "@exception", and Trace: @trace',
|
|
|
|
array('@exception' => $e->getMessage(), '@trace' => $e->getTraceAsString()),
|
|
|
|
WATCHDOG_ERROR
|
|
|
|
);
|
|
|
|
drupal_set_message(
|
|
|
|
t('A problem occured while ingesting "@label" (ID: @pid), please notify the administrator.',
|
|
|
|
array('@label' => $object->label, '@pid' => $object->id)),
|
|
|
|
'error'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// XXX: Foreaching with references can be weird... The reference exists in
|
|
|
|
// the scope outside.
|
|
|
|
unset($object);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a reference to the stored NewFedoraObject's.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* A reference to the stored NewFedoraObjects to be ingested when the final
|
|
|
|
* step submits.
|
|
|
|
*/
|
|
|
|
function &islandora_ingest_form_get_objects(array &$form_state) {
|
|
|
|
return $form_state['islandora']['objects'];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a single object from the stored NewFedoraObject's.
|
|
|
|
*
|
|
|
|
* @note In our current use case we are only dealing with a single object
|
|
|
|
* ingest, this makes it convenient to access it. Ideally the steps
|
|
|
|
* implementations will be abstracted to be indifferent to what object it's
|
|
|
|
* currently working on. This will act as a placeholder for such
|
|
|
|
* functionality.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return NewFedoraObject
|
|
|
|
* Returns the 'current' object in the array of NewFedoraObjects, generally
|
|
|
|
* this is only used when there is one object in the list of objects.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_object(array &$form_state) {
|
|
|
|
$objects = &islandora_ingest_form_get_objects($form_state);
|
|
|
|
return reset($objects);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get general storage for the given/current step.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
* @param array $step_id
|
|
|
|
* The ID of the step.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The given/current step storage if found, NULL otherwise.
|
|
|
|
*/
|
|
|
|
function &islandora_ingest_form_get_step_storage(array &$form_state, $step_id = NULL) {
|
|
|
|
$step_id = isset($step_id) ? $step_id : islandora_ingest_form_get_current_step_id($form_state);
|
|
|
|
if (isset($step_id)) {
|
|
|
|
if (!isset($form_state['islandora']['step_storage'][$step_id])) {
|
|
|
|
$form_state['islandora']['step_storage'][$step_id] = array();
|
|
|
|
}
|
|
|
|
return $form_state['islandora']['step_storage'][$step_id];
|
|
|
|
}
|
|
|
|
$undefined_step_storage = array();
|
|
|
|
return $undefined_step_storage;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the configuration used to create the multi-page ingest form.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The configuration used to generate the multi-page ingest forms.
|
|
|
|
*/
|
|
|
|
function &islandora_ingest_form_get_shared_storage(array &$form_state) {
|
|
|
|
return $form_state['islandora']['shared_storage'];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call form_load_include, for the current step if it defines a required file.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_load_include(array &$form_state) {
|
|
|
|
form_load_include($form_state, 'inc', 'islandora', 'includes/ingest.form');
|
|
|
|
$step = islandora_ingest_form_get_step($form_state);
|
|
|
|
// Load include files.
|
|
|
|
if (isset($step['file']) && isset($step['module'])) {
|
|
|
|
$matches = array();
|
|
|
|
preg_match('/^(.*)\.(.*)$/', $step['file'], $matches);
|
|
|
|
list($file, $name, $type) = $matches;
|
|
|
|
form_load_include($form_state, $type, $step['module'], $name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Buildes the initial list of ingest steps.
|
|
|
|
*
|
|
|
|
* Sorted by weight expected range between -50 to 50.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The list of sorted ingest steps as defined by all implementers
|
|
|
|
* of ISLANDORA_INGEST_STEP_HOOK.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_steps(array &$form_state) {
|
|
|
|
module_load_include('inc', 'islandora', 'includes/utilities');
|
|
|
|
$steps = &drupal_static(__FUNCTION__);
|
|
|
|
if (isset($steps)) {
|
|
|
|
return $steps;
|
|
|
|
}
|
|
|
|
$steps = array();
|
|
|
|
$shared_storage = &islandora_ingest_form_get_shared_storage($form_state);
|
|
|
|
foreach (islandora_build_hook_list(ISLANDORA_INGEST_STEP_HOOK, $shared_storage['models']) as $hook) {
|
|
|
|
// Required for pass by reference.
|
|
|
|
// @todo Change this around so that it isn't passed by reference, there
|
|
|
|
// Is an alter below that can handle that requirement.
|
|
|
|
foreach (module_implements($hook) as $module) {
|
|
|
|
$function = $module . '_' . $hook;
|
|
|
|
$module_steps = (array) $function($form_state);
|
|
|
|
$steps = array_merge($steps, $module_steps);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$steps = array_filter($steps);
|
|
|
|
foreach (islandora_build_hook_list(ISLANDORA_INGEST_STEP_HOOK, $shared_storage['models']) as $hook) {
|
|
|
|
drupal_alter($hook, $steps, $form_state);
|
|
|
|
}
|
|
|
|
// Add any defaults.
|
|
|
|
foreach ($steps as $key => &$step) {
|
|
|
|
$step['id'] = $key;
|
|
|
|
}
|
|
|
|
islandora_sort_ordered($steps, array('weight'));
|
|
|
|
return $steps;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filter the ingest steps to only steps of type 'form'.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The list of sorted ingest form steps as defined by all implementers
|
|
|
|
* of ISLANDORA_INGEST_STEP_HOOK.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_form_steps(array &$form_state) {
|
|
|
|
$steps = islandora_ingest_form_get_steps($form_state);
|
|
|
|
$form_step_filter = function($o) {
|
|
|
|
return $o['type'] == 'form';
|
|
|
|
};
|
|
|
|
return array_filter($steps, $form_step_filter);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filter the ingest steps to only steps of type 'form'.
|
|
|
|
*
|
|
|
|
* @param array $form_state
|
|
|
|
* The Drupal form state.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* The list of sorted ingest callback steps as defined by all implementers
|
|
|
|
* of ISLANDORA_INGEST_STEP_HOOK.
|
|
|
|
*/
|
|
|
|
function islandora_ingest_form_get_callback_steps(array &$form_state) {
|
|
|
|
$steps = islandora_ingest_form_get_steps($form_state);
|
|
|
|
$callback_step_filter = function($o) {
|
|
|
|
return $o['type'] == 'callback';
|
|
|
|
};
|
|
|
|
return array_filter($steps, $callback_step_filter);
|
|
|
|
}
|