问题

我需要了解TextKit的工作原理,以及如何使用它来构建文本编辑器.我需要弄清楚如何仅绘制最终用户与之交互的可见文本,或者确定如何将属性延伸到可视文本中,而不是在processEditing方法中对整个更改的文本范围应用属性.

背景

iOS 7出来了TextKit.我有一个完全实现TextKit的tokenizer和代码(参考Apple的TextKitDemo项目 – 下面提供了一个链接)…它的工作原理.但是,它真的很慢.当文本被解析时,由NSTextStorage,它需要您在同一个线程上将editEditing方法中的编辑文本的ENTIRE范围进行着色.将工作卸载到线程没有帮助.这太慢了我得到了我可以重新归类修改的范围的点,但是如果范围太大,则过程将会很慢.在某些情况下,整个文档可能在更改后无效.

这里有一些想法.请让我知道,如果有任何这些工作,或者可能推动我正确的方向.

1)多个NSTextContainers

阅读文档,我可以在NSLayoutManager中添加多个NSTextContainers.我假设通过这样做,我应该能够定义不仅可以在NSTextContainer中绘制的行数,而且我还应该能够知道最终用户可以看到哪个NSTextContainer.我知道如果我去这条路线,我需要投入大量的时间来看看是否可行.初始测试表明您只需要一个NSTextContainer.所以我必须对NSLayout进行子类化,或者创建一个包装器,其中布局管理器确定哪个文本在哪个文本容器中.呸.此外,我不知道TextKit如何让我知道是时候画一个特定的NSTextContainer …也许这不是它的工作原理!

2)无效的范围与NSLayoutManager

无效的layoutManager使用invalidateLayoutForCharacterRange:actualCharacterRange :.但是,这实际上是做什么的,如何卸载文本归属阶段?什么时候让我知道一个特定的文本需要突出显示?另外,我看到NSLayoutManager会懒洋洋地画字形…怎么样?什么时候?这如何帮助我?如何点击这个调用,以便我可以在支持字符串实际放置文本之前对其进行归属?

3)超越NSLayoutManager drawGlyphsForGlyphRange:atPoint:方法.

我真的不想这样做话虽如此,在Mac OS X中,NSAttributedStrings具有临时属性的概念,其中样式信息仅用于呈现.这加快了突出重点的过程!问题是,它在iOS 7 TextKit框架中不存在(或者它在那里,我只是不知道).我相信,通过超越这种方法,它将给我相同类型的速度,你会得到使用临时属性…因为我可以回答所有的布局,颜色和格式化问题在这种方法,而不用触摸NSTextStorage归因串.唯一的问题是,我不知道这个方法如何与NSLayoutManager类中提供的其他方法相关.它是否保持宽度和高度的状态?当NSTextContainer太小时会修改它的大小吗?此外,它只为文本缓冲区中添加的字符绘制字形.它不会重绘整个屏幕.只有它的一小部分…这是非常好的.我有一些想法,我可以如何工作这个…但我真的不想排列字形.这是太多的工作,我没有找到一个很好的例子,这样做.

我非常感谢您提供的任何帮助.

感谢您,我列出了过去几年中使用过的所有框架和参考资料,帮助我了解了现在我希望他们对您有所帮助.

语法高亮框架:

> http://colorer.sourceforge.net/
> https://github.com/MikeJ1971/Glint(不支持字符串,注释等)
> https://projects.gnome.org/gtksourceview/features.html(为令牌生成提供了一个体面的lib),与GTK配合使用,但可能会为不同的布局管理员重新编写)
> http://parsekit.com/(很好的解析器,但是,当需要修复范围时,必须打包API来创建状态机)
> http://svn.gna.org/viewcvs/etoile/trunk/Etoile/Languages/LanguageKit/
> https://github.com/CodaFi/IDEKit(AMAZING工作,这个代码中有很多好的想法,我的问题是我完全不知道它如何修复范围,甚至是这样)
> http://www.crimsoneditor.com/(一个旧的Windows代码编辑器,它有一个非常好的tokenizer – 虽然有点难以阅读,它不使用正则表达式,也就是说,我基于我的令牌逻辑从这个代码,它是更快比上述任何框架)

资源:

> http://cocoafactory.com/blog/2012/10/29/how-to-use-custom-nsattributedstring-attributes/
> https://github.com/objcio/issue-5-textkit
> http://alexgorbatchev.com/SyntaxHighlighter/
> http://docs.xamarin.com/samples/TextKitDemo/(Apple的演示)
> http://cocoadev.com/ImplementSyntaxHighlighting(一定要阅读所有的小孩子文章,好东西)

大多数框架是相同的.他们或者不考虑上下文切换(或者你必须编写包装器以提供上下文)范围,或者当用户修改文本(例如字符串,多行注释等)时,它们不会修复上下文范围.最后一个要求是非常重要的.因为如果一个记分器不能确定哪个范围受到一个变化的影响,那么你最终将必须解析并再次对整个字符串进行归属.唯一的例外是Crimson Editor.该标记器的问题是它不会在标记化时保存状态.在绘制时,算法使用令牌来确定绘图状态.它从文档的顶部开始,直到它到达可见的文本范围.不用说,我通过在文档的某些部分缓存文档的状态来优化.

另一个问题是框架不符合Apple所做的相同MVC模式 – 这是预期的.具有完整工作编辑器的框架都使用由它们构建的API(即GTK,Windows等)提供的钩子,为他们提供何时以及何时绘制到屏幕的哪个部分的信息.在我的情况下,TextKit似乎要求您在processEditing中归因整个更改的范围.

也许我的观点是错误的. (我希望他们是!!)也许,例如ParseKit,将为我需要做的工作,我根本不明白如何使用它.如果是这样,请让我知道!再次感谢

解决方法

我想到了.我没有使用上面的建议.话虽如此,我现在得到的表现是令人难以置信的.请记住,YMMV.标记和缓存有关字符串的元数据的方式可能与我不同.但是,我可以输入一个1400行的PHP文件,只需0.015秒即可完成任何一个更改.简直令人难以置信

这是我采取的方法:

我的UIViewController是UITextViewDelegate和uiscrollviewdelegate的委托.

当UITextViewDelegate.textViewDidChange:被调用时,我确定最终用户当前可以看到哪个文本范围.我通过使用我现有的分类的UITextView来添加这个方法:

- (NSRange)visibleRangeOfText
{
    CGRect bounds = self.bounds;
    UITextPosition *start = [self characterRangeAtPoint:bounds.origin].start;
    UITextPosition *end = [self characterRangeAtPoint:CGPointMake(CGRectGetMaxX(bounds),CGRectGetMaxY(bounds))].end;
    return NSMakeRange([self offsetFromPosition:self.beginningOfDocument toPosition:start],[self offsetFromPosition:start toPosition:end]);
}

之后,我将范围传递给子类的NSTextStorage对象,然后它将执行魔术来确定哪些行需要突出显示.

UIScollViewDelegate方法调用也一样.根据视图的哪个部分,我将可见范围传递到我的子类NSTextStorage调用,并确定行是否已被归因等.

我意识到我要留下很多读者.我最终使用了我目前所做的,并调整了一些工作与上述实现.

我想分享一些我发现有趣的发现:

1)如果您尝试突出显示当前行中的任何文本,光标所在的位置,您可能会看到光标在视图中“跳转”,然后返回到原始位置.我几乎是积极的,这是由NSTextStorage.processEditing方法调用引起的.我能够把它弄到哪里,系统只突出显示出修改的行,所以这个问题现在已经不复存在了.

2)最初我这样做是为了防止光标跳动:

NSRange selectedRange = [textView selectedTextRange];
[textView setScrollEnabled:NO];
NSRange visibleRange = [textView visibleRangeOfText];
[textStorage applyAttributesToRange:visibleRange];
[textView setScrollEnabled:YES];

它工作…但是[textView setScrollEnabled:NO]调用对性能有很大的影响.这个命令单独花了近3/4的时间才能完成1400线文件.我不知道是什么原因是缓慢,但我认为值得一提.

为iOS 7创建文本编辑器的更多相关文章

  1. Swift社交应用文本输入优化汇总

    本文将汇总一下Swift社交应用文本输入优化技巧。

  2. 在 Cocoa 中实现 ICU 文本变换

    ICU库提供了一整套强大的文本变换功能,在处理用户输入、特别是当你的程序需要处理一些英语之外的语言或者非拉丁字符时非常有用。在Apple的平台中,字符串变换一直以来都是通过CoreFoundation的CFStringTranform函数来实现。输入变换结果HELLOWORLDLowerhelloworld仅转换元音字母为小写。拉丁到ASCII这一步会移除变音符以及会把ASCII码范围之外的字符和标点符号转换成ASCII中与之最为接近的版本。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@fox

  3. 可点击 @、# 标记文本实现

    在社交类APP中@、#符号构成的标记文本已经形成了某种通用的意义:前者表示通知某位好友,而后者表示为某个话题或者分类。开始上码的代码首先声明了一个wordType的枚举类型,该类用用于对标示文本进行类型标记。这里之所以使用.character而不是后面的.word的原因是:后者会将@、#这些标示符丢弃,导致一只类似点击到无效区域的情形。当上诉检查通过也就是点击区域有效的时候,我们使用.word,获取点击区域的单词。

  4. Python深度学习之FastText实现文本分类详解

    FastText是一种典型的深度学习词向量的表示方法,它非常简单通过Embedding层将单词映射到稠密空间,然后将句子中所有的单词在Embedding空间中进行平均,进而完成分类操作

  5. php – 如何使用imagick annotateImage中文文本?

    我需要使用中文文本注释图像,我现在正在使用Imagick库.中文文本的一个例子是这是中文使用的中文字体文件是this该文件最初名为华文黑体.ttf它也可以在MacOSX的/Library/Font下找到我已将其重命名为英语STHeiTi.ttf,可以更轻松地在PHP代码中调用该文件.特别是theImagick::annotateImagefunction我也是usingtheanswerfrom“

  6. 正则表达式快速入门教程

    正则表达式30分钟入门教程版本:v2.31作者:deerchao转载请注明来源目录跳过目录本文目标如何使用本教程正则表达式到底是什么东西?正则表达式就是用于描述这些规则的工具。假设你在一篇英文小说里查找hi,你可以使用正则表达式hi。如果同时使用其它元字符,我们就能构造出功能更强大的正则表达式。正则表达式的语法很令人头疼,即使对经常使用它的人来说也是如此。对中文/汉字的特殊处理是由.Net提供的正则表达

  7. 正则表达式30分钟入门教程

    入门测试正则表达式元字符字符转义重复字符类分枝条件反义分组后向引用零宽断言负向零宽断言注释贪婪与懒惰处理选项平衡组/递归匹配还有些什么东西没提到联系作者网上的资源及本文参考文献更新纪录本文目标30分钟内让你明白正则表达式是什么,并对它有一些基本的了解,让你可以在自己的程序或网页里使用它。正则表达式就是用于描述这些规则的工具。正则表达式的语法很令人头疼,即使对经常使用它的人来说也是如此。

  8. 正则表达式经验入门

    正则表达式就是用于描述这些规则的工具。入门学习正则表达式的最好方法是从例子开始,理解例子之后再自己对例子进行修改,实验。如果同时使用其它元字符,我们就能构造出功能更强大的正则表达式。正则表达式的语法很令人头疼,即使对经常使用它的人来说也是如此。对中文/汉字的特殊处理是由.Net提供的正则表达式引擎支持的,其它环境下的具体情况请查看相关文档。

  9. 正则表达式30分钟入门

    说某个字符串匹配某个正则表达式,通常是指这个字符串里有一部分能满足表达式给出的条件。正则表达式就是用于描述这些规则的工具。假设你在一篇英文小说里查找hi,你可以使用正则表达式hi。如果同时使用其它元字符,我们就能构造出功能更强大的正则表达式。正则表达式的语法很令人头疼,即使对经常使用它的人来说也是如此。对中文/汉字的特殊处理是由.Net提供的正则表达式引擎支持的,其它环境下的具体情况请查看相关文档。

  10. 正则表达式学习十一前后查找lookaround

    解决这个问题我们就需要用到前后查找。但是并不是所有的正则表达式实现都支持向后查找,JavaScript就不支持,java语言支持向后查找。这种用法叫正向前查找和正向后查找。)负向后查找比如一段文本中即有价格和数量,我们要找出价格和数量,先来看查找价格:文本:Ipaid$30for10apples,15oranges,and10pears.Isaved$5onthisorder.正则表达式:(?

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部