You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1044 lines
25 KiB
1044 lines
25 KiB
/*
|
|
* @doc INTERNAL
|
|
*
|
|
* @module DISPSL.CPP -- CDisplaySL class |
|
|
*
|
|
* This is the Single-line display engine. See disp.c for the base class
|
|
* methods and dispml.c for the Multi-line display engine.
|
|
*
|
|
* Owner:<nl>
|
|
* Original RichEdit code: David R. Fulmer
|
|
* Christian Fortini
|
|
* Murray Sargent
|
|
*
|
|
* Copyright (c) 1995-2000, Microsoft Corporation. All rights reserved.
|
|
*/
|
|
|
|
#include "_common.h"
|
|
#include "_dispsl.h"
|
|
#include "_measure.h"
|
|
#include "_select.h"
|
|
#include "_render.h"
|
|
#include "_font.h"
|
|
#include "_dfreeze.h"
|
|
|
|
ASSERTDATA
|
|
|
|
const LONG CALC_XSCROLL_FROM_FIRST_VISIBLE = -2;
|
|
|
|
/*
|
|
* CDisplaySL::CDisplaySL
|
|
*
|
|
* Purpose
|
|
* Constructor
|
|
*/
|
|
CDisplaySL::CDisplaySL ( CTxtEdit* ped )
|
|
: CDisplay( ped )
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::CDisplaySL");
|
|
|
|
Assert(!_fMultiLine);
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::Init()
|
|
*
|
|
* @mfunc
|
|
* Init this display for the screen
|
|
*
|
|
* @rdesc
|
|
* TRUE iff initialization succeeded
|
|
*/
|
|
BOOL CDisplaySL::Init()
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::Init");
|
|
|
|
// Initialize our base class
|
|
if(!CDisplay::Init())
|
|
return FALSE;
|
|
|
|
SetWordWrap(FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::InitVars()
|
|
*
|
|
*/
|
|
void CDisplaySL::InitVars()
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::InitVars");
|
|
|
|
_upScroll = 0;
|
|
SetCpFirstVisible(0);
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::RecalcView(fUpdateScrollBars, prc)
|
|
*
|
|
* @mfunc
|
|
* Recalc all lines breaks and update first visible line
|
|
*
|
|
* @rdesc
|
|
* TRUE if success
|
|
*/
|
|
BOOL CDisplaySL::RecalcView(
|
|
BOOL fUpdateScrollBars, //@param TRUE - update scroll bars
|
|
RECTUV* prc)
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::RecalcView");
|
|
LONG dupOld = _dup;
|
|
|
|
if(!RecalcLine())
|
|
return FALSE;
|
|
|
|
if(_fViewChanged)
|
|
{
|
|
if(IsActive() || _dup <= GetDupView())
|
|
{
|
|
_upScroll = 0;
|
|
SetCpFirstVisible(0);
|
|
}
|
|
else if(CALC_XSCROLL_FROM_FIRST_VISIBLE == _upScroll)
|
|
{
|
|
// In this case we want to set our upScroll by a visible. The
|
|
// only way to get here is if the active view has been cloned
|
|
// for displaying an inactive view.
|
|
_upScroll = 0; // Assume that first visible is 0
|
|
if(GetFirstVisibleCp()) // Check first visible
|
|
{
|
|
CMeasurer me(this); // Start at cp 0
|
|
me.NewLine(*this); // Measure from there to where we are
|
|
|
|
// Scroll is length to character
|
|
_upScroll = CLine::UpFromCch(me, GetFirstVisibleCp(), TA_TOP);
|
|
}
|
|
}
|
|
if(fUpdateScrollBars)
|
|
UpdateScrollBar(SB_HORZ, TRUE);
|
|
|
|
_fViewChanged = FALSE;
|
|
}
|
|
|
|
// We only resize if width of single line control has changed.
|
|
if(_dup != dupOld)
|
|
{
|
|
if (FAILED(RequestResize()))
|
|
_ped->GetCallMgr()->SetOutOfMemory();
|
|
|
|
else if (prc && _ped->_fInOurHost) /*bug fix# 5830, forms3 relies on old behavior*/
|
|
_ped->TxGetClientRect(prc);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::RecalcLine()
|
|
*
|
|
* @mfunc
|
|
* Recalculate a line
|
|
*
|
|
* @rdesc
|
|
* TRUE if success <nl>
|
|
* FALSE if failure <nl>
|
|
*/
|
|
BOOL CDisplaySL::RecalcLine()
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::RecalcLine");
|
|
|
|
Assert( _ped );
|
|
|
|
// Create a measurer starting at cp = 0
|
|
CMeasurer me(this);
|
|
|
|
LONG dupOld = _dup;
|
|
BOOL fMeasured = CLine::Measure(me, MEASURE_FIRSTINPARA);
|
|
|
|
if(!fMeasured)
|
|
{
|
|
Assert(FALSE); // Should succeed
|
|
InitVars();
|
|
return FALSE;
|
|
}
|
|
|
|
_fNeedRecalc = FALSE;
|
|
_fRecalcDone = TRUE;
|
|
|
|
if(dupOld != _dup)
|
|
_fViewChanged = TRUE;
|
|
|
|
_fLineRecalcErr = FALSE;
|
|
return fMeasured;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::Render(rcView, rcRender)
|
|
*
|
|
* @mfunc
|
|
* Renders this line
|
|
*/
|
|
void CDisplaySL::Render(
|
|
const RECTUV &rcView, //@parm View RECT
|
|
const RECTUV &rcRender) //@parm RECT to render (must be contained in
|
|
// client rect)
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::Render");
|
|
POINTUV pt;
|
|
|
|
_fRectInvalid = FALSE;
|
|
|
|
CRenderer re(this);
|
|
|
|
if(!re.StartRender(rcView, rcRender))
|
|
return;
|
|
|
|
// Set renderer at top/left of view rect
|
|
pt.u = rcView.left - _upScroll;
|
|
pt.v = rcView.top;
|
|
re.SetCurPoint(pt);
|
|
// Renderer is set at cp = 0 at the moment
|
|
CLine::Render(re, TRUE);
|
|
re.EndRender(this, this, 0, pt);
|
|
|
|
if(_fOffscreenOnce)
|
|
_fUseOffscreenDC = _fOffscreenOnce = FALSE;
|
|
|
|
// If line metrics are not yet up to date, get them from renderer
|
|
if(_dup == -1)
|
|
{
|
|
_dup = re._li._dup;
|
|
_dvpHeight = re._li._dvpHeight;
|
|
_dvpDescent = re._li._dvpDescent;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::WaitForRecalcIli(ili)
|
|
*
|
|
* @mfunc
|
|
* Wait until line array is recalculated up to line <p ili>
|
|
*
|
|
* @rdesc
|
|
* Returns TRUE if lines were recalc'd up to ili (TRUE if ili == 0)
|
|
*/
|
|
BOOL CDisplaySL::WaitForRecalcIli (
|
|
LONG ili) //@parm Line index to recalculate line array up to
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::WaitForRecalcIli");
|
|
|
|
return ili == 0;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::GetScrollRange(nBar)
|
|
*
|
|
* @mfunc
|
|
* Returns the max part of a scrollbar range for scrollbar <p nBar>
|
|
*
|
|
* @rdesc
|
|
* LONG max part of scrollbar range
|
|
*/
|
|
LONG CDisplaySL::GetScrollRange(
|
|
INT nBar) const //@parm Scroll bar to interrogate (SB_VERT or SB_HORZ)
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetScrollRange");
|
|
|
|
Assert( IsMain() );
|
|
|
|
LONG lRange = 0;
|
|
|
|
if(nBar != SB_VERT && _fUScrollEnabled)
|
|
{
|
|
if(_ped->TxGetScrollBars() & WS_HSCROLL)
|
|
{
|
|
lRange = max(0, _dup + _ped->GetCaretWidth());
|
|
lRange = min(lRange, _UI16_MAX);
|
|
}
|
|
}
|
|
return lRange;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::UpdateScrollBar(nBar, fUpdateRange)
|
|
*
|
|
* @mfunc
|
|
* Update horizontal scroll bar
|
|
* Also figure whether the scroll bar should be visible or not
|
|
*
|
|
* @rdesc
|
|
* BOOL
|
|
*/
|
|
BOOL CDisplaySL::UpdateScrollBar (
|
|
INT nBar, //@parm Which scrollbar: SB_HORZ or SB_VERT
|
|
BOOL fUpdateRange) //@parm Should range be recomputed and updated
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::UpdateScrollBar");
|
|
|
|
// Note: In the old days we didn't allow autosize & scroll bars, so to keep
|
|
// forms working, we need this special logic with respect to autosize.
|
|
if (!IsActive() || SB_VERT == nBar ||
|
|
!_ped->fInOurHost() && _ped->TxGetAutoSize())
|
|
{
|
|
// Scroll bars are only updated on active views.
|
|
return FALSE;
|
|
}
|
|
|
|
const DWORD dwScrollBars = _ped->TxGetScrollBars();
|
|
const BOOL fHide = !(dwScrollBars & ES_DISABLENOSCROLL);
|
|
BOOL fReturn = FALSE;
|
|
BOOL fEnabled = TRUE;
|
|
BOOL fEnabledOld = FALSE;
|
|
LONG lScroll = 0;
|
|
CTxtSelection *psel = _ped->GetSelNC();
|
|
|
|
// Get scrolling position
|
|
if(nBar == SB_HORZ)
|
|
{
|
|
if(!(dwScrollBars & WS_HSCROLL))
|
|
{
|
|
// even if we don't have scrollbars, we may allow horizontal
|
|
// scrolling.
|
|
|
|
if(!_fUScrollEnabled)
|
|
_fUScrollEnabled = !!(dwScrollBars & ES_AUTOHSCROLL);
|
|
return FALSE;
|
|
}
|
|
|
|
fEnabledOld = _fUScrollEnabled;
|
|
lScroll = ConvertUPosToScrollPos(_upScroll);
|
|
|
|
if(_dup <= _dupView)
|
|
fEnabled = FALSE;
|
|
}
|
|
|
|
// !s beforehand because all true values aren't necessarily equal
|
|
if(!fEnabled != !fEnabledOld)
|
|
{
|
|
if(_fDeferUpdateScrollBar)
|
|
_fUpdateScrollBarDeferred = TRUE;
|
|
else
|
|
{
|
|
if (nBar == SB_HORZ)
|
|
_fUScrollEnabled = fEnabled;
|
|
else
|
|
_fVScrollEnabled = fEnabled;
|
|
}
|
|
|
|
if(!_fDeferUpdateScrollBar)
|
|
{
|
|
if(!fHide) // Don't hide scrollbar, just disable
|
|
_ped->TxEnableScrollBar(nBar, fEnabled ? ESB_ENABLE_BOTH : ESB_DISABLE_BOTH);
|
|
else
|
|
{
|
|
fReturn = TRUE;
|
|
// Make sure to hide caret before showing scrollbar
|
|
if(psel)
|
|
psel->ShowCaret(FALSE);
|
|
|
|
// Hide or show scroll bar
|
|
_ped->TxShowScrollBar(nBar, fEnabled);
|
|
|
|
if(psel)
|
|
psel->ShowCaret(TRUE);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Set scrollbar range and thumb position
|
|
if(fEnabled)
|
|
{
|
|
if(fUpdateRange)
|
|
{
|
|
if(!_fDeferUpdateScrollBar)
|
|
_ped->TxSetScrollRange(nBar, 0, GetScrollRange(nBar), FALSE);
|
|
}
|
|
|
|
if(_fDeferUpdateScrollBar)
|
|
_fUpdateScrollBarDeferred = TRUE;
|
|
else
|
|
_ped->TxSetScrollPos(nBar, lScroll, TRUE);
|
|
}
|
|
else if (!_fDeferUpdateScrollBar)
|
|
{
|
|
// This turns off the scroll bar and only needs to happen when a change
|
|
// occurs so we can count on the change in state check above to set
|
|
// _fUpdateScrollBarDeferred.
|
|
if (!fEnabled && fEnabledOld)
|
|
_ped->TxSetScrollRange(nBar, 0, 0, FALSE);
|
|
}
|
|
return fReturn;
|
|
}
|
|
|
|
BOOL CDisplaySL::IsMain() const
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::IsMain");
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
LONG CDisplaySL::GetDupLineMax() const
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetWidth");
|
|
|
|
return CLine::_dup;
|
|
}
|
|
|
|
LONG CDisplaySL::GetHeight() const
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetHeight");
|
|
|
|
return CLine::_dvpHeight;
|
|
}
|
|
|
|
LONG CDisplaySL::GetResizeHeight() const
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetResizeHeight");
|
|
|
|
return CLine::_dvpHeight;
|
|
}
|
|
|
|
LONG CDisplaySL::LineCount() const
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::LineCount");
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::GetCliVisible()
|
|
*
|
|
* @mfunc
|
|
* Get count of visible lines and update GetCp()MostVisible for PageDown()
|
|
*
|
|
* @rdesc
|
|
* count of visible lines
|
|
*/
|
|
LONG CDisplaySL::GetCliVisible (
|
|
LONG* pcpMostVisible, //@parm Returns cpMostVisible
|
|
BOOL fLastCharOfLastVisible) const //@parm Want cp of last visible char
|
|
// (ignored here).
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetCliVisible");
|
|
|
|
if (pcpMostVisible)
|
|
*pcpMostVisible = CLine::_cch;
|
|
|
|
return 1;
|
|
}
|
|
|
|
LONG CDisplaySL::GetFirstVisibleLine() const
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetFirstVisibleLine");
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::GetLineText(ili, pchBuff, cchMost)
|
|
*
|
|
* @mfunc
|
|
* Copy given line of this display into a character buffer
|
|
*
|
|
* @rdesc
|
|
* number of character copied
|
|
*/
|
|
LONG CDisplaySL::GetLineText (
|
|
LONG ili, //@parm Line to get text of
|
|
TCHAR * pchBuff, //@parm Buffer to stuff text into
|
|
LONG cchMost) //@parm Length of buffer
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetLineText");
|
|
|
|
if(ili == 0)
|
|
{
|
|
cchMost = min(cchMost, _ped->GetTextLength());
|
|
if(cchMost > 0)
|
|
{
|
|
CTxtPtr tp(_ped, 0);
|
|
return tp.GetText( cchMost, pchBuff );
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::CpFromLine(ili, pdvp)
|
|
*
|
|
* @mfunc
|
|
* Computes cp at start of given line
|
|
* (and top of line position relative to this display)
|
|
*
|
|
* @rdesc
|
|
* cp of given line; here always 0
|
|
*/
|
|
LONG CDisplaySL::CpFromLine (
|
|
LONG ili, //@parm Line we're interested in (if <lt> 0 means caret line)
|
|
LONG *pdvp) //@parm Returns top of line relative to display
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::CpFromLine");
|
|
// (NULL if don't want that info)
|
|
Assert( ili == 0 );
|
|
|
|
if(pdvp)
|
|
*pdvp = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::LineFromCp(cp, fAtEnd)
|
|
*
|
|
* @mfunc
|
|
* Computes line containing given cp.
|
|
*
|
|
* @rdesc
|
|
* index of line found; here returns 0 always
|
|
*/
|
|
LONG CDisplaySL::LineFromCp(
|
|
LONG cp, //@parm cp to look for
|
|
BOOL fAtEnd) //@parm If true, return previous line for ambiguous cp
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::LineFromCp");
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::CpFromPoint(pt, prcClient, ptp, prp, fAllowEOL, pHit, pdispdim)
|
|
*
|
|
* @mfunc
|
|
* Determine cp at given point
|
|
*
|
|
* @devnote
|
|
* --- Use when in-place active only ---
|
|
*
|
|
* @rdesc
|
|
* Computed cp, -1 if failed
|
|
*/
|
|
LONG CDisplaySL::CpFromPoint(
|
|
POINTUV pt, //@parm Point to compute cp at (client coords)
|
|
const RECTUV *prcClient,//@parm Client rectangle (can be NULL if active).
|
|
CRchTxtPtr * const ptp, //@parm Returns text pointer at cp (may be NULL)
|
|
CLinePtr * const prp, //@parm Returns line pointer at cp (may be NULL)
|
|
BOOL fAllowEOL, //@parm Click at EOL returns cp after CRLF
|
|
HITTEST * phit, //@parm Out parm for hit-test value
|
|
CDispDim * pdispdim, //@parm Out parm for display dimensions
|
|
LONG *pcpActual, //@parm Out cp that pt is above
|
|
CLine * pliParent) //@parm Parent pli for table row displays
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::CpFromPoint");
|
|
|
|
RECTUV rcView;
|
|
LONG cp = 0;
|
|
LONG cch = 0;
|
|
|
|
HITTEST hit = HT_Nothing;
|
|
|
|
GetViewRect(rcView, prcClient);
|
|
|
|
if (pt.v >= rcView.top && pt.v <= rcView.bottom)
|
|
{
|
|
|
|
// Create measurer at cp(0)
|
|
CMeasurer me(this);
|
|
|
|
// Get character in the line
|
|
pt.u += _upScroll - rcView.left; // Transform to galley coordinates
|
|
pt.v -= rcView.top;
|
|
cch = CLine::CchFromUp(me, pt, pdispdim, &hit, pcpActual);
|
|
|
|
// Don't allow click at EOL to select EOL marker and take into account
|
|
// single line edits as well
|
|
if(!fAllowEOL && cch == CLine::_cch && CLine::_cchEOP)
|
|
me._rpTX.BackupCRLF();
|
|
|
|
cp = me.GetCp();
|
|
}
|
|
|
|
if (phit)
|
|
*phit = hit;
|
|
if(ptp)
|
|
ptp->SetCp(cp);
|
|
if(prp)
|
|
prp->Set(0, cch);
|
|
|
|
return cp;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::PointFromTp(tp, fAtEnd, pt, prp, taMode, pdx)
|
|
*
|
|
* @mfunc
|
|
* Determine coordinates at given tp
|
|
*
|
|
* @devnote
|
|
* --- Use when in-place active only ---
|
|
*
|
|
* @rdesc
|
|
* line index at cp, -1 if error
|
|
*/
|
|
LONG CDisplaySL::PointFromTp(
|
|
const CRchTxtPtr &rtp, //@parm Text ptr to get coordinates at
|
|
const RECTUV *prcClient,//@parm Client rectangle (can be NULL if active).
|
|
BOOL fAtEnd, //@parm Return end of previous line for ambiguous cp
|
|
POINTUV & pt, //@parm Returns point at cp in client coords
|
|
CLinePtr * const prp, //@parm Returns line pointer at tp (may be null)
|
|
UINT taMode, //@parm Text Align mode: top, baseline, bottom
|
|
CDispDim * pdispdim) //@parm Out parm for display dimensions
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::PointFromTp");
|
|
|
|
LONG dy = 0;
|
|
RECTUV rcView;
|
|
CLinePtr rp(this);
|
|
|
|
if(!rp.SetCp(rtp.GetCp(), fAtEnd))
|
|
return -1;
|
|
|
|
AssertSz(_ped->_fInPlaceActive || prcClient,
|
|
"CDisplaySL::PointFromTp() called with invalid client rect");
|
|
|
|
GetViewRect(rcView, prcClient);
|
|
|
|
pt.u = rcView.left - _upScroll;
|
|
pt.v = rcView.top;
|
|
|
|
CMeasurer me(this, rtp);
|
|
me.Move(-rp.GetIch()); // Backup to start of line
|
|
me.NewLine(*rp); // Measure from there to where we are
|
|
|
|
pt.u += CLine::UpFromCch(me, rp.GetIch(), taMode, pdispdim, &dy);
|
|
pt.v += dy;
|
|
|
|
if(prp)
|
|
*prp = rp;
|
|
return rp;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::UpdateView(&tpFirst, cchOld, cchNew)
|
|
*
|
|
* @mfunc
|
|
* Update visible part of display (the "view" on the screen).
|
|
*
|
|
* @devnote
|
|
* --- Use when in-place active only ---
|
|
*
|
|
* @rdesc
|
|
* TRUE if success
|
|
*/
|
|
BOOL CDisplaySL::UpdateView(
|
|
CRchTxtPtr &tpFirst, //@parm Text ptr where change happened
|
|
LONG cchOld, //@parm Count of chars deleted
|
|
LONG cchNew) //@parm Count of chars inserted
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::UpdateView");
|
|
|
|
BOOL fUpdateScrollBarHorz = FALSE;
|
|
BOOL fReturn = TRUE;
|
|
RECTUV rcView;
|
|
CTxtSelection *psel = _ped->GetSelNC();
|
|
LONG dupOld = _dup;
|
|
BOOL fScrollChanged = FALSE;
|
|
RECTUV rcClient;
|
|
RECTUV rc;
|
|
LONG dvpOld = _dvpHeight;
|
|
|
|
if (_fNoUpdateView)
|
|
return fReturn;
|
|
|
|
AssertSz(_ped->_fInPlaceActive, "CDisplaySL::UpdateView(...) called when inactive");
|
|
|
|
_ped->TxGetClientRect(&rcClient);
|
|
GetViewRect(rcView, &rcClient);
|
|
|
|
if(psel && !psel->PuttingChar())
|
|
psel->ClearCchPending();
|
|
|
|
if(!RecalcLine())
|
|
{
|
|
// We're in deep crap now, the recalc failed. Let's try to get
|
|
// out of this with our head still mostly attached
|
|
fReturn = FALSE;
|
|
}
|
|
|
|
// An update has occurred. Go offscreen to be flicker-free.
|
|
if(!_fUseOffscreenDC)
|
|
_fUseOffscreenDC = _fOffscreenOnce = TRUE;
|
|
|
|
//If we tried to go offscreen, but the edit changed the first
|
|
//character, it might be the WM_SETTEXT case called when dialogs
|
|
//are instantiated, and so it doesn't make sense to go offscreen there.
|
|
if (_fOffscreenOnce && tpFirst.GetCp() == 0)
|
|
_fOffscreenOnce = _fUseOffscreenDC = FALSE;
|
|
|
|
if(_dup <= _dupView)
|
|
{
|
|
// x scroll range is smaller than the view width
|
|
// force x scrolling position = 0
|
|
_upScroll = 0;
|
|
SetCpFirstVisible(0);
|
|
_fViewChanged = TRUE;
|
|
fUpdateScrollBarHorz = TRUE;
|
|
}
|
|
|
|
_fRectInvalid = TRUE;
|
|
|
|
// Only resize a Single Line edit control if the width changes
|
|
if(_dup != dupOld)
|
|
{
|
|
if(FAILED(RequestResize()))
|
|
_ped->GetCallMgr()->SetOutOfMemory();
|
|
}
|
|
|
|
// If view changed, update scroll bars
|
|
if(_fViewChanged)
|
|
{
|
|
_fViewChanged = FALSE;
|
|
fScrollChanged = UpdateScrollBar(SB_HORZ);
|
|
}
|
|
|
|
// Scroll bar state did not change so we need to update screen
|
|
// Build an invalidation rectangle.
|
|
rc = rcClient;
|
|
if(dvpOld == _dvpHeight)
|
|
{
|
|
// Height of control did not change so we can minimize the update
|
|
// rectangle to the height of the control.
|
|
rc.bottom = rcView.top + _dvpHeight;
|
|
}
|
|
|
|
// Tell display to update when it gets a chance
|
|
_ped->TxInvalidateRect(&rc);
|
|
|
|
return fReturn;
|
|
}
|
|
|
|
|
|
/*
|
|
* CDisplaySL::ScrollView(upScroll, vpScroll, fTracking, fFractionalScroll)
|
|
*
|
|
* @mfunc
|
|
* Scroll view to new x and y position
|
|
*
|
|
* @devnote
|
|
* This method tries to adjust the y scroll pos before
|
|
* scrolling to display complete line at top. x scroll
|
|
* pos is adjusted to avoid scrolling all text off the
|
|
* view rectangle.
|
|
*
|
|
* Must be able to handle vpScroll <gt> pdp->yHeight and vpScroll <lt> 0
|
|
*
|
|
* @rdesc
|
|
* TRUE if actual scrolling occurred,
|
|
* FALSE if no change
|
|
*/
|
|
BOOL CDisplaySL::ScrollView (
|
|
LONG upScroll, //@parm New x scroll position
|
|
LONG vpScroll, //@parm New y scroll position
|
|
BOOL fTracking, //@parm TRUE indicates we are tracking scrollbar thumb
|
|
BOOL fFractionalScroll)
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::ScrollView");
|
|
LONG dupMax;
|
|
LONG dx = 0;
|
|
RECTUV rcClient, rcClip;
|
|
CTxtSelection *psel = _ped->GetSelNC();
|
|
COleObject *pipo;
|
|
|
|
AssertSz(_ped->_fInPlaceActive, "CDisplaySL::ScrollView() called when not in-place");
|
|
|
|
if(upScroll == -1)
|
|
return FALSE;
|
|
|
|
//For scrolling purposes, we clip to rcView's top and bottom, but rcClient's left and right
|
|
_ped->TxGetClientRect(&rcClient);
|
|
GetViewRect(rcClip, &rcClient);
|
|
rcClip.left = rcClient.left;
|
|
rcClip.right = rcClient.right;
|
|
|
|
// Determine horizontal scrolling pos.
|
|
dupMax = _dup;
|
|
upScroll = min(upScroll, dupMax);
|
|
upScroll = max(0, upScroll);
|
|
|
|
dx = _upScroll - upScroll;
|
|
if(dx)
|
|
{
|
|
_upScroll = upScroll;
|
|
|
|
// Calculate new first visible
|
|
CMeasurer me(this); // Create measurer at cp(0)
|
|
POINTUV pt = {upScroll, 0};
|
|
CDispDim dispdim;
|
|
HITTEST Hit;
|
|
LONG cpActual;
|
|
CLine::CchFromUp(me, pt, &dispdim, &Hit, &cpActual);
|
|
|
|
SetCpFirstVisible(cpActual); // Save character position
|
|
}
|
|
|
|
AssertSz(IsMain(), "CDisplaySL::ScrollView non-main SL control");
|
|
|
|
// Now perform the actual scrolling
|
|
if(dx)
|
|
{
|
|
if(!_fRectInvalid)
|
|
{
|
|
// Scroll only if scrolling < view dimensions and we are in-place
|
|
// Note that we only scroll the active view and we can be in-place
|
|
// active and have multiple inactive views.
|
|
if(IsActive() && !IsTransparent() && dx < _dupView)
|
|
{
|
|
if(psel)
|
|
psel->ShowCaret(FALSE);
|
|
|
|
RECT rcxyClip;
|
|
RectFromRectuv(rcxyClip, rcClip);
|
|
|
|
_ped->TxScrollWindowEx(dx, 0, NULL, &rcxyClip);
|
|
|
|
if(psel)
|
|
psel->ShowCaret(TRUE);
|
|
}
|
|
else
|
|
_ped->TxInvalidateRect(&rcClip);
|
|
}
|
|
|
|
if(psel)
|
|
psel->UpdateCaret(FALSE);
|
|
|
|
if(!fTracking && dx)
|
|
{
|
|
_ped->SendScrollEvent(EN_HSCROLL);
|
|
UpdateScrollBar(SB_HORZ);
|
|
}
|
|
|
|
_ped->TxUpdateWindow();
|
|
|
|
// FUTURE: since we're now repositioning in place active
|
|
// objects every time we draw, this call seems to be
|
|
// superfluous (AndreiB)
|
|
|
|
// Tell object subsystem to reposition any in place objects
|
|
if(_ped->GetObjectCount())
|
|
{
|
|
pipo = _ped->GetObjectMgr()->GetInPlaceActiveObject();
|
|
if(pipo)
|
|
pipo->OnReposition();
|
|
}
|
|
}
|
|
return dx;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::InvertRange(cp, cch, selAction)
|
|
*
|
|
* @mfunc
|
|
* Invert a given range on screen (for selection)
|
|
*
|
|
* @devnote
|
|
* --- Use when in-place active only ---
|
|
*
|
|
* @rdesc
|
|
* TRUE if success
|
|
*/
|
|
BOOL CDisplaySL::InvertRange (
|
|
LONG cp, //@parm Active end of range to invert
|
|
LONG cch, //@parm Signed length of range
|
|
SELDISPLAYACTION selAction) //@parm What we are doing to the selection
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::InvertRange");
|
|
|
|
if(cch < 0) //make cch negative, make cp the minimum
|
|
cch = -cch;
|
|
else
|
|
cp -= cch;
|
|
|
|
#ifndef NOLINESERVICES
|
|
if (g_pols)
|
|
g_pols->DestroyLine(this);
|
|
#endif
|
|
|
|
// If display is frozen, just update recalc region and move on.
|
|
if(_padc)
|
|
{
|
|
AssertSz(cp >= 0, "CDisplayML::InvertRange: range (cp) goes below"
|
|
"zero!!" );
|
|
// Make sure these values are bounded.
|
|
if(cp > _ped->GetTextLength()) // Don't bother updating region;
|
|
return TRUE; // it's out of bounds
|
|
|
|
if(cp + cch > _ped->GetTextLength())
|
|
cch -= cp + cch - _ped->GetTextLength();
|
|
|
|
_padc->UpdateRecalcRegion(cp, cch, cch);
|
|
return TRUE;
|
|
}
|
|
|
|
// Ensure all visible lines are recalced
|
|
if(!WaitForRecalcView())
|
|
return FALSE;
|
|
|
|
//REVIEW (keithcu) CDisplayML::InvertRange() has a different order WRT
|
|
//frozen displays and OLE objects.
|
|
|
|
// If an object is being inverted, and that is all that
|
|
// is being inverted, delegate to the ObjectMgr.
|
|
if (cch == 1 && _ped->GetObjectCount() &&
|
|
(selAction == selSetNormal || selAction == selSetHiLite) )
|
|
{
|
|
CObjectMgr* pobjmgr = _ped->GetObjectMgr();
|
|
CTxtPtr tp(_ped, cp);
|
|
|
|
if(tp.GetChar() == WCH_EMBEDDING && pobjmgr)
|
|
pobjmgr->HandleSingleSelect(_ped, cp, selAction == selSetHiLite);
|
|
}
|
|
|
|
// Get view rectangle
|
|
AssertSz(_ped->_fInPlaceActive, "CDisplaySL::InvertRange() called when not in-place active");
|
|
|
|
_ped->TxInvalidate();
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::InitLinePtr (&lp)
|
|
*
|
|
* @mfunc
|
|
* Initialize a CLinePtr properly
|
|
*/
|
|
void CDisplaySL::InitLinePtr (
|
|
CLinePtr & lp ) //@parm Ptr to line to initialize
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::InitLinePtr");
|
|
|
|
lp.Init(*this);
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::GetNaturalSize(hdcDraw, hicTarget, dwMode, pwidth, pheight)
|
|
*
|
|
* @mfunc
|
|
* Recalculate display to input width & height
|
|
*
|
|
* @rdesc
|
|
* S_OK - Call completed successfully <nl>
|
|
*/
|
|
HRESULT CDisplaySL::GetNaturalSize(
|
|
HDC hdcDraw, //@parm DC for drawing
|
|
HDC hicTarget, //@parm DC for information
|
|
DWORD dwMode, //@parm Type of natural size required
|
|
LONG *pwidth, //@parm Input/output width parm
|
|
LONG *pheight) //@parm Input/output height parm
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetNaturalSize");
|
|
|
|
// Assume this won't work
|
|
HRESULT hr = E_FAIL;
|
|
|
|
// Set height temporarily so zoom factor will work out
|
|
LONG yOrigHeightClient = SetClientHeight(*pheight);
|
|
|
|
// Adjust height and width by view inset
|
|
LONG widthView = *pwidth;
|
|
LONG heightView = *pheight;
|
|
GetViewDim(widthView, heightView);
|
|
|
|
// Store adjustment so we can restore it to height & width
|
|
LONG widthAdj = *pwidth - widthView;
|
|
LONG heightAdj = *pheight - heightView;
|
|
LONG uCaret = !_ped->fInOurHost() && !_ped->fInHost2() ? duCaret : 0;
|
|
|
|
// Recalculate size needed
|
|
// Create a measurer starting at cp = 0
|
|
CMeasurer me(this);
|
|
CLine liNew;
|
|
|
|
if(liNew.Measure(me, MEASURE_FIRSTINPARA))
|
|
{
|
|
*pwidth = liNew._dup;
|
|
if(dwMode == TXTNS_FITTOCONTENT2)
|
|
*pwidth += liNew._upStart + me.GetRightIndent();
|
|
*pheight = liNew._dvpHeight;
|
|
hr = S_OK;
|
|
}
|
|
|
|
// Restore insets so output reflects true client rect needed
|
|
*pwidth += widthAdj + uCaret;
|
|
*pheight += heightAdj;
|
|
|
|
// Restore client height to match current cache
|
|
SetClientHeight(yOrigHeightClient);
|
|
return hr;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::GetWordWrap()
|
|
*
|
|
* @mfunc
|
|
* Gets the wrap flag
|
|
*
|
|
* @rdesc
|
|
* TRUE - Word wrap
|
|
* FALSE - No word Word wrap
|
|
*
|
|
* @devnote
|
|
* Single line controls cannot word wrap.
|
|
*/
|
|
BOOL CDisplaySL::GetWordWrap() const
|
|
{
|
|
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetNoWrap");
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::Clone()
|
|
*
|
|
* @mfunc
|
|
* Make a copy of this object
|
|
*
|
|
* @rdesc
|
|
* NULL - failed
|
|
* CDisplay *
|
|
*/
|
|
CDisplay *CDisplaySL::Clone() const
|
|
{
|
|
CDisplaySL *pdp = new CDisplaySL(_ped);
|
|
if(pdp)
|
|
{
|
|
// Initialize our base class
|
|
if(pdp->CDisplay::Init())
|
|
{
|
|
pdp->InitFromDisplay(this);
|
|
|
|
// Setting scroll to 0 means use the first visible character
|
|
pdp->_upScroll = CALC_XSCROLL_FROM_FIRST_VISIBLE;
|
|
pdp->_fVScrollEnabled = _fVScrollEnabled;
|
|
pdp->_fWordWrap = _fWordWrap;
|
|
pdp->ResetDrawInfo(this);
|
|
pdp->SetCpFirstVisible(GetFirstVisibleCp());
|
|
|
|
// This can't be active view since it is a clone of some view
|
|
pdp->SetActiveFlag(FALSE);
|
|
}
|
|
}
|
|
return pdp;
|
|
}
|
|
|
|
/*
|
|
* CDisplaySL::GetMaxUScroll()
|
|
*
|
|
* @mfunc
|
|
* Get the maximum u scroll value
|
|
*
|
|
* @rdesc
|
|
* Maximum u scroll value
|
|
*
|
|
*/
|
|
LONG CDisplaySL::GetMaxUScroll() const
|
|
{
|
|
LONG uCaret = !_ped->fInOurHost() && !_ped->fInHost2() ? duCaret : 0;
|
|
return _dup + uCaret;
|
|
}
|