STTextView

STTextView

基于TextKit 2的macOS和iOS高性能文本编辑组件

STTextView是基于TextKit 2框架开发的macOS和iOS文本编辑组件,旨在替代NSTextView和UITextView。它提供行号显示、语法高亮、多光标编辑和搜索替换等功能,性能优异且高度可定制。STTextView通过Swift Package Manager分发,并支持插件扩展,适用于构建高效的源代码编辑器。

STTextView文本编辑器macOSiOSTextKit 2SwiftGithub开源项目
<img height="45" src="https://yellow-cdn.veclightyear.com/835a84d5/f137f030-8142-4244-94a6-5e9dd02bc2e6.png">

STTextView

具有行号和更多功能的高性能 macOS 和 iOS 文本视图。(NSTextView/UITextView 重新实现)

本项目的目标是利用 TextKit 2 框架构建 NSTextView/UITextView 替代的可重用组件。由于许多充分理由

该组件主要是为满足 Swift Studio 作为源代码编辑器的需求而开发的。

海报

TextKit 2 在 WWDC 2021 期间被宣布为 TextKit 1 的替代品,用于文本布局等功能。苹果宣布专门用于文本编辑的视图组件 NSTextView 将采用 TextKit 2,并提供与 TextKit 1 相当的支持。当我开始深入了解 NSTextView + TextKit 2 时,我意识到截至今天(2022 年 2 月),NSTextView 和 TextKit 2 类都还未完全发挥作用。在此过程中,我向苹果报告了几个错误报告,并请求 DTS(支持票据)。最终,我被某些特定的错误阻碍,这促使我开始了这个项目。

✨ 特性

  • macOS 文本系统集成
  • 高性能文本编辑
  • 标尺视图中的行号
  • 支持标尺标记
  • 颜色和字体的自定义
  • 开启和关闭换行
  • 调整行高
  • 在文本视图中高亮/选择范围
  • 多光标编辑
  • 搜索/替换文本
  • 拼写和语法检查
  • 听写
  • 显示不可见字符
  • 可自定义的自动完成支持
  • 长内容的平滑滚动
  • LTR(从左到右)/ RTL(从右到左)布局
  • 撤销/重做
  • 插件
  • 锚定注释(通过插件)
  • 源代码语法高亮(通过插件)
<div align="center"> <video src="https://github.com/user-attachments/assets/e18c058b-8a58-47e0-a57c-a3b01f3d93db" width="90%" /> </div>

🤝 支持与赞助

项目的财务可持续性得益于我们 GitHub 赞助者的持续贡献

🗓️ 路线图

STTextView 已经非常适合作为文本编辑器组件使用。然而,在发布 v1.0 版本之前,它仍需要一些改进

建议投票支持新功能:功能请求

已知问题

  • 文本拖动(部分实现)

🚀 入门

STTextView 通过 Swift Package Manager 分发。通过在 Package.swift 清单中添加依赖项或通过 Xcode 项目设置中的"Package Dependencies"将其安装到项目中

let package = Package( dependencies: [ .package(url: "https://github.com/krzyzanowskim/STTextView", from: "0.9.6") ] )

演示应用

演示应用 TextEdit 和 TextEdit.SwiftUI 让你可以探索该库。

插件

STTextView 组件中的插件提供了超出简单文本显示之外的额外功能和定制选项。它们通过添加语法高亮、字数统计等功能来增强文本视图的核心能力。这些插件在保持模块化和适应性软件结构的同时,扩展了 STTextView 的实用性。

  • Plugin-Neon:使用 TreeSitter 和 Neon 进行源代码语法高亮。
  • Plugin-TextFormation:使用 TextFormation 进行输入完成。
  • Plugin-Annotations:锚定注释(例如内联错误消息)插件。
  • Plugin-Template:准备构建新插件的虚拟插件模板仓库。
  • ... 添加更多插件

使用方法

SwiftUI

TextView 是一个包装 STTextView 的 SwiftUI 视图。

import STTextViewSwiftUI struct ContentView: View { @State private var text = AttributedString("Hello World!") @State private var selection: NSRange? var body: some View { TextView( text: $text, selection: $selection, options: [.wrapLines, .highlightSelectedLine], plugins: [plugin1(), plugin2()] ) .textViewFont(.preferredFont(forTextStyle: .body)) } }

创建 TextView

STTextView 是 NSView 的子类,可以像其他视图一样初始化。它的 API 与 NSTextView 类似。

import STTextView let textView = STTextView() view.addSubView(textView)

(macOS)添加到滚动视图

let textView = STTextView() let scrollView = NSScrollView() scrollView.documentView = textView
let scrollView = STTextView.scrollableTextView() let textView = scrollView.documentView as! STTextView

自定义

文本视图可以通过多种方式进行自定义。

let paragraph = NSParagraphStyle.default.mutableCopy() as! NSMutableParagraphStyle // 将行高设置为 110% paragraph.lineHeightMultiple = 1.1 paragraph.defaultTabInterval = 28 // 默认段落样式 textView.typingAttributes[.paragraphStyle] = paragraph // 设置默认字体 textView.font = NSFont.monospacedSystemFont(ofSize: 14, weight: .regular) // 设置默认文本颜色 textView.textColor = .textColor // 设置文本值 textView.text = "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Aenean ornare lobortis sem a vulputate." textView.addAttributes([.foregroundColor: NSColor.red], range: NSRange(location: 10, length: 5))

// 将行包裹/不包裹到编辑器宽度 textView.isHorizontallyResizable = true

// 高亮选中的行 textView.highlightSelectedLine = true


使用专门的 `STLineNumberRulerView`(`NSRulerView` 的专门子类)添加行号

```swift
let textView = STTextView()
let scrollView = NSScrollView()
scrollView.documentView = textView

// 行号
textView.showLineNumbers = true
textView.gutterView?.drawSeparator = true

(macOS) 在视图内启用可选的搜索和替换查找界面,通常是在滚动视图中。

textView.isIncrementalSearchingEnabled = true textView.textFinder.incrementalSearchingShouldDimContentView = true

🐛 TextKit 2 错误报告列表

以下是我迄今为止向 Apple 报告的与 NSTextView 和 TextKit 框架相关的 TextKit 2 问题和错误列表:

  • FB9856587: TextKit2 最后一行出现意外的额外行片段
  • FB9925766: NSTextSelectionNavigation.deletionRanges 只在单词末尾有效
  • FB9925647: NSTextLayoutManager.replaceContents(in range: with attributedString:) 已有文档但不是公共 API 的一部分
  • FB9907261: NSTextElementProvider.replaceContents(in:with:) 未按文档所述替换内容
  • FB9692714: 渲染属性无法正确绘制
  • FB9886911: NSTextView 无法正确布局和显示长行(这个问题很棘手,因为它会导致视图在文本属性更新时"跳动")
  • FB9713415: NSTextView drawInsertionPoint(in:color:turnedOn) 从未被调用
  • FB9971054: NSLayoutManager.enumerateCaretOffsetsInLineFragment 忽略起始位置
  • FB9971054: 使用 TextKit2 设置时 NSTextView 在选择时断言
  • FB9743449, FB10019859: NSTextContentStorage.textElements(for:) 不返回元素,而 enumerateTextElements 返回元素
  • FB11898356: textSelections(interactingAt:inContainerAt:anchors:modifiers:selecting:bounds:) 对某些位置产生错误的选择
  • FB12726775: NSTextParagraph.paragraphContentRange 的文档不正确
  • FB13272586: NSTextContainer.size 的默认值与文档不符
  • FB13290979: NSTextContainer.lineFragmentPadding 不影响 usageBoundsForTextContainer 矩形的片段末端
  • FB13291926: NSTextLayoutManager.usageBoundsForTextContainer 观察器从未被触发
  • FB13789916: NSTextInputClient.setMarkedText 为中文键盘提供错误的选择范围
  • FB14700414: NSTextList 自 macOS 14 起不再工作(回归)

... 我知道这个问题列表并不完整。我在 STTextView 中设法解决了大部分问题。

为什么是 ST?

ST 前缀代表 "Swift sTudio",因为自 1939 年以来 SS 不是一个好前缀)

建议或反馈

开启一个新的讨论主题或提交拉取请求。

我很乐意听到您的声音!通过 X/Twitter @krzyzanowskim 或 Mastodon @krzyzanowskim@mastodon.social 与我联系。

许可证

开源许可证

如果您正在创建一个与 GNU GPL 许可证 v3 兼容的开源应用程序,您可以在 GPLv3 条款下使用 STTextView。

商业许可证

从 €5 起获取一个。

如果您想使用 STTextView 开发非开源产品和应用程序,商业许可证是适合的。通过这个选项,您的源代码将保持专有。您无需将整个应用程序源代码更改为开源许可证。购买 STTextView 商业许可证

编辑推荐精选

GPT Image 2中文站

GPT Image 2中文站

AI 图片生成平台

GPT Image 2 是面向用户的 AI 图片生成平台,支持文生图、图生图及多模型创意工作流。

Nano Banana Pro 中文站

Nano Banana Pro 中文站

AI 图片生成工具

输入简单文字,生成想要的图片。支持Nano Banana/gptimage-2等最新模型。

Vecbase

Vecbase

你的AI Agent团队

Vecbase 是专为 AI 团队打造的智能工作空间,将数据管理、模型协作与知识沉淀整合于一处。算法、产品与业务在同一平台无缝协同,让从数据到 AI 应用的落地更快一步。

音述AI

音述AI

全球首个AI音乐社区

音述AI是全球首个AI音乐社区,致力让每个人都能用音乐表达自我。音述AI提供零门槛AI创作工具,独创GETI法则帮助用户精准定义音乐风格,AI润色功能支持自动优化作品质感。音述AI支持交流讨论、二次创作与价值变现。针对中文用户的语言习惯与文化背景进行专门优化,支持国风融合、C-pop等本土音乐标签,让技术更好地承载人文表达。

QoderWork

QoderWork

阿里Qoder团队推出的桌面端AI智能体

QoderWork 是阿里推出的本地优先桌面 AI 智能体,适配 macOS14+/Windows10+,以自然语言交互实现文件管理、数据分析、AI 视觉生成、浏览器自动化等办公任务,自主拆解执行复杂工作流,数据本地运行零上传,技能市场可无限扩展,是高效的 Agentic 生产力办公助手。

lynote.ai

lynote.ai

一站式搞定所有学习需求

不再被海量信息淹没,开始真正理解知识。Lynote 可摘要 YouTube 视频、PDF、文章等内容。即时创建笔记,检测 AI 内容并下载资料,将您的学习效率提升 10 倍。

AniShort

AniShort

为AI短剧协作而生

专为AI短剧协作而生的AniShort正式发布,深度重构AI短剧全流程生产模式,整合创意策划、制作执行、实时协作、在线审片、资产复用等全链路功能,独创无限画布、双轨并行工业化工作流与Ani智能体助手,集成多款主流AI大模型,破解素材零散、版本混乱、沟通低效等行业痛点,助力3人团队效率提升800%,打造标准化、可追溯的AI短剧量产体系,是AI短剧团队协同创作、提升制作效率的核心工具。

seedancetwo2.0

seedancetwo2.0

能听懂你表达的视频模型

Seedance two是基于seedance2.0的中国大模型,支持图像、视频、音频、文本四种模态输入,表达方式更丰富,生成也更可控。

nano-banana纳米香蕉中文站

nano-banana纳米香蕉中文站

国内直接访问,限时3折

输入简单文字,生成想要的图片,纳米香蕉中文站基于 Google 模型的 AI 图片生成网站,支持文字生图、图生图。官网价格限时3折活动

扣子-AI办公

扣子-AI办公

职场AI,就用扣子

AI办公助手,复杂任务高效处理。办公效率低?扣子空间AI助手支持播客生成、PPT制作、网页开发及报告写作,覆盖科研、商业、舆情等领域的专家Agent 7x24小时响应,生活工作无缝切换,提升50%效率!

下拉加载更多