App-Chart

 view release on metacpan or  search on metacpan

lib/App/Chart/Gtk2/Graph.pm  view on Meta::CPAN


# # 'size-allocate' class closure
# sub _do_size_allocate {
#   my ($self, $alloc) = @_;
#   ### Graph _do_size_allocate(): $alloc->width."x".$alloc->height
#   $self->signal_chain_from_overridden($alloc);
# 
#   # after superclass has set $alloc into $self->allocation
#   if ($alloc->width != 1 && $alloc->height != 1) {
#     if (delete $self->{'waiting_initial_allocate'}) {
#       _initial_scale ($self);
#     }
#   }
# }
# 
# sub _do_notify {
#   my ($self, $pspec) = @_;
#   if ($pspec->get_name eq 'parent') {
#     if (! $self->get_parent) {
#       $self->{'waiting_initial_allocate'} = 1;
#     }
#   }
#   return shift->signal_chain_from_overridden(@_);
# }

sub initial_scale {
  my ($self) = @_;
  $self->{'initial_scale'} = 1;
}

sub _initial_scale {
  my ($self) = @_;
  ### Graph _initial_scale(): "$self"

#   {
#     my $alloc = $self->allocation;
#     ### Graph _initial_scale() size: $alloc->width."x".$alloc->height
#     return if ($alloc->width == 1 || $alloc->height == 1);
#   }

  my $series_list = $self->{'series_list'};
  my $series = $series_list->[0] || do {
    ### no series...
    return;
  };

  my $hadj = $self->{'hadjustment'};
  my $vadj = $self->{'vadjustment'};
  my ($lo, $hi) = $hadj->value_range_inc;
  ### hadj: "$lo $hi, on ".ref($series)

  if (my ($p_lo, $p_hi) = $series->initial_range ($lo, $hi)) {
    ### series initial_range(): ($p_lo//'undef')." to ".($p_hi//'undef')." from $lo to $hi on ".ref($series)
    ($p_lo, $p_hi) = stretch_range ($p_lo, $p_hi);
    ### stretched to: ($p_lo//'undef')." to ".($p_hi//'undef')
    if (defined $p_lo) {
      $vadj->set_page_range ($p_lo, $p_hi);
    }
    $self->queue_draw;
  }
  # expanding on initial ...
  $self->{'vrange_span'} = undef;
  update_v_range ($self);
}

# 'set-scroll-adjustments' class closure
sub _do_set_scroll_adjustments {
  my ($self, $hadj, $vadj) = @_;
  $self->set (hadjustment => $hadj,
              vadjustment => $vadj);
}

sub scale_x_step {
  my ($self) = @_;
  return $self->{'hadjustment'}->get_pixel_per_value;
}

sub scale_x {
  my ($self, $t) = @_;
  return $self->{'hadjustment'}->value_to_pixel ($t);
}

sub scale_x_proc {
  my ($self) = @_;
  return $self->{'hadjustment'}->value_to_pixel_proc;
}

sub x_to_date {
  my ($self, $x) = @_;
  return POSIX::floor ($self->{'hadjustment'}->pixel_to_value ($x));
}

sub scale_y {
  my ($self, $y) = @_;
  return $self->{'vadjustment'}->value_to_pixel ($y);
}

sub scale_y_proc {
  my ($self) = @_;
  return $self->{'vadjustment'}->value_to_pixel_proc;

#   my $price_lo = $self->{'vadjustment'}->get_value;
#   my $price_height = $self->{'vadjustment'}->page_size;
#   if ($price_height == 0) { $price_height = 1; }
#   my ($win_width, $win_height) = $self->window->get_size();
#   my $factor = $win_height / $price_height;
#
#   return sub {
#     my ($price) = @_;
#     return $win_height - $factor * ($price - $price_lo);
#   };
}

sub y_to_value {
  my ($self, $y) = @_;
  return $self->{'vadjustment'}->pixel_to_value ($y);

#   my $win_height = $self->allocation->height;
#   my $vadj = $self->{'vadjustment'};
#   my $factor = $vadj->value
#     + ($win_height - $y) * $vadj->page_size / $win_height;

lib/App/Chart/Gtk2/Graph.pm  view on Meta::CPAN


  my $region = $event->region;
  Gtk2::Ex::GdkBits::window_clear_region ($self->window, $region);

  if (! @$series_list) {
    App::Chart::Gtk2::GUI::draw_text_centred
        ($self, $event, __('Use File/Open to open or add a symbol'));

  } else {
    _draw_region ($self, $region);
  }
  ### Graph _do_expose_event() end ...
  return Gtk2::EVENT_PROPAGATE;
}

sub _draw_region {
  my ($self, $region) = @_;
  ### Graph _draw_region() ...

  my $series_list = $self->get('series_list');
  my $any = 0;
  foreach my $series (@$series_list) {
    ### Graph draw_region linestyle: $series->linestyle_class
    my $class = $series->linestyle_class // next;
    Module::Load::load ($class);
    $any |= $class->draw ($self, $series, $region);
  }
  if (! $any) {
    ### Graph _draw_region() no data ...
    App::Chart::Gtk2::GUI::draw_text_centred ($self, $region, __('no data'));
  }

  foreach my $class (@plugins) {
    $class->draw ($self, $region);
  }
  ### Graph _draw_region() end ...
}

# 'changed' and 'value-changed' signals on vadjustment
sub _do_vadj_changed {
  my ($adj, $ref_weak_self) = @_;
  my $self = $$ref_weak_self || return;
  ### Graph vadj changed, redraw: "value=@{[$adj->value]} upper=@{[$adj->upper]} lower=@{[$adj->lower]}"
  #                if ($adj->value < 0) {
  #                  require Devel::StackTrace;
  #                  my $trace = Devel::StackTrace->new;
  #                  print $trace->as_string; # like carp
  #                }

  $self->queue_draw;
}

# Expand $adj so its lower/upper covers all of @values.
#
# If lower==upper==page_size==0 in the existing $adj settings it's treated
# as uninitialized and that 0 lower/upper is ignored, just @values is used.
#
# undefs in @values are ignored, and if all are undef then $adj is not
# changed.
#
sub adjustment_expand {
  my ($adj, @values) = @_;
  @values = grep {defined} @values;
  ### Graph adjustment_expand(): join(' ',@values)
  if (! @values) { return; }

  my ($new_lower, $new_upper) = List::MoreUtils::minmax (@values);
  ### base: "   new_lower $new_lower  new_upper $new_upper"
  ($new_lower, $new_upper) = stretch_range ($new_lower, $new_upper);
  ### stretch: "new_lower $new_lower  new_upper $new_upper"

  my $old_lower = $adj->lower;
  my $old_upper = $adj->upper;
  ### old: "    old_lower $old_lower  old_upper $old_upper  old_page ".$adj->page_size
  if (! ($old_lower == 0 && $old_upper == 0 && $adj->page_size == 0)) {
    ($new_lower, $new_upper) = List::MoreUtils::minmax
      ($new_lower, $new_upper, $old_lower, $old_upper);
  }

  ### new: "    new_lower $new_lower  new_upper $new_upper"
  Gtk2::Ex::AdjustmentBits::set_maybe
      ($adj,
       lower => $new_lower,
       upper => $new_upper);
}

sub stretch_range {
  my ($lo, $hi) = @_;
  my $extra = ($hi - $lo) * 0.1;
  if ($lo < 0) {
    $lo -= $extra;
  } else {
    $lo = max ($lo - $extra, $lo * 0.5);
  }
  $hi += $extra;
  return ($lo, $hi);
}

sub update_v_range {
  my ($self) = @_;
  my $vadj = $self->{'vadjustment'} || return;

  my ($lo, $hi) = $self->draw_t_range;
  my $vrange_span = ($self->{'vrange_span'} ||= do {
    require Set::IntSpan::Fast;
    Set::IntSpan::Fast->new
    });
  if ($vrange_span->contains_all_range ($lo, $hi)) { return; }

  ### Graph update_v_range for: "$lo to $hi"
  my $series_list = $self->{'series_list'};
  adjustment_expand ($vadj,
                     (map {
                       $_->vrange ($self, $series_list);
                     } @plugins),
                     (map {
                       _series_v_range($_, $lo, $hi)
                     } @$series_list));
  $vrange_span->add_range ($lo, $hi);
}
sub _series_v_range {
  my ($series, $lo, $hi) = @_;
  my @ret;
  my $min = $series->minimum;  push @ret, $min;
  my $max = $series->maximum;  push @ret, $max;
  $series->fill ($lo, $hi);

  foreach my $p ($series->filled_low, $series->filled_high) {
    $p // next;
    push @ret, $p;
#     foreach my $w ($p * 1.1, $p / 1.1) {
#       if (defined $min) { $w = max ($w, $min); }
#       if (defined $max) { $w = min ($w, $max); }
#       push @ret, $w;
#     }
  }
  return @ret;
}

# 'value-changed' signal on hadjustment
sub _do_hadj_value_changed {
  my ($adj, $ref_weak_self) = @_;
  my $self = $$ref_weak_self || return;
  ### Graph hadj changed, v_range and redraw: "value=@{[$adj->value]} upper=@{[$adj->upper]} lower=@{[$adj->lower]}"

  update_v_range ($self);
  $self->queue_draw;
}

# 'changed' signal on hadjustment
*_do_hadj_other_changed = \&_do_hadj_value_changed;
#   my ($adj, $ref_weak_self) = @_;
#   my $self = $$ref_weak_self || return;
#   update_v_range ($self);
#   $self->queue_draw;
# }

# 'button-press-event' class closure
sub _do_button_press {
  my ($self, $event) = @_;
  ### Graph _do_button_press(): $event->button
  require App::Chart::Gtk2::Ex::BindingBits;
  App::Chart::Gtk2::Ex::BindingBits::activate_button_event
      ('App__Chart__Gtk2__Graph_keys', $event, $self);
  return shift->signal_chain_from_overridden(@_);
}

sub centre {
  my ($self) = @_;
  ### Graph centre()
  my $vadj = $self->{'vadjustment'};
  my $page = $vadj->page_size * 0.9; # gap at ends

lib/App/Chart/Gtk2/Graph.pm  view on Meta::CPAN

    if ($new_page <= $page) {
      $l = $new_l;
      $h = $new_h;
      return 1;
    }

    if ($new_l < $l) {
      $l = $h - $page;
    } else {
      $h = $l + $page;
    }
    return 0;
  };

  if (my $series = $series_list->[0]) {
    if (defined (my $symbol = $series->symbol)) {
      my ($latest_lo,$latest_hi)
        = App::Chart::Gtk2::Graph::Plugin::Latest->hrange ($self, $series_list);
      ### latest hrange: $latest_lo,$latest_hi
      if (defined $lo
          && App::Chart::overlap_inclusive_p ($lo,$hi,
                                              $latest_lo,$latest_hi)) {
        my $latest = App::Chart::Latest->get($symbol);
        if ($series->isa('App::Chart::Series::Derived::Volume')) {
          $accumulate->($latest->{'volume'});
        } else {
          $accumulate->($latest->{'last'})
            and $accumulate->($latest->{'bid'})
              and $accumulate->($latest->{'offer'})
                and $accumulate->($latest->{'high'})
                  and $accumulate->($latest->{'low'});
        }
      }
    }
  }

  my @arrays;
  foreach my $series (@$series_list) {
    $series->fill ($lo, $hi);
    my $values = $series->values_array;
    push @arrays, $values;
    if (my $highs = $series->array('highs')) {
      if ($highs != $values) { push @arrays, $highs; }
    }
    if (my $lows = $series->array('lows')) {
      if ($lows != $values) { push @arrays, $lows; }
    }
  }

 OUTER: for (my $i = $hi; $i >= $lo; $i--) {
    foreach my $array (@arrays) {
      $accumulate->($array->[$i])
        or last OUTER;
    }
  }
  if (! defined $l) { return; }
  ### decided: "$l to $h"

  my $extra = $page - ($h - $l);
  $l -= $extra / 2;
  ### expand to: "low $l on page $page"
  $vadj->set_value ($l);
}


#------------------------------------------------------------------------------
# scrolling

# 'scroll-event' class closure
sub _do_scroll_event {
  my ($self, $event) = @_;
  ### Graph _do_scroll_event(): "$self->{'hadjustment'}, $self->{'vadjustment'}"

  my $direction = $event->direction;
  if    ($direction eq 'up')    { $self->{'vadjustment'}->scroll_step(1); }
  elsif ($direction eq 'down')  { $self->{'vadjustment'}->scroll_step(-1); }
  elsif ($direction eq 'left')  { $self->{'hadjustment'}->scroll_step(1); }
  elsif ($direction eq 'right') { $self->{'hadjustment'}->scroll_step(-1); }

  return $self->signal_chain_from_overridden ($event);
}


#------------------------------------------------------------------------------
# action signal handlers

sub _do_start_drag {
  my ($self, $button) = @_;
  my $hadj = $self->{'hadjustment'} || return; # only when adj set
  my $vadj = $self->{'vadjustment'} || return; # only when adj set
  my $dragger = ($self->{'dragger'} ||= do {
    require Gtk2::Ex::Dragger;
    Gtk2::Ex::Dragger->new (widget => $self,
                            hadjustment => $hadj,
                            vadjustment => $vadj,
                            vinverted   => 1,
                            confine     => 1,
                            cursor      => 'fleur')
    });
  $dragger->start (Gtk2->get_current_event);
}

sub _do_start_lasso {
  my ($self, $button) = @_;
  my $lasso = ($self->{'lasso'} ||= do {
    require Gtk2::Ex::Lasso;
    my $l = Gtk2::Ex::Lasso->new (widget => $self);
    $l->signal_connect (ended => \&_do_lasso_ended);
    $l
  });
  $lasso->start (Gtk2->get_current_event);
}
sub _do_lasso_ended {
  my ($lasso, $x1,$y1, $x2,$y2) = @_;
  my $self = $lasso->get('widget') || return;

  my $hadj = $self->{'hadjustment'};
  my $t1 = $self->x_to_date ($x1);
  my $t2 = $self->x_to_date ($x2);
  $hadj->set_value_range (min($t1,$t2), max($t1,$t2));



( run in 0.445 second using v1.01-cache-2.11-cpan-5623c5533a1 )