windows-nt/Source/XPSP1/NT/windows/richedit/re30/dispsl.cpp
2020-09-26 16:20:57 +08:00

1068 lines
25 KiB
C++

/*
* @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-1998, 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");
_xScroll = 0;
SetCpFirstVisible(0);
}
/*
* CDisplaySL::RecalcView(fUpdateScrollBars)
*
* @mfunc
* Recalc all lines breaks and update first visible line
*
* @rdesc
* TRUE if success
*/
BOOL CDisplaySL::RecalcView(
BOOL fUpdateScrollBars, RECT* prc) //@param TRUE - update scroll bars
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::RecalcView");
LONG xWidthOld = _xWidth + _xLineOverhang;
if(!RecalcLine())
return FALSE;
if(_fViewChanged)
{
if(IsActive() || _xWidth + _xLineOverhang <= GetViewWidth())
{
_xScroll = 0;
SetCpFirstVisible(0);
}
else if(CALC_XSCROLL_FROM_FIRST_VISIBLE == _xScroll)
{
// In this case we want to set our xScroll by a visible. The
// only way to get here is if the active view has been cloned
// for displaying an inactive view.
_xScroll = 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
_xScroll = CLine::XposFromCch(me, GetFirstVisibleCp(), TA_TOP);
}
}
if(fUpdateScrollBars)
UpdateScrollBar(SB_HORZ, TRUE);
_fViewChanged = FALSE;
}
// We only resize if width of single line control has changed.
if(_xWidth + _xLineOverhang != xWidthOld)
{
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 xWidthOld = CLine::_xWidth + CLine::_xLineOverhang;
BOOL fMeasured = CLine::Measure(me, -1, -1, MEASURE_FIRSTINPARA);
if(!fMeasured)
{
Assert(FALSE); // Should succeed
InitVars();
return FALSE;
}
_fNeedRecalc = FALSE;
_fRecalcDone = TRUE;
if(_fViewChanged || xWidthOld != (CLine::_xWidth + CLine::_xLineOverhang))
_fViewChanged = TRUE;
_fLineRecalcErr = FALSE;
return fMeasured;
}
/*
* CDisplaySL::Render(rcView, rcRender)
*
* @mfunc
* Renders this line
*/
void CDisplaySL::Render(
const RECT &rcView, //@parm View RECT
const RECT &rcRender) //@parm RECT to render (must be contained in
// client rect)
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::Render");
POINT pt;
LONG yHeightBitmap = 0;
_fRectInvalid = FALSE;
CRenderer re(this);
if(!IsMetafile() && !IsTransparent() && (_bFlags & fliUseOffScreenDC))
yHeightBitmap = _yHeight;
if(!re.StartRender(rcView, rcRender, yHeightBitmap))
return;
// Set renderer at top/left of view rect
pt.x = rcView.left - _xScroll;
pt.y = rcView.top;
re.SetCurPoint(pt);
// Renderer is set at cp = 0 at the moment
CLine::Render(re);
if(_bFlags & fliOffScreenOnce)
_bFlags &= ~(fliUseOffScreenDC | fliOffScreenOnce);
// If line metrics are not yet up to date, get them from renderer
if(_xWidth == -1)
{
_xWidth = re._li._xWidth;
_xLineOverhang = re._li._xLineOverhang;
_yHeight = re._li._yHeight;
_yDescent = re._li._yDescent;
}
re.EndRender();
}
/*
* 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 && _fHScrollEnabled)
{
if(_ped->TxGetScrollBars() & WS_HSCROLL)
{
lRange = max(0, _xWidth + dxCaret);
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(!_fHScrollEnabled)
_fHScrollEnabled = !!(dwScrollBars & ES_AUTOHSCROLL);
return FALSE;
}
fEnabledOld = _fHScrollEnabled;
lScroll = ConvertXPosToScrollPos(_xScroll);
if(_xWidth <= _xWidthView)
fEnabled = FALSE;
}
// !s beforehand because all true values aren't necessarily equal
if(!fEnabled != !fEnabledOld)
{
if(_fDeferUpdateScrollBar)
_fUpdateScrollBarDeferred = TRUE;
else
{
if (nBar == SB_HORZ)
_fHScrollEnabled = 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::GetMaxWidth() const
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetMaxWidth");
return 0;
}
LONG CDisplaySL::GetMaxPixelWidth() const
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetMaxPixelWidth");
return GetViewWidth();
}
LONG CDisplaySL::GetMaxHeight() const
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetMaxHeight");
return 0;
}
LONG CDisplaySL::GetWidth() const
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetWidth");
return CLine::_xWidth + CLine::_xLineOverhang;
}
LONG CDisplaySL::GetHeight() const
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetHeight");
return CLine::_yHeight;
}
LONG CDisplaySL::GetResizeHeight() const
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::GetResizeHeight");
return CLine::_yHeight;
}
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, pyHeight)
*
* @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 *pyLine) //@parm Returns top of line relative to display
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::CpFromLine");
// (NULL if don't want that info)
Assert( ili == 0 );
if(pyLine)
*pyLine = 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(
POINT pt, //@parm Point to compute cp at (client coords)
const RECT *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
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::CpFromPoint");
RECT rcView;
LONG cp = 0;
LONG cch = 0;
HITTEST hit = HT_Nothing;
GetViewRect(rcView, prcClient);
if (pt.y >= rcView.top && pt.y <= rcView.bottom)
{
// Create measurer at cp(0)
CMeasurer me(this);
// Get character in the line
pt.x += _xScroll - rcView.left; // Transform to galley coordinates
pt.y -= rcView.top;
cch = CLine::CchFromXpos(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.BackupCpCRLF();
cp = me.GetCp();
}
if (phit)
*phit = hit;
if(ptp)
ptp->SetCp(cp);
if(prp)
prp->RpSet(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 RECT *prcClient, //@parm Client rectangle (can be NULL if active).
BOOL fAtEnd, //@parm Return end of previous line for ambiguous cp
POINT & 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;
RECT rcView;
CLinePtr rp(this);
if(!rp.RpSetCp(rtp.GetCp(), fAtEnd))
return -1;
AssertSz(_ped->_fInPlaceActive || prcClient,
"CDisplaySL::PointFromTp() called with invalid client rect");
GetViewRect(rcView, prcClient);
pt.x = rcView.left - _xScroll;
pt.y = rcView.top;
CMeasurer me(this, rtp);
me.Advance(-rp.RpGetIch()); // Backup to start of line
me.NewLine(*rp); // Measure from there to where we are
pt.x += CLine::XposFromCch(me, rp.RpGetIch(), taMode, pdispdim, &dy);
pt.y += 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(
const 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;
RECT rcView;
CTxtSelection *psel = _ped->GetSelNC();
LONG xWidthOld = _xWidth + _xLineOverhang;
BOOL fScrollChanged = FALSE;
RECT rcClient;
RECT rc;
LONG yHeightOld = _yHeight;
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 trouble now, the recalc failed. Let's try to get
// out of this with our head still mostly attached
fReturn = FALSE;
}
// An update has occurred. If it isn't already off screen make it
// off screen so that there is no flicker.
if(!(_bFlags & fliUseOffScreenDC))
_bFlags |= (fliUseOffScreenDC | fliOffScreenOnce);
if(_xWidth <= _xWidthView)
{
// x scroll range is smaller than the view width
// force x scrolling position = 0
_xScroll = 0;
SetCpFirstVisible(0);
_fViewChanged = TRUE;
fUpdateScrollBarHorz = TRUE;
}
_fRectInvalid = TRUE;
// Only resize a Single Line edit control if the width changes
if(_xWidth + _xLineOverhang != xWidthOld)
{
if(FAILED(RequestResize()))
_ped->GetCallMgr()->SetOutOfMemory();
}
// If view changed, update scroll bars
if(_fViewChanged)
{
_fViewChanged = FALSE;
fScrollChanged = UpdateScrollBar(SB_HORZ);
}
if(!fScrollChanged)
{
// Scroll bar state did not change so we need to update screen
// Build an invalidation rectangle.
rc = rcClient;
if(yHeightOld == _yHeight)
{
// Height of control did not change so we can minimize the update
// rectangle to the height of the control.
rc.bottom = rcView.top + _yHeight;
}
// Tell display to update when it gets a chance
_ped->TxInvalidateRect(&rc, FALSE);
}
return fReturn;
}
/*
* CDisplaySL::ScrollView(xScroll, yScroll, 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 yScroll <gt> pdp->yHeight and yScroll <lt> 0
*
* @rdesc
* TRUE if actual scrolling occurred,
* FALSE if no change
*/
BOOL CDisplaySL::ScrollView (
LONG xScroll, //@parm New x scroll position
LONG yScroll, //@parm New y scroll position
BOOL fTracking, //@parm TRUE indicates we are tracking scrollbar thumb
BOOL fFractionalScroll)
{
TRACEBEGIN(TRCSUBSYSDISP, TRCSCOPEINTERN, "CDisplaySL::ScrollView");
// (don't update the scrollbar pos)
RECT rcUpdate; // ??? we may want use a region here but ScrollView is
// rarely called with both a xScroll and yScroll value.
LONG xWidthMax;
LONG dx = 0;
RECT rcView;
CTxtSelection *psel = _ped->GetSelNC();
COleObject *pipo;
AssertSz(_ped->_fInPlaceActive, "CDisplaySL::ScrollView() called when not in-place");
if(xScroll == -1)
return FALSE;
GetViewRect(rcView);
// Determine horizontal scrolling pos.
xWidthMax = _xWidth;
xScroll = min(xScroll, xWidthMax);
xScroll = max(0, xScroll);
dx = _xScroll - xScroll;
if(dx)
{
_xScroll = xScroll;
// Calculate new first visible
CMeasurer me(this); // Create measurer at cp(0)
POINT pt = {xScroll, 0};
CDispDim dispdim;
HITTEST Hit;
LONG cpActual;
CLine::CchFromXpos(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 < _xWidthView)
{
if(psel)
psel->ShowCaret(FALSE);
_ped->TxScrollWindowEx((INT) dx, 0, NULL, &rcView,
NULL, &rcUpdate, 0);
_ped->TxInvalidateRect(&rcUpdate, FALSE);
if(psel)
psel->ShowCaret(TRUE);
}
else
_ped->TxInvalidateRect(&rcView, FALSE);
}
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(dx, 0);
}
}
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");
RECT rcView;
RECT rc;
if(cch < 0) //make cch negative, make cp the minimum
cch = -cch;
else
cp -= cch;
#ifdef LINESERVICES
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->TxGetClientRect(&rc);
GetViewRect(rcView, &rc);
_ped->TxInvalidateRect(NULL, FALSE);
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;
// Recalculate size needed
// Create a measurer starting at cp = 0
CMeasurer me(this);
CLine liNew;
if(liNew.Measure(me, -1, -1, MEASURE_FIRSTINPARA))
{
*pwidth = liNew._xWidth + liNew._xLineOverhang;
*pheight = liNew._yHeight;
hr = S_OK;
}
// Restore insets so output reflects true client rect needed
*pwidth += widthAdj + dxCaret;
*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->_xScroll = 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::GetMaxXScroll()
*
* @mfunc
* Get the maximum x scroll value
*
* @rdesc
* Maximum x scroll value
*
*/
LONG CDisplaySL::GetMaxXScroll() const
{
return _xWidth + dxCaret;
}