/* Copyright (c) 2017, Computer History Museum All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted (subject to the limitations in the disclaimer below) provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of Computer History Museum nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY THIS LICENSE. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "petepch.h" void ResetScrollbars(DocumentInfoHandle docInfo) { long winHeight, height, width, winWidth; #ifdef SHOWHIDESCROLLS Rect controlRect; PETEPortInfo savedPortInfo; #endif if(!(**docInfo).flags.docHeightValid) { RecalcDocHeight(docInfo); if(!(**docInfo).flags.docHeightValid) { (**docInfo).flags.scrollsDirty = true; return; } } if((**docInfo).flags.reposition) { RepositionDocument(docInfo); } if((**docInfo).flags.hasHScroll) { winWidth = RectWidth(&(**docInfo).viewRect); width = GetScrollingWidth(docInfo) - winWidth; if(width < 0L) { width = 0L; } DI_MEMCANTFAIL(docInfo); #if TARGET_CPU_PPC if((**(**docInfo).globals).flags.hasControlManager) { SetControl32BitMinimum((**docInfo).hScroll, 0L); SetControl32BitMaximum((**docInfo).hScroll, width < (**docInfo).hPosition ? (**docInfo).hPosition : width); SetControlViewSize((**docInfo).hScroll, winWidth); } else #endif { SetControlMaximum((**docInfo).hScroll, width < (**docInfo).hPosition ? (**docInfo).hPosition : width); } DI_MEMCANFAIL(docInfo); SetScrollPosition(docInfo, true); #ifdef SHOWHIDESCROLLS DI_MEMCANTFAIL(docInfo); GetControlRect((**docInfo).hScroll, &controlRect); if((controlRect.right - controlRect.left < kTooSmallForScroll) || !(**docInfo).flags.scrollsVisible) { HideControl((**docInfo).hScroll); SavePortInfo(GetWindowPort((**docInfo).docWindow), &savedPortInfo); EraseRect(&controlRect); FrameRect(&controlRect); ResetPortInfo(&savedPortInfo); } else { ShowControl((**docInfo).hScroll); } DI_MEMCANFAIL(docInfo); #endif } if((**docInfo).flags.hasVScroll) { winHeight = RectHeight(&(**docInfo).viewRect); height = (**docInfo).docHeight - winHeight; if(height < 0L) { height = 0L; } DI_MEMCANTFAIL(docInfo); #if TARGET_CPU_PPC if((**(**docInfo).globals).flags.hasControlManager) { SetControl32BitMinimum((**docInfo).vScroll, 0L); SetControl32BitMaximum((**docInfo).vScroll, height < (**docInfo).vPosition ? (**docInfo).vPosition : height); SetControlViewSize((**docInfo).vScroll, winHeight); } else #endif { if(height < (**docInfo).vPosition) { height = (**docInfo).vPosition; } SetControlMaximum((**docInfo).vScroll, height > winHeight ? winHeight : height); } DI_MEMCANFAIL(docInfo); SetScrollPosition(docInfo, false); #ifdef SHOWHIDESCROLLS DI_MEMCANTFAIL(docInfo); GetControlRect((**docInfo).vScroll, &controlRect); if((controlRect.bottom - controlRect.top < kTooSmallForScroll) || !(**docInfo).flags.scrollsVisible) { HideControl((**docInfo).vScroll); SavePortInfo(GetWindowPort((**docInfo).docWindow), &savedPortInfo); EraseRect(&controlRect); FrameRect(&controlRect); ResetPortInfo(&savedPortInfo); } else { ShowControl((**docInfo).vScroll); } DI_MEMCANFAIL(docInfo); #endif } (**docInfo).flags.scrollsDirty = false; } void ScrollDoc(DocumentInfoHandle docInfo, short dh, short dv) { PETEPortInfo savedPortInfo; Rect viewRect; RgnHandle clipRgn; if((dh == 0) && (dv == 0)) { return; } if(dh != 0) { (**docInfo).hPosition += dh; SetScrollPosition(docInfo, true); } if(dv != 0) { (**docInfo).vPosition += dv; SetScrollPosition(docInfo, false); } clipRgn = NewRgn(); if(clipRgn != nil) { SavePortInfo(GetWindowPort((**docInfo).docWindow), &savedPortInfo); viewRect = (**docInfo).viewRect; ScrollRectInDoc(docInfo, &viewRect, -dh, -dv, clipRgn); OffsetRgn((**docInfo).hiliteRgn, -dh, -dv); if(CalcSelectionRgn(docInfo, true) == noErr) { Boolean progressLoopCalled = (**docInfo).flags.progressLoopCalled; (**docInfo).flags.progressLoopCalled = true; DrawDocument(docInfo, clipRgn, true); (**docInfo).flags.progressLoopCalled = progressLoopCalled; } DisposeRgn(clipRgn); ResetPortInfo(&savedPortInfo); } } void SetDocPosition(DocumentInfoHandle docInfo, long vPosition, short hPosition) { long vOffset; short hOffset; PETEPortInfo savedPortInfo; hOffset = hPosition - (**docInfo).hPosition; vOffset = vPosition - (**docInfo).vPosition; if((hOffset == 0L) && (vOffset == 0L)) { return; } if((vOffset > SHRT_MAX) || (vOffset < SHRT_MIN)) { (**docInfo).vPosition += vOffset; (**docInfo).hPosition += hOffset; SavePortInfo(GetWindowPort((**docInfo).docWindow), &savedPortInfo); if(CalcSelectionRgn(docInfo, true) == noErr) { DrawDocument(docInfo, nil, true); } ResetPortInfo(&savedPortInfo); } else { ScrollDoc(docInfo, hOffset, vOffset); } } Boolean TrackScrolls(DocumentInfoHandle docInfo, ControlRef theControl, Point where, short partCode) { Boolean noScrollProc = ((partCode == kControlIndicatorPart) && (!(**(**docInfo).globals).flags.hasAppearanceManager || !(**(**docInfo).globals).flags.useLiveScrolling)); long saved; short rect; Boolean vertical; unsigned long lastClickTime; DECLARE_UPP(ScrollActionProc,ControlAction); INIT_UPP(ScrollActionProc,ControlAction); if(!(vertical = (theControl == (**docInfo).vScroll))) { if(theControl != (**docInfo).hScroll) { return false; } } if(vertical) { RecalcDocHeight(docInfo); saved = (**docInfo).docHeight; rect = RectHeight(&(**docInfo).viewRect); if((**docInfo).vPosition > (**docInfo).docHeight - rect) { (**docInfo).docHeight = (**docInfo).vPosition + rect; } } else { saved = (**docInfo).docWidth; rect = RectWidth(&(**docInfo).viewRect); (**docInfo).docWidth = GetScrollingWidth(docInfo); if((**docInfo).hPosition > (**docInfo).docWidth - rect) { (**docInfo).docWidth = (**docInfo).hPosition + rect; } } /* If the caret was on, turn it off first */ EraseCaret(docInfo); lastClickTime = (**docInfo).lastClickTime; (**docInfo).lastClickTime = 0L; DI_MEMCANTFAIL(docInfo); partCode = TrackControl(theControl, where, noScrollProc ? nil : ScrollActionProcUPP); DI_MEMCANFAIL(docInfo); (**docInfo).lastClickTime = lastClickTime; if(noScrollProc) { SetPositionFromScroll(docInfo, vertical, !vertical); } if(vertical) { (**docInfo).docHeight = saved; } else { (**docInfo).docWidth = saved; } ResetScrollbars(docInfo); return true; } Fixed GetScrollRatio(ControlRef theControl); Fixed GetScrollRatio(ControlRef theControl) { short max, value; if((max = GetControlMaximum(theControl)) == 0) { return -1L; } else if((value = GetControlValue(theControl)) == 0) { return 0L; } else { return FixRatio(value, max); } } /* Set the doc position according to the scroll values */ void SetPositionFromScroll(DocumentInfoHandle docInfo, Boolean vScroll, Boolean hScroll) { long vPosition; short hPosition; Fixed hRatio, vRatio; DI_MEMCANTFAIL(docInfo); #if TARGET_CPU_PPC if((**(**docInfo).globals).flags.hasControlManager) { vPosition = vScroll ? GetControl32BitValue((**docInfo).vScroll) : (**docInfo).vPosition; hPosition = hScroll ? GetControl32BitValue((**docInfo).hScroll) : (**docInfo).hPosition; } else #endif { vRatio = vScroll ? GetScrollRatio((**docInfo).vScroll) : -1; hRatio = hScroll ? GetScrollRatio((**docInfo).hScroll) : -1; if(vRatio >= 0) { vPosition = FixMul((**docInfo).docHeight - RectHeight(&(**docInfo).viewRect), vRatio); } else { vPosition = (**docInfo).vPosition; } if(hRatio >= 0) { hPosition = FixMul((**docInfo).docWidth - RectWidth(&(**docInfo).viewRect), hRatio); } else { hPosition = (**docInfo).hPosition; } } DI_MEMCANFAIL(docInfo); SetDocPosition(docInfo, vPosition, hPosition); } /* Set the scroll value according to the doc position */ void SetScrollPosition(DocumentInfoHandle docInfo, Boolean horizontal) { long position, total, screen; Fract scrollFraction; ControlHandle theControl; short max, value; if(horizontal) { if((**docInfo).flags.hasHScroll) { theControl = (**docInfo).hScroll; position = (**docInfo).hPosition; total = GetScrollingWidth(docInfo); screen = RectWidth(&(**docInfo).viewRect); } else { return; } } else { if((**docInfo).flags.hasVScroll) { theControl = (**docInfo).vScroll; position = (**docInfo).vPosition; total = (**docInfo).docHeight; screen = RectHeight(&(**docInfo).viewRect); } else { return; } } DI_MEMCANTFAIL(docInfo); #if TARGET_CPU_PPC if((**(**docInfo).globals).flags.hasControlManager) { if(position != GetControl32BitValue(theControl)) { SetControl32BitValue(theControl, position); } } else #endif { if(total - position < screen) { scrollFraction = fract1; } else if(position <= 0L) { scrollFraction = 0L; } else { scrollFraction = FracDiv(position, total - screen); } max = GetControlMaximum(theControl); value = FixRound(FracMul(Long2Fix(max), scrollFraction)); if(value != GetControlValue(theControl)) { SetControlValue(theControl, value); } } DI_MEMCANFAIL(docInfo); } void GetVScrollbarLocation(Rect *docRect, Rect *controlRect, Boolean hasHScroll, Boolean hasGrowBox) { /* Top is 1 pixel underneath the title bar */ controlRect->top = docRect->top - 1; /* Left is the scrollbar width from the right, plus 1 pixel overlap */ controlRect->left = docRect->right - kScrollbarAdjust; /* Bottom is 1 pixel past the bottom of the window */ controlRect->bottom = docRect->bottom + 1; /* If there is a horizontal scrollbar or grow box, subtract from the bottom */ if(hasHScroll || hasGrowBox) { controlRect->bottom -= kScrollbarAdjust; } /* Right is 1 pixel past the right side of the window */ controlRect->right = docRect->right + 1; } void GetHScrollbarLocation(Rect *docRect, Rect *controlRect, Boolean hasVScroll, Boolean hasGrowBox) { /* Top is the scrollbar width from the bottom, plus 1 pixel overlap */ controlRect->top = docRect->bottom - kScrollbarAdjust; /* Left is 1 pixel before the left side of the window */ controlRect->left = docRect->left - 1; /* Bottom is 1 pixel past the bottom of the window */ controlRect->bottom = docRect->bottom + 1; /* Right is 1 pixel past the right side of the window */ controlRect->right = docRect->right + 1; /* If there is a vertical scrollbar or grow box, subtract from the right */ if(hasVScroll || hasGrowBox) { controlRect->right -= kScrollbarAdjust; } } /* Return whether docWidth or the longest line determines the scrollbar width */ short GetScrollingWidth(DocumentInfoHandle docInfo) { if((**docInfo).flags.useHLineWidth) { if((**docInfo).textLen == 0L) { (**docInfo).longLineWidth = 0L; } return (**docInfo).longLineWidth; } else { return (**docInfo).docWidth; } } /* Move the document if there is extra space */ void RepositionDocument(DocumentInfoHandle docInfo) { short rightSpace; long bottomSpace; Rect viewRect; if(!(**docInfo).flags.docHeightValid) { RecalcDocHeight(docInfo); } if((**docInfo).flags.recalcOff || !(**docInfo).flags.docHeightValid) { (**docInfo).flags.reposition = true; return; } (**docInfo).flags.reposition = false; viewRect = (**docInfo).viewRect; bottomSpace = (**docInfo).docHeight - (**docInfo).vPosition - RectHeight(&viewRect); if((bottomSpace >= 0L) || ((**docInfo).vPosition <= 0)) { bottomSpace = 0L; } else if(-bottomSpace > (**docInfo).vPosition) { bottomSpace = -(**docInfo).vPosition; } rightSpace = GetScrollingWidth(docInfo) - (**docInfo).hPosition - RectWidth(&viewRect); if((rightSpace >= 0L) || ((**docInfo).hPosition <= 0)) { rightSpace = 0L; } else if(-rightSpace > (**docInfo).hPosition) { rightSpace = -(**docInfo).hPosition; } if((rightSpace != 0L) || (bottomSpace != 0)) { PETEPortInfo savedPortInfo; (**docInfo).hPosition += rightSpace; (**docInfo).vPosition += bottomSpace; SavePortInfo(GetWindowPort((**docInfo).docWindow), &savedPortInfo); InvalWindowRect((**docInfo).docWindow,&viewRect); ResetPortInfo(&savedPortInfo); (**docInfo).flags.scrollsDirty = true; } } /* Change the size of the viewing and scrollbar rectangle; no margin change */ void ResizeDocRect(DocumentInfoHandle docInfo, short h, short v) { PETEPortInfo savedPortInfo; Rect docRect, controlRect; RgnHandle oldRgn, scratchRgn; short dh, dv; dh = h - RectWidth(&(**docInfo).docRect); dv = v - RectHeight(&(**docInfo).docRect); if((dh == 0) && (dv == 0)) { return; } scratchRgn = (**(**docInfo).globals).scratchRgn; oldRgn = NewRgn(); if(oldRgn == nil) { return; } EraseCaret(docInfo); SavePortInfo(GetWindowPort((**docInfo).docWindow), &savedPortInfo); SetRectRgn(oldRgn, (**docInfo).viewRect.left, (**docInfo).viewRect.top, (**docInfo).viewRect.right, (**docInfo).viewRect.bottom); (**docInfo).viewRect.right += dh; (**docInfo).viewRect.bottom += dv; (**docInfo).docRect.right += dh; (**docInfo).docRect.bottom += dv; SetRectRgn(scratchRgn, (**docInfo).viewRect.left, (**docInfo).viewRect.top, (**docInfo).viewRect.right, (**docInfo).viewRect.bottom); DiffRgn(scratchRgn, oldRgn, oldRgn); InvalWindowRgn((**docInfo).docWindow,oldRgn); SetRectRgn(scratchRgn, (**docInfo).docRect.left, (**docInfo).docRect.top, (**docInfo).docRect.right, (**docInfo).docRect.bottom); DiffRgn(scratchRgn, oldRgn, oldRgn); InvalWindowRgn((**docInfo).docWindow,oldRgn); DisposeRgn(oldRgn); docRect = (**docInfo).docRect; DI_MEMCANTFAIL(docInfo); if((**docInfo).flags.hasVScroll) { GetVScrollbarLocation(&docRect, &controlRect, (**docInfo).flags.hasHScroll, (**docInfo).flags.hasGrowBox); #ifdef SHOWHIDESCROLLS if((controlRect.bottom - controlRect.top >= kTooSmallForScroll) && (**docInfo).flags.scrollsVisible) { ShowControl((**docInfo).vScroll); } else { HideControl((**docInfo).vScroll); } #endif SizeControl((**docInfo).vScroll, controlRect.right - controlRect.left, controlRect.bottom - controlRect.top); MoveControl((**docInfo).vScroll, controlRect.left, controlRect.top); } if((**docInfo).flags.hasHScroll) { GetHScrollbarLocation(&docRect, &controlRect, (**docInfo).flags.hasVScroll, (**docInfo).flags.hasGrowBox); #ifdef SHOWHIDESCROLLS if((controlRect.right - controlRect.left >= kTooSmallForScroll) && (**docInfo).flags.scrollsVisible) { ShowControl((**docInfo).hScroll); } else { HideControl((**docInfo).hScroll); } #endif SizeControl((**docInfo).hScroll, controlRect.right - controlRect.left, controlRect.bottom - controlRect.top); MoveControl((**docInfo).hScroll, controlRect.left, controlRect.top); } DI_MEMCANFAIL(docInfo); CalcSelectionRgn(docInfo, true); ResetScrollbars(docInfo); ResetPortInfo(&savedPortInfo); } /* Move the document rectangle to another position in the window */ void MoveDocRect(DocumentInfoHandle docInfo, short h, short v) { PETEPortInfo savedPortInfo; Rect tempRect, controlRect; short dh, dv; tempRect = (**docInfo).docRect; dh = h - tempRect.left; dv = v - tempRect.top; if((dh == 0) && (dv == 0)) { return; } EraseCaret(docInfo); SavePortInfo(GetWindowPort((**docInfo).docWindow), &savedPortInfo); InvalWindowRect((**docInfo).docWindow,&tempRect); OffsetRect(&tempRect, dh, dv); InvalWindowRect((**docInfo).docWindow,&tempRect); (**docInfo).docRect = tempRect; tempRect = (**docInfo).viewRect; OffsetRect(&tempRect, dh, dv); (**docInfo).viewRect = tempRect; DI_MEMCANTFAIL(docInfo); if((**docInfo).flags.hasVScroll) { GetVScrollbarLocation(&(**docInfo).docRect, &controlRect, (**docInfo).flags.hasHScroll, (**docInfo).flags.hasGrowBox); MoveControl((**docInfo).vScroll, controlRect.left, controlRect.top); } if((**docInfo).flags.hasHScroll) { GetHScrollbarLocation(&(**docInfo).docRect, &controlRect, (**docInfo).flags.hasVScroll, (**docInfo).flags.hasGrowBox); MoveControl((**docInfo).hScroll, controlRect.left, controlRect.top); } DI_MEMCANFAIL(docInfo); CalcSelectionRgn(docInfo, true); ResetPortInfo(&savedPortInfo); } void ChangeDocWidth(DocumentInfoHandle docInfo, short docWidth, Boolean pinMargins) { short widthDiff; long paraIndex; ParagraphInfoPtr paraInfo; if(docWidth < 0L) { docWidth = RectWidth(&(**docInfo).viewRect); } widthDiff = docWidth - (**docInfo).docWidth; if(widthDiff != 0L) { (**docInfo).docWidth = docWidth; FlushDocInfoLineCache(docInfo); if(pinMargins) { for(paraIndex = 0L; paraIndex < (**docInfo).paraCount; ++paraIndex) { paraInfo = *((**docInfo).paraArray[paraIndex].paraInfo); if(paraInfo->direction == leftCaret) { if(paraInfo->endMargin > 0L) { paraInfo->endMargin += widthDiff; } } else { paraInfo->startMargin -= widthDiff; } SetParaDirty(docInfo, paraIndex); } RecalcDocHeight(docInfo); } SetEmptyRgn((**docInfo).hiliteRgn); InvalidateDocument(docInfo, true); (**docInfo).flags.selectionDirty = (**docInfo).flags.scrollsDirty = true; ResetScrollbars(docInfo); } } pascal void ScrollActionProc(ControlRef theControl, short partCode) { DocumentInfoHandle docInfo; Boolean horizontal; short amount; long tickDiff; unsigned long tickCount; docInfo = (DocumentInfoHandle)GetControlReference(theControl); horizontal = (theControl == (**docInfo).hScroll); switch(partCode) { case kControlIndicatorPart : SetPositionFromScroll(docInfo, !horizontal, horizontal); return; case kControlPageUpPart : amount = psePageUp; break; case kControlPageDownPart : amount = psePageDn; break; case kControlUpButtonPart : amount = pseLineUp; break; case kControlDownButtonPart : amount = pseLineDn; break; default : amount = pseNoScroll; } tickCount = TickCount(); tickDiff = (**(**docInfo).globals).autoScrollTicks - (tickCount - (**docInfo).lastClickTime); if(tickDiff > 0L) { Delay(tickDiff, &tickCount); } (**docInfo).lastClickTime = tickCount; DoScroll(docInfo, horizontal ? amount : pseNoScroll, horizontal ? pseNoScroll : amount); } void GetSelectionVisibleDistance(DocumentInfoHandle docInfo, SelDataPtr selection, Rect *viewRect, long *vDistance, short *hDistance) { long vPosition; short hPosition; Rect tempViewRect; if(viewRect == nil) { tempViewRect = (**docInfo).viewRect; viewRect = &tempViewRect; } vPosition = (**docInfo).vPosition + (viewRect->top - (**docInfo).viewRect.top); if(selection->vLineHeight > RectHeight(viewRect)) { vPosition = (selection->vPosition + (selection->lastLine ? selection->vLineHeight - RectHeight(viewRect) : 0)) - vPosition; // vPosition = selection->vPosition + (RectHeight(viewRect) / 2) - (selection->vLineHeight / 2) - vPosition; } else { if(selection->vPosition < vPosition) { vPosition = selection->vPosition - vPosition; } else { vPosition += (RectHeight(viewRect) - selection->vLineHeight); if(selection->vPosition > vPosition) { vPosition = selection->vPosition - vPosition; } else { vPosition = 0L; } } } if((viewRect->top != (**docInfo).viewRect.top) || (viewRect->bottom != (**docInfo).viewRect.bottom)) { if(vPosition < 0L) { if((**docInfo).vPosition + vPosition < 0L) { if((**docInfo).vPosition >= 0L) { vPosition = -(**docInfo).vPosition; } else { vPosition = 0L; } } } else { if((**docInfo).vPosition + vPosition + RectHeight(&(**docInfo).viewRect) > (**docInfo).docHeight) { if((**docInfo).vPosition + RectHeight(&(**docInfo).viewRect) < (**docInfo).docHeight) { vPosition = (**docInfo).docHeight - (**docInfo).vPosition + RectHeight(&(**docInfo).viewRect); } else { vPosition = 0L; } } } } hPosition = (**docInfo).hPosition + (viewRect->left - (**docInfo).viewRect.left); if(selection->lPosition < (**docInfo).hPosition) { hPosition = selection->lPosition - hPosition; } else { hPosition += RectWidth(viewRect); if(selection->rPosition > hPosition) { hPosition = selection->rPosition - hPosition; } else { hPosition = 0L; } } if((viewRect->left != (**docInfo).viewRect.left) || (viewRect->right != (**docInfo).viewRect.right)) { if(hPosition < 0L) { if((**docInfo).hPosition + hPosition < 0L) { if((**docInfo).hPosition > 0L) { hPosition = -(**docInfo).hPosition; } else { hPosition = 0L; } } } else { if((**docInfo).hPosition + hPosition + RectWidth(&(**docInfo).viewRect) > GetScrollingWidth(docInfo)) { if((**docInfo).hPosition + RectWidth(&(**docInfo).viewRect) < GetScrollingWidth(docInfo)) { hPosition = GetScrollingWidth(docInfo) - (**docInfo).hPosition + RectWidth(&(**docInfo).viewRect); } else { hPosition = 0L; } } } } *vDistance = vPosition; *hDistance = hPosition; } void MakeSelectionVisible(DocumentInfoHandle docInfo, SelDataPtr selection) { long vPosition; short hPosition; Rect viewRect; PETEPortInfo savedPortInfo; GetSelectionVisibleDistance(docInfo, selection, nil, &vPosition, &hPosition); if((vPosition > SHRT_MAX) || (vPosition < SHRT_MIN)) { (**docInfo).vPosition += vPosition; (**docInfo).hPosition += hPosition; SavePortInfo(GetWindowPort((**docInfo).docWindow), &savedPortInfo); CalcSelectionRgn(docInfo, true); viewRect = (**docInfo).viewRect; InvalWindowRect((**docInfo).docWindow,&viewRect); ResetPortInfo(&savedPortInfo); } else { ScrollDoc(docInfo, hPosition, vPosition); } if((vPosition != 0L) || (hPosition != 0L)) { (**docInfo).flags.reposition = true; ResetScrollbars(docInfo); } } Boolean ShouldCenterSelection(DocumentInfoHandle docInfo, Boolean force) { if(!force && ((**docInfo).selStart.vPosition >= (**docInfo).vPosition) && ((**docInfo).selEnd.vPosition + (**docInfo).selEnd.vLineHeight <= (**docInfo).vPosition + RectHeight(&(**docInfo).viewRect))) { return false; } if((**docInfo).selStart.vPosition >= (**docInfo).vPosition + RectHeight(&(**docInfo).viewRect)) { return true; } if((**docInfo).selEnd.vPosition + (**docInfo).selEnd.vLineHeight <= (**docInfo).vPosition) { return true; } return(((**docInfo).selEnd.vPosition + (**docInfo).selEnd.vLineHeight - (**docInfo).selStart.vPosition) < RectHeight(&(**docInfo).viewRect)); } OSErr DoScroll(DocumentInfoHandle docInfo, short horizontal, short vertical) { SelData selection; long vPosition, vMax; short hPosition, distance, hMax; OSErr vErrCode, hErrCode; Boolean doHorizontal, doVertical; vErrCode = hErrCode = noErr; if((horizontal == pseSelection) || (vertical == pseSelection)) { selection = (**docInfo).selStart; MakeSelectionVisible(docInfo, &selection); } vPosition = (**docInfo).vPosition; hPosition = (**docInfo).hPosition; vMax = (**docInfo).docHeight - RectHeight(&(**docInfo).viewRect); if(vMax < vPosition) { vMax = vPosition; } hMax = GetScrollingWidth(docInfo) - RectWidth(&(**docInfo).viewRect); if(hMax < (**docInfo).hPosition) { hMax = (**docInfo).hPosition; } switch(vertical) { case psePageUp : case psePageDn : if(vMax > vPosition) { vMax = GetDocHeight(docInfo) - (**docInfo).shortLineHeight; distance = vMax; } distance = RectHeight(&(**docInfo).viewRect) - (**docInfo).shortLineHeight; goto AddVerticalDistance; case pseLineUp : case pseLineDn : distance = (**docInfo).shortLineHeight; AddVerticalDistance: if((vertical == pseLineUp) || (vertical == psePageUp)) { distance = -distance; } vPosition += distance; case pseSelection : case pseNoScroll : break; case pseCenterSelection : case pseForceCenterSelection : if(ShouldCenterSelection(docInfo, (vertical == pseForceCenterSelection))) { vPosition = (**docInfo).selStart.vPosition; distance = (RectHeight(&(**docInfo).viewRect) / 2) - ((**docInfo).selStart.vLineHeight / 2); vPosition -= distance; } break; default : if(vertical >= 0) { vPosition = FixMul(GetDocHeight(docInfo) - RectHeight(&(**docInfo).viewRect), FixRatio(vertical, SHRT_MAX)); } else { vErrCode = paramErr; } } if(vErrCode == noErr) { if(vPosition < 0L) { if((**docInfo).vPosition > 0) { vPosition = 0L; } else { vPosition = (**docInfo).vPosition; vErrCode = errTopOfDocument; } } else if(vMax <= vPosition) { vPosition = vMax; if(vMax <= (**docInfo).vPosition) { vErrCode = errEndOfDocument; } } } switch(horizontal) { case psePageUp : case psePageDn : distance = RectWidth(&(**docInfo).viewRect); goto AddHorizontalDistance; case pseLineUp : case pseLineDn : distance = 1; AddHorizontalDistance : if((horizontal == pseLineUp) || (horizontal == psePageUp)) { distance = -distance; } hPosition += distance; case pseSelection : case pseNoScroll : case pseCenterSelection : break; default : if(horizontal >= 0) { hPosition = FixMul(GetScrollingWidth(docInfo) - RectWidth(&(**docInfo).viewRect), FixRatio(horizontal, SHRT_MAX)); } else { hErrCode = paramErr; } } if(hErrCode == noErr) { if(hPosition < 0L) { if((**docInfo).hPosition > 0L) { hPosition = 0L; } else { hPosition = (**docInfo).hPosition; hErrCode = errTopOfDocument; } } else if(hPosition > hMax) { if(hMax > (**docInfo).hPosition) { hPosition = hMax; } else { hErrCode = errEndOfDocument; } } } doHorizontal = ((hErrCode == noErr) && (hPosition != (**docInfo).hPosition)); doVertical = ((vErrCode == noErr) && ((**docInfo).vPosition != vPosition)); if(doHorizontal || doVertical) { SetDocPosition(docInfo, vPosition, hPosition); ResetScrollbars(docInfo); } if(vErrCode != noErr) { return vErrCode; } else { return hErrCode; } } void ScrollRectInDoc(DocumentInfoHandle docInfo, Rect *scrollRect, short dh, short dv, RgnHandle scrollRgn) { RgnHandle scratchRgn, graphicRgn; RGBColor oldColor, newColor; Boolean colorChange = false; newColor = DocOrGlobalBGColor(nil, docInfo); if(!IsPETEDefaultColor(newColor)) { colorChange = true; GetBackColor(&oldColor); RGBBackColorAndPat(&newColor); } graphicRgn = (**docInfo).graphicRgn; scratchRgn = (**(**docInfo).globals).scratchRgn; RectRgn(scrollRgn, scrollRect); DiffRgn(graphicRgn, scrollRgn, scratchRgn); OffsetRgn(graphicRgn, dh, dv); SectRgn(scrollRgn, graphicRgn, graphicRgn); UnionRgn(scratchRgn, graphicRgn, graphicRgn); GetWindowUpdateRgn((**docInfo).docWindow, scratchRgn); GlobalToLocalRgn(scratchRgn); SectRgn(scrollRgn, scratchRgn, scratchRgn); OffsetRgn(scratchRgn, dh, dv); ScrollRect(scrollRect, dh, dv, scrollRgn); UnionRgn(scratchRgn, scrollRgn, scrollRgn); if(colorChange) { RGBBackColorAndPat(&oldColor); } }