lilypond-1.3.9
[lilypond.git] / flower / dstream.cc
blobe51f97a39e1ef5ea2f802d57f7f17c91f0f77e57
1 /*
2 dstream.cc -- implement Dstream
4 source file of the Flower Library
6 (c) 1996, 1997--1999 Han-Wen Nienhuys <hanwen@cs.uu.nl>
7 */
9 #include <fstream.h>
10 #include "dictionary-iter.hh"
11 #include "dstream.hh"
13 #include "text-db.hh"
14 #include "string-convert.hh"
15 #include "rational.hh"
17 /// amount of indentation for each level.
18 const int INDTAB = 2;
21 should use Regexp library.
23 static String
24 strip_pretty (String pretty_str)
26 int i = pretty_str.index_i ('(');
27 if (i>=0)
28 pretty_str = pretty_str.left_str (i);
30 int l = pretty_str.index_last_i (' '); // strip until last ' '
31 if (l>=0)
32 pretty_str = pretty_str.nomid_str (0,l+1);
33 return pretty_str;
36 static String
37 strip_member (String pret)
39 int l=pret.index_last_i (':')-1;
40 if (l>=0)
41 pret = pret.left_str (l);
42 return pret;
45 Dstream&
46 Dstream::identify_as (String name)
48 if (!os_l_)
49 return *this;
51 String mem (strip_pretty (name));
52 String cl (strip_member (mem));
53 String idx = cl;
55 if (silent_dict_p_->elem_b (mem))
56 idx = mem;
57 else if (silent_dict_p_->elem_b (cl))
58 idx = cl;
59 else
61 (*silent_dict_p_)[idx] = default_silence_b_;
63 local_silence_b_ = (*silent_dict_p_)[idx];
64 if (current_classname_str_ != idx && !local_silence_b_)
66 current_classname_str_=idx;
67 if (!(*silent_dict_p_)["Dstream"])
68 *os_l_ << "[" << current_classname_str_ << ":]"; // messy.
70 return *this;
73 bool
74 Dstream::silent_b (String s) const
76 if (!silent_dict_p_->elem_b (s))
77 return false;
78 return (*silent_dict_p_)[s];
81 Dstream &
82 Dstream::operator<<(void const *v_l)
84 output (String_convert::pointer_str (v_l));
85 return *this;
88 Dstream &
89 Dstream::operator <<(String s)
91 output (s);
92 return *this;
95 Dstream &
96 Dstream::operator <<(const char * s)
98 output (String (s));
99 return *this;
102 Dstream &
103 Dstream::operator <<(char c)
105 output (to_str (c));
106 return *this;
109 Dstream&
110 Dstream::operator << (Real r)
112 output (to_str (r));
113 return *this;
115 Dstream &
116 Dstream::operator <<(Rational c)
118 output (c.str ());
119 return *this;
121 Dstream &
122 Dstream::operator <<(int i)
124 output (to_str(i));
125 return *this;
128 void
129 Dstream::output (String s)
131 if (local_silence_b_|| !os_l_)
132 return ;
134 for (char const *cp = s.ch_C (); *cp; cp++)
135 switch (*cp)
137 case '{':
138 case '[':
139 case '(': indent_level_i_ += INDTAB;
140 *os_l_ << *cp;
141 break;
143 case ')':
144 case ']':
145 case '}':
146 indent_level_i_ -= INDTAB;
147 *os_l_ << *cp ;
149 assert (indent_level_i_>=0) ;
150 break;
152 case '\n':
153 *os_l_ << '\n' << to_str (' ', indent_level_i_) << flush;
154 break;
155 default:
156 *os_l_ << *cp;
157 break;
159 return ;
163 Dstream::Dstream (ostream *r, char const * cfg_nm)
165 os_l_ = r;
166 silent_dict_p_ = new Dictionary<bool>;
167 default_silence_b_ = false;
168 indent_level_i_ = 0;
169 if (!os_l_)
170 return;
172 char const * fn =cfg_nm ? cfg_nm : ".dstreamrc";
174 ifstream ifs (fn); // can 't open
175 if (!ifs)
176 return;
179 Text_db cfg (fn);
180 while (!cfg.eof_b ()){
181 Text_record r (cfg++);
182 if (r.size() != 2)
184 r.message (_ ("not enough fields in Dstream init"));
185 continue;
187 (*silent_dict_p_)[r[0]] = r[1] == "1";
190 if ((*silent_dict_p_).elem_b ("Dstream_default_silence"))
191 default_silence_b_ = (*silent_dict_p_)["Dstream_default_silence"];
195 Dstream::~Dstream()
197 delete silent_dict_p_;
198 assert (!indent_level_i_) ;
201 void
202 Dstream::clear_silence()
204 for (Dictionary_iter<bool> i (*silent_dict_p_); i.ok(); i++)
206 i.val_ref() = false;