historical/toontown-classic.git/panda/include/cMetaInterval.h
2024-01-16 11:20:27 -06:00

204 lines
5.8 KiB
C++

/**
* PANDA 3D SOFTWARE
* Copyright (c) Carnegie Mellon University. All rights reserved.
*
* All use of this software is subject to the terms of the revised BSD
* license. You should have received a copy of this license along
* with this source code in a file named "LICENSE."
*
* @file cMetaInterval.h
* @author drose
* @date 2002-08-27
*/
#ifndef CMETAINTERVAL_H
#define CMETAINTERVAL_H
#include "directbase.h"
#include "cInterval.h"
#include "pointerTo.h"
#include "pdeque.h"
#include "pvector.h"
#include "plist.h"
#include "pset.h"
#include <math.h>
/**
* This interval contains a list of nested intervals, each of which has its
* own begin and end times. Some of them may overlap and some of them may
* not.
*/
class EXPCL_DIRECT_INTERVAL CMetaInterval : public CInterval {
PUBLISHED:
explicit CMetaInterval(const std::string &name);
virtual ~CMetaInterval();
enum RelativeStart {
RS_previous_end,
RS_previous_begin,
RS_level_begin,
};
INLINE void set_precision(double precision);
INLINE double get_precision() const;
void clear_intervals();
int push_level(const std::string &name,
double rel_time, RelativeStart rel_to);
int add_c_interval(CInterval *c_interval,
double rel_time = 0.0f,
RelativeStart rel_to = RS_previous_end);
int add_ext_index(int ext_index, const std::string &name,
double duration, bool open_ended,
double rel_time, RelativeStart rel_to);
int pop_level(double duration = -1.0);
bool set_interval_start_time(const std::string &name, double rel_time,
RelativeStart rel_to = RS_level_begin);
double get_interval_start_time(const std::string &name) const;
double get_interval_end_time(const std::string &name) const;
enum DefType {
DT_c_interval,
DT_ext_index,
DT_push_level,
DT_pop_level
};
INLINE int get_num_defs() const;
INLINE DefType get_def_type(int n) const;
INLINE CInterval *get_c_interval(int n) const;
INLINE int get_ext_index(int n) const;
virtual void priv_initialize(double t);
virtual void priv_instant();
virtual void priv_step(double t);
virtual void priv_finalize();
virtual void priv_reverse_initialize(double t);
virtual void priv_reverse_instant();
virtual void priv_reverse_finalize();
virtual void priv_interrupt();
INLINE bool is_event_ready();
INLINE int get_event_index() const;
INLINE double get_event_t() const;
INLINE EventType get_event_type() const;
void pop_event();
virtual void write(std::ostream &out, int indent_level) const;
void timeline(std::ostream &out) const;
protected:
virtual void do_recompute();
private:
class IntervalDef {
public:
DefType _type;
PT(CInterval) _c_interval;
int _ext_index;
std::string _ext_name;
double _ext_duration;
bool _ext_open_ended;
double _rel_time;
RelativeStart _rel_to;
int _actual_begin_time;
};
enum PlaybackEventType {
PET_begin,
PET_end,
PET_instant
};
class PlaybackEvent {
public:
INLINE PlaybackEvent(int time, int n, PlaybackEventType type);
INLINE bool operator < (const PlaybackEvent &other) const;
int _time;
int _n;
PlaybackEventType _type;
PlaybackEvent *_begin_event;
};
class EventQueueEntry {
public:
INLINE EventQueueEntry(int n, EventType event_type, int time);
int _n;
EventType _event_type;
int _time;
};
typedef pvector<IntervalDef> Defs;
typedef pvector<PlaybackEvent *> PlaybackEvents;
// ActiveEvents must be either a list or a vector--something that preserves
// order--so we can call priv_step() on the currently active intervals in
// the order they were encountered.
typedef plist<PlaybackEvent *> ActiveEvents;
typedef pdeque<EventQueueEntry> EventQueue;
INLINE int double_to_int_time(double t) const;
INLINE double int_to_double_time(int time) const;
void clear_events();
void do_event_forward(PlaybackEvent *event, ActiveEvents &new_active,
bool is_initial);
void finish_events_forward(int now, ActiveEvents &new_active);
void do_event_reverse(PlaybackEvent *event, ActiveEvents &new_active,
bool is_initial);
void finish_events_reverse(int now, ActiveEvents &new_active);
void enqueue_event(int n, CInterval::EventType event_type, bool is_initial,
int time = 0);
void enqueue_self_event(CInterval::EventType event_type, double t = 0.0);
void enqueue_done_event();
bool service_event_queue();
int recompute_level(int n, int level_begin, int &level_end);
int get_begin_time(const IntervalDef &def, int level_begin,
int previous_begin, int previous_end);
void write_event_desc(std::ostream &out, const IntervalDef &def,
int &extra_indent_level) const;
double _precision;
Defs _defs;
int _current_nesting_level;
PlaybackEvents _events;
ActiveEvents _active;
int _end_time;
size_t _next_event_index;
bool _processing_events;
// This is the queue of events that have occurred due to a recent
// priv_initialize(), priv_step(), etc., but have not yet been serviced, due
// to an embedded external (e.g. Python) interval that the scripting
// language must service. This queue should be considered precious, and
// should never be arbitrarily flushed without servicing all of its events.
EventQueue _event_queue;
public:
static TypeHandle get_class_type() {
return _type_handle;
}
static void init_type() {
CInterval::init_type();
register_type(_type_handle, "CMetaInterval",
CInterval::get_class_type());
}
virtual TypeHandle get_type() const {
return get_class_type();
}
virtual TypeHandle force_init_type() {init_type(); return get_class_type();}
private:
static TypeHandle _type_handle;
};
#include "cMetaInterval.I"
#endif