data__sequence.inc 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. <?php
  2. class data__sequence extends TripalField {
  3. // --------------------------------------------------------------------------
  4. // EDITABLE STATIC CONSTANTS
  5. //
  6. // The following constants SHOULD be set for each descendent class. They are
  7. // used by the static functions to provide information to Drupal about
  8. // the field and it's default widget and formatter.
  9. // --------------------------------------------------------------------------
  10. // The term that this field maps to. The format for the term should be:
  11. // [vocab]:[accession] where [vocab] is the short name of the vocabulary
  12. // and [acession] is the unique accession number for the term. This term
  13. // must already exist in the vocabulary storage backend. This
  14. // value should never be changed once fields exist for this type.
  15. public static $term = 'data:2044';
  16. // The default lable for this field.
  17. public static $label = 'Sequence';
  18. // The default description for this field.
  19. public static $description = 'A field for managing nucleotide and protein residues.';
  20. // Provide a list of global settings. These can be accessed witihn the
  21. // globalSettingsForm. When the globalSettingsForm is submitted then
  22. // Drupal will automatically change these settings for all fields.
  23. public static $settings = array(
  24. 'chado_table' => '',
  25. 'chado_column' => '',
  26. 'base_table' => '',
  27. );
  28. // Provide a list of instance specific settings. These can be access within
  29. // the instanceSettingsForm. When the instanceSettingsForm is submitted
  30. // then Drupal with automatically change these settings for the instnace.
  31. // It is recommended to put settings at the instance level whenever possible.
  32. public static $instance_settings = array();
  33. // Set this to the name of the storage backend that by default will support
  34. // this field.
  35. public static $storage = 'tripal_no_storage';
  36. // The default widget for this field.
  37. public static $default_widget = 'data__sequence_widget';
  38. // The default formatter for this field.
  39. public static $default_formatter = 'data__sequence_formatter';
  40. // --------------------------------------------------------------------------
  41. // PROTECTED CLASS MEMBERS -- DO NOT OVERRIDE
  42. // --------------------------------------------------------------------------
  43. // An array containing details about the field. The format of this array
  44. // is the same as that returned by field_info_fields()
  45. protected $field;
  46. // An array containing details about an instance of the field. A field does
  47. // not have to have an instance. But if dealing with an instance (such as
  48. // when using the widgetForm, formatterSettingsForm, etc.) it should be set.
  49. protected $instance;
  50. public function load($entity, $details = array()) {
  51. $field_name = $this->field['field_name'];
  52. $feature = $details['record'];
  53. $num_seqs = 0;
  54. // We don't want to get the sequence for traditionally large types. They are
  55. // too big, bog down the web browser, take longer to load and it's not
  56. // reasonable to print them on a page.
  57. if(strcmp($feature->type_id->name,'scaffold') == 0 or
  58. strcmp($feature->type_id->name,'chromosome') == 0 or
  59. strcmp($feature->type_id->name,'supercontig') == 0 or
  60. strcmp($feature->type_id->name,'pseudomolecule') == 0) {
  61. $entity->{$field_name}['und'][$num_seqs]['value'] = array(
  62. '@type' => 'SO:0000110',
  63. 'type' => 'sequence_feature',
  64. 'label' => 'Residues',
  65. 'defline' => ">This sequence is too large for this display.",
  66. 'residues' => '',
  67. );
  68. $entity->{$field_name}['und'][$num_seqs]['chado-feature__residues'] = '';
  69. }
  70. else {
  71. $feature = chado_expand_var($feature,'field','feature.residues');
  72. if ($feature->residues) {
  73. $entity->{$field_name}['und'][$num_seqs]['value'] = array(
  74. '@type' => 'SO:0000110',
  75. 'type' => 'sequence_feature',
  76. 'label' => 'Raw Sequence',
  77. 'defline' => tripal_get_fasta_defline($feature, '', NULL, '', strlen($feature->residues)),
  78. 'residues' => $feature->residues,
  79. );
  80. $entity->{$field_name}['und'][$num_seqs]['chado-feature__residues'] = $feature->residues;
  81. }
  82. else {
  83. $entity->{$field_name}['und'][$num_seqs]['value'] = array();
  84. $entity->{$field_name}['und'][$num_seqs]['chado-feature__residues'] = '';
  85. }
  86. }
  87. $num_seqs++;
  88. // Add in the protein sequences. It's faster to provide the SQL rather than
  89. // to use chado_generate_var based on the type.
  90. $sql = "
  91. SELECT F.*
  92. FROM {feature_relationship} FR
  93. INNER JOIN {feature} F on FR.subject_id = F.feature_id
  94. INNER JOIN {cvterm} CVT on CVT.cvterm_id = F.type_id
  95. INNER JOIN {cvterm} RCVT on RCVT.cvterm_id = FR.type_id
  96. WHERE
  97. FR.object_id = :feature_id and
  98. CVT.name = 'polypeptide' and
  99. RCVT.name = 'derives_from'
  100. ORDER BY FR.rank ASC
  101. ";
  102. $results = chado_query($sql, array(':feature_id' => $feature->feature_id));
  103. while ($protein = $results->fetchObject()) {
  104. if ($protein->residues) {
  105. $entity->{$field_name}['und'][$num_seqs++]['value'] = array(
  106. '@type' => 'SO:0000104',
  107. 'type' => 'polypeptide',
  108. 'label' => 'Protein Sequence',
  109. 'defline' => tripal_get_fasta_defline($protein, '', NULL, '', strlen($protein->residues)),
  110. 'residues' => $protein->residues,
  111. );
  112. }
  113. }
  114. // Add in sequences from alignments.
  115. $options = array(
  116. 'return_array' => 1,
  117. 'include_fk' => array(
  118. 'srcfeature_id' => array(
  119. 'type_id' => 1
  120. ),
  121. 'feature_id' => array(
  122. 'type_id' => 1
  123. ),
  124. ),
  125. );
  126. $feature = chado_expand_var($feature, 'table', 'featureloc', $options);
  127. $featureloc_sequences = $this->get_featureloc_sequences($feature->feature_id, $feature->featureloc->feature_id);
  128. // Add in the coding sequences. It's faster to provide the SQL rather than
  129. // to use chado_generate_var based on the type.
  130. $sql = "
  131. SELECT F.*
  132. FROM {feature_relationship} FR
  133. INNER JOIN {feature} F on FR.subject_id = F.feature_id
  134. INNER JOIN {cvterm} CVT on CVT.cvterm_id = F.type_id
  135. INNER JOIN {cvterm} RCVT on RCVT.cvterm_id = FR.type_id
  136. INNER JOIN {featureloc} FL on FL.feature_id = F.feature_id
  137. WHERE
  138. FR.object_id = :feature_id and
  139. CVT.name = 'CDS' and
  140. RCVT.name = 'part_of'
  141. ORDER BY FR.rank ASC
  142. ";
  143. $results = chado_query($sql, array(':feature_id' => $feature->feature_id));
  144. $coding_seq = '';
  145. while ($CDS = $results->fetchObject()) {
  146. if ($CDS->residues) {
  147. $coding_seq .= $CDS->residues;
  148. }
  149. }
  150. if ($coding_seq) {
  151. $entity->{$field_name}['und'][$num_seqs++]['value'] = array(
  152. '@type' => 'SO:0000316',
  153. 'type' => 'coding_sequence',
  154. 'label' => 'Coding sequence (CDS)',
  155. 'defline' => tripal_get_fasta_defline($feature, 'CDS', NULL, '', strlen($coding_seq)),
  156. 'residues' => $coding_seq,
  157. );
  158. }
  159. foreach($featureloc_sequences as $src => $attrs){
  160. // the $attrs array has the following keys
  161. // * id: a unique identifier combining the feature id with the cvterm id
  162. // * type: the type of sequence (e.g. mRNA, etc)
  163. // * location: the alignment location
  164. // * defline: the definition line
  165. // * formatted_seq: the formatted sequences
  166. // * featureloc: the feature object aligned to
  167. $entity->{$field_name}['und'][$num_seqs++]['value'] = array(
  168. 'residues' => $attrs['residues'],
  169. '@type' => 'SO:0000110',
  170. 'type' => 'sequence_feature',
  171. 'defline' => tripal_get_fasta_defline($feature, '', $attrs['featureloc'], 'CDS', strlen($attrs['residues'])),
  172. 'label' => 'Sequence from alignment at ' . $attrs['location'],
  173. );
  174. // check to see if this alignment has any CDS. If so, generate a CDS sequence
  175. $cds_sequence = tripal_get_feature_sequences(
  176. array(
  177. 'feature_id' => $feature->feature_id,
  178. 'parent_id' => $attrs['featureloc']->srcfeature_id->feature_id,
  179. 'name' => $feature->name,
  180. 'featureloc_id' => $attrs['featureloc']->featureloc_id,
  181. ),
  182. array(
  183. 'derive_from_parent' => 1, // CDS are in parent-child relationships so we want to use the sequence from the parent
  184. 'aggregate' => 1, // we want to combine all CDS for this feature into a single sequence
  185. 'sub_feature_types' => array('CDS'), // we're looking for CDS features
  186. 'is_html' => 0
  187. )
  188. );
  189. if (count($cds_sequence) > 0) {
  190. // the tripal_get_feature_sequences() function can return multiple sequences
  191. // if a feature is aligned to multiple places. In the case of CDSs we expect
  192. // that one mRNA is only aligned to a single location on the assembly so we
  193. // can access the CDS sequence with index 0.
  194. if ($cds_sequence[0]['residues']) {
  195. $entity->{$field_name}['und'][$num_seqs++]['value'] = array(
  196. 'residues' => $cds_sequence[0]['residues'],
  197. '@type' => 'SO:0000316',
  198. 'type' => 'coding_sequence',
  199. 'defline' => tripal_get_fasta_defline($feature, '', $attrs['featureloc'], 'CDS', $cds_sequence[0]['length']),
  200. 'label' => 'Coding sequence (CDS) from alignment at ' . $attrs['location'],
  201. );
  202. }
  203. }
  204. }
  205. }
  206. /**
  207. *
  208. * @param unknown $feature_id
  209. * @param unknown $featurelocs
  210. * @return multitype:|Ambigous <multitype:, an>
  211. */
  212. private function get_featureloc_sequences($feature_id, $featurelocs) {
  213. // if we don't have any featurelocs then no point in continuing
  214. if (!$featurelocs) {
  215. return array();
  216. }
  217. // get the list of relationships (including any aggregators) and iterate
  218. // through each one to find information needed to color-code the reference sequence
  219. $relationships = $this->get_aggregate_relationships($feature_id);
  220. if (!$relationships) {
  221. return array();
  222. }
  223. // iterate through each of the realtionships features and get their
  224. // locations
  225. foreach ($relationships as $rindex => $rel) {
  226. // get the featurelocs for each of the relationship features
  227. $rel_featurelocs = $this->get_featurelocs($rel->subject_id, 'as_child', 0);
  228. foreach ($rel_featurelocs as $rfindex => $rel_featureloc) {
  229. // keep track of this unique source feature
  230. $src = $rel_featureloc->src_feature_id . "-" . $rel_featureloc->src_cvterm_id;
  231. // copy over the results to the relationship object. Since there can
  232. // be more than one feature location for each relationship feature we
  233. // use the '$src' variable to keep track of these.
  234. $rel->featurelocs = new stdClass();
  235. $rel->featurelocs->$src = new stdClass();
  236. $rel->featurelocs->$src->src_uniquename = $rel_featureloc->src_uniquename;
  237. $rel->featurelocs->$src->src_cvterm_id = $rel_featureloc->src_cvterm_id;
  238. $rel->featurelocs->$src->src_cvname = $rel_featureloc->src_cvname;
  239. $rel->featurelocs->$src->fmin = $rel_featureloc->fmin;
  240. $rel->featurelocs->$src->fmax = $rel_featureloc->fmax;
  241. $rel->featurelocs->$src->src_name = $rel_featureloc->src_name;
  242. // keep track of the individual parts for each relationship
  243. $start = $rel->featurelocs->$src->fmin;
  244. $end = $rel->featurelocs->$src->fmax;
  245. $type = $rel->subject_type;
  246. $rel_locs[$src]['parts'][$start][$type]['start'] = $start;
  247. $rel_locs[$src]['parts'][$start][$type]['end'] = $end;
  248. $rel_locs[$src]['parts'][$start][$type]['type'] = $type;
  249. }
  250. }
  251. // the featurelocs array provided to the function contains the locations
  252. // where this feature is found. We want to get the sequence for each
  253. // location and then annotate it with the parts found from the relationships
  254. // locations determiend above.
  255. $floc_sequences = array();
  256. foreach ($featurelocs as $featureloc) {
  257. // build the src name so we can keep track of the different parts for each feature
  258. $src = $featureloc->srcfeature_id->feature_id . "-" . $featureloc->srcfeature_id->type_id->cvterm_id;
  259. // orient the parts to the beginning of the feature sequence
  260. if (!empty($rel_locs[$src]['parts'])) {
  261. $parts = $rel_locs[$src]['parts'];
  262. $rparts = array(); // we will fill this up if we're on the reverse strand
  263. foreach ($parts as $start => $types) {
  264. foreach ($types as $type_name => $type) {
  265. if ($featureloc->strand >= 0) {
  266. // this is on the forward strand. We need to convert the start on the src feature to the
  267. // start on this feature's sequence
  268. $parts[$start][$type_name]['start'] = $parts[$start][$type_name]['start'] - $featureloc->fmin;
  269. $parts[$start][$type_name]['end'] = $parts[$start][$type_name]['end'] - $featureloc->fmin;
  270. $parts[$start][$type_name]['type'] = $type_name;
  271. }
  272. else {
  273. // this is on the reverse strand. We need to swap the start and stop and calculate from the
  274. // begining of the reverse sequence
  275. $size = ($featureloc->fmax - $featureloc->fmin);
  276. $start_orig = $parts[$start][$type_name]['start'];
  277. $end_orig = $parts[$start][$type_name]['end'];
  278. $new_start = $size - ($end_orig - $featureloc->fmin);
  279. $new_end = $size - ($start_orig - $featureloc->fmin);
  280. $rparts[$new_start][$type_name]['start'] = $new_start;
  281. $rparts[$new_start][$type_name]['end'] = $new_end;
  282. $rparts[$new_start][$type_name]['type'] = $type_name;
  283. }
  284. }
  285. }
  286. // now sort the parts
  287. // if we're on the reverse strand we need to resort
  288. if ($featureloc->strand >= 0) {
  289. usort($parts, 'chado_feature__residues_sort_rel_parts_by_start');
  290. }
  291. else {
  292. usort($rparts, 'chado_feature__residues_sort_rel_parts_by_start');
  293. $parts = $rparts;
  294. }
  295. $floc_sequences[$src]['id'] = $src;
  296. $floc_sequences[$src]['type'] = $featureloc->feature_id->type_id->name;
  297. $args = array(':feature_id' => $featureloc->srcfeature_id->feature_id);
  298. $start = $featureloc->fmin + 1;
  299. $size = $featureloc->fmax - $featureloc->fmin;
  300. // TODO: fix the hard coded $start and $size
  301. // the $start and $size variables are hard-coded in the SQL statement
  302. // because the db_query function places quotes around all placeholders
  303. // (e.g. :start & :size) and screws up the substring function
  304. $sql = "
  305. SELECT substring(residues from $start for $size) as residues
  306. FROM {feature}
  307. WHERE feature_id = :feature_id
  308. ";
  309. $sequence = chado_query($sql, $args)->fetchObject();
  310. $residues = $sequence->residues;
  311. if ($featureloc->strand < 0) {
  312. $residues = tripal_reverse_compliment_sequence($residues);
  313. }
  314. $strand = '.';
  315. if ($featureloc->strand == 1) {
  316. $strand = '+';
  317. }
  318. elseif ($featureloc->strand == -1) {
  319. $strand = '-';
  320. }
  321. $floc_sequences[$src]['location'] = tripal_get_location_string($featureloc);
  322. $floc_sequences[$src]['defline'] = tripal_get_fasta_defline($featureloc->feature_id, '', $featureloc, '', strlen($residues));
  323. $floc_sequences[$src]['featureloc'] = $featureloc;
  324. $floc_sequences[$src]['residues'] = $residues;
  325. //$floc_sequences[$src]['formatted_seq'] = tripal_feature_color_sequence($residues, $parts, $floc_sequences[$src]['defline']);
  326. }
  327. }
  328. return $floc_sequences;
  329. }
  330. /**
  331. * Get features related to the current feature to a given depth. Recursive function.
  332. *
  333. * @param $feature_id
  334. * @param $substitute
  335. * @param $levels
  336. * @param $base_type_id
  337. * @param $depth
  338. *
  339. * @ingroup tripal_feature
  340. */
  341. private function get_aggregate_relationships($feature_id, $substitute=1,
  342. $levels=0, $base_type_id=NULL, $depth=0) {
  343. // we only want to recurse to as many levels deep as indicated by the
  344. // $levels variable, but only if this variable is > 0. If 0 then we
  345. // recurse until we reach the end of the relationships tree.
  346. if ($levels > 0 and $levels == $depth) {
  347. return NULL;
  348. }
  349. // first get the relationships for this feature
  350. return $this->get_relationships($feature_id, 'as_object');
  351. }
  352. /**
  353. * Get the relationships for a feature.
  354. *
  355. * @param $feature_id
  356. * The feature to get relationships for
  357. * @param $side
  358. * The side of the relationship this feature is (ie: 'as_subject' or 'as_object')
  359. *
  360. * @ingroup tripal_feature
  361. */
  362. private function get_relationships($feature_id, $side = 'as_subject') {
  363. // get the relationships for this feature. The query below is used for both
  364. // querying the object and subject relationships
  365. $sql = "
  366. SELECT
  367. FS.name as subject_name, FS.uniquename as subject_uniquename,
  368. CVTS.name as subject_type, CVTS.cvterm_id as subject_type_id,
  369. FR.subject_id, FR.type_id as relationship_type_id, FR.object_id, FR.rank,
  370. CVT.name as rel_type,
  371. FO.name as object_name, FO.uniquename as object_uniquename,
  372. CVTO.name as object_type, CVTO.cvterm_id as object_type_id
  373. FROM {feature_relationship} FR
  374. INNER JOIN {cvterm} CVT ON FR.type_id = CVT.cvterm_id
  375. INNER JOIN {feature} FS ON FS.feature_id = FR.subject_id
  376. INNER JOIN {feature} FO ON FO.feature_id = FR.object_id
  377. INNER JOIN {cvterm} CVTO ON FO.type_id = CVTO.cvterm_id
  378. INNER JOIN {cvterm} CVTS ON FS.type_id = CVTS.cvterm_id
  379. ";
  380. if (strcmp($side, 'as_object')==0) {
  381. $sql .= " WHERE FR.object_id = :feature_id";
  382. }
  383. if (strcmp($side, 'as_subject')==0) {
  384. $sql .= " WHERE FR.subject_id = :feature_id";
  385. }
  386. $sql .= " ORDER BY FR.rank";
  387. // get the relationships
  388. $results = chado_query($sql, array(':feature_id' => $feature_id));
  389. // iterate through the relationships, put these in an array and add
  390. // in the Drupal node id if one exists
  391. $i=0;
  392. $esql = "
  393. SELECT entity_id
  394. FROM {chado_entity}
  395. WHERE data_table = 'feature' AND record_id = :feature_id";
  396. $relationships = array();
  397. while ($rel = $results->fetchObject()) {
  398. $entity = db_query($esql, array(':feature_id' => $rel->subject_id))->fetchObject();
  399. if ($entity) {
  400. $rel->subject_entity_id = $entity->entity_id;
  401. }
  402. $entity = db_query($esql, array(':feature_id' => $rel->object_id))->fetchObject();
  403. if ($entity) {
  404. $rel->object_entity_id = $entity->entity_id;
  405. }
  406. $relationships[$i++] = $rel;
  407. }
  408. return $relationships;
  409. }
  410. /**
  411. * Load the locations for a given feature
  412. *
  413. * @param $feature_id
  414. * The feature to look up locations for
  415. * @param $side
  416. * Whether the feature is the scrfeature, 'as_parent', or feature, 'as_child'
  417. * @param $aggregate
  418. * Whether or not to get the locations for related features
  419. *
  420. * @ingroup tripal_feature
  421. */
  422. private function get_featurelocs($feature_id, $side = 'as_parent', $aggregate = 1) {
  423. $sql = "
  424. SELECT
  425. F.name, F.feature_id, F.uniquename,
  426. FS.name as src_name, FS.feature_id as src_feature_id, FS.uniquename as src_uniquename,
  427. CVT.name as cvname, CVT.cvterm_id,
  428. CVTS.name as src_cvname, CVTS.cvterm_id as src_cvterm_id,
  429. FL.fmin, FL.fmax, FL.is_fmin_partial, FL.is_fmax_partial,FL.strand, FL.phase
  430. FROM {featureloc} FL
  431. INNER JOIN {feature} F ON FL.feature_id = F.feature_id
  432. INNER JOIN {feature} FS ON FS.feature_id = FL.srcfeature_id
  433. INNER JOIN {cvterm} CVT ON F.type_id = CVT.cvterm_id
  434. INNER JOIN {cvterm} CVTS ON FS.type_id = CVTS.cvterm_id
  435. ";
  436. if (strcmp($side, 'as_parent')==0) {
  437. $sql .= "WHERE FL.srcfeature_id = :feature_id ";
  438. }
  439. if (strcmp($side, 'as_child')==0) {
  440. $sql .= "WHERE FL.feature_id = :feature_id ";
  441. }
  442. $flresults = chado_query($sql, array(':feature_id' => $feature_id));
  443. // copy the results into an array
  444. $i=0;
  445. $featurelocs = array();
  446. while ($loc = $flresults->fetchObject()) {
  447. // if a drupal node exists for this feature then add the nid to the
  448. // results object
  449. $loc->feid = tripal_get_chado_entity_id('feature', $loc->feature_id);
  450. $loc->seid = tripal_get_chado_entity_id('feature', $loc->src_feature_id);
  451. // add the result to the array
  452. $featurelocs[$i++] = $loc;
  453. }
  454. // Add the relationship feature locs if aggregate is turned on
  455. if ($aggregate and strcmp($side, 'as_parent')==0) {
  456. // get the relationships for this feature without substituting any children
  457. // for the parent. We want all relationships
  458. $relationships = tripal_feature_get_aggregate_relationships($feature_id, 0);
  459. foreach ($relationships as $rindex => $rel) {
  460. // get the featurelocs for each of the relationship features
  461. $rel_featurelocs = tripal_feature_load_featurelocs($rel->subject_id, 'as_child', 0);
  462. foreach ($rel_featurelocs as $findex => $rfloc) {
  463. $featurelocs[$i++] = $rfloc;
  464. }
  465. }
  466. }
  467. usort($featurelocs, 'chado_feature__residues_sort_locations');
  468. return $featurelocs;
  469. }
  470. }
  471. /**
  472. * Used to sort the list of relationship parts by start position
  473. *
  474. * @ingroup tripal_feature
  475. */
  476. function chado_feature__residues_sort_rel_parts_by_start($a, $b) {
  477. foreach ($a as $type_name => $details) {
  478. $astart = $a[$type_name]['start'];
  479. break;
  480. }
  481. foreach ($b as $type_name => $details) {
  482. $bstart = $b[$type_name]['start'];
  483. break;
  484. }
  485. return strnatcmp($astart, $bstart);
  486. }
  487. /**
  488. * Used to sort the feature locs by start position
  489. *
  490. * @param $a
  491. * One featureloc record (as an object)
  492. * @param $b
  493. * The other featureloc record (as an object)
  494. *
  495. * @return
  496. * Which feature location comes first
  497. *
  498. * @ingroup tripal_feature
  499. */
  500. function chado_feature__residues_sort_locations($a, $b) {
  501. return strnatcmp($a->fmin, $b->fmin);
  502. }