Blame lib/Date/Manip/Delta.pm

Packit 95306a
package Date::Manip::Delta;
Packit 95306a
# Copyright (c) 1995-2017 Sullivan Beck. All rights reserved.
Packit 95306a
# This program is free software; you can redistribute it and/or modify it
Packit 95306a
# under the same terms as Perl itself.
Packit 95306a
Packit 95306a
########################################################################
Packit 95306a
# Any routine that starts with an underscore (_) is NOT intended for
Packit 95306a
# public use.  They are for internal use in the the Date::Manip
Packit 95306a
# modules and are subject to change without warning or notice.
Packit 95306a
#
Packit 95306a
# ABSOLUTELY NO USER SUPPORT IS OFFERED FOR THESE ROUTINES!
Packit 95306a
########################################################################
Packit 95306a
Packit 95306a
use Date::Manip::Obj;
Packit 95306a
@ISA = ('Date::Manip::Obj');
Packit 95306a
Packit 95306a
require 5.010000;
Packit 95306a
use warnings;
Packit 95306a
use strict;
Packit 95306a
use utf8;
Packit 95306a
use IO::File;
Packit 95306a
#use re 'debug';
Packit 95306a
Packit 95306a
use Date::Manip::Base;
Packit 95306a
use Date::Manip::TZ;
Packit 95306a
Packit 95306a
our $VERSION;
Packit 95306a
$VERSION='6.60';
Packit 95306a
END { undef $VERSION; }
Packit 95306a
Packit 95306a
########################################################################
Packit 95306a
# BASE METHODS
Packit 95306a
########################################################################
Packit 95306a
Packit 95306a
sub is_delta {
Packit 95306a
   return 1;
Packit 95306a
}
Packit 95306a
Packit 95306a
sub config {
Packit 95306a
   my($self,@args) = @_;
Packit 95306a
   $self->SUPER::config(@args);
Packit 95306a
Packit 95306a
   # A new config can change the value of the format fields, so clear them.
Packit 95306a
   $$self{'data'}{'f'}    = {};
Packit 95306a
   $$self{'data'}{'flen'} = {};
Packit 95306a
}
Packit 95306a
Packit 95306a
# Call this every time a new delta is put in to make sure everything is
Packit 95306a
# correctly initialized.
Packit 95306a
#
Packit 95306a
sub _init {
Packit 95306a
   my($self) = @_;
Packit 95306a
Packit 95306a
   my $def = [0,0,0,0,0,0,0];
Packit 95306a
   my $dmt = $$self{'tz'};
Packit 95306a
   my $dmb = $$dmt{'base'};
Packit 95306a
Packit 95306a
   $$self{'err'}  = '';
Packit 95306a
   $$self{'data'} = {
Packit 95306a
                     'delta'      => $def,  # the delta (all negative fields signed)
Packit 95306a
                     'in'         => '',    # the string that was parsed (if any)
Packit 95306a
                     'length'     => 0,     # length of delta (in seconds)
Packit 95306a
Packit 95306a
                     'gotmode'    => 0,     # 1 if mode set explicitly
Packit 95306a
                     'business'   => 0,     # 1 for a business delta
Packit 95306a
Packit 95306a
                     'f'          => {},    # format fields
Packit 95306a
                     'flen'       => {},    # field lengths
Packit 95306a
                    }
Packit 95306a
}
Packit 95306a
Packit 95306a
sub _init_args {
Packit 95306a
   my($self) = @_;
Packit 95306a
Packit 95306a
   my @args = @{ $$self{'args'} };
Packit 95306a
   $self->parse(@args);
Packit 95306a
}
Packit 95306a
Packit 95306a
sub value {
Packit 95306a
   my($self) = @_;
Packit 95306a
   my $dmt = $$self{'tz'};
Packit 95306a
   my $dmb = $$dmt{'base'};
Packit 95306a
Packit 95306a
   return ''  if ($$self{'err'});
Packit 95306a
   if (wantarray) {
Packit 95306a
      return @{ $$self{'data'}{'delta'} };
Packit 95306a
   } else {
Packit 95306a
      my @delta = @{ $$self{'data'}{'delta'} };
Packit 95306a
      my $err;
Packit 95306a
      ($err,@delta) = $dmb->_delta_fields( { 'nonorm'  => 1,
Packit 95306a
                                             'source'  => 'delta',
Packit 95306a
                                             'sign'    => 0 },
Packit 95306a
                                           [@delta]);
Packit 95306a
      return ''  if ($err);
Packit 95306a
      return join(':',@delta);
Packit 95306a
   }
Packit 95306a
}
Packit 95306a
Packit 95306a
sub input {
Packit 95306a
   my($self) = @_;
Packit 95306a
   return  $$self{'data'}{'in'};
Packit 95306a
}
Packit 95306a
Packit 95306a
########################################################################
Packit 95306a
# DELTA METHODS
Packit 95306a
########################################################################
Packit 95306a
Packit 95306a
BEGIN {
Packit 95306a
   my %ops = map { $_,1 } qw( delta business normal standard );
Packit 95306a
   my %f   = qw( y 0  M 1  w 2  d 3  h 4  m 5  s 6 );
Packit 95306a
Packit 95306a
   sub set {
Packit 95306a
      my($self,$field,$val,$no_normalize) = @_;
Packit 95306a
Packit 95306a
      my $dmt        = $$self{'tz'};
Packit 95306a
      my $dmb        = $$dmt{'base'};
Packit 95306a
      my $zone       = $$self{'data'}{'tz'};
Packit 95306a
      my $gotmode    = $$self{'data'}{'gotmode'};
Packit 95306a
      my $business   = 0;
Packit 95306a
Packit 95306a
      my (@delta,$err);
Packit 95306a
Packit 95306a
      if (exists $ops{lc($field)}) {
Packit 95306a
         $field       = lc($field);
Packit 95306a
Packit 95306a
         if ($field eq 'business') {
Packit 95306a
            $business = 1;
Packit 95306a
            $gotmode  = 1;
Packit 95306a
         } elsif ($field eq 'normal'  ||  $field eq 'standard') {
Packit 95306a
            $business = 0;
Packit 95306a
            $gotmode  = 1;
Packit 95306a
         } elsif ($field eq 'delta') {
Packit 95306a
            $business = $$self{'data'}{'business'};
Packit 95306a
            $gotmode  = $$self{'data'}{'gotmode'};
Packit 95306a
         }
Packit 95306a
Packit 95306a
         ($err,@delta) = $dmb->_delta_fields( { 'nonorm'   => $no_normalize,
Packit 95306a
                                                'business' => $business,
Packit 95306a
                                                'source'   => 'delta',
Packit 95306a
                                                'sign'     => -1 },
Packit 95306a
                                              $val);
Packit 95306a
Packit 95306a
      } elsif (exists $f{$field}) {
Packit 95306a
Packit 95306a
         if ($$self{'err'}) {
Packit 95306a
            $$self{'err'} = "[set] Invalid delta";
Packit 95306a
            return 1;
Packit 95306a
         }
Packit 95306a
Packit 95306a
         @delta             = @{ $$self{'data'}{'delta'} };
Packit 95306a
         $business          = $$self{'data'}{'business'};
Packit 95306a
         $delta[$f{$field}] = $val;
Packit 95306a
Packit 95306a
         ($err,@delta) = $dmb->_delta_fields( { 'nonorm'   => $no_normalize,
Packit 95306a
                                                'business' => $business,
Packit 95306a
                                                'source'   => 'delta',
Packit 95306a
                                                'sign'     => -1 },
Packit 95306a
                                              [@delta]);
Packit 95306a
Packit 95306a
      } elsif (lc($field) eq 'mode') {
Packit 95306a
Packit 95306a
         @delta             = @{ $$self{'data'}{'delta'} };
Packit 95306a
         $val               = lc($val);
Packit 95306a
         if ($val eq 'business'  ||  $val eq 'normal'  ||  $val eq 'standard') {
Packit 95306a
            $gotmode        = 1;
Packit 95306a
            $business       = ($val eq 'business' ? 1 : 0);
Packit 95306a
Packit 95306a
         } else {
Packit 95306a
            $$self{'err'} = "[set] Invalid mode: $val";
Packit 95306a
            return 1;
Packit 95306a
         }
Packit 95306a
Packit 95306a
      } else {
Packit 95306a
Packit 95306a
         $$self{'err'} = "[set] Invalid field: $field";
Packit 95306a
         return 1;
Packit 95306a
Packit 95306a
      }
Packit 95306a
Packit 95306a
      if ($err) {
Packit 95306a
         $$self{'err'} = "[set] Invalid field value: $field";
Packit 95306a
         return 1;
Packit 95306a
      }
Packit 95306a
Packit 95306a
      $self->_init();
Packit 95306a
      $$self{'data'}{'delta'}      = [ @delta ];
Packit 95306a
      $$self{'data'}{'business'}   = $business;
Packit 95306a
      $$self{'data'}{'gotmode'}    = $gotmode;
Packit 95306a
      $$self{'data'}{'length'}     = 'unknown';
Packit 95306a
Packit 95306a
      return 0;
Packit 95306a
   }
Packit 95306a
}
Packit 95306a
Packit 95306a
sub _rx {
Packit 95306a
   my($self,$rx) = @_;
Packit 95306a
   my $dmt = $$self{'tz'};
Packit 95306a
   my $dmb = $$dmt{'base'};
Packit 95306a
Packit 95306a
   return $$dmb{'data'}{'rx'}{'delta'}{$rx}
Packit 95306a
     if (exists $$dmb{'data'}{'rx'}{'delta'}{$rx});
Packit 95306a
Packit 95306a
   if ($rx eq 'expanded') {
Packit 95306a
      my $sign    = '[-+]?\s*';
Packit 95306a
      my $sep     = '(?:,\s*|\s+|$)';
Packit 95306a
Packit 95306a
      my $nth     = $$dmb{'data'}{'rx'}{'nth'}[0];
Packit 95306a
      my $yf      = $$dmb{data}{rx}{fields}[1];
Packit 95306a
      my $mf      = $$dmb{data}{rx}{fields}[2];
Packit 95306a
      my $wf      = $$dmb{data}{rx}{fields}[3];
Packit 95306a
      my $df      = $$dmb{data}{rx}{fields}[4];
Packit 95306a
      my $hf      = $$dmb{data}{rx}{fields}[5];
Packit 95306a
      my $mnf     = $$dmb{data}{rx}{fields}[6];
Packit 95306a
      my $sf      = $$dmb{data}{rx}{fields}[7];
Packit 95306a
      my $num     = '(?:\d+(?:\.\d*)?|\.\d+)';
Packit 95306a
Packit 95306a
      my $y       = "(?:(?:(?<y>$sign$num)|(?<y>$nth))\\s*(?:$yf)$sep)";
Packit 95306a
      my $m       = "(?:(?:(?<m>$sign$num)|(?<m>$nth))\\s*(?:$mf)$sep)";
Packit 95306a
      my $w       = "(?:(?:(?<w>$sign$num)|(?<w>$nth))\\s*(?:$wf)$sep)";
Packit 95306a
      my $d       = "(?:(?:(?<d>$sign$num)|(?<d>$nth))\\s*(?:$df)$sep)";
Packit 95306a
      my $h       = "(?:(?:(?<h>$sign$num)|(?<h>$nth))\\s*(?:$hf)$sep)";
Packit 95306a
      my $mn      = "(?:(?:(?<mn>$sign$num)|(?<mn>$nth))\\s*(?:$mnf)$sep)";
Packit 95306a
      my $s       = "(?:(?:(?<s>$sign$num)|(?<s>$nth))\\s*(?:$sf)?)";
Packit 95306a
Packit 95306a
      my $exprx   = qr/^\s*$y?$m?$w?$d?$h?$mn?$s?\s*$/i;
Packit 95306a
      $$dmb{'data'}{'rx'}{'delta'}{$rx} = $exprx;
Packit 95306a
Packit 95306a
   } elsif ($rx eq 'mode') {
Packit 95306a
Packit 95306a
      my $mode = qr/\b($$dmb{'data'}{'rx'}{'mode'}[0])\b/i;
Packit 95306a
      $$dmb{'data'}{'rx'}{'delta'}{$rx} = $mode;
Packit 95306a
Packit 95306a
   } elsif ($rx eq 'when') {
Packit 95306a
Packit 95306a
      my $when = qr/\b($$dmb{'data'}{'rx'}{'when'}[0])\b/i;
Packit 95306a
      $$dmb{'data'}{'rx'}{'delta'}{$rx} = $when;
Packit 95306a
Packit 95306a
   }
Packit 95306a
Packit 95306a
   return $$dmb{'data'}{'rx'}{'delta'}{$rx};
Packit 95306a
}
Packit 95306a
Packit 95306a
sub parse {
Packit 95306a
   my($self,$instring,@args) = @_;
Packit 95306a
   my($business,$no_normalize,$gotmode,$err,@delta);
Packit 95306a
Packit 95306a
   if (@args == 2) {
Packit 95306a
      ($business,$no_normalize) = (lc($args[0]),lc($args[1]));
Packit 95306a
      if      ($business eq 'standard') {
Packit 95306a
         $business = 0;
Packit 95306a
      } elsif ($business eq 'business') {
Packit 95306a
         $business = 1;
Packit 95306a
      } elsif ($business) {
Packit 95306a
         $business = 1;
Packit 95306a
      } else {
Packit 95306a
         $business = 0;
Packit 95306a
      }
Packit 95306a
      if ($no_normalize) {
Packit 95306a
         $no_normalize = 1;
Packit 95306a
      } else {
Packit 95306a
         $no_normalize = 0;
Packit 95306a
      }
Packit 95306a
      $gotmode = 1;
Packit 95306a
Packit 95306a
   } elsif (@args == 1) {
Packit 95306a
      my $arg = lc($args[0]);
Packit 95306a
      if      ($arg eq 'standard') {
Packit 95306a
         $business     = 0;
Packit 95306a
         $no_normalize = 0;
Packit 95306a
         $gotmode      = 1;
Packit 95306a
      } elsif ($arg eq 'business') {
Packit 95306a
         $business     = 1;
Packit 95306a
         $no_normalize = 0;
Packit 95306a
         $gotmode      = 1;
Packit 95306a
      } elsif ($arg eq 'nonormalize') {
Packit 95306a
         $business     = 0;
Packit 95306a
         $no_normalize = 1;
Packit 95306a
         $gotmode      = 0;
Packit 95306a
      } elsif ($arg) {
Packit 95306a
         $business     = 1;
Packit 95306a
         $no_normalize = 0;
Packit 95306a
         $gotmode      = 1;
Packit 95306a
      } else {
Packit 95306a
         $business     = 0;
Packit 95306a
         $no_normalize = 0;
Packit 95306a
         $gotmode      = 0;
Packit 95306a
      }
Packit 95306a
   } elsif (@args == 0) {
Packit 95306a
      $business     = 0;
Packit 95306a
      $no_normalize = 0;
Packit 95306a
      $gotmode      = 0;
Packit 95306a
   } else {
Packit 95306a
      $$self{'err'} = "[parse] Unknown arguments";
Packit 95306a
      return 1;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   my $dmt = $$self{'tz'};
Packit 95306a
   my $dmb = $$dmt{'base'};
Packit 95306a
   $self->_init();
Packit 95306a
Packit 95306a
   if (! $instring) {
Packit 95306a
      $$self{'err'} = '[parse] Empty delta string';
Packit 95306a
      return 1;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   #
Packit 95306a
   # Parse the string
Packit 95306a
   #
Packit 95306a
Packit 95306a
   $$self{'err'} = '';
Packit 95306a
   $instring     =~ s/^\s*//;
Packit 95306a
   $instring     =~ s/\s*$//;
Packit 95306a
Packit 95306a
 PARSE: {
Packit 95306a
Packit 95306a
      # First, we'll try the standard format (without a mode string)
Packit 95306a
Packit 95306a
      ($err,@delta) = $dmb->_split_delta($instring);
Packit 95306a
      last PARSE  if (! $err);
Packit 95306a
Packit 95306a
      # Next, we'll need to get a list of all the encodings and look
Packit 95306a
      # for (and remove) the mode string from each.  We'll also recheck
Packit 95306a
      # the standard format for each.
Packit 95306a
Packit 95306a
      my @strings = $dmb->_encoding($instring);
Packit 95306a
      my $moderx  = $self->_rx('mode');
Packit 95306a
      my %mode    = ();
Packit 95306a
Packit 95306a
      foreach my $string (@strings) {
Packit 95306a
         if ($string =~ s/\s*$moderx\s*//i) {
Packit 95306a
            my $b = $1;
Packit 95306a
            if ($$dmb{'data'}{'wordmatch'}{'mode'}{lc($b)} == 1) {
Packit 95306a
               $b = 0;
Packit 95306a
            } else {
Packit 95306a
               $b = 1;
Packit 95306a
            }
Packit 95306a
Packit 95306a
            ($err,@delta) = $dmb->_split_delta($string);
Packit 95306a
            if (! $err) {
Packit 95306a
               $business = $b;
Packit 95306a
               $gotmode  = 1;
Packit 95306a
               last PARSE;
Packit 95306a
            }
Packit 95306a
Packit 95306a
            $mode{$string} = $b;
Packit 95306a
         }
Packit 95306a
      }
Packit 95306a
Packit 95306a
      # Now we'll check each string for an expanded form delta.
Packit 95306a
Packit 95306a
      foreach my $string (@strings) {
Packit 95306a
         my($b,$g);
Packit 95306a
         if (exists $mode{$string}) {
Packit 95306a
            $b = $mode{$string};
Packit 95306a
            $g = 1;
Packit 95306a
         } else {
Packit 95306a
            $b = $business;
Packit 95306a
            $g = 0;
Packit 95306a
         }
Packit 95306a
Packit 95306a
         my $past    = 0;
Packit 95306a
Packit 95306a
         my $whenrx  = $self->_rx('when');
Packit 95306a
         if ($string  &&
Packit 95306a
             $string =~ s/$whenrx//i) {
Packit 95306a
            my $when = $1;
Packit 95306a
            if ($$dmb{'data'}{'wordmatch'}{'when'}{lc($when)} == 1) {
Packit 95306a
               $past   = 1;
Packit 95306a
            }
Packit 95306a
         }
Packit 95306a
Packit 95306a
         my $rx        = $self->_rx('expanded');
Packit 95306a
         if ($string  &&
Packit 95306a
             $string   =~ $rx) {
Packit 95306a
            $business  = $b;
Packit 95306a
            $gotmode   = $g;
Packit 95306a
            @delta     = @+{qw(y m w d h mn s)};
Packit 95306a
            foreach my $f (@delta) {
Packit 95306a
               if (! defined $f) {
Packit 95306a
                  $f = 0;
Packit 95306a
               } elsif (exists $$dmb{'data'}{'wordmatch'}{'nth'}{lc($f)}) {
Packit 95306a
                  $f = $$dmb{'data'}{'wordmatch'}{'nth'}{lc($f)};
Packit 95306a
               } else {
Packit 95306a
                  $f =~ s/\s//g;
Packit 95306a
               }
Packit 95306a
            }
Packit 95306a
Packit 95306a
            # if $past, reverse the signs
Packit 95306a
            if ($past) {
Packit 95306a
               foreach my $v (@delta) {
Packit 95306a
                  $v *= -1;
Packit 95306a
               }
Packit 95306a
            }
Packit 95306a
Packit 95306a
            last PARSE;
Packit 95306a
         }
Packit 95306a
      }
Packit 95306a
   }
Packit 95306a
Packit 95306a
   if (! @delta) {
Packit 95306a
      $$self{'err'} = "[parse] Invalid delta string";
Packit 95306a
      return 1;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   ($err,@delta) = $dmb->_delta_fields( { 'nonorm'   => $no_normalize,
Packit 95306a
                                          'business' => $business,
Packit 95306a
                                          'source'   => 'string',
Packit 95306a
                                          'sign'     => -1 },
Packit 95306a
                                        [@delta]);
Packit 95306a
Packit 95306a
   if ($err) {
Packit 95306a
      $$self{'err'} = "[parse] Invalid delta string";
Packit 95306a
      return 1;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   $$self{'data'}{'in'}         = $instring;
Packit 95306a
   $$self{'data'}{'delta'}      = [@delta];
Packit 95306a
   $$self{'data'}{'business'}   = $business;
Packit 95306a
   $$self{'data'}{'gotmode'}    = $gotmode;
Packit 95306a
   $$self{'data'}{'length'}     = 'unknown';
Packit 95306a
   return 0;
Packit 95306a
}
Packit 95306a
Packit 95306a
sub printf {
Packit 95306a
   my($self,@in) = @_;
Packit 95306a
   if ($$self{'err'}) {
Packit 95306a
      warn "WARNING: [printf] Object must contain a valid delta\n";
Packit 95306a
      return undef;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   my($y,$M,$w,$d,$h,$m,$s) = @{ $$self{'data'}{'delta'} };
Packit 95306a
Packit 95306a
   my @out;
Packit 95306a
   foreach my $in (@in) {
Packit 95306a
      my $out = '';
Packit 95306a
      while ($in) {
Packit 95306a
         if ($in =~ s/^([^%]+)//) {
Packit 95306a
            $out .= $1;
Packit 95306a
Packit 95306a
         } elsif ($in =~ s/^%%//) {
Packit 95306a
            $out .= "%";
Packit 95306a
Packit 95306a
         } elsif ($in =~ s/^%
Packit 95306a
                           (\+)?                   # sign
Packit 95306a
                           ([<>0])?                # pad
Packit 95306a
                           (\d+)?                  # width
Packit 95306a
                           ([yMwdhms])             # field
Packit 95306a
                           v                       # type
Packit 95306a
                          //ox) {
Packit 95306a
            my($sign,$pad,$width,$field) = ($1,$2,$3,$4);
Packit 95306a
            $out .= $self->_printf_field($sign,$pad,$width,0,$field);
Packit 95306a
Packit 95306a
         } elsif ($in =~ s/^(%
Packit 95306a
                              (\+)?                   # sign
Packit 95306a
                              ([<>0])?                # pad
Packit 95306a
                              (\d+)?                  # width
Packit 95306a
                              (?:\.(\d+))?            # precision
Packit 95306a
                              ([yMwdhms])             # field
Packit 95306a
                              ([yMwdhms])             # field0
Packit 95306a
                              ([yMwdhms])             # field1
Packit 95306a
                           )//ox) {
Packit 95306a
            my($match,$sign,$pad,$width,$precision,$field,$field0,$field1) =
Packit 95306a
              ($1,$2,$3,$4,$5,$6,$7,$8);
Packit 95306a
Packit 95306a
            # Get the list of fields we're expressing
Packit 95306a
Packit 95306a
            my @field = qw(y M w d h m s);
Packit 95306a
            while (@field  &&  $field[0] ne $field0) {
Packit 95306a
               shift(@field);
Packit 95306a
            }
Packit 95306a
            while (@field  &&  $field[$#field] ne $field1) {
Packit 95306a
               pop(@field);
Packit 95306a
            }
Packit 95306a
Packit 95306a
            if (! @field) {
Packit 95306a
               $out .= $match;
Packit 95306a
            } else {
Packit 95306a
               $out .=
Packit 95306a
                 $self->_printf_field($sign,$pad,$width,$precision,$field,@field);
Packit 95306a
            }
Packit 95306a
Packit 95306a
         } elsif ($in =~ s/^%
Packit 95306a
                           (\+)?                   # sign
Packit 95306a
                           ([<>])?                 # pad
Packit 95306a
                           (\d+)?                  # width
Packit 95306a
                           Dt
Packit 95306a
                          //ox) {
Packit 95306a
            my($sign,$pad,$width) = ($1,$2,$3);
Packit 95306a
            $out .= $self->_printf_delta($sign,$pad,$width,'y','s');
Packit 95306a
Packit 95306a
         } elsif ($in =~ s/^(%
Packit 95306a
                              (\+)?                   # sign
Packit 95306a
                              ([<>])?                 # pad
Packit 95306a
                              (\d+)?                  # width
Packit 95306a
                              D
Packit 95306a
                              ([yMwdhms])             # field0
Packit 95306a
                              ([yMwdhms])             # field1
Packit 95306a
                           )//ox) {
Packit 95306a
            my($match,$sign,$pad,$width,$field0,$field1) = ($1,$2,$3,$4,$5,$6);
Packit 95306a
Packit 95306a
            # Get the list of fields we're expressing
Packit 95306a
Packit 95306a
            my @field = qw(y M w d h m s);
Packit 95306a
            while (@field  &&  $field[0] ne $field0) {
Packit 95306a
               shift(@field);
Packit 95306a
            }
Packit 95306a
            while (@field  &&  $field[$#field] ne $field1) {
Packit 95306a
               pop(@field);
Packit 95306a
            }
Packit 95306a
Packit 95306a
            if (! @field) {
Packit 95306a
               $out .= $match;
Packit 95306a
            } else {
Packit 95306a
               $out .= $self->_printf_delta($sign,$pad,$width,$field[0],
Packit 95306a
                                            $field[$#field]);
Packit 95306a
            }
Packit 95306a
Packit 95306a
         } else {
Packit 95306a
            $in =~ s/^(%[^%]*)//;
Packit 95306a
            $out .= $1;
Packit 95306a
         }
Packit 95306a
      }
Packit 95306a
      push(@out,$out);
Packit 95306a
   }
Packit 95306a
Packit 95306a
   if (wantarray) {
Packit 95306a
      return @out;
Packit 95306a
   } elsif (@out == 1) {
Packit 95306a
      return $out[0];
Packit 95306a
   }
Packit 95306a
Packit 95306a
   return ''
Packit 95306a
}
Packit 95306a
Packit 95306a
sub _printf_delta {
Packit 95306a
   my($self,$sign,$pad,$width,$field0,$field1) = @_;
Packit 95306a
   my $dmt = $$self{'tz'};
Packit 95306a
   my $dmb = $$dmt{'base'};
Packit 95306a
   my @delta = @{ $$self{'data'}{'delta'} };
Packit 95306a
   my $delta;
Packit 95306a
   my %tmp   = qw(y 0 M 1 w 2 d 3 h 4 m 5 s 6);
Packit 95306a
Packit 95306a
   # Add a sign to each field
Packit 95306a
Packit 95306a
   my $s = "+";
Packit 95306a
   foreach my $f (@delta) {
Packit 95306a
      if ($f < 0) {
Packit 95306a
         $s = "-";
Packit 95306a
      } elsif ($f > 0) {
Packit 95306a
         $s = "+";
Packit 95306a
         $f *= 1;
Packit 95306a
         $f = "+$f";
Packit 95306a
      } else {
Packit 95306a
         $f = "$s$f";
Packit 95306a
      }
Packit 95306a
   }
Packit 95306a
Packit 95306a
   # Split the delta into field sets containing only those fields to
Packit 95306a
   # print.
Packit 95306a
   #
Packit 95306a
   # @set = ( [SETa] [SETb] ....)
Packit 95306a
   #   where [SETx] is a listref of fields from one set of fields
Packit 95306a
Packit 95306a
   my @set;
Packit 95306a
   my $business = $$self{'data'}{'business'};
Packit 95306a
Packit 95306a
   my $f0 = $tmp{$field0};
Packit 95306a
   my $f1 = $tmp{$field1};
Packit 95306a
Packit 95306a
   if ($field0 eq $field1) {
Packit 95306a
      @set = ( [ $delta[$f0] ] );
Packit 95306a
Packit 95306a
   } elsif ($business) {
Packit 95306a
Packit 95306a
      if ($f0 <= 1) {
Packit 95306a
         # if (field0 = y or M)
Packit 95306a
         #    add [y,M]
Packit 95306a
         #    field0 = w   OR   done if field1 = M
Packit 95306a
         push(@set, [ @delta[0..1] ]);
Packit 95306a
         $f0 = ($f1 == 1 ? 7 : 2);
Packit 95306a
      }
Packit 95306a
Packit 95306a
      if ($f0 == 2) {
Packit 95306a
         # if (field0 = w)
Packit 95306a
         #    add [w]
Packit 95306a
         #    field0 = d  OR  done if field1 = w
Packit 95306a
         push(@set, [ $delta[2] ]);
Packit 95306a
         $f0 = ($f1 == 2 ? 7 : 3);
Packit 95306a
      }
Packit 95306a
Packit 95306a
      if ($f0 <= 6) {
Packit 95306a
         push(@set, [ @delta[$f0..$f1] ]);
Packit 95306a
      }
Packit 95306a
Packit 95306a
   } else {
Packit 95306a
Packit 95306a
      if ($f0 <= 1) {
Packit 95306a
         # if (field0 = y or M)
Packit 95306a
         #    add [y,M]
Packit 95306a
         #    field0 = w   OR   done if field1 = M
Packit 95306a
         push(@set, [ @delta[0..1] ]);
Packit 95306a
         $f0 = ($f1 == 1 ? 7 : 2);
Packit 95306a
      }
Packit 95306a
Packit 95306a
      if ($f0 <= 6) {
Packit 95306a
         push(@set, [ @delta[$f0..$f1] ]);
Packit 95306a
      }
Packit 95306a
   }
Packit 95306a
Packit 95306a
   # If we're not forcing signs, remove signs from all fields
Packit 95306a
   # except the first in each set.
Packit 95306a
Packit 95306a
   my @ret;
Packit 95306a
Packit 95306a
   foreach my $set (@set) {
Packit 95306a
      my @f = @$set;
Packit 95306a
Packit 95306a
      if (defined($sign)  &&  $sign eq "+") {
Packit 95306a
         push(@ret,@f);
Packit 95306a
      } else {
Packit 95306a
         push(@ret,shift(@f));
Packit 95306a
         foreach my $f (@f) {
Packit 95306a
            $f =~ s/[-+]//;
Packit 95306a
            push(@ret,$f);
Packit 95306a
         }
Packit 95306a
      }
Packit 95306a
   }
Packit 95306a
Packit 95306a
   # Width/pad
Packit 95306a
Packit 95306a
   my $ret = join(':',@ret);
Packit 95306a
   if ($width  &&  length($ret) < $width) {
Packit 95306a
      if (defined $pad  &&  $pad eq ">") {
Packit 95306a
         $ret .= ' 'x($width-length($ret));
Packit 95306a
      } else {
Packit 95306a
         $ret = ' 'x($width-length($ret)) . $ret;
Packit 95306a
      }
Packit 95306a
   }
Packit 95306a
Packit 95306a
   return $ret;
Packit 95306a
}
Packit 95306a
Packit 95306a
sub _printf_field {
Packit 95306a
   my($self,$sign,$pad,$width,$precision,$field,@field) = @_;
Packit 95306a
Packit 95306a
   my $val = $self->_printf_field_val($field,@field);
Packit 95306a
   $pad    = "<"  if (! defined($pad));
Packit 95306a
Packit 95306a
   # Strip off the sign.
Packit 95306a
Packit 95306a
   my $s = '';
Packit 95306a
Packit 95306a
   if ($val < 0) {
Packit 95306a
      $s   = "-";
Packit 95306a
      $val *= -1;
Packit 95306a
   } elsif ($sign) {
Packit 95306a
      $s   = "+";
Packit 95306a
   }
Packit 95306a
Packit 95306a
   # Handle the precision.
Packit 95306a
Packit 95306a
   if (defined($precision)) {
Packit 95306a
      $val = sprintf("%.${precision}f",$val);
Packit 95306a
Packit 95306a
   } elsif (defined($width)) {
Packit 95306a
      my $i = $s . int($val) . '.';
Packit 95306a
      if (length($i) < $width) {
Packit 95306a
         $precision = $width-length($i);
Packit 95306a
         $val = sprintf("%.${precision}f",$val);
Packit 95306a
      }
Packit 95306a
   }
Packit 95306a
Packit 95306a
   # Handle padding.
Packit 95306a
Packit 95306a
   if ($width) {
Packit 95306a
      if      ($pad eq ">") {
Packit 95306a
         $val = "$s$val";
Packit 95306a
         my $pad = ($width > length($val) ? $width - length($val) : 0);
Packit 95306a
         $val .= ' 'x$pad;
Packit 95306a
Packit 95306a
      } elsif ($pad eq "<") {
Packit 95306a
         $val = "$s$val";
Packit 95306a
         my $pad = ($width > length($val) ? $width - length($val) : 0);
Packit 95306a
         $val = ' 'x$pad . $val;
Packit 95306a
Packit 95306a
      } else {
Packit 95306a
         my $pad = ($width > length($val)-length($s) ?
Packit 95306a
                    $width - length($val) - length($s): 0);
Packit 95306a
         $val = $s . '0'x$pad . $val;
Packit 95306a
      }
Packit 95306a
   } else {
Packit 95306a
      $val = "$s$val";
Packit 95306a
   }
Packit 95306a
Packit 95306a
   return $val;
Packit 95306a
}
Packit 95306a
Packit 95306a
# $$self{'data'}{'f'}{X}{Y} is the value of field X expressed in terms of Y.
Packit 95306a
#
Packit 95306a
sub _printf_field_val {
Packit 95306a
   my($self,$field,@field) = @_;
Packit 95306a
Packit 95306a
   if (! exists $$self{'data'}{'f'}{'y'}  &&
Packit 95306a
       ! exists $$self{'data'}{'f'}{'y'}{'y'}) {
Packit 95306a
Packit 95306a
      my($yv,$Mv,$wv,$dv,$hv,$mv,$sv) = map { $_*1 } @{ $$self{'data'}{'delta'} };
Packit 95306a
      $$self{'data'}{'f'}{'y'}{'y'} = $yv;
Packit 95306a
      $$self{'data'}{'f'}{'M'}{'M'} = $Mv;
Packit 95306a
      $$self{'data'}{'f'}{'w'}{'w'} = $wv;
Packit 95306a
      $$self{'data'}{'f'}{'d'}{'d'} = $dv;
Packit 95306a
      $$self{'data'}{'f'}{'h'}{'h'} = $hv;
Packit 95306a
      $$self{'data'}{'f'}{'m'}{'m'} = $mv;
Packit 95306a
      $$self{'data'}{'f'}{'s'}{'s'} = $sv;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   # A single field
Packit 95306a
Packit 95306a
   if (! @field) {
Packit 95306a
      return $$self{'data'}{'f'}{$field}{$field};
Packit 95306a
   }
Packit 95306a
Packit 95306a
   # Find the length of 1 unit of each field in terms of seconds.
Packit 95306a
Packit 95306a
   if (! exists $$self{'data'}{'flen'}{'s'}) {
Packit 95306a
      my $business = $$self{'data'}{'business'};
Packit 95306a
      my $dmb      = $self->base();
Packit 95306a
      $$self{'data'}{'flen'} = { 's'  => 1,
Packit 95306a
                                 'm'  => 60,
Packit 95306a
                                 'h'  => 3600,
Packit 95306a
                                 'd'  => $$dmb{'data'}{'len'}{$business}{'dl'},
Packit 95306a
                                 'w'  => $$dmb{'data'}{'len'}{$business}{'wl'},
Packit 95306a
                                 'M'  => $$dmb{'data'}{'len'}{$business}{'ml'},
Packit 95306a
                                 'y'  => $$dmb{'data'}{'len'}{$business}{'yl'},
Packit 95306a
                               };
Packit 95306a
   }
Packit 95306a
Packit 95306a
   # Calculate the value for each field.
Packit 95306a
Packit 95306a
   my $val = 0;
Packit 95306a
   foreach my $f (@field) {
Packit 95306a
Packit 95306a
      # We want the value of $f expressed in terms of $field
Packit 95306a
Packit 95306a
      if (! exists $$self{'data'}{'f'}{$f}{$field}) {
Packit 95306a
Packit 95306a
         # Get the value of $f expressed in seconds
Packit 95306a
Packit 95306a
         if (! exists $$self{'data'}{'f'}{$f}{'s'}) {
Packit 95306a
            $$self{'data'}{'f'}{$f}{'s'} =
Packit 95306a
              $$self{'data'}{'f'}{$f}{$f} * $$self{'data'}{'flen'}{$f};
Packit 95306a
         }
Packit 95306a
Packit 95306a
         # Get the value of $f expressed in terms of $field
Packit 95306a
Packit 95306a
         $$self{'data'}{'f'}{$f}{$field} =
Packit 95306a
           $$self{'data'}{'f'}{$f}{'s'} / $$self{'data'}{'flen'}{$field};
Packit 95306a
      }
Packit 95306a
Packit 95306a
      $val += $$self{'data'}{'f'}{$f}{$field};
Packit 95306a
   }
Packit 95306a
Packit 95306a
   return $val;
Packit 95306a
}
Packit 95306a
Packit 95306a
sub type {
Packit 95306a
   my($self,$op) = @_;
Packit 95306a
   $op = lc($op);
Packit 95306a
Packit 95306a
   if      ($op eq 'business') {
Packit 95306a
      return $$self{'data'}{'business'};
Packit 95306a
   } elsif ($op eq 'standard') {
Packit 95306a
      return 1-$$self{'data'}{'business'};
Packit 95306a
   }
Packit 95306a
Packit 95306a
   my($exact,$semi,$approx) = (0,0,0);
Packit 95306a
   my($y,$m,$w,$d,$h,$mn,$s) = @{ $$self{'data'}{'delta'} };
Packit 95306a
   if ($y  ||  $m) {
Packit 95306a
      $approx = 1;
Packit 95306a
   } elsif ($w  ||  (! $$self{'data'}{'business'}  &&  $d)) {
Packit 95306a
      $semi = 1;
Packit 95306a
   } else {
Packit 95306a
      $exact = 1;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   if      ($op eq 'exact') {
Packit 95306a
      return $exact;
Packit 95306a
   } elsif ($op eq 'semi') {
Packit 95306a
      return $semi;
Packit 95306a
   } elsif ($op eq 'approx') {
Packit 95306a
      return $approx;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   return undef;
Packit 95306a
}
Packit 95306a
Packit 95306a
sub calc {
Packit 95306a
   my($self,$obj,$subtract,$no_normalize) = @_;
Packit 95306a
   if ($$self{'err'}) {
Packit 95306a
      $$self{'err'} = "[calc] First object invalid (delta)";
Packit 95306a
      return undef;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   if      (ref($obj) eq 'Date::Manip::Date') {
Packit 95306a
      if ($$obj{'err'}) {
Packit 95306a
         $$self{'err'} = "[calc] Second object invalid (date)";
Packit 95306a
         return undef;
Packit 95306a
      }
Packit 95306a
      return $obj->calc($self,$subtract);
Packit 95306a
Packit 95306a
   } elsif (ref($obj) eq 'Date::Manip::Delta') {
Packit 95306a
      if ($$obj{'err'}) {
Packit 95306a
         $$self{'err'} = "[calc] Second object invalid (delta)";
Packit 95306a
         return undef;
Packit 95306a
      }
Packit 95306a
      return $self->_calc_delta_delta($obj,$subtract,$no_normalize);
Packit 95306a
Packit 95306a
   } else {
Packit 95306a
      $$self{'err'} = "[calc] Second object must be a Date/Delta object";
Packit 95306a
      return undef;
Packit 95306a
   }
Packit 95306a
}
Packit 95306a
Packit 95306a
sub _calc_delta_delta {
Packit 95306a
   my($self,$delta,@args) = @_;
Packit 95306a
   my $dmt = $$self{'tz'};
Packit 95306a
   my $dmb = $$dmt{'base'};
Packit 95306a
   my $ret = $self->new_delta;
Packit 95306a
Packit 95306a
   if ($self->err()) {
Packit 95306a
      $$ret{'err'} = "[calc] First delta object invalid";
Packit 95306a
      return $ret;
Packit 95306a
   } elsif ($delta->err()) {
Packit 95306a
      $$ret{'err'} = "[calc] Second delta object invalid";
Packit 95306a
      return $ret;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   my($subtract,$no_normalize);
Packit 95306a
   if (@args == 2) {
Packit 95306a
      ($subtract,$no_normalize) = @args;
Packit 95306a
   } elsif ($args[0] eq 'nonormalize') {
Packit 95306a
      $subtract     = 0;
Packit 95306a
      $no_normalize = 1;
Packit 95306a
   } else {
Packit 95306a
      $subtract     = 0;
Packit 95306a
      $no_normalize = 0;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   my $business = 0;
Packit 95306a
   if ($$self{'data'}{'business'} != $$delta{'data'}{'business'}) {
Packit 95306a
      $$ret{'err'} = "[calc] Delta/delta calculation objects must be of " .
Packit 95306a
        'the same type';
Packit 95306a
      return $ret;
Packit 95306a
   } else {
Packit 95306a
      $business = $$self{'data'}{'business'};
Packit 95306a
   }
Packit 95306a
Packit 95306a
   my ($err,@delta);
Packit 95306a
   for (my $i=0; $i<7; $i++) {
Packit 95306a
      if ($subtract) {
Packit 95306a
         $delta[$i] = $$self{'data'}{'delta'}[$i] - $$delta{'data'}{'delta'}[$i];
Packit 95306a
      } else {
Packit 95306a
         $delta[$i] = $$self{'data'}{'delta'}[$i] + $$delta{'data'}{'delta'}[$i];
Packit 95306a
      }
Packit 95306a
   }
Packit 95306a
Packit 95306a
   ($err,@delta) = $dmb->_delta_fields( { 'nonorm'  => 0,
Packit 95306a
                                          'source'  => 'delta',
Packit 95306a
                                          'sign'    => -1 },
Packit 95306a
                                        [@delta])  if (! $no_normalize);
Packit 95306a
Packit 95306a
   $$ret{'data'}{'delta'}       = [@delta];
Packit 95306a
   $$ret{'data'}{'business'}    = $business;
Packit 95306a
   $$self{'data'}{'length'}     = 'unknown';
Packit 95306a
Packit 95306a
   return $ret;
Packit 95306a
}
Packit 95306a
Packit 95306a
sub convert {
Packit 95306a
   my($self,$to) = @_;
Packit 95306a
Packit 95306a
   # What mode are we currently in
Packit 95306a
Packit 95306a
   my $from;
Packit 95306a
   my($y,$m,$w,$d,$h,$mn,$s) = @{ $$self{'data'}{'delta'} };
Packit 95306a
   if ($y  ||  $m) {
Packit 95306a
      $from = 'approx';
Packit 95306a
   } elsif ($w  ||  (! $$self{'data'}{'business'}  &&  $d)) {
Packit 95306a
      $from = 'semi';
Packit 95306a
   } else {
Packit 95306a
      $from = 'exact';
Packit 95306a
   }
Packit 95306a
Packit 95306a
   my $business = $$self{'data'}{'business'};
Packit 95306a
Packit 95306a
   #
Packit 95306a
   # Do the conversion
Packit 95306a
   #
Packit 95306a
Packit 95306a
   {
Packit 95306a
      no integer;
Packit 95306a
Packit 95306a
      my $dmb = $self->base();
Packit 95306a
      my $yl  = $$dmb{'data'}{'len'}{$business}{'yl'};
Packit 95306a
      my $ml  = $$dmb{'data'}{'len'}{$business}{'ml'};
Packit 95306a
      my $wl  = $$dmb{'data'}{'len'}{$business}{'wl'};
Packit 95306a
      my $dl  = $$dmb{'data'}{'len'}{$business}{'dl'};
Packit 95306a
Packit 95306a
      # Convert it to seconds
Packit 95306a
Packit 95306a
      $s += $y*$yl + $m*$ml + $w*$wl + $d*$dl + $h*3600 + $mn*60;
Packit 95306a
      ($y,$m,$w,$d,$h,$mn) = (0,0,0,0,0,0);
Packit 95306a
Packit 95306a
      # Convert it to $to
Packit 95306a
Packit 95306a
      if ($to eq 'approx') {
Packit 95306a
         # Figure out how many months there are
Packit 95306a
         $m          = int($s/$ml);
Packit 95306a
         $s         -= $m*$ml;
Packit 95306a
      }
Packit 95306a
Packit 95306a
      if ($to eq 'approx'  ||  $to eq 'semi') {
Packit 95306a
         if ($business) {
Packit 95306a
            $w       = int($s/$wl);
Packit 95306a
            $s      -= $w*$wl;
Packit 95306a
         } else {
Packit 95306a
            $d       = int($s/$dl);
Packit 95306a
            $s      -= $d*$dl;
Packit 95306a
         }
Packit 95306a
      }
Packit 95306a
Packit 95306a
      $s = int($s);
Packit 95306a
   }
Packit 95306a
Packit 95306a
   $self->set('delta',[$y,$m,$w,$d,$h,$mn,$s]);
Packit 95306a
}
Packit 95306a
Packit 95306a
sub cmp {
Packit 95306a
   my($self,$delta) = @_;
Packit 95306a
Packit 95306a
   if ($$self{'err'}) {
Packit 95306a
      warn "WARNING: [cmp] Arguments must be valid deltas: delta1\n";
Packit 95306a
      return undef;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   if (! ref($delta) eq 'Date::Manip::Delta') {
Packit 95306a
      warn "WARNING: [cmp] Argument must be a Date::Manip::Delta object\n";
Packit 95306a
      return undef;
Packit 95306a
   }
Packit 95306a
   if ($$delta{'err'}) {
Packit 95306a
      warn "WARNING: [cmp] Arguments must be valid deltas: delta2\n";
Packit 95306a
      return undef;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   if ($$self{'data'}{'business'} != $$delta{'data'}{'business'}) {
Packit 95306a
      warn "WARNING: [cmp] Deltas must both be business or standard\n";
Packit 95306a
      return undef;
Packit 95306a
   }
Packit 95306a
Packit 95306a
   my $business = $$self{'data'}{'business'};
Packit 95306a
   my $dmb      = $self->base();
Packit 95306a
   my $yl       = $$dmb{'data'}{'len'}{$business}{'yl'};
Packit 95306a
   my $ml       = $$dmb{'data'}{'len'}{$business}{'ml'};
Packit 95306a
   my $wl       = $$dmb{'data'}{'len'}{$business}{'wl'};
Packit 95306a
   my $dl       = $$dmb{'data'}{'len'}{$business}{'dl'};
Packit 95306a
Packit 95306a
   if ($$self{'data'}{'length'} eq 'unknown') {
Packit 95306a
      my($y,$m,$w,$d,$h,$mn,$s) = @{ $$self{'data'}{'delta'} };
Packit 95306a
Packit 95306a
      no integer;
Packit 95306a
      $$self{'data'}{'length'}  = int($y*$yl + $m*$ml + $w*$wl +
Packit 95306a
                                      $d*$dl + $h*3600 + $mn*60 + $s);
Packit 95306a
   }
Packit 95306a
Packit 95306a
   if ($$delta{'data'}{'length'} eq 'unknown') {
Packit 95306a
      my($y,$m,$w,$d,$h,$mn,$s) = @{ $$delta{'data'}{'delta'} };
Packit 95306a
Packit 95306a
      no integer;
Packit 95306a
      $$delta{'data'}{'length'}  = int($y*$yl + $m*$ml + $w*$wl +
Packit 95306a
                                       $d*$dl + $h*3600 + $mn*60 + $s);
Packit 95306a
   }
Packit 95306a
Packit 95306a
   return ($$self{'data'}{'length'} <=> $$delta{'data'}{'length'});
Packit 95306a
}
Packit 95306a
Packit 95306a
1;
Packit 95306a
# Local Variables:
Packit 95306a
# mode: cperl
Packit 95306a
# indent-tabs-mode: nil
Packit 95306a
# cperl-indent-level: 3
Packit 95306a
# cperl-continued-statement-offset: 2
Packit 95306a
# cperl-continued-brace-offset: 0
Packit 95306a
# cperl-brace-offset: 0
Packit 95306a
# cperl-brace-imaginary-offset: 0
Packit 95306a
# cperl-label-offset: 0
Packit 95306a
# End: