]> git.donarmstrong.com Git - lilypond.git/blob - lily/span-bar.cc
release: 1.3.83
[lilypond.git] / lily / span-bar.cc
1 /*
2   span-bar.cc -- implement Span_bar
3
4   source file of the GNU LilyPond music typesetter
5
6   (c)  1997--2000 Han-Wen Nienhuys <hanwen@cs.uu.nl>
7 */
8
9 #include "span-bar.hh"
10 #include "lookup.hh"
11 #include "dimensions.hh"
12 #include "paper-def.hh"
13 #include "molecule.hh"
14 #include "warn.hh"
15 #include "axis-group-interface.hh"
16 #include "group-interface.hh"
17 #include "score-element.hh"
18 #include "bar.hh"
19
20 void
21 Span_bar::add_bar (Score_element*me, Score_element*b)
22 {
23   Pointer_group_interface gi (me);
24   gi.add_element (b);
25
26   me->add_dependency (b);
27 }
28
29 Interval
30 Span_bar::width_callback (Score_element *se, Axis )
31 {
32   String gl = ly_scm2string (se->get_elt_property ("glyph"));
33
34   /*
35     urg.
36    */
37   Molecule m = Bar::compound_barline (se, gl, 40 PT);
38   
39   return m.extent (X_AXIS);
40 }
41
42 MAKE_SCHEME_CALLBACK(Span_bar,before_line_breaking);
43 SCM
44 Span_bar::before_line_breaking (SCM smob)
45 {
46   evaluate_empty (unsmob_element (smob));
47   evaluate_glyph (unsmob_element (smob));
48
49   /*
50     no need to call   Bar::before_line_breaking (), because the info
51     in ELEMENTS already has been procced by Bar::before_line_breaking().
52    */
53   return SCM_UNSPECIFIED;
54 }
55
56 Real
57 Span_bar::center_on_spanned_callback (Score_element * me, Axis a)
58 {
59   assert (a == Y_AXIS);
60   Interval i (get_spanned_interval (me));
61
62   /*
63     Bar::brew_molecule delivers a barline of y-extent (-h/2,h/2), so
64     we have to translate ourselves to be in the center of the 
65     interval that we span.  */
66
67   return i.center ();
68 }
69
70 void
71 Span_bar::evaluate_empty (Score_element*me)
72 {
73   /*
74     TODO: filter all hara-kiried out of ELEMENS list, and then
75     optionally do suicide. Call this cleanage function from
76     center_on_spanned_callback() as well.
77     
78    */
79   if (!gh_pair_p (me->get_elt_property ("elements")))
80     {
81       me->suicide ();
82     }
83 }
84
85 void
86 Span_bar::evaluate_glyph (Score_element*me)
87 {
88   SCM elts = me->get_elt_property ("elements");
89   Score_element * b = unsmob_element (gh_car (elts));
90   SCM glsym =ly_symbol2scm ("glyph");
91   SCM gl =b ->get_elt_property (glsym);
92   if (!gh_string_p (gl))
93     {
94       me->suicide ();
95       return ; 
96     }
97
98   String type = ly_scm2string (gl);
99   
100   if (type == "|:") 
101     {
102       type = ".|";
103     }
104   else if (type== ":|")
105     {
106       type = "|.";
107     }
108   else if (type== ":|:")
109     {
110       type = ".|.";
111     }
112
113   gl = ly_str02scm (type.ch_C());
114   if (scm_equal_p (me->get_elt_property (glsym), gl) != SCM_BOOL_T)
115     me->set_elt_property (glsym, gl);
116 }
117
118 Interval
119 Span_bar::get_spanned_interval (Score_element*me) 
120 {
121   return Axis_group_interface::group_extent_callback (me, Y_AXIS);  
122 }
123
124
125 MAKE_SCHEME_CALLBACK(Span_bar,get_bar_size);
126 SCM
127 Span_bar::get_bar_size (SCM smob)
128 {
129   Score_element* me =  unsmob_element (smob);
130   Interval iv (get_spanned_interval (me));
131   if (iv.empty_b ())
132     {
133       /*
134         This happens if the bars are hara-kiried from under us.
135        */
136       me->suicide ();
137       return gh_double2scm (-1);
138     }
139   return gh_double2scm (iv.length ());
140 }
141 void
142 Span_bar::set_interface (Score_element *me)
143 {
144   Bar::set_interface (me);
145   
146   Pointer_group_interface(me).set_interface ();
147   me->set_extent_callback (width_callback, X_AXIS);
148   me->add_offset_callback (center_on_spanned_callback, Y_AXIS);
149   me->set_interface (ly_symbol2scm ("span-bar-interface"));
150   me->set_extent_callback (0, Y_AXIS);
151 }
152
153 bool
154 Span_bar::has_interface (Score_element*m)
155 {
156   return m && m->has_interface (ly_symbol2scm ("span-bar-interface"));
157 }