2020-09-30 16:53:55 +02:00

2823 lines
92 KiB
C

/************************************************************/
/* Windows Write, Copyright 1985-1992 Microsoft Corporation */
/************************************************************/
/* disp.c -- MW display routines */
#define NOKEYSTATE
#define NOSYSCOMMANDS
#define NOSHOWWINDOW
//#define NOATOM
#define NOCLIPBOARD
#define NOGDICAPMASKS
#define NOCTLMGR
#define NOWINSTYLES
//#define NOVIRTUALKEYCODES
#ifndef DBCS
#define NOSYSMETRICS
#endif
#define NOMENUS
#define NOSOUND
#define NOCOMM
#define NOOPENFILE
#define NOWH
#define NOWINOFFSETS
#define NOMETAFILE
#ifndef DBCS
#define NOMB
#endif
#define NODRAWTEXT
#include <windows.h>
#define NOUAC
#include "mw.h"
#include "debug.h"
#include "cmddefs.h"
#include "dispdefs.h"
#include "wwdefs.h"
#define NOKCCODES /* Removes all kc code defines */
#include "ch.h"
#include "docdefs.h"
#include "fmtdefs.h"
#include "propdefs.h"
#include "macro.h"
#include "printdef.h"
#include "fontdefs.h"
#if defined(OLE)
#include "obj.h"
#endif
#ifdef DBCS
#include "dbcs.h"
#include "kanji.h"
#endif
#ifdef DBCS_IME
#include <ime.h>
#ifdef JAPAN
#include "prmdefs.h" //IME3.1J
BOOL ConvertEnable = FALSE;
BOOL bClearCall = TRUE;
#endif /* JAPAN */
#endif /* DBCS_IME */
#ifdef CASHMERE /* No VisiMode in WinMemo */
extern int vfVisiMode;
#endif /* CASHMERE */
extern int vcchBlted;
extern int vidxpInsertCache;
extern int vdlIns;
extern int vfInsLast;
extern struct PAP vpapAbs;
extern struct SEP vsepAbs;
extern int rgval[];
extern struct DOD (**hpdocdod)[];
extern typeCP cpMacCur;
extern int vfSelHidden;
extern struct WWD rgwwd[];
extern int wwCur, wwMac;
extern struct FLI vfli;
extern struct SEL selCur;
extern struct WWD *pwwdCur;
extern int docCur;
extern struct CHP (**vhgchpFormat)[];
extern int vichpFormat;
extern typeCP cpMinCur;
extern typeCP cpMinDocument;
extern int vfInsertOn;
extern int vfTextBltValid;
extern typeCP vcpFirstParaCache;
extern typeCP vcpLimParaCache;
extern unsigned vpgn;
extern struct SEP vsepAbs;
extern CHAR stBuf[];
extern typeCP CpEdge();
extern typeCP CpMacText();
extern int vdocPageCache;
extern int vfPictSel;
extern int vfAwfulNoise;
extern int vfSkipNextBlink;
extern int dypMax;
extern HDC vhMDC;
extern HWND vhWndPageInfo;
extern struct FMI vfmiScreen;
extern int docScrap;
extern long rgbBkgrnd;
extern long ropErase;
extern BOOL vfMonochrome;
extern int dxpbmMDC;
extern int dypbmMDC;
extern HBITMAP hbmNull;
extern int vfOutOfMemory;
extern int vfSeeSel;
extern int vfInsEnd; /* Is insert point at end-of-line? */
extern int vipgd;
extern typeCP vcpMinPageCache;
extern typeCP vcpMacPageCache;
/* actual position of the cursor line */
extern int vxpCursLine;
extern int vypCursLine;
extern int vdypCursLine;
extern int vfScrollInval; /* means scroll did not take and UpdateWw must be repeated */
extern BOOL vfDead;
extern HRGN vhrgnClip;
#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
extern typeCP selUncpFirst;
extern typeCP selUncpLim;
extern HANDLE hImeUnAttrib;
extern int vfImeHidden;
int HiddenTextTop = 0;
int HiddenTextBottom = 0;
//if TRUE we are managing IR_UNDETERMINE
BOOL whileUndetermine = FALSE; //12/28/92
LPSTR Attrib;
WORD AttribPos = 0;
#define IMEDEFCOLORS 6 // IME Define colors
#define IMESPOT 0 // IME Spot background color
#define IMESPOTTEXT 1 // IME Spot text color
#define IMEINPUT 2 // IME Input background color
#define IMEINPUTTEXT 3 // IME Input text color
#define IMEOTHER 4 // IME Other background color
#define IMEOTHERTEXT 5 // IME Other text color
COLORREF rgbIMEHidden[IMEDEFCOLORS] = {0L,0L,0L,0L,0L,0L};
#endif
/* G L O B A L S
int dlsMac = 0;*/
#ifdef DBCS
int donteat = 0; /* propagate not to eat message */
#endif
#if defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
#define FKana(_ch) FALSE
#endif
/* D I S P L A Y F L I */
/* Display formatted line in window ww at line dl */
DisplayFli(ww, dl, fDontDisplay)
int ww;
int dl;
int fDontDisplay; /* True if we set up dl info but don't display */
{
typeCP dcp;
typeCP dcpMac;
struct WWD *pwwd = &rgwwd[ww];
HDC hDC = pwwd->hDC;
int xp; /* Current xp to write text */
int yp; /* Current yp to write text */
int xpMin = pwwd->xpMin; /* Minimum xp in window */
int xpMac = pwwd->xpMac; /* Maximum xp in window */
int ypLine; /* Screen yp for current line */
int dxp; /* Width of current run */
int dyp; /* Line height */
int dxpExtra; /* Width of pad for each space */
typeCP cpMin;
typeCP cpMac;
int xpSel; /* xp of the start of the selection */
int dxpSel = 0; /* Width of the selection. */
CHAR chMark = '\0'; /* style character */
struct CHP *pchp;
BOOL fTabsKludge = (vfli.ichLastTab >= 0);
BOOL fInsertOn = FALSE;
int cBreakRun; /* break characters in run (no relation to Dick or Jane) */
#ifdef SMFONT
RECT rcOpaque;
#endif /* SMFONT */
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
extern int vfWordWrap; /* WordWrap flag : TRUE=ON, FALSE=OFF */
extern int iNonWideSpaces;
int iRun;
typeCP RealcpFirst;
#elif defined(TAIWAN) || defined(PRC)
extern int vfWordWrap; /* WordWrap flag : TRUE=ON, FALSE=OFF */
extern int iNonWideSpaces;
int iRun;
typeCP RealcpFirst;
#endif
#ifdef DDISP
CommSzNumNum(" DisplayFli: dl/fDontDisplay ", dl, fDontDisplay);
#endif
Assert(ww >= 0 && ww < wwMax);
#ifdef SMFONT
Assert(!fDontDisplay || vfli.fGraphics)
#endif /* SMFONT */
Scribble(5,'D');
/* Fill up EDL and set some useful locals */
{
register struct EDL *pedl = &(**pwwd->hdndl)[dl];
if (dl == vdlIns)
{
/* Overwriting chars blted during fast insert; reset blt count */
vcchBlted = 0;
vidxpInsertCache = -1;
}
pedl->xpLeft = vfli.xpLeft;
pedl->xpMac = vfli.xpReal;
cpMin = pedl->cpMin = vfli.cpMin;
#ifdef JAPAN
RealcpFirst = cpMin;
#endif
pedl->dcpMac = (cpMac = vfli.cpMac) - cpMin;
dyp = pedl->dyp = vfli.dypLine;
pedl->ichCpMin = vfli.ichCpMin;
pedl->dcpDepend = (cpMin == cpMac) ? 0xff : vfli.dcpDepend;
pedl->fValid = TRUE;
pedl->fGraphics = vfli.fGraphics;
pedl->fSplat = vfli.fSplat;
/* The position of current line equals the position of the previous line
+ height of this line. */
#ifdef SMFONT
pedl->yp = rcOpaque.bottom = dyp + (ypLine = rcOpaque.top = (dl == 0 ?
pwwd->ypMin : (pedl - 1)->yp));
#else /* not SMFONT */
pedl->yp = dyp + (ypLine = (dl == 0 ? pwwd->ypMin :
(pedl - 1)->yp));
#endif /* SMFONT */
if (pedl->fIchCpIncr = (vfli.ichCpMac != 0))
{
/* Look at final text column */
++cpMac;
/* Since this is true, we can compress pedl->ichCpMac to 1 bit. */
Assert(vfli.ichCpMac == pedl->ichCpMin + 1);
}
}
if (vfli.doc == docNil)
{
/* This is the space beyond the end mark. */
PatBlt(hDC, 0, ypLine, xpMac, dyp, ropErase);
goto Finished;
}
/* Is there a character in the "style bar"? */
if (cpMin != cpMac)
{
#ifdef CASHMERE
/* This line is not completely empty (not after the end mark); check for
painting marks on the style bar. */
if (cpMin == vcpFirstParaCache && vpapAbs.rhc != 0)
{
/* This is a running-head. */
chMark = chStatRH;
}
else if ((**hpdocdod)[vfli.doc].hpgtb != 0)
#else /* not CASHMERE */
if (vpapAbs.rhc == 0 && (**hpdocdod)[vfli.doc].hpgtb != 0)
#endif /* CASHMERE */
{
if (vdocPageCache != vfli.doc || cpMac > vcpMacPageCache || cpMac <=
vcpMinPageCache)
{
CachePage(vfli.doc, cpMac - 1);
}
/* We are now guaranteed that cpMac is within the cached page. */
if (cpMin <= vcpMinPageCache && (!vfli.fGraphics || vfli.ichCpMin ==
0))
{
/* This is the first line of new page; show page mark. */
chMark = chStatPage;
}
}
}
#ifdef SMFONT
#ifdef DDISP
/* black out this line to test how efficiently/correctly we
overwrite pixels from previously-resident lines of text */
PatBlt(hDC, 0, ypLine, xpMac, dyp, BLACKNESS);
{ long int i; for (i=0; i < 500000; i++) ; }
#endif
/* Calculate dcpMac now, so we might be able to know how much to erase. */
dcpMac = vfli.fSplat ? vfli.ichMac : vfli.ichReal;
/* Erase any character that might be in the style bar. */
dxp = xpSelBar + 1;
if (!vfli.fGraphics)
{
dxp = xpMac; // clear the whole line
}
PatBlt(hDC, 0, ypLine, dxp, dyp, ropErase);
/* If this is graphics then go draw any characters in the style bar. */
if (vfli.fGraphics)
{
goto DrawMark;
}
/* If there are no "real" characters on this line then we can skip alot of
this. */
if (dcpMac == 0)
{
goto EndLine2;
}
#else /* not SMFONT */
if (vfli.fGraphics || fDontDisplay)
{
/* Erase any character that might be in the style bar. */
PatBlt(hDC, 0, ypLine, xpSelBar, dyp, ropErase);
goto DrawMark;
}
#endif /* SMFONT */
ValidateMemoryDC();
if (vhMDC == NULL)
{
Error:
/* Notify the user that an error has occured and simply erase this line.
*/
WinFailure();
PatBlt(hDC, xpSelBar, ypLine, xpMac - xpSelBar, dyp, ropErase);
goto Finished;
}
#ifndef SMFONT
/* Create a new bitmap for the memory DC if the current bitmap is not big
enough. */
if (xpMac > dxpbmMDC || dyp > dypbmMDC)
{
HBITMAP hbm;
/* If there is an old bitmap, then delete it. */
if (dxpbmMDC != 0 || dypbmMDC != 0)
{
DeleteObject(SelectObject(vhMDC, hbmNull));
}
/* Create the new bitmap and select it in. */
if ((hbm = CreateBitmap(dxpbmMDC = xpMac, dypbmMDC = dyp, 1, 1,
(LPSTR)NULL)) == NULL)
{
/* There should be a graceful way to recover if the bitmap is ever
NULL (e.g we don't have enough memory for it). */
dxpbmMDC = dypbmMDC = 0;
goto Error;
}
SelectObject(vhMDC, hbm);
}
/* Erase the are of the bitmap we are going to use. */
PatBlt(vhMDC, xpSelBar, 0, xpMac, dyp, vfMonochrome ? ropErase : WHITENESS);
#endif /* not SMFONT */
/* Initialize some of the variables we'll need. */
pchp = &(**vhgchpFormat)[0];
#ifdef SMFONT
xp = rcOpaque.left = rcOpaque.right = vfli.xpLeft + xpSelBar - xpMin + 1;
#else /* not SMFONT */
dcpMac = vfli.fSplat ? vfli.ichMac : vfli.ichReal;
xp = vfli.xpLeft + xpSelBar - xpMin + 1;
#endif /* SMFONT */
dxpExtra = fTabsKludge ? 0 : vfli.dxpExtra;
#ifdef SMFONT
/* If we are horizontally scrolled, then set the clip area to the area
outside of the selection bar. */
if (xpMin != 0)
{
IntersectClipRect(hDC, xpSelBar, rcOpaque.top, xpMac, rcOpaque.bottom);
}
#endif /* SMFONT */
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
iRun = 0;
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
iRun = 0;
#endif
for (dcp = 0; dcp < dcpMac; pchp++)
{
/* For all runs do: */
int ichFirst; /* First character in the current run */
int cchRun; /* Number of characters in the current run */
dcp = ichFirst = pchp->ichRun;
dcp += pchp->cchRun;
if (dcp > dcpMac)
{
dcp = dcpMac;
}
cchRun = dcp - ichFirst;
/* Compute dxp = sum of width of characters in current run (formerly
DxaFromIcpDcp). */
{
register int *pdxp;
register int cchT = cchRun;
PCH pch = vfli.rgch + ichFirst;
dxp = cBreakRun = 0;
pdxp = &vfli.rgdxp[ichFirst];
while (cchT-- > 0)
{
dxp += *pdxp++;
if (*pch++ == chSpace)
++cBreakRun;
}
#ifdef DDISP
CommSzNum(" dxp=",dxp);
#endif
}
if (dxp > 0)
{
int cchDone;
PCH pch = &vfli.rgch[ichFirst];
#ifdef JAPAN // added 08 Jul. 1992 by Hiraisi
int *pdxpT = &vfli.rgdxp[ichFirst];
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
int *pdxpT = &vfli.rgdxp[ichFirst];
#endif
LoadFont(vfli.doc, pchp, mdFontScreen);
yp = (dyp - (vfli.dypBase + (pchp->hpsPos != 0 ? (pchp->hpsPos <
hpsNegMin ? ypSubSuper : -ypSubSuper) : 0))) -
vfmiScreen.dypBaseline;
/* Note: tabs and other special characters are guaranteed to come at
the start of a run. */
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
if( vpapAbs.jc != jcBoth || fTabsKludge )
SetTextJustification(vhMDC, dxpExtra * cBreakRun, cBreakRun);
#elif defined(TAIWAN) || defined(PRC) // Daniel/MSTC, 1993/02/25, for jcBoth
if( vpapAbs.jc != jcBoth)
SetTextJustification(vhMDC, dxpExtra * cBreakRun, cBreakRun);
#else
SetTextJustification(vhMDC, dxpExtra * cBreakRun, cBreakRun);
#endif /* JAPAN */
#ifdef SMFONT
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
if( vpapAbs.jc != jcBoth || fTabsKludge )
SetTextJustification(hDC, dxpExtra * cBreakRun, cBreakRun);
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
if( vpapAbs.jc != jcBoth)
SetTextJustification(hDC, dxpExtra * cBreakRun, cBreakRun);
#else
SetTextJustification(hDC, dxpExtra * cBreakRun, cBreakRun);
#endif /* JAPAN */
#endif /* SMFONT */
cchDone = 0;
while (cchDone < cchRun)
{
int cch;
/* Does the wide-space zone begin in this run? */
if (vfli.fAdjSpace && (vfli.ichFirstWide < ichFirst + cchRun) &&
(ichFirst + cchDone <= vfli.ichFirstWide))
{
int cchDoneT = cchDone;
/* Is this the beginning of the wide-space zone? */
if (ichFirst + cchDone == vfli.ichFirstWide)
{
/* Reset the width of the spaces. */
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
if( vpapAbs.jc != jcBoth || fTabsKludge )
SetTextJustification(vhMDC, ++dxpExtra * cBreakRun,
cBreakRun);
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
if( vpapAbs.jc != jcBoth || fTabsKludge )
SetTextJustification(vhMDC, ++dxpExtra * cBreakRun,
cBreakRun);
#else
SetTextJustification(vhMDC, ++dxpExtra * cBreakRun, cBreakRun);
#endif /* JAPAN */
#ifdef SMFONT
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
if( vpapAbs.jc != jcBoth || fTabsKludge )
SetTextJustification(hDC, dxpExtra * cBreakRun,
cBreakRun);
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
if( vpapAbs.jc != jcBoth)
SetTextJustification(hDC, dxpExtra * cBreakRun,
cBreakRun);
#else
SetTextJustification(hDC, dxpExtra * cBreakRun, cBreakRun);
#endif /* JAPAN */
#endif /* SMFONT */
cch = cchRun - cchDone;
cchDone = cchRun;
}
else
{
cchDone = cch = vfli.ichFirstWide - ichFirst;
}
/* This run is cut short because of a wide space, so we need
to calculate a new width. */
{
register int *pdxp;
register int cchT = cch;
PCH pch = &vfli.rgch[ichFirst + cchDoneT];
dxp = 0;
pdxp = &vfli.rgdxp[ichFirst + cchDoneT];
while (cchT-- > 0)
{
dxp += *pdxp++;
if (*pch++ == chSpace)
++cBreakRun;
}
}
}
else
{
cchDone = cch = cchRun;
}
while (cch > 0)
{
switch (*pch)
{
CHAR ch;
int dxpT;
case chTab:
#ifdef CASHMERE
/* chLeader contains tab leader character (see
FormatLine) */
if ((ch = pchp->chLeader) != chSpace)
{
int cxpTab;
CHAR rgch[32];
int dxpLeader = CharWidth(ch);
int xpT = xp;
int iLevelT = SaveDC(vhMDC);
SetBytes(&rgch[0], ch, 32);
dxpT = vfli.rgdxp[ichFirst];
cxpTab = ((dxpT + dxpLeader - 1) / dxpLeader + 31)
>> 5;
xp += dxpT;
while (cxpTab-- > 0)
{
TextOut(vhMDC, xpT, yp, (LPSTR)rgch, 32);
xpT += dxpLeader << 5;
}
RestoreDC(vhMDC, iLevelT);
}
else
#endif /* CASHMERE */
{
#ifdef SMFONT
/* Expand the opaque rectangle to include the tab.
*/
rcOpaque.right += vfli.rgdxp[ichFirst];
#endif /* SMFONT */
xp += vfli.rgdxp[ichFirst];
}
if (fTabsKludge && ichFirst >= vfli.ichLastTab)
{
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
if( vpapAbs.jc != jcBoth )
SetTextJustification(vhMDC, (dxpExtra =
vfli.dxpExtra) * cBreakRun, cBreakRun);
else
dxpExtra = vfli.dxpExtra;
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
if( vpapAbs.jc != jcBoth )
SetTextJustification(vhMDC, (dxpExtra =
vfli.dxpExtra) * cBreakRun, cBreakRun);
else
dxpExtra = vfli.dxpExtra;
#else
SetTextJustification(vhMDC, (dxpExtra =
vfli.dxpExtra) * cBreakRun, cBreakRun);
#endif /* JAPAN */
#ifdef SMFONT
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
if( vpapAbs.jc != jcBoth )
SetTextJustification(hDC, dxpExtra * cBreakRun,
cBreakRun);
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
if( vpapAbs.jc != jcBoth )
SetTextJustification(hDC, dxpExtra * cBreakRun,
cBreakRun);
#else
SetTextJustification(hDC, dxpExtra * cBreakRun, cBreakRun);
#endif /* JAPAN */
#endif /* SMFONT */
fTabsKludge = FALSE;
}
dxp -= vfli.rgdxp[ichFirst];
pch++;
cch--;
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
iRun++;
pdxpT++;
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
iRun++;
pdxpT++;
#endif
goto EndLoop;
#ifdef CASHMERE
case schPage:
if (!pchp->fSpecial)
{
goto EndLoop;
}
stBuf[0] = CchExpPgn(&stBuf[1], vpgn, vsepAbs.nfcPgn,
flmSandMode, ichMaxLine);
goto DrawSpecial;
case schFootnote:
if (!pchp->fSpecial)
{
goto EndLoop;
}
stBuf[0] = CchExpFtn(&stBuf[1], cpMin + ichFirst,
flmSandMode, ichMaxLine);
DrawSpecial:
#else /* not CASHMERE */
case schPage:
case schFootnote:
if (!pchp->fSpecial)
{
goto EndLoop;
}
stBuf[0] = *pch == schPage && (wwdCurrentDoc.fEditHeader
|| wwdCurrentDoc.fEditFooter) ? CchExpPgn(&stBuf[1],
vpgn, 0, flmSandMode, ichMaxLine) :
CchExpUnknown(&stBuf[1], flmSandMode, ichMaxLine);
#endif /* not CASHMERE */
#ifdef SMFONT
/* Calculate the opaque rectangle. */
rcOpaque.right += vfli.rgdxp[ichFirst] +
vfmiScreen.dxpOverhang;
TextOut(hDC, xp, ypLine+yp, &stBuf[1], stBuf[0]);
#else /* not SMFONT */
TextOut(vhMDC, xp, yp, (LPSTR)&stBuf[1], stBuf[0]);
#endif /* SMFONT */
break;
#ifdef DBCS
#if !defined(JAPAN) && !defined(KOREA)
/* Write ver3.1j endmark is 1charcter t-yoshio May 26,92*/
case chMark1:
if(*(pch+1) == chEMark)
{ /* This run only contains EndMark */
rcOpaque.right += dxp + vfmiScreen.dxpOverhang;
#if defined (TAIWAN) || defined(PRC) // solve italic font overhang truncation problem, MSTC - pisuih, 2/19/93
TextOut(hDC, xp, (yp>0?(ypLine+yp):ypLine), (LPSTR)pch, cch );
#else
ExtTextOut(hDC, xp, (yp>0?(ypLine+yp):ypLine),
2, (LPRECT)&rcOpaque,(LPSTR)pch, cch, (LPSTR)NULL);
#endif
pch += cch;
cch = 0;
xp += dxp;
rcOpaque.left
= (rcOpaque.right = xp) + vfmiScreen.dxpOverhang;
}
/* else fall through */
#endif /*JAPAN*/
#endif
default:
goto EndLoop;
}
dxp -= vfli.rgdxp[ichFirst];
#ifdef SMFONT
/* End the line if no more will fit into the window. */
if ((xp += vfli.rgdxp[ichFirst++]) >= xpMac) {
goto EndLine;
}
rcOpaque.left = (rcOpaque.right = xp) +
vfmiScreen.dxpOverhang;
#else /* not SMFONT */
xp += vfli.rgdxp[ichFirst++];
#endif /* SMFONT */
pch++;
cch--;
#ifdef JAPAN // added 09 Jul. 1992 by Hiraisi
pdxpT++;
#endif
}
EndLoop:
#ifdef SMFONT
if (cch == 0)
{
Assert(dxp == 0);
}
else
{
/* Calculate the opaque rectangle. */
rcOpaque.right += dxp + vfmiScreen.dxpOverhang;
#if 0
{
char msg[180];
wsprintf(msg,"putting out %d characters\n\r",cch);
OutputDebugString(msg);
}
#endif
#ifdef JAPAN // added 19 Jun. 1992 by Hiraisi
if( vpapAbs.jc == jcBoth&&!fTabsKludge ) {
CHAR *ptr1, *ptr2;
int len, cnt;
int iExtra, iSpace, iWid;
BOOL bFlag;
typeCP RealcpEnd;
ptr2 = pch;
for( cnt=0 ; cnt < cch ; ) {
ptr1 = ptr2;
iExtra = dxpExtra;
iWid = len = 0;
bFlag = TRUE;
if( IsDBCSLeadByte(*ptr2) ) {
for( ; cnt < cch ; ) {
iWid += *pdxpT;
pdxpT += 2;
cnt += 2;
len += 2;
iRun += 2;
ptr2 += 2;
if( --iNonWideSpaces == 0 ) {
dxpExtra++;
break;
}
if( iRun == dcp - 2 )
break;
if( iRun == dcp ) { /* lastDBC(maybe) */
iExtra = 0;
break;
}
if( !IsDBCSLeadByte(*ptr2) )
break;
}
}
else {
if( FKana( (int)*ptr2) ) {
for( ; cnt < cch ; ) {
iWid += *pdxpT++;
cnt++;
len++;
iRun++;
ptr2++;
if( --iNonWideSpaces == 0 ) {
dxpExtra++;
break;
}
if( iRun == dcp - 1 )
break;
if( iRun == dcp ) { /* last SBC(maybe) */
iExtra = 0;
break;
}
if( !FKana( (int)*ptr2) )
break;
}
}
else {
for( bFlag = FALSE,iSpace = 0; cnt < cch ; ) {
iWid += *pdxpT++;
cnt++;
len++;
iRun++;
if( *ptr2++ == chSpace || !vfWordWrap ) {
iSpace++;
if( --iNonWideSpaces == 0 ) {
dxpExtra++;
break;
}
}
if( iRun == dcp - 1 )
break;
if( iRun == dcp ) { /* lastSBC(maybe) */
iExtra = 0;
break;
}
if( IsDBCSLeadByte(*ptr2 ) ||
FKana((int)*ptr2) )
break;
}
}
}
if( vfWordWrap && !bFlag ) {
SetTextCharacterExtra( hDC, 0 );
SetTextJustification( hDC,
iExtra*iSpace,iSpace);
}
else {
SetTextJustification(hDC,0,0);
SetTextCharacterExtra(hDC,iExtra);
}
/*-TextOut-*/
#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
/*Undetermine*/
if( (selUncpFirst != selUncpLim) &&
*ptr1 != chEMark) {
RealcpEnd = RealcpFirst + len;
UndetermineString(hDC, xp, ypLine+yp, ptr1,
len, RealcpFirst, RealcpEnd);
} else
TextOut(hDC,xp,ypLine+yp,ptr1,len);
#else
TextOut(hDC,xp,ypLine+yp,ptr1,len);
#endif
RealcpFirst+=len;
xp+=iWid;
}
}
else{
iRun += cch;
SetTextCharacterExtra( hDC, 0 );
/*Undetermine*/
#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
if((selUncpFirst != selUncpLim) && *pch != chEMark)
UndetermineString( hDC, xp, ypLine+yp,
pch, cch, RealcpFirst, RealcpFirst+cch );
else
/* Output cch characters starting at pch */
TextOut(hDC, xp, ypLine+yp, pch, cch);
#else
/* Output cch characters starting at pch */
TextOut(hDC, xp, ypLine+yp, pch, cch);
#endif
/* End the line if no more will fit into the window. */
xp += dxp;
RealcpFirst+=cch;
}
if (xp >= xpMac)
#elif defined(TAIWAN) || defined(PRC) //Daniel/MSTC, 1993/02/25, for jcBoth
if( vpapAbs.jc == jcBoth) { //&& !fTabsKludge ) {
CHAR *ptr1, *ptr2;
int len, cnt;
int iExtra, iSpace, iWid;
BOOL bFlag;
typeCP RealcpEnd;
ptr2 = pch;
for( cnt=0 ; cnt < cch ; ) {
ptr1 = ptr2;
iExtra = dxpExtra;
iWid = len = 0;
bFlag = TRUE;
if( IsDBCSLeadByte(*ptr2) ) {
for( ; cnt < cch ; ) {
iWid += *pdxpT;
pdxpT += 2;
cnt += 2;
len += 2;
iRun += 2;
ptr2 += 2;
if( --iNonWideSpaces == 0 ) {
dxpExtra++;
break;
}
if( iRun == dcp - 2 )
break;
if( iRun == dcp ) { /* lastDBC(maybe) */
iExtra = 0;
break;
}
if( !IsDBCSLeadByte(*ptr2) )
break;
}
}
else {
if( FKana( (int)*ptr2) ) {
for( ; cnt < cch ; ) {
iWid += *pdxpT++;
cnt++;
len++;
iRun++;
ptr2++;
if( --iNonWideSpaces == 0 ) {
dxpExtra++;
break;
}
if( iRun == dcp - 1 )
break;
if( iRun == dcp ) { /* last SBC(maybe) */
iExtra = 0;
break;
}
if( !FKana( (int)*ptr2) )
break;
}
}
else {
for( bFlag = FALSE,iSpace = 0; cnt < cch ; ) {
iWid += *pdxpT++;
cnt++;
len++;
iRun++;
if( *ptr2++ == chSpace || !vfWordWrap ) {
iSpace++;
if( --iNonWideSpaces == 0 ) {
dxpExtra++;
break;
}
}
if( iRun == dcp - 1 )
break;
if( iRun == dcp ) { /* lastSBC(maybe) */
iExtra = 0;
break;
}
if( IsDBCSLeadByte(*ptr2 ) ||
FKana((int)*ptr2) )
break;
}
}
}
if( vfWordWrap && !bFlag ) {
SetTextCharacterExtra( hDC, 0 );
SetTextJustification( hDC,
iExtra*iSpace,iSpace);
}
else {
SetTextJustification(hDC,0,0);
SetTextCharacterExtra(hDC,iExtra);
}
/*-TextOut-*/
//#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
// /*Undetermine*/
// if( (selUncpFirst != selUncpLim) &&
// *ptr1 != chEMark) {
// RealcpEnd = RealcpFirst + len;
// UndetermineString(hDC, xp, ypLine+yp, ptr1,
// len, RealcpFirst, RealcpEnd);
// } else
// TextOut(hDC,xp,ypLine+yp,ptr1,len);
//#else
TextOut(hDC,xp,ypLine+yp,ptr1,len);
//#endif
RealcpFirst+=len;
xp+=iWid;
}
}
else{
iRun += cch;
SetTextCharacterExtra( hDC, 0 );
/*Undetermine*/
//#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
// if((selUncpFirst != selUncpLim) && *pch != chEMark)
// UndetermineString( hDC, xp, ypLine+yp,
// pch, cch, RealcpFirst, RealcpFirst+cch );
// else
// /* Output cch characters starting at pch */
// TextOut(hDC, xp, ypLine+yp, pch, cch);
//#else
/* Output cch characters starting at pch */
TextOut(hDC, xp, ypLine+yp, pch, cch);
//#endif
/* End the line if no more will fit into the window. */
xp += dxp;
RealcpFirst+=cch;
}
if (xp >= xpMac)
#else
/* Output cch characters starting at pch */
#if defined(KOREA)
if ((cch == 1) && (pch[0] == chEMark))
{
int iPrevBkMode;
iPrevBkMode = SetBkMode(hDC, OPAQUE);
TextOut(hDC, xp, ypLine+yp, pch, cch);
SetBkMode(hDC, iPrevBkMode);
}
else
TextOut(hDC, xp, ypLine+yp, pch, cch);
#else
TextOut(hDC, xp, ypLine+yp, pch, cch);
#endif
/* End the line if no more will fit into the window. */
if ((xp += dxp) >= xpMac)
#endif // JAPAN
{
goto EndLine;
}
rcOpaque.left = (rcOpaque.right = xp) +
vfmiScreen.dxpOverhang;
pch += cch;
}
#else /* not SMFONT */
/* Output cch characters starting at pch */
TextOut(vhMDC, xp, yp, (LPSTR)pch, cch);
xp += dxp;
pch += cch;
#endif /* SMFONT */
} /* end while (cchDone<cchRun) */
} /* end if (dxp>0) */
} /* end for dcp=0..dcpMac */
#ifdef SMFONT
EndLine:
/* Restore the clip region if need be. */
if (xpMin != 0)
{
SelectClipRgn(hDC, NULL);
}
EndLine2:
#endif /* SMFONT */
#ifdef CASHMERE
if (vfVisiMode)
{
AddVisiSpaces(ww, &(**pwwd->hdndl)[dl], vfli.dypBase, vfli.dypAfter +
vfli.dypFont);
}
#endif /* CASHMERE */
vfTextBltValid = FALSE;
if ((ww == wwCur) && (pwwd->doc != docScrap) && !vfSelHidden &&
(selCur.cpLim >= cpMin))
{
if (selCur.cpFirst <= cpMac)
{
/* Show selection */
int xpFirst;
int xpLim;
#ifdef ENABLE
if (vfli.fSplatNext && selCur.cpFirst == selCur.cpLim &&
selCur.cpFirst == cpMac)
{
vfInsEnd = TRUE;
ClearInsertLine();
}
vfInsertOn = FALSE;
#endif /* ENABLE */
if (selCur.cpFirst <= cpMin && selCur.cpLim >= cpMac)
{
xpFirst = vfli.xpLeft;
xpLim = vfli.xpReal;
}
else if (selCur.cpFirst < cpMac || (selCur.cpLim == cpMac &&
vfInsEnd))
{
typeCP cpBegin = CpMax(cpMin, selCur.cpFirst);
typeCP cpEnd = CpMin(cpMac, selCur.cpLim);
dxp = DxpDiff((int)(cpBegin - cpMin), (int)(cpEnd - cpBegin),
&xpFirst);
xpLim = min(xpMin + vfli.xpReal, xpFirst + dxp);
}
else
{
goto DidntHighlight;
}
xpSel = xpSelBar + max(xpFirst - xpMin, 0);
if (xpLim > xpFirst)
{
/* Set highlighting at desired screen position. */
dxpSel = max(xpLim - max(xpFirst, xpMin), 0);
}
else if (selCur.cpFirst == selCur.cpLim && ((selCur.cpLim != cpMac)
^ vfInsEnd))
{
vfInsertOn = FALSE; /* Because we redisplayed insert pt line */
#ifdef CASHMERE
vdypCursLine = min(vfli.dypFont, vfli.dypLine - vfli.dypAfter);
vypCursLine = ypLine + dyp - vfli.dypAfter;
#else /* not CASHMERE */
#ifdef DBCS // Some double byte fonts have bigger character
// face than the line hight.
vdypCursLine = min(vfli.dypFont, vfli.dypLine - vfli.dypAfter);
#else
vdypCursLine = vfli.dypFont;
#endif
vypCursLine = ypLine + dyp;
#endif /* not CASHMERE */
vxpCursLine = xpSel;
/* Start blinking in a while */
vfSkipNextBlink = TRUE;
fInsertOn = xpFirst >= xpMin;
}
DidntHighlight:;
}
}
#ifdef SMFONT
/* Invert the selection */
if (dxpSel != 0) {
PatBlt(hDC, xpSel, ypLine, dxpSel, dyp, DSTINVERT);
}
#else /* not SMFONT */
/* Blt the line of text onto the screen. */
PatBlt(vhMDC, 0, 0, xpSelBar, dyp, vfMonochrome ? ropErase : WHITENESS);
if (dxpSel == 0)
{
BitBlt(hDC, 0, ypLine, xpMac, dyp, vhMDC, 0, 0, SRCCOPY);
}
else
{
BitBlt(hDC, 0, ypLine, xpSel, dyp, vhMDC, 0, 0, SRCCOPY);
BitBlt(hDC, xpSel, ypLine, dxpSel, dyp, vhMDC, xpSel, 0, NOTSRCCOPY);
xpSel += dxpSel;
BitBlt(hDC, xpSel, ypLine, xpMac - xpSel, dyp, vhMDC, xpSel, 0,
SRCCOPY);
}
#endif /* SMFONT */
/* Draw the insertion bar if necessary. */
if (fInsertOn)
{
DrawInsertLine();
}
#if defined(JAPAN) & defined(DBCS_IME) // Set a flag for IME management.
else
{
bClearCall = TRUE;
}
#endif
DrawMark:
/* Draw the character in the style bar if necessary. */
if (chMark != '\0')
{
#ifdef SYSENDMARK
#ifdef DBCS // prepare buf for double-byte end mark.
CHAR rgch[cchKanji];
#endif
struct CHP chpT;
extern struct CHP vchpNormal;
blt(&vchpNormal, &chpT, cwCHP);
chpT.ftc = ftcSystem;
chpT.ftcXtra = 0;
chpT.hps = hpsDefault;
/* Draw the style character in the standard font. */
LoadFont(vfli.doc, &chpT, mdFontScreen);
#ifdef DBCS // we use double byte end mark.
#if defined(JAPAN) || defined(KOREA) /*t-yoshio May 26,92*/
#if defined(KOREA)
{
int iPrevBkMode;
iPrevBkMode = SetBkMode(hDC, OPAQUE);
TextOut(hDC, 0, ypLine + dyp - vfli.dypBase - vfmiScreen.dypBaseline,
(LPSTR)&chMark, 1);
SetBkMode(hDC, iPrevBkMode);
}
#else
TextOut(hDC, 0, ypLine + dyp - vfli.dypBase - vfmiScreen.dypBaseline,
(LPSTR)&chMark, 1);
#endif
#else
rgch[0] = chMark1;
rgch[1] = chMark;
rcOpaque.left = 0;
rcOpaque.right = DxpFromCh(chMark1,FALSE);
rcOpaque.top = ypLine;
rcOpaque.bottom = ypLine+dyp;
/* When this line string is lower than system font,it remains dust,
so we must clip this mark. */
ExtTextOut(hDC, 0, ypLine + dyp - vfli.dypBase - vfmiScreen.dypBaseline,
ETO_CLIPPED, (LPRECT)&rcOpaque,(LPSTR)rgch, cchKanji, (LPSTR)NULL);
#endif /*JAPAN*/
#else
TextOut(hDC, 0, ypLine + dyp - vfli.dypBase - vfmiScreen.dypBaseline,
(LPSTR)&chMark, 1);
#endif /* DBCS */
#else /* ifdef SYSENDMARK */
/*T-HIROYN sync 3.0 disp.c */
#if 0
/* Draw the style character in the standard font. */
LoadFont(vfli.doc, NULL, mdFontScreen);
TextOut(hDC, 0, ypLine + dyp - vfli.dypBase - vfmiScreen.dypBaseline,
(LPSTR)&chMark, 1);
#endif
#if defined(JAPAN) || defined(KOREA) /* KenjiK '90-10-26 */
/* Draw the style character in the standard font. */
{
CHAR rgch[cchKanji];
LoadFont(vfli.doc, NULL, mdFontScreen);
rgch[0] = chMark1;
rgch[1] = chMark;
#if defined(KOREA)
{
int iPrevBkMode;
iPrevBkMode = SetBkMode(hDC, OPAQUE);
ExtTextOut(hDC, 0, ypLine + dyp - vfli.dypBase - vfmiScreen.dypBaseline,ETO_CLIPPED, (LPRECT)&rcOpaque,(LPSTR)rgch, cchKanji, (LPSTR)NULL);
SetBkMode(hDC, iPrevBkMode);
}
#else
ExtTextOut(hDC, 0, ypLine + dyp - vfli.dypBase - vfmiScreen.dypBaseline,ETO_CLIPPED, (LPRECT)&rcOpaque,(LPSTR)rgch, cchKanji, (LPSTR)NULL);
#endif
}
#else /* JAPAN */
/* Draw the style character in the standard font. */
LoadFont(vfli.doc, NULL, mdFontScreen);
TextOut(hDC, 0, ypLine + dyp - vfli.dypBase - vfmiScreen.dypBaseline,
(LPSTR)&chMark, 1);
#endif /* JAPAN */
#endif /* if-else-def SYSENDMARK */
}
if (vfli.fGraphics)
{
DisplayGraphics(ww, dl, fDontDisplay);
}
Finished:
Scribble(5,' ');
}
/* D X P D I F F */
DxpDiff(dcpFirst, dcp, pdxpFirst)
int dcpFirst;
int dcp;
int *pdxpFirst;
{
#if 1
register int *pdxp = &vfli.rgdxp[0];
register int cch;
int dxp = vfli.xpLeft;
#ifdef ENABLE /* Not used */
int ichLim = dcpFirst + dcp;
#endif
if (dcp > vfli.ichMac - dcpFirst)
{ /* This should not be, but is when we have a CR */
//Assert( dcpFirst < vfli.ichMac );
dcp = vfli.ichMac - dcpFirst;
}
for (cch = 0; cch < dcpFirst; ++cch)
{
dxp += *pdxp++;
}
*pdxpFirst = dxp;
dxp = 0;
for (cch = 0; cch < dcp; ++cch)
{
dxp += *pdxp++;
}
return dxp;
#else
int dxp;
if (dcp > vfli.ichMac - dcpFirst)
{ /* This should not be, but is when we have a CR */
Assert( dcpFirst < vfli.ichMac );
dcp = vfli.ichMac - dcpFirst;
}
/* first get space up to first character */
*pdxpFirst = LOWORD(GetTextExtent(hDC,vfli.rgch,dcpFirst)) + vfli.xpLeft;
/* now get space between first and first+dcp */
dxp = LOWORD(GetTextExtent(hDC,vfli.rgch+dcpFirst,dcp));
return dxp;
#endif
}
UpdateDisplay(fAbortOK)
int fAbortOK;
{
int ww;
if (wwMac <= 0)
{
return;
}
#ifdef CASHMERE
for (ww = 0; ww < wwMac; ww++)
if ( rgwwd[ww].doc != docScrap )
{
UpdateWw(ww, fAbortOK);
if (rgwwd[ww].fDirty || vfOutOfMemory)
{
return; /* update has been interrupted */
}
}
#else /* not CASHMERE */
UpdateWw(wwDocument, fAbortOK);
if (wwdCurrentDoc.fDirty || vfOutOfMemory)
{
/* Update has been interrupted */
return;
}
#endif /* not CASHMERE */
if (wwdCurrentDoc.fRuler)
{
UpdateRuler();
}
}
/* U P D A T E W W */
UpdateWw(ww, fAbortOK)
int ww, fAbortOK;
{ /* Redisplay ww as necessary */
extern int vfWholePictInvalid;
register struct WWD *pwwd = &rgwwd[ww];
int dlMac;
int dlOld, dlNew;
int doc;
int ichCp;
struct EDL *pedlNew;
register struct EDL *pedl;
struct EDL (**hdndl)[]=pwwd->hdndl;
int dypDiff;
int ypTop;
int ypFirstInval;
int dr;
int fLastNotShown;
typeCP cp, cpMacWw;
if (!pwwd->fDirty)
{
return;
}
if (!((**hpdocdod)[pwwd->doc].fDisplayable))
return;
if (fAbortOK && FImportantMsgPresent())
return;
#if 0 // how to get first and last cp's in invalid rect?
#if defined(OLE)
/*
Load visible objects. Do it now rather than in DisplayGraphics()
because here it has less chance of disrupting the state variables
upon which UpdateWw depends.
*/
ObjEnumInRange(docCur,cpMinCur,cpMacCur,ObjLoadObjectInDoc);
#endif
#endif
dlMac = pwwd->dlMac;
ypTop = pwwd->ypMin;
Assert( ww >= 0 && ww < wwMax );
vfli.doc = docNil; /* An aid to Fast Insert */
UpdateInvalid(); /* InvalBand for what Windows considers to be invalid */
ypFirstInval = pwwd->ypFirstInval;
#ifndef CASHMERE
Assert( ww == wwCur ); /* A MEMO-only assumption */
#endif /* CASHMERE */
Scribble(5, 'U');
ValidateMemoryDC(); /* to do any update, we need a good memory DC */
if (vhMDC == NULL)
{
WinFailure();
return;
}
doc = pwwd->doc;
vfli.doc = docNil;
if (pwwd->fCpBad)
{
/* cp first displayed has not been blessed */
#ifdef CASHMERE /* Must do this if ww != wwCur assertion is FALSE */
int wwT = wwCur;
if (ww != wwCur && wwCur >= 0)
/* CtrBackTrs cache is only good for wwCur. Treat != case */
{
if (pwwdCur->fDirty) /* Do wwCur first, saving cache */
UpdateWw(wwCur, fAbortOK);
if (fAbortOK && FImportantMsgPresent())
return;
ChangeWw(ww, false);
CtrBackDypCtr( 0, 0 ); /* Validate pwwdCur->cpFirst */
ChangeWw(wwT, false);
}
else
#endif /* CASHMERE */
{
if (fAbortOK && FImportantMsgPresent())
return;
CtrBackDypCtr( 0, 0 ); /* Validate pwwdCur->cpFirst */
}
}
/* check for cpMin accessible in this ww */
RestartUpdate:
vfWholePictInvalid = fTrue; /* Tells DisplayGraphics to
abandon accumulated partial pict rect */
fLastNotShown = fFalse;
cp = CpMax(pwwd->cpMin, pwwd->cpFirst);
cpMacWw = pwwd->cpMac;
ichCp = pwwd->ichCpFirst;
/* Note test for dlNew==0 that guarantees that there will be at least
one dl -- this was added for WRITE because we do not have
the ability to enforce a minimum window size */
for (dlNew = dlOld = 0; ypTop < pwwd->ypMac || (dlNew == 0) ; dlNew++)
/* we have: cp, ichCP: pints to text desired on the coming line dlNew
ypTop: desired position for top of dlNew -1
dlOld: next line to be considered for re-use
*/
/* check for having to extend dndl array */
{
if (dlNew >= (int)pwwd->dlMax)
{
/* extend the array with uninitialized dl's, increment max, break if no space.
We assume that dlMac(Old) was <= dlMax, so the dl's will not be looked at
but used only to store new lines */
#define ddlIncr 5
if (!FChngSizeH(hdndl, (pwwd->dlMax + ddlIncr) * cwEDL, fFalse))
break;
pwwd->dlMax += ddlIncr;
}
/* discard unusable dl's */
for (; dlOld < dlMac; dlOld++)
{ /* Set dlOld and pedl to the next good dl */
int ypTopOld, ypOld;
/* Re-entrant Heap Movement */
if (fAbortOK && !fLastNotShown && FImportantMsgPresent())
goto RetInval;
pedl = &(**hdndl)[dlOld];
ypOld = pedl->yp;
/* loop if: invalid, passed over in cp space, passed over in dl space,
passed over in yp space,
in invalid band, passed over in ich space */
if (!pedl->fValid || dlOld < dlNew || pedl->cpMin < cp
|| (ypTopOld = (ypOld - pedl->dyp)) < ypTop
|| (ypOld >= ypFirstInval && ypTopOld <= pwwd->ypLastInval)
|| (pedl->cpMin == cp && pedl->ichCpMin < ichCp))
continue;
/* now we have dlOld, an acceptable if not necessarily useful dl.
now compute dlNew either from scratch or by re-using dlOld. To be
re-useable, dlOld must have right cp/ichCp pair, plus be totally on screen
or, if it is a partial line, it must stay still or move down - not up */
if (pedl->cpMin == cp && pedl->ichCpMin == ichCp &&
(ypOld <= pwwd->ypMac || ypTopOld <= ypTop))
{
/* Re-use this dl */
int yp = ypTop;
if (fLastNotShown)
{
/* HEAP MOVEMENT */
DisplayFli(ww, dlNew - 1, fLastNotShown = fFalse);
pedl = &(**hdndl)[dlOld];
}
cp = pedl->cpMin + pedl->dcpMac;
ichCp = pedl->fIchCpIncr ? pedl->ichCpMin + 1 : 0;
ypTop += pedl->dyp;
if (dlOld != dlNew || ypTopOld != yp)
{
DypScroll(ww, dlOld, dlNew - dlOld, yp);
if (vfScrollInval)
{
/* There was a popup; invalid region might have changed */
/* fLastNotShown test is for interrupting picture display */
/* before we've really displayed it */
(**hdndl) [dlOld].fValid = fFalse;
goto Restart1;
}
dlMac += dlNew - dlOld;
}
dlOld = dlNew + 1;
goto NextDlNew;
}
break;
}
/* cpMin > cp, the line is not anywhere so it will have to be formatted
from scratch */
if (fAbortOK && !fLastNotShown && FImportantMsgPresent())
goto RetInval;
FormatLine(doc, cp, ichCp, cpMacWw, flmSandMode); /* Creates vfli */
if (vfOutOfMemory)
goto RetInval;
ichCp = vfli.ichCpMac;
cp = vfli.cpMac;
/* advance invalid band so that update can resume after an interruption */
pwwd->ypFirstInval = (ypTop += vfli.dypLine);
pedl = &(**hdndl)[dlOld];
if (dlOld < dlMac && pedl->cpMin == cp && pedl->ichCpMin == ichCp)
{
int dlT = dlOld;
/* line at dlOld is a valid, existing line that will abutt the line just about
to be displayed. */
if (dlOld == dlNew && pedl->yp - pedl->dyp <= ypTop)
/* the line about to be overwritten will be re-used in the next loop.
Hence, it is worthwhile to save this line and its dl */
DypScroll(ww, dlOld++, 1, ypTop);
else
/* Move the next line to its abutting position. We know that it has not yet been
overwritten (yp, dlOld all > than ypTop, dlNew) */
DypScroll(ww, dlOld, 0, ypTop);
if (vfScrollInval)
{
/* There was a popup; invalid region might have changed */
/* fLastNotShown test is for interrupting picture display */
/* before we've really displayed it */
(**hdndl) [dlT].fValid = fFalse;
Restart1:
if (fLastNotShown)
{
pwwd->ypFirstInval = pwwd->ypMin;
}
ypFirstInval = pwwd->ypFirstInval;
ypTop = pwwd->ypMin;
goto RestartUpdate;
}
}
/* true in 3rd param means put off picture redisplay till later */
/* condition: graphics & not last in picture & not last in y space and
not in front of a invalid or valid transition in the picture */
DisplayFli(ww, dlNew, fLastNotShown =
(vfli.fGraphics && vfli.ichCpMac!=0 && ypTop < pwwd->ypMac));
NextDlNew:;
}
Break1:
pwwd->dlMac = dlNew;
#ifdef CASHMERE
/* condition is here to avoid swapping */
if (pwwd->fSplit && rgwwd[pwwd->ww].fFtn)
CalcFtnLimits(pwwd);
#endif /* CASHMERE */
SetCurWwVScrollPos(); /* Set Scroll bar position */
vfTextBltValid = false;
/* reset invalid indications */
pwwd->fDirty = false;
pwwd->ypFirstInval = ypMaxAll;
pwwd->ypLastInval = 0; /* so that max in InvalBand will work */
Scribble(5, ' ');
goto Validate;
/* Before returning from an interrupt, invalidate lines that were overwritten
within the present update. */
RetInval:
Scribble(5, ' ');
for (; dlOld < dlMac; dlOld++)
{
pedl = &(**hdndl)[dlOld];
if ((pedl->yp - pedl->dyp) < ypTop)
pedl->fValid = fFalse;
else
break;
}
Validate: ;
#ifdef ENABLE /* We will let UpdateInvalid handle this in case
further invalidation occurred during the update */
{ /* Tell Windows that the part we updated is valid */
RECT rc;
rc.left = 0;
rc.top = pwwd->ypMin;
rc.right = pwwd->xpMac;
rc.bottom = imin( pwwd->ypMac, ypTop );
ValidateRect( pwwd->wwptr, (LPRECT)&rc );
}
#endif
}
/* D Y P S C R O L L */
DypScroll(ww, dlFirst, ddl, ypTo)
int ww, dlFirst, ddl, ypTo;
{
/* Scroll dl's in a window, from dlFirst to end, down ddl lines (or up -ddl).
Bitmap is moved from top of dlFirst to ypTo. The yp's of the dl's are updated.
Returns the amount scrolled. (positive means down). */
register struct WWD *pwwd = &rgwwd[ww];
int dlMac;
int dlT;
int ypFrom;
int dypChange;
int cdlBelow;
struct EDL *pedl;
struct EDL *pedlT;
/* Do not call procedures while dndl is loaded up to avoid heap movement */
struct EDL *dndl = &(**(pwwd->hdndl))[0];
Assert( ww >= 0 && ww < wwMax );
vfScrollInval = fFalse;
/* Number of dl's below (and including) the first one to be scrolled */
cdlBelow = pwwd->dlMac - dlFirst;
pwwd->dlMac = min(pwwd->dlMac + ddl, pwwd->dlMax);
cdlBelow = max(0, min(cdlBelow, pwwd->dlMac - ddl - dlFirst));
pedlT = &dndl[dlFirst];
ypFrom = pedlT->yp - pedlT->dyp;
/* Length of area to be moved */
dypChange = ypTo - ypFrom;
if (cdlBelow > 0)
{
int dlTo = dlFirst + ddl;
int ypMac = pwwd->ypMac;
pedlT = &dndl[dlTo];
if (ddl != 0)
{
blt(&dndl[dlFirst], pedlT, cwEDL * cdlBelow);
}
for (dlT = dlTo; dlT < pwwd->dlMac; ++dlT, ++pedlT)
{
if (dypChange < 0 && pedlT->yp > ypMac)
{
/* Invalidate dl's that are pulled in from the ozone below ypMac
*/
pedlT->fValid = fFalse;
}
else
{
pedlT->yp += dypChange;
}
}
}
if (dypChange != 0)
{
RECT rc;
SetRect( (LPRECT)&rc, 0, min(ypFrom, ypTo),
pwwd->xpMac, pwwd->ypMac );
Assert( ww == wwCur ); /* A MEMO-only assumption */
ScrollCurWw( &rc, 0, dypChange );
}
return dypChange;
}
FImportantMsgPresent()
{
/* If the next message is important enough to interrupt a screen update, we
return TRUE; if it can wait, we return FALSE */
BOOL fToggledKey;
extern MSG vmsgLast;
#ifdef DEBUG
unsigned wHeapVal = *(pLocalHeap + 1);
Assert( wHeapVal == 0 ); /* Heap should not be frozen */
#endif
#ifdef DBCS
if( donteat )
return TRUE;
#endif
while (PeekMessage((LPMSG) &vmsgLast, NULL, NULL, NULL, PM_NOREMOVE))
{
#if defined(JAPAN) & defined(DBCS_IME)
/* If IME Cnv window open,we have to avoid ourselves to get into AlphaMode.
** So we do getmessage here to fill vmsgLast with app queue message.
*/
extern BOOL bImeCnvOpen;
#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
if( bImeCnvOpen || vfImeHidden) {
#else
if( bImeCnvOpen ) {
#endif //JAPAN & IME_HIDDEN
GetMessage((LPMSG) &vmsgLast, NULL, NULL, NULL);
donteat = TRUE;
return TRUE;
}
#endif
/*T-HIROYN sync 3.1 disp.c */
#if 0
/* Filter uninteresting or easily handled events */
if (fToggledKey = FCheckToggleKeyMessage(&vmsgLast) ||
(vmsgLast.message == WM_KEYUP && vmsgLast.hwnd == wwdCurrentDoc.wwptr))
{
if (((vmsgLast.wParam == VK_MENU) || (vmsgLast.wParam == VK_CONTROL)))
return TRUE;
#endif
#ifdef JAPAN // 03/22/93
#ifdef PENWIN // 03/22/93
if (((vmsgLast.message == WM_KEYDOWN) || (vmsgLast.message == WM_KEYUP)
|| (vmsgLast.message == WM_SYSKEYDOWN) || (vmsgLast.message == WM_SYSKEYUP))
&& ((vmsgLast.wParam == VK_MENU) || (vmsgLast.wParam == VK_CONTROL)))
#else //PENWIN
if (((vmsgLast.wParam == VK_MENU) || (vmsgLast.wParam == VK_CONTROL)))
#endif
#elif defined(KOREA)
#define VK_PROCESSKEY 0xE5 // New finalize message. bklee.
// bug #3191, In HWin3.1, WM_NCLBUTTONDOWN message is reached this if statement.
// Because HTBOTTOMDOWN's value is same with VK_CONTROL's, we can't resize write window
if (((vmsgLast.message == WM_KEYDOWN) || (vmsgLast.message == WM_KEYUP)
|| (vmsgLast.message == WM_SYSKEYDOWN) || (vmsgLast.message == WM_SYSKEYUP))
&& ((vmsgLast.wParam == VK_MENU) ||
(vmsgLast.wParam == VK_CONTROL) ||
(vmsgLast.wParam == VK_PROCESSKEY) ))
#else //JAPAN
if (((vmsgLast.wParam == VK_MENU) || (vmsgLast.wParam == VK_CONTROL)))
#endif
{
if (vmsgLast.wParam == VK_CONTROL)
{
GetMessage((LPMSG) &vmsgLast, NULL, NULL, NULL);
SetShiftFlags();
}
return TRUE;
}
/* Filter uninteresting or easily handled events */
else if (fToggledKey = FCheckToggleKeyMessage(&vmsgLast) ||
(vmsgLast.message == WM_KEYUP && vmsgLast.hwnd == wwdCurrentDoc.wwptr))
{
/* This is so the Windows keyboard interface mechanism will see toggle
key and key-up transitions */
GetMessage((LPMSG) &vmsgLast, NULL, NULL, NULL);
#ifdef WIN30
/* PeekMessage has been changed in Win 3.0 so that GetKeyState()
called from FCheckToggleKeyMessage() is really only valid if
you've done a PeekMessage(...,PM_REMOVE) or GetMessage() first.
That is, while the FCheckToggleKeyMessage() call might succeed
above, it will NOT have set the vfShiftKey/vfCommandKey flags
correctly -- so we do it here ..pault */
if (fToggledKey)
FCheckToggleKeyMessage(&vmsgLast);
#endif
if (vmsgLast.hwnd != wwdCurrentDoc.wwptr)
{
/* Just in case a modeless dialog's window proc cares */
TranslateMessage((LPMSG)&vmsgLast);
DispatchMessage((LPMSG)&vmsgLast);
}
#ifdef DBCS /* I hate it */
if (vmsgLast.message == WM_CHAR
#ifdef JAPAN
// We've been reported by one of OEM about LED not disapearing
// problem.
// they know it is bug of write. So I added this code from win2.x
// write source. 27 sep 91 Yutakan
|| (vmsgLast.message == WM_KEYUP && vmsgLast.wParam == VK_KANA)
#endif
#ifdef KOREA
|| vmsgLast.message == WM_INTERIM
#endif
|| vmsgLast.message == WM_KEYDOWN) {
donteat = TRUE;
return( TRUE );
} /* else Ok, you are KEYUP message. do normal */
#endif
}
else
{
switch (vmsgLast.message)
{
case WM_MOUSEMOVE:
#ifdef KOREA
case WM_NCMOUSEMOVE:
#endif
/* Process mouse move messages immediately; they are not really
important. NOTE: This assumes that we have not captured all mouse
events; in which case, they are important. */
DispatchMessage((LPMSG)&vmsgLast);
case WM_TIMER:
case WM_SYSTIMER:
/* Remove timer and mouse move messages from the queue. */
GetMessage((LPMSG) &vmsgLast, NULL, NULL, NULL);
break;
default:
Assert( *(pLocalHeap+1) == 0 ); /* Heap should still not be frozen */
return (TRUE);
}
}
}
Assert( *(pLocalHeap + 1) == 0 ); /* Heap should still not be frozen */
return (FALSE);
}
/* C P B E G I N L I N E */
typeCP CpBeginLine(pdl, cp)
int *pdl;
typeCP cp;
{ /* return the cp and dl containing cp */
int dlMin, dlLim;
typeCP cpGuess;
struct EDL *dndl;
do
{
UpdateWw(wwCur, false);
PutCpInWwVert(cp); /* Ensure cp on screen */
} while (pwwdCur->fDirty && !vfOutOfMemory);
dndl = &(**(pwwdCur->hdndl))[0];
dlMin = 0;
dlLim = pwwdCur->dlMac;
while (dlMin + 1 < dlLim)
{ /* Binary search the ww */
int dlGuess = (dlMin + dlLim) >> 1;
struct EDL *pedl = &dndl[dlGuess];
if ((cpGuess = pedl->cpMin) <= cp && (cpGuess != cp || pedl->ichCpMin == 0))
{ /* guess is low or right */
dlMin = dlGuess;
if (cp == cpGuess && pedl->cpMin + pedl->dcpMac != cp)
break; /* Got it right */
}
else /* Guess is high */
dlLim = dlGuess;
}
*pdl = dlMin;
return dndl[dlMin].cpMin;
}
/* T O G G L E S E L */
ToggleSel(cpFirst, cpLim, fOn)
typeCP cpFirst, cpLim; /* selection bounds */
int fOn;
{ /* Flip selection highlighting on and off */
extern int vfPMS;
struct EDL *pedl;
int dlT;
int xpMin;
int dxpRoom;
int xpFirst;
int xpLim;
int fInsertPoint = (cpFirst == cpLim);
if (vfSelHidden || cpFirst > cpLim || cpLim < /*cp0*/ cpMinCur || vfDead)
return;
if ( vfPictSel && vfPMS &&
(CachePara( docCur, cpFirst ), vpapAbs.fGraphics) &&
(vcpLimParaCache == cpLim) )
{ /* Don't show inversion if we're moving or sizing a picture */
return;
}
dxpRoom = pwwdCur->xpMac - xpSelBar;
xpMin = pwwdCur->xpMin;
for (dlT = 0; dlT < pwwdCur->dlMac; dlT++)
{
typeCP cpMin, cpMac; /* line bounds */
pedl = &(**(pwwdCur->hdndl))[dlT];
if (!pedl->fValid)
continue;
cpMin = pedl->cpMin;
if (cpMin > cpLim || cpMin > cpMacCur || (cpMin == cpLim && cpLim != cpFirst))
break;
cpMac = cpMin + pedl->dcpMac;
if (cpFirst <= cpMin && cpLim >= cpMac)
{
/* entire line is highlighted */
xpFirst = pedl->xpLeft;
if (pedl->fGraphics && cpLim == cpMac && cpMin == cpMac)
/* Special kludge for graphics paras */
xpLim = xpFirst;
else
xpLim = pedl->xpMac;
}
else if (fInsertPoint && cpFirst == cpMac && vfInsEnd)
{ /* Special kludge for an insert point at the end of a line */
xpLim = xpFirst = pedl->xpMac;
}
else if (cpFirst < cpMac)
{
/* Bite the bullet */
int dxp;
typeCP cpBegin = CpMax(cpMin, cpFirst);
typeCP cpEnd = CpMin(cpMac, cpLim);
FormatLine(docCur, cpMin, pedl->ichCpMin, cpMacCur, flmSandMode);
dxp = DxpDiff((int) (cpBegin - cpMin),
(int) (cpEnd - cpBegin), &xpFirst);
xpLim = xpFirst + dxp;
/* reload pedl because procedures were called */
pedl = &(**(pwwdCur->hdndl))[dlT];
}
else
continue;
/* now we have: pedl valid, xpFirst, xpLast describe highlight */
/* xpFirst = max(xpFirst, xpMin); */
xpLim = min(xpLim, xpMin + pedl->xpMac);
if (xpLim > xpFirst)
{
if (xpLim > xpMin)
{
RECT rc;
rc.top = pedl->yp - pedl->dyp;
rc.left = xpSelBar + max(xpFirst - xpMin, 0);
rc.bottom = pedl->yp;
rc.right = xpSelBar + xpLim - xpMin;
InvertRect( wwdCurrentDoc.hDC, (LPRECT)&rc);
}
}
/* ToggleSel modified 7/28/85 -- added explicit check for fInsertPoint, since
the xpLim == xpFirst test sometimes succeeded bogusly when a selection
was extended backwards. BL */
else if (fInsertPoint && (xpLim == xpFirst)) /* Insertion point */
{
/* vfli should usually be cached already, so will be fast. */
int yp = pedl->yp;
FormatLine(docCur, cpMin, pedl->ichCpMin, cpMacCur, flmSandMode);
if (fOn ^ vfInsertOn)
{
if (!vfInsertOn)
{
vxpCursLine = xpSelBar + xpFirst - xpMin;
vypCursLine = yp - vfli.dypAfter;
vdypCursLine = min(vfli.dypFont, vfli.dypLine - vfli.dypAfter);
/* Start blinking in a while */
vfSkipNextBlink = TRUE;
}
DrawInsertLine();
}
return;
}
}
}
/* T R A S H W W */
TrashWw(ww)
{ /* Invalidate all dl's in ww */
Assert( ww >= 0 && ww < wwMax );
InvalBand(&rgwwd[ww], 0, ypMaxAll);
}
/* I N V A L B A N D */
/* invalidate the band ypFirst, ypLast inclusive */
InvalBand(pwwd, ypFirst, ypLast)
struct WWD *pwwd; int ypFirst, ypLast;
{
/* this covers some peculiar rects received from update event after a
window resize by 1 pixel. CS */
if (ypLast < 0 || ypFirst == ypLast) return;
pwwd->fDirty = true;
pwwd->ypFirstInval = min(pwwd->ypFirstInval, ypFirst);
pwwd->ypLastInval = max(ypLast, pwwd->ypLastInval);
}
/* T R A S H A L L W W S */
TrashAllWws()
{ /* trash them all */
int ww;
#ifdef CASHMERE
for (ww = 0; ww < wwMac; ++ww)
TrashWw(ww);
#else
TrashWw( wwDocument );
#endif
vfli.doc = docNil; /* Mark vfli invalid */
}
/* T U R N O F F S E L */
TurnOffSel()
{ /* Remove sel highlighting from screen */
/* HideSel has no effect */
if (!vfSelHidden)
{
ToggleSel(selCur.cpFirst, selCur.cpLim, false);
vfSelHidden = true;
}
}
#ifdef JAPAN
/* We handle IME convert window. */
//int FontHeight = 0; 01/19/93
// 03/29/93 int ImePosSize = 0; //01/19/93
int ImePosSize = 256; //03/29/93 #5484
BOOL bGetFocus = FALSE;
// Handle to the IME communication block - 061491 Yukini
HANDLE hImeMem = NULL;
// to prevent unwanted caret traveling - 061591 Yukini
BOOL bForceBlock = FALSE;
// To avoid ilreagal setting for IME rectangle
static BOOL bResetIMERect=FALSE;
static BOOL bDefaultBlock = FALSE; //IME3.1J t-hiroyn
/*
* ForceImeBlock controls IME input display to the screen.
* When Write program is writing string to the screen, the caret
* doesn't move with string. After completing text display,
* then caret is moved to appropriate position. During this
* period, if user type something by using IME, char is displayed on the
* screen. This makes mixing ugly result. This routine will
* prevent this. Yukini
*/
void ForceImeBlock( hWnd, bFlag )
HWND hWnd;
BOOL bFlag;
{
WORD x, y;
LPIMESTRUCT lpmem;
if (bForceBlock = bFlag) {
if (lpmem = (LPIMESTRUCT)GlobalLock(hImeMem)) {
/* Move bounding rectangle to out of the world */
lpmem->fnc = IME_MOVECONVERTWINDOW;
x = GetSystemMetrics(SM_CXSCREEN);
y = GetSystemMetrics(SM_CYSCREEN);
lpmem->wParam = MCW_SCREEN | MCW_RECT;
#if 1 //01/19/93
lpmem->lParam1 = (DWORD)MAKELONG(x+x/2,y+y/2);
lpmem->lParam2 = (DWORD)MAKELONG(x+x/2,y+y/2);
lpmem->lParam3 = (DWORD)MAKELONG(x*2,y*2);
#else
lpmem->lParam1 = (DWORD)MAKELONG(x+1,y+1);
lpmem->lParam2 = (DWORD)MAKELONG(x+1,y+1);
lpmem->lParam3 = (DWORD)MAKELONG(x*2,y*2);
#endif
GlobalUnlock(hImeMem);
MySendIMEMessageEx(hWnd,MAKELONG(hImeMem,NULL));
bResetIMERect = FALSE;
}
}
}
void DefaultImeBlock(hWnd)
HWND hWnd;
{
LPIMESTRUCT lpmem;
if (bDefaultBlock == FALSE) {
if (lpmem = (LPIMESTRUCT)GlobalLock(hImeMem)) {
/* Move bounding rectangle to default */
lpmem->fnc = IME_MOVECONVERTWINDOW;
lpmem->wParam = MCW_DEFAULT;
GlobalUnlock(hImeMem);
MySendIMEMessageEx(hWnd,MAKELONG(hImeMem,NULL));
bResetIMERect = FALSE;
}
}
}
/* use these veriables to optimize IME call. This will prevent task switching
* overhead. - Yukini
*/
static RECT rcOldRect = {-1, -1, -1, -1};
static DWORD dwCurpos = -1;
static WORD wModeMCW = MCW_DEFAULT;
void SetIMEConvertWindow(hWnd,x,y,bFlag)
HWND hWnd;
int x,y;
BOOL bFlag;
{
LPIMESTRUCT lpmem;
//Yukini:HANDLE hIMEBlock;
DWORD dwXY = MAKELONG(x,y);
BOOL bRetSendIme;
RECT rcRect;
extern BOOL bImeCnvOpen; // Yutakan:08/06/91
/* Do nothing if in text drawing to the screen */
if (bForceBlock)
return;
/* we allocate the Ime communication area. freeing of this
* area will be done by wrap up routine MmwDestroy() of Quit.C.
* This will improve the performance than previous code. - Yukini
*/
if (hImeMem == NULL) {
if ((hImeMem = GlobalAlloc(GMEM_MOVEABLE|GMEM_SHARE|GMEM_LOWER,
(DWORD)sizeof(IMESTRUCT))) == NULL)
return; // something wrong
}
//Yukini: /* Get comunication area with IME */
//Yukini: hIMEBlock=GlobalAlloc(GMEM_MOVEABLE | GMEM_SHARE | GMEM_LOWER,
//Yukini: (DWORD)sizeof(IMESTRUCT));
//Yukini: if(!hIMEBlock) return;
if(!bGetFocus)
bFlag = FALSE;
GetWindowRect(hWnd,&rcRect);
// [yutakan:08/06/91] IF out of Window Rect, force MCW_DEFAULT.
// if(rcRect.top > y || rcRect.bottom < y) bFlag = TRUE;
//
if (bFlag) {
/* Add ResetIMERect check . If we've not done MOVECONVERTWINDOW after
** ForceIMEblock(), don't pass by SendIMEMessage to avoid ilreagal setting
** for bounding rectangle. [Yutakan.]
*/
// bResetIMERect FALSE when just after ForceImeBlock()
if ( bResetIMERect == TRUE
&& dwCurpos == dwXY && EqualRect(&rcRect, &rcOldRect)){
//OutputDebugString("Write:optimized\r\n");
return;
}
} else
dwCurpos = -1; // invalidate cache
//Yukini: lpmem = (LPIMESTRUCT)GlobalLock(hIMEBlock);
//Yukini: lpmem->fnc = IME_MOVECONVERTWINDOW;
//Yukini: lpmem->wParam = bFlag?MCW_WINDOW:MCW_DEFAULT;
//Yukini: lpmem->lParam1 = (DWORD)MAKELONG(x,y);
//Yukini: GlobalUnlock(hIMEBlock);
//Yukini: SendIMEMessage(hWnd,MAKELONG(hIMEBlock,NULL));
//Yukini: GlobalFree(hIMEBlock);
#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
if(vfImeHidden) {
DWORD dwXY1, dwXY2, dwXY3;
int x1,x2,x3,y1,y2,y3;
RECT hiRect;
//12/28/92
if(whileUndetermine == TRUE) // we are managing IR_UNDETERMINE
return;
CopyRect(&hiRect, &rcRect);
if(x < 0) x = 0;
x1 = x + hiRect.left;
y1 = (vypCursLine - vdypCursLine) + hiRect.top;
if(y1 < 0) y1 = 0;
x2 = hiRect.left;
if(HiddenTextTop == 0)
y2 = y1;
else
y2 = hiRect.top + HiddenTextTop;
x3 = hiRect.right;
if(vdypCursLine <= 0)
y3 = y1 + (19+1);
else {
if(HiddenTextBottom == 0)
y3 = y1 + (vdypCursLine+1);
else
y3 = hiRect.top + HiddenTextBottom + 1;
if(y3 < (y1 + (vdypCursLine+1)))
y3 = y1 + (vdypCursLine+1);
}
if(y2 > y1)
y2 = y1;
if(x3 <= x1 || y3 <= y1)
goto dontHidden;
if(x3 <= x2 || y3 <= y2)
goto dontHidden;
dwXY1 = MAKELONG(x1, y1);
dwXY2 = MAKELONG(x2, y2);
dwXY3 = MAKELONG(x3, y3);
if (lpmem = (LPIMESTRUCT)GlobalLock(hImeMem)) {
lpmem->fnc = IME_MOVECONVERTWINDOW;
lpmem->wParam = MCW_HIDDEN;
dwCurpos = dwXY;
CopyRect(&rcOldRect, &rcRect);
lpmem->lParam1 = dwXY1;
lpmem->lParam2 = dwXY2;
lpmem->lParam3 = dwXY3;
GlobalUnlock(hImeMem);
bResetIMERect = TRUE;
if(MySendIMEMessageEx(hWnd,MAKELONG(hImeMem,NULL))) {
vfImeHidden = 1; //Hidden OK
return;
} else {
vfImeHidden = 0; //Hidden ERR we set MCW_WINDOW
}
}
}
dontHidden:
if(selUncpLim > selUncpFirst) {
return;
}
#endif
if (lpmem = (LPIMESTRUCT)GlobalLock(hImeMem)) {
lpmem->fnc = IME_MOVECONVERTWINDOW;
if ((lpmem->wParam = bFlag ? MCW_WINDOW : MCW_DEFAULT) == MCW_WINDOW) {
/* update previous state */
dwCurpos = dwXY;
CopyRect(&rcOldRect, &rcRect);
lpmem->lParam1 = dwXY;
wModeMCW = MCW_WINDOW; //01/25/93
} else {
wModeMCW = MCW_DEFAULT; //01/25/93
}
GlobalUnlock(hImeMem);
if(FALSE == MySendIMEMessageEx(hWnd,MAKELONG(hImeMem,NULL)))
wModeMCW = MCW_DEFAULT; //01/25/93
bResetIMERect = TRUE; // yutakan:08/06/91
}
}
HANDLE hImeSetFont = NULL;
BOOL bImeFontEx = FALSE; //T-HIROYN 02/25/93
SetImeFont(HWND hWnd)
{
LPIMESTRUCT lpmem;
HANDLE hImeLogfont;
LOGFONT lf;
LPLOGFONT lpLogfont;
HANDLE hfont;
void IMEManage(BOOL);
//Get IME use LOGFONT
if (FALSE == GetLogfontImeFont((LPLOGFONT)(&lf)))
return;
if(TRUE == bImeFontEx) {
if ((hImeLogfont = GlobalAlloc(GMEM_MOVEABLE | GMEM_SHARE | GMEM_LOWER,
(DWORD)sizeof(LOGFONT))) == NULL)
return; // something wrong
if (lpmem = (LPIMESTRUCT)GlobalLock(hImeMem)) {
lpmem->fnc = IME_SETCONVERSIONFONTEX;
lpmem->lParam1 = 0L;
if (lpLogfont = (LPLOGFONT)GlobalLock(hImeLogfont)) {
bltbx((LPLOGFONT)(&lf), lpLogfont, sizeof(LOGFONT));
lpmem->lParam1 = MAKELONG(hImeLogfont,NULL);
GlobalUnlock(hImeLogfont);
}
GlobalUnlock(hImeMem);
MySendIMEMessageEx(hWnd,MAKELONG(hImeMem,NULL));
}
GlobalFree(hImeLogfont);
} else {
hfont = CreateFontIndirect(&lf);
if (lpmem = (LPIMESTRUCT)GlobalLock(hImeMem)) {
lpmem->fnc = IME_SETCONVERSIONFONT;
lpmem->wParam = hfont;
GlobalUnlock(hImeMem);
MySendIMEMessageEx(hWnd,MAKELONG(hImeMem,NULL));
}
// prev font deleate;
if(hImeSetFont != NULL) {
HDC hdc;
HANDLE oldhfont;
hdc = GetDC(NULL);
oldhfont = SelectObject(hdc,hImeSetFont);
SelectObject(hdc,oldhfont);
DeleteObject(hImeSetFont);
ReleaseDC(NULL, hdc);
}
hImeSetFont = hfont;
}
//display:
bResetIMERect = FALSE;
IMEManage( FALSE );
}
int GetLogfontImeFont(lplf)
LPLOGFONT lplf;
{
extern struct FCE *vpfceScreen;
extern struct CHP vchpSel;
struct CHP chp;
blt(&vchpSel, &chp, cwCHP); /* CHP for use in comparisons */
//change CHARSET to KANJI_CHARSET
{
int ftc;
if( ftcNil != (ftc = GetKanjiFtc(&chp))) {
ApplyCLooks(&chp, sprmCFtc, ftc);
}
}
LoadFont(docCur, &chp, mdFontJam);
bltbcx(lplf, 0, sizeof(LOGFONT));
GetObject(vpfceScreen->hfont, sizeof(LOGFONT), lplf);
//Ime SETCONVERSIONWINDOW set point (vypCursLine - ImePosSize)
ImePosSize = vfmiScreen.dypBaseline + vfli.dypBase;
if(chp.hpsPos != 0 ) {
if (chp.hpsPos < hpsNegMin)
ImePosSize += ypSubSuper;
else
ImePosSize -= ypSubSuper;
}
return TRUE;
}
void
IMEManage( bFlag )
BOOL bFlag; /* Flag to indicate real default or virtual default in case
of requesting IME convert line to be default. Real default
(value TRUE) is selected when I am loosing the focus. In
this case, IME convert line will be default. Virtual
default (value FALSE) is selected when I lost a caret
position. In this case, IME convert line will be out of
the screen. You can type characters, but not displayed
to the screen correctly. Yukini
*/
{
int x,y;
BOOL bCE = ConvertEnable;
/*IME3.1J
if(!FontHeight)
{
TEXTMETRIC tm;
GetTextMetrics(wwdCurrentDoc.hDC,&tm);
FontHeight = tm.tmHeight;
}
*/
x = vxpCursLine;
// y = vypCursLine-(vdypCursLine+FontHeight)/2; IME3.1J
// y = vypCursLine - FontHeight;
if(ImePosSize > vdypCursLine)
y = vypCursLine - vdypCursLine;
else
y = vypCursLine - ImePosSize;
if (x < 0 || y < 0)
bCE = FALSE; /* Sometime caret position will be
less than zero. Do no set IME
window in this case. Yukini */
#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
if (bCE == FALSE && bFlag == FALSE && bForceBlock == FALSE
&& 0 == vfImeHidden) {
#else
if (bCE == FALSE && bFlag == FALSE && bForceBlock == FALSE) {
#endif
// get out of the world, babe.
//IME3.1J
if( vypCursLine > 0 && ImePosSize > vypCursLine ) {
DefaultImeBlock( wwdCurrentDoc.wwptr); //t-hiroyn
bDefaultBlock = TRUE; //IME3.1J
} else {
ForceImeBlock( wwdCurrentDoc.wwptr, TRUE );
bForceBlock = FALSE;
bDefaultBlock = FALSE; //IME3.1J
}
// yutakan:08/16/91 dwCurpos = 1; // invalidate optimize cache
dwCurpos = -1;
} else {
SetIMEConvertWindow(wwdCurrentDoc.wwptr, x, y, bCE);
bDefaultBlock = FALSE; //IME3.1J
}
}
#endif // JAPAN
#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
//IME3.1J call by MENU selecting.
ChangeImeConversionMode()
{
extern BOOL bImeCnvOpen;
vfImeHidden = (vfImeHidden ? 0 : 1);
// 12/28/92 if(vfImeHidden)
// bImeCnvOpen = TRUE;
bResetIMERect = FALSE;
IMEManage( FALSE );
}
#endif
/* D R A W I N S E R T L I N E */
DrawInsertLine()
{
#if defined(JAPAN) & defined(DBCS_IME)
extern BOOL bImeCnvOpen;
#if defined(IME_HIDDEN)
extern BOOL ImeClearInsert;
#endif
#endif
/* Draw (in Xor mode) a vertical bar at screen position v*CursLine */
/* Toggles both the display and the vfInsertOn flag */
/* Adjustments in cursor draw must be reflected in DisplayFli, above */
/* Last-minute correction for a bug: assure that the insert line
does not extend above ypMin */
if (!vfInsertOn && vdypCursLine > vypCursLine - wwdCurrentDoc.ypMin)
vdypCursLine = vypCursLine - wwdCurrentDoc.ypMin;
/* Tell GDI to invert the caret line */
//#3221 01/25/93
#if defined(JAPAN) & defined(DBCS_IME)
#if defined(IME_HIDDEN)
if(ImeClearInsert || (bImeCnvOpen && wModeMCW == MCW_WINDOW) ) {
#else
if(bImeCnvOpen && wModeMCW == MCW_WINDOW) {
#endif
if(vfInsertOn) {
PatBlt( wwdCurrentDoc.hDC, vxpCursLine,
vypCursLine - vdypCursLine, 2, vdypCursLine , DSTINVERT );
vfInsertOn = 1 - vfInsertOn;
}
} else {
PatBlt( wwdCurrentDoc.hDC, vxpCursLine,
vypCursLine - vdypCursLine, 2, vdypCursLine , DSTINVERT );
vfInsertOn = 1 - vfInsertOn;
}
#else
PatBlt( wwdCurrentDoc.hDC, vxpCursLine, vypCursLine - vdypCursLine,
2, vdypCursLine , DSTINVERT );
vfInsertOn = 1 - vfInsertOn;
#endif
/*T-HIROYN sync 3.0 disp.c */
#ifdef JAPAN /* KenjiK '90-10-30 */
if(bClearCall)
{
if((vxpCursLine>=0) && (vxpCursLine<=wwdCurrentDoc.xpMac)
&& (vypCursLine>=0) && (vypCursLine<=wwdCurrentDoc.ypMac))
ConvertEnable = TRUE;
}
IMEManage( FALSE );
#endif
}
/* C L E A R I N S E R T L I N E */
ClearInsertLine()
{
#if defined(JAPAN) & defined(DBCS_IME)
/* So we do some IME manage when clearning the line */
if((vxpCursLine<0) || (vxpCursLine>=wwdCurrentDoc.xpMac)
|| (vypCursLine<0) || (vypCursLine>=wwdCurrentDoc.ypMac)) {
ConvertEnable = FALSE;
//OutputDebugString("ClearInsertLine\r\n");
}
bClearCall = FALSE;
if ( vfInsertOn) DrawInsertLine();
else IMEManage( FALSE );
bClearCall = TRUE;
#else
if ( vfInsertOn) DrawInsertLine();
#endif
}
#if defined(JAPAN) & defined(IME_HIDDEN) //IME3.1J
UndetermineTextOut(HDC hDC, int xp, int yp, PCH ptr, int cch, LPSTR Attrib)
{
int Length;
long rgbBack;
long rgbText;
int bkMode;
PCH ptr1 = ptr;
TEXTMETRIC tm;
GetTextMetrics(hDC, &tm);
bkMode = SetBkMode(hDC, OPAQUE);
rgbBack = GetBkColor(hDC);
rgbText = GetTextColor(hDC);
while( cch ) {
switch((*Attrib) & 0x03)
{
case 1:
SetBkColor(hDC,
GetNearestColor(hDC, rgbIMEHidden[IMESPOT]));
SetTextColor(hDC,
GetNearestColor(hDC, rgbIMEHidden[IMESPOTTEXT]));
// 12/28/92
if(HiddenTextTop == 0)
HiddenTextTop = yp;
if(HiddenTextTop > yp)
HiddenTextTop = yp;
if(HiddenTextBottom == 0)
HiddenTextBottom = yp + tm.tmHeight;
if(HiddenTextBottom > (yp + tm.tmHeight))
HiddenTextBottom = yp + tm.tmHeight;
break;
case 0:
SetBkColor(hDC,
GetNearestColor(hDC, rgbIMEHidden[IMEINPUT]));
SetTextColor(hDC,
GetNearestColor(hDC, rgbIMEHidden[IMEINPUTTEXT]));
break;
case 2:
default:
SetBkColor(hDC,
GetNearestColor(hDC, rgbIMEHidden[IMEOTHER]));
SetTextColor(hDC,
GetNearestColor(hDC, rgbIMEHidden[IMEOTHERTEXT]));
break;
}
Length = ( (IsDBCSLeadByte((BYTE)(*ptr1)) ) ? 2 : 1 );
TextOut(hDC, xp, yp, ptr1, Length);
xp += LOWORD(GetTextExtent(hDC, ptr1, Length));
xp-=tm.tmOverhang;
ptr1+=Length;
Attrib+=Length;
AttribPos+=Length;
cch-=Length;
}
SetBkColor(hDC,rgbBack);
SetTextColor(hDC,rgbText);
SetBkMode(hDC, bkMode);
}
UndetermineString(HDC hDC, int xp, int yp, PCH ptr, int cch, typeCP cpFirst,
typeCP cpEnd)
{
int Length;
int len = cch;
PCH ptr1;
TEXTMETRIC tm;
GetTextMetrics(hDC, &tm);
ptr1 = ptr;
if((cpEnd <= selUncpFirst) || (cpFirst >= selUncpLim))
TextOut(hDC, xp, yp, ptr1, len);
else {
Attrib = GlobalLock(hImeUnAttrib);
if(cpFirst < selUncpFirst) {
Length = selUncpFirst - cpFirst;
TextOut(hDC, xp, yp, ptr1, Length);
xp+=LOWORD(GetTextExtent(hDC, ptr1, Length));
xp-=tm.tmOverhang;
len-=Length;
ptr1+=Length;
}
if(selUncpLim <= cpEnd) {
if(cpFirst > selUncpFirst) {
Length = (int)(selUncpLim - cpFirst);
Attrib += (cpFirst-selUncpFirst);
}
else {
Length = (int)(selUncpLim - selUncpFirst);
}
UndetermineTextOut(hDC, xp, yp, ptr1, Length, Attrib);
AttribPos = 0;
xp+=LOWORD(GetTextExtent(hDC, ptr1, Length));
xp-=tm.tmOverhang;
ptr1+=Length;
len-=Length;
if ( Length = (int)(cpEnd - selUncpLim) ) {
TextOut(hDC, xp, yp, ptr1, Length);
}
}
else if(Attrib) {
if(cpFirst > selUncpFirst) {
Attrib += (cpFirst-selUncpFirst);
}
UndetermineTextOut(hDC, xp, yp, ptr1,len, Attrib);
}
GlobalUnlock(hImeUnAttrib);
Attrib = NULL;
}
}
DoHiddenRectSend()
{
bResetIMERect = FALSE;
IMEManage( FALSE );
}
GetImeHiddenTextColors()
{
COLORREF NEAR PASCAL GetIMEColor(WORD);
int i;
for (i = 0; i < IMEDEFCOLORS; i++)
rgbIMEHidden[i] = GetIMEColor(i);
}
/*
GetIMEColor -
Retrieve IME color scheme from [colors] in win.ini
*/
#define MAX_SCHEMESIZE 190
COLORREF NEAR PASCAL GetIMEColor(WORD wIndex)
{
static char *pszWinStrings[IMEDEFCOLORS] = {
"IMESpot",
"IMESpotText",
"IMEInput",
"IMEInputText",
"IMEOther",
"IMEOtherText"};
BYTE szTemp[MAX_SCHEMESIZE];
LPBYTE szLP;
COLORREF colRGB;
int i, v;
if (wIndex >= IMEDEFCOLORS)
return RGB(0,0,0);
GetProfileString("colors",
pszWinStrings[wIndex],
"",
szTemp,
MAX_SCHEMESIZE);
if (!lstrlen(szTemp)) {
switch(wIndex) {
case IMESPOT:
return GetSysColor(COLOR_HIGHLIGHTTEXT);
case IMESPOTTEXT:
return GetSysColor(COLOR_HIGHLIGHT);
case IMEINPUT:
return GetSysColor(COLOR_WINDOW);
case IMEINPUTTEXT:
return GetSysColor(COLOR_HIGHLIGHT);
case IMEOTHER:
return GetSysColor(COLOR_WINDOW);
case IMEOTHERTEXT:
return GetSysColor(COLOR_HIGHLIGHT);
}
}
colRGB = RGB(0,0,0);
szLP = szTemp;
for (i = 0; i < 3; i++) {
v = 0;
while(*szLP < '0' || *szLP > '9') szLP++;
while(*szLP >= '0' && *szLP <= '9') v = v * 10 + (*szLP++ - '0');
switch(i) {
case 0:
colRGB |= RGB(v,0,0);
break;
case 1:
colRGB |= RGB(0,v,0);
break;
case 2:
colRGB |= RGB(0,0,v);
break;
}
}
return colRGB;
}
#endif