reshuffling of dlls
[reactos.git] / reactos / dll / riched20 / row.c
1 /*
2 * RichEdit - Operations on rows of text (rows are recreated during
3 * wrapping and are used for displaying the document, they don't keep any
4 * true document content; delete all rows, rewrap all paragraphs and
5 * you get them back).
6 *
7 * Copyright 2004 by Krzysztof Foltman
8 *
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
13 *
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
18 *
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 */
23
24
25 #include "editor.h"
26
27 ME_DisplayItem *ME_FindRowStart(ME_Context *c, ME_DisplayItem *item,
28 int nRelPos) {
29 ME_DisplayItem *para = ME_GetParagraph(item);
30 ME_MustBeWrapped(c, para);
31 if(nRelPos>=0) { /* if this or preceding row */
32 while(nRelPos<=0) {
33 ME_DisplayItem *item2 = ME_FindItemBack(item, diStartRowOrParagraph);
34 if (item2->type == diParagraph)
35 {
36 if (item2->member.para.prev_para == NULL)
37 return item;
38 /* if skipping to the preceding paragraph, ensure it's wrapped */
39 ME_MustBeWrapped(c, item2->member.para.prev_para);
40 item = item2;
41 continue;
42 }
43 else if (item2->type == diStartRow)
44 {
45 nRelPos++;
46 if (nRelPos>0)
47 return item;
48 item = item2;
49 continue;
50 }
51 assert(0 == "bug in FindItemBack(item, diStartRowOrParagraph)");
52 item = item2;
53 }
54 return item;
55 }
56 while(nRelPos>0) { /* if one of the next rows */
57 ME_DisplayItem *item2 = ME_FindItemFwd(item, diStartRowOrParagraph);
58 if (!item2)
59 return item;
60 if (item2->type == diParagraph)
61 {
62 if (item2->member.para.next_para == NULL)
63 return item;
64 continue;
65 }
66 item = item2;
67 nRelPos--;
68 }
69 return item;
70 }
71
72 /* I'm sure these functions would simplify some code in caret ops etc,
73 * I just didn't remember them when I wrote that code
74 */
75
76 ME_DisplayItem *ME_RowStart(ME_DisplayItem *item) {
77 return ME_FindItemBackOrHere(item, diStartRow);
78 }
79
80 /*
81 ME_DisplayItem *ME_RowEnd(ME_DisplayItem *item) {
82 ME_DisplayItem *item2 = ME_FindItemFwd(item, diStartRowOrParagraphOrEnd);
83 if (!item2) return NULL;
84 return ME_FindItemBack(item, diRun);
85 }
86 */
87
88 ME_DisplayItem *
89 ME_FindRowWithNumber(ME_TextEditor *editor, int nRow)
90 {
91 ME_DisplayItem *item = ME_FindItemFwd(editor->pBuffer->pFirst, diParagraph);
92 int nCount = 0;
93
94 while (item && nCount + item->member.para.nRows <= nRow)
95 {
96 nCount += item->member.para.nRows;
97 item = ME_FindItemFwd(item, diParagraph);
98 }
99 if (!item)
100 return item;
101 for (item = ME_FindItemFwd(item, diStartRow); item && nCount < nRow; nCount++)
102 item = ME_FindItemFwd(item, diStartRow);
103 return item;
104 }
105
106
107 int
108 ME_RowNumberFromCharOfs(ME_TextEditor *editor, int nOfs)
109 {
110 ME_DisplayItem *item = editor->pBuffer->pFirst->next;
111 int nRow = 0;
112
113 while (item && item->member.para.next_para->member.para.nCharOfs <= nOfs)
114 {
115 nRow += item->member.para.nRows;
116 item = ME_FindItemFwd(item, diParagraph);
117 }
118 if (item)
119 {
120 ME_DisplayItem *next_para = item->member.para.next_para;
121
122 nOfs -= item->member.para.nCharOfs;
123 item = ME_FindItemFwd(item, diRun);
124 while ((item = ME_FindItemFwd(item, diStartRowOrParagraph)) != NULL)
125 {
126 if (item == next_para)
127 break;
128 item = ME_FindItemFwd(item, diRun);
129 if (item->member.run.nCharOfs > nOfs)
130 break;
131 nRow++;
132 }
133 }
134 return nRow;
135 }