diff --git a/.gitignore b/.gitignore index 2359847d9..f2cbbf418 100644 --- a/.gitignore +++ b/.gitignore @@ -21,3 +21,4 @@ biodiverse.iml /biodiverseb.iml /.idea /AGS_bin +/etc/benchmark_scripts/nytprof diff --git a/lib/Biodiverse/GUI/Dendrogram.pm b/lib/Biodiverse/GUI/Dendrogram.pm index 094f7961e..2466ab830 100644 --- a/lib/Biodiverse/GUI/Dendrogram.pm +++ b/lib/Biodiverse/GUI/Dendrogram.pm @@ -872,19 +872,6 @@ sub recolour_cluster_elements { my $cluster_colour_mode = $self->get_cluster_colour_mode(); my $colour_callback; - my %list_and_index = (list => $list_name, index => $list_index); - my $is_canape = $list_name =~ />>CANAPE>>/ && $list_index =~ /CANAPE/; - my $is_zscore = eval { - $parent_tab->index_is_zscore(%list_and_index); - }; - my $is_prank = $list_name =~ />>p_rank>>/; - my $is_ratio - = !$is_prank && !$is_zscore - && eval {$parent_tab->index_is_ratio(%list_and_index)}; - my $is_divergent - = !$is_prank && !$is_zscore && !$is_ratio - && eval {$parent_tab->index_is_divergent(%list_and_index)}; - if ($cluster_colour_mode eq 'palette') { # sets colours according to palette $colour_callback = sub { @@ -924,17 +911,13 @@ sub recolour_cluster_elements { }; } elsif ($cluster_colour_mode eq 'list-values') { - my $abs_extreme; - if ($is_ratio) { - $abs_extreme = exp (max (abs log $analysis_min, log $analysis_max)); - $analysis_min = 1 / $abs_extreme; - $analysis_max = $abs_extreme; - } - elsif ($is_divergent) { # assumes zero - needs work - $abs_extreme = max(abs $analysis_min, abs $analysis_max); - $analysis_min = 0; - $analysis_max = $abs_extreme; - } + my $legend = $map->get_legend; + $legend->set_colour_mode_from_list_and_index ( + list => $list_name, + index => $list_index, + ); + my @minmax_args = ($analysis_min, $analysis_max); + my $colour_method = $legend->get_colour_method; # sets colours according to (usually spatial) # list value for the element's cluster @@ -952,12 +935,7 @@ sub recolour_cluster_elements { my $val = $list_ref->{$list_index} // return $colour_for_undef; - return $is_canape ? $map->get_colour_canape ($val) - : $is_zscore ? $map->get_colour_zscore ($val) - : $is_prank ? $map->get_colour_prank ($val) - : $is_ratio ? $map->get_colour_ratio ($val, $abs_extreme) - : $is_divergent ? $map->get_colour_divergent ($val, 0, $abs_extreme) - : $map->get_colour($val, $analysis_min, $analysis_max); + return $legend->$colour_method ($val, @minmax_args); } else { return exists $terminal_elements->{$elt} @@ -974,12 +952,6 @@ sub recolour_cluster_elements { $map->colour ($colour_callback); - $map->get_legend->set_canape_mode($is_canape); - $map->get_legend->set_zscore_mode($is_zscore); - $map->get_legend->set_prank_mode($is_prank); - $map->get_legend->set_ratio_mode($is_ratio); - $map->get_legend->set_divergent_mode($is_divergent); - if ($cluster_colour_mode eq 'list-values') { $map->set_legend_min_max($analysis_min, $analysis_max); } @@ -1259,31 +1231,13 @@ sub recolour_cluster_lines { my $analysis_max = $self->{analysis_max}; my $colour_mode = $self->get_cluster_colour_mode(); - my %list_and_index = (list => $list_name, index => $list_index); - my $is_canape = $list_name =~ />>CANAPE>>/ && $list_index =~ /^CANAPE/; - my $is_zscore = eval { - $self->{parent_tab}->index_is_zscore (%list_and_index); - }; - my $is_prank = $list_name =~ />>p_rank>>/; - my $is_ratio - = !$is_prank && !$is_zscore - && eval {$self->{parent_tab}->index_is_ratio(%list_and_index)}; - my $is_divergent - = !$is_prank && !$is_zscore && !$is_ratio - && eval {$self->{parent_tab}->index_is_divergent(%list_and_index)}; - - my $abs_extreme; - if ($is_ratio) { - $abs_extreme = exp (max (abs log $analysis_min, log $analysis_max)); - $analysis_min = 1 / $abs_extreme; - $analysis_max = $abs_extreme; - } - elsif ($is_divergent) { # assumes zero - needs work - $abs_extreme = max(abs $analysis_min, abs $analysis_max); - $analysis_min = 0; - $analysis_max = $abs_extreme; - } - + my $legend = $map->get_legend; + $legend->set_colour_mode_from_list_and_index ( + list => $list_name, + index => $list_index, + ); + my @minmax_args = ($analysis_min, $analysis_max); + my $colour_method = $legend->get_colour_method; foreach my $node_ref (@$cluster_nodes) { @@ -1307,13 +1261,7 @@ sub recolour_cluster_lines { : undef; # allows for missing lists $colour_ref = defined $val - ? ( $is_canape ? $map->get_colour_canape($val) : - $is_zscore ? $map->get_colour_zscore($val) : - $is_prank ? $map->get_colour_prank($val) : - $is_ratio ? $map->get_colour_ratio ($val, $abs_extreme) : - $is_divergent ? $map->get_colour_divergent ($val, 0, $abs_extreme) : - $map->get_colour ($val, $analysis_min, $analysis_max) - ) + ? $legend->$colour_method ($val, @minmax_args) : undef; } else { diff --git a/lib/Biodiverse/GUI/Grid.pm b/lib/Biodiverse/GUI/Grid.pm index e47d83d6d..7e30e81b6 100644 --- a/lib/Biodiverse/GUI/Grid.pm +++ b/lib/Biodiverse/GUI/Grid.pm @@ -1135,37 +1135,6 @@ sub set_colour_for_undef { $self->{colour_none} = $colour; } -# need factory generation for the next few -sub get_colour { - my $self = shift; - return $self->get_legend->get_colour (@_); -} - -sub get_colour_canape { - my $self = shift; - return $self->get_legend->get_colour_canape (@_); -} - -sub get_colour_zscore { - my $self = shift; - return $self->get_legend->get_colour_zscore (@_); -} - -sub get_colour_prank { - my $self = shift; - return $self->get_legend->get_colour_prank (@_); -} - -sub get_colour_ratio { - my $self = shift; - return $self->get_legend->get_colour_ratio (@_); -} - -sub get_colour_divergent { - my $self = shift; - return $self->get_legend->get_colour_divergent (@_); -} - ########################################################## # Data extraction utilities diff --git a/lib/Biodiverse/GUI/Legend.pm b/lib/Biodiverse/GUI/Legend.pm index 2e4f7181c..09c595c0d 100644 --- a/lib/Biodiverse/GUI/Legend.pm +++ b/lib/Biodiverse/GUI/Legend.pm @@ -109,20 +109,9 @@ sub new { # reverse might not be needed but ensures the array is the correct size from the start foreach my $i (reverse 0..3) { - $self->{marks}[$i] = $self->make_mark($self->{legend_marks}[$i]); - } - # clunky that we need to do it here - my @anchors = ('nw', ('w') x 3, 'sw'); - foreach my $i (reverse 0..4) { - $self->{canape_marks}[$i] = $self->make_mark($anchors[$i]); - $self->{divergent_marks}[$i] = $self->make_mark($anchors[$i]); - $self->{ratio_marks}[$i] = $self->make_mark($anchors[$i]); - } - @anchors = ('nw', ('w') x 5, 'sw'); - foreach my $i (reverse 0..6) { - $self->{zscore_marks}[$i] = $self->make_mark($anchors[$i]); - $self->{prank_marks}[$i] = $self->make_mark($anchors[$i]); + $self->{marks}{default}[$i] = $self->make_mark($self->{legend_marks}[$i]); } + $self->{marks}{current} = $self->{marks}{default}; # debug stuff #my $sub = sub { @@ -198,11 +187,24 @@ sub make_rect { my $n = (scalar keys %canape_colour_hash) - 1; foreach my $row (0..($height - 1)) { - my $class = int (0.5 + $n * $row / ($height - 1)); + my $class = $n - int (0.5 + $n * $row / ($height - 1)); my $colour = $self->get_colour_canape ($class); $self->add_row($self->{legend_colours_group}, $row, $colour); } } + elsif ($self->get_categorical_mode) { + ($width, $height) = ($self->get_width, 255); + $self->{legend_height} = $height; + my $label_hash = $self->{categorical}{labels}; + + my $n = (scalar keys %$label_hash) - 1; + foreach my $row (0..($height - 1)) { + # cat 0 at the top + my $class = $n - int (0.5 + $n * $row / ($height - 1)); + my $colour = $self->get_colour_categorical ($class); + $self->add_row($self->{legend_colours_group}, $row, $colour); + } + } elsif ($self->get_zscore_mode) { ($width, $height) = ($self->get_width, 255); @@ -252,10 +254,10 @@ sub make_rect { local $self->{log_mode} = 0; # hacky override my $mid = ($height - 1) / 2; - foreach my $row (0..($height - 1)) { + foreach my $row (reverse 0..($height - 1)) { my $val = $row < $mid ? 1 / ($mid - $row) : $row - $mid; # invert again so colours match legend text - my $colour = $self->get_colour_ratio (1 / $val, $mid); + my $colour = $self->get_colour_ratio (1/$val, 1/$mid, $mid); $self->add_row($self->{legend_colours_group}, $row, $colour); } } @@ -267,9 +269,10 @@ sub make_rect { my $centre = ($height - 1) / 2; my $extreme = $height - $centre; - foreach my $row (0..($height - 1)) { - # ensure colours match plot since 0 is the top - my $colour = $self->get_colour_divergent ($height - $row, $centre, $extreme); + + # ensure colours match plot since 0 is the top + foreach my $row (reverse 0..($height - 1)) { + my $colour = $self->get_colour_divergent ($centre - $row, -$extreme, $extreme); $self->add_row($self->{legend_colours_group}, $row, $colour); } } @@ -358,6 +361,15 @@ sub make_mark { return $mark; } +sub hide_current_marks { + my $self = shift; + my $marks = $self->{marks}{current} // []; + foreach my $mark (@$marks) { + $mark->hide; + } + return; +} + sub set_gt_flag { my $self = shift; my $flag = shift; @@ -387,6 +399,14 @@ sub get_height { return $self->{back_rect_height} // LEGEND_HEIGHT; } +sub refresh_legend { + my $self = shift; + $self->make_rect; + # trigger a redisplay of the legend + $self->reposition($self->{width_px}, $self->{height_px}); + 1; +} + # Updates position of legend and value box # when canvas is resized or scrolled sub reposition { @@ -427,13 +447,7 @@ sub reposition { $self->{legend_colours_group}->affine_absolute($matrix); # Reposition the "mark" textboxes - my @mark_arr - = $self->get_zscore_mode ? @{$self->{zscore_marks}} - : $self->get_prank_mode ? @{$self->{prank_marks}} - : $self->get_canape_mode ? @{$self->{canape_marks}} - : $self->get_divergent_mode ? @{$self->{divergent_marks}} - : $self->get_ratio_mode ? @{$self->{ratio_marks}} - : @{$self->{marks}}; + my @mark_arr = @{$self->{marks}{current} // []}; foreach my $i (0..$#mark_arr) { my $mark = $mark_arr[$#mark_arr - $i]; # move the mark to right align with the legend @@ -496,6 +510,8 @@ sub set_mode { #$self->colour_cells(); # Update legend + $self->hide_current_marks; + # could use refresh_legend here? if ($self->{legend}) { # && $self->{width_px} && $self->{height_px}) { $self->{legend} = $self->make_rect(); $self->reposition($self->{width_px}, $self->{height_px}); # trigger a redisplay of the legend @@ -602,6 +618,17 @@ sub get_colour { return $self->$method($val, $min, $max); } +sub get_colour_categorical { + my ($self, $val) = @_; + $val //= -1; # avoid undef key warnings + my $colour_hash = $self->{categorical}{colours} //= {}; + my $colour = $colour_hash->{$val} || COLOUR_WHITE; + # should not need to do this + if (!blessed $colour) { + $colour = $colour_hash->{$val} = Gtk2::Gdk::Color->parse($colour); + } + return $colour; +} sub get_colour_canape { my ($self, $val) = @_; @@ -658,16 +685,17 @@ sub get_colour_prank { } sub get_colour_divergent { - my ($self, $val, $centre, $max_dist) = @_; + my ($self, $val, $min, $max) = @_; state $default_colour = Gtk2::Gdk::Color->new(0, 0, 0); return $default_colour - if ! defined $max_dist; + if ! (defined $max && defined $min); state $centre_colour = Gtk2::Gdk::Color->parse('#ffffbf'); - $centre //= 0; + my $centre = 0; + my $max_dist = max (abs($min), abs($max)); return $centre_colour if $val == $centre || $max_dist == 0; @@ -705,18 +733,23 @@ sub get_colour_divergent { } sub get_colour_ratio { - my ($self, $val, $extreme) = @_; + my ($self, $val, $min, $max) = @_; state $default_colour = Gtk2::Gdk::Color->new(0, 0, 0); return $default_colour - if ! defined $extreme; + if ! (defined $min && defined $max); state $centre_colour = Gtk2::Gdk::Color->parse('#ffffbf'); + my $extreme = exp (max (abs log $min, log $max)); + return $centre_colour if $val == 1 || $extreme == 1; + # $min = 1 / $extreme; + # $max = $extreme; + # simplify logic below if ($extreme < 1) { $extreme = 1 / $extreme; @@ -909,18 +942,28 @@ sub set_min_max { return $self->set_text_marks_prank if $self->get_prank_mode; - # foreach my $mark (@{$self->{marks}}) { + # foreach my $mark (@{$self->{marks}{default}}) { # $mark->show; # } return $self->set_text_marks_canape if $self->get_canape_mode; - return $self->set_text_marks_divergent ($val1, $val2) - if $self->get_divergent_mode; + return $self->set_text_marks_categorical + if $self->get_categorical_mode; - return $self->set_text_marks_ratio ($val2) - if $self->get_ratio_mode; + if ($self->get_divergent_mode) { + my $abs_extreme = max(abs $val1, abs $val2); + my $min = 0; + my $max = $abs_extreme; + return $self->set_text_marks_divergent($min, $max); + } + elsif ($self->get_ratio_mode) { + my $abs_extreme = exp (max (abs log $val1, log $val2)); + my $min = 1 / $abs_extreme; + my $max = $abs_extreme; + return $self->set_text_marks_ratio($min, $max) + } my $min = $val1 //= $self->{last_min}; my $max = $val2 //= $self->{last_max}; @@ -929,13 +972,15 @@ sub set_min_max { $self->{last_max} = $max; - return if ! ($self->{marks} + return if ! ($self->{marks}{default} && defined $min && defined $max ); # Set legend textbox markers - my @mark_arr = @{$self->{marks}}; + $self->hide_current_marks; + my @mark_arr = @{$self->{marks}{default}}; + $self->{marks}{current} = $self->{marks}{default}; my $marker_step = ($max - $min) / $#mark_arr; foreach my $i (0..$#mark_arr) { my $val = $min + $i * $marker_step; @@ -959,7 +1004,7 @@ sub set_min_max { $text = ' ' . $text; } - my $mark = $self->{marks}[$#mark_arr - $i]; + my $mark = $self->{marks}{default}[$#mark_arr - $i]; $mark->set( text => $text ); # move the mark to right align with the legend my @bounds = $mark->get_bounds; @@ -980,66 +1025,32 @@ sub set_min_max { sub set_text_marks_canape { my $self = shift; - return if !$self->{marks}; - - foreach my $mark (@{$self->{marks}}) { - $mark->hide; - } - - my @strings = qw /super mixed palaeo neo non-sig/; + my @strings = reverse (qw /super mixed palaeo neo non-sig/); + return $self->set_text_marks_for_labels (\@strings); +} - my $mark_arr = $self->{canape_marks} //= []; - if (!@$mark_arr) { - foreach my $i (0 .. $#strings) { - my $anchor_loc = $i == 0 ? 'nw' : $i == $#strings ? 'sw' : 'w'; - $mark_arr->[$i] = $self->make_mark($anchor_loc); - } - } +sub set_text_marks_categorical { + my $self = shift; - # Set legend textbox markers - foreach my $i (0..$#strings) { - my $mark = $mark_arr->[$#$mark_arr - $i]; - $mark->set( text => $strings[$i] ); - $mark->raise_to_top; - } + my $label_hash = $self->{categorical}{labels} // {}; + my @strings = @$label_hash{sort {$a <=> $b} keys %$label_hash}; - return; + return $self->set_text_marks_for_labels (\@strings); } sub set_text_marks_zscore { my $self = shift; - # needed? seem to remember it avoids triggering marks if grid is not set up - return if !$self->{marks}; - - foreach my $mark (@{$self->{marks}}) { - $mark->hide; - } - my @strings = ('<-2.58', '[-2.58,-1.96)', '[-1.96,-1.65)', '[-1.65,1.65]', '(1.65,1.96]', '(1.96,2.58]', '>2.58'); - - my $mark_arr = $self->{zscore_marks} //= []; - if (!@$mark_arr) { - foreach my $i (0 .. $#strings) { - my $anchor_loc = $i == 0 ? 'nw' : $i == $#strings ? 'sw' : 'w'; - $mark_arr->[$i] = $self->make_mark($anchor_loc); - } - } - - # Set legend textbox markers - foreach my $i (0 .. $#strings) { - my $mark = $mark_arr->[$#strings - $i]; - $mark->set( text => $strings[$i] ); - # $mark->show; - $mark->raise_to_top; - } - - return; + return $self->set_text_marks_for_labels (\@strings); } # refactor needed sub set_text_marks_divergent { - my ($self, $mid, $extent) = @_; + my ($self, $min, $max) = @_; + + my $extent = max (abs($min), abs ($max)); + my $mid = 0; my $mid2 = ($mid + $extent) / 2; my @strings = ( @@ -1071,11 +1082,11 @@ sub set_text_marks_divergent { } # say join ' ', @strings; - $self->set_text_marks_for_labels (\@strings, $self->{divergent_marks}); + $self->set_text_marks_for_labels (\@strings); } sub set_text_marks_ratio { - my ($self, $max) = @_; + my ($self, $min, $max) = @_; $max //= 1; my $mid = 1 + ($max - 1) / 2; @@ -1103,13 +1114,13 @@ sub set_text_marks_ratio { $strings[-1] = ">=$strings[-1]"; } - $self->set_text_marks_for_labels (\@strings, $self->{ratio_marks}); + $self->set_text_marks_for_labels (\@strings); } sub set_text_marks_prank { my $self = shift; my @strings = ('<0.01', '<0.025', '<0.05', '[0.05,0.95]', '>0.95', '>0.975', '>0.99'); - $self->set_text_marks_for_labels (\@strings, $self->{prank_marks}); + $self->set_text_marks_for_labels (\@strings); } # generalises z-score version - need to simplify it @@ -1117,21 +1128,22 @@ sub set_text_marks_for_labels { my ($self, \@strings, $mark_arr) = @_; # needed? seem to remember it avoids triggering marks if grid is not set up - return if !$self->{marks}; + return if !$self->{marks}{default}; $mark_arr //= []; - carp "Mark count does not match label count" - if scalar(@strings) != scalar @$mark_arr; + $self->{marks}{current} //= $self->{marks}{default}; - foreach my $mark (@{$self->{marks}}) { - $mark->hide; - } + $self->hide_current_marks; if (!@$mark_arr) { - foreach my $i (0 .. $#strings) { - my $anchor_loc = $i == 0 ? 'nw' : $i == $#strings ? 'sw' : 'w'; - $mark_arr->[$i] = $self->make_mark($anchor_loc); + my $n = scalar @strings; + $mark_arr = $self->{marks}{$n} //= []; + if (!@$mark_arr) { # populate if needed + foreach my $i (0 .. $#strings) { + my $anchor_loc = $i == 0 ? 'nw' : $i == $#strings ? 'sw' : 'w'; + $mark_arr->[$i] = $self->make_mark($anchor_loc); + } } } @@ -1143,6 +1155,8 @@ sub set_text_marks_for_labels { $mark->raise_to_top; } + $self->{marks}{current} = $mark_arr; + return; } @@ -1161,206 +1175,162 @@ sub get_log_mode { $_[0]->{log_mode}; } -sub set_canape_mode_on { - my ($self) = @_; - my $prev_val = $self->{canape_mode}; - $self->{canape_mode} = 1; - if (!$prev_val) { # update legend colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}) # trigger a redisplay of the legend - } - return 1; -} - -sub set_canape_mode_off { - my ($self) = @_; - my $prev_val = $self->{canape_mode}; - $self->{canape_mode} = 0; - foreach my $mark (@{$self->{canape_marks}}) { - $mark->hide; - } - if ($prev_val) { # give back our colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}) # trigger a redisplay of the legend - } - return 0; -} - -sub get_canape_mode { - $_[0]->{canape_mode}; -} - -sub set_canape_mode { - my ($self, $bool) = @_; - if ($bool) { - $self->set_canape_mode_on; - } - else { - $self->set_canape_mode_off; - } - return $self->{canape_mode}; -} - -sub set_zscore_mode_on { - my ($self) = @_; - my $prev_val = $self->{zscore_mode}; - $self->{zscore_mode} = 1; - if (!$prev_val) { # update legend colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}); # trigger a redisplay of the legend - } - return 1; -} - -sub set_zscore_mode_off { - my ($self) = @_; - my $prev_val = $self->{zscore_mode}; - $self->{zscore_mode} = 0; - foreach my $mark (@{$self->{zscore_marks}}) { - $mark->hide; - } - if ($prev_val) { # give back our colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}) # trigger a redisplay of the legend - } - return 0; -} - -sub get_zscore_mode { - $_[0]->{zscore_mode}; -} - -sub set_zscore_mode { - my ($self, $bool) = @_; - if ($bool) { - $self->set_zscore_mode_on; - } - else { - $self->set_zscore_mode_off; - } - return $self->{zscore_mode}; -} - -sub set_divergent_mode_on { - my ($self) = @_; - my $prev_val = $self->{divergent_mode}; - $self->{divergent_mode} = 1; - if (!$prev_val) { # update legend colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}); # trigger a redisplay of the legend - } - return 1; +# need a better name +sub _get_nonbasic_plot_modes { + my @modes = qw/canape zscore prank ratio divergent categorical/; + return wantarray ? @modes : \@modes; } -sub set_divergent_mode_off { - my ($self) = @_; - my $prev_val = $self->{divergent_mode}; - $self->{divergent_mode} = 0; - foreach my $mark (@{$self->{divergent_marks}}) { - $mark->hide; - } - if ($prev_val) { # give back our colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}) # trigger a redisplay of the legend - } - return 0; -} +sub set_colour_mode_from_list_and_index { + my ($self, %args) = @_; + my $index = $args{index} // ''; + my $list = $args{list} // ''; -sub get_divergent_mode { - $_[0]->{divergent_mode}; -} + state $bd_obj = Biodiverse::BaseData->new ( + NAME => 'colour-mode', + CELL_SIZES => [1], + CELL_ORIGINS => [0] + ); + state $indices_object = Biodiverse::Indices->new ( + BASEDATA_REF => $bd_obj, + ); -sub set_divergent_mode { - my ($self, $bool) = @_; - if ($bool) { - $self->set_divergent_mode_on; + my $is_list = $list && $list !~ />>/ && $indices_object->index_is_list (index => $list); + if ($is_list) { + $index = $list + } + + # check list name then index name + my %h = (index => $index); + my $mode + = $list =~ />>z_scores>>/ ? 'zscore' + : $list =~ />>p_rank>>/ ? 'prank' + : $list =~ />>CANAPE>>/ && $index =~ /^CANAPE/ ? 'canape' + : $indices_object->index_is_zscore (%h) ? 'zscore' + : $indices_object->index_is_ratio (%h) ? 'ratio' + : $indices_object->index_is_divergent (%h) ? 'divergent' + : $indices_object->index_is_categorical (%h) ? 'categorical' + : ''; + + # clunky to have to iterate over these but they trigger things turning off + # Update - might not be the case now but process does not take long + foreach my $possmode (_get_nonbasic_plot_modes()) { + my $method = "set_${possmode}_mode"; + $self->$method ($mode eq $possmode); + } + + if ($mode eq 'categorical') { + my $labels = $indices_object->get_index_category_labels (index => $index) // {}; + my $colours = $indices_object->get_index_category_colours (index => $index) // {}; + $self->{categorical}{labels} = $labels; + foreach my $key (keys %$colours) { + my $colour = $colours->{$key}; + $colours->{$key} => Gtk2::Gdk::Color->parse($colour); + } + $self->{categorical}{colours} = $colours; } - else { - $self->set_divergent_mode_off; + elsif (!$mode && $list =~ />>CANAPE>>/) { + # special handling for CANAPE indices + my %codes = ( + NEO => 1, PALAEO => 2, MIXED => 3, SUPER => 4, + ); + # special handling + my $colour = $canape_colour_hash{$codes{$index} // 0}; + $self->{categorical}{colours} = { + 0 => $canape_colour_hash{0}, + 1 => $colour, + }; + $self->{categorical}{labels} = { + 0 => 'other', + 1 => lc $index, + }; + $self->set_categorical_mode(1); } - return $self->{divergent_mode}; -} -sub set_ratio_mode_on { - my ($self) = @_; - my $prev_val = $self->{ratio_mode}; - $self->{ratio_mode} = 1; - if (!$prev_val) { # update legend colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}); # trigger a redisplay of the legend - } - return 1; + return; } -sub set_ratio_mode_off { - my ($self) = @_; - my $prev_val = $self->{ratio_mode}; - $self->{ratio_mode} = 0; - foreach my $mark (@{$self->{ratio_marks}}) { - $mark->hide; - } - if ($prev_val) { # give back our colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}) # trigger a redisplay of the legend - } - return 0; -} +sub get_colour_method { + my $self = shift; -sub get_ratio_mode { - $_[0]->{ratio_mode}; -} + my $method = 'get_colour'; -sub set_ratio_mode { - my ($self, $bool) = @_; - if ($bool) { - $self->set_ratio_mode_on; - } - else { - $self->set_ratio_mode_off; + # clunky to have to iterate over these, + # even if we use a lookup table + foreach my $mode (_get_nonbasic_plot_modes()) { + my $check_method = "get_${mode}_mode"; + if ($self->$check_method) { + $method = "get_colour_${mode}"; + } } - return $self->{ratio_mode}; -} -sub set_prank_mode_on { - my ($self) = @_; - my $prev_val = $self->{prank_mode}; - $self->{prank_mode} = 1; - if (!$prev_val) { # update legend colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}); # trigger a redisplay of the legend - } - return 1; + return $method; } -sub set_prank_mode_off { - my ($self) = @_; - my $prev_val = $self->{prank_mode}; - $self->{prank_mode} = 0; - foreach my $mark (@{$self->{prank_marks}}) { - $mark->hide; +# a few factory methods +sub _make_nonbasic_methods { + my ($pkg) = shift || __PACKAGE__; + my @methods = _get_nonbasic_plot_modes(); + print "Calling _make_access_methods for $pkg"; + no strict 'refs'; + foreach my $key (@methods) { + my $method = "get_${key}_mode"; + my $mode_key = "${key}_mode"; + # next if $pkg->can($method); # do not override + *{"${pkg}::${method}"} = + do { + sub { + $_[0]->{$mode_key}; + }; + }; + $method = "set_${key}_mode_on"; + # say STDERR "==== Building $method in package $pkg"; + *{"${pkg}::${method}"} = + do { + sub { + my ($self) = @_; + my $prev_val = $self->{$mode_key}; + $self->{$mode_key} = 1; + if (!$prev_val) { # update legend colours + $self->refresh_legend; + } + return 1; + }; + }; + $method = "set_${key}_mode_off"; + # say STDERR "==== Building $method in package $pkg"; + *{"${pkg}::${method}"} = + do { + sub { + my ($self) = @_; + my $prev_val = $self->{$mode_key}; + $self->{$mode_key} = 0; + $self->hide_current_marks; + if ($prev_val) { # give back our colours + $self->refresh_legend; + } + return 0; + }; + }; + $method = "set_${key}_mode"; + my $mode_off_method = "set_${key}_mode_off"; + my $mode_on_method = "set_${key}_mode_on"; + # say STDERR "==== Building $method in package $pkg"; + *{"${pkg}::${method}"} = + do { + sub { + my ($self, $bool) = @_; + my $method_name = $bool ? $mode_on_method : $mode_off_method; + $self->$method_name; + return $self->{$mode_key}; + }; + }; } - if ($prev_val) { # give back our colours - $self->make_rect; - $self->reposition($self->{width_px}, $self->{height_px}) # trigger a redisplay of the legend - } - return 0; -} - -sub get_prank_mode { - $_[0]->{prank_mode}; -} -sub set_prank_mode { - my ($self, $bool) = @_; - if ($bool) { - $self->set_prank_mode_on; - } - else { - $self->set_prank_mode_off; - } - return $self->{prank_mode}; + return; } +_make_nonbasic_methods(); # dup from Tab.pm - need to inherit from single source diff --git a/lib/Biodiverse/GUI/Tabs/Spatial.pm b/lib/Biodiverse/GUI/Tabs/Spatial.pm index e56a45f3b..2c4f4e3ff 100644 --- a/lib/Biodiverse/GUI/Tabs/Spatial.pm +++ b/lib/Biodiverse/GUI/Tabs/Spatial.pm @@ -1574,11 +1574,10 @@ sub colour_branches_on_dendrogram { my $output_ref = $self->{output_ref}; my $legend = $dendrogram->get_legend; - - my $is_zscore = $self->index_is_zscore (list => $list_name); - $legend->set_zscore_mode ($is_zscore); - my $is_prank = $list_name =~ />>p_rank>>/; - $legend->set_prank_mode ($is_prank); + $legend->set_colour_mode_from_list_and_index ( + list => $list_name, + index => '', + ); my $log_check_box = $self->{xmlPage}->get_object('menuitem_spatial_tree_log_scale'); if ($log_check_box->get_active) { @@ -1598,16 +1597,12 @@ sub colour_branches_on_dendrogram { my $minmax = $self->get_index_min_max_values_across_full_list ($list_name); - - $legend->set_min_max (@$minmax); my ($min, $max) = @$minmax; # should not need to pass this + $legend->set_min_max ($min, $max); - # currently does not handle divergent, ratio or CANAPE - these do not yet apply for tree branches - my @minmax_args = ($is_zscore || $is_prank) ? () : ($min, $max); - my $colour_method - = $is_zscore ? 'get_colour_zscore' - : $is_prank ? 'get_colour_prank' - : 'get_colour'; + # currently does not handle ratio or CANAPE - these do not yet apply for tree branches + my @minmax_args = ($min, $max); + my $colour_method = $legend->get_colour_method; my $checkbox_show_legend = $self->{xmlPage}->get_object('menuitem_spatial_tree_show_legend'); if ($checkbox_show_legend->get_active) { @@ -2005,28 +2000,13 @@ sub recolour { #delete @{$colour_cache}{keys %$colour_cache}; # temp for debug my $ccache = $colour_cache->{$list}{$index} //= {}; - my %list_and_index = (list => $list, index => $index); - my $is_canape = $list =~ />>CANAPE>>/ && $index =~ /^CANAPE/; - my $is_zscore = $self->index_is_zscore (%list_and_index); - my $is_prank = $list =~ />>p_rank>>/; - my $is_ratio - = !$is_prank && !$is_zscore - && $self->index_is_ratio (%list_and_index); - my $is_divergent - = !$is_prank && !$is_zscore && !$is_ratio - && $self->index_is_divergent (%list_and_index); - - my $abs_extreme; - if ($is_ratio) { - $abs_extreme = exp (max (abs log $min, log $max)); - $min = 1 / $abs_extreme; - $max = $abs_extreme; - } - elsif ($is_divergent) { # assumes zero - needs work - $abs_extreme = max(abs $min, abs $max); - $min = 0; - $max = $abs_extreme; - } + my $legend = $grid->get_legend; + + $legend->set_colour_mode_from_list_and_index ( + list => $list, + index => $index, + ); + my $colour_method = $legend->get_colour_method; my $colour_func = sub { my $elt = shift // return; @@ -2040,14 +2020,8 @@ sub recolour { # should use a method here my $val = $elements_hash->{$elt}{$list}{$index}; $colour - = defined $val ? ( - $is_canape ? $grid->get_colour_canape ($val) : - $is_zscore ? $grid->get_colour_zscore ($val) : - $is_prank ? $grid->get_colour_prank ($val) : - $is_ratio ? $grid->get_colour_ratio ($val, $abs_extreme) : - $is_divergent ? $grid->get_colour_divergent ($val, 0, $abs_extreme) : - $grid->get_colour($val, $min, $max) - ) + = defined $val + ? $legend->$colour_method ($val, $min, $max) : $colour_none; } @@ -2061,14 +2035,6 @@ sub recolour { # !$output_ref->group_passed_def_query(group => $elt); #}; - my $legend = $self->{grid}->get_legend; - # This is getting messy but ensures cleanup of old labels. - # Should register active labels. - $legend->set_canape_mode($is_canape); - $legend->set_zscore_mode($is_zscore); - $legend->set_prank_mode($is_prank); - $legend->set_ratio_mode($is_ratio); - $legend->set_divergent_mode($is_divergent); $self->show_legend; $grid->colour($colour_func); diff --git a/lib/Biodiverse/GUI/Tabs/Tab.pm b/lib/Biodiverse/GUI/Tabs/Tab.pm index 34b250743..4b7deb280 100644 --- a/lib/Biodiverse/GUI/Tabs/Tab.pm +++ b/lib/Biodiverse/GUI/Tabs/Tab.pm @@ -534,9 +534,8 @@ sub index_is_ratio { my %args = @_; # check list and then check index - my $list = $args{list} // ''; - - return if $args{list} ne 'SPATIAL_RESULTS'; + my $list = $args{list} // ''; + my $index = $args{index} // ''; state $bd_obj = Biodiverse::BaseData->new ( NAME => 'rationing', @@ -547,7 +546,9 @@ sub index_is_ratio { BASEDATA_REF => $bd_obj, ); - my $index = $args{index} // ''; + return 1 + if $indices_object->index_is_list (index => $list) + && $indices_object->index_is_ratio (index => $list); return $indices_object->index_is_ratio (index => $index); } @@ -557,9 +558,8 @@ sub index_is_divergent { my %args = @_; # check list and then check index - my $list = $args{list} // ''; - - return if $args{list} ne 'SPATIAL_RESULTS'; + my $list = $args{list} // ''; + my $index = $args{index} // ''; state $bd_obj = Biodiverse::BaseData->new ( NAME => 'divergency', @@ -570,7 +570,9 @@ sub index_is_divergent { BASEDATA_REF => $bd_obj, ); - my $index = $args{index} // ''; + return 1 + if $indices_object->index_is_list (index => $list) + && $indices_object->index_is_divergent (index => $list); return $indices_object->index_is_divergent (index => $index); } diff --git a/lib/Biodiverse/Indices.pm b/lib/Biodiverse/Indices.pm index 358da0309..0bb39e570 100644 --- a/lib/Biodiverse/Indices.pm +++ b/lib/Biodiverse/Indices.pm @@ -1275,6 +1275,24 @@ sub get_valid_region_grower_indices { return wantarray ? %indices : \%indices; } +sub get_categorical_indices { + my $self = shift; + my %args = @_; + my $list = $args{calculations} || $self->get_calculations_as_flat_hash; + + my %indices; + foreach my $calculations ( keys %$list ) { + my $meta = $self->get_metadata( sub => $calculations ); + INDEX: + foreach my $index ( keys %{ $meta->get_indices } ) { + next INDEX if !$meta->get_index_is_categorical($index); + $indices{$index} = $meta->get_index_description($index); + } + } + + return wantarray ? %indices : \%indices; +} + sub get_list_indices { my $self = shift; my %args = @_; @@ -1298,13 +1316,27 @@ sub index_is_list { my %args = @_; croak 'argument index not defined' - if !defined $args{index}; + if !defined $args{index}; my $hash = $self->get_list_indices; return $hash->{$args{index} // ''}; } +sub index_is_categorical { + my $self = shift; + my %args = @_; + + croak 'argument index not defined' + if !defined $args{index}; + + my $hash = $self->get_categorical_indices; + + return $hash->{$args{index} // ''}; +} + + + # almost identical to get_list_indices except the "next INDEX" condition sub get_scalar_indices { my $self = shift; @@ -1431,6 +1463,30 @@ sub get_index_bounds { return $bounds; } +sub get_index_category_colours { + my ($self, %args) = @_; + my $index = $args{index}; + + my $index_source = $self->get_index_source(index => $index); + my $meta = $self->get_metadata( sub => $index_source ); + + my $result = $meta->get_index_category_colours ($index); + + return $result; +} + +sub get_index_category_labels { + my ($self, %args) = @_; + my $index = $args{index}; + + my $index_source = $self->get_index_source(index => $index); + my $meta = $self->get_metadata( sub => $index_source ); + + my $result = $meta->get_index_category_labels ($index); + + return $result; +} + sub index_distribution_is_valid { my $self = shift; my %args = @_; diff --git a/lib/Biodiverse/Metadata/Indices.pm b/lib/Biodiverse/Metadata/Indices.pm index dcab5ac17..611049dc8 100644 --- a/lib/Biodiverse/Metadata/Indices.pm +++ b/lib/Biodiverse/Metadata/Indices.pm @@ -219,7 +219,12 @@ sub index_distribution_is_valid { sub get_index_is_ratio { my ($self, $index) = @_; - return return $self->get_index_distribution($index) =~ /ratio$/; + return $self->get_index_distribution($index) =~ /ratio$/; +} + +sub get_index_is_categorical { + my ($self, $index) = @_; + return $self->get_index_distribution($index) eq 'categorical'; } sub get_index_is_nonnegative { @@ -245,6 +250,30 @@ sub get_index_distribution { return $indices->{$index}{distribution} // $self->{distribution} // 'sequential'; } +sub get_index_category_labels { + my ($self, $index) = @_; + + return if !$self->get_index_is_categorical($index); + + no autovivification; + + my $indices = $self->get_indices; + my $hash = $indices->{$index}{labels}; + return wantarray ? %$hash : $hash; +} + +sub get_index_category_colours { + my ($self, $index) = @_; + + return if !$self->get_index_is_categorical($index); + + no autovivification; + + my $indices = $self->get_indices; + my $hash = $indices->{$index}{colours}; + return wantarray ? %$hash : $hash; +} + __PACKAGE__->_make_distribution_methods (keys %valid_distributions); sub _make_distribution_methods { diff --git a/t/23-Indices.t b/t/23-Indices.t index 3bdbef41f..2b5865c64 100644 --- a/t/23-Indices.t +++ b/t/23-Indices.t @@ -5,6 +5,7 @@ use English qw { -no_match_vars }; use Carp; use Test2::V0; +use Test2::Tools::Compare qw/hash/; use rlib; local $| = 1; @@ -242,17 +243,37 @@ sub test_index_bounds { foreach my $index (sort keys %$indices) { my $bounds = $indices_object->get_index_bounds (index => $index); - like $bounds, - [$RE_bound, $RE_bound], - "Bounds for scalar index $index match expected pattern"; my $index_source = $indices_object->get_index_source(index => $index); my $metadata = $indices_object->get_metadata( sub => $index_source ); - my $expected - = $metadata->get_index_is_unit_interval ($index) ? [0,1] - : $metadata->get_index_is_nonnegative ($index) ? [0,'Inf'] - : $metadata->get_index_is_categorical ($index) ? [] - : ['-Inf','Inf']; - is $bounds, $expected, "Bounds correct for $index"; + if ($metadata->get_index_is_categorical($index)) { + # some of these structures still need to be finalised + is $bounds, undef, + "Bounds undefined for categorical index $index"; + my $labels = $indices_object->get_index_category_labels (index => $index); + is $labels, hash { + all_vals D(); + etc() + }, "Categorical index $index: labels all defined"; + my $colours = $indices_object->get_index_category_colours (index => $index) // {}; + if (keys %$colours) { + is $colours, hash { + all_vals D(); + etc() + }, "Categorical index $index: colours all defined"; + } + } + else { + like $bounds, + [ $RE_bound, $RE_bound ], + "Bounds for scalar index $index match expected pattern"; + my $expected + = $metadata->get_index_is_unit_interval($index) ? [ 0, 1 ] + : $metadata->get_index_is_nonnegative($index) ? [ 0, 'Inf' ] + : [ '-Inf', 'Inf' ]; + is $bounds, $expected, "Bounds correct for $index"; + my $labels = $indices_object->get_index_category_labels (index => $index); + is $labels, undef, "No labels defined for continuous index $index"; + } } }