linedata.c revision 50027b5b
150027b5bSmrg/* $XTermId: linedata.c,v 1.101 2021/12/27 23:43:28 tom Exp $ */
2956cc18dSsnj
33367019cSmrg/*
48f44fb3bSmrg * Copyright 2009-2019,2021 by Thomas E. Dickey
53367019cSmrg *
63367019cSmrg *                         All Rights Reserved
73367019cSmrg *
83367019cSmrg * Permission is hereby granted, free of charge, to any person obtaining a
93367019cSmrg * copy of this software and associated documentation files (the
103367019cSmrg * "Software"), to deal in the Software without restriction, including
113367019cSmrg * without limitation the rights to use, copy, modify, merge, publish,
123367019cSmrg * distribute, sublicense, and/or sell copies of the Software, and to
133367019cSmrg * permit persons to whom the Software is furnished to do so, subject to
143367019cSmrg * the following conditions:
153367019cSmrg *
163367019cSmrg * The above copyright notice and this permission notice shall be included
173367019cSmrg * in all copies or substantial portions of the Software.
183367019cSmrg *
193367019cSmrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
203367019cSmrg * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
213367019cSmrg * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
223367019cSmrg * IN NO EVENT SHALL THE ABOVE LISTED COPYRIGHT HOLDER(S) BE LIABLE FOR ANY
233367019cSmrg * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
243367019cSmrg * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
253367019cSmrg * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
263367019cSmrg *
273367019cSmrg * Except as contained in this notice, the name(s) of the above copyright
283367019cSmrg * holders shall not be used in advertising or otherwise to promote the
293367019cSmrg * sale, use or other dealings in this Software without prior written
303367019cSmrg * authorization.
313367019cSmrg */
32956cc18dSsnj
33956cc18dSsnj#include <xterm.h>
34d4fba8b9Smrg#include <data.h>
35956cc18dSsnj
36956cc18dSsnj#include <assert.h>
37956cc18dSsnj
38956cc18dSsnj/*
39956cc18dSsnj * Given a row-number, find the corresponding data for the line in the VT100
40956cc18dSsnj * widget.  Row numbers can be positive or negative.
41956cc18dSsnj *
42956cc18dSsnj * If the data comes from the scrollback, defer that to getScrollback().
43956cc18dSsnj */
44956cc18dSsnjLineData *
4594644356SmrggetLineData(TScreen *screen, int row)
46956cc18dSsnj{
47956cc18dSsnj    LineData *result = 0;
48956cc18dSsnj    ScrnBuf buffer;
4950027b5bSmrg    int max_row = LastRowNumber(screen);
50956cc18dSsnj
51956cc18dSsnj    if (row >= 0) {
52956cc18dSsnj	buffer = screen->visbuf;
53956cc18dSsnj    } else {
54956cc18dSsnj	buffer = 0;
55956cc18dSsnj	result = getScrollback(screen, row);
56956cc18dSsnj    }
5720d2c4d2Smrg    if (row >= 0 && row <= max_row) {
58956cc18dSsnj	result = (LineData *) scrnHeadAddr(screen, buffer, (unsigned) row);
59956cc18dSsnj    }
60956cc18dSsnj
61956cc18dSsnj    return result;
62956cc18dSsnj}
63956cc18dSsnj
64956cc18dSsnj/*
65956cc18dSsnj * Copy line's data, e.g., from one screen buffer to another, given the preset
66956cc18dSsnj * pointers for the destination.
67956cc18dSsnj *
68956cc18dSsnj * TODO: optionally prune unused combining character data from the result.
69956cc18dSsnj */
70956cc18dSsnjvoid
7194644356SmrgcopyLineData(LineData *dst, CLineData *src)
72956cc18dSsnj{
73d4fba8b9Smrg    if (dst == NULL || src == NULL)
74d4fba8b9Smrg	return;
75d4fba8b9Smrg
76956cc18dSsnj    dst->bufHead = src->bufHead;
77956cc18dSsnj
78956cc18dSsnj#if OPT_WIDE_CHARS
79956cc18dSsnj    dst->combSize = src->combSize;
80956cc18dSsnj#endif
81956cc18dSsnj
82956cc18dSsnj    /*
83956cc18dSsnj     * Usually we're copying the same-sized line; a memcpy is faster than
84956cc18dSsnj     * several loops.
85956cc18dSsnj     */
86956cc18dSsnj    if (dst->lineSize == src->lineSize) {
87956cc18dSsnj	size_t size = (sizeof(dst->attribs[0])
88956cc18dSsnj#if OPT_ISO_COLORS
89956cc18dSsnj		       + sizeof(dst->color[0])
90956cc18dSsnj#endif
91956cc18dSsnj		       + sizeof(dst->charData[0])
92956cc18dSsnj#if OPT_WIDE_CHARS
93956cc18dSsnj		       + sizeof(dst->combData[0][0]) * dst->combSize
94956cc18dSsnj#endif
95956cc18dSsnj	);
96956cc18dSsnj
97956cc18dSsnj	memcpy(dst->attribs, src->attribs, size * dst->lineSize);
98956cc18dSsnj    } else {
99956cc18dSsnj	Dimension col;
100956cc18dSsnj	Dimension limit = ((dst->lineSize < src->lineSize)
101956cc18dSsnj			   ? dst->lineSize
102956cc18dSsnj			   : src->lineSize);
103956cc18dSsnj#if OPT_WIDE_CHARS
104956cc18dSsnj	Char comb;
105956cc18dSsnj#endif
106956cc18dSsnj
107956cc18dSsnj	for (col = 0; col < limit; ++col) {
108956cc18dSsnj	    dst->attribs[col] = src->attribs[col];
109956cc18dSsnj#if OPT_ISO_COLORS
110956cc18dSsnj	    dst->color[col] = src->color[col];
111956cc18dSsnj#endif
112956cc18dSsnj	    dst->charData[col] = src->charData[col];
113956cc18dSsnj#if OPT_WIDE_CHARS
114956cc18dSsnj	    for (comb = 0; comb < dst->combSize; ++comb) {
115956cc18dSsnj		dst->combData[comb][col] = src->combData[comb][col];
116956cc18dSsnj	    }
117956cc18dSsnj#endif
118956cc18dSsnj	}
119956cc18dSsnj	for (col = limit; col < dst->lineSize; ++col) {
120956cc18dSsnj	    dst->attribs[col] = 0;
121956cc18dSsnj#if OPT_ISO_COLORS
122d4fba8b9Smrg	    dst->color[col] = initCColor;
123956cc18dSsnj#endif
124956cc18dSsnj	    dst->charData[col] = 0;
125956cc18dSsnj#if OPT_WIDE_CHARS
126956cc18dSsnj	    for (comb = 0; comb < dst->combSize; ++comb) {
127956cc18dSsnj		dst->combData[comb][col] = 0;
128956cc18dSsnj	    }
129956cc18dSsnj#endif
130956cc18dSsnj	}
131956cc18dSsnj    }
132956cc18dSsnj}
133956cc18dSsnj
134956cc18dSsnj#if OPT_WIDE_CHARS
135956cc18dSsnj#define initLineExtra(screen) \
136d4fba8b9Smrg    screen->lineExtra = ((size_t) (screen->max_combining) * sizeof(IChar *)); \
137d4fba8b9Smrg    screen->cellExtra = ((size_t) (screen->max_combining) * sizeof(IChar))
138956cc18dSsnj#else
139956cc18dSsnj#define initLineExtra(screen) \
140d4fba8b9Smrg    screen->lineExtra = 0; \
141d4fba8b9Smrg    screen->cellExtra = 0
142956cc18dSsnj#endif
143956cc18dSsnj
144d4fba8b9Smrg/*
145d4fba8b9Smrg * CellData size depends on the "combiningChars" resource.
146d4fba8b9Smrg */
147d4fba8b9Smrg#define CellDataSize(screen) (SizeOfCellData + screen->cellExtra)
148d4fba8b9Smrg
149d4fba8b9Smrg#define CellDataAddr(screen, data, cell) \
150d4fba8b9Smrg	( (CellData *)(void *) ((char *)data + (cell * CellDataSize(screen))) )
151d4fba8b9Smrg#define ConstCellDataAddr(screen, data, cell) \
152d4fba8b9Smrg	( (const CellData *)(const void *) ( \
153d4fba8b9Smrg	      (const char *)data + (cell * CellDataSize(screen))) )
154d4fba8b9Smrg
155956cc18dSsnjvoid
156956cc18dSsnjinitLineData(XtermWidget xw)
157956cc18dSsnj{
15820d2c4d2Smrg    TScreen *screen = TScreenOf(xw);
159956cc18dSsnj
160956cc18dSsnj    initLineExtra(screen);
161956cc18dSsnj
162d4fba8b9Smrg#if OPT_WIDE_CHARS
163d4fba8b9Smrg    TRACE(("initLineData %lu (%d combining chars)\n",
164d4fba8b9Smrg	   (unsigned long) screen->lineExtra, screen->max_combining));
165d4fba8b9Smrg#else
166d4fba8b9Smrg    TRACE(("initLineData\n"));
167956cc18dSsnj#endif
168d4fba8b9Smrg
169d4fba8b9Smrg    /*
170d4fba8b9Smrg     * Per-line size/offsets.
171d4fba8b9Smrg     */
172d4fba8b9Smrg    TRACE(("** sizeof(LineData)  %lu\n", (unsigned long) sizeof(LineData)));
173d4fba8b9Smrg    TRACE(("   offset(lineSize)  %lu\n", (unsigned long) offsetof(LineData, lineSize)));
174d4fba8b9Smrg    TRACE(("   offset(bufHead)   %lu\n", (unsigned long) offsetof(LineData, bufHead)));
175956cc18dSsnj#if OPT_WIDE_CHARS
176d4fba8b9Smrg    TRACE(("   offset(combSize)  %lu\n", (unsigned long) offsetof(LineData, combSize)));
177956cc18dSsnj#endif
178d4fba8b9Smrg    TRACE(("   offset(*attribs)  %lu\n", (unsigned long) offsetof(LineData, attribs)));
179956cc18dSsnj#if OPT_ISO_COLORS
180d4fba8b9Smrg    TRACE(("   offset(*color)    %lu\n", (unsigned long) offsetof(LineData, color)));
181956cc18dSsnj#endif
182d4fba8b9Smrg    TRACE(("   offset(*charData) %lu\n", (unsigned long) offsetof(LineData, charData)));
183d4fba8b9Smrg    TRACE(("   offset(*combData) %lu\n", (unsigned long) offsetof(LineData, combData)));
184956cc18dSsnj
185d4fba8b9Smrg    /*
186d4fba8b9Smrg     * Per-cell size/offsets.
187d4fba8b9Smrg     */
188d4fba8b9Smrg    TRACE(("** sizeof(CellData)  %lu\n", (unsigned long) CellDataSize(screen)));
189d4fba8b9Smrg    TRACE(("   offset(attribs)   %lu\n", (unsigned long) offsetof(CellData, attribs)));
190d4fba8b9Smrg#if OPT_WIDE_CHARS
191d4fba8b9Smrg    TRACE(("   offset(combSize)  %lu\n", (unsigned long) offsetof(CellData, combSize)));
192d4fba8b9Smrg#endif
193d4fba8b9Smrg#if OPT_ISO_COLORS
194d4fba8b9Smrg    TRACE(("   offset(color)     %lu\n", (unsigned long) offsetof(CellData, color)));
195d4fba8b9Smrg#endif
196d4fba8b9Smrg    TRACE(("   offset(charData)  %lu\n", (unsigned long) offsetof(CellData, charData)));
197d4fba8b9Smrg    TRACE(("   offset(combData)  %lu\n", (unsigned long) offsetof(CellData, combData)));
198956cc18dSsnj
199d4fba8b9Smrg    /*
200d4fba8b9Smrg     * Data-type sizes.
201d4fba8b9Smrg     */
202d4fba8b9Smrg#if OPT_ISO_COLORS
203d4fba8b9Smrg    TRACE(("** sizeof(CellColor) %lu\n", (unsigned long) sizeof(CellColor)));
204d4fba8b9Smrg#endif
205d4fba8b9Smrg    TRACE(("** sizeof(IAttr)     %lu\n", (unsigned long) sizeof(IAttr)));
206d4fba8b9Smrg    TRACE(("** sizeof(IChar)     %lu\n", (unsigned long) sizeof(IChar)));
207d4fba8b9Smrg    TRACE(("** sizeof(RowData)   %lu\n", (unsigned long) sizeof(RowData)));
208d4fba8b9Smrg}
209956cc18dSsnj
210956cc18dSsnjCellData *
211956cc18dSsnjnewCellData(XtermWidget xw, Cardinal count)
212956cc18dSsnj{
213956cc18dSsnj    CellData *result;
21420d2c4d2Smrg    TScreen *screen = TScreenOf(xw);
215956cc18dSsnj
216956cc18dSsnj    initLineExtra(screen);
217956cc18dSsnj    result = (CellData *) calloc((size_t) count, (size_t) CellDataSize(screen));
218956cc18dSsnj    return result;
219956cc18dSsnj}
220956cc18dSsnj
221956cc18dSsnjvoid
22294644356SmrgsaveCellData(TScreen *screen,
22394644356Smrg	     CellData *data,
224956cc18dSsnj	     Cardinal cell,
22594644356Smrg	     CLineData *ld,
226d4fba8b9Smrg	     XTermRect *limits,
227956cc18dSsnj	     int column)
228956cc18dSsnj{
229956cc18dSsnj    CellData *item = CellDataAddr(screen, data, cell);
230956cc18dSsnj
231d4fba8b9Smrg    (void) limits;
232956cc18dSsnj    if (column < MaxCols(screen)) {
233956cc18dSsnj	item->attribs = ld->attribs[column];
234d4fba8b9Smrg	if_OPT_ISO_COLORS(screen, {
235d4fba8b9Smrg	    item->color = ld->color[column];
236d4fba8b9Smrg	});
237956cc18dSsnj	item->charData = ld->charData[column];
238956cc18dSsnj	if_OPT_WIDE_CHARS(screen, {
239956cc18dSsnj	    size_t off;
240d4fba8b9Smrg	    Bool blank = (((item->charData == HIDDEN_CHAR)
241d4fba8b9Smrg			   && (limits == NULL
242d4fba8b9Smrg			       || (column + 1) == limits->left))
243d4fba8b9Smrg			  || (item->charData != HIDDEN_CHAR
244d4fba8b9Smrg			      && WideCells(item->charData) > 1
245d4fba8b9Smrg			      && (limits == NULL
246d4fba8b9Smrg				  || (column + 1) >= limits->right)));
247d4fba8b9Smrg	    if (blank) {
2488f44fb3bSmrg		item->charData = (CharData) ' ';
249d4fba8b9Smrg	    }
250d4fba8b9Smrg	    item->combSize = blank ? 0 : ld->combSize;
251d4fba8b9Smrg	    for_each_combData(off, item) {
252956cc18dSsnj		item->combData[off] = ld->combData[off][column];
253956cc18dSsnj	    }
254d4fba8b9Smrg	});
255d4fba8b9Smrg	TRACE2(("SAVED::%s\n", visibleIChars(&(item->charData), 1)));
256956cc18dSsnj    }
257956cc18dSsnj}
258956cc18dSsnj
259956cc18dSsnjvoid
26094644356SmrgrestoreCellData(TScreen *screen,
26194644356Smrg		const CellData *data,
262956cc18dSsnj		Cardinal cell,
26394644356Smrg		LineData *ld,
264d4fba8b9Smrg		XTermRect *limits,
265956cc18dSsnj		int column)
266956cc18dSsnj{
26794644356Smrg    const CellData *item = ConstCellDataAddr(screen, data, cell);
268956cc18dSsnj
269d4fba8b9Smrg    (void) limits;
270956cc18dSsnj    if (column < MaxCols(screen)) {
271956cc18dSsnj	ld->attribs[column] = item->attribs;
272d4fba8b9Smrg	TRACE2(("BEFORE:%2d:%s\n", column + 1, visibleIChars(ld->charData, ld->lineSize)));
273d4fba8b9Smrg	if_OPT_ISO_COLORS(screen, {
274d4fba8b9Smrg	    ld->color[column] = item->color;
275d4fba8b9Smrg	});
276956cc18dSsnj	ld->charData[column] = item->charData;
277956cc18dSsnj	if_OPT_WIDE_CHARS(screen, {
278956cc18dSsnj	    size_t off;
279956cc18dSsnj	    ld->combSize = item->combSize;
280956cc18dSsnj	    for_each_combData(off, ld) {
281956cc18dSsnj		ld->combData[off][column] = item->combData[off];
282956cc18dSsnj	    }
283d4fba8b9Smrg	});
284d4fba8b9Smrg	TRACE2(("AFTER::%2d:%s\n", column + 1, visibleIChars(ld->charData, ld->lineSize)));
285956cc18dSsnj    }
286956cc18dSsnj}
287