- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试放大 NSScrollView
其中包含 NSTextView
并始终使其以其内容为中心。 NSTextView
具有左/右插图,以保持自动换行的一致性,并使段落很好地位于 View 的中心。
两者[NSScrollView scaleUnitSquareToSize:...]
和setMagnification:...
有自己的怪癖和问题,但目前 setMagnification 似乎是更好的选择,因为它不是相对的。
调整大小时,我更新插图:
CGFloat inset = self.textScrollView.frame.size.width / 2 - _documentWidth / 2;
self.textView.textContainerInset = NSMakeSize(inset, TEXT_INSET_TOP);
self.textView.textContainer.size = NSMakeSize(_documentWidth, self.textView.textContainer.size.height);
放大:
CGFloat magnification = [self.textScrollView magnification];
NSPoint center = NSMakePoint(self.textScrollView.frame.size.width / 2, self.textScrollView.frame.size.height / 2);
if (zoomIn) magnification += .05; else magnification -= .05;
[self.textScrollView setMagnification:magnification centeredAtPoint:center];
一切都会暂时有效。有时,根据窗口调整大小的窗口角,ScrollView 会失去其中心,并且我还没有找到将放大的 View 重新居中的解决方案 NSScrollView
.
放大后,调整窗口大小时布局约束也可能会被破坏,特别是当textContainer
时被剪切到 View 之外,并且应用程序崩溃并出现以下错误: *** Assertion failure in -[NSISLinearExpression addVariable:coefficient:], /Library/Caches/com.apple.xbs/Sources/Foundation/Foundation-1349.91/Layout.subproj/IncrementalSimplex/NSISLinearExpression.m:716
一个问题可能是我根据 UIScrollView
设置插图帧大小,因为包含的 NSTextView 的坐标在放大后似乎不是相对的而是绝对的。
是否有任何安全的方法来放大此类 View 并使其始终以其内容为中心?为什么我的限制会被打破?
最佳答案
我也遇到过类似的问题,不幸的是我最终自己进行了居中。以下是我的解决方案的一些亮点。
代码如下:
@interface FlippedParentView : NSView
@end
@implementation FlippedParentView
- (BOOL) isFlipped { return YES; }
@end
- (void)awakeFromNib
{
[self resetMouseInfo];
[[self window] setAcceptsMouseMovedEvents:YES];
needsFullRedraw = YES;
[self setAcceptsTouchEvents:YES];
// problem: when zoomed-in, CALayer backed NSOpenGLView becomes too large
// and hurts performance.
// solution: create a fullsizeView for the NSScrollView to resize,
// and make NSOpenGLView a subview. Keep NSOpenGLView size the same as visibleRect,
// positioning it as needed on the fullsizeView.
NSScrollView *scrollvw = [self enclosingScrollView];
[scrollvw setBackgroundColor:[NSColor darkStrokeColor]];
fullsizeView = [[FlippedParentView alloc] initWithFrame: [self frame]];
[scrollvw setDocumentView:fullsizeView];
[fullsizeView setAutoresizesSubviews:NO];
//printf("mask %d\n", [self autoresizingMask]);
[fullsizeView setAutoresizingMask: NSViewHeightSizable | NSViewWidthSizable | NSViewMinYMargin | NSViewMaxYMargin | NSViewMaxXMargin | NSViewMinXMargin];
[self setAutoresizingMask: NSViewNotSizable];
[fullsizeView addSubview:self];
}
- (NSRect) visibleRect
{
NSRect visRect = [super visibleRect];
if ( visRect.size.width == 0 )
{
visRect = [[self superview] visibleRect];
if ( visRect.size.width == 0 )
{
// this jacks up everything
DUMP( @"bad visibleRect" );
}
visRect.origin = NSZeroPoint;
}
return visRect;
}
- (void) _my_zoom: (double)newZoom
{
mouseFocusPt = [self focusPt];
NSRect oldVisRect = [[self superview] visibleRect];
if ( newZoom < 1.0 )
newZoom = 1.0;
if ( newZoom > kZoomFactorMax ) newZoom = kZoomFactorMax;
float xpct = (mouseFocusPt.x - oldVisRect.origin.x) /
( NSMaxX(oldVisRect) - oldVisRect.origin.x );
float ypct = (mouseFocusPt.y - oldVisRect.origin.y) /
( NSMaxY(oldVisRect) - oldVisRect.origin.y );
float oldZoom = zoomFactor;
zoomFactor = newZoom;
/////////////////////////////////////////////////////////////////////////////////////////////////////
// Stay locked on users' relative mouse location, so user can zoom in and back out without
// the view scrolling out from under the mouse location.
NSPoint newFocusPt = NSMakePoint (mouseFocusPt.x * newZoom/oldZoom,
mouseFocusPt.y * newZoom/oldZoom) ;
NSRect myFrame = fullsizeFrame; // [self frame];
float marginPercent = (myFrame.size.height - drawableSizeWithMargins.height) / drawableSizeWithMargins.height;
[self updateContext];
NSRect newVisRect;
newVisRect.size = [self visibleRect].size;
newVisRect.origin.x = (newFocusPt.x) - (xpct * newVisRect.size.width);
//DLog( @"xpct %0.2f, zoomFactor %0.2f, newVisRect.origin.x %0.2f", xpct, zoomFactor, newVisRect.origin.x);
myFrame = fullsizeFrame; // [self frame];
float marginPercent2 = (myFrame.size.height - drawableSizeWithMargins.height) / drawableSizeWithMargins.height;
float marginDiff = (marginPercent - marginPercent2) * drawableSizeWithMargins.height;
newVisRect.origin.y = (newFocusPt.y ) - (ypct * newVisRect.size.height) - marginDiff;
//DLog( @"ypct %0.2f, zoomFactor %0.2f, newVisRect.origin.y %0.2f", ypct, zoomFactor, newVisRect.origin.y);
//DLog( @"marginPercent %0.2f newVisRect %@", marginPercent, NSStringFromRect(newVisRect) );
if ( newVisRect.origin.x < 1 ) newVisRect.origin.x = 1;
if ( newVisRect.origin.y < 1 ) newVisRect.origin.y = 1;
// NSLog( @"zoom scrollRectToVisible %@ bounds %@", NSStringFromRect(newVisRect), NSStringFromRect([[self superview] bounds]) );
// if ( iUseMousePt || isSlider )
[[self superview] scrollRectToVisible:newVisRect];
}
// - zoomFactor of 1.0 is defined as the zoomFactor needed to show entire selected context within visibleRect,
// including margins of 5% of the context size
// - zoomFactor > 1.0 will make pixels look bigger (view a subsection of a larger total drawableSize)
// - zoomFactor < 1.0 will make pixels look smaller (selectedContext size will be less than drawableSize)
-(void)updateContext
{
static BOOL sRecursing = NO;
if ( sRecursing ) return; // prevent recursion
sRecursing = YES;
//NSRect scrollRect = [[self superview] frame];
NSRect clipViewRect = [[[self enclosingScrollView] contentView] frame];
NSRect visRect = [[self superview] visibleRect]; // careful... visibleRect is sometimes NSZeroRect
float layoutWidth = clipViewRect.size.width;
float layoutHeight = clipViewRect.size.height;
marginPct = layoutHeight / (layoutHeight - (overlayViewMargin*2) );
// Satisfy the constraints fully-zoomed-out case:
// 1) the drawable rect is centered in the view with at margins.
// Allow for 5% margins (1.025 = 2.5% left, right, top, bottom)
// 2) guarantee the drawable rect does not overlap the mini-map in upper right corner.
NSRect baseRect = NSZeroRect;
baseRect.size = visRect.size;
NSRect drawableBaseRect = getCenteredRectFloat(baseRect, metaUnionRect.size );
//drawableSizeWithMargins = nsIntegralSize( nsScaleSize( drawableBaseRect.size, zoomFactor ) );
drawableSizeWithMargins = nsScaleSize( drawableBaseRect.size, zoomFactor );
// drawableSize will NOT include the margins. We loop until we've satisfied
// the constraints above.
drawableSize = drawableSizeWithMargins;
do
{
NSSize shrunkSize;
shrunkSize.width = layoutWidth / marginPct;
shrunkSize.height = layoutHeight / marginPct;
//drawableSize = nsIntegralSize( nsScaleSize( drawableBaseRect.size, zoomFactor / marginPct ));
drawableSize = nsScaleSize( drawableBaseRect.size, zoomFactor / marginPct );
[self calculateMiniMapRect]; // get approx. size. Will calculate once more below.
NSRect shrunkRect = getCenteredRectNoScaling(baseRect, shrunkSize );
// DLog( @"rough miniMapRect %@ shrunk %@", NSStringFromRect(miniMapRect), NSStringFromRect(shrunkRect));
// make sure minimap doesn't overlap drawable when you scroll to top-left
NSRect topMiniMapRect = miniMapRect;
topMiniMapRect.origin.x -= visRect.origin.x;
topMiniMapRect.origin.y = 0;
if ( !NSIntersectsRect( topMiniMapRect, shrunkRect ) )
{
topMarginPercent = fabs(shrunkRect.origin.y - drawableBaseRect.origin.y) / baseRect.size.height;
break;
}
float topMarginOffset = shrunkRect.size.height + (baseRect.size.height * 0.025);
shrunkRect.origin.y = NSMaxY(baseRect) - topMarginOffset;
if ( !NSIntersectsRect( topMiniMapRect, shrunkRect ) )
{
topMarginPercent = fabs(shrunkRect.origin.y - drawableBaseRect.origin.y) / baseRect.size.height;
break;
}
marginPct *= 1.025;
} while (1);
fullsizeFrame.origin = NSZeroPoint;
fullsizeFrame.size.width = fmax(drawableSizeWithMargins.width, layoutWidth);
fullsizeFrame.size.height = fmax(drawableSizeWithMargins.height, layoutHeight);
[fullsizeView setFrame:fullsizeFrame];
NSRect myNewFrame = [fullsizeView visibleRect];
if (myNewFrame.size.width > 0)
[self setFrame: myNewFrame]; //NSView
sRecursing = NO;
}
关于objective-c - 调整窗口大小时保持缩放后的 NSScrollView 的内容居中且可见,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55950198/
我有一个 NSScrollview 嵌套在另一个 NSScrollview 中。如何使内部 View 仅处理水平滚动?垂直滚动应该移动外部 View 。 目前我将 scrollWheel: 事件从内部
有谁知道将 NSScrollView 滚动到顶部的正确方法?我正在寻找与 UIView 的 scrollToTop 方法等效的方法。 到目前为止,这是我所拥有的,但在所有情况下都不完全正确 [self
我阅读了文档 Synchronizing Scroll Views ,并且完全按照文档进行,但是有一个问题。 我想同步一个 NSTableView 和一个 NSTextView。先让NSTableVi
当您轻轻滚动 NSScrollView 时,Cocoa 标记为脏的矩形并传递给 drawRect,通常很小(可能是对于垂直 ScrollView ,高度小至一或两个像素)。框架显然已经知道大部分内容是
我有一个 NSScrollview 其中包含两个 View ,让我们分别将它们称为View A 和View B,现在它们都可以增长高度,我希望 View A 出现在顶部和底部的 View B ,当两者
我在 Mac 上使用简单的 NSScrollView 时遇到了一个奇怪的问题。我刚刚开始使用 Cocoa 编程,但我之前有过在 iOS 上使用 UIKit 的经验。我将 NSScrollView 添加
我正在尝试将链接属性添加到 NSTextView 的 NSMutableString。 由于我只想使用 TextView 将链接添加到可见文本,因此我通过 的 NSViewBoundsDidChang
我打算在 NSScrollView 中使用 NSOutlineView,它不会紧贴顶部和底部边缘,因为我需要在顶部和底部滚动边缘插入阴影(重叠渐变)。轮廓 View 应该可以滚动到垂直顶部阴影下方和底
我的主窗口中有一个 NSScrollView 项目。每当我启动该程序时, ScrollView 中都会有文本,并且它从中心开始。用户不应该从顶部开始阅读,但为什么应用程序要在中心启动它?谢谢! 最佳答
我有一个 NSCollectionView,它列出了核心数据源中的项目。每个项目都有一个按钮,触发时会滑下操作 View 。当此 View 向下滑动时, Collection View 会缩小,以便两
我有一个 NSScrollView ,我向其中添加了一堆具有 NSImage subview 的 NSViews 。基本上它是 ScrollView 中包含的一长排缩略图。 一切都很好......直到
NSScrollView 在 Mac 应用程序中如何工作?我编写了以下代码,但滚动不起作用。 NSDictionary *temp=[[ NSDictionary alloc] init ]; NS
我正在尝试构建一个无尽的 NSScrollView,即可以在任一方向无限滚动的 ScrollView 。这是通过具有固定尺寸的 ScrollView 来实现的,一旦它太靠近任一边缘,该 ScrollV
我正在尝试使用 Lion 中的 NSScrollView 构建类似于在 iPhone 应用程序中如此流行的拉动刷新概念的东西,但是没有像 contentOffset 这样的属性,并且当我滚动到上面时,
我有一个 Cocoa 窗口,其内容 View 包含一个 NSScrollView,而 NSScrollView 又包含一个固定大小的 NSView。 启动程序后,最初显示的滚动条太小,就好像内容大小比
我目前在 NSScrollView 中有一个 NSView。NSView很大,我需要滚动来管理里面的对象。 我想在 NSView 的中心绘制一个静态矩形而不滚动。我想在NSView的drawRect方
我试图弄清楚 NSScrollView 的内容放大是如何工作的: 我有一个 NSScrollView,我试图在其中添加一个包含 NSTextField 的 NSView。 我使用文本字段创建 NSVi
我正在以编程方式在 NSScrollView 内创建 NSTableView,但是当我尝试将框架设置为 ScrollView 时,出现约束错误。 Unable to simultaneously sa
我需要在更大的滚动窗口内使用固定大小的 NSTextView。 IB 要求 TextView 位于它们自己的 NSScrollView 内部,即使它们的最小/最大尺寸是固定的,因此它们实际上不会滚动。
我有一个 NSScrollView,我试图用复选框填充它(NSButton/NSToggleButton)。我添加的复选框数量应该等于数组中的项目数量。这部分正在工作。 以下代码成功地将复选框添加到
我是一名优秀的程序员,十分优秀!