xref: /haiku/src/apps/terminal/TerminalLine.h (revision e1c4049fed1047bdb957b0529e1921e97ef94770)
1 /*
2  * Copyright 2013, Haiku, Inc. All rights reserved.
3  * Copyright 2008, Ingo Weinhold, ingo_weinhold@gmx.de.
4  * Distributed under the terms of the MIT License.
5  *
6  * Authors:
7  *		Ingo Weinhold, ingo_weinhold@gmx.de
8  *		Siarzhuk Zharski, zharik@gmx.li
9  */
10 #ifndef TERMINAL_LINE_H
11 #define TERMINAL_LINE_H
12 
13 #include <GraphicsDefs.h>
14 #include <SupportDefs.h>
15 
16 #include "TermConst.h"
17 
18 #include "UTF8Char.h"
19 
20 
21 struct Attributes {
22 	uint32 state;
23 	uint32 foreground;
24 	uint32 background;
25 	uint32 underline;
26 	int underlineStyle;
27 
28 	Attributes() : state(0), foreground(0), background(0), underline(0), underlineStyle(0) {}
29 
30 	inline void Reset()
31 	{
32 		state = 0;
33 		foreground = 0;
34 		background = 0;
35 		underline = 0;
36 		underlineStyle = 0;
37 	}
38 
39 	inline bool IsWidth() const { return (state & A_WIDTH) == A_WIDTH; }
40 	inline bool IsBold() const { return (state & BOLD) == BOLD; }
41 	inline bool IsUnder() const { return (state & UNDERLINE) == UNDERLINE; }
42 	inline bool IsInverse() const { return (state & INVERSE) == INVERSE; }
43 	inline bool IsMouse() const { return (state & MOUSE) == MOUSE; }
44 	inline bool IsForeSet() const { return (state & FORESET) == FORESET; }
45 	inline bool IsBackSet() const { return (state & BACKSET) == BACKSET; }
46 	inline bool IsUnderSet() const { return (state & UNDERSET) == UNDERSET; }
47 	inline bool IsFont() const { return (state & FONT) == FONT; }
48 	inline bool IsCR() const { return (state & DUMPCR) == DUMPCR; }
49 
50 	inline void SetDirectForeground(uint8 red, uint8 green, uint8 blue)
51 	{
52 		foreground = 0x80000000 | (red << 16) | (green << 8) | blue;
53 		state &= ~FORECOLOR;
54 		state |= FORESET;
55 	}
56 
57 	inline void SetDirectBackground(uint8 red, uint8 green, uint8 blue)
58 	{
59 		background = 0x80000000 | (red << 16) | (green << 8) | blue;
60 		state &= ~BACKCOLOR;
61 		state |= BACKSET;
62 	}
63 
64 	inline void SetDirectUnderline(uint8 red, uint8 green, uint8 blue)
65 	{
66 		underline = 0x80000000 | (red << 16) | (green << 8) | blue;
67 		state |= UNDERSET;
68 	}
69 
70 	inline void SetIndexedForeground(uint32 index)
71 	{
72 		state &= ~FORECOLOR;
73 		state |= FORESET;
74 		state |= FORECOLORED(index);
75 		foreground = 0;
76 	}
77 
78 	inline void SetIndexedBackground(uint32 index)
79 	{
80 		state &= ~BACKCOLOR;
81 		state |= BACKSET;
82 		state |= BACKCOLORED(index);
83 		background = 0;
84 	}
85 
86 	inline void SetIndexedUnderline(uint32 index)
87 	{
88 		state |= UNDERSET;
89 		underline = index;
90 	}
91 
92 	inline void SetUnder(int style)
93 	{
94 		underlineStyle = style;
95 		state |= UNDERLINE;
96 	}
97 
98 	inline void UnsetForeground()
99 	{
100 		state &= ~FORESET;
101 		foreground = 0;
102 	}
103 
104 	inline void UnsetBackground()
105 	{
106 		state &= ~BACKSET;
107 		background = 0;
108 	}
109 
110 	inline void UnsetUnderline()
111 	{
112 		state &= ~UNDERSET;
113 		underline = 0;
114 	}
115 
116 	inline void UnsetUnder()
117 	{
118 		underlineStyle = 0;
119 		state &= ~UNDERLINE;
120 	}
121 
122 	inline rgb_color
123 	ForegroundColor(const rgb_color* indexedColors) const
124 	{
125 		if ((foreground & 0x80000000) != 0)
126 			return make_color((foreground >> 16) & 0xFF,
127 				(foreground >> 8) & 0xFF,
128 				foreground & 0xFF);
129 		else
130 			return indexedColors[(state & FORECOLOR) >> 16];
131 	}
132 
133 	inline rgb_color
134 	BackgroundColor(const rgb_color* indexedColors) const
135 	{
136 		if ((background & 0x80000000) != 0)
137 			return make_color((background >> 16) & 0xFF,
138 				(background >> 8) & 0xFF,
139 				background & 0xFF);
140 		else
141 			return indexedColors[(state & BACKCOLOR) >> 24];
142 	}
143 
144 	inline rgb_color
145 	UnderlineColor(const rgb_color* indexedColors) const
146 	{
147 		if ((underline & 0x80000000) != 0)
148 			return make_color((underline >> 16) & 0xFF,
149 				(underline >> 8) & 0xFF,
150 				underline & 0xFF);
151 		else
152 			return indexedColors[underline];
153 	}
154 
155 	inline int
156 	UnderlineStyle() const
157 	{
158 		return underlineStyle;
159 	}
160 
161 	inline Attributes&
162 	operator&=(uint32 value) { state &= value; return *this; }
163 
164 	inline Attributes&
165 	operator|=(uint32 value) { state |= value; return *this; }
166 
167 	inline uint32
168 	operator|(uint32 value) { return state | value; }
169 
170 	inline uint32
171 	operator&(uint32 value) { return state & value; }
172 
173 	inline bool
174 	operator==(const Attributes& other) const
175 	{
176 		return state == other.state
177 			&& foreground == other.foreground
178 			&& background == other.background
179 			&& underline == other.underline
180 			&& underlineStyle == other.underlineStyle;
181 	}
182 
183 	inline bool
184 	operator!=(const Attributes& other) const
185 	{
186 		return state != other.state
187 			|| foreground != other.foreground
188 			|| background != other.background
189 			|| underline != other.underline
190 			|| underlineStyle != other.underlineStyle;
191 	}
192 };
193 
194 
195 struct TerminalCell {
196 	UTF8Char			character;
197 	Attributes			attributes;
198 
199 	inline bool
200 	operator!=(const Attributes& other) const
201 	{
202 		return (attributes.state & CHAR_ATTRIBUTES)
203 				!= (other.state & CHAR_ATTRIBUTES)
204 			|| attributes.foreground != other.foreground
205 			|| attributes.background != other.background;
206 	}
207 };
208 
209 
210 struct TerminalLine {
211 	uint16			length;
212 	bool			softBreak;	// soft line break
213 	Attributes		attributes;
214 	TerminalCell	cells[1];
215 
216 	inline void Clear()
217 	{
218 		Clear(Attributes());
219 	}
220 
221 	inline void Clear(size_t count)
222 	{
223 		Clear(Attributes(), count);
224 	}
225 
226 	inline void Clear(Attributes attr, size_t count = 0)
227 	{
228 		length = 0;
229 		attributes = attr;
230 		softBreak = false;
231 		for (size_t i = 0; i < count; i++)
232 			cells[i].attributes = attr;
233 	}
234 };
235 
236 
237 struct AttributesRun {
238 	Attributes	attributes;
239 	uint16	offset;			// character offset
240 	uint16	length;			// length of the run in characters
241 };
242 
243 
244 struct HistoryLine {
245 	AttributesRun*	attributesRuns;
246 	uint16			attributesRunCount;	// number of attribute runs
247 	uint16			byteLength : 15;	// number of bytes in the line
248 	bool			softBreak : 1;		// soft line break;
249 	Attributes		attributes;
250 
251 	AttributesRun* AttributesRuns() const
252 	{
253 		return attributesRuns;
254 	}
255 
256 	char* Chars() const
257 	{
258 		return (char*)(attributesRuns + attributesRunCount);
259 	}
260 
261 	int32 BufferSize() const
262 	{
263 		return attributesRunCount * sizeof(AttributesRun) + byteLength;
264 	}
265 };
266 
267 
268 #endif	// TERMINAL_LINE_H
269