');
$breadcrumb[] = l('Administration', 'admin');
$breadcrumb[] = l('Tripal', 'admin/tripal');
$breadcrumb[] = l('Biological Data', 'admin/tripal/bundles');
drupal_set_breadcrumb($breadcrumb);
/* // Add the view
$view = views_embed_view('tripal_feature_admin_features','default');
if (isset($view)) {
$output .= $view;
}
else {
$output .= 'The Feature module uses primarily views to provide an '
. 'administrative interface. Currently one or more views needed for this '
. 'administrative interface are disabled. Click each of the following links to '
. 'enable the pertinent views:
';
$output .= '';
$output .= '- '.l('Features View', 'admin/tripal/chado/tripal_feature/views/features/enable').'
';
$output .= '
';
}
// Add a summary chart.
//-----------------------------------
$output .= theme('tripal_feature_bar_chart_type_organism_summary');
drupal_add_js('
Drupal.behaviors.tripalFeature_moveAdminSummaryChart = {
attach: function (context, settings) {
jQuery("#tripal-feature-admin-summary").insertBefore( jQuery(".view-filters") );
}};
', 'inline'); */
return $output;
}
/**
*
*/
function tripal_entities_content_view() {
$form = drupal_get_form('tripal_entities_content_overview_form');
$output = drupal_render($form);
// set the breadcrumb
$breadcrumb = array();
$breadcrumb[] = l('Home', '');
$breadcrumb[] = l('Administration', 'admin');
drupal_set_breadcrumb($breadcrumb);
return $output;
}
/**
*
* @param unknown $form
* @param unknown $form_state
* @return multitype:
*/
function tripal_entities_content_overview_form($form, &$form_state) {
$form = array();
$entities = db_select('tripal_entity', 'td')
->fields('td')
->orderBy('created', 'DESC')//ORDER BY created
->range(0,25)
->execute();
$headers = array('Title', 'Vocabulary', 'Term', 'Author', 'Status', 'Updated', 'Operations');
$rows = array();
while ($entity = $entities->fetchObject()) {
$cvterm = chado_generate_var('cvterm', array('cvterm_id' => $entity->cvterm_id));
$author = user_load($entity->uid);
$rows[] = array(
l($entity->title, 'BioData/' . $entity->id),
$cvterm->cv_id->name . ' (' . $cvterm->dbxref_id->db_id->name . ')',
$cvterm->name,
l($author->name, 'user/' . $entity->uid),
$entity->status == 1 ? 'published' : 'unpublished',
format_date($entity->changed, 'short'),
l('edit', 'BioData/' . $entity->id . '/edit') . ' ' .
l('delete', 'BioData/' . $entity->id . '/delete')
);
}
$table_vars = array(
'header' => $headers,
'rows' => $rows,
'attributes' => array(),
'sticky' => TRUE,
'caption' => '',
'colgroups' => array(),
'empty' => '',
);
$results_table = theme('table', $table_vars);
$form['results'] = array(
'#markup' => $results_table
);
return $form;
}
/**
*
* @param unknown $form
* @param unknown $form_state
* @return multitype:
*/
function tripal_entities_admin_bundles_form($form, &$form_state) {
$form = array();
// Set the defaults.
$cv_id = NULL;
$term_name = NULL;
// Set defaults using the form state.
if (array_key_exists('values', $form_state)) {
$cv_id = array_key_exists('cv_id', $form_state['values']) ? $form_state['values']['cv_id'] : NULL;
$term_name = array_key_exists('term_name', $form_state['values']) ? $form_state['values']['term_name'] : NULL;
}
// Let the user select the vocabulary and tripal_entity but only if they haven't
// already selected a tripal_entity.
$sql = "
SELECT CV.cv_id, CV.name
FROM {cv} CV
ORDER BY CV.name
";
$vocabs = chado_query($sql);
$cvs = array();
while ($vocab = $vocabs->fetchObject()) {
$cvs[$vocab->cv_id] = $vocab->name;
}
$form['cv_id'] = array(
'#type' => 'select',
'#title' => t('Vocabulary'),
'#options' => $cvs,
'#required' => FALSE,
'#description' => t('Select a vocabulary to view potential data types in the chart below. Limit the chart to only published data types by selecting the checkbox.'),
'#default_value' => $cv_id,
'#ajax' => array(
'callback' => "tripal_entities_admin_bundles_form_ajax_callback",
'wrapper' => 'tripal_entities_admin_bundles_form',
'effect' => 'fade',
'method' => 'replace'
)
);
$form['refresh_bundles'] = array(
'#type' => 'submit',
'#value' => t('Refresh Data Types'),
'#name' => 'refresh_bundles',
);
$form['publish_new_data'] = array(
'#type' => 'submit',
'#value' => t('Publish New Data'),
'#name' => 'publish_new_data',
);
$form['#prefix'] = '';
$form['#suffix'] = '
';
return $form;
}
/**
* Submit a job to populate the entity tables
* This operation makes available data types in the database publishable
*/
function tripal_entities_admin_bundles_form_submit($form, $form_state) {
global $user;
if ($form_state['clicked_button']['#name'] == 'refresh_bundles') {
tripal_add_job('Create publishable data types', 'tripal_entity', 'tripal_entities_populate_entity_tables', array(), $user->uid);
}
if ($form_state['clicked_button']['#name'] == 'publish_new_data') {
}
}
/**
*
* @param unknown $form
* @param unknown $form_state
* @return multitype:
*/
function tripal_entities_admin_add_type_form($form, &$form_state) {
$term_name = '';
$num_terms = 0;
$cv_id = '';
// Set defaults using the form state.
if (array_key_exists('storage', $form_state) and
array_key_exists('terms', $form_state['storage'])) {
$terms = $form_state['storage']['terms'];
$num_terms = count($terms);
}
if (array_key_exists('values', $form_state) and
array_key_exists('term_name', $form_state['values'])) {
$term_name = $form_state['values']['term_name'];
}
// If no term has been selected yet then provide the auto complete field.
if ($num_terms == 0) {
$form['term_name'] = array(
'#title' => t('Biological Data Type'),
'#type' => 'textfield',
'#description' => t("Please enter the type of data that you want to add.
Once added, priviledged users can add new records of the selected
type. As you type, suggestions will be provided."),
'#required' => TRUE,
'#default_value' => $term_name,
'#autocomplete_path' => "admin/tripal/chado/tripal_cv/cvterm/auto_name/$cv_id",
);
}
// If the term belongs to more than one vocabulary then add additional fields
// to let the user select the vocabulary.
if ($num_terms > 1) {
$cvs = array();
foreach ($terms as $term) {
$cvs[$term->cv_id->cv_id] = $term->cv_id->name;
}
$form['cv_id'] = array(
'#type' => 'radios',
'#title' => t('Select the appropriate vocabulary'),
'#options' => $cvs,
'#description' => t('The term belongs to more than one vocabulary. Please
indicate the proper vocabulary for the term.')
);
}
// Add in the button for the cases of no terms or too many.
if ($num_terms != 1) {
$form['select_button'] = array(
'#type' => 'submit',
'#value' => t('Use this term'),
'#name' => 'select_cvterm'
);
}
return $form;
}
/**
* Implements hook_validate() for the tripal_entities_admin_publish_form.
*
*/
function tripal_entities_admin_add_type_form_validate($form, &$form_state) {
// Check if this term and vocabulary is in the tripal_vocabulary usage tables.
// If not then add it.
if (array_key_exists('clicked_button', $form_state) and
$form_state['clicked_button']['#name'] =='select_cvterm') {
// First, make sure the term is unique. If not then we can't check it.
$term_name = $form_state['values']['term_name'];
$cv_id = array_key_exists('cv_id', $form_state['values']) ? $form_state['values']['cv_id'] : '';
$cvterm = NULL;
// If a term and cv_id are provided then we can look for the term using
// both and we should find a unique term. If only ther term is provided
// we can still look for a unique term but there must only be one.
if ($term_name and !$cv_id) {
$match = array(
'name' => $term_name,
);
}
else {
$match = array(
'name' => $term_name,
'cv_id' => $cv_id,
);
}
$terms = chado_generate_var('cvterm', $match, array('return_array' => TRUE));
// Add the cvterm to the storage element so we don't have to keep
// looking it up in the submit function or on a rebuild of the form.
$form_state['storage']['terms'] = $terms;
// If we only have one term then we found a unique match and we can do
// some further checking.
if (count($terms) == 1) {
$cvterm = $terms[0];
// Make sure the term is set as published.
tripal_entities_add_term_usage($cvterm, $form_state);
}
// If we do not have any terms then the term provided by the user does not
// exists and we need to provide an error message.
if (count($terms) == 0) {
form_set_error('term_name', t('The term does not exist in this database.'));
}
// If we have more than one term then we need to set an error so that the
// form can provide a list of vocabularies to select from.
if (count($terms) > 1) {
form_set_error('', t('The term is not unique. A list of vocabularies has been provided where this term is present. Please select the appropriate one.'));
}
}
}
/**
* Implements hook_submit() for the tripal_entities_admin_publish_form.
*
*/
function tripal_entities_admin_add_type_form_submit($form, &$form_state) {
if ($form_state['clicked_button']['#name'] =='select_cvterm') {
$cvterm = $form_state['storage']['terms'][0];
$bundle_id = 'dbxref_' . $cvterm->dbxref_id->dbxref_id;
// Before we try to add this type, check to see if it already exists
// as a bundle.
$einfo = entity_get_info('BioData');
if (!in_array($bundle_id, array_keys($einfo['bundles']))) {
tripal_entities_add_term_usage($cvterm, $form_state);
tripal_entities_add_bundle($cvterm);
drupal_set_message('New biological data type created. Fields are added automatically to this type.');
$form_state['redirect'] = "admin/structure/BioData";
}
else {
drupal_set_message('This type already exists.', 'warning');
}
}
}
/**
* Implements hook_submit() for the tripal_entities_admin_publish_form.
*
*/
function tripal_entities_add_bundle($cvterm) {
// Create the bundle name and entity type name. The bundle name is the
// dbxref ID. This isn't very human readable, but the alternative is to
// use the accession which may not always be alpha-numeric.
$bundle_name = 'dbxref_' . $cvterm->dbxref_id->dbxref_id;
// Check to see if this bundle exists. If not then create it
$bundle = db_select('tripal_bundle', 't')
->fields('t')
->condition('type', 'BioData')
->condition('bundle', $bundle_name)
->execute()
->fetchObject();
if (!$bundle) {
// The TripalBundle Entity manages the bundles we have available.
// Therefore, we need to add a new entity for each bundle "type".
$vals = array(
'label' => $cvterm->name,
'type' => 'BioData',
'bundle' => $bundle_name,
'data' => serialize(array()),
'module' => 'tripal_entities'
);
$tripal_bundle = new TripalBundle($vals, 'BioData_bundles');
$tripal_bundle->save();
}
// Allow modules to now add fields to the bundle
module_invoke_all('add_bundle_fields', 'BioData', $bundle_name, $cvterm);
}
/**
* Implements hook_add_bundle_fields().
*
* @param $entity_type_name
* @param $bundle_name
* @param $cvterm
*/
function tripal_entities_add_bundle_fields($entity_type_name, $bundle_name, $cvterm) {
// Adds the fields for the base table to the entity.
tripal_entities_add_bundle_base_fields($entity_type_name, $bundle_name, $cvterm);
// Check to see if there are any kv-property tables associated to this
// base table. If so, add the fields for that type of table.
tripal_entities_add_bundle_kvproperty_adder_field($entity_type_name, $bundle_name, 'featureprop');
}
/**
* Adds the fields for a kv-property table fields
*
* @param $entity_type_name
* @param $bundle_name
* @param $kv_table
*/
function tripal_entities_add_bundle_kvproperty_adder_field($entity_type_name, $bundle_name, $kv_table) {
// First add a generic property field so that users can add new proeprty types.
$field_name = $kv_table;
// Initialize the field array.
$field_info = array(
'field_type' => 'kvproperty_adder',
'widget_type' => 'tripal_fields_kvproperty_adder_widget',
'field_settings' => array(),
'widget_settings' => array('display_label' => 1),
'description' => '',
'label' => 'Additional Properties',
'is_required' => 0,
);
tripal_add_bundle_field($field_name, $field_info, $entity_type_name, $bundle_name);
}
/**
* Adds the fields for the base table to the entity.
*/
function tripal_entities_add_bundle_base_fields($entity_type_name, $bundle_name, $cvterm) {
// Get the list of tables where this cvterm is used.
$match = array('cvterm_id' => $cvterm->cvterm_id);
$term = chado_select_record('tripal_term', array('*'), $match);
$values = array('term_id' => $term[0]->term_id);
$tables = chado_select_record('tripal_term_usage', array('*'), $values);
// Iterate through the tables.
foreach ($tables as $table) {
$table_name = $table->data_table;
$type_table = $table->type_table;
$type_field = $table->field;
// We only want to look at base tables.
if ($table_name == 'cvterm_dbxref' || $table_name == 'cvterm_relationship' ||
$table_name == 'cvtermpath' || $table_name == 'cvtermprop' || $table_name == 'chadoprop' ||
$table_name == 'cvtermsynonym' || preg_match('/_relationship$/', $table_name) ||
preg_match('/_cvterm$/', $table_name)) {
continue;
}
// Iterate through the columns of the table and see if fields have been
// created for each one. If not, then create them.
$schema = chado_get_schema($table_name);
$columns = $schema['fields'];
foreach ($columns as $column_name => $details) {
$field_name = $table_name . '__' . $column_name;
// Skip the primary key field.
if ($column_name == $schema['primary key'][0]) {
continue;
}
// Skip the type field.
if ($table_name == $type_table and $column_name == $type_field) {
continue;
}
// Get the field defaults for this column.
$field_info = tripal_entities_get_table_column_field_default($table_name, $schema, $column_name);
// Determine if the field is required.
if (array_key_exists('not null', $details) and $details['not null'] === TRUE) {
$field_info['is_required'] = array_key_exists('default', $details) ? 0 : 1;
}
// If we don't have a field type then we don't need to create a field.
if (!$field_info['field_type']) {
// If we don't have a field type but it is required and doesn't have
// a default value then we are in trouble.
if ($field_info['is_required'] and !array_key_exists('default', $details)) {
throw new Exception(t('The %table.%field type, %type, is not yet supported for Entity fields, but it is required,',
array('%table' => $table_name, '%field' => $column_name, '%type' => $details['type'])));
}
continue;
}
// If this field is a foreign key field then we will have a special custom
// field provided by Tripal.
$is_fk = FALSE;
if (array_key_exists('foreign keys', $schema)) {
foreach ($schema['foreign keys'] as $remote_table => $fk_details) {
if (array_key_exists($column_name, $fk_details['columns'])) {
$is_fk = TRUE;
}
}
}
// Add the field to the bundle.
tripal_add_bundle_field($field_name, $field_info, $entity_type_name, $bundle_name);
}
}
}
/**
* Returns a $field_info array for a field based on a databaes column.
*
*/
function tripal_entities_get_table_column_field_default($table_name, $schema, $column_name) {
$details = $schema['fields'][$column_name];
// Create an array with information about this field.
$field_info = array(
'field_type' => '',
'widget_type' => '',
'field_settings' => array(
'chado_table' => $table_name,
'chado_column' => $column_name,
),
'widget_settings' => array('display_label' => 1),
'description' => '',
'label' => ucwords(preg_replace('/_/', ' ', $column_name)),
'is_required' => 0,
);
// Alter the field info array depending on the column details.
switch($details['type']) {
case 'char':
$field_info['field_type'] = 'text';
$field_info['widget_type'] = 'text_textfield';
$field_info['field_settings']['max_length'] = $details['length'];
break;
case 'varchar':
$field_info['field_type'] = 'text';
$field_info['widget_type'] = 'text_textfield';
$field_info['field_settings']['max_length'] = $details['length'];
break;
case 'text':
$field_info['field_type'] = 'text';
$field_info['widget_type'] = 'text_textarea';
$field_info['field_settings']['max_length'] = 17179869184;
break;
case 'blob':
// not sure how to support a blob field.
continue;
break;
case 'int':
$field_info['field_type'] = 'number_integer';
$field_info['widget_type'] = 'number';
break;
case 'float':
$field_info['field_type'] = 'number_float';
$field_info['widget_type'] = 'number';
$field_info['field_settings']['precision'] = 10;
$field_info['field_settings']['scale'] = 2;
$field_info['field_settings']['decimal_separator'] = '.';
break;
case 'numeric':
$field_info['field_type'] = 'number_decimal';
$field_info['widget_type'] = 'number';
break;
case 'serial':
// Serial fields are most likely not needed as a field.
break;
case 'boolean':
$field_info['field_type'] = 'list_boolean';
$field_info['widget_type'] = 'options_onoff';
$field_info['field_settings']['allowed_values'] = array(0 => "No", 1 => "Yes");
break;
case 'datetime':
// Use the Drupal Date and Date API to create the field/widget
$field_info['field_type'] = 'datetime';
$field_info['widget_type'] = 'date_select';
$field_info['widget_settings']['increment'] = 1;
$field_info['widget_settings']['tz_handling'] = 'none';
$field_info['widget_settings']['collapsible'] = TRUE;
// TODO: Add settings so that the minutes increment by 1.
// And turn off the timezone, as the Chado field doesn't support it.
break;
}
return $field_info;
}
/**
* Adds a vocabulary and term to the Tripal term usage tables.
*
* This function is meant to be called only by the
* tripal_entities_entity_form_validate() function. This code is
* separated to simplify that function. Therefore, if errors occur with adding
* of terms then the form_set_error() is called.
*
* @param $cvterm
*/
function tripal_entities_add_term_usage($cvterm, &$form_state) {
// Before creating the entity we mut add records to the tripal_vocabulary
// tripal_vocabulary_usage, tripal_term, and tripal_term_usage tables.
$match = array('cv_id' => $cvterm->cv_id->cv_id);
$vocab = chado_select_record('tripal_vocabulary', array('*'), $match);
if (count($vocab) == 0) {
$values = array(
'cv_id' => $cvterm->cv_id->cv_id,
'db_id' => $cvterm->dbxref_id->db_id->db_id,
'publish' => 1,
);
$values = chado_insert_record('tripal_vocabulary', $values);
if (!$values) {
form_set_error('', 'Could not add vocabulary to tripal_vocabluary table.');
return FALSE;
}
// Convert the values array into an object.
$vocab = new stdClass();
$vocab->vocabulary_id = $values['vocabulary_id'];
$vocab->cv_id = $values['cv_id'];
}
else {
// Make sure the vocabulary is set to publish
$values = array('publish' => 1);
chado_update_record('tripal_vocabulary', $match, $values);
$vocab = $vocab[0];
}
// Does this vocabulary have a record in the tripal_vocabulary_usage
// table? If not then add one.
$match = array('vocabulary_id' => $vocab->vocabulary_id);
$vocab_usage = chado_select_record('tripal_vocabulary_usage', array('*'), $match);
if (count($vocab_usage) == 0) {
// Look to see if this vocabulary is used as a default for any table. If
// so then we can use that to populate the tripal_vocabulary_usage table.
$default = db_select('tripal_cv_defaults', 't')
->fields('t')
->condition('cv_id', $vocab->cv_id)
->execute()
->fetchObject();
if ($default) {
$values = array(
'vocabulary_id' => $vocab->vocabulary_id,
'data_table' => $default->table_name,
'type_table' => $default->table_name,
'field' => $default->field_name,
);
$values = chado_insert_record('tripal_vocabulary_usage', $values);
if (!$values) {
form_set_error('', 'Could not add vocabulary to tripal_vocabulary_usage table.');
return FALSE;
}
}
// If there is no default table then we have an error, and we should
// set a variable so that the form can help the user deal with the problem.
else {
$form_state['storage']['cvterm_has_default'] = FALSE;
form_set_error('', t('There is no default mapping of this term\'s
vocabulary to a table in Chado. Therefore, it is not possible to
determine how to store data of this type.'));
return FALSE;
}
$vocab_usage = new stdClass();
$vocab_usage->vocabulary_id = $values['vocabulary_id'];
$vocab_usage->data_table = $values['data_table'];
$vocab_usage->type_table = $values['type_table'];
$vocab_usage->field = $values['field'];
}
else {
$vocab_usage = $vocab_usage[0];
}
// Now add the tripal_term record if it doesn't already exist.
$match = array(
'vocabulary_id' => $vocab->vocabulary_id,
'cvterm_id' => $cvterm->cvterm_id,
);
$term = chado_select_record('tripal_term', array('*'), $match);
if (count($term) == 0) {
$values = array(
'vocabulary_id' => $vocab->vocabulary_id,
'cvterm_id' => $cvterm->cvterm_id,
);
$values = chado_insert_record('tripal_term', $values);
if (!$values) {
form_set_error('', 'Could not add term to tripal_term table..');
return FALSE;
}
$term = new stdClass();
$term->term_id = $values['term_id'];
}
else {
$values = array('publish' => 1);
chado_update_record('tripal_term', $match, $values);
$term = $term[0];
}
// Finally, add the tripal_term_usage record if it doesn't already exist.
$match = array('term_id' => $term->term_id);
$options = array('has_record' => TRUE);
if (!chado_select_record('tripal_term_usage', array('*'), $match, $options)) {
$values = array(
'term_id' => $term->term_id,
'data_table' => $vocab_usage->data_table,
'type_table' => $vocab_usage->type_table,
'field' => $vocab_usage->field,
);
$values = chado_insert_record('tripal_term_usage', $values);
if (!$values) {
form_set_error('', 'Could not add term to tripal_term table..');
return FALSE;
}
}
// Clear the entity cache so that Drupal will read our
// hook_entity_info() implementation which now will have the entities
// described because we set the publish column to 1 in the tripal_term
// table.
global $language;
$langcode = $language->language;
cache_clear_all("entity_info:$langcode", 'cache');
return TRUE;
}
/**
* Implements hook_chado_field_alter.
*
* This function is used when new Chado fields are addd to an Entity. It
* allows modules to customize the field, widget types and settings for
* a field before it is created.
*
* @param $field
*/
function hook_chado_field_alter(&$field) {
// TODO: add example code for how to use this hook.
}
/**
*
* @param unknown $form
* @param unknown $form_state
* @return multitype:
*/
function tripal_entities_admin_access_form($form, &$form_state) {
$form = array();
return $form;
}