lilypond-1.3.0
[lilypond.git] / flower / dstream.cc
blobba46d69ed6851890f0601eaa9e80011e16fd283b
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"
12 #include "scalar.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 <<(Scalar s)
91 output (s);
92 return *this;
95 Dstream &
96 Dstream::operator <<(const char * s)
98 output (String (s));
99 return *this;
102 void
103 Dstream::output (String s)
105 if (local_silence_b_|| !os_l_)
106 return ;
108 for (char const *cp = s.ch_C (); *cp; cp++)
109 switch (*cp)
111 case '{':
112 case '[':
113 case '(': indent_level_i_ += INDTAB;
114 *os_l_ << *cp;
115 break;
117 case ')':
118 case ']':
119 case '}':
120 indent_level_i_ -= INDTAB;
121 *os_l_ << *cp ;
123 assert (indent_level_i_>=0) ;
124 break;
126 case '\n':
127 *os_l_ << '\n' << to_str (' ', indent_level_i_) << flush;
128 break;
129 default:
130 *os_l_ << *cp;
131 break;
133 return ;
137 Dstream::Dstream (ostream *r, char const * cfg_nm)
139 os_l_ = r;
140 silent_dict_p_ = new Dictionary<bool>;
141 default_silence_b_ = false;
142 indent_level_i_ = 0;
143 if (!os_l_)
144 return;
146 char const * fn =cfg_nm ? cfg_nm : ".dstreamrc";
148 ifstream ifs (fn); // can 't open
149 if (!ifs)
150 return;
153 Text_db cfg (fn);
154 while (!cfg.eof_b ()){
155 Text_record r (cfg++);
156 if (r.size() != 2)
158 r.message (_ ("not enough fields in Dstream init"));
159 continue;
161 (*silent_dict_p_)[r[0]] = (bool)(int)(Scalar (r[1]));
164 if ((*silent_dict_p_).elem_b ("Dstream_default_silence"))
165 default_silence_b_ = (*silent_dict_p_)["Dstream_default_silence"];
169 Dstream::~Dstream()
171 delete silent_dict_p_;
172 assert (!indent_level_i_) ;
175 void
176 Dstream::clear_silence()
178 for (Dictionary_iter<bool> i (*silent_dict_p_); i.ok(); i++)
180 i.val_ref() = false;