2 // duration.cc -- implement Duration, Plet, Duration_convert, Duration_iterator
4 // copyright 1997 Jan Nieuwenhuizen <jan@digicash.com>
11 #include "source-file.hh"
14 #include "duration.hh"
15 #include "debug.hh" //ugh
17 Duration::Duration( int type_i, int dots_i = 0, Plet* plet_l )
25 Duration::Duration( Duration const& dur_c_r )
39 Duration::operator =( Duration const& dur_c_r )
41 if ( &dur_c_r == this )
44 type_i_ = dur_c_r.type_i_;
45 dots_i_ = dur_c_r.dots_i_;
46 set_plet( dur_c_r.plet_p_ );
52 Duration::set_plet( Plet* plet_l )
57 plet_p_ = new Plet( *plet_l );
60 Plet::Plet( int iso_i, int type_i )
66 Plet::Plet( Plet const& plet_c_r )
68 iso_i_ = plet_c_r.iso_i_;
69 type_i_ = plet_c_r.type_i_;
73 Duration_convert::dur2_str( Duration dur )
75 String str( dur.type_i_ );
76 str += String( '.', dur.dots_i_ );
78 str += String( "*" ) + String( dur.plet_p_->iso_i_ )
79 + String( "/" ) + String( dur.plet_p_->type_i_ );
84 Duration_convert::dur2_i( Duration dur, int division_1_i )
86 return dur2_mom( dur ) * Moment( division_1_i );
90 Duration_convert::dur2_mom( Duration dur )
95 Moment mom = Moment( 1 , dur.type_i_ );
98 while ( dur.dots_i_-- ) {
103 return mom * plet_factor_mom( dur );
107 Duration_convert::mom2_dur( Moment mom )
110 but filling an array using Duration_iterator
111 might speed things up, a little
113 Duration_iterator iter_dur;
116 Duration lower_dur = iter_dur++;
117 Duration upper_dur( 0 );
119 upper_dur = iter_dur();
120 Moment lower_mom = dur2_mom( lower_dur );
121 Moment upper_mom = dur2_mom( upper_dur );
122 if ( mom == lower_mom )
124 if ( mom == upper_mom ) // don-t miss last (sic)
126 if ( ( mom >= lower_mom ) && ( mom <= upper_mom ) ) {
127 warning( String( "duration not exact: " ) + String( (Real)mom ) , 0 );
128 if ( abs( mom - lower_mom ) < abs( mom - upper_mom ) )
133 lower_dur = upper_dur;
135 return Duration( 0 );
139 Duration_convert::plet_factor_mom( Duration dur )
143 return Moment( dur.plet_p_->iso_i_, dur.plet_p_->type_i_ );
147 Duration_convert::sync_f( Duration dur, Moment mom )
149 return mom / dur2_mom( dur );
153 Duration_convert::i2_mom( int time_i, int division_1_i )
158 if ( division_1_i > 0 )
159 return Moment( time_i, division_1_i );
161 return Moment( -division_1_i, time_i );
164 Duration_iterator::Duration_iterator()
166 cursor_dur_.type_i_ = 128;
167 cursor_dur_.set_plet( 0 );
171 Duration_iterator::operator ++(int)
173 return forward_dur();
177 Duration_iterator::operator ()()
182 Duration_iterator::operator bool()
188 Duration_iterator::dur()
194 Duration_iterator::forward_dur()
196 // should do smart table? guessing:
209 Duration dur = cursor_dur_;
211 if ( !cursor_dur_.dots_i_ && !cursor_dur_.plet_p_ ) {
212 cursor_dur_.type_i_ *= 2;
213 cursor_dur_.dots_i_ = 2;
215 else if ( cursor_dur_.dots_i_ == 2 ) {
216 assert( !cursor_dur_.plet_p_ );
217 cursor_dur_.dots_i_ = 0;
218 cursor_dur_.type_i_ /= 4;
219 cursor_dur_.set_plet( &Plet( 2, 3 ) );
221 else if ( cursor_dur_.plet_p_
222 && ( cursor_dur_.plet_p_->iso_i_ == 2 )
223 && ( cursor_dur_.plet_p_->type_i_ == 3 ) ) {
224 assert( !cursor_dur_.dots_i_ );
225 cursor_dur_.set_plet( 0 );
226 cursor_dur_.type_i_ *= 2;
227 cursor_dur_.dots_i_ = 1;
229 else if ( cursor_dur_.dots_i_ == 1 ) {
230 assert( !cursor_dur_.plet_p_ );
231 cursor_dur_.dots_i_ = 0;
232 cursor_dur_.type_i_ /= 2;
236 if ( no_triplets_bo_g && cursor_dur_.plet_p_ && ok() )
243 Duration_iterator::ok()
245 return ( cursor_dur_.type_i_
246 && !( ( cursor_dur_.type_i_ == 1 ) && ( cursor_dur_.dots_i_ > 2 ) ) );