如何禁用NSTextView的自动换行?

时间:2010-07-04 08:15:07

标签: cocoa word-wrap nstextview

默认情况下,NSTextView会自动换行。我怎么能禁用它? 我正在制作一个JSON代码查看器,所以我必须禁用它。

5 个答案:

答案 0 :(得分:25)

我从http://lists.apple.com/archives/cocoa-dev/2008/May/msg02396.html

获得了解决方案

您必须将NSTextView的最大宽度设置为非常大的数字才能使其正常工作。 (只需复制最大高度) 并启用NSScrollView的水平滚动,这是NSTextView的超级视图。 见这些图片:

http://www.flickr.com/photos/47601728@N06/4759470529/

alt text

http://www.flickr.com/photos/47601728@N06/4759470533/

alt text

更新

我发现我的旧示例代码不足以使其完全正常工作。 (因为SDK版本?) 也 这是我的完整源代码片段,它禁用OSX 10.8 SDK中的自动换行。

[self setMaxSize:CGSizeMake(FLT_MAX, FLT_MAX)];    
[self setHorizontallyResizable:YES];               
[[self textContainer] setWidthTracksTextView:NO];  
[[self textContainer] setContainerSize:CGSizeMake(FLT_MAX, FLT_MAX)];  

更新2

现在Apple is providing an official guide正确创建NSTextView。我希望这会有所帮助。

更新3

我发布了example project on Github。有关具体实施,请参阅此页面:https://github.com/Eonil/CocoaProgrammaticHowtoCollection/blob/master/ComponentUsages/TextView/ExampleApplicationController.swift?ts=4

以下是示例项目的代码段。

        if wordwrap {
            /// Matching width is also important here.
            let sz1                                     =   scroll1.contentSize
            text1.frame                                 =   CGRect(x: 0, y: 0, width: sz1.width, height: 0)
            text1.textContainer!.containerSize          =   CGSize(width: sz1.width, height: CGFloat.max)
            text1.textContainer!.widthTracksTextView    =   true
        } else {
            text1.textContainer!.widthTracksTextView    =   false
            text1.textContainer!.containerSize          =   CGSize(width: CGFloat.max, height: CGFloat.max)
        }

答案 1 :(得分:5)

Eonil单独接受的答案中给出的三行对我来说不起作用,生成的文本视图没有水平滚动,因此无法看到长行的剪切部分。

来自引用的cocoa-dev线程的完整代码片段确实产生了正确的结果。具体来说,这一系列步骤:

NSSize layoutSize = [textView maxSize];
layoutSize.width = layoutSize.height;
[textView setMaxSize:layoutSize];
[[textView textContainer] setWidthTracksTextView:NO];
[[textView textContainer] setContainerSize:layoutSize];

答案 2 :(得分:0)

我发现这个解决方案确实阻止了换行,但是对NSTextView有一些不幸的绘制后果。问题是当包含文本视图的NSPanel被拖动到比文本行更宽时,NSView的边界rect被调整(由底层布局管理器)到比文本略长 - 显示NSPanel的背景。不是我想到的......

我的解决方案是在NSPanel中调整大小,通过设置最小和最大尺寸将其约束到水平,并关闭文本视图本身的水平调整大小:

// some fragments from the creation of the NSPanel
// resizable NSPanel created
mMonitorPanel = [[NSPanel alloc] initWithContentRect: monitorRect
              styleMask: (NSTitledWindowMask| NSMiniaturizableWindowMask | NSResizableWindowMask)
                backing: NSBackingStoreBuffered defer: NO];

 // constrain resizing to horizontal only, with obvious limits ...
[mMonitorPanel setContentMaxSize: NSMakeSize(1000, 600)];
[mMonitorPanel setContentMinSize: NSMakeSize(300,  600)];

// a fragment from the init that creates an instance of a subclass of NSTextView ...

[self setVerticallyResizable: NO];
[self setHorizontallyResizable: NO];  // rather than YES as prior code snippet

// the rest is the same as above
[[self textContainer] setContainerSize: NSMakeSize(FLT_MAX, FLT_MAX)];
[[self textContainer] setWidthTracksTextView: NO];

这会导致显示器面板可以水平调整大小,不会包裹线条,并在其中包含NSPanel正确绘制。

希望这可以帮助别人 - 如果没有我在本网站上找到的所有好的修复,我就不知道自己在哪里!

答案 3 :(得分:0)

我在NSTextView上写了扩展名。

extension NSTextView {
    var wrapsLines: Bool {
        get {
            return self.textContainer?.widthTracksTextView ?? false
        }
        set {
            guard
                newValue != self.wrapsLines,
                let scrollView = enclosingScrollView,
                let textContainer = self.textContainer
                else { return }

            scrollView.hasHorizontalScroller = !newValue
            isHorizontallyResizable = !newValue
            textContainer.widthTracksTextView = newValue

            if newValue {
                self.frame.size[keyPath: \NSSize.width] = scrollView.contentSize.width
                maxSize = NSSize(width: scrollView.contentSize.width, height: CGFloat.greatestFiniteMagnitude)
                textContainer.size.width = scrollView.contentSize.width
            } else {
                let infiniteSize = CGSize(width: CGFloat.greatestFiniteMagnitude, height: CGFloat.greatestFiniteMagnitude)
                maxSize = infiniteSize
                textContainer.size = infiniteSize
            }
        }
    }
}

答案 4 :(得分:-7)

最简单的方法是在Interface Builder中,在检查器中只需更改下拉菜单。