草庐IT

ios - 将 TextView 的属性字符串分割为(可引用的)部分

coder 2023-07-30 原文

我正在尝试实现一个非常简单的文本编辑器,它应该与 macOS 上的 NSTextView 和 iOS 上的 UITextView 一起工作。此文本编辑器有一个工具栏按钮“Section Break”,每次单击它都会在当前光标位置插入一个新节。一个部分应该是:

  1. 在视觉上可识别作为一个部分(通过在两个后续部分之间添加视觉分隔符并可能添加一些垂直空白),

  2. 可引用。用户应该能够在编辑器中看到所有部分的列表,并且通过单击该列表中的一个项目, TextView 应立即滚动到该部分的开头。

another question我问了如何解决第一个问题,不幸的是,我还没有找到关于那部分的答案(即使有 solution that works on macOS only)。

然而,这个问题关注的是第二方面:
如何在我的 TextView 中维护所有部分的列表,其中每个部分都准确引用相关文本?

此任务的复杂性在于用户可以随时复制和粘贴或简单地编辑文本的任何部分。因此,我不能保留一个简单的段落编号数组或类似的东西。


我尝试过的以及为什么这看起来是一项如此困难的任务:

  1. 我的一个想法是子类化 NSTextStorage 并使用可变属性字符串数组作为其内部存储。然后,我将使用 NSTextAttachment 的一个特殊子类,并将其用作我的文本存储中的分节符指示符。问题在于 the text view only calls the following methods whenever the user edits text :

    func replaceCharacters(in range: NSRange, 
                           with str: String)
    

    func setAttributes(_ attrs: [NSAttributedString.Key : Any]?, 
                         range: NSRange)
    

    第一种方法只传递没有任何属性的纯字符串,第二种方法只获取属性。这意味着在第一种方法中,我无法判断替换字符是否真的应该是分节符,因此我无法决定在我的内部文本存储数组中的何处创建新元素。

    在第二种方法中,我不知道用户是否真的添加了新文本(在这种情况下我必须在我的文本存储数组中为每个分节符属性添加一个新元素)或者用户是否只是更改了一些现有文本的属性(在这种情况下,之前已经创建了新的数组元素)。

  2. 我还考虑过在 TableView 或堆栈 View 中使用多个 TextView 的想法。但是,这不起作用,因为它会阻止用户选择(和删除)多个后续部分中的文本。

  3. 最后,我尝试了对 NSLayoutManager 进行子类化,但是关于它的文档真的很薄而且对我来说似乎是错误的地方。 (毕竟,布局管理器的职责是文本的布局,而不是跟踪其结构。)

有什么想法吗?

最佳答案

How can I maintain a list of all sections in my text view where each section keeps an accurate reference to the related text?

首先确定什么是部分。直观上看,它就像是一段连续的文本,其中几段串在一起形成了一个完整的文档。但尝试更深入地探讨:一个部分中的所有文本是否都具有相同的样式?边距一样吗?您真的需要跟踪所有部分,还是仅跟踪分节符是否也有效?

接下来,由于您已经决定买入 TextKit通过使用 NSTextViewUITextView,并且由于您的需求超出了这些类最常见的“其中包含一些可编辑文本的 View ”用例,因此您需要了解有关底层类如何组合在一起的更多信息。主要的类有NSTextStorageNSTextContainerNSLayoutManager,当然还有NSTextView(或者UITextView).

了解 TextKit 类的工作原理以及它们为您提供的功能将帮助您了解如何实现“部分”概念。以下是您可以采用的几个方法示例:

  • 构建您自己的 TextView 。 TextView 所做的大部分工作实际上是由其他类完成的:文本容器、布局管理器和文本存储。您可以构建自己的包含一系列文本容器的 View ,以便每个部分由不同的容器表示,每个容器都有自己的布局管理器和存储对象。这显然是最多的工作,但它可以对部分的显示方式进行大量控制。此外,如果您构建的 View 可以在两个平台上运行,您就不必担心 UITextViewNSTextView 之间的差异。

  • 让 TextView 完成所有工作。部分的轻量级实现可能涉及为 TextView 或文本存储对象设置委托(delegate)。当发生编辑时,这两个调用委托(delegate)方法,因此如果您的委托(delegate)维护代表部分的索引或范围列表,它可以使用这些方法来更新其部分边界数据。您仍然需要弄清楚如何直观地显示部分边界,但这可以像 inserting an image 一样简单。在每个部分边界。

关于ios - 将 TextView 的属性字符串分割为(可引用的)部分,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55263313/

有关ios - 将 TextView 的属性字符串分割为(可引用的)部分的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. Ruby 解析字符串 - 2

    我有一个字符串input="maybe(thisis|thatwas)some((nice|ugly)(day|night)|(strange(weather|time)))"Ruby中解析该字符串的最佳方法是什么?我的意思是脚本应该能够像这样构建句子:maybethisissomeuglynightmaybethatwassomenicenightmaybethiswassomestrangetime等等,你明白了......我应该一个字符一个字符地读取字符串并构建一个带有堆栈的状态机来存储括号值以供以后计算,还是有更好的方法?也许为此目的准备了一个开箱即用的库?

  3. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  4. ruby-on-rails - unicode 字符串的长度 - 2

    在我的Rails(2.3,Ruby1.8.7)应用程序中,我需要将字符串截断到一定长度。该字符串是unicode,在控制台中运行测试时,例如'א'.length,我意识到返回了双倍长度。我想要一个与编码无关的长度,以便对unicode字符串或latin1编码字符串进行相同的截断。我已经了解了Ruby的大部分unicode资料,但仍然有些一头雾水。应该如何解决这个问题? 最佳答案 Rails有一个返回多字节字符的mb_chars方法。试试unicode_string.mb_chars.slice(0,50)

  5. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  6. ruby-on-rails - Rails 常用字符串(用于通知和错误信息等) - 2

    大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje

  7. ruby - 如何以所有可能的方式将字符串拆分为长度最多为 3 的连续子字符串? - 2

    我试图获取一个长度在1到10之间的字符串,并输出将字符串分解为大小为1、2或3的连续子字符串的所有可能方式。例如:输入:123456将整数分割成单个字符,然后继续查找组合。该代码将返回以下所有数组。[1,2,3,4,5,6][12,3,4,5,6][1,23,4,5,6][1,2,34,5,6][1,2,3,45,6][1,2,3,4,56][12,34,5,6][12,3,45,6][12,3,4,56][1,23,45,6][1,2,34,56][1,23,4,56][12,34,56][123,4,5,6][1,234,5,6][1,2,345,6][1,2,3,456][123

  8. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  9. ruby-on-rails - 如果为空或不验证数值,则使属性默认为 0 - 2

    我希望我的UserPrice模型的属性在它们为空或不验证数值时默认为0。这些属性是tax_rate、shipping_cost和price。classCreateUserPrices8,:scale=>2t.decimal:tax_rate,:precision=>8,:scale=>2t.decimal:shipping_cost,:precision=>8,:scale=>2endendend起初,我将所有3列的:default=>0放在表格中,但我不想要这样,因为它已经填充了字段,我想使用占位符。这是我的UserPrice模型:classUserPrice回答before_val

  10. ruby - 如何使用文字标量样式在 YAML 中转储字符串? - 2

    我有一大串格式化数据(例如JSON),我想使用Psychinruby​​同时保留格式转储到YAML。基本上,我希望JSON使用literalstyle出现在YAML中:---json:|{"page":1,"results":["item","another"],"total_pages":0}但是,当我使用YAML.dump时,它不使用文字样式。我得到这样的东西:---json:!"{\n\"page\":1,\n\"results\":[\n\"item\",\"another\"\n],\n\"total_pages\":0\n}\n"我如何告诉Psych以想要的样式转储标量?解

随机推荐