ReactOS 0.4.16-dev-937-g7afcd2a
wrap.c File Reference
#include "editor.h"
Include dependency graph for wrap.c:

Go to the source code of this file.

Classes

struct  tagME_WrapContext
 
struct  repaint_range
 

Typedefs

typedef struct tagME_WrapContext ME_WrapContext
 

Functions

 WINE_DEFAULT_DEBUG_CHANNEL (richedit)
 
static BOOL get_run_glyph_buffers (ME_Run *run)
 
static HRESULT shape_run (ME_Context *c, ME_Run *run)
 
static void calc_run_extent (ME_Context *c, const ME_Paragraph *para, int startx, ME_Run *run)
 
static ME_Runsplit_run_extents (ME_WrapContext *wc, ME_Run *run, int nVChar)
 
static int find_split_point (ME_Context *c, int cx, ME_Run *run)
 
static ME_Rowrow_create (int height, int baseline, int width)
 
static void ME_BeginRow (ME_WrapContext *wc)
 
static void layout_row (ME_Run *start, ME_Run *last)
 
static void ME_InsertRowStart (ME_WrapContext *wc, ME_Run *last)
 
static void ME_WrapEndParagraph (ME_WrapContext *wc)
 
static void ME_WrapSizeRun (ME_WrapContext *wc, ME_Run *run)
 
static int find_non_whitespace (const WCHAR *s, int len, int start)
 
static int reverse_find_non_whitespace (const WCHAR *s, int start)
 
static int reverse_find_whitespace (const WCHAR *s, int start)
 
static ME_RunME_MaximizeSplit (ME_WrapContext *wc, ME_Run *run, int i)
 
static ME_RunME_SplitByBacktracking (ME_WrapContext *wc, ME_Run *run, int loc)
 
static ME_RunME_WrapHandleRun (ME_WrapContext *wc, ME_Run *run)
 
static int ME_GetParaLineSpace (ME_Context *c, ME_Paragraph *para)
 
static void ME_PrepareParagraphForWrapping (ME_TextEditor *editor, ME_Context *c, ME_Paragraph *para)
 
static HRESULT itemize_para (ME_Context *c, ME_Paragraph *para)
 
static HRESULT shape_para (ME_Context *c, ME_Paragraph *para)
 
static void ME_WrapTextParagraph (ME_TextEditor *editor, ME_Context *c, ME_Paragraph *para)
 
static void update_repaint (ME_Paragraph *para, struct repaint_range *repaint)
 
static void adjust_para_y (ME_Paragraph *para, ME_Context *c, struct repaint_range *repaint)
 
BOOL wrap_marked_paras_dc (ME_TextEditor *editor, HDC hdc, BOOL invalidate)
 
BOOL ME_WrapMarkedParagraphs (ME_TextEditor *editor)
 
void para_range_invalidate (ME_TextEditor *editor, ME_Paragraph *start_para, ME_Paragraph *last_para)
 
void ME_SendRequestResize (ME_TextEditor *editor, BOOL force)
 

Typedef Documentation

◆ ME_WrapContext

Function Documentation

◆ adjust_para_y()

static void adjust_para_y ( ME_Paragraph para,
ME_Context c,
struct repaint_range repaint 
)
static

Definition at line 919 of file wrap.c.

920{
921 ME_Cell *cell;
922
923 if (para->nFlags & MEPF_ROWSTART)
924 {
925 ME_Paragraph *end_row_para = table_row_end( para );
926 int borderWidth = 0;
927
928 cell = table_row_first_cell( para );
929 cell->pt = c->pt;
930 /* Offset the text by the largest top border width. */
931 while (cell_next( cell ))
932 {
933 borderWidth = max( borderWidth, cell->border.top.width );
934 cell = cell_next( cell );
935 }
936 if (borderWidth > 0)
937 {
938 borderWidth = max(ME_twips2pointsY(c, borderWidth), 1);
939 while (cell)
940 {
941 cell->yTextOffset = borderWidth;
942 cell = cell_prev( cell );
943 }
944 c->pt.y += borderWidth;
945 }
946 if (end_row_para->fmt.dxStartIndent > 0)
947 {
948 cell = table_row_first_cell( para );
949 cell->pt.x += ME_twips2pointsX( c, end_row_para->fmt.dxStartIndent );
950 c->pt.x = cell->pt.x;
951 }
952 }
953 else if (para->nFlags & MEPF_ROWEND)
954 {
955 /* Set all the cells to the height of the largest cell */
956 ME_Paragraph *start_row_para = table_row_start( para );
957 int prevHeight, nHeight, bottomBorder = 0;
958
959 cell = table_row_end_cell( para );
960 para->nWidth = cell->pt.x + cell->nWidth;
961 if (!(para_next( para )->nFlags & MEPF_ROWSTART))
962 {
963 /* Last row, the bottom border is added to the height. */
964 while ((cell = cell_prev( cell )))
965 bottomBorder = max( bottomBorder, cell->border.bottom.width );
966
967 bottomBorder = ME_twips2pointsY(c, bottomBorder);
968 cell = table_row_end_cell( para );
969 }
970 prevHeight = cell->nHeight;
971 nHeight = cell_prev( cell )->nHeight + bottomBorder;
972 cell->nHeight = nHeight;
973 para->nHeight = nHeight;
974 while (cell_prev( cell ))
975 {
976 cell = cell_prev( cell );
977 cell->nHeight = nHeight;
978 }
979
980 /* Also set the height of the start row paragraph */
981 start_row_para->nHeight = nHeight;
982 c->pt.x = start_row_para->pt.x;
983 c->pt.y = cell->pt.y + nHeight;
984 if (prevHeight < nHeight)
985 {
986 /* The height of the cells has grown, so invalidate the bottom of
987 * the cells. */
988 update_repaint( para, repaint );
989 cell = cell_prev( table_row_end_cell( para ) );
990 while (cell)
991 {
992 update_repaint( cell_end_para( cell ), repaint );
993 cell = cell_prev( cell );
994 }
995 }
996 }
997 else if ((cell = para_cell( para )) && para == cell_end_para( cell ))
998 {
999 /* The next paragraph is in the next cell in the table row. */
1000 cell->nHeight = c->pt.y + para->nHeight - cell->pt.y;
1001
1002 /* Propagate the largest height to the end so that it can be easily
1003 * sent back to all the cells at the end of the row. */
1004 if (cell_prev( cell ))
1005 cell->nHeight = max( cell->nHeight, cell_prev( cell )->nHeight );
1006
1007 c->pt.x = cell->pt.x + cell->nWidth;
1008 c->pt.y = cell->pt.y;
1009 cell_next( cell )->pt = c->pt;
1010 if (!(para_next( para )->nFlags & MEPF_ROWEND))
1011 c->pt.y += cell->yTextOffset;
1012 }
1013 else
1014 {
1015 if ((cell = para_cell( para ))) /* Next paragraph in the same cell. */
1016 c->pt.x = cell->pt.x;
1017 else /* Normal paragraph */
1018 c->pt.x = 0;
1019 c->pt.y += para->nHeight;
1020 }
1021}
ME_Paragraph * cell_end_para(ME_Cell *cell) DECLSPEC_HIDDEN
Definition: table.c:207
ME_Cell * para_cell(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: para.c:127
int ME_twips2pointsY(const ME_Context *c, int y) DECLSPEC_HIDDEN
Definition: paint.c:175
int ME_twips2pointsX(const ME_Context *c, int x) DECLSPEC_HIDDEN
Definition: paint.c:167
ME_Paragraph * table_row_start(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: table.c:142
ME_Paragraph * para_next(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: para.c:57
ME_Cell * table_row_first_cell(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: table.c:170
ME_Cell * cell_next(ME_Cell *cell) DECLSPEC_HIDDEN
Definition: table.c:192
ME_Cell * table_row_end_cell(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: table.c:178
ME_Cell * cell_prev(ME_Cell *cell) DECLSPEC_HIDDEN
Definition: table.c:197
ME_Paragraph * table_row_end(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: table.c:127
#define MEPF_ROWSTART
Definition: editstr.h:143
#define MEPF_ROWEND
Definition: editstr.h:144
const GLubyte * c
Definition: glext.h:8905
LONG dxStartIndent
Definition: richedit.h:670
ME_Border bottom
Definition: editstr.h:189
ME_Border top
Definition: editstr.h:187
POINT pt
Definition: editstr.h:225
int nHeight
Definition: editstr.h:226
ME_BorderRect border
Definition: editstr.h:224
int yTextOffset
Definition: editstr.h:227
int nWidth
Definition: editstr.h:226
PARAFORMAT2 fmt
Definition: editstr.h:203
long y
Definition: polytest.cpp:48
long x
Definition: polytest.cpp:48
#define max(a, b)
Definition: svc.c:63
static void update_repaint(ME_Paragraph *para, struct repaint_range *repaint)
Definition: wrap.c:913

Referenced by wrap_marked_paras_dc().

◆ calc_run_extent()

static void calc_run_extent ( ME_Context c,
const ME_Paragraph para,
int  startx,
ME_Run run 
)
static

Definition at line 114 of file wrap.c.

115{
116 if (run->nFlags & MERF_HIDDEN) run->nWidth = 0;
117 else
118 {
119 SIZE size = ME_GetRunSizeCommon( c, para, run, run->len, startx, &run->nAscent, &run->nDescent );
120 run->nWidth = size.cx;
121 }
122}
SIZE ME_GetRunSizeCommon(ME_Context *c, const ME_Paragraph *para, ME_Run *run, int nLen, int startx, int *pAscent, int *pDescent) DECLSPEC_HIDDEN
Definition: run.c:669
#define MERF_HIDDEN
Definition: editstr.h:125
GLsizeiptr size
Definition: glext.h:5919
int nAscent
Definition: editstr.h:165
int nDescent
Definition: editstr.h:165
int nWidth
Definition: editstr.h:163
int nFlags
Definition: editstr.h:164
int len
Definition: editstr.h:162

Referenced by ME_WrapHandleRun(), ME_WrapSizeRun(), and split_run_extents().

◆ find_non_whitespace()

static int find_non_whitespace ( const WCHAR s,
int  len,
int  start 
)
static

Definition at line 395 of file wrap.c.

396{
397 int i;
398 for (i = start; i < len && ME_IsWSpace( s[i] ); i++)
399 ;
400
401 return i;
402}
static int ME_IsWSpace(WCHAR ch)
Definition: editor.h:99
GLuint start
Definition: gl.h:1545
GLdouble s
Definition: gl.h:2039
GLenum GLsizei len
Definition: glext.h:6722
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint i
Definition: glfuncs.h:248

Referenced by ME_WrapHandleRun().

◆ find_split_point()

static int find_split_point ( ME_Context c,
int  cx,
ME_Run run 
)
static

Definition at line 170 of file wrap.c.

171{
172 if (!run->len || cx <= 0) return 0;
173 return ME_CharFromPointContext( c, cx, run, FALSE, FALSE );
174}
#define FALSE
Definition: types.h:117
int ME_CharFromPointContext(ME_Context *c, int cx, ME_Run *run, BOOL closest, BOOL visual_order) DECLSPEC_HIDDEN
Definition: run.c:511
_Out_opt_ int * cx
Definition: commctrl.h:585

Referenced by ME_SplitByBacktracking().

◆ get_run_glyph_buffers()

static BOOL get_run_glyph_buffers ( ME_Run run)
static

Definition at line 51 of file wrap.c.

52{
53 heap_free( run->glyphs );
54 run->glyphs = heap_alloc( run->max_glyphs * (sizeof(WORD) + sizeof(SCRIPT_VISATTR) + sizeof(int) + sizeof(GOFFSET)) );
55 if (!run->glyphs) return FALSE;
56
57 run->vis_attrs = (SCRIPT_VISATTR*)((char*)run->glyphs + run->max_glyphs * sizeof(WORD));
58 run->advances = (int*)((char*)run->glyphs + run->max_glyphs * (sizeof(WORD) + sizeof(SCRIPT_VISATTR)));
59 run->offsets = (GOFFSET*)((char*)run->glyphs + run->max_glyphs * (sizeof(WORD) + sizeof(SCRIPT_VISATTR) + sizeof(int)));
60
61 return TRUE;
62}
static void * heap_alloc(size_t len)
Definition: appwiz.h:66
static BOOL heap_free(void *mem)
Definition: appwiz.h:76
#define TRUE
Definition: types.h:120
unsigned int(__cdecl typeof(jpeg_read_scanlines))(struct jpeg_decompress_struct *
Definition: typeof.h:31
unsigned short WORD
Definition: ntddk_ex.h:93
GOFFSET * offsets
Definition: editstr.h:174
SCRIPT_VISATTR * vis_attrs
Definition: editstr.h:172
int * advances
Definition: editstr.h:173
WORD * glyphs
Definition: editstr.h:171
int max_glyphs
Definition: editstr.h:170

Referenced by shape_run().

◆ itemize_para()

static HRESULT itemize_para ( ME_Context c,
ME_Paragraph para 
)
static

Definition at line 724 of file wrap.c.

725{
726 ME_Run *run;
727 SCRIPT_ITEM buf[16], *items = buf;
728 int items_passed = ARRAY_SIZE( buf ), num_items, cur_item;
730 FALSE, FALSE, 0 };
732 HRESULT hr;
733
734 if (para->fmt.dwMask & PFM_RTLPARA && para->fmt.wEffects & PFE_RTLPARA)
735 state.uBidiLevel = 1;
736
737 TRACE( "Base embedding level %d\n", state.uBidiLevel );
738
739 while (1)
740 {
741 hr = ScriptItemize( para->text->szData, para->text->nLen, items_passed, &control,
742 &state, items, &num_items );
743 if (hr != E_OUTOFMEMORY) break; /* may not be enough items if hr == E_OUTOFMEMORY */
744 if (items_passed > para->text->nLen + 1) break; /* something else has gone wrong */
745 items_passed *= 2;
746 if (items == buf)
747 items = heap_alloc( items_passed * sizeof( *items ) );
748 else
749 items = heap_realloc( items, items_passed * sizeof( *items ) );
750 if (!items) break;
751 }
752 if (FAILED( hr )) goto end;
753
754 if (TRACE_ON( richedit ))
755 {
756 TRACE( "got items:\n" );
757 for (cur_item = 0; cur_item < num_items; cur_item++)
758 {
759 TRACE( "\t%d - %d RTL %d bidi level %d\n", items[cur_item].iCharPos, items[cur_item+1].iCharPos - 1,
760 items[cur_item].a.fRTL, items[cur_item].a.s.uBidiLevel );
761 }
762
763 TRACE( "before splitting runs into ranges\n" );
764 for (run = para_first_run( para ); run; run = run_next( run ))
765 TRACE( "\t%d: %s\n", run->nCharOfs, debugstr_run( run ) );
766 }
767
768 /* split runs into ranges at item boundaries */
769 for (run = para_first_run( para ), cur_item = 0; run; run = run_next( run ))
770 {
771 if (run->nCharOfs == items[cur_item+1].iCharPos) cur_item++;
772
773 items[cur_item].a.fLogicalOrder = TRUE;
774 run->script_analysis = items[cur_item].a;
775
776 if (run->nFlags & MERF_ENDPARA) break; /* don't split eop runs */
777
778 if (run->nCharOfs + run->len > items[cur_item+1].iCharPos)
779 {
780 ME_Cursor cursor = { para, run, items[cur_item + 1].iCharPos - run->nCharOfs };
781 run_split( c->editor, &cursor );
782 }
783 }
784
785 if (TRACE_ON( richedit ))
786 {
787 TRACE( "after splitting into ranges\n" );
788 for (run = para_first_run( para ); run; run = run_next( run ))
789 TRACE( "\t%d: %s\n", run->nCharOfs, debugstr_run( run ) );
790 }
791
792 para->nFlags |= MEPF_COMPLEX;
793
794end:
795 if (items != buf) heap_free( items );
796 return hr;
797}
static int state
Definition: maze.c:121
static void * heap_realloc(void *mem, size_t len)
Definition: appwiz.h:71
#define ARRAY_SIZE(A)
Definition: main.h:20
#define E_OUTOFMEMORY
Definition: ddrawi.h:100
#define TRACE_ON(x)
Definition: compat.h:75
HRESULT WINAPI ScriptItemize(const WCHAR *pwcInChars, int cInChars, int cMaxItems, const SCRIPT_CONTROL *psControl, const SCRIPT_STATE *psState, SCRIPT_ITEM *pItems, int *pcItems)
Definition: usp10.c:1853
ME_Run * para_first_run(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: para.c:104
ME_Run * run_next(ME_Run *run) DECLSPEC_HIDDEN
Definition: run.c:68
ME_Run * run_split(ME_TextEditor *editor, ME_Cursor *cursor) DECLSPEC_HIDDEN
Definition: run.c:305
static const char * debugstr_run(const ME_Run *run)
Definition: editor.h:47
#define MERF_ENDPARA
Definition: editstr.h:121
#define MEPF_COMPLEX
Definition: editstr.h:145
GLuint GLuint end
Definition: gl.h:1545
GLenum GLuint GLenum GLsizei const GLchar * buf
Definition: glext.h:7751
GLboolean GLboolean GLboolean GLboolean a
Definition: glext.h:6204
const char cursor[]
Definition: icontest.c:13
#define FAILED(hr)
Definition: intsafe.h:51
static TCHAR * items[]
Definition: page1.c:45
#define PFE_RTLPARA
Definition: richedit.h:932
#define PFM_RTLPARA
Definition: richedit.h:856
HRESULT hr
Definition: shlfolder.c:183
#define TRACE(s)
Definition: solgame.cpp:4
DWORD dwMask
Definition: richedit.h:667
WORD wEffects
Definition: richedit.h:669
Definition: dialog.c:52
ME_String * text
Definition: editstr.h:204
int nCharOfs
Definition: editstr.h:161
SCRIPT_ANALYSIS script_analysis
Definition: editstr.h:169
WCHAR * szData
Definition: editstr.h:55
int nLen
Definition: editstr.h:56
#define LANG_USER_DEFAULT
Definition: tnerror.cpp:50

Referenced by ME_WrapTextParagraph().

◆ layout_row()

static void layout_row ( ME_Run start,
ME_Run last 
)
static

Definition at line 233 of file wrap.c.

234{
235 ME_Run *run;
236 int i, num_runs = 0;
237 int buf[16 * 5]; /* 5 arrays - 4 of int & 1 of BYTE, alloc space for 5 of ints */
238 int *vis_to_log = buf, *log_to_vis, *widths, *pos;
239 BYTE *levels;
240 BOOL found_black = FALSE;
241
242 for (run = last; run; run = run_prev( run ))
243 {
244 if (!found_black) found_black = !(run->nFlags & (MERF_WHITESPACE | MERF_ENDPARA));
245 if (found_black) num_runs++;
246 if (run == start) break;
247 }
248
249 TRACE("%d runs\n", num_runs);
250 if (!num_runs) return;
251
252 if (num_runs > ARRAY_SIZE( buf ) / 5)
253 vis_to_log = heap_alloc( num_runs * sizeof(int) * 5 );
254
255 log_to_vis = vis_to_log + num_runs;
256 widths = vis_to_log + 2 * num_runs;
257 pos = vis_to_log + 3 * num_runs;
258 levels = (BYTE*)(vis_to_log + 4 * num_runs);
259
260 for (i = 0, run = start; i < num_runs; run = run_next( run ))
261 {
263 widths[i] = run->nWidth;
264 TRACE( "%d: level %d width %d\n", i, levels[i], widths[i] );
265 i++;
266 }
267
268 ScriptLayout( num_runs, levels, vis_to_log, log_to_vis );
269
270 pos[0] = run->para->pt.x;
271 for (i = 1; i < num_runs; i++)
272 pos[i] = pos[i - 1] + widths[ vis_to_log[ i - 1 ] ];
273
274 for (i = 0, run = start; i < num_runs; run = run_next( run ))
275 {
276 run->pt.x = pos[ log_to_vis[ i ] ];
277 TRACE( "%d: x = %d\n", i, run->pt.x );
278 i++;
279 }
280
281 if (vis_to_log != buf) heap_free( vis_to_log );
282}
HRESULT WINAPI ScriptLayout(int runs, const BYTE *level, int *vistolog, int *logtovis)
Definition: usp10.c:3752
ME_Run * run_prev(ME_Run *run) DECLSPEC_HIDDEN
Definition: run.c:82
#define MERF_WHITESPACE
Definition: editstr.h:119
unsigned int BOOL
Definition: ntddk_ex.h:94
GLsizei levels
Definition: glext.h:7884
static UINT UINT last
Definition: font.c:45
POINT pt
Definition: editstr.h:166
struct tagME_Paragraph * para
Definition: editstr.h:160
SCRIPT_STATE s
Definition: usp10.h:146
WORD uBidiLevel
Definition: usp10.h:125
unsigned char BYTE
Definition: xxhash.c:193

Referenced by ME_InsertRowStart().

◆ ME_BeginRow()

static void ME_BeginRow ( ME_WrapContext wc)
static

Definition at line 186 of file wrap.c.

187{
188 ME_Cell *cell;
189
190 wc->pRowStart = NULL;
191 wc->bOverflown = FALSE;
193 wc->bWordWrap = wc->context->editor->bWordWrap;
194 if (wc->para->nFlags & (MEPF_ROWSTART | MEPF_ROWEND))
195 {
196 wc->nAvailWidth = 0;
197 wc->bWordWrap = FALSE;
198 if (wc->para->nFlags & MEPF_ROWEND)
199 {
200 cell = table_row_end_cell( wc->para );
201 cell->nWidth = 0;
202 }
203 }
204 else if (para_cell( wc->para ))
205 {
206 int width;
207
208 cell = para_cell( wc->para );
209 width = cell->nRightBoundary;
210 if (cell_prev( cell )) width -= cell_prev( cell )->nRightBoundary;
211 else
212 {
213 int rowIndent = table_row_end( wc->para )->fmt.dxStartIndent;
214 width -= rowIndent;
215 }
216 cell->nWidth = max(ME_twips2pointsX(wc->context, width), 0);
217
218 wc->nAvailWidth = cell->nWidth
219 - (wc->nRow ? wc->nLeftMargin : wc->nFirstMargin) - wc->nRightMargin;
220 wc->bWordWrap = TRUE;
221 }
222 else
224 - (wc->nRow ? wc->nLeftMargin : wc->nFirstMargin) - wc->nRightMargin;
225
226 wc->pt.x = wc->context->pt.x;
227 if (wc->context->editor->bEmulateVersion10 && /* v1.0 - 3.0 */
228 para_in_table( wc->para ))
229 /* Shift the text down because of the border. */
230 wc->pt.y++;
231}
#define NULL
Definition: types.h:112
BOOL para_in_table(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: para.c:122
GLint GLint GLsizei width
Definition: gl.h:1546
if(dx< 0)
Definition: linetemp.h:194
int nRightBoundary
Definition: editstr.h:223
int nAvailWidth
Definition: editstr.h:460
POINT pt
Definition: editstr.h:457
ME_TextEditor * editor
Definition: editstr.h:465
unsigned int bEmulateVersion10
Definition: editstr.h:385
int nLeftMargin
Definition: wrap.c:39
int nAvailWidth
Definition: wrap.c:42
int nRightMargin
Definition: wrap.c:39
int nFirstMargin
Definition: wrap.c:40
ME_Run * pRowStart
Definition: wrap.c:47
ME_Paragraph * para
Definition: wrap.c:46
POINT pt
Definition: wrap.c:44
BOOL bOverflown
Definition: wrap.c:45
ME_Context * context
Definition: wrap.c:38
BOOL bWordWrap
Definition: wrap.c:45
ME_Run * pLastSplittableRun
Definition: wrap.c:48

Referenced by ME_InsertRowStart(), and ME_WrapTextParagraph().

◆ ME_GetParaLineSpace()

static int ME_GetParaLineSpace ( ME_Context c,
ME_Paragraph para 
)
static

Definition at line 669 of file wrap.c.

670{
671 int sp = 0, ls = 0;
672 if (!(para->fmt.dwMask & PFM_LINESPACING)) return 0;
673
674 /* FIXME: how to compute simply the line space in ls ??? */
675 /* FIXME: does line spacing include the line itself ??? */
676 switch (para->fmt.bLineSpacingRule)
677 {
678 case 0: sp = ls; break;
679 case 1: sp = (3 * ls) / 2; break;
680 case 2: sp = 2 * ls; break;
681 case 3: sp = ME_twips2pointsY(c, para->fmt.dyLineSpacing); if (sp < ls) sp = ls; break;
682 case 4: sp = ME_twips2pointsY(c, para->fmt.dyLineSpacing); break;
683 case 5: sp = para->fmt.dyLineSpacing / 20; break;
684 default: FIXME("Unsupported spacing rule value %d\n", para->fmt.bLineSpacingRule);
685 }
686 if (c->editor->nZoomNumerator == 0)
687 return sp;
688 else
689 return sp * c->editor->nZoomNumerator / c->editor->nZoomDenominator;
690}
void ls(int argc, const char *argv[])
Definition: cmds.c:1136
#define FIXME(fmt,...)
Definition: precomp.h:53
static const WCHAR sp[]
Definition: suminfo.c:287
#define PFM_LINESPACING
Definition: richedit.h:849
BYTE bLineSpacingRule
Definition: richedit.h:678
LONG dyLineSpacing
Definition: richedit.h:676

Referenced by ME_WrapTextParagraph().

◆ ME_InsertRowStart()

static void ME_InsertRowStart ( ME_WrapContext wc,
ME_Run last 
)
static

Definition at line 284 of file wrap.c.

285{
286 ME_Run *run;
287 ME_Row *row;
288 BOOL bSkippingSpaces = TRUE;
289 int ascent = 0, descent = 0, width = 0, shift = 0, align = 0;
290
291 /* Include height of para numbering label */
292 if (wc->nRow == 0 && wc->para->fmt.wNumbering)
293 {
294 ascent = wc->para->para_num.style->tm.tmAscent;
295 descent = wc->para->para_num.style->tm.tmDescent;
296 }
297
298 for (run = last; run; run = run_prev( run ))
299 {
300 /* ENDPARA run shouldn't affect row height, except if it's the only run in the paragraph */
301 if (run == wc->pRowStart || !(run->nFlags & MERF_ENDPARA))
302 {
303 if (run->nAscent > ascent) ascent = run->nAscent;
304 if (run->nDescent > descent) descent = run->nDescent;
305 if (bSkippingSpaces)
306 {
307 /* Exclude space characters from run width.
308 * Other whitespace or delimiters are not treated this way. */
309 int len = run->len;
310 WCHAR *text = get_text( run, len - 1 );
311
312 assert(len);
313 if (~run->nFlags & MERF_GRAPHICS)
314 while (len && *(text--) == ' ') len--;
315 if (len)
316 {
317 if (len == run->len)
318 width += run->nWidth;
319 else
321 }
322 bSkippingSpaces = !len;
323 }
324 else if (!(run->nFlags & MERF_ENDPARA))
325 width += run->nWidth;
326 }
327 if (run == wc->pRowStart) break;
328 }
329
330 wc->para->nWidth = max( wc->para->nWidth, width );
331 row = row_create( ascent + descent, ascent, width );
332 if (wc->context->editor->bEmulateVersion10 && /* v1.0 - 3.0 */
333 (wc->para->fmt.dwMask & PFM_TABLE) && (wc->para->fmt.wEffects & PFE_TABLE))
334 {
335 /* The text was shifted down in ME_BeginRow so move the wrap context
336 * back to where it should be. */
337 wc->pt.y--;
338 /* The height of the row is increased by the borders. */
339 row->nHeight += 2;
340 }
341 row->pt = wc->pt;
342 row->nLMargin = (!wc->nRow ? wc->nFirstMargin : wc->nLeftMargin);
343 row->nRMargin = wc->nRightMargin;
345 align = wc->para->fmt.wAlignment;
346 if (align == PFA_CENTER) shift = max((wc->nAvailWidth-width)/2, 0);
347 if (align == PFA_RIGHT) shift = max(wc->nAvailWidth-width, 0);
348
349 if (wc->para->nFlags & MEPF_COMPLEX) layout_row( wc->pRowStart, last );
350
351 row->pt.x = row->nLMargin + shift;
352
353 for (run = wc->pRowStart; run; run = run_next( run ))
354 {
355 run->pt.x += row->nLMargin + shift;
356 if (run == last) break;
357 }
358
359 if (wc->nRow == 0 && wc->para->fmt.wNumbering)
360 {
361 wc->para->para_num.pt.x = wc->nParaNumOffset + shift;
362 wc->para->para_num.pt.y = wc->pt.y + row->nBaseline;
363 }
364
366 wc->nRow++;
367 wc->pt.y += row->nHeight;
368 ME_BeginRow( wc );
369}
const WCHAR * text
Definition: package.c:1794
struct png_info_def *typedef unsigned char **typedef struct png_info_def *typedef struct png_info_def *typedef struct png_info_def *typedef unsigned char ** row
Definition: typeof.h:78
#define assert(x)
Definition: debug.h:53
int align(int length, int align)
Definition: dsound8.c:36
int ME_PointFromCharContext(ME_Context *c, ME_Run *pRun, int nOffset, BOOL visual_order) DECLSPEC_HIDDEN
Definition: run.c:608
static ME_DisplayItem * run_get_di(ME_Run *run)
Definition: editor.h:163
static ME_DisplayItem * row_get_di(ME_Row *row)
Definition: editor.h:126
void ME_InsertBefore(ME_DisplayItem *diWhere, ME_DisplayItem *diWhat) DECLSPEC_HIDDEN
Definition: list.c:26
static WCHAR * get_text(const ME_Run *run, int offset)
Definition: editor.h:42
#define MERF_GRAPHICS
Definition: editstr.h:104
#define shift
Definition: input.c:1755
#define PFA_RIGHT
Definition: richedit.h:922
#define PFA_CENTER
Definition: richedit.h:923
#define PFM_TABLE
Definition: richedit.h:870
#define PFE_TABLE
Definition: richedit.h:944
#define PFM_ALIGNMENT
Definition: richedit.h:841
WORD wAlignment
Definition: richedit.h:673
WORD wNumbering
Definition: richedit.h:668
struct para_num para_num
Definition: editstr.h:214
int nParaNumOffset
Definition: wrap.c:41
static ME_Row * row_create(int height, int baseline, int width)
Definition: wrap.c:176
static void layout_row(ME_Run *start, ME_Run *last)
Definition: wrap.c:233
static void ME_BeginRow(ME_WrapContext *wc)
Definition: wrap.c:186
__wchar_t WCHAR
Definition: xmlstorage.h:180

Referenced by ME_WrapEndParagraph(), and ME_WrapHandleRun().

◆ ME_MaximizeSplit()

static ME_Run * ME_MaximizeSplit ( ME_WrapContext wc,
ME_Run run,
int  i 
)
static

Definition at line 427 of file wrap.c.

428{
429 ME_Run *new_run, *iter = run;
430 int j;
431 if (!i)
432 return NULL;
433 j = reverse_find_non_whitespace( get_text( run, 0 ), i );
434 if (j > 0)
435 {
436 new_run = split_run_extents( wc, iter, j );
437 wc->pt.x += iter->nWidth;
438 return new_run;
439 }
440 else
441 {
442 new_run = iter;
443 /* omit all spaces before split point */
444 while (iter != wc->pRowStart)
445 {
446 iter = run_prev( iter );
447 if (iter->nFlags & MERF_WHITESPACE)
448 {
449 new_run = iter;
450 continue;
451 }
452 if (iter->nFlags & MERF_ENDWHITE)
453 {
454 i = reverse_find_non_whitespace( get_text( iter, 0 ), iter->len );
455 new_run = split_run_extents( wc, iter, i );
456 wc->pt = new_run->pt;
457 return new_run;
458 }
459 /* this run is the end of spaces, so the run edge is a good point to split */
460 wc->pt = new_run->pt;
461 wc->bOverflown = TRUE;
462 TRACE( "Split point is: %s|%s\n", debugstr_run( iter ), debugstr_run( new_run ) );
463 return new_run;
464 }
465 wc->pt = iter->pt;
466 return iter;
467 }
468}
#define MERF_ENDWHITE
Definition: editstr.h:117
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint GLint GLint j
Definition: glfuncs.h:250
static int reverse_find_non_whitespace(const WCHAR *s, int start)
Definition: wrap.c:408
static ME_Run * split_run_extents(ME_WrapContext *wc, ME_Run *run, int nVChar)
Definition: wrap.c:130

Referenced by ME_SplitByBacktracking().

◆ ME_PrepareParagraphForWrapping()

static void ME_PrepareParagraphForWrapping ( ME_TextEditor editor,
ME_Context c,
ME_Paragraph para 
)
static

Definition at line 692 of file wrap.c.

693{
695
696 para->nWidth = 0;
697 /* remove row start items as they will be reinserted by the
698 * paragraph wrapper anyway */
699 editor->total_rows -= para->nRows;
700 para->nRows = 0;
701 for (p = para_get_di( para ); p != para->next_para; p = p->next)
702 {
703 if (p->type == diStartRow)
704 {
705 ME_DisplayItem *pRow = p;
706 p = p->prev;
707 ME_Remove( pRow );
708 ME_DestroyDisplayItem( pRow );
709 }
710 }
711
712 /* join runs that can be joined */
713 for (p = para_get_di( para )->next; p != para->next_para; p = p->next)
714 {
715 assert(p->type != diStartRow); /* should have been deleted above */
716 if (p->type == diRun)
717 {
718 while (p->next->type == diRun && ME_CanJoinRuns( &p->member.run, &p->next->member.run ))
719 run_join( c->editor, &p->member.run );
720 }
721 }
722}
static ME_DisplayItem * para_get_di(ME_Paragraph *para)
Definition: editor.h:237
void ME_DestroyDisplayItem(ME_DisplayItem *item) DECLSPEC_HIDDEN
Definition: list.c:115
void run_join(ME_TextEditor *editor, ME_Run *run) DECLSPEC_HIDDEN
Definition: run.c:273
void ME_Remove(ME_DisplayItem *diWhere) DECLSPEC_HIDDEN
Definition: list.c:35
BOOL ME_CanJoinRuns(const ME_Run *run1, const ME_Run *run2) DECLSPEC_HIDDEN
Definition: run.c:129
@ diStartRow
Definition: editstr.h:87
@ diRun
Definition: editstr.h:86
GLfloat GLfloat p
Definition: glext.h:8902
static unsigned __int64 next
Definition: rand_nt.c:6
struct tagME_DisplayItem * next_para
Definition: editstr.h:216

Referenced by ME_WrapTextParagraph().

◆ ME_SendRequestResize()

void ME_SendRequestResize ( ME_TextEditor editor,
BOOL  force 
)

Definition at line 1128 of file wrap.c.

1129{
1130 if (editor->nEventMask & ENM_REQUESTRESIZE)
1131 {
1132 RECT rc;
1133
1134 ITextHost_TxGetClientRect(editor->texthost, &rc);
1135
1136 if (force || rc.bottom != editor->nTotalLength)
1137 {
1139
1140 info.nmhdr.hwndFrom = NULL;
1141 info.nmhdr.idFrom = 0;
1142 info.nmhdr.code = EN_REQUESTRESIZE;
1143 info.rc = rc;
1144 info.rc.right = editor->nTotalWidth;
1145 info.rc.bottom = editor->nTotalLength;
1146
1147 editor->nEventMask &= ~ENM_REQUESTRESIZE;
1148 ITextHost_TxNotify(editor->texthost, info.nmhdr.code, &info);
1149 editor->nEventMask |= ENM_REQUESTRESIZE;
1150 }
1151 }
1152}
#define ITextHost_TxNotify(This, a, b)
Definition: editor.h:368
#define ITextHost_TxGetClientRect(This, a)
Definition: editor.h:354
#define EN_REQUESTRESIZE
Definition: richedit.h:192
#define ENM_REQUESTRESIZE
Definition: richedit.h:477
ITextHost2 * texthost
Definition: editstr.h:384
LONG bottom
Definition: windef.h:309

Referenced by editor_draw(), editor_handle_message(), ME_KeyDown(), and ME_StreamIn().

◆ ME_SplitByBacktracking()

static ME_Run * ME_SplitByBacktracking ( ME_WrapContext wc,
ME_Run run,
int  loc 
)
static

Definition at line 470 of file wrap.c.

471{
472 ME_Run *new_run;
473 int i, idesp, len;
474
475 idesp = i = find_split_point( wc->context, loc, run );
476 len = run->len;
477 assert( len > 0 );
478 assert( i < len );
479 if (i)
480 {
481 /* don't split words */
482 i = reverse_find_whitespace( get_text( run, 0 ), i );
483 new_run = ME_MaximizeSplit(wc, run, i);
484 if (new_run) return new_run;
485 }
486 TRACE("Must backtrack to split at: %s\n", debugstr_run( run ));
487 if (wc->pLastSplittableRun)
488 {
490 {
491 wc->pt = wc->pLastSplittableRun->pt;
492 return wc->pLastSplittableRun;
493 }
495 {
496 /* the following two lines are just to check if we forgot to call UpdateRunFlags earlier,
497 they serve no other purpose */
500
501 run = wc->pLastSplittableRun;
502 len = run->len;
503 /* don't split words */
504 i = reverse_find_whitespace( get_text( run, 0 ), len );
505 if (i == len)
507 new_run = split_run_extents(wc, run, i);
508 wc->pt = new_run->pt;
509 return new_run;
510 }
511 else
512 {
513 /* restart from the first run beginning with spaces */
514 wc->pt = wc->pLastSplittableRun->pt;
515 return wc->pLastSplittableRun;
516 }
517 }
518 TRACE("Backtracking failed, trying desperate: %s\n", debugstr_run( run ));
519 /* OK, no better idea, so assume we MAY split words if we can split at all*/
520 if (idesp)
521 return split_run_extents(wc, run, idesp);
522 else
523 if (wc->pRowStart && run != wc->pRowStart)
524 {
525 /* don't need to break current run, because it's possible to split
526 before this run */
527 wc->bOverflown = TRUE;
528 return run;
529 }
530 else
531 {
532 /* split point inside first character - no choice but split after that char */
533 if (len != 1)
534 /* the run is more than 1 char, so we may split */
535 return split_run_extents( wc, run, 1 );
536
537 /* the run is one char, can't split it */
538 return run;
539 }
540}
void ME_UpdateRunFlags(ME_TextEditor *editor, ME_Run *run) DECLSPEC_HIDDEN
Definition: run.c:462
#define MERF_TAB
Definition: editstr.h:106
#define MERF_SPLITTABLE
Definition: editstr.h:113
static int reverse_find_whitespace(const WCHAR *s, int start)
Definition: wrap.c:418
static ME_Run * ME_MaximizeSplit(ME_WrapContext *wc, ME_Run *run, int i)
Definition: wrap.c:427
static int find_split_point(ME_Context *c, int cx, ME_Run *run)
Definition: wrap.c:170

Referenced by ME_WrapHandleRun().

◆ ME_WrapEndParagraph()

static void ME_WrapEndParagraph ( ME_WrapContext wc)
static

Definition at line 371 of file wrap.c.

372{
373 if (wc->pRowStart) ME_InsertRowStart( wc, wc->para->eop_run );
374
375 if (wc->context->editor->bEmulateVersion10 && /* v1.0 - 3.0 */
377 {
378 /* ME_BeginRow was called an extra time for the paragraph, and it shifts the
379 * text down by one pixel for the border, so fix up the wrap context. */
380 wc->pt.y--;
381 }
382}
ME_Run * eop_run
Definition: editstr.h:215
static void ME_InsertRowStart(ME_WrapContext *wc, ME_Run *last)
Definition: wrap.c:284

Referenced by ME_WrapTextParagraph().

◆ ME_WrapHandleRun()

static ME_Run * ME_WrapHandleRun ( ME_WrapContext wc,
ME_Run run 
)
static

Definition at line 542 of file wrap.c.

543{
544 ME_Run *new_run;
545 int len;
546
547 if (!wc->pRowStart) wc->pRowStart = run;
548 run->pt.x = wc->pt.x;
549 run->pt.y = wc->pt.y;
550 ME_WrapSizeRun( wc, run );
551 len = run->len;
552
553 if (wc->bOverflown) /* just skipping final whitespaces */
554 {
555 /* End paragraph run can't overflow to the next line by itself. */
556 if (run->nFlags & MERF_ENDPARA) return run_next( run );
557
558 if (run->nFlags & MERF_WHITESPACE)
559 {
560 wc->pt.x += run->nWidth;
561 /* skip runs consisting of only whitespaces */
562 return run_next( run );
563 }
564
565 if (run->nFlags & MERF_STARTWHITE)
566 {
567 /* try to split the run at the first non-white char */
568 int black;
569 black = find_non_whitespace( get_text( run, 0 ), run->len, 0 );
570 if (black)
571 {
572 ME_Run *new_run;
573 wc->bOverflown = FALSE;
574 new_run = split_run_extents( wc, run, black );
575 calc_run_extent( wc->context, wc->para,
576 wc->nRow ? wc->nLeftMargin : wc->nFirstMargin, run );
577 ME_InsertRowStart( wc, run );
578 return new_run;
579 }
580 }
581 /* black run: the row goes from pRowStart to the previous run */
582 ME_InsertRowStart( wc, run_prev( run ) );
583 return run;
584 }
585 /* simply end the current row and move on to next one */
586 if (run->nFlags & MERF_ENDROW)
587 {
588 ME_InsertRowStart( wc, run );
589 return run_next( run );
590 }
591
592 /* will current run fit? */
593 if (wc->bWordWrap &&
594 wc->pt.x + run->nWidth - wc->context->pt.x > wc->nAvailWidth)
595 {
596 int loc = wc->context->pt.x + wc->nAvailWidth - wc->pt.x;
597 /* total white run or end para */
598 if (run->nFlags & (MERF_WHITESPACE | MERF_ENDPARA)) {
599 /* let the overflow logic handle it */
600 wc->bOverflown = TRUE;
601 return run;
602 }
603 /* TAB: we can split before */
604 if (run->nFlags & MERF_TAB) {
605 wc->bOverflown = TRUE;
606 if (wc->pRowStart == run)
607 /* Don't split before the start of the run, or we will get an
608 * endless loop. */
609 return run_next( run );
610 else
611 return run;
612 }
613 /* graphics: we can split before, if run's width is smaller than row's width */
614 if ((run->nFlags & MERF_GRAPHICS) && run->nWidth <= wc->nAvailWidth) {
615 wc->bOverflown = TRUE;
616 return run;
617 }
618 /* can we separate out the last spaces ? (to use overflow logic later) */
619 if (run->nFlags & MERF_ENDWHITE)
620 {
621 /* we aren't sure if it's *really* necessary, it's a good start however */
622 int black = reverse_find_non_whitespace( get_text( run, 0 ), len );
623 split_run_extents( wc, run, black );
624 /* handle both parts again */
625 return run;
626 }
627 /* determine the split point by backtracking */
628 new_run = ME_SplitByBacktracking( wc, run, loc );
629 if (new_run == wc->pRowStart)
630 {
631 if (run->nFlags & MERF_STARTWHITE)
632 {
633 /* We had only spaces so far, so we must be on the first line of the
634 * paragraph (or the first line after MERF_ENDROW forced the line
635 * break within the paragraph), since no other lines of the paragraph
636 * start with spaces. */
637
638 /* The lines will only contain spaces, and the rest of the run will
639 * overflow onto the next line. */
640 wc->bOverflown = TRUE;
641 return run;
642 }
643 /* Couldn't split the first run, possible because we have a large font
644 * with a single character that caused an overflow.
645 */
646 wc->pt.x += run->nWidth;
647 return run_next( run );
648 }
649 if (run != new_run) /* found a suitable split point */
650 {
651 wc->bOverflown = TRUE;
652 return new_run;
653 }
654 /* we detected that it's best to split on start of this run */
655 if (wc->bOverflown)
656 return new_run;
657 ERR("failure!\n");
658 /* not found anything - writing over margins is the only option left */
659 }
660 if ((run->nFlags & (MERF_SPLITTABLE | MERF_STARTWHITE))
661 || ((run->nFlags & (MERF_GRAPHICS|MERF_TAB)) && (run != wc->pRowStart)))
662 {
663 wc->pLastSplittableRun = run;
664 }
665 wc->pt.x += run->nWidth;
666 return run_next( run );
667}
#define ERR(fmt,...)
Definition: precomp.h:57
#define MERF_STARTWHITE
Definition: editstr.h:115
#define MERF_ENDROW
Definition: editstr.h:123
static void calc_run_extent(ME_Context *c, const ME_Paragraph *para, int startx, ME_Run *run)
Definition: wrap.c:114
static ME_Run * ME_SplitByBacktracking(ME_WrapContext *wc, ME_Run *run, int loc)
Definition: wrap.c:470
static void ME_WrapSizeRun(ME_WrapContext *wc, ME_Run *run)
Definition: wrap.c:384
static int find_non_whitespace(const WCHAR *s, int len, int start)
Definition: wrap.c:395

Referenced by ME_WrapTextParagraph().

◆ ME_WrapMarkedParagraphs()

BOOL ME_WrapMarkedParagraphs ( ME_TextEditor editor)

Definition at line 1094 of file wrap.c.

1095{
1096 HDC hdc = ITextHost_TxGetDC( editor->texthost );
1097 BOOL ret = wrap_marked_paras_dc( editor, hdc, TRUE );
1099 return ret;
1100}
#define ITextHost_TxGetDC(This)
Definition: editor.h:333
#define ITextHost_TxReleaseDC(This, a)
Definition: editor.h:334
HDC hdc
Definition: main.c:9
static HDC
Definition: imagelist.c:88
int ret
BOOL wrap_marked_paras_dc(ME_TextEditor *editor, HDC hdc, BOOL invalidate)
Definition: wrap.c:1023

Referenced by editor_handle_message(), handle_EM_SETCHARFORMAT(), ME_InvalidateSelection(), ME_Repaint(), ME_RewrapRepaint(), ME_UpdateRepaint(), ME_UpdateScrollBar(), and table_select_next_cell_or_append().

◆ ME_WrapSizeRun()

static void ME_WrapSizeRun ( ME_WrapContext wc,
ME_Run run 
)
static

Definition at line 384 of file wrap.c.

385{
386 /* FIXME compose style (out of character and paragraph styles) here */
387
388 ME_UpdateRunFlags( wc->context->editor, run );
389
390 calc_run_extent( wc->context, wc->para,
391 wc->nRow ? wc->nLeftMargin : wc->nFirstMargin, run );
392}

Referenced by ME_WrapHandleRun().

◆ ME_WrapTextParagraph()

static void ME_WrapTextParagraph ( ME_TextEditor editor,
ME_Context c,
ME_Paragraph para 
)
static

Definition at line 817 of file wrap.c.

818{
819 ME_Run *run;
821 int border = 0;
822 int linespace = 0;
823
824 if (!(para->nFlags & MEPF_REWRAP)) return;
825
826 ME_PrepareParagraphForWrapping( editor, c, para );
827
828 /* Calculate paragraph numbering label */
829 para_num_init( c, para );
830
831 /* For now treating all non-password text as complex for better testing */
832 if (!c->editor->password_char /* &&
833 ScriptIsComplex( tp->member.para.text->szData, tp->member.para.text->nLen, SIC_COMPLEX ) == S_OK */)
834 {
835 if (SUCCEEDED( itemize_para( c, para ) ))
836 shape_para( c, para );
837 }
838
839 wc.context = c;
840 wc.para = para;
841 wc.style = NULL;
842 wc.nParaNumOffset = 0;
843 if (para->nFlags & MEPF_ROWEND)
844 wc.nFirstMargin = wc.nLeftMargin = wc.nRightMargin = 0;
845 else
846 {
847 int dxStartIndent = para->fmt.dxStartIndent;
848 if (para_cell( wc.para )) dxStartIndent += table_row_end( para )->fmt.dxOffset;
849
850 wc.nLeftMargin = ME_twips2pointsX( c, dxStartIndent + para->fmt.dxOffset );
851 wc.nFirstMargin = ME_twips2pointsX( c, dxStartIndent );
852 if (para->fmt.wNumbering)
853 {
855 dxStartIndent = max( ME_twips2pointsX(c, para->fmt.wNumberingTab),
856 para->para_num.width );
857 wc.nFirstMargin += dxStartIndent;
858 }
860
861 if (wc.nFirstMargin < 0) wc.nFirstMargin = 0;
862 if (wc.nLeftMargin < 0) wc.nLeftMargin = 0;
863 }
864 if (c->editor->bEmulateVersion10 && /* v1.0 - 3.0 */
865 para->fmt.dwMask & PFM_TABLE && para->fmt.wEffects & PFE_TABLE)
866 {
867 wc.nFirstMargin += ME_twips2pointsX( c, para->fmt.dxOffset * 2 );
868 }
869 wc.nRow = 0;
870 wc.pt.y = 0;
871 if (para->fmt.dwMask & PFM_SPACEBEFORE)
872 wc.pt.y += ME_twips2pointsY( c, para->fmt.dySpaceBefore );
873 if (!(para->fmt.dwMask & PFM_TABLE && para->fmt.wEffects & PFE_TABLE) &&
874 para->fmt.dwMask & PFM_BORDER)
875 {
877 if (para->fmt.wBorders & 1)
878 {
879 wc.nFirstMargin += border;
880 wc.nLeftMargin += border;
881 }
882 if (para->fmt.wBorders & 2) wc.nRightMargin -= border;
883 if (para->fmt.wBorders & 4) wc.pt.y += border;
884 }
885
886 linespace = ME_GetParaLineSpace( c, para );
887
888 ME_BeginRow( &wc );
889 run = para_first_run( para );
890 while (run)
891 {
892 run = ME_WrapHandleRun( &wc, run );
893 if (wc.nRow && run == wc.pRowStart) wc.pt.y += linespace;
894 }
895 ME_WrapEndParagraph( &wc );
896 if (!(para->fmt.dwMask & PFM_TABLE && para->fmt.wEffects & PFE_TABLE) &&
897 (para->fmt.dwMask & PFM_BORDER) && (para->fmt.wBorders & 8))
898 wc.pt.y += border;
899 if (para->fmt.dwMask & PFM_SPACEAFTER)
900 wc.pt.y += ME_twips2pointsY( c, para->fmt.dySpaceAfter );
901
902 para->nFlags &= ~MEPF_REWRAP;
903 para->nHeight = wc.pt.y;
904 para->nRows = wc.nRow;
905 editor->total_rows += wc.nRow;
906}
int ME_GetParaBorderWidth(const ME_Context *c, int flags) DECLSPEC_HIDDEN
Definition: paint.c:537
void para_num_init(ME_Context *c, ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: para.c:392
#define MEPF_REWRAP
Definition: editstr.h:140
GLint GLint GLsizei GLsizei GLsizei GLint border
Definition: gl.h:1546
#define SUCCEEDED(hr)
Definition: intsafe.h:50
#define c
Definition: ke_i.h:80
#define PFM_SPACEBEFORE
Definition: richedit.h:847
#define PFM_BORDER
Definition: richedit.h:851
#define PFM_SPACEAFTER
Definition: richedit.h:848
LONG dySpaceBefore
Definition: richedit.h:676
LONG dxRightIndent
Definition: richedit.h:671
LONG dySpaceAfter
Definition: richedit.h:676
LONG dxOffset
Definition: richedit.h:672
WORD wBorders
Definition: richedit.h:681
WORD wNumberingTab
Definition: richedit.h:680
ME_Style * style
Definition: wrap.c:37
static int ME_GetParaLineSpace(ME_Context *c, ME_Paragraph *para)
Definition: wrap.c:669
static HRESULT shape_para(ME_Context *c, ME_Paragraph *para)
Definition: wrap.c:800
static HRESULT itemize_para(ME_Context *c, ME_Paragraph *para)
Definition: wrap.c:724
static void ME_PrepareParagraphForWrapping(ME_TextEditor *editor, ME_Context *c, ME_Paragraph *para)
Definition: wrap.c:692
static void ME_WrapEndParagraph(ME_WrapContext *wc)
Definition: wrap.c:371
static ME_Run * ME_WrapHandleRun(ME_WrapContext *wc, ME_Run *run)
Definition: wrap.c:542

Referenced by wrap_marked_paras_dc().

◆ para_range_invalidate()

void para_range_invalidate ( ME_TextEditor editor,
ME_Paragraph start_para,
ME_Paragraph last_para 
)

Definition at line 1102 of file wrap.c.

1104{
1105 RECT rc;
1106 int ofs;
1107
1108 rc = editor->rcFormat;
1109 ofs = editor->vert_si.nPos;
1110
1111 if (start_para)
1112 {
1113 start_para = table_outer_para( start_para );
1114 last_para = table_outer_para( last_para );
1115 rc.top += start_para->pt.y - ofs;
1116 } else {
1117 rc.top += editor->nTotalLength - ofs;
1118 }
1119 if (editor->nTotalLength < editor->nLastTotalLength)
1120 rc.bottom = editor->rcFormat.top + editor->nLastTotalLength - ofs;
1121 else
1122 rc.bottom = editor->rcFormat.top + last_para->pt.y + last_para->nHeight - ofs;
1124}
ME_Paragraph * table_outer_para(ME_Paragraph *para) DECLSPEC_HIDDEN
Definition: table.c:158
#define ITextHost_TxInvalidateRect(This, a, b)
Definition: editor.h:339
SCROLLINFO vert_si
Definition: editstr.h:440
int nLastTotalLength
Definition: editstr.h:397
LONG top
Definition: windef.h:307

Referenced by ME_InvalidateSelection(), and wrap_marked_paras_dc().

◆ reverse_find_non_whitespace()

static int reverse_find_non_whitespace ( const WCHAR s,
int  start 
)
static

Definition at line 408 of file wrap.c.

409{
410 int i;
411 for (i = start; i > 0 && ME_IsWSpace( s[i - 1] ); i--)
412 ;
413
414 return i;
415}

Referenced by ME_MaximizeSplit(), ME_SplitByBacktracking(), and ME_WrapHandleRun().

◆ reverse_find_whitespace()

static int reverse_find_whitespace ( const WCHAR s,
int  start 
)
static

Definition at line 418 of file wrap.c.

419{
420 int i;
421 for (i = start; i > 0 && !ME_IsWSpace( s[i - 1] ); i--)
422 ;
423
424 return i;
425}

Referenced by ME_SplitByBacktracking().

◆ row_create()

static ME_Row * row_create ( int  height,
int  baseline,
int  width 
)
static

Definition at line 176 of file wrap.c.

177{
179
180 item->member.row.nHeight = height;
181 item->member.row.nBaseline = baseline;
182 item->member.row.nWidth = width;
183 return &item->member.row;
184}
ME_DisplayItem * ME_MakeDI(ME_DIType type) DECLSPEC_HIDDEN
Definition: list.c:133
GLint GLint GLsizei GLsizei height
Definition: gl.h:1546
static ATOM item
Definition: dde.c:856

Referenced by ME_InsertRowStart().

◆ shape_para()

static HRESULT shape_para ( ME_Context c,
ME_Paragraph para 
)
static

Definition at line 800 of file wrap.c.

801{
802 ME_Run *run;
803 HRESULT hr;
804
805 for (run = para_first_run( para ); run; run = run_next( run ))
806 {
807 hr = shape_run( c, run );
808 if (FAILED( hr ))
809 {
810 para->nFlags &= ~MEPF_COMPLEX;
811 return hr;
812 }
813 }
814 return hr;
815}
static HRESULT shape_run(ME_Context *c, ME_Run *run)
Definition: wrap.c:64

Referenced by ME_WrapTextParagraph().

◆ shape_run()

static HRESULT shape_run ( ME_Context c,
ME_Run run 
)
static

Definition at line 64 of file wrap.c.

65{
66 HRESULT hr;
67 int i;
68
69 if (!run->glyphs)
70 {
71 run->max_glyphs = 1.5 * run->len + 16; /* This is suggested in the uniscribe documentation */
72 run->max_glyphs = (run->max_glyphs + 7) & ~7; /* Keep alignment simple */
74 }
75
76 if (run->max_clusters < run->len)
77 {
78 heap_free( run->clusters );
79 run->max_clusters = run->len * 2;
80 run->clusters = heap_alloc( run->max_clusters * sizeof(WORD) );
81 }
82
83 select_style( c, run->style );
84 while (1)
85 {
86 hr = ScriptShape( c->hDC, &run->style->script_cache, get_text( run, 0 ), run->len, run->max_glyphs,
87 &run->script_analysis, run->glyphs, run->clusters, run->vis_attrs, &run->num_glyphs );
88 if (hr != E_OUTOFMEMORY) break;
89 if (run->max_glyphs > 10 * run->len) break; /* something has clearly gone wrong */
90 run->max_glyphs *= 2;
92 }
93
94 if (SUCCEEDED(hr))
95 hr = ScriptPlace( c->hDC, &run->style->script_cache, run->glyphs, run->num_glyphs, run->vis_attrs,
96 &run->script_analysis, run->advances, run->offsets, NULL );
97
98 if (SUCCEEDED(hr))
99 {
100 for (i = 0, run->nWidth = 0; i < run->num_glyphs; i++)
101 run->nWidth += run->advances[i];
102 }
103
104 return hr;
105}
HRESULT WINAPI ScriptPlace(HDC hdc, SCRIPT_CACHE *psc, const WORD *pwGlyphs, int cGlyphs, const SCRIPT_VISATTR *psva, SCRIPT_ANALYSIS *psa, int *piAdvance, GOFFSET *pGoffset, ABC *pABC)
Definition: usp10.c:3510
HRESULT WINAPI ScriptShape(HDC hdc, SCRIPT_CACHE *psc, const WCHAR *pwcChars, int cChars, int cMaxGlyphs, SCRIPT_ANALYSIS *psa, WORD *pwOutGlyphs, WORD *pwLogClust, SCRIPT_VISATTR *psva, int *pcGlyphs)
Definition: usp10.c:3321
void select_style(ME_Context *c, ME_Style *s) DECLSPEC_HIDDEN
Definition: style.c:369
int num_glyphs
Definition: editstr.h:170
int max_clusters
Definition: editstr.h:175
ME_Style * style
Definition: editstr.h:159
WORD * clusters
Definition: editstr.h:176
SCRIPT_CACHE script_cache
Definition: editstr.h:77
static BOOL get_run_glyph_buffers(ME_Run *run)
Definition: wrap.c:51

Referenced by shape_para(), and split_run_extents().

◆ split_run_extents()

static ME_Run * split_run_extents ( ME_WrapContext wc,
ME_Run run,
int  nVChar 
)
static

Definition at line 130 of file wrap.c.

131{
132 ME_TextEditor *editor = wc->context->editor;
133 ME_Run *run2;
134 ME_Cursor cursor = { wc->para, run, nVChar };
135
136 assert( run->nCharOfs != -1 );
137 ME_CheckCharOffsets(editor);
138
139 TRACE("Before split: %s(%d, %d)\n", debugstr_run( run ),
140 run->pt.x, run->pt.y);
141
142 run_split( editor, &cursor );
143
144 run2 = cursor.run;
145 run2->script_analysis = run->script_analysis;
146
147 shape_run( wc->context, run );
148 shape_run( wc->context, run2 );
149 calc_run_extent(wc->context, wc->para, wc->nRow ? wc->nLeftMargin : wc->nFirstMargin, run);
150
151 run2->pt.x = run->pt.x + run->nWidth;
152 run2->pt.y = run->pt.y;
153
154 ME_CheckCharOffsets(editor);
155
156 TRACE("After split: %s(%d, %d), %s(%d, %d)\n",
157 debugstr_run( run ), run->pt.x, run->pt.y,
158 debugstr_run( run2 ), run2->pt.x, run2->pt.y);
159
160 return cursor.run;
161}
void ME_CheckCharOffsets(ME_TextEditor *editor) DECLSPEC_HIDDEN
Definition: run.c:173

Referenced by ME_MaximizeSplit(), ME_SplitByBacktracking(), and ME_WrapHandleRun().

◆ update_repaint()

static void update_repaint ( ME_Paragraph para,
struct repaint_range repaint 
)
static

Definition at line 913 of file wrap.c.

914{
915 if (!repaint->start) repaint->start = para;
916 repaint->end = para;
917}
ME_Paragraph * start
Definition: wrap.c:910
ME_Paragraph * end
Definition: wrap.c:910

Referenced by adjust_para_y(), and wrap_marked_paras_dc().

◆ WINE_DEFAULT_DEBUG_CHANNEL()

WINE_DEFAULT_DEBUG_CHANNEL ( richedit  )

◆ wrap_marked_paras_dc()

BOOL wrap_marked_paras_dc ( ME_TextEditor editor,
HDC  hdc,
BOOL  invalidate 
)

Definition at line 1023 of file wrap.c.

1024{
1025 ME_Paragraph *para, *next;
1026 struct wine_rb_entry *entry, *next_entry = NULL;
1027 ME_Context c;
1028 int totalWidth = editor->nTotalWidth, prev_width;
1029 struct repaint_range repaint = { NULL, NULL };
1030
1031 if (!editor->marked_paras.root) return FALSE;
1032
1033 ME_InitContext( &c, editor, hdc );
1034
1035 entry = wine_rb_head( editor->marked_paras.root );
1036 while (entry)
1037 {
1038 para = WINE_RB_ENTRY_VALUE( entry, ME_Paragraph, marked_entry );
1039
1040 /* If the first entry lies inside a table, go back to the start
1041 of the table to ensure cell heights are kept in sync. */
1042 if (!next_entry && para_in_table( para ) && para != table_outer_para( para ))
1043 {
1044 para = table_outer_para( para );
1045 next_entry = entry;
1046 }
1047 else
1048 next_entry = wine_rb_next( entry );
1049
1050 c.pt = para->pt;
1051 prev_width = para->nWidth;
1052 ME_WrapTextParagraph( editor, &c, para );
1053 if (prev_width == totalWidth && para->nWidth < totalWidth)
1054 totalWidth = get_total_width(editor);
1055 else
1056 totalWidth = max(totalWidth, para->nWidth);
1057
1058 update_repaint( para, &repaint );
1059 adjust_para_y( para, &c, &repaint );
1060
1061 if (para_next( para ))
1062 {
1063 if (c.pt.x != para_next( para )->pt.x || c.pt.y != para_next( para )->pt.y ||
1064 para_in_table( para ))
1065 {
1066 next = para;
1067 while (para_next( next ) && &next->marked_entry != next_entry &&
1068 next != &editor->pBuffer->pLast->member.para)
1069 {
1070 update_repaint( para_next( next ), &repaint );
1071 para_next( next )->pt = c.pt;
1072 adjust_para_y( para_next( next ), &c, &repaint );
1073 next = para_next( next );
1074 }
1075 }
1076 }
1077 entry = next_entry;
1078 }
1079 wine_rb_clear( &editor->marked_paras, NULL, NULL );
1080
1081 editor->sizeWindow.cx = c.rcView.right-c.rcView.left;
1082 editor->sizeWindow.cy = c.rcView.bottom-c.rcView.top;
1083
1084 editor->nTotalLength = editor->pBuffer->pLast->member.para.pt.y;
1085 editor->nTotalWidth = totalWidth;
1086
1088
1089 if (invalidate && (repaint.start || editor->nTotalLength < editor->nLastTotalLength))
1090 para_range_invalidate( editor, repaint.start, repaint.end);
1091 return !!repaint.start;
1092}
void ME_DestroyContext(ME_Context *c)
Definition: context.c:44
void ME_InitContext(ME_Context *c, ME_TextEditor *editor, HDC hDC)
Definition: context.c:23
#define pt(x, y)
Definition: drawing.c:79
int get_total_width(ME_TextEditor *editor) DECLSPEC_HIDDEN
Definition: para.c:69
uint32_t entry
Definition: isohybrid.c:63
#define WINE_RB_ENTRY_VALUE(element, type, field)
Definition: rbtree.h:31
static struct wine_rb_entry * wine_rb_head(struct wine_rb_entry *iter)
Definition: rbtree.h:103
static void wine_rb_clear(struct wine_rb_tree *tree, wine_rb_traverse_func_t *callback, void *context)
Definition: rbtree.h:191
static struct wine_rb_entry * wine_rb_next(struct wine_rb_entry *iter)
Definition: rbtree.h:117
LONG cx
Definition: kdterminal.h:27
LONG cy
Definition: kdterminal.h:28
union tagME_DisplayItem::@558 member
ME_Paragraph para
Definition: editstr.h:261
ME_DisplayItem * pLast
Definition: editstr.h:267
ME_TextBuffer * pBuffer
Definition: editstr.h:388
struct wine_rb_tree marked_paras
Definition: editstr.h:451
Definition: rbtree.h:36
static void invalidate()
static void ME_WrapTextParagraph(ME_TextEditor *editor, ME_Context *c, ME_Paragraph *para)
Definition: wrap.c:817
void para_range_invalidate(ME_TextEditor *editor, ME_Paragraph *start_para, ME_Paragraph *last_para)
Definition: wrap.c:1102
static void adjust_para_y(ME_Paragraph *para, ME_Context *c, struct repaint_range *repaint)
Definition: wrap.c:919

Referenced by fnTextSrv_TxDraw(), fnTextSrv_TxGetNaturalSize(), and ME_WrapMarkedParagraphs().