]> git.lyx.org Git - lyx.git/blob - src/mathed/MathStream.cpp
Special handing of space characters during text output.
[lyx.git] / src / mathed / MathStream.cpp
1 /**
2  * \file MathStream.cpp
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author André Pönitz
7  *
8  * Full author contact details are available in file CREDITS.
9  */
10
11 #include <config.h>
12
13 #include "MathStream.h"
14
15 #include "MathData.h"
16 #include "MathExtern.h"
17
18 #include "support/textutils.h"
19 #include "support/docstring.h"
20
21 #include <algorithm>
22 #include <cstring>
23 #include <ostream>
24
25 using namespace std;
26
27 namespace lyx {
28
29
30 //////////////////////////////////////////////////////////////////////
31
32
33 NormalStream & operator<<(NormalStream & ns, MathAtom const & at)
34 {
35         at->normalize(ns);
36         return ns;
37 }
38
39
40 NormalStream & operator<<(NormalStream & ns, MathData const & ar)
41 {
42         normalize(ar, ns);
43         return ns;
44 }
45
46
47 NormalStream & operator<<(NormalStream & ns, docstring const & s)
48 {
49         ns.os() << s;
50         return ns;
51 }
52
53
54 NormalStream & operator<<(NormalStream & ns, const string & s)
55 {
56         ns.os() << from_utf8(s);
57         return ns;
58 }
59
60
61 NormalStream & operator<<(NormalStream & ns, char const * s)
62 {
63         ns.os() << s;
64         return ns;
65 }
66
67
68 NormalStream & operator<<(NormalStream & ns, char c)
69 {
70         ns.os() << c;
71         return ns;
72 }
73
74
75 NormalStream & operator<<(NormalStream & ns, int i)
76 {
77         ns.os() << i;
78         return ns;
79 }
80
81
82
83 /////////////////////////////////////////////////////////////////
84
85
86 WriteStream & operator<<(WriteStream & ws, docstring const & s)
87 {
88         if (ws.pendingBrace()) {
89                 ws.os() << '}';
90                 ws.pendingBrace(false);
91                 ws.pendingSpace(false);
92                 ws.textMode(true);
93         } else if (ws.pendingSpace() && s.length() > 0) {
94                 if (isAlphaASCII(s[0]))
95                         ws.os() << ' ';
96                 else if (s[0] == ' ' && ws.textMode())
97                         ws.os() << '\\';
98                 ws.pendingSpace(false);
99         }
100         ws.os() << s;
101         int lf = 0;
102         docstring::const_iterator dit = s.begin();
103         docstring::const_iterator end = s.end();
104         for (; dit != end; ++dit)
105                 if ((*dit) == '\n')
106                         ++lf;
107         ws.addlines(lf);
108         return ws;
109 }
110
111
112 WriteStream::WriteStream(odocstream & os, bool fragile, bool latex, OutputType output,
113                         Encoding const * encoding)
114         : os_(os), fragile_(fragile), firstitem_(false), latex_(latex),
115           output_(output), pendingspace_(false), pendingbrace_(false),
116           textmode_(false), locked_(0), line_(0), encoding_(encoding)
117 {}
118
119
120 WriteStream::WriteStream(odocstream & os)
121         : os_(os), fragile_(false), firstitem_(false), latex_(false),
122           output_(wsDefault), pendingspace_(false), pendingbrace_(false),
123           textmode_(false), locked_(0), line_(0), encoding_(0)
124 {}
125
126
127 WriteStream::~WriteStream()
128 {
129         if (pendingbrace_)
130                 os_ << '}';
131         else if (pendingspace_)
132                 os_ << ' ';
133 }
134
135
136 void WriteStream::addlines(unsigned int n)
137 {
138         line_ += n;
139 }
140
141
142 void WriteStream::pendingSpace(bool how)
143 {
144         pendingspace_ = how;
145 }
146
147
148 void WriteStream::pendingBrace(bool brace)
149 {
150         pendingbrace_ = brace;
151 }
152
153
154 void WriteStream::textMode(bool textmode)
155 {
156         textmode_ = textmode;
157 }
158
159
160 void WriteStream::lockedMode(bool locked)
161 {
162         locked_ = locked;
163 }
164
165
166 WriteStream & operator<<(WriteStream & ws, MathAtom const & at)
167 {
168         at->write(ws);
169         return ws;
170 }
171
172
173 WriteStream & operator<<(WriteStream & ws, MathData const & ar)
174 {
175         write(ar, ws);
176         return ws;
177 }
178
179
180 WriteStream & operator<<(WriteStream & ws, char const * s)
181 {
182         if (ws.pendingBrace()) {
183                 ws.os() << '}';
184                 ws.pendingBrace(false);
185                 ws.pendingSpace(false);
186                 ws.textMode(true);
187         } else if (ws.pendingSpace() && strlen(s) > 0) {
188                 if (isAlphaASCII(s[0]))
189                         ws.os() << ' ';
190                 else if (s[0] == ' ' && ws.textMode())
191                         ws.os() << '\\';
192                 ws.pendingSpace(false);
193         }
194         ws.os() << s;
195         ws.addlines(int(count(s, s + strlen(s), '\n')));
196         return ws;
197 }
198
199
200 WriteStream & operator<<(WriteStream & ws, char c)
201 {
202         if (ws.pendingBrace()) {
203                 ws.os() << '}';
204                 ws.pendingBrace(false);
205                 ws.pendingSpace(false);
206                 ws.textMode(true);
207         } else if (ws.pendingSpace()) {
208                 if (isAlphaASCII(c))
209                         ws.os() << ' ';
210                 else if (c == ' ' && ws.textMode())
211                         ws.os() << '\\';
212                 ws.pendingSpace(false);
213         }
214         ws.os() << c;
215         if (c == '\n')
216                 ws.addlines(1);
217         return ws;
218 }
219
220
221 WriteStream & operator<<(WriteStream & ws, int i)
222 {
223         if (ws.pendingBrace()) {
224                 ws.os() << '}';
225                 ws.pendingBrace(false);
226                 ws.textMode(true);
227         }
228         ws.os() << i;
229         return ws;
230 }
231
232
233 WriteStream & operator<<(WriteStream & ws, unsigned int i)
234 {
235         if (ws.pendingBrace()) {
236                 ws.os() << '}';
237                 ws.pendingBrace(false);
238                 ws.textMode(true);
239         }
240         ws.os() << i;
241         return ws;
242 }
243
244
245 //////////////////////////////////////////////////////////////////////
246
247
248 MathStream::MathStream(odocstream & os)
249         : os_(os), tab_(0), line_(0), lastchar_(0), in_text_(false)
250 {}
251
252
253 void MathStream::cr()
254 {
255         os() << '\n';
256         for (int i = 0; i < tab(); ++i)
257                 os() << ' ';
258 }
259
260
261 void MathStream::defer(docstring const & s)
262 {
263         deferred_ << s;
264 }
265
266
267 void MathStream::defer(string const & s)
268 {
269         deferred_ << from_utf8(s);
270 }
271
272
273 docstring MathStream::deferred() const
274
275         return deferred_.str();
276 }
277
278
279 MathStream & operator<<(MathStream & ms, MathAtom const & at)
280 {
281         at->mathmlize(ms);
282         return ms;
283 }
284
285
286 MathStream & operator<<(MathStream & ms, MathData const & ar)
287 {
288         mathmlize(ar, ms);
289         return ms;
290 }
291
292
293 MathStream & operator<<(MathStream & ms, char const * s)
294 {
295         ms.os() << s;
296         return ms;
297 }
298
299
300 MathStream & operator<<(MathStream & ms, char c)
301 {
302         ms.os() << c;
303         return ms;
304 }
305
306
307 MathStream & operator<<(MathStream & ms, char_type c)
308 {
309         ms.os() << c;
310         return ms;
311 }
312
313
314 MathStream & operator<<(MathStream & ms, MTag const & t)
315 {
316         ++ms.tab();
317         ms.cr();
318         ms.os() << '<' << from_ascii(t.tag_) << '>';
319         return ms;
320 }
321
322
323 MathStream & operator<<(MathStream & ms, ETag const & t)
324 {
325         ms.cr();
326         if (ms.tab() > 0)
327                 --ms.tab();
328         ms.os() << "</" << from_ascii(t.tag_) << '>';
329         return ms;
330 }
331
332
333 MathStream & operator<<(MathStream & ms, docstring const & s)
334 {
335         ms.os() << s;
336         return ms;
337 }
338
339
340 SetMode::SetMode(MathStream & os, bool text, docstring attrs)
341                 : os_(os)
342 {
343         was_text_ = os.inText();
344         if (was_text_)
345                 os << "</mtext>";
346         if (text) {
347                 os.setTextMode();
348                 os << "<mtext";
349                 if (!attrs.empty())
350                         os << " " << attrs;
351                 os << ">";
352         } else {
353                 if (!attrs.empty())
354                         os << "<mstyle " << attrs << ">";
355                 os.setMathMode();
356         }
357 }
358
359
360 SetMode::~SetMode()
361 {
362         if (os_.inText())
363                 os_ << "</mtext>";
364         if (was_text_) {
365                 os_.setTextMode();
366                 os_ << "<mtext>";
367         } else {
368                 os_.setMathMode();
369         }
370 }
371
372
373 //////////////////////////////////////////////////////////////////////
374
375
376 MapleStream & operator<<(MapleStream & ms, MathAtom const & at)
377 {
378         at->maple(ms);
379         return ms;
380 }
381
382
383 MapleStream & operator<<(MapleStream & ms, MathData const & ar)
384 {
385         maple(ar, ms);
386         return ms;
387 }
388
389
390 MapleStream & operator<<(MapleStream & ms, char const * s)
391 {
392         ms.os() << s;
393         return ms;
394 }
395
396
397 MapleStream & operator<<(MapleStream & ms, char c)
398 {
399         ms.os() << c;
400         return ms;
401 }
402
403
404 MapleStream & operator<<(MapleStream & ms, int i)
405 {
406         ms.os() << i;
407         return ms;
408 }
409
410
411 MapleStream & operator<<(MapleStream & ms, char_type c)
412 {
413         ms.os().put(c);
414         return ms;
415 }
416
417
418 MapleStream & operator<<(MapleStream & ms, docstring const & s)
419 {
420         ms.os() << s;
421         return ms;
422 }
423
424
425 //////////////////////////////////////////////////////////////////////
426
427
428 MaximaStream & operator<<(MaximaStream & ms, MathAtom const & at)
429 {
430         at->maxima(ms);
431         return ms;
432 }
433
434
435 MaximaStream & operator<<(MaximaStream & ms, MathData const & ar)
436 {
437         maxima(ar, ms);
438         return ms;
439 }
440
441
442 MaximaStream & operator<<(MaximaStream & ms, char const * s)
443 {
444         ms.os() << s;
445         return ms;
446 }
447
448
449 MaximaStream & operator<<(MaximaStream & ms, char c)
450 {
451         ms.os() << c;
452         return ms;
453 }
454
455
456 MaximaStream & operator<<(MaximaStream & ms, int i)
457 {
458         ms.os() << i;
459         return ms;
460 }
461
462
463 MaximaStream & operator<<(MaximaStream & ms, docstring const & s)
464 {
465         ms.os() << s;
466         return ms;
467 }
468
469
470 MaximaStream & operator<<(MaximaStream & ms, char_type c)
471 {
472         ms.os().put(c);
473         return ms;
474 }
475
476
477 //////////////////////////////////////////////////////////////////////
478
479
480 MathematicaStream & operator<<(MathematicaStream & ms, MathAtom const & at)
481 {
482         at->mathematica(ms);
483         return ms;
484 }
485
486
487 MathematicaStream & operator<<(MathematicaStream & ms, MathData const & ar)
488 {
489         mathematica(ar, ms);
490         return ms;
491 }
492
493
494 MathematicaStream & operator<<(MathematicaStream & ms, char const * s)
495 {
496         ms.os() << s;
497         return ms;
498 }
499
500
501 MathematicaStream & operator<<(MathematicaStream & ms, char c)
502 {
503         ms.os() << c;
504         return ms;
505 }
506
507
508 MathematicaStream & operator<<(MathematicaStream & ms, int i)
509 {
510         ms.os() << i;
511         return ms;
512 }
513
514
515 MathematicaStream & operator<<(MathematicaStream & ms, docstring const & s)
516 {
517         ms.os() << s;
518         return ms;
519 }
520
521
522 MathematicaStream & operator<<(MathematicaStream & ms, char_type c)
523 {
524         ms.os().put(c);
525         return ms;
526 }
527
528
529 //////////////////////////////////////////////////////////////////////
530
531
532 OctaveStream & operator<<(OctaveStream & ns, MathAtom const & at)
533 {
534         at->octave(ns);
535         return ns;
536 }
537
538
539 OctaveStream & operator<<(OctaveStream & ns, MathData const & ar)
540 {
541         octave(ar, ns);
542         return ns;
543 }
544
545
546 OctaveStream & operator<<(OctaveStream & ns, char const * s)
547 {
548         ns.os() << s;
549         return ns;
550 }
551
552
553 OctaveStream & operator<<(OctaveStream & ns, char c)
554 {
555         ns.os() << c;
556         return ns;
557 }
558
559
560 OctaveStream & operator<<(OctaveStream & ns, int i)
561 {
562         ns.os() << i;
563         return ns;
564 }
565
566
567 OctaveStream & operator<<(OctaveStream & ns, docstring const & s)
568 {
569         ns.os() << s;
570         return ns;
571 }
572
573
574 OctaveStream & operator<<(OctaveStream & ns, char_type c)
575 {
576         ns.os().put(c);
577         return ns;
578 }
579
580
581 OctaveStream & operator<<(OctaveStream & os, string const & s)
582 {
583         os.os() << from_utf8(s);
584         return os;
585 }
586
587
588 } // namespace lyx