You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tqt3/src/kernel/qtextlayout.cpp

644 lines
17 KiB

/****************************************************************************
**
** ???
**
** Copyright (C) 2005-2008 Trolltech ASA. All rights reserved.
**
** This file is part of the kernel module of the TQt GUI Toolkit.
**
** This file may be used under the terms of the GNU General
** Public License versions 2.0 or 3.0 as published by the Free
** Software Foundation and appearing in the files LICENSE.GPL2
** and LICENSE.GPL3 included in the packaging of this file.
** Alternatively you may (at your option) use any later version
** of the GNU General Public License if such license has been
** publicly approved by Trolltech ASA (or its successors, if any)
** and the KDE Free TQt Foundation.
**
** Please review the following information to ensure GNU General
** Public Licensing requirements will be met:
** http://trolltech.com/products/qt/licenses/licensing/opensource/.
** If you are unsure which license is appropriate for your use, please
** review the following information:
** http://trolltech.com/products/qt/licenses/licensing/licensingoverview
** or contact the sales department at sales@trolltech.com.
**
** This file may be used under the terms of the Q Public License as
** defined by Trolltech ASA and appearing in the file LICENSE.TQPL
** included in the packaging of this file. Licensees holding valid TQt
** Commercial licenses may use this file in accordance with the TQt
** Commercial License Agreement provided with the Software.
**
** This file is provided "AS IS" with NO WARRANTY OF ANY KIND,
** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR
** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted
** herein.
**
**********************************************************************/
#include "qtextlayout_p.h"
#include "qtextengine_p.h"
#include <ntqfont.h>
#include <ntqapplication.h>
#include <ntqpainter.h>
TQRect TQTextItem::rect() const
{
TQScriptItem& si = engine->items[item];
return TQRect( si.x, si.y, si.width, si.ascent+si.descent );
}
int TQTextItem::x() const
{
return engine->items[item].x;
}
int TQTextItem::y() const
{
return engine->items[item].y;
}
int TQTextItem::width() const
{
return engine->items[item].width;
}
int TQTextItem::ascent() const
{
return engine->items[item].ascent;
}
int TQTextItem::descent() const
{
return engine->items[item].descent;
}
void TQTextItem::setWidth( int w )
{
engine->items[item].width = w;
}
void TQTextItem::setAscent( int a )
{
engine->items[item].ascent = a;
}
void TQTextItem::setDescent( int d )
{
engine->items[item].descent = d;
}
int TQTextItem::from() const
{
return engine->items[item].position;
}
int TQTextItem::length() const
{
return engine->length(item);
}
int TQTextItem::cursorToX( int *cPos, Edge edge ) const
{
int pos = *cPos;
TQScriptItem *si = &engine->items[item];
engine->shape( item );
advance_t *advances = engine->advances( si );
GlyphAttributes *glyphAttributes = engine->glyphAttributes( si );
unsigned short *logClusters = engine->logClusters( si );
int l = engine->length( item );
if ( pos > l )
pos = l;
if ( pos < 0 )
pos = 0;
int glyph_pos = pos == l ? si->num_glyphs : logClusters[pos];
if ( edge == Trailing ) {
// trailing edge is leading edge of next cluster
while ( glyph_pos < si->num_glyphs && !glyphAttributes[glyph_pos].clusterStart )
glyph_pos++;
}
int x = 0;
bool reverse = engine->items[item].analysis.bidiLevel % 2;
if ( reverse ) {
for ( int i = si->num_glyphs-1; i >= glyph_pos; i-- )
x += advances[i];
} else {
for ( int i = 0; i < glyph_pos; i++ )
x += advances[i];
}
// tqDebug("cursorToX: pos=%d, gpos=%d x=%d", pos, glyph_pos, x );
*cPos = pos;
return x;
}
int TQTextItem::xToCursor( int x, CursorPosition cpos ) const
{
TQScriptItem *si = &engine->items[item];
engine->shape( item );
advance_t *advances = engine->advances( si );
unsigned short *logClusters = engine->logClusters( si );
int l = engine->length( item );
bool reverse = si->analysis.bidiLevel % 2;
if ( x < 0 )
return reverse ? l : 0;
if ( reverse ) {
int width = 0;
for ( int i = 0; i < si->num_glyphs; i++ ) {
width += advances[i];
}
x = -x + width;
}
int cp_before = 0;
int cp_after = 0;
int x_before = 0;
int x_after = 0;
int lastCluster = 0;
for ( int i = 1; i <= l; i++ ) {
int newCluster = i < l ? logClusters[i] : si->num_glyphs;
if ( newCluster != lastCluster ) {
// calculate cluster width
cp_before = cp_after;
x_before = x_after;
cp_after = i;
for ( int j = lastCluster; j < newCluster; j++ )
x_after += advances[j];
// tqDebug("cluster boundary: lastCluster=%d, newCluster=%d, x_before=%d, x_after=%d",
// lastCluster, newCluster, x_before, x_after );
if ( x_after > x )
break;
lastCluster = newCluster;
}
}
bool before = ( cpos == OnCharacters || (x - x_before) < (x_after - x) );
// tqDebug("got cursor position for %d: %d/%d, x_ba=%d/%d using %d",
// x, cp_before,cp_after, x_before, x_after, before ? cp_before : cp_after );
return before ? cp_before : cp_after;
}
bool TQTextItem::isRightToLeft() const
{
return (engine->items[item].analysis.bidiLevel % 2);
}
bool TQTextItem::isObject() const
{
return engine->items[item].isObject;
}
bool TQTextItem::isSpace() const
{
return engine->items[item].isSpace;
}
bool TQTextItem::isTab() const
{
return engine->items[item].isTab;
}
TQTextLayout::TQTextLayout()
:d(0) {}
TQTextLayout::TQTextLayout( const TQString& string, TQPainter *p )
{
TQFontPrivate *f = p ? ( p->pfont ? p->pfont->d : p->cfont.d ) : TQApplication::font().d;
d = new TQTextEngine( (string.isNull() ? (const TQString&)TQString::fromLatin1("") : string), f );
}
TQTextLayout::TQTextLayout( const TQString& string, const TQFont& fnt )
{
d = new TQTextEngine( (string.isNull() ? (const TQString&)TQString::fromLatin1("") : string), fnt.d );
}
TQTextLayout::~TQTextLayout()
{
delete d;
}
void TQTextLayout::setText( const TQString& string, const TQFont& fnt )
{
delete d;
d = new TQTextEngine( (string.isNull() ? (const TQString&)TQString::fromLatin1("") : string), fnt.d );
}
/* add an additional item boundary eg. for style change */
void TQTextLayout::setBoundary( int strPos )
{
if ( strPos <= 0 || strPos >= (int)d->string.length() )
return;
int itemToSplit = 0;
while ( itemToSplit < d->items.size() && d->items[itemToSplit].position <= strPos )
itemToSplit++;
itemToSplit--;
if ( d->items[itemToSplit].position == strPos ) {
// already a split at the requested position
return;
}
d->splitItem( itemToSplit, strPos - d->items[itemToSplit].position );
}
int TQTextLayout::numItems() const
{
return d->items.size();
}
TQTextItem TQTextLayout::itemAt( int i ) const
{
return TQTextItem( i, d );
}
TQTextItem TQTextLayout::findItem( int strPos ) const
{
if ( strPos == 0 && d->items.size() )
return TQTextItem( 0, d );
// ## TODO use bsearch
for ( int i = d->items.size()-1; i >= 0; --i ) {
if ( d->items[i].position < strPos )
return TQTextItem( i, d );
}
return TQTextItem();
}
void TQTextLayout::beginLayout( TQTextLayout::LayoutMode m )
{
d->items.clear();
TQTextEngine::Mode mode = TQTextEngine::Full;
if (m == NoBidi)
mode = TQTextEngine::NoBidi;
else if (m == SingleLine)
mode = TQTextEngine::SingleLine;
d->itemize( mode );
d->currentItem = 0;
d->firstItemInLine = -1;
}
void TQTextLayout::beginLine( int width )
{
d->lineWidth = width;
d->widthUsed = 0;
d->firstItemInLine = -1;
}
bool TQTextLayout::atEnd() const
{
return d->currentItem >= d->items.size();
}
TQTextItem TQTextLayout::nextItem()
{
d->currentItem++;
if ( d->currentItem >= d->items.size() )
return TQTextItem();
d->shape( d->currentItem );
return TQTextItem( d->currentItem, d );
}
TQTextItem TQTextLayout::currentItem()
{
if ( d->currentItem >= d->items.size() )
return TQTextItem();
d->shape( d->currentItem );
return TQTextItem( d->currentItem, d );
}
/* ## maybe also currentItem() */
void TQTextLayout::setLineWidth( int newWidth )
{
d->lineWidth = newWidth;
}
int TQTextLayout::lineWidth() const
{
return d->lineWidth;
}
int TQTextLayout::widthUsed() const
{
return d->widthUsed;
}
int TQTextLayout::availableWidth() const
{
return d->lineWidth - d->widthUsed;
}
/* returns true if completely added */
TQTextLayout::Result TQTextLayout::addCurrentItem()
{
if ( d->firstItemInLine == -1 )
d->firstItemInLine = d->currentItem;
TQScriptItem &current = d->items[d->currentItem];
d->shape( d->currentItem );
d->widthUsed += current.width;
// tqDebug("trying to add item %d with width %d, remaining %d", d->currentItem, current.width, d->lineWidth-d->widthUsed );
d->currentItem++;
return (d->widthUsed <= d->lineWidth
|| (d->currentItem < d->items.size() && d->items[d->currentItem].isSpace)) ? Ok : LineFull;
}
TQTextLayout::Result TQTextLayout::endLine( int x, int y, int alignment,
int *ascent, int *descent, int *lineLeft, int *lineRight )
{
int available = d->lineWidth;
int numRuns = 0;
int numSpaceItems = 0;
TQ_UINT8 _levels[128];
int _visual[128];
TQ_UINT8 *levels = _levels;
int *visual = _visual;
int i;
TQTextLayout::Result result = LineEmpty;
// tqDebug("endLine x=%d, y=%d, first=%d, current=%d lw=%d wu=%d", x, y, d->firstItemInLine, d->currentItem, d->lineWidth, d->widthUsed );
int width_nobreak_found = d->widthUsed;
if ( d->firstItemInLine == -1 )
goto end;
if ( !(alignment & (TQt::SingleLine|TQt::IncludeTrailingSpaces))
&& d->currentItem > d->firstItemInLine && d->items[d->currentItem-1].isSpace ) {
int i = d->currentItem-1;
while ( i > d->firstItemInLine && d->items[i].isSpace ) {
numSpaceItems++;
d->widthUsed -= d->items[i--].width;
}
}
if ( (alignment & (TQt::WordBreak|TQt::BreakAnywhere)) &&
d->widthUsed > d->lineWidth ) {
// find linebreak
// even though we removed trailing spaces the line was too wide. We'll have to break at an earlier
// position. To not confuse the layouting below, reset the number of space items
numSpaceItems = 0;
bool breakany = alignment & TQt::BreakAnywhere;
const TQCharAttributes *attrs = d->attributes();
int w = 0;
int itemWidth = 0;
int breakItem = d->firstItemInLine;
int breakPosition = -1;
#if 0
// we iterate backwards or forward depending on what we guess is closer
if ( d->widthUsed - d->lineWidth < d->lineWidth ) {
// backwards search should be faster
} else
#endif
{
int tmpWidth = 0;
int swidth = 0;
// forward search is probably faster
for ( int i = d->firstItemInLine; i < d->currentItem; i++ ) {
const TQScriptItem *si = &d->items[i];
int length = d->length( i );
const TQCharAttributes *itemAttrs = attrs + si->position;
advance_t *advances = d->advances( si );
unsigned short *logClusters = d->logClusters( si );
int lastGlyph = 0;
int tmpItemWidth = 0;
// tqDebug("looking for break in item %d, isSpace=%d", i, si->isSpace );
if(si->isSpace && !(alignment & (TQt::SingleLine|TQt::IncludeTrailingSpaces))) {
swidth += si->width;
} else {
tmpWidth += swidth;
swidth = 0;
for ( int pos = 0; pos < length; pos++ ) {
// tqDebug("advance=%d, w=%d, tmpWidth=%d, softbreak=%d, whitespace=%d",
// *advances, w, tmpWidth, itemAttrs->softBreak, itemAttrs->whiteSpace );
int glyph = logClusters[pos];
if ( lastGlyph != glyph ) {
while ( lastGlyph < glyph )
tmpItemWidth += advances[lastGlyph++];
if ( breakPosition != -1 && w + tmpWidth + tmpItemWidth > d->lineWidth ) {
// tqDebug("found break at w=%d, tmpWidth=%d, tmpItemWidth=%d", w, tmpWidth, tmpItemWidth);
d->widthUsed = w;
goto found;
}
}
if ( (itemAttrs->softBreak ||
( breakany && itemAttrs->charStop ) ) &&
(i != d->firstItemInLine || pos != 0) ) {
if ( breakItem != i )
itemWidth = 0;
if (itemAttrs->softBreak)
breakany = FALSE;
breakItem = i;
breakPosition = pos;
// tqDebug("found possible break at item %d, position %d (absolute=%d), w=%d, tmpWidth=%d, tmpItemWidth=%d", breakItem, breakPosition, d->items[breakItem].position+breakPosition, w, tmpWidth, tmpItemWidth);
w += tmpWidth + tmpItemWidth;
itemWidth += tmpItemWidth;
tmpWidth = 0;
tmpItemWidth = 0;
}
itemAttrs++;
}
while ( lastGlyph < si->num_glyphs )
tmpItemWidth += advances[lastGlyph++];
tmpWidth += tmpItemWidth;
if ( w + tmpWidth > d->lineWidth ) {
d->widthUsed = w;
goto found;
}
}
}
}
found:
// no valid break point found
if ( breakPosition == -1 ) {
d->widthUsed = width_nobreak_found;
goto nobreak;
}
// tqDebug("linebreak at item %d, position %d, wu=%d", breakItem, breakPosition, d->widthUsed );
// split the line
if ( breakPosition > 0 ) {
// int length = d->length( breakItem );
// tqDebug("splitting item, itemWidth=%d", itemWidth);
// not a full item, need to break
d->splitItem( breakItem, breakPosition );
d->currentItem = breakItem+1;
} else {
d->currentItem = breakItem;
}
}
result = Ok;
nobreak:
// position the objects in the line
available -= d->widthUsed;
numRuns = d->currentItem - d->firstItemInLine - numSpaceItems;
if ( numRuns > 127 ) {
levels = new TQ_UINT8[numRuns];
visual = new int[numRuns];
}
// tqDebug("reordering %d runs, numSpaceItems=%d", numRuns, numSpaceItems );
for ( i = 0; i < numRuns; i++ ) {
levels[i] = d->items[i+d->firstItemInLine].analysis.bidiLevel;
// tqDebug(" level = %d", d->items[i+d->firstItemInLine].analysis.bidiLevel );
}
d->bidiReorder( numRuns, levels, visual );
end:
// ### FIXME
if ( alignment & TQt::AlignJustify ) {
// #### justify items
alignment = TQt::AlignAuto;
}
if ( (alignment & TQt::AlignHorizontal_Mask) == TQt::AlignAuto )
alignment = TQt::AlignLeft;
if ( alignment & TQt::AlignRight )
x += available;
else if ( alignment & TQt::AlignHCenter )
x += available/2;
int asc = ascent ? *ascent : 0;
int desc = descent ? *descent : 0;
for ( i = 0; i < numRuns; i++ ) {
TQScriptItem &si = d->items[d->firstItemInLine+visual[i]];
asc = TQMAX( asc, si.ascent );
desc = TQMAX( desc, si.descent );
}
int left = x;
for ( i = 0; i < numRuns; i++ ) {
TQScriptItem &si = d->items[d->firstItemInLine+visual[i]];
// tqDebug("positioning item %d with width %d (from=%d/length=%d) at %d", d->firstItemInLine+visual[i], si.width, si.position,
// d->length(d->firstItemInLine+visual[i]), x );
si.x = x;
si.y = y + asc;
x += si.width;
}
int right = x;
if ( numSpaceItems ) {
if ( d->items[d->firstItemInLine+numRuns].analysis.bidiLevel % 2 ) {
x = left;
for ( i = 0; i < numSpaceItems; i++ ) {
TQScriptItem &si = d->items[d->firstItemInLine + numRuns + i];
x -= si.width;
si.x = x;
si.y = y + asc;
}
} else {
for ( i = 0; i < numSpaceItems; i++ ) {
TQScriptItem &si = d->items[d->firstItemInLine + numRuns + i];
si.x = x;
si.y = y + asc;
x += si.width;
}
}
}
if ( lineLeft )
*lineLeft = left;
if ( lineRight )
*lineRight = right;
if ( ascent )
*ascent = asc;
if ( descent )
*descent = desc;
if (levels != _levels)
delete []levels;
if (visual != _visual)
delete []visual;
return result;
}
void TQTextLayout::endLayout()
{
// nothing to do currently
}
int TQTextLayout::nextCursorPosition( int oldPos, CursorMode mode ) const
{
// tqDebug("looking for next cursor pos for %d", oldPos );
const TQCharAttributes *attributes = d->attributes();
int len = d->string.length();
if ( oldPos >= len )
return oldPos;
oldPos++;
if ( mode == SkipCharacters ) {
while ( oldPos < len && !attributes[oldPos].charStop )
oldPos++;
} else {
while ( oldPos < len && !attributes[oldPos].wordStop && !attributes[oldPos-1].whiteSpace )
oldPos++;
}
// tqDebug(" -> %d", oldPos );
return oldPos;
}
int TQTextLayout::previousCursorPosition( int oldPos, CursorMode mode ) const
{
// tqDebug("looking for previous cursor pos for %d", oldPos );
const TQCharAttributes *attributes = d->attributes();
if ( oldPos <= 0 )
return 0;
oldPos--;
if ( mode == SkipCharacters ) {
while ( oldPos && !attributes[oldPos].charStop )
oldPos--;
} else {
while ( oldPos && !attributes[oldPos].wordStop && !attributes[oldPos-1].whiteSpace )
oldPos--;
}
// tqDebug(" -> %d", oldPos );
return oldPos;
}
bool TQTextLayout::validCursorPosition( int pos ) const
{
const TQCharAttributes *attributes = d->attributes();
if ( pos < 0 || pos > (int)d->string.length() )
return FALSE;
return attributes[pos].charStop;
}
void TQTextLayout::setDirection(TQChar::Direction dir)
{
d->direction = dir;
}