Module: TaxonNamesHelper
- Defined in:
- app/helpers/taxon_names_helper.rb
Constant Summary collapse
- VALID_MARK =
checkmark
'✓'.html_safe.freeze
- INVALID_MARK =
'❌'.html_safe.freeze
- COMBINATION_MARK =
'[c]'.freeze
Instance Method Summary collapse
- #ancestor_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) ⇒ Object
-
#author_chart_data(author_data) ⇒ Hash
Format author data for Chartkick column chart.
-
#author_coauthorship_data(taxon_names) ⇒ Hash
Calculate co-authorship relationships for Sankey diagram.
-
#author_individual_chart_data(author_years) ⇒ Array
Format individual author data for Chartkick column chart Scopes to min/max year observed per author.
- #cached_classified_as_tag(taxon_name) ⇒ Object
- #current_author_year(taxon_name) ⇒ String
-
#defined_full_original_taxon_name_tag(taxon_name) ⇒ String?
!! This is used in taxon_name attributes now! TODO: Refactor our logic for display contexts and value contexts to better reflect presence of data vs.
- #descendant_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) ⇒ Object
- #document_names_per_year(names) ⇒ Object
- #edit_original_combination_task_link(taxon_name) ⇒ Object
- #edit_taxon_name_link(taxon_name, target: nil) ⇒ Object
-
#edit_taxon_name_path_string(taxon_name) ⇒ Object
See #edit_object_path_string in navigation_helper.rb.
-
#full_original_taxon_name_label(taxon_name) ⇒ String
The name in original combination, with author year, without HTML.
-
#full_original_taxon_name_tag(taxon_name) ⇒ String
The name in original combination, with author year, with HTML.
-
#full_taxon_name_tag(taxon_name) ⇒ String
The current name/combination with author year, with HTML.
-
#label_for_taxon_name(taxon_name) ⇒ String
!! Unified deprecated taxon_name_name_string() here.
- #next_sibling_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) ⇒ Object
-
#original_author_year(taxon_name) ⇒ String
Removes parens.
- #original_taxon_name_link(taxon_name) ⇒ Object
-
#original_taxon_name_tag(taxon_name) ⇒ String
The taxon name in original combination, without author year, with HTML.
-
#parent_taxon_name_for_select(taxon_name) ⇒ Object
@taxon_name.parent.andand.display_name(:type => :for_select_list).
- #previous_sibling_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) ⇒ Object
- #rank_tag(taxon_name) ⇒ Object
-
#simple_hierarchy_tag(names, selected_names = nil) ⇒ String
!! Does not try to sort names, works best in combination with
ancestrify: truein ::Queries::TaxonNames::Filter TODO: there is some missalignment on the name matching, you'll see some names that likely matched not linked. -
#summarize_authors_by_year(taxon_names) ⇒ Hash
Summarize People instances through taxon_name_author roles per year.
-
#taxon_name_autocomplete_selected_tag(taxon_name) ⇒ String
No HTML inside .
- #taxon_name_autocomplete_tag(taxon_name, term) ⇒ Object
-
#taxon_name_autoselect_info(taxon_name) ⇒ Object
Disambiguation info Array for the autoselect dropdown (right-justified).
-
#taxon_name_autoselect_tag(taxon_name) ⇒ Object
HTML label for the autoselect dropdown (left-justified).
- #taxon_name_browse_link(taxon_name) ⇒ Object
- #taxon_name_decorator_status(taxon_name) ⇒ Object
- #taxon_name_for_select(taxon_name) ⇒ Object
- #taxon_name_gender_sentence_tag(taxon_name) ⇒ Object
- #taxon_name_inferred_combination_tag(taxon_name) ⇒ Object
- #taxon_name_inventory_stats(taxon_name) ⇒ Object
- #taxon_name_latinization_tag(taxon_name) ⇒ Object
- #taxon_name_link(taxon_name) ⇒ Object
- #taxon_name_link_path(taxon_name, path) ⇒ Object protected
-
#taxon_name_now_tag(taxon_name, css_class = [:feedback, 'feedback-warning', 'feedback-thin']) ⇒ Object
Styling indicating the current valid name.
- #taxon_name_original_combination_tag(taxon_name, css_class = [:feedback, 'feedback-notice', 'feedback-thin']) ⇒ Object
- #taxon_name_otus_links(taxon_name) ⇒ Object
- #taxon_name_parent_navigator_item_link(taxon_name, target = :taxon_name_path) ⇒ Object
- #taxon_name_parent_tag(taxon_name, css_class = [:feedback, 'feedback-secondary', 'feedback-thin']) ⇒ Object
-
#taxon_name_rank_select_tag(taxon_name: TaxonName.new, code: nil) ⇒ Object
TODO: Scope to code.
- #taxon_name_rank_tag(taxon_name, css_class = [:feedback, 'feedback-info', 'feedback-thin']) ⇒ Object
- #taxon_name_short_status(taxon_name) ⇒ Object
- #taxon_name_short_status_label(taxon_name) ⇒ Object
- #taxon_name_status_label(taxon_name) ⇒ Object
-
#taxon_name_tag(taxon_name) ⇒ String
The taxon name without author year, with HTML.
- #taxon_name_type_short_tag(taxon_name) ⇒ Object
- #taxon_name_year_data_table(data, *attributes) ⇒ Object
- #taxon_names_by_year_count(names) ⇒ Object
-
#taxon_names_count_by_validity_and_year(scope = nil) ⇒ Object
Perhaps a /lib/catalog method.
-
#taxon_names_cumulative_count_by_validity_and_year(scope = nil) ⇒ Object
Perhaps a /lib/catalog method.
- #taxon_names_per_year(totals) ⇒ Object
- #taxon_names_search_form ⇒ Object
- #taxonomic_tree(taxon_name, include_ancestors = true, include_count = true) ⇒ Object
- #taxonomic_tree_ancestors(taxon_name, include_count) ⇒ Object
- #taxonomic_tree_descendants(taxon_name, include_count) ⇒ Object
- #taxonomic_tree_node(taxon_name, include_count) ⇒ Object
Instance Method Details
#ancestor_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) ⇒ Object
349 350 351 352 353 354 355 356 357 358 |
# File 'app/helpers/taxon_names_helper.rb', line 349 def ancestor_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) text = 'Up' if taxon_name.ancestors.any? a = taxon_name.ancestors.first. text = object_tag(a) link_to(content_tag(:span, text, data: {icon: 'arrow-up'}, class: 'small-icon'), taxon_name_link_path(a, path), class: 'navigation-item', data: {arrow: 'ancestor'}) else content_tag(:div, content_tag(:span, text, class: 'small-icon', data: {icon: 'arrow-up'}), class: 'navigation-item disable') end end |
#author_chart_data(author_data) ⇒ Hash
Format author data for Chartkick column chart
763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 |
# File 'app/helpers/taxon_names_helper.rb', line 763 def () = {} # Count unique authors per year .each do |person_id, | [:years].each_key do |year| [year] = ([year] || 0) + 1 end end # Return single series of unique author counts { data: [ { name: 'Unique Authors', data: } ] } end |
#author_coauthorship_data(taxon_names) ⇒ Hash
Calculate co-authorship relationships for Sankey diagram
827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 |
# File 'app/helpers/taxon_names_helper.rb', line 827 def (taxon_names) = {} = {} # Find all taxon names with multiple authors taxon_names.left_joins(:taxon_name_authors).find_each do |taxon_name| = taxon_name..to_a next if .length < 2 # Store author names .each { || [.id] = .cached } # Count co-authorships (combinations of authors on same taxon name) .combination(2).each do |, | # Create consistent ordering for the pair source_id, target_id = [.id, .id].sort key = "#{source_id}-#{target_id}" [key] ||= { source_id: source_id, target_id: target_id, count: 0 } [key][:count] += 1 end end # Only include authors that have links = .values.flat_map { |link| [link[:source_id], link[:target_id]] }.uniq nodes_data = .select { |id, name| .include?(id) } # Create node array with indices for d3-sankey nodes = nodes_data.map.with_index { |(id, name), index| { id: index, name: name, person_id: id } } # Create id to index mapping id_to_index = {} nodes.each { |node| id_to_index[node[:person_id]] = node[:id] } # Convert links to use node indices links = .values.map do |link| { source: id_to_index[link[:source_id]], target: id_to_index[link[:target_id]], value: link[:count] } end { nodes: nodes, links: links } end |
#author_individual_chart_data(author_years) ⇒ Array
Format individual author data for Chartkick column chart Scopes to min/max year observed per author
785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 |
# File 'app/helpers/taxon_names_helper.rb', line 785 def () return { data: [], width: '0px' } if .empty? valid_data = {} invalid_data = {} years = .keys.reject { |y| y == 'Unknown' }.sort min_year = years.min max_year = years.max if min_year && max_year (min_year..max_year).each do |year| if [year] valid_data[year] = [year][:valid] || 0 invalid_data[year] = [year][:invalid] || 0 else valid_data[year] = 0 invalid_data[year] = 0 end end end if ['Unknown'] valid_data['Unknown'] = ['Unknown'][:valid] || 0 invalid_data['Unknown'] = ['Unknown'][:invalid] || 0 end year_span = (max_year.to_i - min_year.to_i) chart_width = [120 + year_span * 20, 800].min { data: [ { name: 'Valid', data: valid_data }, { name: 'Invalid', data: invalid_data } ], width: "#{chart_width}px" } end |
#cached_classified_as_tag(taxon_name) ⇒ Object
247 248 249 |
# File 'app/helpers/taxon_names_helper.rb', line 247 def cached_classified_as_tag(taxon_name) taxon_name.cached_classified_as ? taxon_name.cached_classified_as.strip.html_safe : '' end |
#current_author_year(taxon_name) ⇒ String
136 137 138 139 |
# File 'app/helpers/taxon_names_helper.rb', line 136 def (taxon_name) return nil if taxon_name.nil? || taxon_name..nil? taxon_name. end |
#defined_full_original_taxon_name_tag(taxon_name) ⇒ String?
!! This is used in taxon_name attributes now! TODO: Refactor our logic for display contexts and value contexts to better reflect presence of data vs. utility of report.
114 115 116 117 |
# File 'app/helpers/taxon_names_helper.rb', line 114 def defined_full_original_taxon_name_tag(taxon_name) return nil if taxon_name.nil? || taxon_name.cached_original_combination_html.blank? full_original_taxon_name_tag(taxon_name) end |
#descendant_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) ⇒ Object
360 361 362 363 364 365 366 367 368 369 |
# File 'app/helpers/taxon_names_helper.rb', line 360 def descendant_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) text = 'Down' if taxon_name.descendants.unscope(:order).any? a = taxon_name.descendants.first. text = taxon_name_tag(a) link_to(content_tag(:span, text, data: {icon: 'arrow-down'}, class: 'small-icon'), taxon_name_link_path(a, path), class: 'navigation-item', data: {arrow: 'descendant'}) else content_tag(:div, content_tag(:span, text, class: 'small-icon', data: {icon: 'arrow-down'}), class: 'navigation-item disable') end end |
#document_names_per_year(names) ⇒ Object
639 640 641 642 643 |
# File 'app/helpers/taxon_names_helper.rb', line 639 def document_names_per_year(names) taxon_names_per_year( taxon_names_by_year_count(names) ) end |
#edit_original_combination_task_link(taxon_name) ⇒ Object
301 302 303 |
# File 'app/helpers/taxon_names_helper.rb', line 301 def edit_original_combination_task_link(taxon_name) link_to('Edit original combination', edit_protonym_original_combination_task_path(taxon_name)) if GENUS_AND_SPECIES_RANK_NAMES.include?(taxon_name.rank_string) end |
#edit_taxon_name_link(taxon_name, target: nil) ⇒ Object
316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 |
# File 'app/helpers/taxon_names_helper.rb', line 316 def edit_taxon_name_link(taxon_name, target: nil) i = {'Combination': :combination, 'Protonym': :taxon_name}[taxon_name.type.to_sym] t = taxon_name. icon = content_tag(:span, '', data: { icon: 'edit' }, class: 'small-icon') case target when :edit_task path = case i when :taxon_name new_taxon_name_task_path(taxon_name_id: t.id) when :combination new_combination_task_path(taxon_name_id: t.id, literal: URI.encode_www_form_component(t.cached)) # only spaces should be an issue end link_to(safe_join([icon, 'Edit (task)'], ''), path, class: 'navigation-item', 'data-task' => 'new_taxon_name') else link_to(safe_join([icon, 'Edit'], ''), send("edit_#{i}_path}", taxon_name.), 'class' => 'navigation-item') end end |
#edit_taxon_name_path_string(taxon_name) ⇒ Object
See #edit_object_path_string in navigation_helper.rb
306 307 308 309 310 311 312 313 314 |
# File 'app/helpers/taxon_names_helper.rb', line 306 def edit_taxon_name_path_string(taxon_name) if taxon_name.type == 'Protonym' 'edit_taxon_name_path' elsif taxon_name.type == 'Combination' 'edit_combination_path' else nil end end |
#full_original_taxon_name_label(taxon_name) ⇒ String
Returns the name in original combination, with author year, without HTML.
121 122 123 124 125 126 |
# File 'app/helpers/taxon_names_helper.rb', line 121 def full_original_taxon_name_label(taxon_name) return nil if taxon_name.nil? || taxon_name.cached_original_combination.nil? [ taxon_name.cached_original_combination, taxon_name. ].compact.join(' ') end |
#full_original_taxon_name_tag(taxon_name) ⇒ String
Returns the name in original combination, with author year, with HTML.
102 103 104 105 106 107 |
# File 'app/helpers/taxon_names_helper.rb', line 102 def full_original_taxon_name_tag(taxon_name) return nil if taxon_name.nil? [ original_taxon_name_tag(taxon_name), (taxon_name) ].compact.join(' ').html_safe end |
#full_taxon_name_tag(taxon_name) ⇒ String
Returns the current name/combination with author year, with HTML.
95 96 97 98 |
# File 'app/helpers/taxon_names_helper.rb', line 95 def full_taxon_name_tag(taxon_name) return nil if taxon_name.nil? [taxon_name_tag(taxon_name), taxon_name.].compact.join(' ').html_safe end |
#label_for_taxon_name(taxon_name) ⇒ String
!! Unified deprecated taxon_name_name_string() here
37 38 39 40 |
# File 'app/helpers/taxon_names_helper.rb', line 37 def label_for_taxon_name(taxon_name) return nil if taxon_name.nil? [taxon_name.cached, taxon_name.].compact.join(' ') end |
#next_sibling_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) ⇒ Object
371 372 373 374 375 376 377 378 379 |
# File 'app/helpers/taxon_names_helper.rb', line 371 def next_sibling_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) text = 'Next' link_object = taxon_name.next_sibling if link_object.nil? content_tag(:div, content_tag(:span, text), class: 'navigation-item disable') else link_to(text, taxon_name_link_path(link_object, path), title: taxon_name_tag(link_object), class: 'navigation-item', data: { button: 'next' }) end end |
#original_author_year(taxon_name) ⇒ String
Returns removes parens.
130 131 132 133 |
# File 'app/helpers/taxon_names_helper.rb', line 130 def (taxon_name) return nil if taxon_name.nil? || taxon_name..nil? taxon_name. || '' end |
#original_taxon_name_link(taxon_name) ⇒ Object
278 279 280 281 |
# File 'app/helpers/taxon_names_helper.rb', line 278 def original_taxon_name_link(taxon_name) return nil if taxon_name.nil? [ link_to(original_taxon_name_tag(taxon_name).html_safe, browse_nomenclature_task_path(taxon_name_id: taxon_name.id)).html_safe, taxon_name.].compact.join(' ').html_safe end |
#original_taxon_name_tag(taxon_name) ⇒ String
Returns the taxon name in original combination, without author year, with HTML.
84 85 86 87 88 89 90 91 |
# File 'app/helpers/taxon_names_helper.rb', line 84 def original_taxon_name_tag(taxon_name) return nil if taxon_name.nil? if taxon_name.cached_original_combination_html.nil? taxon_name_tag(taxon_name) else taxon_name.cached_original_combination_html.html_safe end end |
#parent_taxon_name_for_select(taxon_name) ⇒ Object
@taxon_name.parent.andand.display_name(:type => :for_select_list)
288 289 290 |
# File 'app/helpers/taxon_names_helper.rb', line 288 def parent_taxon_name_for_select(taxon_name) taxon_name.parent ? taxon_name_for_select(taxon_name.parent) : nil end |
#previous_sibling_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) ⇒ Object
381 382 383 384 385 386 387 388 389 390 |
# File 'app/helpers/taxon_names_helper.rb', line 381 def previous_sibling_browse_taxon_name_link(taxon_name, path = :browse_nomenclature_task_path) text = 'Previous' link_object = taxon_name.previous_sibling if link_object.nil? content_tag(:div, content_tag(:span, text), class: 'navigation-item disable') else link_to(text, taxon_name_link_path(link_object, path), class: 'navigation-item', data: { button: 'back' }) end end |
#rank_tag(taxon_name) ⇒ Object
336 337 338 339 340 341 342 343 344 345 346 347 |
# File 'app/helpers/taxon_names_helper.rb', line 336 def rank_tag(taxon_name) case taxon_name.type when 'Protonym' if taxon_name.rank_class taxon_name.rank.downcase else content_tag(:em, 'ERROR') end when 'Combination' content_tag(:em, 'Combination') end end |
#simple_hierarchy_tag(names, selected_names = nil) ⇒ String
!! Does not try to sort names, works best in combination with ancestrify: true in ::Queries::TaxonNames::Filter
TODO: there is some missalignment on the name matching, you'll see some names that likely matched not linked.
650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 |
# File 'app/helpers/taxon_names_helper.rb', line 650 def simple_hierarchy_tag(names, selected_names = nil) match = [] if selected_names match = selected_names.select("CASE WHEN taxon_names.type = 'Protonym' THEN taxon_names.id ELSE taxon_names.cached_valid_taxon_name_id END as id").pluck(:id) end # taxon_names.cached as alias, \ objects = names.left_joins(:valid_taxon_name) .select("CASE WHEN taxon_names.type = 'Protonym' THEN taxon_names.id ELSE taxon_names.cached_valid_taxon_name_id END as id, \ CASE WHEN taxon_names.type = 'Protonym' THEN taxon_names.parent_id ELSE valid_taxon_names_taxon_names.parent_id END as parent_id, \ COALESCE(taxon_names.name, valid_taxon_names_taxon_names.name, valid_taxon_names_taxon_names.name, valid_taxon_Names_taxon_names.cached) as label") .order('parent_id, label') .distinct d = Utilities::Hierarchy.new(objects:, match:).to_a rows = [] d.each do |r| s = ' ' * r[3] * 10 # space a = (r[2] ? " [#{r[2]}]" : '') # alias if r[4] # matched rows.push s + link_to( tag.b(r[1] + a), browse_nomenclature_task_path(taxon_name_id: r[0])) else # unmatched rows.push s + r[1] + a end end rows.join('<br>').html_safe end |
#summarize_authors_by_year(taxon_names) ⇒ Hash
Summarize People instances through taxon_name_author roles per year
734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 |
# File 'app/helpers/taxon_names_helper.rb', line 734 def (taxon_names) = {} taxon_names.left_joins(:taxon_name_authors).find_each do |taxon_name| year = taxon_name.cached_nomenclature_date&.year || 'Unknown' is_valid = taxon_name.cached_is_valid taxon_name..each do |person| next unless person = person.id [] ||= { name: person.cached, years: {} } [][:years][year] ||= { valid: 0, invalid: 0, total: 0 } if is_valid [][:years][year][:valid] += 1 else [][:years][year][:invalid] += 1 end [][:years][year][:total] += 1 end end end |
#taxon_name_autocomplete_selected_tag(taxon_name) ⇒ String
Returns no HTML inside .
57 58 59 |
# File 'app/helpers/taxon_names_helper.rb', line 57 def taxon_name_autocomplete_selected_tag(taxon_name) label_for_taxon_name(taxon_name) end |
#taxon_name_autocomplete_tag(taxon_name, term) ⇒ Object
42 43 44 45 46 47 48 49 50 51 52 53 |
# File 'app/helpers/taxon_names_helper.rb', line 42 def taxon_name_autocomplete_tag(taxon_name, term) return nil if taxon_name.nil? klass = taxon_name.rank_class ? taxon_name.rank_class.nomenclatural_code : nil a = [ content_tag(:span, mark_tag(taxon_name., term), class: :klass), taxon_name_rank_tag(taxon_name), taxon_name_parent_tag(taxon_name), taxon_name_original_combination_tag(taxon_name), taxon_name_type_short_tag(taxon_name) # " [#{taxon_name.sml_t}]" ].compact.join(' ').html_safe end |
#taxon_name_autoselect_info(taxon_name) ⇒ Object
Disambiguation info Array for the autoselect dropdown (right-justified). Only called for real TaxonName AR records (external levels render their own info).
16 17 18 19 20 21 22 23 24 |
# File 'app/helpers/taxon_names_helper.rb', line 16 def taxon_name_autoselect_info(taxon_name) return [] if taxon_name.nil? [ taxon_name_rank_tag(taxon_name), taxon_name_parent_tag(taxon_name), taxon_name_original_combination_tag(taxon_name), taxon_name_type_short_tag(taxon_name) ] end |
#taxon_name_autoselect_tag(taxon_name) ⇒ Object
HTML label for the autoselect dropdown (left-justified). Uses cached_html for real records; falls back to cached for CoL pseudo-records.
9 10 11 12 |
# File 'app/helpers/taxon_names_helper.rb', line 9 def taxon_name_autoselect_tag(taxon_name) return nil if taxon_name.nil? tag.span( taxon_name..html_safe, class: :klass) end |
#taxon_name_browse_link(taxon_name) ⇒ Object
261 262 263 264 |
# File 'app/helpers/taxon_names_helper.rb', line 261 def taxon_name_browse_link(taxon_name) return nil if taxon_name.nil? [ link_to(taxon_name_tag(taxon_name), browse_nomenclature_task_path(taxon_name_id: taxon_name.id)).html_safe, taxon_name.].compact.join(' ').html_safe end |
#taxon_name_decorator_status(taxon_name) ⇒ Object
225 226 227 228 229 230 231 |
# File 'app/helpers/taxon_names_helper.rb', line 225 def taxon_name_decorator_status(taxon_name) return nil if taxon_name.nil? taxon_name.taxon_name_classifications .where(taxon_name_classifications: {type: TAXON_NAME_CLASSIFICATIONS_FOR_DECORATION}) .select('taxon_name_classifications.type') .map{|a| a.type.demodulize.underscore.gsub(/(\d+)/, ' \1').gsub('_', ' ').capitalize} end |
#taxon_name_for_select(taxon_name) ⇒ Object
283 284 285 |
# File 'app/helpers/taxon_names_helper.rb', line 283 def taxon_name_for_select(taxon_name) taxon_name.name if taxon_name end |
#taxon_name_gender_sentence_tag(taxon_name) ⇒ Object
242 243 244 245 |
# File 'app/helpers/taxon_names_helper.rb', line 242 def taxon_name_gender_sentence_tag(taxon_name) return nil if taxon_name.nil? "The name is #{taxon_name.cached_gender}." if taxon_name.cached_gender end |
#taxon_name_inferred_combination_tag(taxon_name) ⇒ Object
233 234 235 236 237 238 239 240 |
# File 'app/helpers/taxon_names_helper.rb', line 233 def taxon_name_inferred_combination_tag(taxon_name) return nil if taxon_name.nil? || taxon_name.is_combination? || taxon_name.is_valid? if taxon_name.is_protonym? return nil if taxon_name.cached_primary_homonym == taxon_name.cached_secondary_homonym end tag.span(tag.em('inferred combination'), class: :subtle) end |
#taxon_name_inventory_stats(taxon_name) ⇒ Object
405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 |
# File 'app/helpers/taxon_names_helper.rb', line 405 def taxon_name_inventory_stats(taxon_name) # Code mostly by chatgpt 5 (with comment/naming revsisions) d = [] # Query 1. Get all ranks for ordering ranks = ::Queries::TaxonName::Filter .new(synonymify: true, descendants: false, taxon_name_id: taxon_name.id) .all .where(type: 'Protonym') .distinct .pluck(:rank_class) .compact .sort_by { |r| RANKS.index(r) || RANKS.length } return [] if ranks.empty? # Query 2. VALID valid = ::Queries::TaxonName::Filter.new( validity: true, descendants: false, taxon_name_id: taxon_name.id, taxon_name_type: 'Protonym' ).all valid_by_rank = valid .group('rank_class') .count # Query 3. VALID FOSSILS valid_fossil_names_by_rank = valid .joins(:taxon_name_classifications) .where(taxon_name_classifications: { type: TAXON_NAME_CLASSIFICATIONS_FOR_FOSSILS }) .group('rank_class') .count # Query 4. INVALID invalid_scope = ::Queries::TaxonName::Filter.new( descendants: false, synonymify: true, taxon_name_id: taxon_name.id, taxon_name_type: 'Protonym' ).all.that_is_invalid # Count invalid names at the rank of their valid name. invalid_by_rank = TaxonName .from("(#{invalid_scope.to_sql}) invalid") .joins('JOIN taxon_names valid ON valid.id = invalid.cached_valid_taxon_name_id') .group('valid.rank_class') .count # Query 5: Coordinatified OTU counts # In brief: Let S be the subtree of taxon_name. # 1) To each name in S, assign the rank of its valid name (may be itself). # 2) Coordinatify all otus corresponding to S (expand in both directions # from S via valid-name-of/invalid-name-of name in S). # 3) Join 2) to 1) via the expansion described in 2). # 4) Group that join by the valid rank assigned to elements of S in 1). # 5) Count by that valid rank. base_scope = ::Queries::TaxonName::Filter.new( descendants: false, taxon_name_id: taxon_name.id, taxon_name_type: 'Protonym' ).all # Count invalid names with their valid name's rank. valid_rank = TaxonName .from(base_scope, :tn) .joins('LEFT JOIN taxon_names valid ON valid.id = tn.cached_valid_taxon_name_id') .select( 'tn.id, COALESCE(valid.rank_class, tn.rank_class) AS valid_rank, tn.cached_is_valid'.squish ) otus_scope = ::Otu.where(taxon_name_id: valid_rank.except(:select).select('tn.id')) # This is a little janky, but it's what allows us to avoid an extra query # (and it gives project_id context). otus_coordinatified = ::Queries::Otu::Filter.new({}).coordinatify_result(otus_scope) rows = TaxonName .with(valid_rank:) .from(otus_coordinatified, :o) .joins('JOIN taxon_names tn ON tn.id = o.taxon_name_id') .joins('JOIN valid_rank vr ON tn.id = vr.id OR tn.cached_valid_taxon_name_id = vr.id') .group('vr.valid_rank') .pluck('vr.valid_rank', Arel.sql('COUNT(DISTINCT o.id)')) otu_by_rank = rows.each_with_object({}) { |(rank, cnt), h| h[rank] = cnt.to_i } # Stitch results per rank ranks.each do |rank_class| n = rank_class.safe_constantize.rank_name.to_sym valid = valid_by_rank[rank_class] || 0 valid_fossil = valid_fossil_names_by_rank[rank_class] || 0 invalid = invalid_by_rank[rank_class] || 0 taxa = otu_by_rank[rank_class] || 0 d << { rank: n, taxa: taxa, names: { valid: valid, valid_fossil: valid_fossil, valid_extant: valid - valid_fossil, invalid: invalid } } end d end |
#taxon_name_latinization_tag(taxon_name) ⇒ Object
251 252 253 254 |
# File 'app/helpers/taxon_names_helper.rb', line 251 def taxon_name_latinization_tag(taxon_name) list = taxon_name.taxon_name_classifications.with_type_array(LATINIZED_TAXON_NAME_CLASSIFICATION_NAMES).map(&:classification_label) content_tag(:span, "The word \"#{taxon_name.name}\" has the following Latin-based classifications: #{list.to_sentence}.", class: 'history__latinized_classifications') if list.any? end |
#taxon_name_link(taxon_name) ⇒ Object
256 257 258 259 |
# File 'app/helpers/taxon_names_helper.rb', line 256 def taxon_name_link(taxon_name) return nil if taxon_name.nil? link_to(taxon_name_tag(taxon_name), taxon_name.).html_safe end |
#taxon_name_link_path(taxon_name, path) ⇒ Object (protected)
877 878 879 880 881 882 883 |
# File 'app/helpers/taxon_names_helper.rb', line 877 def taxon_name_link_path(taxon_name, path) if path == :taxon_name_path send(path, taxon_name) else send(path, taxon_name_id: taxon_name.id) end end |
#taxon_name_now_tag(taxon_name, css_class = [:feedback, 'feedback-warning', 'feedback-thin']) ⇒ Object
Styling indicating the current valid name
77 78 79 80 |
# File 'app/helpers/taxon_names_helper.rb', line 77 def taxon_name_now_tag(taxon_name, css_class = [:feedback, 'feedback-warning', 'feedback-thin'] ) return nil if taxon_name.nil? || !taxon_name.is_valid? content_tag(:span, ('now ' + taxon_name.cached_html).html_safe, class: css_class) end |
#taxon_name_original_combination_tag(taxon_name, css_class = [:feedback, 'feedback-notice', 'feedback-thin']) ⇒ Object
71 72 73 74 |
# File 'app/helpers/taxon_names_helper.rb', line 71 def taxon_name_original_combination_tag(taxon_name, css_class = [:feedback, 'feedback-notice', 'feedback-thin'] ) return nil if taxon_name.nil? || taxon_name.cached_original_combination.blank? content_tag(:span, taxon_name.cached_original_combination, class: css_class) end |
#taxon_name_otus_links(taxon_name) ⇒ Object
392 393 394 395 396 397 398 399 400 401 402 403 |
# File 'app/helpers/taxon_names_helper.rb', line 392 def taxon_name_otus_links(taxon_name) if taxon_name.otus.load.any? ('The following Otus are linked to this name: ' + content_tag(:ul, class: 'no_bullets') do taxon_name.otus.each do |o| concat(content_tag(:li, otu_link(o) )) end end.html_safe).html_safe else content_tag(:em, 'There are no Otus linked to this name.') end end |
#taxon_name_parent_navigator_item_link(taxon_name, target = :taxon_name_path) ⇒ Object
266 267 268 269 270 271 272 273 274 275 276 |
# File 'app/helpers/taxon_names_helper.rb', line 266 def taxon_name_parent_navigator_item_link(taxon_name, target = :taxon_name_path) return nil if taxon_name.nil? || target.nil? if target case target.to_sym when :taxon_name_path link_to(taxon_name_tag(taxon_name), taxon_name.) else link_to(taxon_name_tag(taxon_name), send(target, {taxon_name_id: taxon_name.id})) end end end |
#taxon_name_parent_tag(taxon_name, css_class = [:feedback, 'feedback-secondary', 'feedback-thin']) ⇒ Object
66 67 68 69 |
# File 'app/helpers/taxon_names_helper.rb', line 66 def taxon_name_parent_tag(taxon_name, css_class = [:feedback, 'feedback-secondary', 'feedback-thin'] ) return nil if taxon_name.nil? || taxon_name.parent_id.nil? content_tag(:span, taxon_name_tag(taxon_name.parent).html_safe, class: css_class) end |
#taxon_name_rank_select_tag(taxon_name: TaxonName.new, code: nil) ⇒ Object
TODO: Scope to code
293 294 295 |
# File 'app/helpers/taxon_names_helper.rb', line 293 def taxon_name_rank_select_tag(taxon_name: TaxonName.new, code: nil) select(:taxon_name, :rank_class, (RANKS_SELECT_OPTIONS, selected: taxon_name.rank_string) ) end |
#taxon_name_rank_tag(taxon_name, css_class = [:feedback, 'feedback-info', 'feedback-thin']) ⇒ Object
61 62 63 64 |
# File 'app/helpers/taxon_names_helper.rb', line 61 def taxon_name_rank_tag(taxon_name, css_class = [:feedback, 'feedback-info', 'feedback-thin'] ) return nil if taxon_name.nil? content_tag(:span, taxon_name.rank || 'Combination', class: css_class) end |
#taxon_name_short_status(taxon_name) ⇒ Object
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 |
# File 'app/helpers/taxon_names_helper.rb', line 150 def taxon_name_short_status(taxon_name) if taxon_name.is_combination? n = taxon_name.finest_protonym s = ['This name is a subsequent combination of'] if n.is_valid? s += [ link_to(original_taxon_name_tag(n), browse_nomenclature_task_path(taxon_name_id: n.id)), (n), ] else v = n.valid_taxon_name s += [ original_taxon_name_tag(n), (n), 'whose valid/accepted name is', link_to(taxon_name_tag(v), browse_nomenclature_task_path(taxon_name_id: v.id) ), v. ] end (s.join(' ') + '.').html_safe else if taxon_name.is_valid? # taxon_name.unavailable_or_invalid? content_tag(:span, safe_join([ content_tag(:span, '',data: {icon: :ok, status: :valid }), content_tag(:span, 'This name is valid/accepted.', data: { status: :valid }) ], ''), class: :brief_status, data: { status: :valid }) else if taxon_name.is_ambiguously_invalid? tag.span('This name is not valid/accepted.'.html_safe, class: :brief_status, data: {icon: :attention, status: :invalid}) else tag.span("This name is not valid/accepted.<br>The valid name is #{taxon_name_browse_link(taxon_name.valid_taxon_name)}.".html_safe, class: :brief_status, data: {icon: :attention, status: :invalid}) end end end end |
#taxon_name_short_status_label(taxon_name) ⇒ Object
191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 |
# File 'app/helpers/taxon_names_helper.rb', line 191 def taxon_name_short_status_label(taxon_name) if taxon_name.is_combination? n = taxon_name.finest_protonym s = ['This name is subsequent combination of'] if n.is_valid? s += [ original_taxon_name_tag(n), (n), ] else v = n.valid_taxon_name s += [ original_taxon_name_tag(n), (n), 'whose valid/accepted name is', taxon_name_tag(v), v. ] end (s.join(' ') + '.') else if taxon_name.is_valid? # taxon_name.unavailable_or_invalid? 'This name is valid/accepted.' else if taxon_name.is_ambiguously_invalid? 'This name is not valid/accepted.' else "This name is not valid/accepted. The valid name is #{taxon_name.valid_taxon_name.cached}." end end end end |
#taxon_name_status_label(taxon_name) ⇒ Object
187 188 189 |
# File 'app/helpers/taxon_names_helper.rb', line 187 def taxon_name_status_label(taxon_name) taxon_name.combined_statuses.collect{|s| s}.join('; ') end |
#taxon_name_tag(taxon_name) ⇒ String
Returns the taxon name without author year, with HTML.
28 29 30 31 32 |
# File 'app/helpers/taxon_names_helper.rb', line 28 def taxon_name_tag(taxon_name) return nil if taxon_name.nil? return taxon_name.name if taxon_name.new_record? taxon_name.cached_html.try(:html_safe) || taxon_name.name end |
#taxon_name_type_short_tag(taxon_name) ⇒ Object
141 142 143 144 145 146 147 148 |
# File 'app/helpers/taxon_names_helper.rb', line 141 def taxon_name_type_short_tag(taxon_name) return nil if taxon_name.nil? if taxon_name.is_valid? VALID_MARK else taxon_name.is_combination? ? COMBINATION_MARK : INVALID_MARK # c or X end end |
#taxon_name_year_data_table(data, *attributes) ⇒ Object
551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 |
# File 'app/helpers/taxon_names_helper.rb', line 551 def taxon_name_year_data_table(data, *attributes) a = data[:data].first b = data[:data].second content_tag(:table, safe_join([ tag.thead( tag.tr( safe_join [tag.th('Year'), tag.th(a[:name]), tag.th(b[:name])] ) ), safe_join((data[:metadata][:min_year]..data[:metadata][:max_year]).collect{|y| tag.tr( safe_join([ tag.td(y), tag.td(a[:data][y]), tag.td(b[:data][y]) ]) ) }) ]), *attributes ) end |
#taxon_names_by_year_count(names) ⇒ Object
634 635 636 637 |
# File 'app/helpers/taxon_names_helper.rb', line 634 def taxon_names_by_year_count(names) t = names.select('EXTRACT(YEAR FROM taxon_names.cached_nomenclature_date) AS year, COUNT(*) AS count').group('year').inject({}){|hsh, r| hsh[r.year.to_i] = r.count; hsh} t end |
#taxon_names_count_by_validity_and_year(scope = nil) ⇒ Object
Perhaps a /lib/catalog method
520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 |
# File 'app/helpers/taxon_names_helper.rb', line 520 def taxon_names_count_by_validity_and_year(scope = nil) return {} if scope.nil? invalid = taxon_names_by_year_count(scope.that_is_invalid) valid = taxon_names_by_year_count(scope.that_is_valid) min = [invalid.keys.sort.first, invalid.keys.sort.first].compact.sort.first || 0 max = [valid.keys.sort.last, valid.keys.sort.last].compact.sort.first || 0 min = 1759 if min < 1759 max = Time.current.year if max > Time.current.year invalid_data = {} valid_data = {} (min..max).each do |y| invalid_data[y] = invalid[y].present? ? invalid[y].to_i : 0 valid_data[y] = valid[y].present? ? valid[y].to_i : 0 end return { metadata: { max_year: max, min_year: min, }, data: [ { name: 'Valid', data: valid_data}, { name: 'Invalid', data: invalid_data} ] } end |
#taxon_names_cumulative_count_by_validity_and_year(scope = nil) ⇒ Object
Perhaps a /lib/catalog method
577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 |
# File 'app/helpers/taxon_names_helper.rb', line 577 def taxon_names_cumulative_count_by_validity_and_year(scope = nil) return {} if scope.nil? invalid = taxon_names_by_year_count(scope.that_is_invalid) valid = taxon_names_by_year_count(scope.that_is_valid) min = [invalid.keys.sort.first, invalid.keys.sort.first].compact.sort.first || 0 max = [valid.keys.sort.last, valid.keys.sort.last].compact.sort.first || 0 min = 1759 if min < 1759 max = Time.current.year if max > Time.current.year invalid_data = {} valid_data = {} invalid_total = 0 valid_total = 0 (min..max).each do |y| i = ( invalid[y].present? ? invalid[y].to_i : 0 ) v = ( valid[y].present? ? valid[y].to_i : 0 ) invalid_total += i valid_total += v invalid_data[y] = invalid_total valid_data[y] = valid_total end return { metadata: { max_year: max, min_year: min, }, data: [ { name: 'Valid', data: valid_data}, { name: 'Invalid', data: invalid_data} ] } end |
#taxon_names_per_year(totals) ⇒ Object
618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 |
# File 'app/helpers/taxon_names_helper.rb', line 618 def taxon_names_per_year(totals) min = totals.keys.sort.first || 0 max = totals.keys.sort.last || 0 min = 1759 if min < 1759 max = Time.current.year if max > Time.current.year data = {} (min..max).each do |y| data[y] = totals[y].present? ? totals[y].to_i : 0 end data end |
#taxon_names_search_form ⇒ Object
297 298 299 |
# File 'app/helpers/taxon_names_helper.rb', line 297 def taxon_names_search_form render '/taxon_names/quick_search_form' end |
#taxonomic_tree(taxon_name, include_ancestors = true, include_count = true) ⇒ Object
717 718 719 720 721 722 723 724 725 726 727 728 |
# File 'app/helpers/taxon_names_helper.rb', line 717 def taxonomic_tree(taxon_name, include_ancestors = true, include_count = true) node = { taxon_name: taxonomic_tree_node(taxon_name, include_count), descendants: taxonomic_tree_descendants(taxon_name, include_count) } if (include_ancestors) node[:ancestors] = taxonomic_tree_ancestors(taxon_name, include_count) end node end |
#taxonomic_tree_ancestors(taxon_name, include_count) ⇒ Object
703 704 705 |
# File 'app/helpers/taxon_names_helper.rb', line 703 def taxonomic_tree_ancestors(taxon_name, include_count) taxon_name.ancestor_protonyms.map { |ancestor| taxonomic_tree_node(ancestor, include_count) } end |
#taxonomic_tree_descendants(taxon_name, include_count) ⇒ Object
707 708 709 710 711 712 713 714 715 |
# File 'app/helpers/taxon_names_helper.rb', line 707 def taxonomic_tree_descendants(taxon_name, include_count) taxon_name .children .where(type: 'Protonym') .sort_by { |a| [RANKS.index(a.rank_string), a.cached, a. || ''] } .map { |child| taxonomic_tree_node(child, include_count) } end |
#taxonomic_tree_node(taxon_name, include_count) ⇒ Object
683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 |
# File 'app/helpers/taxon_names_helper.rb', line 683 def taxonomic_tree_node(taxon_name, include_count) node = { id: taxon_name.id, parent_id: taxon_name.parent_id, label: taxon_name., is_valid: taxon_name.cached_is_valid, cached_valid_taxon_name_id: taxon_name.cached_valid_taxon_name_id, rank_string: taxon_name.rank_string, synonyms: taxon_name_synonyms_list(taxon_name).map { |syn| taxon_name_synonym_li(syn) }, leaf_node: taxon_name.descendants.unscope(:order).empty? } if include_count node[:valid_descendants] = taxon_name.descendants.unscope(:order).that_is_valid.count node[:invalid_descendants] = taxon_name.descendants.unscope(:order).that_is_invalid.count end node end |