]> git.donarmstrong.com Git - lilypond.git/blob - lily/include/translator.icc
Issue 4842/4: Replace Translator_method et al
[lilypond.git] / lily / include / translator.icc
1 /* -*- c++ -*-
2   This file is part of LilyPond, the GNU music typesetter.
3
4   Copyright (C) 2005--2015 Han-Wen Nienhuys <hanwen@xs4all.nl>
5
6   LilyPond is free software: you can redistribute it and/or modify
7   it under the terms of the GNU General Public License as published by
8   the Free Software Foundation, either version 3 of the License, or
9   (at your option) any later version.
10
11   LilyPond is distributed in the hope that it will be useful,
12   but WITHOUT ANY WARRANTY; without even the implied warranty of
13   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14   GNU General Public License for more details.
15
16   You should have received a copy of the GNU General Public License
17   along with LilyPond.  If not, see <http://www.gnu.org/licenses/>.
18 */
19
20 #ifndef TRANSLATOR_ICC
21 #define TRANSLATOR_ICC
22
23 #include "callback.hh"
24 #include "std-vector.hh"
25 #include "engraver.hh"
26
27 /*
28   TODO: derive "foo-bar-interface" from Foo_bar classname.
29  */
30
31 /**
32    A macro to automate administration of translators.
33 */
34 #define ADD_THIS_TRANSLATOR(T)                                          \
35   SCM T::static_description_ = SCM_EOL;                                 \
36   static void _ ## T ## _adder ()                                       \
37   {                                                                     \
38     T *t = new T;                                                       \
39     T::static_description_ =                                            \
40       scm_permanent_object (t->static_translator_description ());       \
41     add_translator (t);                                                 \
42   }                                                                     \
43   SCM T::translator_description () const                                \
44   {                                                                     \
45     return static_description_;                                         \
46   }                                                                     \
47   ADD_GLOBAL_CTOR (_ ## T ## _adder); \
48   /* end define */
49
50 #define DEFINE_TRANSLATOR_LISTENER_LIST(T) \
51   Protected_scm T::listener_list_ (SCM_EOL);    \
52   /* end define */
53
54 #define DEFINE_ACKNOWLEDGERS(classname) \
55   Drul_array< vector<Acknowledge_information> > classname::acknowledge_static_array_drul_;      \
56   SCM                                                                   \
57   classname::static_get_acknowledger (SCM sym)                          \
58   {                                                                     \
59     return generic_get_acknowledger (sym, &acknowledge_static_array_drul_[START]);      \
60   }                                                                     \
61   SCM                                                                   \
62   classname::static_get_end_acknowledger (SCM sym)                      \
63   {                                                                     \
64     return generic_get_acknowledger (sym, &acknowledge_static_array_drul_[STOP]);       \
65   }                                                                     \
66   /* end define */
67
68 #define DEFINE_TRANSLATOR_DOC(classname, desc, grobs, read, write)              \
69   SCM                                                                   \
70   classname::static_translator_description () const                     \
71   {                                                                     \
72     return Translator::static_translator_description (grobs, desc, listener_list_, read, write); \
73   }
74
75 #define ADD_TRANSLATOR_FAMILY(classname)                                \
76   IMPLEMENT_FETCH_PRECOMPUTABLE_METHODS (classname);                    \
77   DEFINE_ACKNOWLEDGERS(classname)                                       \
78
79 #define ADD_TRANSLATOR(classname, desc, grobs, read, write)             \
80   ADD_TRANSLATOR_FAMILY (classname);                                    \
81   ADD_THIS_TRANSLATOR (classname);                                      \
82   DEFINE_TRANSLATOR_DOC(classname, desc, grobs, read, write)            \
83   DEFINE_TRANSLATOR_LISTENER_LIST(classname)                            \
84
85 #define IMPLEMENT_FETCH_PRECOMPUTABLE_METHODS(T)                        \
86   void                                                                  \
87   T::fetch_precomputable_methods (SCM ptrs[])                           \
88   {                                                                     \
89     ptrs[START_TRANSLATION_TIMESTEP] =                                  \
90       method_finder <&T::start_translation_timestep> ();                \
91                                                                         \
92     ptrs[STOP_TRANSLATION_TIMESTEP] =                                   \
93       method_finder <&T::stop_translation_timestep> ();                 \
94                                                                         \
95     ptrs[PROCESS_MUSIC] =                                               \
96       method_finder <&T::process_music> ();                             \
97                                                                         \
98     ptrs[PROCESS_ACKNOWLEDGED] =                                        \
99       method_finder <&T::process_acknowledged> ();                      \
100   }
101
102 void add_acknowledger (SCM ptr,
103                        char const *func_name,
104                        vector<Acknowledge_information> *ack_array);
105
106 SCM
107 generic_get_acknowledger (SCM sym,
108                           vector<Acknowledge_information> const *ack_array);
109
110 #define ADD_ACKNOWLEDGER(CLASS, NAME)                                   \
111   void CLASS ## NAME ## _ack_adder ()                                   \
112   {                                                                     \
113     add_acknowledger (CLASS::ack_finder<&CLASS::acknowledge_ ## NAME> (), \
114                       #NAME, &CLASS::acknowledge_static_array_drul_[START]); \
115   }                                                                     \
116   ADD_SCM_INIT_FUNC (CLASS ## NAME ## _ack_adder_initclass, CLASS ## NAME ## _ack_adder);
117
118 #define ADD_END_ACKNOWLEDGER(CLASS, NAME)                               \
119   void CLASS ## NAME ## _end_ack_adder ()                               \
120   {                                                                     \
121     add_acknowledger (CLASS::ack_finder<&CLASS::acknowledge_end_ ## NAME> (), \
122                       #NAME, &CLASS::acknowledge_static_array_drul_[STOP]); \
123   }                                                                     \
124   ADD_SCM_INIT_FUNC (CLASS ## NAME ## _end_ack_adder_initclass, CLASS ## NAME ## _end_ack_adder);
125
126 /*
127   Implement the method cl::listen_##m, and make it listen to stream
128   events of class m.
129  */
130 #define IMPLEMENT_TRANSLATOR_LISTENER(cl, m)            \
131 void                                                    \
132 cl :: _internal_declare_ ## m ()                        \
133 {                                                       \
134   listener_list_ = scm_acons                                            \
135     (event_class_symbol (#m),                                           \
136      Callback_wrapper::make_smob                                        \
137      <trampoline <cl, &cl::listen_ ## m> > (), listener_list_);         \
138 }                                                                       \
139                                                                         \
140 ADD_SCM_INIT_FUNC (cl ## _declare_event_ ## m, cl::_internal_declare_ ## m);
141
142 #endif /* TRANSLATOR_ICC */