Windows2003-3790/base/mvdm/wow16/write/d_form1.c

3022 lines
104 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/************************************************************/
/* Windows Write, Copyright 1985-1992 Microsoft Corporation */
/************************************************************/
/*--- Module not really used, just the idea behind FORMAT.ASM ---*/
#define NOCLIPBOARD
#define NOGDICAPMASKS
#define NOCTLMGR
#define NOVIRTUALKEYCODES
#define NOWINMESSAGES
#define NOWINSTYLES
#define NOKEYSTATE
#define NOSYSCOMMANDS
#define NORASTEROPS
#define NOSHOWWINDOW
#define NOCLIPBOARD
#define NOWINSTYLES
#define NOSYSMETRICS
#define NOMENUS
#define NOATOM
#define NOCREATESTRUCT
#define NODRAWTEXT
#define NOMB
#define NOMEMMGR
#define NOMETAFILE
#define NOWH
#define NOWINOFFSETS
#define NOWNDCLASS
#include <windows.h>
/* #include "wwsmall.h" */
#include "mw.h"
#include "cmddefs.h"
#include "fmtdefs.h"
#include "propdefs.h"
#include "ch.h"
#include "docdefs.h"
#include "ffdefs.h"
#include "filedefs.h"
#include "fkpdefs.h"
#include "dispdefs.h"
#include "scrndefs.h"
#include "macro.h"
#include "debug.h"
#include "fontdefs.h"
#include "str.h"
#include "wwdefs.h"
#ifdef DBCS
#include "dbcs.h"
/* We move several hard code Kanji code table from this source file
to kanji.h as external variables. Define CODE_TABLE will define
those variables */
#define CODE_TABLE
#include "kanji.h"
#endif
#if defined(TAIWAN) || defined(PRC)
int WINAPI GetFontAssocStatus(HDC);
#endif
#ifdef DFLI
#define Dfli(x) x /* Enable debug-format-line info */
#else
#define Dfli(x)
#endif
#ifdef CASHMERE
#define cchSmcapMax 16
#endif /* CASHMERE */
static int ichpFormat;
#ifdef CASHMERE
static CHAR mptlcch[] = " .-_";
#endif /* CASHMERE */
#if defined(JAPAN) || defined(KOREA) // added 22 Jun. 1992 by Hiraisi
int iWidenChar; /* counter for widened characters except (KANJI) space */
/* Ex.) DBCS, Japanese KANA */
int iNonWideSpaces;
#elif defined(TAIWAN) || defined(PRC)// Daniel/MSTC, 1993/02/25, for jcBoth
int iWidenChar; /* counter for widened characters except (KANJI) space */
/* Ex.) DBCS, Japanese KANA */
int iNonWideSpaces;
extern int vfWordWrap;
#define FKana(_ch) FALSE
#endif
/* T-HIROYN sync format.asm */
/*extern int docHelp;*/
extern struct FLI vfli;
extern struct CHP (**vhgchpFormat)[];
extern int ichpMacFormat;
extern struct CHP vchpAbs;
extern struct PAP vpapAbs;
extern struct SEP vsepAbs;
extern struct SEP vsepPage;
extern struct CHP vchpNormal;
extern struct DOD (**hpdocdod)[];
extern typeCP vcpLimSectCache;
extern typeCP vcpFirstParaCache;
extern typeCP vcpLimParaCache;
extern typeCP vcpFetch;
extern int vichFetch;
extern int vccpFetch;
extern CHAR *vpchFetch;
extern int vcchFetch;
extern int vftc;
extern int ypSubSuper;
extern int ypSubSuperPr;
extern HDC vhMDC;
extern HDC vhDCPrinter;
extern int dxpLogInch;
extern int dypLogInch;
extern int dxaPrPage;
extern int dyaPrPage;
extern int dxpPrPage;
extern int dypPrPage;
extern int dypMax;
extern struct FMI vfmiScreen, vfmiPrint;
extern int vfOutOfMemory;
extern CHAR vchDecimal; /* "decimal point" character */
extern int vzaTabDflt; /* width of default tab */
#if defined(JAPAN) || defined(KOREA)
extern int vfWordWrap; /*t-Yoshio WordWrap flag*/
#endif
#ifdef CASHMERE
extern int vfVisiMode;
#endif /* CASHMERE */
/* F O R M A T L I N E */
FormatLine(doc, cp, ichCp, cpMac, flm)
int doc;
typeCP cp;
int ichCp;
typeCP cpMac;
int flm;
{
/* Fills up vfli with a line of text */
int near Justify(struct IFI *, unsigned, int);
int near FGrowFormatHeap(void);
int near FFirstIch(int);
#ifdef DBCS
BOOL near FAdmitCh1(CHAR);
BOOL near FAdmitCh2(CHAR, CHAR);
BOOL near FOptAdmitCh1(CHAR);
BOOL near FOptAdmitCh2(CHAR, CHAR);
int DBCSDxpFromCh(int,int,int);
#endif
struct IFI ifi;
struct TBD *ptbd;
struct CHP chpLocal;
int xpTab;
#ifdef CASHMERE
int dypBefore;
#endif /* CASHMERE */
int dypAscent;
int dypDescent;
int dypAscentMac;
int dypDescentMac;
unsigned xaLeft;
unsigned xaRight;
struct PAP *ppap;
struct SEP *psep;
int fFlmPrinting = flm & flmPrinting;
int dxaFormat;
int dyaFormat;
int dxpFormat;
int dypFormat;
int ypSubSuperFormat;
int fTruncated = false; /* if the run was truncated */
int ichpNRH;
#ifdef DBCS
int dichSpaceAdjust;
int dypLeading;
int dypIntLeading;
int dypPAscent; /* Pure Ascent */
int dypLeadingMac;
int dypIntLeadingMac;
int dypPAscentMac;
BOOL fKanjiBreakOppr = false;
BOOL fKanjiPrevious = false;
/* true iff we already have a hanging character on the line. */
BOOL fKanjiHanging = false;
/* true iff the first and second bytes of a kanji character
were in two different runs. */
BOOL fOddBoundary = false;
typeCP cpFetchSave;
typeCP cpSeqFetch;
int ichFetchSave;
int cchUsedSave;
extern int utCur;
extern int dxaAdjustPer5Ch;
extern unsigned cxaCh;
#endif /* ifdef DBCS */
#ifdef CASHMERE
struct FNTB **hfntb;
int fVisiMode;
#endif /* CASHMERE */
/* Check for fli current */
if (vfli.doc == doc && vfli.cpMin == cp && vfli.ichCpMin == ichCp &&
vfli.flm == flm)
{
/* Just did this one */
return;
}
#ifdef JAPAN // added by Hiraisi
// When printing, WRITE doesn't redraw the screen.
{
extern BOOL fPrinting;
if( fPrinting && !fFlmPrinting )
return;
}
#endif
Scribble(5, 'F');
bltc(&vfli, 0, cwFLIBase);
/* This means:
vfli.fSplat = false;
vfli.dcpDepend = 0;
vfli.ichCpMac = 0;
vfli.dypLine = 0;
vfli.dypAfter = 0;
vfli.dypFont = 0;
vfli.dypBase = 0;
*/
/* vfSplatNext = FALSE; No longer used. */
/* Rest of format loads up cache with current data */
vfli.doc = doc;
vfli.cpMin = cp;
vfli.ichCpMin = ichCp;
vfli.flm = flm;
if (cp > cpMac)
{
/* Space after the endmark. Reset the cache because the footnotes come
at the same cp in the footnote window */
vfli.doc = docNil;
vfli.cpMac = cp;
vfli.rgdxp[0] = 0;
/* Line after end mark is taller than screen */
#ifdef CASHMERE
vfli.dypBase = vfli.dypFont = vfli.dypAfter = ((vfli.dypLine = dypMax)
>> 1);
#else /* not CASHMERE */
vfli.dypBase = vfli.dypFont = ((vfli.dypLine = dypMax) >> 1);
#endif /* not CASHMERE */
Scribble(5, ' ');
return;
}
/* Initialize run tables */
ichpFormat = 0;
/* Cache section and paragraph properties */
#ifdef CASHMERE
hfntb = (**hpdocdod)[doc].hfntb;
if (hfntb == 0 || cp < (**hfntb).rgfnd[0].cpFtn)
{
/* Normal text */
CacheSect(doc, cp);
}
else
{
/* Footnote section properties come from the section of the footnote
reference. */
CacheSect(doc, CpRefFromFtn(doc, cp));
}
#else /* not CASHMERE */
CacheSect(doc, cp);
#endif /* not CASHMERE */
psep = &vsepAbs;
CachePara(doc, cp);
ppap = &vpapAbs;
/* Now we have:
ppap paragraph properties
psep division properties
*/
if (ppap->fGraphics)
{
/* Format a picture paragraph in a special way (see picture.c) */
FormatGraphics(doc, cp, ichCp, cpMac, flm);
Scribble(5, ' ');
return;
}
/* Assure we have a good memory DC for font stuff */
ValidateMemoryDC();
if (vhMDC == NULL || vhDCPrinter == NULL)
{
Scribble(5, ' ');
return;
}
#ifdef CASHMERE
/* When printing, don't show visible characters */
fVisiMode = vfVisiMode && !fFlmPrinting;
#endif /* CASHMERE */
bltc(&ifi, 0, cwIFI);
/* This means:
ifi.ich = 0;
ifi.ichPrev = 0;
ifi.ichFetch = 0;
ifi.cchSpace = 0;
ifi.ichLeft = 0;
*/
ifi.jc = jcTabLeft;
ifi.fPrevSpace = true;
/* Set up some variables that have different value depending on whether we
are printing or not. */
if (fFlmPrinting)
{
dxaFormat = dxaPrPage;
dyaFormat = dyaPrPage;
dxpFormat = dxpPrPage;
dypFormat = dypPrPage;
ypSubSuperFormat = ypSubSuperPr;
}
else
{
dxaFormat = dyaFormat = czaInch;
dxpFormat = dxpLogInch;
dypFormat = dypLogInch;
ypSubSuperFormat = ypSubSuper;
}
/* Calculate line height and width measures. Compute
xaLeft left indent 0 means at left margin
xaRight width of column measured from left margin (not from left
indent).
*/
xaLeft = ppap->dxaLeft;
/* If this is the first line of a paragraph, adjust xaLeft for the first
line indent. (Also, set dypBefore, since its handy.) */
if (cp == vcpFirstParaCache)
{
xaLeft += ppap->dxaLeft1;
#ifdef CASHMERE
dypBefore = MultDiv(ppap->dyaBefore, dypLogInch, czaInch);
#endif /* CASHMERE */
}
#ifdef CASHMERE
else
{
dypBefore = 0;
}
#endif /* CASHMERE */
/* Now, set xaRight (width measured in twips). */
#ifdef CASHMERE
xaRight = (ppap->rhc ? vsepPage.xaMac - vsepPage.dxaGutter :
psep->dxaText) - ppap->dxaRight;
#else /* not CASHMERE */
xaRight = psep->dxaText - ppap->dxaRight;
#endif /* not CASHMERE */
/* Do necessary checks on xaLeft and xaRight */
if (xaRight > xaRightMax)
{
xaRight = xaRightMax;
}
if (xaLeft > xaRightMax)
{
xaLeft = xaRightMax;
}
if (xaLeft < 0)
{
xaLeft = 0;
}
if (xaRight < xaLeft)
{
xaRight = xaLeft + 1;
}
vfli.xpLeft = ifi.xp = ifi.xpLeft = MultDiv(xaLeft, dxpFormat, dxaFormat);
vfli.xpMarg = ifi.xpRight = MultDiv(xaRight, dxpFormat, dxaFormat);
ifi.xpPr = MultDiv(xaLeft, dxpPrPage, dxaPrPage);
ifi.xpPrRight = MultDiv(xaRight, dxpPrPage, dxaPrPage);
#ifndef JAPAN // added by Hiraisi (BUG#3542)
#ifdef DBCS /* was in JAPAN */
/* at least one kanji is displayed */
/* DxpFromCh() --> DBCSDxpFromCh() 03 Oct 1991 YutakaN */
{
int dxpPr;
if ( ifi.xpPrRight - ifi.xpPr < (dxpPr = DBCSDxpFromCh(bKanjiSpace1,bKanjiSpace2, TRUE) ) )
{
ifi.xpPrRight = ifi.xpPr + dxpPr + 1;
}
}
#endif
#endif
/* Get a pointer to the tab-stop table. */
ptbd = ppap->rgtbd;
/* Turn off justification. */
SetTextJustification(fFlmPrinting ? vhDCPrinter : vhMDC, 0, 0);
/* Initialize the line height information. */
dypAscentMac = dypDescentMac = 0;
/*T-HIROYN add from 3.0*/
#if defined(JAPAN) || defined(KOREA)
dypLeadingMac = dypIntLeadingMac = dypPAscentMac = 0;
#endif /* JAPAN */
/* To tell if there were any tabs */
ifi.ichLeft = -1;
#if defined(JAPAN) || defined(KOREA) // added 22 Jun. 1992 by Hiraisi
iWidenChar=0;
#elif defined(TAIWAN) || defined(PRC) // Daniel/MSTC, 1993/02/25, for jcBoth
iWidenChar=0;
#endif
/* Get the first run, and away we go... */
FetchCp(doc, cp, ichCp, fcmBoth + fcmParseCaps);
goto FirstCps;
for ( ; ; )
{
int iichNew;
int xpPrev;
int dxp;
int dxpPr;
/* The number of characters to process (usually vcchFetch) */
int cch;
/* The number of characters in current run already used */
int cchUsed;
/* A pointer to the current list of characters (usually vpchFetch) */
CHAR *pch;
#ifdef CASHMERE
CHAR rgchSmcap[cchSmcapMax];
#endif /* CASHMERE */
if (ifi.ichFetch == cch)
{
/* End of a run */
int dich;
BOOL fSizeChanged;
if (ifi.ich >= ichMaxLine )
/* End of run because of line length limit has been reached. */
{
goto DoBreak;
}
if (fTruncated)
{
cchUsed += cch;
pch = vpchFetch + cchUsed;
cch = vcchFetch - cchUsed;
fTruncated = false;
goto OldRun; /* use the rest of the old run */
}
NullRun:
#ifdef DBCS
if (!fOddBoundary)
{
/* The last fetch did not mess up a sequential access. */
FetchCp(docNil, cpNil, 0, fcmBoth + fcmParseCaps);
}
else
{
/* Previous fetch was an odd one. Set it up again. */
FetchCp(doc, cpSeqFetch, 0, fcmBoth + fcmParseCaps);
}
fOddBoundary = false;
#else
FetchCp(docNil, cpNil, 0, fcmBoth + fcmParseCaps);
#endif
FirstCps:
cchUsed = 0;
/* Continue fetching runs until a run is found with a nonzero
length. */
if ((cch = vcchFetch) == 0)
{
goto NullRun;
}
pch = vpchFetch;
if (vcpFetch >= cpMac || (!fFlmPrinting && *pch == chSect))
{
#ifdef SYSENDMARK
/* Force end mark and section mark to be in standard system
font. */
blt(&vchpNormal, &vchpAbs, cwCHP);
vchpAbs.ftc = ftcSystem;
vchpAbs.ftcXtra = 0;
vchpAbs.hps = hpsDefault;
#else
#ifdef REVIEW
/* The following comment is absolutely misleading! Ftc==0
doesn't give you a system font. It gives you the first
entry in the font table. */
#endif /* REVIEW */
/* Force end mark and section mark to be in standard system
font. */
blt(&vchpNormal, &vchpAbs, cwCHP);
vchpAbs.ftc = 0;
vchpAbs.ftcXtra = 0;
vchpAbs.hps = hpsDefault;
#endif /* if-else-def KANJI */
}
#ifdef CASHMERE
/* Adjust the size of the font for "small capitals". */
if (vchpAbs.csm == csmSmallCaps)
{
vchpAbs.hps = HpsAlter(vchpAbs.hps, -1);
}
#endif /* CASHMERE */
/* Now we have:
ichpFormat index into gchp table
vcpFetch first cp of current run
vfli.cpMin first cp of line
ifi.ich ???
*/
/* since LoadFont could change vchpAbs, and we don't want
that to happen, we copy vchpAbs into vchpLocal and use
vchpLocal in place of vchpAbs hereafter. Note that vchpAbs
is intentionally used above for handling the endmark. */
blt(&vchpAbs, &chpLocal, cwCHP);
if (fFlmPrinting)
{
LoadFont(doc, &chpLocal, mdFontPrint);
dypAscent = vfmiPrint.dypAscent + vfmiPrint.dypLeading;
dypDescent = vfmiPrint.dypDescent;
#ifdef DBCS /* was in JAPAN */
dypPAscent = vfmiPrint.dypAscent;
dypLeading = vfmiPrint.dypLeading;
dypIntLeading = vfmiPrint.dypIntLeading;
#endif
}
else
{
LoadFont(doc, &chpLocal, mdFontJam);
dypAscent = vfmiScreen.dypAscent + vfmiScreen.dypLeading;
dypDescent = vfmiScreen.dypDescent;
#ifdef DBCS /* was in JAPAN */
dypPAscent = vfmiScreen.dypAscent;
dypLeading = vfmiScreen.dypLeading;
dypIntLeading = vfmiScreen.dypIntLeading;
#endif
}
#ifdef ENABLE /* BRYANL 8/27/87: New philosophy for handling
font selection failures is: font selection
ALWAYS succeeds. This prevents FormatLine
returns that do not advance. */
/* Bail out if there is a memory failure. */
if (vfOutOfMemory)
{
goto DoBreak;
}
#endif /* ENABLE */
/* Floating line size algorithm */
if (chpLocal.hpsPos != 0)
{
/* Modify font for subscript/superscript */
if (chpLocal.hpsPos < hpsNegMin)
{
dypAscent += ypSubSuperFormat;
#ifdef DBCS /* was in JAPAN */
dypPAscent += ypSubSuperFormat;
#endif
}
else
{
dypDescent += ypSubSuperFormat;
}
}
/* Update the maximum ascent and descent of the line. */
fSizeChanged = FALSE;
if (dypDescentMac < dypDescent)
{
dypDescentMac = dypDescent;
fSizeChanged = TRUE;
}
if (dypAscentMac < dypAscent)
{
dypAscentMac = dypAscent;
fSizeChanged = TRUE;
}
#ifdef DBCS /* was in JAPAN */
if (dypPAscentMac < dypPAscent)
{
dypPAscentMac = dypPAscent;
fSizeChanged = TRUE;
}
if (dypIntLeadingMac < dypIntLeading)
{
dypIntLeadingMac = dypIntLeading;
fSizeChanged = TRUE;
}
if (dypLeadingMac < dypLeading)
{
dypLeadingMac = dypLeading;
fSizeChanged = TRUE;
}
#endif
if (fSizeChanged)
{
#ifdef AUTO_SPACING
/* This is the original Mac Word code that assumed line spacing
of 0 in a PAP meant auto line spacing. PC Word defaults to 1
line invalidating this assumption. */
if (ppap->dyaLine == 0)
{
#ifdef CASHMERE
ifi.dypLineSize = dypDescentMac + dypAscentMac + dypBefore;
#else /* not CASHMERE */
ifi.dypLineSize = dypDescentMac + dypAscentMac;
#endif /* not CASHMERE */
}
else
{
#ifdef CASHMERE
ifi.dypLineSize = imax(MultDiv(ppap->dyaLine, dypFormat,
dyaFormat) + dypBefore, dypBefore + 1);
#else /* not CASHMERE */
ifi.dypLineSize = imax(MultDiv(ppap->dyaLine, dypFormat,
dyaFormat), 1);
#endif /* not CASHMERE */
}
#else /* not AUTO_SPACING */
/* This code forces auto line spacing except in the case where
the user specifies a line spacing greater than the auto line
spacing. */
{
#ifdef DBCS /* was in JAPAN */
#if defined(TAIWAN) || defined(PRC)
register int dypAuto = dypDescentMac + dypAscentMac;
if (ppap->dyaLine > czaLine)
{
register int dypUser = imax(MultDiv(ppap->dyaLine,
dypFormat, dyaFormat), 1);
ifi.dypLineSize = max(dypAuto, dypUser);
}
else
{
ifi.dypLineSize = dypAuto;
}
#else /* TAIWAN */
register int dypAuto = dypDescentMac + dypAscentMac;
int cHalfLine;
int dypSingle = dypPAscentMac + dypDescentMac;
cHalfLine = (ppap->dyaLine + (czaLine / 4)) / (czaLine / 2);
ifi.dypLineSize = (cHalfLine == 3) ? (dypSingle*3)/2 :
((cHalfLine <= 2) ?
dypSingle :
(dypSingle * 2));
#endif /* TAIWAN */
#else // DBCS
#ifdef CASHMERE
register int dypAuto = dypDescentMac + dypAscentMac +
dypBefore;
#else /* not CASHMERE */
register int dypAuto = dypDescentMac + dypAscentMac;
#endif /* not CASHMERE */
if (ppap->dyaLine > czaLine)
{
#ifdef CASHMERE
register int dypUser = imax(MultDiv(ppap->dyaLine,
dypFormat, dyaFormat) + dypBefore, dypBefore + 1);
#else /* not CASHMERE */
register int dypUser = imax(MultDiv(ppap->dyaLine,
dypFormat, dyaFormat), 1);
#endif /* not CASHMERE */
ifi.dypLineSize = max(dypAuto, dypUser);
}
else
{
ifi.dypLineSize = dypAuto;
}
#endif /* DBCS */
}
#endif /* not AUTO_SPACING */
}
OldRun:
/* Calculate length of the run but no greater than 256 */
iichNew = (int)(vcpFetch - vfli.cpMin);
if (iichNew >= ichMaxLine)
{
iichNew = ichMaxLine - 1;
}
dich = iichNew - ifi.ich;
/* Ensure that all tab and non-required hyphen characters start at
beginning of run */
if (ichpFormat <= 0 || dich > 0 || CchDiffer(&chpLocal,
&(**vhgchpFormat)[ichpFormat - 1], cchCHPUsed) != 0 || *pch ==
chTab || *pch == chNRHFile)
{
#ifdef DFLI
if (*pch == chNRHFile)
CommSz("CHNRHFILE at beginning of run");
#endif
if (ichpFormat != ichpMacFormat || FGrowFormatHeap())
{
register struct CHP *pchp = &(**vhgchpFormat)[ichpFormat -
1];
if (ichpFormat > 0)
{
pchp->cchRun = ifi.ich - ifi.ichPrev;
pchp->ichRun = ifi.ichPrev;
}
blt(&chpLocal, ++pchp, cwCHP);
#ifdef ENABLE /* font codes */
pchp->ftc = vftc;
pchp->ftcXtra = (vftc & 0x01c0) >> 6;
pchp->hps = vhps;
#endif /* ENABLE */
pchp->cchRun = ichMaxLine;
if (dich <= 0)
{
pchp->ichRun = ifi.ich;
}
else
{
/* Q&D insert */
bltc(&vfli.rgdxp[ifi.ich], 0, dich);
bltbc(&vfli.rgch[ifi.ich], 0, dich);
pchp->ichRun = ifi.ich = iichNew;
}
ifi.ichPrev = ifi.ich;
ichpFormat++;
}
}
if (vcpFetch >= cpMac)
{
/* End of doc reached */
if (!ifi.fPrevSpace || vcpFetch == cp)
{
vfli.ichReal = ifi.ich;
vfli.xpReal = ifi.xpReal = ifi.xp;
}
/* T-HIROYN sync fromat.asm */
/* if (!fFlmPrinting && (doc != docHelp))*/
if (!fFlmPrinting)
{
/*
* Write 3.1j endmark is 1-bytecharcter
* t-Yoshio May 26,92
*/
#if defined(JAPAN) || defined(KOREA)
vfli.rgch[ifi.ich] = chEMark;
vfli.xpReal += (vfli.rgdxp[ifi.ich++] = DxpFromCh(chEMark,
false));
#else
#ifdef DBCS /* was in JAPAN */
/* We use Double byte character to chEMark in Japan */
if (ifi.ich + cchKanji > ichMaxLine) {
/* vfli.rgch has no room for the two-byte
end mark. Too bad do the break and
wait for the next time around. */
goto DoBreak;
}
vfli.rgch[ifi.ich] = chMark1;
vfli.xpReal += (vfli.rgdxp[ifi.ich++] = DxpFromCh(chMark1,
false));
vfli.rgch[ifi.ich] = chEMark;
vfli.rgdxp[ifi.ich++] = 0;
#if !defined(TAIWAN) && !defined(PRC)
ifi.dypLineSize += 2;
#endif
#else /* DBCS */
vfli.rgch[ifi.ich] = chEMark;
vfli.xpReal += (vfli.rgdxp[ifi.ich++] = DxpFromCh(chEMark,
false));
#endif
#endif /*JAPAN*/
}
vfli.dypLine = ifi.dypLineSize;
vfli.dypBase = dypDescentMac;
vfli.dypFont = dypAscentMac + dypDescentMac;
vfli.ichMac = vfli.ichReal = ifi.ich;
vfli.cpMac = cpMac + 1;
goto JustEol; /* dcpDepend == 0 */
}
/* Here we have ifi.ich, cch */
if (ifi.ich + cch > ichMaxLine)
/* If this run would put the line over 255, truncate it and set a
flag. */
{
cch = ichMaxLine - ifi.ich;
fTruncated = true;
}
ifi.ichFetch = 0;
#ifdef CASHMERE
if (chpLocal.csm != csmNormal)
{
int ich;
CHAR *pchT = &rgchSmcap[0];
/* We can handle only a run of cchSmcapMax small capital
characters. If the run is larger then truncate. */
if (cch > cchSmcapMax)
{
cch = cchSmcapMax;
fTruncated = true;
}
/* Raise the case of the characters. */
for (ich = 0 ; ich < cch ; ich++)
{
*pchT++ = ChUpper(*pch++);
}
pch = &rgchSmcap[0];
}
#endif /* CASHMERE */
/* Do "special" characters here */
if (chpLocal.fSpecial)
{
if (!FFormatSpecials(&ifi, flm, vsepAbs.nfcPgn))
{
if (ifi.chBreak == 0 ) /* No breaks in this line */
{
goto Unbroken;
}
else
{
vfli.dcpDepend = vcpFetch + ifi.ichFetch - vfli.cpMac;
goto JustBreak;
}
}
}
continue;
}
/* End of new run treatment. We are back in the "for every character"
section. */
{
register int ch = pch[ifi.ichFetch++];
NormChar:
#ifdef DBCS
/* Unless it is a kanji space, we only need to adjust
by 1 byte. */
dichSpaceAdjust = 1;
#endif
#if defined(JAPAN) || defined(KOREA) /*t-Yoshio*/
if (ch == chSpace && vfWordWrap)
#else
if (ch == chSpace)
#endif
{
/* Speed kludge for spaces */
ifi.xp += (vfli.rgdxp[ifi.ich] = dxp =
fFlmPrinting ? vfmiPrint.dxpSpace : vfmiScreen.dxpSpace);
ifi.xpPr += (dxpPr = vfmiPrint.dxpSpace);
vfli.rgch[ifi.ich++] = chSpace;
#ifdef DFLI
{
char rgch[100];
wsprintf(rgch," chSpace , xp==%d/%d, xpPr==%d/%d",
ifi.xp, ifi.xpRight, ifi.xpPr, ifi.xpPrRight);
CommSz(rgch);
}
#endif
goto BreakOppr;
}
#ifndef TEMP_KOREA
/* If the printer width is not in the printer width table, then get
it. */
if (ch < chFmiMin || ch >= chFmiMax || (dxpPr =
vfmiPrint.mpchdxp[ch]) == dxpNil)
{
#ifdef DBCS
/* Don't pass to DxpFromCh() DBCS LeadByte except for '8140H'.
** Because the function can make elleagal ShiftJIS and pass it
** to GetTextExtent(). GetTextExtent might return SBC space
** when the code is undefined. this will cause win-hang-up at
** formatting line. yutakan
*/
#if defined(TAIWAN) || defined(PRC) //solve BkSp single byte (>0x80) infinite loop problem, MSTC - pisuih, 2/25/93
dxpPr=DBCSDxpFromCh(ch, ( ((cp + ifi.ich) < cpMac) ?
pch[ifi.ichFetch] : 0 ), TRUE);
#else
dxpPr=DBCSDxpFromCh(ch,pch[ifi.ichFetch],TRUE);
#endif //TAIWAN
#else
dxpPr = DxpFromCh(ch, TRUE);
#endif
}
if (fFlmPrinting)
{
/* If we are printing, then there is no need to bother with the
screen width. */
dxp = dxpPr;
}
else if (ch < chFmiMin || ch >= chFmiMax ||
(dxp = vfmiScreen.mpchdxp[ch]) == dxpNil)
#ifdef DBCS
#if defined(TAIWAN) || defined(PRC) //solve BkSp single byte (>0x80) infinite loop problem, MSTC - pisuih, 2/25/93
dxp = DBCSDxpFromCh(ch, ( ((cp + ifi.ich) < cpMac) ?
pch[ifi.ichFetch] : 0 ), FALSE);
#else
// yutakan:
dxp = DBCSDxpFromCh(ch,pch[ifi.ichFetch],FALSE);
#endif //TAIWAN
#else
dxp = DxpFromCh(ch, FALSE);
#endif /* ifdef DBCS */
#endif /* ifndef KOREA */
#ifdef DBCS /* was in JAPAN */
//solve BkSp single byte (>0x80) infinite loop problem, MSTC - pisuih, 2/25/93
//#ifdef TAIWAN
// if (IsDBCSLeadByte(ch) && !pch[ifi.ichFetch]) {
// ifi.xp += (vfli.rgdxp[ifi.ich] = (dxp/2));
// ifi.xpPr += (dxpPr/2);
// vfli.rgch[ifi.ich++] = ch;
// vfli.rgch[ifi.ich] = NULL;
// goto OnlyDBCSPaste;
// }
//#endif
/*T-HIROYN add from 3.0*/
ifi.xp += (vfli.rgdxp[ifi.ich] = dxp);
ifi.xpPr += dxpPr;
vfli.rgch[ifi.ich++] = ch;
#if defined(TAIWAN) || defined(PRC) //solve BkSp single byte (>0x80) infinite loop problem, MSTC - pisuih, 2/25/93
// BUG 5477: Echen: add NON FA font, LeadByte + 2nd byte checking
if (((cp + ifi.ich) < cpMac) && IsDBCSLeadByte(ch) &&
GetFontAssocStatus(vhMDC)) {
#else
if (IsDBCSLeadByte(ch)) {
#endif //TAIWAN
CHAR ch2;
if (ifi.ich + 1 >= ichMaxLine) {
/* It's full. Do the break without this kanji character. */
#ifndef TEMP_KOREA
ifi.ich--;
#endif
ifi.ichFetch--; /* We don't want to skip the first byte. */
#ifndef TEMP_KOREA
ifi.xp -= dxp;
ifi.xpPr -= dxpPr;
#endif
lblFull2: /* new label of line full case ( for kanji and kana ) */
goto DoBreak;
}
/* Now all is well. Get the second byte of the kanji
character of interest from the current run. */
/* Get the second byte of the kanji character from
the current run. If we run of the current run,
use FetchRgch() to staple us over. */
#ifdef TEMP_KOREA /* for variable width, 90.12.26 by sangl */
vfli.rgch[ifi.ich++] = ch;
#endif
if (ifi.ichFetch == cch)
{
if (fTruncated)
{
cchUsed += cch;
pch = vpchFetch + cchUsed;
cch = vcchFetch - cchUsed;
fTruncated = false;
ch2 = vfli.rgch[ifi.ich] = pch[ifi.ichFetch++];
}
else {
int cchFetch;
/* Save parameters needed for the re-fetch. */
cpFetchSave = vcpFetch;
ichFetchSave = vichFetch;
cchUsedSave = cchUsed;
cpSeqFetch = vcpFetch + cch + 1;
FetchRgch(&cchFetch, &ch2, docNil, cpNil,
vcpFetch + cch + 1, 1);
fOddBoundary = true;
Assert(cchFetch != 0); /* Better fetched something for us. */
#if defined(TAIWAN) || defined(PRC) //solve BkSp single byte (>0x80) infinite loop problem, MSTC - pisuih, 2/25/93
if ( !cchFetch ) goto SingleDBCS;
#endif //TAIWAN
/* Now, let's settle related parameters. */
pch = &ch2;
cch = cchFetch;
ifi.ichFetch = 1; /* == cch */
cchUsed = 0;
vfli.rgch[ifi.ich] = ch2;
#if defined(TAIWAN) || defined(PRC) //solve BkSp single byte (>0x80) infinite loop problem, MSTC - pisuih, 2/25/93
//adjust DBCS char width by the new fetched 2nd byte
{
int dxpPr2, dxp2;
dxpPr2 = DBCSDxpFromCh(ch, ch2, TRUE);
if (fFlmPrinting) dxp2 = dxpPr2;
else dxp2 = DBCSDxpFromCh(ch, ch2, FALSE);
vfli.rgdxp[ifi.ich - 1] += (dxp2 - dxp);
ifi.xp += (dxp2 - dxp);
ifi.xpPr += (dxpPr2 - dxpPr);
}
#endif //TAIWAN
}
}
else
{
ch2 = vfli.rgch[ifi.ich] = pch[ifi.ichFetch++];
}
#ifdef TEMP_KOREA /* For variable width, 90.12.26 by sangl */
{ unsigned int wd;
wd = (ch<<8) + ch2;
dxpPr = DxpFromCh(wd, TRUE);
if (fFlmPrinting) /* if we are printing, then there is */
/* no need to bother with the screen width */
dxp = dxpPr;
else
dxp = DxpFromCh(wd, FALSE);
ifi.xp += (vfli.rgdxp[ifi.ich-1] = dxp);
ifi.xpPr += dxpPr;
vfli.rgdxp[ifi.ich++] = 0;
}
#else
vfli.rgdxp[ifi.ich++] = 0; /* The second byte has 0 width. */
#endif /* KOREA */
#if defined(JAPAN) || defined(KOREA) /*t-Yoshio*/
if (FKanjiSpace(ch, ch2) && vfWordWrap)
#else
if (FKanjiSpace(ch, ch2))
#endif
{
fKanjiPrevious = true;
fKanjiBreakOppr = false; /* Treat it like a regular space. */
dichSpaceAdjust = cchKanji;
goto BreakOppr;
}
if (ifi.xpPr > ifi.xpPrRight ) {
fKanjiBreakOppr = false; /* Reset the flag */
if (FAdmitCh2(ch, ch2) ||
(fKanjiPrevious && FOptAdmitCh2(ch, ch2))) {
/* We do a line break including this odd character. */
/* Make sure non-printables won't start a new line. */
/* If we already have a hanging character on the */
/* line, we don't want to treat this character as */
/* a hanging one. */
if (!fKanjiHanging )
{
fKanjiHanging = TRUE;
ch = chHyphen;
goto BreakOppr;
}
}
#ifndef JAPAN // added by Hiraisi (BUG#3542)
#if defined(JAPAN) || defined(KOREA) /*t-Yoshio*/
if(vfWordWrap)
#endif
ifi.ich--;
#endif
/* If this run was the result of an odd boundary run,
re-fetch. */
if (fOddBoundary && ifi.ichFetch == 1 )
{
FetchCp(doc, cpFetchSave, ichFetchSave,
fcmBoth + fcmParseCaps);
/* This fetch is guaranteed to result to non-null run. */
fOddBoundary = false;
pch = vpchFetch;
ifi.ichFetch = cch = vcchFetch;
#ifdef JAPAN // added by Hiraisi (BUG#3542)
ifi.ichFetch++;
#endif
cchUsed = cchUsedSave;
}
#ifndef JAPAN // added by Hiraisi (BUG#3542)
else
{
#if defined(JAPAN) || defined(KOREA) /*t-Yoshio*/
if(vfWordWrap)
#endif
ifi.ichFetch--;
}
#endif
/* ifi.xp and ifi.xpPr hasn't changed yet. */
goto lblFull2;
#ifdef TEMP_KOREA /* 90.12.26 : For variable width, 90.12.26 by sangl */
ifi.xp -= dxp;
ifi.xpPr -= dxpPr;
#endif /* KOREA */
}
#if defined(JAPAN) || defined(KOREA) // added 26 Jun. 1992 by Hiraisi
iWidenChar++;
#elif defined(TAIWAN) || defined(PRC)//Daniel/MSTC, 1993/02/25 , for jcBoth
iWidenChar++;
#endif
/* Record the line break opportunity while processing
it as a regular character at the same time. */
fKanjiBreakOppr = true;
fKanjiPrevious = true;
#if defined(JAPAN) || defined(KOREA) /*t-Yoshio*/
if(!vfWordWrap)
goto DefaultCh;
#endif
goto BreakOppr;
}
else {
#if defined(JAPAN) || defined(KOREA) /*t-Yoshio add WordWrap flag*/
if (FKana(ch) && vfWordWrap) {
/* If it is a 1-byte kana letter, we want to treat it
in the same way as a kanji letter. */
if (ifi.xpPr > ifi.xpPrRight) {
fKanjiBreakOppr = false; /* Reset the flag */
if (FAdmitCh1(ch)) {
/* Make sure non-printables won't start a new line. */
/* If we already have a hanging character on the */
/* line, we don't want to treat this character as */
/* a hanging one. */
if (!fKanjiHanging) {
fKanjiHanging = TRUE;
ch = chHyphen;
goto BreakOppr;
}
}
goto lblFull2;
}
#if defined(JAPAN) || defined(KOREA) // added 22 Jun. 1992 by Hiraisi
iWidenChar++;
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25 , for jcBoth
iWidenChar++;
#endif
fKanjiPrevious = true;
fKanjiBreakOppr = true;
/* Go through the break opprotunity processing, then the
default character processing. */
goto BreakOppr;
}
else {
#endif /* JAPAN */
#ifdef TEMP_KOREA /* For variable width by sangl 90.12.26 */
if (ch < chFmiMin || ch >= chFmiMax || (dxpPr =
vfmiPrint.mpchdxp[ch]) == dxpNil)
{
dxpPr = DxpFromCh(ch, TRUE);
}
if (fFlmPrinting)
{
/* If we are printing, then there is no need to bother
with the screen width */
dxp = dxpPr;
}
else if (ch < chFmiMin || ch >= chFmiMax ||
(dxp = vfmiScreen.mpchdxp[ch]) == dxpNil)
dxp = dxpFromCh(ch, FALSE);
ifi.xp += (vfli.rgdxp[ifi.ich] = dxp);
ifi.xpPr += dxpPr;
vfli.rgch[ifi.ich++] = ch;
#endif /* KOREA */
#if defined(TAIWAN) || defined(PRC) //solve BkSp single byte (>0x80) infinite loop problem, MSTC - pisuih, 2/25/93
SingleDBCS:
#endif //TAIWAN
if (fKanjiPrevious && FOptAdmitCh1(ch)) {
fKanjiPrevious = false;
if (ifi.xpPr > ifi.xpPrRight) {
fKanjiBreakOppr = false;
/* If we already have a hanging character past the
margin, we don't want to treat this as a
hanging character. */
if (!fKanjiHanging) {
fKanjiHanging = true;
ch = chHyphen;
goto BreakOppr;
}
}
else {
/* We can treat this character as though a Kanji
punctuation, as far as line breaking is
is concerned. */
fKanjiBreakOppr = true;
goto BreakOppr;
}
}
else {
/* Just go on with a regular English formatting. */
fKanjiBreakOppr = false;
fKanjiPrevious = false;
}
}
#if defined(JAPAN) || defined(KOREA)
}
#endif
#else /* DBCS */
ifi.xp += (vfli.rgdxp[ifi.ich] = dxp);
ifi.xpPr += dxpPr;
vfli.rgch[ifi.ich++] = ch;
#endif
#if defined (TAIWAN)
OnlyDBCSPaste:
#endif
/* special case "normal characters" above hyphen */
if (ch > chHyphen)
goto DefaultCh;
#if defined(JAPAN) || defined(KOREA) /*t-Yoshio*/
/*Non Word wrap Not Hyphen break*/
if(!vfWordWrap) {
if (ch == chHyphen)
goto DefaultCh;
}
#endif
switch (ch)
{
#ifdef CRLF
case chReturn:
/* Undo damage */
ifi.ich--;
ifi.xp -= dxp;
ifi.xpPr -= dxpPr;
continue;
#endif /* CRLF */
case chNRHFile:
/* Undo damage */
ifi.ich--;
ifi.xp -= dxp;
ifi.xpPr -= dxpPr;
ichpNRH = ichpFormat - 1;
#ifdef DFLI
{
char rgch[100];
wsprintf(rgch," OptHyph: width==%d, xpPr==%d/%d\n\r",
DxpFromCh(chHyphen,true), ifi.xpPr,ifi.xpPrRight);
CommSz(rgch);
}
#endif
if (ifi.xpPr + DxpFromCh(chHyphen, true) > ifi.xpPrRight)
{
/* Won't fit, force a break */
goto DoBreak;
}
#ifdef CASHMERE
else if (fVisiMode)
{
/* Treat just like a normal hyphen */
ch = chHyphen;
goto NormChar;
}
#endif /* CASHMERE */
xpPrev = ifi.xp;
vfli.rgch[ifi.ich] = chTab;
goto Tab0;
case chSect:
/* Undo damage */
ifi.ich--;
ifi.xp -= dxp;
ifi.xpPr -= dxpPr;
vfli.dypFont = vfli.dypLine = (dypAscentMac + (vfli.dypBase
= dypDescentMac));
vfli.cpMac = vcpFetch + ifi.ichFetch;
if (FFirstIch(ifi.ich))
{
/* Beginning of line; return a splat */
vfli.fSplat = true;
if (!fFlmPrinting)
{
#ifdef CASHMERE
int chT = vfli.cpMac == vcpLimSectCache ?
chSectSplat : chSplat;
#else /* not CASHMERE */
int chT = chSplat;
#endif /* not CASHMERE */
int dxpCh = DxpFromCh(chT, false);
/* Set the width of the splat to be about 8.5" */
int cch = min((dxpLogInch * 17 / 2) / dxpCh,
ichMaxLine - 32);
bltbc(&vfli.rgch[ifi.ich], chT, cch);
bltc(&vfli.rgdxp[ifi.ich], dxpCh, cch);
vfli.ichMac = cch + ifi.ich;
vfli.xpReal = LOWORD(GetTextExtent(vhMDC,
(LPSTR)vfli.rgch, cch));
vfli.xpLeft = 0;
}
else
{
vfli.ichMac = 0;
}
goto EndFormat;
}
/* The section character is in the middle of a line, the
line will terminate in front of the character. */
/* vfSplatNext = TRUE; No longer used*/
vfli.cpMac += cchUsed - 1;
vfli.dcpDepend = 1;
if (!ifi.fPrevSpace)
{
ifi.cBreak = ifi.cchSpace;
vfli.ichReal = ifi.ich;
vfli.xpReal = ifi.xpReal = ifi.xp;
}
vfli.ichMac = ifi.ich;
vfli.dypLine = ifi.dypLineSize;
goto JustBreak;
case chTab:
/* Undo damage */
ifi.ich--;
ifi.xp -= dxp;
ifi.xpPr -= dxpPr;
if (ifi.xpPr < ifi.xpPrRight)
{
register struct CHP *pchp;
unsigned xaPr;
unsigned xaTab;
if (!ifi.fPrevSpace)
{
/* Remember number of spaces to left and number of
real chars in line for justification */
ifi.cBreak = ifi.cchSpace;
vfli.ichReal = ifi.ich;
ifi.xpReal = ifi.xp;
}
if (ifi.jc != jcTabLeft)
{
Justify(&ifi, xpTab, flm);
}
xpPrev = ifi.xp;
/* Now get info about this tab */
xaPr = MultDiv(ifi.xpPr, dxaPrPage, dxpPrPage);
while ((xaTab = ptbd->dxa) != 0)
{
#ifdef DBCS /* was in JAPAN */
if (xaTab >= xaRight)
#else
if (xaTab > xaRight)
#endif
{
/* Don't let tabs extend past right margin. */
#ifdef DBCS /* was in JAPAN */
break; // Stop to examin next tab-stop
#else
xaTab = xaRight;
#endif
}
if (xaTab >= xaPr)
{
/* Use tab stop information */
#ifdef CASHMERE
ifi.tlc = ptbd->tlc;
#endif /* CASHMERE */
ifi.jc = jcTabMin + (ptbd++)->jc;
#ifdef ENABLE /* we do the mapping in HgtbdCreate */
if (ifi.jc != jcTabDecimal)
{
ifi.jc = jcTabLeft;
}
#endif
goto TabFound;
}
ptbd++;
}
/* Out of set tabs; go to next nth column */
xaTab = (xaPr / (vzaTabDflt) + 1) * (vzaTabDflt);
#ifdef CASHMERE
ifi.tlc = tlcWhite;
#endif /* CASHMERE */
ifi.jc = jcTabLeft;
TabFound:
xpTab = imax(MultDiv(xaTab, dxpFormat, dxaFormat),
ifi.xp);
/* Do left-justified tabs immediately */
if (ifi.jc == jcTabLeft)
{
ifi.xp = xpTab;
ifi.xpPr = MultDiv(xaTab, dxpPrPage, dxaPrPage);
}
ifi.xpLeft = ifi.xp;
ifi.ichLeft = ifi.ich;
ifi.cchSpace = 0;
ifi.chBreak = 0;
#if defined(JAPAN) || defined(KOREA) // added 02 Jul. 1992 by Hiraisi
iWidenChar=0;
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
iWidenChar=0;
#endif
Tab0:
ifi.fPrevSpace = false;
vfli.ichMac = ifi.ich;
vfli.xpReal = ifi.xp;
vfli.dypLine = ifi.dypLineSize;
vfli.dypBase = dypDescentMac;
vfli.dypFont = dypAscentMac + dypDescentMac;
if (ifi.ichFetch != 1 && (ichpFormat != ichpMacFormat
|| FGrowFormatHeap()))
{
/* Probably in real trouble if FGrowFormatHeap fails
at this point */
pchp = &(**vhgchpFormat)[ichpFormat - 1];
if (ichpFormat > 0)
{
/* Finish off previous run */
pchp->ichRun = ifi.ichPrev;
pchp->cchRun = ifi.ich - ifi.ichPrev;
}
blt(&chpLocal, ++pchp, cwCHP);
ichpFormat++;
}
else
{
pchp = &(**vhgchpFormat)[ichpFormat - 1];
}
pchp->ichRun = ifi.ich;
pchp->cchRun = ichMaxLine;
#ifdef CASHMERE
pchp->chLeader = mptlcch[ifi.tlc];
#endif /* CASHMERE */
vfli.rgdxp[ifi.ichPrev = ifi.ich++] = ifi.xp - xpPrev;
if (ch != chTab)
{
/* This character is a non-required hyphen. */
Dfli(CommSz("ch is really OptHyph "));
goto BreakOppr;
}
continue;
}
else
{
ch = chNBSFile;
goto NormChar;
}
case chHyphen:
if (ifi.xpPr > ifi.xpPrRight)
{
goto DoBreak;
}
BreakOppr:
Dfli(CommSz(" BKOPPR\n\r"));
/* this case never used in switch - always goto here */
/* case chSpace: */
if (ifi.ich >= ichMaxLine)
{
Dfli(CommSzNum(" Unbroken, ich>ichMaxLine\n\r"));
goto Unbroken;
}
case chEol:
case chNewLine:
ifi.chBreak = ch;
vfli.cpMac = vcpFetch + cchUsed + ifi.ichFetch;
vfli.xpReal = ifi.xp;
vfli.ichMac = ifi.ich;
vfli.dypLine = ifi.dypLineSize;
vfli.dypFont = dypAscentMac + (vfli.dypBase =
dypDescentMac);
Dfli(CommSzNumNum(" vfli.xpReal, ichMac ",vfli.xpReal,vfli.ichMac));
#ifdef DBCS /* was in JAPAN */
/* We recorded the kanji break opportunity, go default
character processing. */
if (fKanjiBreakOppr)
{
ifi.cBreak = ifi.cchSpace;
vfli.ichReal = ifi.ich;
vfli.xpReal = ifi.xpReal = ifi.xp;
goto DefaultCh;
}
#endif
if (ch == chHyphen || ch == chNRHFile)
{
Dfli(CommSz(" chHyph/OptHyph catch \n\r"));
ifi.cBreak = ifi.cchSpace;
vfli.ichReal = ifi.ich;
vfli.xpReal = ifi.xpReal = ifi.xp;
}
else
{
if (!ifi.fPrevSpace)
{
Dfli(CommSz("!fPrevSpace \n\r"));
ifi.cBreak = ifi.cchSpace;
#ifdef DBCS /* was in JAPAN */
vfli.ichReal = ifi.ich - dichSpaceAdjust;
dichSpaceAdjust = 1;
#else
vfli.ichReal = ifi.ich - 1;
#endif
ifi.xpReal = (vfli.xpReal = ifi.xp) - dxp;
}
if (ch == chEol || ch == chNewLine)
{
#ifdef CASHMERE
if (hfntb != 0 && vfli.cpMac ==
(**hfntb).rgfnd[0].cpFtn)
{
/* End of footnote */
if (!fFlmPrinting)
{
vfli.rgch[ifi.ich - 1] = chEMark;
vfli.xpReal += (vfli.rgdxp[ifi.ich - 1] =
DxpFromCh(chEMark, false)) - dxp;
vfli.ichReal++; /* show this guy */
}
}
else
#endif /* CASHMERE */
{
#ifdef CASHMERE
int chT = fVisiMode ? ChVisible(ch) : chSpace;
#else /* not CASHMERE */
int chT = chSpace;
#endif /* not CASHMERE */
int dxpNew = DxpFromCh(chT, fFlmPrinting);
vfli.rgch[ifi.ich - 1] = chT;
vfli.rgdxp[ifi.ich - 1] = dxpNew;
vfli.xpReal += (vfli.rgdxp[ifi.ich - 1] =
dxpNew) - dxp;
if (!ifi.fPrevSpace)
{
vfli.xpReal += dxpNew - dxp;
#ifdef CASHMERE
vfli.ichReal =
fVisiMode ? ifi.ich : ifi.ich - 1;
#else /* not CASHMERE */
vfli.ichReal = ifi.ich - 1;
#endif /* not CASHMERE */
}
}
if (ch == chEol)
{
JustEol:
if (fFlmPrinting)
{
vfli.ichMac = vfli.ichReal;
}
if (ifi.jc != jcTabLeft)
{
/* Handle last tab's text */
Justify(&ifi, xpTab, flm);
}
else if ((ifi.jc = ppap->jc) != jcBoth &&
ifi.jc != jcLeft)
{
/* Do line justification */
Justify(&ifi, ifi.xpRight, flm);
}
vfli.xpRight = ifi.xpRight;
goto EndFormat;
}
else
{
/* Handle a line break */
goto JustBreak;
}
}
++ifi.cchSpace;
ifi.fPrevSpace = true;
}
break;
DefaultCh:
default:
#ifdef DFLI
{
char rgch[100];
wsprintf(rgch," DefaultCh: %c, xp==%d/%d, xpPr==%d/%d\n\r",
ch, ifi.xp, ifi.xpRight, ifi.xpPr, ifi.xpPrRight);
CommSz(rgch);
}
#endif /* ifdef DFLI */
#ifdef DBCS /* was in JAPAN */
/* Reset the flag for the next character. */
fKanjiBreakOppr = false;
#endif
if (ifi.xpPr > ifi.xpPrRight)
DoBreak:
{
Dfli(CommSz(" BREAK!\n\r"));
if (ifi.chBreak == 0)
Unbroken:
{
/* Admit first character to the line, even if margin
is crossed. First character at ifi.ich - 1 may be
preceded by 0 width characters. */
#ifdef DBCS
if (IsDBCSLeadByte(ch))
{
if (FFirstIch(ifi.ich-2) && ifi.ich<ichMaxLine)
goto PChar;
vfli.cpMac = vcpFetch+cchUsed+ifi.ichFetch-2;
vfli.ichReal = vfli.ichMac = ifi.ich - 2;
vfli.dypLine = ifi.dypLineSize;
vfli.dypFont = dypAscentMac + (vfli.dypBase =
dypDescentMac);
vfli.dcpDepend = 1;
#ifdef KKBUGFIX /*t-Yoshio*/
vfli.xpReal = ifi.xpReal = ifi.xp - dxp;
#else
vfli.xpReal = ifi.xpReal = ifi.xp - (dxp * 2);
#endif
}
else
{
if (FFirstIch(ifi.ich-1) && ifi.ich<ichMaxLine)
goto PChar;
vfli.cpMac = vcpFetch+cchUsed+ifi.ichFetch-1;
vfli.ichReal = vfli.ichMac = ifi.ich - 1;
vfli.dypLine = ifi.dypLineSize;
vfli.dypFont = dypAscentMac + (vfli.dypBase =
dypDescentMac);
vfli.dcpDepend = 1;
vfli.xpReal = ifi.xpReal = ifi.xp - dxp;
}
#else
if (FFirstIch(ifi.ich - 1) && ifi.ich < ichMaxLine)
{
goto PChar;
}
vfli.cpMac = vcpFetch + cchUsed + ifi.ichFetch - 1;
vfli.ichReal = vfli.ichMac = ifi.ich - 1;
vfli.dypLine = ifi.dypLineSize;
vfli.dypFont = dypAscentMac + (vfli.dypBase =
dypDescentMac);
vfli.dcpDepend = 1;
vfli.xpReal = ifi.xpReal = ifi.xp - dxp;
#endif
goto DoJustify;
}
vfli.dcpDepend = vcpFetch + ifi.ichFetch - vfli.cpMac;
JustBreak:
if (ifi.chBreak == chNRHFile)
{
/* Append a non-required hyphen to the end of the
line. (Replace zero length tab previously
inserted) */
Dfli(CommSz(" Breaking line at OptHyphen\n\r"));
ifi.xpReal += (vfli.rgdxp[vfli.ichReal - 1] =
DxpFromCh(chHyphen, fFlmPrinting));
vfli.xpRight = vfli.xpReal = ifi.xpReal;
vfli.rgch[vfli.ichReal - 1] = chHyphen;
vfli.ichMac = vfli.ichReal;
if (ichpNRH < ichpFormat - 1)
{
register struct CHP *pchp =
&(**vhgchpFormat)[ichpNRH];
pchp->cchRun++;
if (pchp->ichRun >= vfli.ichMac)
{
pchp->ichRun = vfli.ichMac - 1;
}
}
}
if (fFlmPrinting)
{
vfli.ichMac = vfli.ichReal;
}
if (ifi.jc != jcTabLeft)
{
Justify(&ifi, xpTab, flm);
}
else
{
DoJustify:
if ((ifi.jc = ppap->jc) != jcLeft)
{
Dfli(CommSzNum(" DoJustify: xpRight ",ifi.xpRight));
Justify(&ifi, ifi.xpRight, flm);
}
}
vfli.xpRight = ifi.xpRight;
EndFormat:
vfli.ichLastTab = ifi.ichLeft;
#ifdef CASHMERE
if (vfli.cpMac == vcpLimParaCache)
{
vfli.dypAfter = vpapAbs.dyaAfter / DyaPerPixFormat;
vfli.dypLine += vfli.dypAfter;
vfli.dypBase += vfli.dypAfter;
}
#endif /* CASHMERE */
Scribble(5, ' ');
return;
}
else
{
PChar:
/* A printing character */
ifi.fPrevSpace = false;
}
break;
} /* Switch */
#ifdef DBCS /* was in KKBUGFIX */
//
// [yutakan:04/02/91]
//
if(vfOutOfMemory == TRUE)
return;
#endif
}
} /* for ( ; ; ) */
Scribble(5, ' ');
}
/* J U S T I F Y */
near Justify(pifi, xpTab, flm)
struct IFI *pifi;
unsigned xpTab;
int flm;
{
int dxp;
int ichT;
int xpLeft;
// justified paragraph is restored in Windows 3.1J by Hiraisi
//#ifdef JAPAN
// /* In the Kanji Write, there is no justified paragraph. */
// if (pifi->jc == jcBoth)
// {
// /* Assert(FALSE); */
// pifi->jc = jcLeft;
// dxp = 0; /* by yutakan / 08/03/91 */
// }
//#endif /* ifdef JAPAN */
xpLeft = pifi->xpLeft;
switch (pifi->jc)
{
CHAR *pch;
unsigned *pdxp;
#ifdef CASHMERE
case jcTabLeft:
case jcLeft:
return;
case jcTabRight:
dxp = xpTab - pifi->xpReal;
break;
case jcTabCenter:
dxp = (xpTab - xpLeft) - ((pifi->xpReal - xpLeft + 1) >> 1);
break;
#endif /* CASHMERE */
case jcTabDecimal:
dxp = xpTab - xpLeft;
for (ichT = pifi->ichLeft + 1; ichT < vfli.ichReal &&
vfli.rgch[ichT] != vchDecimal; ichT++)
{
dxp -= vfli.rgdxp[ichT];
}
break;
case jcCenter:
if ((dxp = xpTab - pifi->xpReal) <= 0)
{
return;
}
dxp = dxp >> 1;
break;
case jcRight:
dxp = xpTab - pifi->xpReal;
break;
case jcBoth:
//#if !defined(JAPAN) // added 22 Jun. 1992 by Hiraisi
#if !defined(JAPAN) && !defined(TAIWAN) && !defined(PRC) // Daniel/MSTC, 1993/02/25, for jcBoth
if (pifi->cBreak == 0)
{
/* Ragged edge forced */
return;
}
#endif
if ((dxp = xpTab - pifi->xpReal) <= 0)
{
/* There is nothing to do. */
return;
}
//#if !defined(JAPAN) // added 22 Jun. 1992 by Hiraisi
#if !defined(JAPAN) && !defined(TAIWAN) && !defined(PRC)//Daniel/MSTC, 1992,02,25, for jcBoth
pifi->xp += dxp;
vfli.xpReal += dxp;
vfli.dxpExtra = dxp / pifi->cBreak;
#endif
/* Rounding becomes a non-existant issue due to brilliant
re-thinking.
"What a piece of work is man
How noble in reason
In form and movement,
how abject and admirable..."
Bill "Shake" Spear [describing Sand Word] */
{
#ifdef JAPAN // added 22 Jun. 1992 by Hiraisi
/*
* In Japan, we examine the buffer from the beginning of the line.
* We find some NULLs in the buffer when a char is deleted,
* but we can ignore all of them.
*/
register CHAR *pch;
register int *pdxp;
CHAR *endPt;
int dxpT = dxp;
int cxpQuotient;
int cNonWideSpaces;
int ichLeft;
if( pifi->ichLeft >= 0 ) /* including some tabs in line */
ichLeft = pifi->ichLeft;
else
ichLeft = 0;
pch = &vfli.rgch[ichLeft];
pdxp = &vfli.rgdxp[ichLeft];
endPt = &vfli.rgch[vfli.ichReal];
if( vfWordWrap ){ /* Word Wrap ON */
/*
* We examine whether there is no break between a non-Japanese
* char and a following Japanese char. The reason is that we
* need to widen the non-Japanese char (except tab and space)
* if we can find no break there.
*/
for( ; pch<endPt ; ){
if( IsDBCSLeadByte( *pch ) ){
pch+=2;
}
else{
if( *pch != chSpace && *pch != chTab &&
!FKana( *pch ) && *pch != NULL ){
CHAR *ptr;
for( ptr = pch+1 ; *ptr == NULL ; ptr++ );
if( IsDBCSLeadByte(*ptr) ){
iWidenChar++;
pch+=2;
}
else{
if( FKana(*ptr) ){
iWidenChar++;
pch++;
}
}
}
pch++;
}
}
/*
* We decrease iWidenChar if last char of the current line
* is Japanese, because it needs not to be widened.
*/
if( *(endPt-1) == NULL ){
for( endPt-- ; *endPt==NULL ; endPt-- );
endPt++;
}
if( IsDBCSLeadByte(*(endPt-2)) ){
iWidenChar--;
}
else{
if( FKana(*(endPt-1)) )
iWidenChar--;
}
iWidenChar += pifi->cBreak;
}
else{ /* Word Wrap OFF */
/* We widen all chars except last char in the line. */
int iDBCS, ichReal;
for( iDBCS=0, ichReal=vfli.ichReal ; pch<endPt ; pch++ ){
if( IsDBCSLeadByte( *pch ) ){
pch++;
iDBCS++;
}
else{
if( *pch == NULL )
ichReal--;
}
}
iWidenChar = ichReal - ichLeft - iDBCS - 1;
}
if( iWidenChar == 0 )
return;
pifi->xp += dxp;
vfli.xpReal += dxp;
vfli.dxpExtra = dxp / iWidenChar;
cNonWideSpaces = iWidenChar - (dxp % iWidenChar);
cxpQuotient = vfli.dxpExtra;
iNonWideSpaces = cNonWideSpaces;
vfli.ichFirstWide = 0;
vfli.fAdjSpace = fTrue;
pch = &vfli.rgch[ichLeft]; /* Reset pch */
for( ; ; ){
if( IsDBCSLeadByte(*pch) ){
if( vfli.ichFirstWide == 0 ){
int *pdxpT = pdxp;
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
}
*pdxp += cxpQuotient;
if( --iWidenChar == 0 )
return;
if( --cNonWideSpaces == 0 )
cxpQuotient++;
pch++;
pdxp++;
}
else{
if( vfWordWrap ){ /* Word Wrap ON */
if( *pch == chSpace || FKana(*pch) ){
if( vfli.ichFirstWide == 0 ){
int *pdxpT = pdxp;
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
}
*pdxp += cxpQuotient;
if( --iWidenChar == 0 )
return;
if( --cNonWideSpaces == 0 )
cxpQuotient++;
}
else{
if( *pch != chTab && *pch != NULL ){
CHAR *ptr;
/*
* We examine whether the following char of
* non-Japanese char is Japanese.
*/
for( ptr = pch+1 ; *ptr == NULL ; ptr++ );
if( IsDBCSLeadByte(*ptr) || FKana(*ptr) ){
if( vfli.ichFirstWide == 0 ){
int *pdxpT = pdxp;
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
}
*pdxp += cxpQuotient;
if( --iWidenChar == 0 )
return;
if( --cNonWideSpaces == 0 )
cxpQuotient++;
}
}
}
}
else{ /* Word Wrap OFF */
if( *pch != NULL ){
if( vfli.ichFirstWide == 0 ){
int *pdxpT = pdxp;
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
}
*pdxp += cxpQuotient;
if( --iWidenChar == 0 )
return;
if( --cNonWideSpaces == 0 )
cxpQuotient++;
}
}
}
pch++;
pdxp++;
}
#elif defined(TAIWAN) || defined(PRC)//Daniel/MSTC, 1992/02/25, for jcBoth
/*
* In Japan, we examine the buffer from the beginning of the line.
* We find some NULLs in the buffer when a char is deleted,
* but we can ignore all of them.
*/
register CHAR *pch;
register int *pdxp;
CHAR *endPt;
int dxpT = dxp;
int cxpQuotient;
int cNonWideSpaces;
int ichLeft;
if( pifi->ichLeft >= 0 ) /* including some tabs in line */
ichLeft = pifi->ichLeft;
else ichLeft = 0;
pch = &vfli.rgch[ichLeft];
pdxp = &vfli.rgdxp[ichLeft];
endPt = &vfli.rgch[vfli.ichReal];
// if( vfWordWrap ){ /* Word Wrap ON */
/*
* We examine whether there is no break between a non-Japanese
* char and a following Japanese char. The reason is that we
* need to widen the non-Japanese char (except tab and space)
* if we can find no break there.
*/
for( ; pch<endPt ; )
{
if( IsDBCSLeadByte( *pch ) ) pch+=2;
else
{
if( *pch != chSpace && *pch != chTab && !FKana( *pch ) && *pch != NULL )
{ CHAR *ptr;
for( ptr = pch+1 ; *ptr == NULL ; ptr++ );
if( IsDBCSLeadByte(*ptr) )
{
iWidenChar++;
pch+=2;
}
else
{
if( FKana(*ptr) )
{
iWidenChar++;
pch++;
}
}
}
pch++;
}
}// for
/*
* We decrease iWidenChar if last char of the current line
* is Japanese, because it needs not to be widened.
*/
if( *(endPt-1) == NULL )
{
for( endPt-- ; *endPt==NULL ; endPt-- );
endPt++;
}
if( IsDBCSLeadByte(*(endPt-2)) ) iWidenChar--;
else
{
if( FKana(*(endPt-1)) ) iWidenChar--;
}
iWidenChar += pifi->cBreak;
// } // vfWordWrap
if( iWidenChar == 0 )
return;
pifi->xp += dxp;
vfli.xpReal += dxp;
vfli.dxpExtra = dxp / iWidenChar;
cNonWideSpaces = iWidenChar - (dxp % iWidenChar);
cxpQuotient = vfli.dxpExtra;
iNonWideSpaces = cNonWideSpaces;
vfli.ichFirstWide = 0;
vfli.fAdjSpace = fTrue;
pch = &vfli.rgch[ichLeft]; /* Reset pch */
for( ; ; )
{
if( IsDBCSLeadByte(*pch) )
{
if( vfli.ichFirstWide == 0 )
{
int *pdxpT = pdxp;
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
}
*pdxp += cxpQuotient;
if( --iWidenChar == 0 ) return;
if( --cNonWideSpaces == 0 ) cxpQuotient++;
pch++;
pdxp++;
}
else
{
// if( vfWordWrap )
if( 1 )
{ /* Word Wrap ON */
if( *pch == chSpace || FKana(*pch) )
{
if( vfli.ichFirstWide == 0 )
{
int *pdxpT = pdxp;
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
}
*pdxp += cxpQuotient;
if( --iWidenChar == 0 ) return;
if( --cNonWideSpaces == 0 ) cxpQuotient++;
}
else
{
if( *pch != chTab && *pch != NULL )
{
CHAR *ptr;
/*
* We examine whether the following char of
* non-Japanese char is Japanese.
*/
for( ptr = pch+1 ; *ptr == NULL ; ptr++ );
if( IsDBCSLeadByte(*ptr) || FKana(*ptr) )
{
if( vfli.ichFirstWide == 0 )
{
int *pdxpT = pdxp;
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
}
*pdxp += cxpQuotient;
if( --iWidenChar == 0 ) return;
if( --cNonWideSpaces == 0 ) cxpQuotient++;
}
}
} //else : ==chSpace || FKana()
} //Word Wrap On
}
pch++;
pdxp++;
}//for
#else // not JAPAN
register CHAR *pch = &vfli.rgch[vfli.ichReal];
register int *pdxp = &vfli.rgdxp[vfli.ichReal];
int dxpT = dxp;
int cBreak = pifi->cBreak;
int cxpQuotient = (dxpT / cBreak) + 1;
int cWideSpaces = dxpT % cBreak;
vfli.fAdjSpace = fTrue;
for ( ; ; )
{
/* Widen blanks */
--pch;
--pdxp;
#if defined(KOREA)
if ((*pch == chSpace) || FKanjiSpace(*pch, *(pch-1)))
{
if (FKanjiSpace(*pch, *(pch-1)))
--pch;
#else
if (*pch == chSpace)
{
#endif
if (cWideSpaces-- == 0)
{
int *pdxpT = pdxp + 1;
while (*pdxpT == 0)
{
pdxpT++;
}
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
cxpQuotient--;
}
*pdxp += cxpQuotient;
if ((dxpT -= cxpQuotient) <= 0)
{
if (pifi->cBreak > 1)
{
int *pdxpT = pdxp + 1;
while (*pdxpT == 0)
{
pdxpT++;
}
vfli.ichFirstWide = pdxpT - vfli.rgdxp;
}
return;
}
pifi->cBreak--;
}
}
#endif // JAPAN
}
} /* Switch */
if (dxp <= 0)
{
/* Nothing to do */
return;
}
pifi->xp += dxp;
if (flm & flmPrinting)
{
pifi->xpPr += dxp;
}
else
{
/* This statememt might introduce rounding errors in pifi->xpPr, but
with luck, they will be small. */
pifi->xpPr += MultDiv(MultDiv(dxp, czaInch, dxpLogInch), dxpPrPage,
dxaPrPage);
}
if (pifi->ichLeft < 0)
{
/* Normal justification */
vfli.xpLeft += dxp;
}
else
{
/* Tab justification */
vfli.rgdxp[pifi->ichLeft] += dxp;
}
vfli.xpReal += dxp;
}
/* F G R O W F O R M A T H E A P */
int near FGrowFormatHeap()
{
/* Grow vhgchpFormat by 20% */
int cchpIncr = ichpMacFormat / 5 + 1;
#ifdef WINHEAP
if (!LocalReAlloc((HANDLE)vhgchpFormat, (ichpMacFormat + cchpIncr) * cchCHP,
NONZEROLHND))
#else /* not WINHEAP */
if (!FChngSizeH(vhgchpFormat, (ichpMacFormat + cchpIncr) * cwCHP, false))
#endif /* not WINHEAP */
{
/* Sorry, charlie */
return false;
}
ichpMacFormat += cchpIncr;
return true;
}
/* #define DBEMG */
/* D X P F R O M C H */
#ifdef DBCS
/* DxpFromCh() assumes that ch passed is the first byte of a DBCS character
if it is a part of such character. */
#endif
int DxpFromCh(ch, fPrinter)
int ch;
int fPrinter;
{
int *pdxp; // changed to int (7.23.91) v-dougk
int dxpDummy; // changed to int (7.23.91) v-dougk
extern int dxpLogCh;
extern struct FCE *vpfceScreen;
/* If the width is not in the width table, then get it. */
if (ch < chFmiMin)
{
switch (ch)
{
case chTab:
case chEol:
case chReturn:
case chSect:
case chNewLine:
case chNRHFile:
/* the width for these characters aren't really important */
pdxp = (CHAR *)(fPrinter ? &vfmiPrint.dxpSpace : &vfmiScreen.dxpSpace);
break;
default:
pdxp = &dxpDummy;
*pdxp = dxpNil;
break;
}
}
else if (ch >= chFmiMax)
{
/* outside the range we hold in our table - kludge it */
pdxp = &dxpDummy;
*pdxp = dxpNil;
}
else
{
/* inside our table */
pdxp = (fPrinter ? vfmiPrint.mpchdxp : vfmiScreen.mpchdxp) + ch;
}
#ifdef DBCS
#ifdef KOREA
if (*pdxp == dxpNil && IsDBCSLeadByte(HIBYTE(ch)) )
#else
if (*pdxp == dxpNil && IsDBCSLeadByte(ch) )
#endif
{
int dxp;
#else
if (*pdxp == dxpNil)
{
int dxp;
#endif
#ifdef DBCS
struct FMI *pfmi;
#if 0 /*T-HIROYN*/
int rgchT[cchDBCS]; // changed to int (7.23.91) v-dougk
#endif
CHAR rgchT[cchDBCS]; // changed to int (7.23.91) v-dougk
int dxpT;
int dxpDBCS;
pfmi = fPrinter ? (&vfmiPrint) : (&vfmiScreen);
Assert(pfmi->bDummy == dxpNil);
if (pfmi->dxpDBCS == dxpNil)
{
#ifdef KOREA /* 90.12.26 For variable width by sangl */
rgchT[0] = HIBYTE(ch);
rgchT[1] = LOBYTE(ch);
#else
/* Get the width from GDI. */
rgchT[0] = rgchT[1] = ch;
#endif
dxpDBCS = (fPrinter ?
LOWORD(GetTextExtent(vhDCPrinter,
(LPSTR) rgchT, cchDBCS)) :
LOWORD(GetTextExtent(vhMDC,
(LPSTR) rgchT, cchDBCS)));
#ifndef TEMP_KOREA /* For variable width by sangl 90.12.26 */
/* Store in fmi, if it fits. */
if (0 <= dxpDBCS && dxpDBCS < dxpNil)
#if defined(JAPAN) || defined(KOREA) || defined(TAIWAN) || defined(PRC) //Win3.1 BYTE-->WORD
pfmi->dxpDBCS = (WORD) dxpDBCS;
#else
pfmi->dxpDBCS = (BYTE) dxpDBCS;
#endif
#endif
return (dxpDBCS - pfmi->dxpOverhang);
}
else
return (pfmi->dxpDBCS - pfmi->dxpOverhang);
}
#if defined(KOREA)
else if (*pdxp == dxpNil) {
#else
else {
#endif
int dxp;
#endif /* DBCS */
/* get width from GDI */
dxp = fPrinter ? LOWORD(GetTextExtent(vhDCPrinter, (LPSTR)&ch, 1)) -
vfmiPrint.dxpOverhang : LOWORD(GetTextExtent(vhMDC, (LPSTR)&ch, 1)) -
vfmiScreen.dxpOverhang;
#ifdef DBEMG
CommSzNum("Get this.... ", dxp);
#endif
//(7.24.91) v-dougk if (dxp >= 0 && dxp < dxpNil)
{
/* only store dxp's that fit in a byte */
*pdxp = dxp;
}
#ifdef DBEMG
{
char szT[10];
CommSzSz("fPrinter: ", (fPrinter ? "Printer" : "Screen"));
if (ch == 0x0D) {
szT[0] = 'C'; szT[1] = 'R'; szT[2] = '\0';
}
else if (ch == 0x0A) {
szT[0] = 'L'; szT[1] = 'F'; szT[2] = '\0';
}
else if (32 <= ch && ch <= 126) {
szT[0] = ch; szT[1] ='\0';
}
else if (FKanji1(ch)) {
szT[0] = 'K'; szT[1] = 'A'; szT[2] = 'N'; szT[3] = 'J';
szT[4] = 'I'; szT[5] = '\0';
}
else {
szT[0] = szT[1] = szT[2] = '-'; szT[3] = '\0';
}
CommSzSz("Character: ", szT);
CommSzNum("Dxp: ", (int) dxp);
CommSzNum("OverHang: ", (int) (fPrinter ? vfmiPrint.dxpOverhang : vfmiScreen.dxpOverhang));
}
#endif
return(dxp);
}
#ifdef DBEMG
{
char szT[10];
CommSzSz("fPrinter: ", (fPrinter ? "Printer" : "Screen"));
if (ch == 0x0D) {
szT[0] = 'C'; szT[1] = 'R'; szT[2] = '\0';
}
else if (ch == 0x0A) {
szT[0] = 'L'; szT[1] = 'F'; szT[2] = '\0';
}
else if (32 <= ch && ch <= 126) {
szT[0] = ch; szT[1] ='\0';
}
else if (FKanji1(ch)) {
szT[0] = 'K'; szT[1] = 'A'; szT[2] = 'N'; szT[3] = 'J';
szT[4] = 'I'; szT[5] = '\0';
}
else {
szT[0] = szT[1] = szT[2] = '-'; szT[3] = '\0';
}
CommSzSz("Character: ", szT);
CommSzNum("Dxp: ", (int) *pdxp);
CommSzNum("OverHang: ", (int) (fPrinter ? vfmiPrint.dxpOverhang : vfmiScreen.dxpOverhang));
}
#endif
return(*pdxp);
}
#ifdef DBCS
//
// DxpFromCh for DBCS
// yutakan, 03 Oct 1991
int DBCSDxpFromCh(ch, ch2, fPrinter)
int ch;
int ch2;
int fPrinter;
{
/* T-HIROYN sync us 3.1*/
int *pdxp; // changed to int (7.23.91) v-dougk
int dxpDummy; // changed to int (7.23.91) v-dougk
extern int dxpLogCh;
extern struct FCE *vpfceScreen;
/* If the width is not in the width table, then get it. */
if (ch < chFmiMin)
{
switch (ch)
{
case chTab:
case chEol:
case chReturn:
case chSect:
case chNewLine:
case chNRHFile:
/* the width for these characters aren't really important */
pdxp = (CHAR *)(fPrinter ? &vfmiPrint.dxpSpace : &vfmiScreen.dxpSpace);
break;
default:
pdxp = &dxpDummy;
*pdxp = dxpNil;
break;
}
}
else if (ch >= chFmiMax)
{
/* outside the range we hold in our table - kludge it */
pdxp = &dxpDummy;
*pdxp = dxpNil;
}
else
{
/* inside our table */
pdxp = (fPrinter ? vfmiPrint.mpchdxp : vfmiScreen.mpchdxp) + ch;
}
if (*pdxp == dxpNil )
{
int dxp;
#if defined(TAIWAN) || defined(PRC) //solve BkSp single byte (>0x80) infinite loop problem, MSTC - pisuih, 2/25/93
// BUG 5477: Echen: add NON FA font, LeadByte + 2nd byte checking
if( ch2 != 0 && IsDBCSLeadByte(ch) && GetFontAssocStatus(vhMDC))
#else
if( IsDBCSLeadByte(ch) )
#endif //TAIWAN
{
struct FMI *pfmi;
#if defined(TAIWAN) || defined(KOREA) || defined(PRC) //for Bug# 3362, MSTC - pisuih, 2/10/93
CHAR rgchT[cchDBCS << 1];
int dxpOverhang;
#else
CHAR rgchT[cchDBCS];
#endif //TAIWAN
int dxpT;
int dxpDBCS;
pfmi = fPrinter ? (&vfmiPrint) : (&vfmiScreen);
Assert(pfmi->bDummy == dxpNil);
#if defined(TAIWAN) || defined(KOREA) || defined(PRC) //fix Italic position error while SBCS's overhang != DBCS's overhang
//for Bug# 3362, MSTC - pisuih, 3/4/93
//fix Go To page too slow, pisuih, 3/4/93
if ( (!pfmi->dxpDBCS) || (pfmi->dxpDBCS == dxpNil) )
{
rgchT[0] = rgchT[2] = ch;
rgchT[1] = rgchT[3] = ch2;
dxpDBCS = LOWORD(GetTextExtent( (fPrinter ? vhDCPrinter : vhMDC),
(LPSTR) rgchT, cchDBCS ));
dxpOverhang = (dxpDBCS << 1) - LOWORD( GetTextExtent(
(fPrinter ? vhDCPrinter : vhMDC), (LPSTR) rgchT, cchDBCS << 1 ));
//for compatible with SBCS's overhang
dxpDBCS += (pfmi->dxpOverhang - dxpOverhang);
/* Store in fmi, if it fits. */
if (0 <= dxpDBCS && dxpDBCS < dxpNil)
pfmi->dxpDBCS = (WORD) dxpDBCS;
return (dxpDBCS - pfmi->dxpOverhang);
}
else
return (pfmi->dxpDBCS - pfmi->dxpOverhang);
#else
if(pfmi->dxpDBCS == dxpNil)
{
/* Get the width from GDI. */
rgchT[0] = ch;
rgchT[1] = ch2;
dxpDBCS = (fPrinter ?
LOWORD(GetTextExtent(vhDCPrinter,
(LPSTR) rgchT, cchDBCS)) :
LOWORD(GetTextExtent(vhMDC,
(LPSTR) rgchT, cchDBCS)));
/* Store in fmi, if it fits. */
if (0 <= dxpDBCS && dxpDBCS < dxpNil)
#if defined(JAPAN) || defined(KOREA) //Win3.1 BYTE-->WORD
pfmi->dxpDBCS = (WORD) dxpDBCS;
#else
pfmi->dxpDBCS = (BYTE) dxpDBCS;
#endif
return (dxpDBCS - pfmi->dxpOverhang);
}
else
return (pfmi->dxpDBCS - pfmi->dxpOverhang);
#endif //TAIWAN
}
else
{
/* get width from GDI */
dxp = fPrinter ? LOWORD(GetTextExtent(vhDCPrinter, (LPSTR)&ch, 1)) -
vfmiPrint.dxpOverhang : LOWORD(GetTextExtent(vhMDC, (LPSTR)&ch, 1)) -
vfmiScreen.dxpOverhang;
}
/*T-HIROYN sync us 3.1*/
//(7.24.91) v-dougk if (dxp >= 0 && dxp < dxpNil)
{
/* only store dxp's that fit in a byte */
*pdxp = dxp;
}
return(dxp);
}
return(*pdxp);
}
#endif
/* F F I R S T I C H */
int near FFirstIch(ich)
int ich;
{
/* Returns true iff ich is 0 or preceded only by 0 width characters */
register int ichT;
register int *pdxp = &vfli.rgdxp[0];
for (ichT = 0; ichT < ich; ichT++)
{
if (*pdxp++)
{
return false;
}
}
return true;
}
ValidateMemoryDC()
{
/* Attempt to assure that vhMDC and vhDCPrinter are valid. If we have not
already run out of memory, then vhDCPrinter is guaranteed, but vhMDC may
fail due to out of memory -- it is the callers responsibility to check for
vhMDC == NULL. */
extern int vfOutOfMemory;
extern HDC vhMDC;
extern BOOL vfMonochrome;
extern long rgbText;
extern struct WWD *pwwdCur;
/* If we are out of memory, then we shouldn't try to gobble it up by getting
DC's. */
if (!vfOutOfMemory)
{
if (vhMDC == NULL)
{
/* Create a memory DC compatible with the screen if necessary. */
vhMDC = CreateCompatibleDC(pwwdCur->hDC);
/* Callers are responsible for checking for vhMDC == NULL case */
if (vhMDC != NULL)
{
/* Put the memory DC in transparent mode. */
SetBkMode(vhMDC, TRANSPARENT);
/* If the display is a monochrome device, then set the text
color for the memory DC. Monochrome bitmaps will not be
converted to the foreground and background colors in this case,
we must do the conversion. */
if (vfMonochrome = (GetDeviceCaps(pwwdCur->hDC, NUMCOLORS) ==
2))
{
SetTextColor(vhMDC, rgbText);
}
}
}
/* If the printer DC is NULL then we need to reestablish it. */
if (vhDCPrinter == NULL)
{
GetPrinterDC(FALSE);
/* GetPrinterDC has already called SetMapperFlags() on vhDCPrinter. */
}
}
}
#ifdef DBCS
/* The following two functions are used to determine if a given kanji
(two byte) character (or 1 byte kana letters) should be admitted
to the current line without causing the line break though it is
passed the right margin.
The table below shows which letters are admitted as a hanging character
on a line. The table below should be updated in sync with the code
itself.
Kanji (2-byte) characters
letter first byte second byte half width
hiragana small a 82 9F
i 82 A1
u 82 A3
e 82 A5
o 82 A7
tsu 82 C1
ya 82 E1
yu 82 E3
yo 82 E5
katakana small a 83 40 85 A5
i 83 42 85 A6
u 83 44 85 A7
e 83 46 85 A8
o 83 48 85 A9
tsu 83 62 85 AD
ya 83 83 85 AA
yu 83 85 85 AB
yo 83 87 85 AC
wa 83 8E
ka 83 95
ke 83 96
blank 81 40
horizontal bar (long) 81 5B 85 AE
(med) 81 5C
(short) 81 5D
touten
(Japanese comma) 81 41 85 A2
kuten
(Japanese period) 81 42 85 9F
handakuten 81 4B 85 DD
dakuten 81 4A 85 DC
kagikakko
(closing Japanese parenthesis)
81 76 85 A1
" (2-byte) 81 68 85 41
' (2-byte) 81 66 85 46
} (2-byte) 81 70 85 9D
] (2-byte) 81 6E 85 7C
) (2-byte) 81 6A 85 48
. (at the center) 81 45 85 A3
... 81 63
.. 81 64
closing angle bracket 81 72
closing double angled bracket
81 74
closing double kagikakko
81 78
closing inversed ) 81 7A
closing half angled bracket
81 6C
thinner ' 81 8C
thinner " 81 8D
1-byte kana characters
letter byte
katakana small a A7
i A8
u A9
e AA
o AB
tsu AF
ya AC
yu AD
yo AE
touten
(Japanese comma) A4
kuten
(Japanese period) A1
handakuten DF
dakuten DE
kagikakko
(closing Japanese parenthesis)
A3
. (at the center) A5
The following 1 or 2 byte characters are treated as a hanging character
if the previous character is a 2-byte kanji character.
letter byte
" 22
' 27
} 7D
] 5D
) 29
. 2E
, 2C
; 3B
: 3A
? 3F
! 21
byte 1 byte 2
. 81 44
, 81 43
; 81 47
: 81 46
? 81 48
! 81 49
. 85 4D
, 85 4B
; 85 4A
: 85 49
? 85 5E
! 85 40
*/
BOOL near FSearchChRgch(ch, rgch, ichLim)
CHAR ch;
CHAR *rgch;
int ichLim;
{
int ichMin;
BOOL fFound;
fFound = FALSE;
ichMin = 0;
while (!fFound && ichMin <= ichLim) {
int ichMid;
CHAR chMid;
/* Save on the dereferencing. */
chMid = rgch[ichMid = (ichMin + ichLim) >> 1];
if (ch == chMid) {
fFound = TRUE;
}
else if (ch < chMid) {
ichLim = ichMid - 1;
}
else {
ichMin = ichMid + 1;
}
}
return (fFound);
}
/* FAdmitCh1() returns true if and only if the given ch is a one-byte
kana code for those letters that can appear beyond the right margin. */
BOOL near FAdmitCh1(ch)
CHAR ch;
{
#ifdef JAPAN
if(!vfWordWrap) /*WordWrap off t-Yoshio*/
return FALSE;
return (
(ch == 0xA1) ||
((0xA3 <= ch) && (ch <= 0xA5)) ||
((0xA7 <= ch) && (ch <= 0xAF)) ||
((0xDE <= ch) && (ch <= 0xDF))
);
#else
return(FALSE);
#endif
}
/* FOptAdmitCh1() returns true if and only if the given ch is a
one-byte character that can be admitted to the end of a line
beyond the right margin, if it appears after a kanji character. */
BOOL near FOptAdmitCh1(ch)
CHAR ch;
{
static CHAR rgchOptAdmit1[]
= {0x21, 0x22, 0x27, 0x29, 0x2C, 0x2E, 0x3A, 0x3B,
0x3F, 0x5D, 0x7D};
#if defined(JAPAN) || defined(KOREA)
if(!vfWordWrap) /*WordWrap off t-Yoshio*/
return FALSE;
#endif
return (FSearchChRgch(ch, rgchOptAdmit1,
(sizeof(rgchOptAdmit1) / sizeof(CHAR)) - 1));
}
/* FAdmitCh2() returns true if and only if the given (ch1, ch2) combination
represents a kanji (2-byte) letter that can appear beyond the right
margin. */
BOOL near FAdmitCh2(ch1, ch2)
CHAR ch1, ch2;
{
int dch=0;
#if defined(JAPAN) || defined(KOREA)
if(!vfWordWrap) /*WordWrap off t-Yoshio*/
return FALSE;
#endif
while((dch < MPDCHRGCHIDX_MAC) && (ch1 != mpdchrgchIdx[dch]))
dch++;
if (dch < MPDCHRGCHIDX_MAC) {
return (FSearchChRgch(ch2, mpdchrgch[dch], mpdchichMax[dch] - 1));
}
else {
return (FALSE);
}
}
/* FOptAdmitCh2() returns true if and only if the given (ch1, ch2) is a
two-byte character combination that can be admitted to the end of a line
beyond the right margin, provided it appears after a kanji character. */
BOOL near FOptAdmitCh2(ch1, ch2)
CHAR ch1, ch2;
{
int i=0;
#if defined(JAPAN) || defined(KOREA)
if(!vfWordWrap) /*WordWrap off t-Yoshio*/
return FALSE;
#endif
while ((i < OPTADMIT2IDX_MAC) && (ch1 != OptAdmit2Idx[i]))
i++;
if (i < OPTADMIT2IDX_MAC){
return (FSearchChRgch(ch2, mpdchrgchOptAdmit2[i], OptAdmit2ichMax[i]));
}
else {
return (FALSE);
}
}
/* FOptAdmitCh() returns TRUE if and only if the given (ch1, ch2) can
be a hanging letter at the end of a line. Otherwise, FALSE. If ch1
is equal to '\0', ch2 is treated as a 1-byte character code. */
BOOL FOptAdmitCh(ch1, ch2)
CHAR ch1, ch2;
{
#if defined(JAPAN) || defined(KOREA)
if(!vfWordWrap) /*WordWrap off t-Yoshio*/
return FALSE;
#endif
if (ch1 == '\0') {
return ((ch2 == chSpace) || FAdmitCh1(ch2) || FOptAdmitCh1(ch2));
}
else {
return (FKanjiSpace(ch1, ch2) || FAdmitCh2(ch1, ch2) ||
FOptAdmitCh2(ch1, ch2));
}
}
#endif /* ifdef DBCS */