| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270 | <?php/** * @file * Intergrates workflow with features. */define('WORKFLOW_FEATURES_AUTHOR_NAME', 'workflow_features_author_name');/** * Workflows are a **faux-exportable** component. *//** * Implements hook_features_export(). */function workflow_features_export($data, &$export, $module_name = '') {  // fontyourface_default_fonts integration is provided by Features.  $export['dependencies']['features'] = 'features';  $export['dependencies']['workflow'] = 'workflow';  foreach (workflow_get_workflows() as $workflow) {    if (in_array($workflow->name, $data)) {      $export['features']['workflow'][$workflow->name] = $workflow->name;    }  }  return $export;}/** * Implements hook_features_export_render(). */function workflow_features_export_render($module, $data) {  $translatables = $code = array();  $code[] = '  $workflows = array();';  $code[] = '';  $workflows = workflow_get_workflows();  foreach ($data as $name) {    if ($workflow = workflow_get_workflows_full_object($name)) {      unset($workflow->wid);      $workflow_export = features_var_export($workflow, '  ');      $workflow_identifier = features_var_export($workflow->name);      $code[] = "  // Exported workflow: $name";      $code[] = "  \$workflows[{$workflow_identifier}] = {$workflow_export};";      $code[] = "";    }  }  $code[] = '  return $workflows;';  $code = implode("\n", $code);  return array('workflow_default_workflows' => $code);}/** * Implements hook_features_export_options(). */function workflow_features_export_options() {  $workflows = array();  foreach (workflow_get_workflows() as $workflow) {    $workflows[$workflow->name] = $workflow->name;  }  return $workflows;}/** * Implements hook_features_revert(). */function workflow_features_revert($module) {  // Including the features inc to make sure this func is available during install of a Features module.  module_load_include('inc', 'features', 'features.export');  foreach (features_get_default('workflow', $module) as $key => $workflow) {    workflow_update_workflows_full_object($workflow);  }}/** * Implements hook_features_export_rebuild(). */function workflow_features_export_rebuild($module) {  workflow_features_revert($module);}/** * CRUD style functions below. *//** * For use by CRUD only, save everything from the CRUD formed object. * * @see workflow_get_workflows_full_object * * @param $workflow *   A fully loaded workflow object to save the states of. * * @return *   Returns whether the workflow was saved fully. */function workflow_update_workflows_full_object($workflow) {  $workflow = (object) $workflow;  // Given a workflow in the format returned from export.  // First we grab the states, transitions and node_maps out.  $states = isset($workflow->states) ? $workflow->states : array();  $transitions = isset($workflow->transitions) ? $workflow->transitions : array();  $node_types = isset($workflow->node_types) ? $workflow->node_types : array();  unset($workflow->states, $workflow->transitions, $workflow->node_types);  // Then make a workflow so we can track by wid.  if ($orig_workflow = workflow_get_workflows_by_name($workflow->name)) {    $workflow->wid = $orig_workflow->wid;  }  workflow_update_workflows($workflow, FALSE);  // Cancel out if workflow failed to save.  if (!isset($workflow->wid) || empty($workflow->wid)) {    return FALSE;  }  // Workflow is now a fully vetted workflow object. We have NOT created a creation state with this.  // Then make states, marking state name to state sid.  $active_states = array();  foreach ($states as $state) {    $state = (object) $state;    $state->wid = $workflow->wid;    if ($orig_state = reset(workflow_get_workflow_states_by_wid_state($state->wid, $state->state))) {      $state->sid = $orig_state->sid;    }    workflow_update_workflow_states($state);    $active_states[$state->state] = $state->sid;  }  // Delete any states *not* in our original construction.  foreach (workflow_get_workflow_states_by_wid($workflow->wid) as $state) {    if (!in_array($state->sid, $active_states)) {      workflow_delete_workflow_states_by_sid($state->sid);    }  }  // Then make transitions with the state mapping.  $active_transitions = array();  foreach ($transitions as $transition) {    $transition = (object) $transition;    $transition->sid = $active_states[$transition->state];    $transition->target_sid = $active_states[$transition->target_state];    // Roles are exported by rolename, so need to translate to RID.    $transition->roles = !empty($transition->roles) ? _workflow_roles_to_rids($transition->roles) : '';    workflow_update_workflow_transitions($transition);    $active_transitions[] = $transition->tid;  }  // Delete any transitions in our workflow that are *not* in our original construction.  foreach (workflow_get_workflow_transitions_by_wid($workflow->wid) as $transition) {    if (!in_array($transition->tid, $active_transitions)) {      workflow_delete_workflow_transitions_by_tid($transition->tid);    }  }  // Then add the node_type mapping.  foreach ($node_types as $node_type) {    $node_type = (object) array(      'type' => $node_type,      'wid' => $workflow->wid    );    // Insert, nodes only have one workflow. Insert will delete any prior workflow assoc.    workflow_insert_workflow_type_map($node_type);  }  return TRUE;}/** * For use by CRUD only, gather everything into the CRUD formed object. * * @param $name *   A string corresponding to a workflow object. * * @return *   A fully loaded workflow object with type and statue mappings. */function workflow_get_workflows_full_object($name) {  if ($workflow = workflow_get_workflows_by_name($name)) {    // Now we need to add data to the object for each state, an array of sub-objects.    $options = array('status' => 1); // We only want active states for this export.    $active_states = array();    foreach (workflow_get_workflow_states_by_wid($workflow->wid, $options) as $index => $state) {      $active_states[$state->sid] = $state->state;      // Remove what we don't need to export.      unset($state->sid);      unset($state->wid);      $workflow->states[] = $state;    }    // Now we need to add data to the export for each transition, an array of sub-objects.    // Same goes for transitions, see above re: states.    foreach ($active_states as $sid => $state) {      // We're going to look everythign up by the start state, not state involved, to avoid dupes.      foreach (workflow_get_workflow_transitions_by_sid($sid, $options) as $transition) {        // And to get the target state (by name) we need to look it up too.        $target_state = workflow_get_workflow_states_by_sid($transition->target_sid);        $transition->state = $state;        $transition->target_state = $target_state->state;        unset($transition->sid, $transition->target_sid);        // Translate to role names so works cross install.        $transition->roles = !empty($transition->roles) ? _workflow_rids_to_roles($transition->roles) : '';        // Remove what we don't need to export.        unset($transition->tid);        $workflow->transitions[] = $transition;      }    }    // Now we need to add data to the export for each type map, an array of sub-objects.    // Same goes for node mappings, see above re: states.    foreach (workflow_get_workflow_type_map_by_wid($workflow->wid) as $index => $type_map) {      $workflow->node_types[] = $type_map->type;    }  }  return $workflow;}/** * Internally cache the user roles as core doesn't. */function _workflow_user_roles($reset = FALSE) {  $roles = &drupal_static(__FUNCTION__);  if ($reset || !isset($roles)) {    $roles = user_roles();  }  return $roles;}/** * Translates a role string to RIDs for importing. * * @param $role_string *   A string of roles or fake 'author' role. * * @return *   A string of RIDs seperated by commas. */function _workflow_roles_to_rids($role_string) {  $roles = _workflow_user_roles();  $rid_array = array();  foreach (explode(',', $role_string) as $role_name) {    if ($role_name === WORKFLOW_FEATURES_AUTHOR_NAME) {      $rid_array[] = 'author';    }    elseif ($role_name && in_array($role_name, $roles)) {      $rid_array[] = array_search($role_name, $roles);    }  }  return implode(',', $rid_array);}/** * Translates a string of rids to role names for exporting. * * @param $rid_string *   A string of rids or fake 'author' role. * * @return *   A string of role names seperated by commas. */function _workflow_rids_to_roles($rid_string) {  $roles = _workflow_user_roles();  $rid_array = explode(',', $rid_string);  // There may be a role named 'author', so make 'author' distinct.  $return = in_array('author', $rid_array) ? WORKFLOW_FEATURES_AUTHOR_NAME . ',' : '';  // Translate RIDs to rolenames.  $return .= implode(',', array_intersect_key($roles, array_flip($rid_array)));  return trim($return, ',');}
 |