在 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 - 从 Nib 实例化总是返回 UIViewController 类型而不是动态类型