tripal_chado.phylotree.inc 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504
  1. <?php
  2. /**
  3. * Prepares a phylogenetic tree for viewing.
  4. *
  5. * @param $phylotree
  6. */
  7. function tripal_phylogeny_prepare_tree_viewer($phylotree) {
  8. // Don't prepare for viewing more than once.
  9. if (property_exists($phylotree, 'prepared_to_view') and
  10. $phylotree->prepared_to_view == TRUE) {
  11. return;
  12. }
  13. $module_path = drupal_get_path('module', 'tripal_chado');
  14. drupal_add_js('https://d3js.org/d3.v3.min.js', 'external');
  15. drupal_add_js("$module_path/theme/js/d3.phylogram.js");
  16. drupal_add_js("$module_path/theme/js/tripal_phylogeny.js");
  17. drupal_add_css("$module_path/theme/css/tripal_phylogeny.css");
  18. drupal_add_library('system', 'ui.dialog');
  19. // Don't show tick marks for the taxonomy tree.
  20. $skip_ticks = 0;
  21. if ($phylotree->type_id->name == 'taxonomy') {
  22. $skip_ticks = 1;
  23. }
  24. // Get the tree options as set by the administrator.
  25. $options = json_encode(array(
  26. 'phylogram_width' => variable_get('tripal_phylogeny_default_phylogram_width', 350),
  27. 'root_node_size' => variable_get('tripal_phylogeny_default_root_node_size', 3),
  28. 'interior_node_size' => variable_get('tripal_phylogeny_default_interior_node_size', 0),
  29. 'leaf_node_size' => variable_get('tripal_phylogeny_default_leaf_node_size', 6),
  30. 'skipTicks' => $skip_ticks,
  31. ));
  32. // Get the node colors as set by the administrator.
  33. $colors = array();
  34. $color_defaults = variable_get("tripal_phylogeny_org_colors", array('1' => array('organism' => '', 'color' => '')));
  35. foreach ($color_defaults as $i => $details) {
  36. if ($details['organism']) {
  37. $colors[$details['organism']] = $details['color'];
  38. }
  39. }
  40. $colors = json_encode($colors);
  41. drupal_add_js(array(
  42. 'tripal_chado' => array(
  43. 'phylotree_url' => url('phylotree/' . $phylotree->phylotree_id),
  44. 'phylotree_theme_url' => url($module_path . '/theme'),
  45. 'tree_options' => $options,
  46. 'org_colors' => $colors,
  47. ),
  48. ), 'setting');
  49. if (!property_exists($phylotree, 'has_nodes')) {
  50. // If the nodes haven't loaded then set a value so the template can
  51. // choose not to show the phylogram.
  52. $values = array('phylotree_id' => $phylotree->phylotree_id);
  53. $options = array('limit' => 1, 'offset' => 0, 'has_record' => 1);
  54. $phylotree->has_nodes = chado_select_record('phylonode', array('phylonode_id'), $values, $options);
  55. }
  56. if (!property_exists($phylotree, 'has_features')) {
  57. // If the nodes haven't loaded then set a value so the template can
  58. // choose not to show the circular dendrogram. The chado_select_record()
  59. // API call can't do this query so we have to do it manually.
  60. $sql = "
  61. SELECT count(*) as num_features
  62. FROM {phylonode}
  63. WHERE NOT feature_id IS NULL and phylotree_id = :phylotree_id
  64. LIMIT 1 OFFSET 0
  65. ";
  66. $phylotree->has_features = chado_query($sql, array(':phylotree_id' => $phylotree->phylotree_id))->fetchField();
  67. }
  68. $phylotree->prepared_to_view = TRUE;
  69. }
  70. /**
  71. * Get json representation of a phylotree id.
  72. *
  73. * This function is meant to be called via AJAX.
  74. *
  75. * @param int $phylotree_id
  76. * the ID of the phylotree node.
  77. *
  78. * @return string json
  79. *
  80. * @ingroup tripal_phylogeny
  81. */
  82. function tripal_phylogeny_ajax_get_tree_json($phylotree_id) {
  83. $phylotree = chado_generate_var('phylotree', array('phylotree_id' => $phylotree_id));
  84. // For backwards compatibility with Tripal v2 and the legacy modules of
  85. // Tripal v3 we have two different SQL statements.
  86. if (module_exists('tripal_phylogeny')) {
  87. // This SQL gets all of the phylonodes for a given tree as well as the
  88. // features and organisms with which it is associated. Each phylonode
  89. // can be associated with an organism in one of two ways: 1) via a
  90. // feature linked by the phylonode.feature_id field or 2) via a
  91. // a record in the phylonde_organsim table. Therefore both types of
  92. // organism records are returned in the query below, but those
  93. // retrieved via a FK link on features are prefixed with 'fo_'.
  94. $sql = "
  95. SELECT
  96. n.phylonode_id, n.parent_phylonode_id, n.label AS name, n.distance AS length,
  97. f.feature_id, f.name AS feature_name,
  98. cvt.name AS cvterm_name,
  99. o.organism_id, o.common_name, o.abbreviation, o.genus, o.species,
  100. fo.organism_id AS fo_organism_id, fo.common_name AS fo_common_name,
  101. fo.abbreviation AS fo_abbreviation, fo.genus as fo_genus, fo.species AS fo_species,
  102. cf.nid AS feature_nid,
  103. fco.nid AS fo_organism_nid,
  104. co.nid AS organism_nid
  105. FROM {phylonode} n
  106. LEFT OUTER JOIN {cvterm} cvt ON n.type_id = cvt.cvterm_id
  107. LEFT OUTER JOIN {feature} f ON n.feature_id = f.feature_id
  108. LEFT OUTER JOIN [chado_feature] cf ON cf.feature_id = f.feature_id
  109. LEFT OUTER JOIN {organism} fo ON f.organism_id = fo.organism_id
  110. LEFT OUTER JOIN [chado_organism] fco ON fco.organism_id = fo.organism_id
  111. LEFT OUTER JOIN {phylonode_organism} po ON po.phylonode_id = n.phylonode_id
  112. LEFT OUTER JOIN {organism} o ON PO.organism_id = o.organism_id
  113. LEFT OUTER JOIN [chado_organism] co ON co.organism_id = o.organism_id
  114. WHERE n.phylotree_id = :phylotree_id
  115. ";
  116. }
  117. else {
  118. $sql = "
  119. SELECT
  120. n.phylonode_id, n.parent_phylonode_id, n.label AS name, n.distance AS length,
  121. f.feature_id, f.name AS feature_name,
  122. cvt.name AS cvterm_name,
  123. o.organism_id, o.common_name, o.abbreviation, o.genus, o.species,
  124. fo.organism_id AS fo_organism_id, fo.common_name AS fo_common_name,
  125. fo.abbreviation AS fo_abbreviation, fo.genus as fo_genus, fo.species AS fo_species
  126. FROM {phylonode} n
  127. LEFT OUTER JOIN {cvterm} cvt ON n.type_id = cvt.cvterm_id
  128. LEFT OUTER JOIN {feature} f ON n.feature_id = f.feature_id
  129. LEFT OUTER JOIN {organism} fo ON f.organism_id = fo.organism_id
  130. LEFT OUTER JOIN {phylonode_organism} po ON po.phylonode_id = n.phylonode_id
  131. LEFT OUTER JOIN {organism} o ON PO.organism_id = o.organism_id
  132. WHERE n.phylotree_id = :phylotree_id
  133. ";
  134. }
  135. $args = array(':phylotree_id' => $phylotree_id);
  136. $results = chado_query($sql, $args);
  137. // Fetch all the phylonodes into an assoc array indexed by phylonode_id.
  138. // Convert from resultset record to array, fixing datatypes. chado_query
  139. // returns numeric as string and fun stuff like that.
  140. $phylonodes = array();
  141. $root_phylonode_ref = null;
  142. if ($results) {
  143. while($r = $results->fetchObject()) {
  144. $phylonode_id = (int) $r->phylonode_id;
  145. // expect all nodes to have these properties
  146. $node = array(
  147. 'phylonode_id' => $phylonode_id,
  148. 'parent_phylonode_id' => (int) $r->parent_phylonode_id,
  149. 'length' => (double) $r->length,
  150. 'cvterm_name' => $r->cvterm_name
  151. );
  152. // If the nodes are taxonomic then set an equal distance
  153. if ($phylotree->type_id->name == 'taxonomy') {
  154. $node['length'] = 0.001;
  155. }
  156. // Other props may exist only for leaf nodes
  157. if ($r->name) {
  158. $node['name'] = $r->name;
  159. }
  160. // If this node is associated with a feature then add in the details
  161. if ($r->feature_id) {
  162. $node['feature_id'] = (int) $r->feature_id;
  163. $node['feature_name'] = $r->feature_name;
  164. if (module_exists('tripal_phylogeny')) {
  165. $node['feature_nid'] = (int) $r->feature_nid;
  166. }
  167. }
  168. // Add in the organism fields when they are available via the
  169. // phylonode_organism table.
  170. if ($r->organism_id) {
  171. $node['organism_id'] = (int) $r->organism_id;
  172. $node['common_name'] = $r->common_name;
  173. $node['abbreviation'] = $r->abbreviation;
  174. $node['genus'] = $r->genus;
  175. $node['species'] = $r->species;
  176. if (module_exists('tripal_phylogeny')) {
  177. $node['organism_nid'] = (int) $r->organism_nid;
  178. }
  179. // If the node does not have a name but is linked to an organism
  180. // then set the name to be that of the genus and species.
  181. if (!$r->name) {
  182. $node['name'] = $r->genus . ' ' . $r->species;
  183. }
  184. }
  185. // Add in the organism fields when they are available via the
  186. // the phylonode.feature_id FK relationship.
  187. if ($r->fo_organism_id) {
  188. $node['fo_organism_id'] = (int) $r->fo_organism_id;
  189. $node['fo_common_name'] = $r->fo_common_name;
  190. $node['fo_abbreviation'] = $r->fo_abbreviation;
  191. $node['fo_genus'] = $r->fo_genus;
  192. $node['fo_species'] = $r->fo_species;
  193. if (module_exists('tripal_phylogeny')) {
  194. $node['fo_organism_nid'] = (int) $r->fo_organism_nid;
  195. }
  196. }
  197. // Add this node to the list, organized by ID.
  198. $phylonodes[$phylonode_id] = $node;
  199. }
  200. // Populate the children[] arrays for each node.
  201. foreach ($phylonodes as $key => &$node) {
  202. if ($node['parent_phylonode_id'] !== 0) {
  203. $parent_ref = &$phylonodes[ $node['parent_phylonode_id']];
  204. // Append node refernce to children.
  205. $parent_ref['children'][] = &$node;
  206. }
  207. else {
  208. $root_phylonode_ref = &$node;
  209. }
  210. }
  211. }
  212. // dump datastructure as json to browser. drupal sets the mime-type correctly.
  213. drupal_json_output($root_phylonode_ref);
  214. }
  215. /**
  216. * @file
  217. * This file contains the functions used for administration of the module
  218. *
  219. */
  220. function tripal_phylogeny_admin_phylotrees_listing() {
  221. $output = '';
  222. // set the breadcrumb
  223. $breadcrumb = array();
  224. $breadcrumb[] = l('Home', '<front>');
  225. $breadcrumb[] = l('Administration', 'admin');
  226. $breadcrumb[] = l('Tripal', 'admin/tripal');
  227. $breadcrumb[] = l('Data Storage', 'admin/tripal/storage');
  228. $breadcrumb[] = l('Chado', 'admin/tripal/storage/chado');
  229. drupal_set_breadcrumb($breadcrumb);
  230. // Add the view
  231. $view = views_embed_view('tripal_phylogeny_admin_phylotree','default');
  232. if (isset($view)) {
  233. $output .= $view;
  234. }
  235. else {
  236. $output .= '<p>The Phylotree module uses primarily views to provide an '
  237. . 'administrative interface. Currently one or more views needed for this '
  238. . 'administrative interface are disabled. <strong>Click each of the following links to '
  239. . 'enable the pertinent views</strong>:</p>';
  240. $output .= '<ul>';
  241. $output .= '<li>'.l('Phylotree View', 'admin/tripal/extension/tripal_phylogeny/views/phylotree/enable').'</li>';
  242. $output .= '</ul>';
  243. }
  244. return $output;
  245. }
  246. /**
  247. *
  248. * @param unknown $form
  249. * @param unknown $form_state
  250. */
  251. function tripal_phylogeny_default_plots_form($form, &$form_state) {
  252. $form = array();
  253. $form['plot_settings'] = array(
  254. '#type' => 'fieldset',
  255. '#title' => t('Plot Settings'),
  256. '#description' => t('You can customize settings for each plot'),
  257. '#collapsible' => TRUE,
  258. '#collapsed' => FALSE
  259. );
  260. $form['plot_settings']['phylogram_width'] = array(
  261. '#type' => 'textfield',
  262. '#title' => 'Tree Width',
  263. '#description' => 'Please specify the width in pixels for the phylogram',
  264. '#default_value' => variable_get('tripal_phylogeny_default_phylogram_width', 350),
  265. '#element_validate' => array(
  266. 'element_validate_integer_positive'
  267. ),
  268. '#size' => 5,
  269. );
  270. $form['node_settings'] = array(
  271. '#type' => 'fieldset',
  272. '#title' => t('Node Settings'),
  273. '#description' => t('You can customize settings for the nodes on the trees.'),
  274. '#collapsible' => TRUE,
  275. '#collapsed' => FALSE
  276. );
  277. $form['node_settings']['root_node_size'] = array(
  278. '#type' => 'textfield',
  279. '#title' => 'Root Node Size',
  280. '#description' => 'Please specify a size for the root node size. If set to zero, the node will not appear.',
  281. '#default_value' => variable_get('tripal_phylogeny_default_root_node_size', 3),
  282. '#element_validate' => array(
  283. 'element_validate_integer'
  284. ),
  285. '#size' => 3,
  286. );
  287. $form['node_settings']['interior_node_size'] = array(
  288. '#type' => 'textfield',
  289. '#title' => 'Interor Node Size',
  290. '#description' => 'Please specify a size for the interior node size. If set to zero, the node will not appear.',
  291. '#default_value' => variable_get('tripal_phylogeny_default_interior_node_size', 0),
  292. '#element_validate' => array(
  293. 'element_validate_integer'
  294. ),
  295. '#size' => 3,
  296. );
  297. $form['node_settings']['leaf_node_size'] = array(
  298. '#type' => 'textfield',
  299. '#title' => 'Leaf Node Size',
  300. '#description' => 'Please specify a size for the leaf node size. If set to zero, the node will not appear.',
  301. '#default_value' => variable_get('tripal_phylogeny_default_leaf_node_size', 6),
  302. '#element_validate' => array(
  303. 'element_validate_integer'
  304. ),
  305. '#size' => 3,
  306. );
  307. // Get the number of organism colors that already exist. If the site admin
  308. // has set colors then those settings will be in a Drupal variable which we
  309. // will retrieve. Otherwise the num_orgs defaults to 1 and a single
  310. // set of fields is provided.
  311. $num_orgs = variable_get("tripal_phylogeny_num_orgs", 1);
  312. if (array_key_exists('values', $form_state) and array_key_exists('num_orgs', $form_state['values'])) {
  313. $num_orgs = $form_state['values']['num_orgs'];
  314. }
  315. // The default values for each organism color are provided in a d
  316. // Drupal variable that gets set when the form is set.
  317. $color_defaults = variable_get("tripal_phylogeny_org_colors", array('1' => array('organism' => '', 'color' => '')));
  318. $form['node_settings']['desc'] = array(
  319. '#type' => 'item',
  320. '#title' => t('Node Colors by Organism'),
  321. '#markup' => t('If the trees are associated with features (e.g. proteins)
  322. then the nodes can be color-coded by their organism. This helps the user
  323. visualize which nodes belong to each organism. Please enter the
  324. name of the organism and it\'s corresponding color in HEX code (e.g. #FF0000 == red).
  325. Organisms that are not given a color will be gray.'),
  326. );
  327. $form['node_settings']['org_table']['num_orgs'] = array(
  328. '#type' => 'value',
  329. '#value' => $num_orgs,
  330. );
  331. // Iterate through the number of organism colors and add a field for each one.
  332. for ($i = 0; $i < $num_orgs; $i++) {
  333. $form['node_settings']['org_table']['organism_' . $i] = array(
  334. '#type' => 'textfield',
  335. '#default_value' => array_key_exists($i, $color_defaults) ? $color_defaults[$i]['organism'] : '',
  336. '#autocomplete_path' => "admin/tripal/legacy/tripal_organism/organism/auto_name",
  337. '#description' => t('Please enter the name of the organism.'),
  338. '#size' => 30,
  339. );
  340. $form['node_settings']['org_table']['color_' . $i] = array(
  341. '#type' => 'textfield',
  342. '#description' => t('Please provide a color in Hex format (e.g. #FF0000).'),
  343. '#default_value' => array_key_exists($i, $color_defaults) ? $color_defaults[$i]['color'] : '',
  344. '#suffix' => "<div id=\"color-box-$i\" style=\"width: 30px;\"></div>",
  345. '#size' => 10,
  346. );
  347. }
  348. $form['node_settings']['org_table']['add'] = array(
  349. '#type' => 'submit',
  350. '#name' => 'add',
  351. '#value' => 'Add',
  352. '#ajax' => array(
  353. 'callback' => "tripal_phylogeny_default_plots_form_ajax_callback",
  354. 'wrapper' => 'tripal_phylogeny_default_plots_form',
  355. 'effect' => 'fade',
  356. 'method' => 'replace',
  357. ),
  358. );
  359. $form['node_settings']['org_table']['remove'] = array(
  360. '#type' => 'submit',
  361. '#name' => 'remove',
  362. '#value' => 'Remove',
  363. '#ajax' => array(
  364. 'callback' => "tripal_phylogeny_default_plots_form_ajax_callback",
  365. 'wrapper' => 'tripal_phylogeny_default_plots_form',
  366. 'effect' => 'fade',
  367. 'method' => 'replace',
  368. ),
  369. );
  370. $form['node_settings']['org_table']['#theme'] = 'tripal_phylogeny_admin_org_color_tables';
  371. $form['node_settings']['org_table']['#prefix'] = '<div id="tripal_phylogeny_default_plots_form">';
  372. $form['node_settings']['org_table']['#suffix'] = '</div>';
  373. $form['submit'] = array(
  374. '#type' => 'submit',
  375. '#name' => 'submit',
  376. '#value' => 'Save Configuration',
  377. );
  378. $form['#submit'][] = 'tripal_phylogeny_default_plots_form_submit';
  379. return $form;
  380. }
  381. /**
  382. * Validate the phylotree settings forms
  383. *
  384. * @ingroup tripal_phylogeny
  385. */
  386. function tripal_phylogeny_default_plots_form_validate($form, &$form_state) {
  387. }
  388. /**
  389. *
  390. * @param unknown $form
  391. * @param unknown $form_state
  392. */
  393. function tripal_phylogeny_default_plots_form_submit($form, &$form_state) {
  394. // Rebuild this form after submission so that any changes are reflected in
  395. // the flat tables.
  396. $form_state['rebuild'] = TRUE;
  397. if ($form_state['clicked_button']['#name'] == 'submit') {
  398. variable_set('tripal_phylogeny_default_phylogram_width', $form_state['values']['phylogram_width']);
  399. variable_set('tripal_phylogeny_default_root_node_size', $form_state['values']['root_node_size']);
  400. variable_set('tripal_phylogeny_default_interior_node_size', $form_state['values']['interior_node_size']);
  401. variable_set('tripal_phylogeny_default_leaf_node_size', $form_state['values']['leaf_node_size']);
  402. $num_orgs = $form_state['values']['num_orgs'];
  403. variable_set("tripal_phylogeny_num_orgs", $num_orgs);
  404. $colors = array();
  405. for ($i = 0; $i < $num_orgs ;$i++) {
  406. $colors[$i] = array(
  407. 'organism' => $form_state['values']['organism_' . $i],
  408. 'color' => $form_state['values']['color_' . $i]
  409. );
  410. }
  411. variable_set("tripal_phylogeny_org_colors", $colors);
  412. }
  413. if ($form_state['clicked_button']['#name'] == 'add') {
  414. $form_state['values']['num_orgs']++;
  415. }
  416. if ($form_state['clicked_button']['#name'] == 'remove') {
  417. $form_state['values']['num_orgs']--;
  418. }
  419. }
  420. /**
  421. *
  422. * @param unknown $variables
  423. */
  424. function theme_tripal_phylogeny_admin_org_color_tables($variables){
  425. $fields = $variables['element'];
  426. $num_orgs = $fields['num_orgs']['#value'];
  427. $headers = array('Organism', 'Color', '');
  428. $rows = array();
  429. for ($i = 0; $i < $num_orgs; $i++) {
  430. $add_button = ($i == $num_orgs - 1) ? drupal_render($fields['add']) : '';
  431. $del_button = ($i == $num_orgs - 1 and $i != 0) ? drupal_render($fields['remove']) : '';
  432. $rows[] = array(
  433. drupal_render($fields['organism_' . $i]),
  434. drupal_render($fields['color_' . $i]),
  435. $add_button . $del_button,
  436. );
  437. }
  438. $table_vars = array(
  439. 'header' => $headers,
  440. 'rows' => $rows,
  441. 'attributes' => array(),
  442. 'sticky' => FALSE,
  443. 'colgroups' => array(),
  444. 'empty' => '',
  445. );
  446. $form['orgs']['num_orgs'] = $fields['num_orgs'];
  447. return theme('table', $table_vars);
  448. }
  449. /**
  450. * Ajax callback function for the gensas_job_view_panel_form.
  451. *
  452. * @param $form
  453. * @param $form_state
  454. */
  455. function tripal_phylogeny_default_plots_form_ajax_callback($form, $form_state) {
  456. return $form['node_settings']['org_table'];
  457. }