在 Swift 中禁用动态类型

Posted

技术标签:

【中文标题】在 Swift 中禁用动态类型【英文标题】:Disabling Dynamic Type in Swift 【发布时间】:2016-08-02 03:23:03 【问题描述】:

我有一个基于 Sprite Kit 的游戏,它在其中一个场景中使用 UIView,我这样做是为了利用 UITableViewController 来呈现游戏设置屏幕。

我遇到的困难是,当用户将他们的 iPad 系统可访问性设置设置为使用(额外的)大字体时,UITableView 中的文本对于单元格来说太大了,而且看起来很傻。

我想做的是直接禁用应用程序中的动态类型,以便它始终在单元格中显示相同大小的类型。

我发现了另一个类似的帖子 (here),但响应提供了 Objective-C 响应:

#import <objc/runtime.h>

@implementation AppDelegate

NSString* swizzled_preferredContentSizeCategory(id self, SEL _cmd) 
    return UIContentSizeCategoryLarge;  // Set category you prefer, Large being ios' default.


- (BOOL)application:(UIApplication*)application didFinishLaunchingWithOptions:(NSDictionary*)launchOptions 
    Method method = class_getInstanceMethod([UIApplication class], @selector(preferredContentSizeCategory));
    method_setImplementation(method, (IMP)swizzled_preferredContentSizeCategory);

    ...

我需要在 Swift 中执行此操作。

在 Xcode 7+ 的 Swift 中做同样事情的正确方法是什么?

【问题讨论】:

这是对许多用户来说非常重要且实际上必要的功能的懒惰回避。不要这样偷工减料。 @AMomchilov 我同意!事实上,我的团队刚刚讨论过它,我们将在未来的修订版中实现这个功能,但现在我们只需要表格看起来不那么不稳定。所以实际上,它是一个临时的创可贴。谢谢你的评论。 @zeeple Tables 看起来不错正是我实现上述代码的原因。我还确保我有一个相当大且可读的默认字体。 @meaning-matters,太棒了。将其转换为 Swift 后,我​​会将代码发布到我的帖子和您的原始帖子中。谢谢! 我工作的一个应用收到了很多用户的请求,希望能够在应用内禁用动态类型。似乎人们更喜欢为某些应用启用动态类型,而不是其他应用。因此,正当理由知道如何做到这一点。 【参考方案1】:

好的,首先让我这样说:虽然我很高兴我能够快速找到一种方法来适应 iOS 辅助功能设置提供的动态文本(我将在几秒钟内显示代码)我认为对于获得原始问题的答案仍然很重要。

也就是说,这是我对表格视图代码所做的,以尊重某些用户需要的较大类型。这是一个两步的过程。首先,添加:

tableView.estimatedRowHeight = 44.0
tableView.rowHeight = UITableViewAutomaticDimension

到 viewDidLoad 方法。然后,在 cellForRowAtIndexPath 方法中,在返回单元格之前添加以下内容:

cell.textLabel!.numberOfLines = 0

祝大家好运,如果您有原始问题的答案,请添加答案:)

【讨论】:

这个问题是关于在你的应用中禁用动态类型功能,而不是在你的表格视图单元格中调整更大的类型?【参考方案2】:

感谢@zeeple 的解决方案。 这是原始问题的答案:

"preferredContentSizeCategory" 在 Objective-C 中是一个方法,但在 Swift 中是一个只读变量。

所以在你的 AppDelegate 中是这样的:

@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate 

    var window: UIWindow?

    // MARK: - UIApplicationDelegate

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey : Any]? = nil) -> Bool 

        UIApplication.classInit

        self.window = UIWindow(frame: UIScreen.main.bounds)
        ...
        self.window?.makeKeyAndVisible()
        return true
    


// MARK: - Fix Dynamic Type

extension UIApplication 

    static let classInit: Void = 
        method_exchangeImplementations(
            class_getInstanceMethod(UIApplication.self, #selector(getter: fixedPreferredContentSizeCategory))!,
            class_getInstanceMethod(UIApplication.self, #selector(getter: preferredContentSizeCategory))!
        )
    ()

    @objc
    var fixedPreferredContentSizeCategory: UIContentSizeCategory 
        return .large
    

【讨论】:

【参考方案3】:

我想要做的是直接禁用应用程序中的动态类型,以便它始终在单元格中显示相同大小的类型。

动态类型仅适用于已实现text styles 的文本。

因此,如果您总是想禁用动态类型在单元格中显示相同大小的类型,请不要在其中使用文本样式或image size adjustment。

但是,如果您确实想使用文本样式,请不要在 Interface Builder 中为每个文本元素勾选 Automatically Adjusts Font(相当于代码中的 adjustsFontForContentSizeCategory

【讨论】:

以上是关于在 Swift 中禁用动态类型的主要内容,如果未能解决你的问题,请参考以下文章

Swift - 将类型动态传递给 JSONDecoder

Swift 动态变量不能是 Printable 类型

Swift - 从 Nib 实例化总是返回 UIViewController 类型而不是动态类型

swift swift中的小动态类型系统

Swift 根据响应类型为 JSON 字段值动态选择 Codable 结构

Swift iOS尝试动态更改UITextAutocapitalization类型