/*
- translator.hh -- declare Translator
+ This file is part of LilyPond, the GNU music typesetter.
- source file of the GNU LilyPond music typesetter
+ Copyright (C) 1997--2015 Han-Wen Nienhuys <hanwen@xs4all.nl>
- (c) 1997--2001 Han-Wen Nienhuys <hanwen@cs.uu.nl>
-*/
+ LilyPond is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ LilyPond is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+ You should have received a copy of the GNU General Public License
+ along with LilyPond. If not, see <http://www.gnu.org/licenses/>.
+*/
#ifndef TRANSLATOR_HH
#define TRANSLATOR_HH
#include "global-ctor.hh"
-#include "string.hh"
#include "lily-proto.hh"
#include "virtual-methods.hh"
-#include "lily-guile.hh"
-#include "parray.hh"
-#include "input.hh"
+#include "input.hh" // for error reporting
#include "smobs.hh"
+#include "std-vector.hh"
+#include "protected-scm.hh"
+
+struct Acknowledge_information
+{
+ SCM symbol_;
+ Engraver_void_function_engraver_grob_info function_;
+
+ Acknowledge_information ()
+ {
+ symbol_ = SCM_EOL;
+ function_ = 0;
+ }
+};
-/** Make some kind of #Element#s from Requests. Elements are made by
- hierarchically grouped #Translator#s
- */
-class Translator : public Input {
+/*
+ Each translator class has a static list of listener records. Each
+ record makes one explains how to register one of the class's stream event
+ listeners to a context.
+*/
+typedef struct translator_listener_record
+{
+ Listener (*get_listener_) (void *, SCM event_class);
+ SCM event_class_;
+ struct translator_listener_record *next_;
+
+ translator_listener_record ()
+ {
+ next_ = 0;
+ event_class_ = SCM_EOL;
+ get_listener_ = 0;
+ }
+
+} translator_listener_record;
+
+#define TRANSLATOR_DECLARATIONS_NO_LISTENER(NAME) \
+private: \
+ public: \
+ NAME (); \
+ VIRTUAL_COPY_CONSTRUCTOR (Translator, NAME); \
+ static SCM static_description_; \
+ static Drul_array<vector<Acknowledge_information> > acknowledge_static_array_drul_; \
+ virtual void fetch_precomputable_methods (Translator_void_method_ptr methods[]); \
+ virtual SCM static_translator_description () const; \
+ virtual SCM translator_description () const; \
+ static Engraver_void_function_engraver_grob_info static_get_acknowledger (SCM sym); \
+ static Engraver_void_function_engraver_grob_info static_get_end_acknowledger(SCM); \
+ virtual Engraver_void_function_engraver_grob_info get_acknowledger (SCM sym) \
+ { \
+ return static_get_acknowledger (sym); \
+ } \
+ virtual Engraver_void_function_engraver_grob_info get_end_acknowledger (SCM sym) \
+ { \
+ return static_get_end_acknowledger (sym); \
+ } \
+ /* end #define */
+
+#define TRANSLATOR_DECLARATIONS(NAME) \
+ TRANSLATOR_DECLARATIONS_NO_LISTENER(NAME) \
+private: \
+ static translator_listener_record *listener_list_; \
+public: \
+ virtual translator_listener_record *get_listener_list () const \
+ { \
+ return listener_list_; \
+ } \
+ /* end #define */
+
+#define DECLARE_TRANSLATOR_LISTENER(m) \
+public: \
+inline void listen_ ## m (Stream_event *); \
+/* Should be private */ \
+static void _internal_declare_ ## m (); \
+private: \
+ static Listener _get_ ## m ## _listener (void *, SCM); \
+DECLARE_LISTENER (_listen_scm_ ## m);
+
+#define DECLARE_ACKNOWLEDGER(x) public : void acknowledge_ ## x (Grob_info); protected:
+#define DECLARE_END_ACKNOWLEDGER(x) public : void acknowledge_end_ ## x (Grob_info); protected:
+
+enum Translator_precompute_index
+{
+ START_TRANSLATION_TIMESTEP,
+ STOP_TRANSLATION_TIMESTEP,
+ PROCESS_MUSIC,
+ PROCESS_ACKNOWLEDGED,
+ TRANSLATOR_METHOD_PRECOMPUTE_COUNT,
+};
+
+/*
+ Translate music into grobs.
+*/
+class Translator : public Smob<Translator>
+{
+public:
+ int print_smob (SCM, scm_print_state *);
+ SCM mark_smob ();
+ static const char type_p_name_[];
+ virtual ~Translator ();
+private:
void init ();
+
public:
- Music_output_def * output_def_l_;
- String type_str_;
-
- virtual const char *name() const;
- bool is_alias_b (String) const;
-
- VIRTUAL_COPY_CONS(Translator);
+ Context *context () const { return daddy_context_; }
+
Translator (Translator const &);
- Translator ();
-
- Translator_group * daddy_trans_l_ ;
-
-
- void pre_move_processing();
- void announces();
- void post_move_processing();
- void removal_processing();
- /**
- ask daddy for a feature
- */
- Music_output_def *output_def_l () const;
-
- SCM get_property (const char *) const;
- SCM get_property (SCM symbol) const;
-
- virtual Moment now_mom () const;
-
- /*
- ugh: bubbled up from Translator_group.
- */
- SCM simple_trans_list_;
- SCM trans_group_list_;
- SCM definition_;
-
- SCM properties_scm_;
- DECLARE_SMOBS(Translator, dummy);
-public:
- /**
- try to fit the request in this engraver
+ SCM internal_get_property (SCM symbol) const;
- @return
- false: not noted, not taken.
+ virtual Output_def *get_output_def () const;
+ virtual Translator_group *get_daddy_translator ()const;
+ virtual Moment now_mom () const;
+ virtual bool must_be_last () const;
- true: request swallowed. Don't try to put the request elsewhere.
+ virtual void initialize ();
+ virtual void finalize ();
- */
- virtual bool try_music (Music *req_l);
- virtual void stop_translation_timestep();
- virtual void start_translation_timestep();
- virtual void do_announces () ;
- virtual void initialize() ;
- virtual void finalize();
-};
+ virtual void connect_to_context (Context *c);
+ virtual void disconnect_from_context (Context *c);
+
+ void stop_translation_timestep ();
+ void start_translation_timestep ();
+ void process_music ();
+ void process_acknowledged ();
+ Context *get_score_context () const;
+ Global_context *get_global_context () const;
-/**
- A macro to automate administration of translators.
- */
-#define ADD_THIS_TRANSLATOR(T) \
-static void _ ## T ## _adder () {\
- T *t = new T;\
- t->type_str_ = classname (t);\
- add_translator (t);\
-}\
-ADD_GLOBAL_CTOR(_ ## T ## _adder);
+ TRANSLATOR_DECLARATIONS (Translator);
+protected: // should be private.
+ Context *daddy_context_;
+ void protect_event (SCM ev);
+ virtual void derived_mark () const;
+ static void add_translator_listener (translator_listener_record **listener_list,
+ translator_listener_record *r,
+ Listener (*get_listener) (void *, SCM),
+ const char *ev_class);
+ SCM static_translator_description (const char *grobs,
+ const char *desc,
+ translator_listener_record *listener_list,
+ const char *read,
+ const char *write) const;
+ friend class Translator_group;
+};
+
+void add_translator (Translator *trans);
+
+Translator *get_translator (SCM s);
+Moment get_event_length (Stream_event *s, Moment now);
+Moment get_event_length (Stream_event *s);
-extern Dictionary<Translator*> *global_translator_dict_p;
-void add_translator (Translator*trans_p);
+/*
+ This helper is only meaningful inside listen_* methods.
+*/
+extern bool internal_event_assignment (Stream_event **old_ev, Stream_event *new_ev, const char *function);
+#define ASSIGN_EVENT_ONCE(o,n) internal_event_assignment (&o, n, __FUNCTION__)
-Translator*get_translator_l (String s);
-Translator *unsmob_translator (SCM);
#endif // TRANSLATOR_HH