转载:http://blog.csdn.net/jianlizhao66/article/details/1480457
我们获得了每个字形的宽度数组piAdvances,以及这个RUN所占用的总宽度abc。
piVdvances对应于每个字符,它取得了每个字形所占用宽度。
如果我们以行为单位来绘制文字,我们可以以一个循环来取得每个RUN的宽度信息,并一直相加,假设我们己取得了一些run的宽度和,假设这个宽度和为LineWidth.当我们再取得链表中下一个run的宽度(假设为lo)与这个宽度相加后超过一个行的宽度。我们假设这个run的索引为n即在链表中的节点顺序由头节点向后遍历的顺序为第n个。
我们可以中断这个run即把这个run分为两个run,所谓划分实际上也就是把这个run的字符分为两部分。
可以这样实现:
hr = ScriptGetLogicalWidths(&pRun->analysis,
pRun->iLen,
cGlyphs,
piAdvances,
pClusters,
pVisattrs,
logwidths);
ScriptGetLogicalWidhts函数转换一个指定字体的字形向前宽度为逻辑宽度。
HRESULT WINAPI ScriptGetLogicalWidths(
const SCRIPT_ANALYSIS *psa,
int cChars,
int cGlyphs,
const int *piGlyphWidth,
const WORD *pwLogClust,
const SCRIPT_VISATTR *psva,
int *piDx,
);
psa
[in]一个 SCRIPT_ANALYSIS 结构指针.
cChars
[in] 在RUN中的逻辑编码点数量
cGlyphs
[in] 在一个RUN中的字形数量
piGlyphWidth
[in] 字形向前宽度的数组指针
pwLogClust
[in] 逻辑Cluster的数组指针
psva
[in] SCRIPT_VISATTR 结构指针
piDx
[out] 逻辑宽度的数组指针
可以看出这个函数的最后一个参数为输出参数,它代表逻辑宽度的数组。即每个字形所占用的宽度数组。
然后我们可以以一个循环来判断具体到哪个字符时满足不超过窗口宽度的条件。
iChars = 0;
iWidth = 0;
while(iChars < pRun->iLen && iWidth + logwidths[iChars] < iMaxWidth)
{
iWidth += logwidths[iChars];
iChars++;
}
pRun代表我们当前的run。iWidth代表当前run中某个字符的逻辑宽度,iMaxWidth代表窗口剩余的宽度即剩余了iMaxWidth的宽度,但这个宽度不足以显示这个run.
通过上面这个循环,我们找到了这个字符的位置。
if(iChars < pRun->iLen )
{
pNewRun = new RUN;
*pNewRun = *pRun;
pRun->pNext = pNewRun;
pRun->iLen = iChars;
pNewRun->iLen -= iChars;
}
我们已经划分了这个RUN。
第二步已经完成,总结第二步,主要目的是确定一行(窗口宽度)要显示多少个RUN,并进行相应的划分。