iOS CoreText 通过 CTFontManagerRegisterGraphicsFont 获取注册字体列表



我正在通过以下方式动态注册字体:

- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation{
if ([url isFileURL])
{
// Handle file being passed in
NSLog(@"handleOpenURL: %@",url.absoluteString);

NSData *inData = [NSData dataWithContentsOfURL:url];
CFErrorRef error;
CGDataProviderRef provider = CGDataProviderCreateWithCFData((CFDataRef)inData);
CGFontRef fontRef = CGFontCreateWithDataProvider(provider);
UIFont *font;
if (!CTFontManagerRegisterGraphicsFont(fontRef, &error)) {
CFStringRef errorDescription = CFErrorCopyDescription(error);
NSLog(@"Failed to load font: %@", error);
CFRelease(errorDescription);
} else {
CFStringRef fontNameRef = CGFontCopyPostScriptName(fontRef);
NSLog(@"fontNameRef: %@",fontNameRef);
font = [UIFont fontWithName:(__bridge NSString *)fontNameRef size:80];
[self.arrayOfFonts addObject:(__bridge NSString *)fontNameRef];
[[NSNotificationCenter defaultCenter] postNotificationName:@"refreshFont" object:nil];
CFRelease(fontNameRef);
}
CFRelease(fontRef);
CFRelease(provider);
return YES;
}
else
{
return NO;
}
}

第一次工作正常。似乎如果我关闭应用程序并尝试再次注册相同的字体,那么它会给我(预期的)错误"Failed to load font: Error Domain=com.apple.CoreText.CTFontManagerErrorDomain Code=105 "Could not register the CGFont '<CGFont (0x1c00f5980): NeuropolXRg-Regular>'" UserInfo={NSDescription=Could not register the CGFont '<CGFont (0x1c00f5980): NeuropolXRg-Regular>', CTFailedCGFont=<CGFont (0x1c00f5980): NeuropolXRg-Regular>}"

这似乎是因为字体已经注册。CTFontManagerRegisterGraphicsFont的文档指出:

"向字体管理器注册指定的图形字体。注册的字体可通过字体描述符匹配来发现。尝试注册已注册或包含与已注册字体相同的 PostScript 名称的字体将失败。

究竟如何"通过字体描述符匹配"??

如何获取已通过CTFontManagerRegisterGraphicsFont方法注册的所有字体的列表,以便在再次注册之前取消注册它们?

编辑:

我尝试使用CTFontManagerCopyAvailablePostScriptNamesCTFontManagerCopyAvailableFontFamilyNames方法,但两者都只打印出iOS上已有的字体名称。不是我通过CTFontManagerRegisterGraphicsFont注册的那些

注意:我不是在询问iOS上已经可用的字体,这些字体可以通过迭代[UIFont familyNames]列出。

我在Apple DTS(开发人员技术支持)上记录了一张票,他们说:

"您需要自己跟踪使用CTFontManagerRegisterGraphicsFont注册的字体。CTFontManagerRegisterGraphicsFont返回的错误代码kCTFontManagerErrorAlreadyRegister将告诉您是否已注册字体。使用字体描述符匹配来发现是否安装了字体可能不是一个好方法,因为系统可能会选择对缺少的字体执行字体替换。使用 CTFontManagerRegisterGraphicsFont 安装字体只是使其可供您的应用程序使用。 它不是用于发现已安装字体的查询服务。如果这还不足以满足您的偏好,那么我认为您最好考虑提交功能请求,要求提供您想要获得的功能。

所以基本上,我们需要跟踪我们自己注册的字体。

解决方案/解决方法我最终使用了:

目前,我的应用程序允许用户使用字体文件上的Action Sheet's"复制到 MYAPP"按钮添加字体。同样的解决方案也适用于我从服务器下载的字体文件。

为了使我的应用列在.ttf.otf文件的操作表中,在我的应用的 info.plist 中,我添加了一种新的文档类型:

<key>CFBundleDocumentTypes</key>
<array>
<dict>
<key>CFBundleTypeIconFiles</key>
<array/>
<key>CFBundleTypeName</key>
<string>Font</string>
<key>LSItemContentTypes</key>
<array>
<string>public.opentype-font</string>
<string>public.truetype-ttf-font</string>
</array>
</dict>
</array>

这允许我的应用显示在任何字体文件的action sheet中。因此,用户可以将字体文件放在保管箱,谷歌云端硬盘或任何其他文件共享应用程序上。然后,他们可以将字体从那里导入我的应用程序。导入后,我需要将字体文件从临时tmpinbox文件夹移动到应用的Documentsfonts目录。fonts目录保留所有自定义字体。之后,我注册此自定义字体并将名称添加到self.arrayOfFonts数组中。这是包含我所有字体列表的数组。

此外,CTFontManagerRegisterGraphicsFont似乎仅适用于会话。因此,当应用程序从应用程序切换器关闭并重新启动时,该字体将不再注册。因此,每次启动后,我都会浏览我的documents/fonts文件夹并重新注册所有字体并将其名称添加到self.arrayOfFonts数组中。

我的其余应用程序代码可以使其正常工作:

- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation{
if ([url isFileURL])
{
// Handle file being passed in
NSLog(@"handleOpenURL: %@, extension: %@",url.absoluteString,url.pathExtension);
[self moveFontFrom:url];
return YES;
}
else
{
return NO;
}
}
-(void)moveFontFrom:(NSURL*)fromurl{
NSString *stringPath = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)objectAtIndex:0] stringByAppendingPathComponent:@"fonts"];
// New Folder is your folder name
NSError *error1 = nil;
if (![[NSFileManager defaultManager] fileExistsAtPath:stringPath]){
[[NSFileManager defaultManager] createDirectoryAtPath:stringPath withIntermediateDirectories:NO attributes:nil error:&error1];
}
NSLog(@"error1: %@", error1.debugDescription);
NSURL *tourl = [NSURL fileURLWithPath:[NSString stringWithFormat:@"%@/%@",stringPath,[[fromurl absoluteString] lastPathComponent]] isDirectory:NO];
NSLog(@"Trying to move from:nn%@nnto:nn%@nn", fromurl.absoluteString,tourl.absoluteString);
NSError* error2;
if ([[NSFileManager defaultManager] fileExistsAtPath:tourl.path]){
[[NSFileManager defaultManager] removeItemAtPath:tourl.path error:&error2];
NSLog(@"Deleting old existing file at %@ error2: %@", tourl.path,error2.debugDescription);
}

NSError* error3;
[[NSFileManager defaultManager] moveItemAtURL:fromurl toURL:tourl error:&error3];
NSLog(@"error3: %@", error3.debugDescription);
if (!error3) {
NSString *fontName = [self registerFont:tourl checkIfNotify:YES];
if (fontName) {
if (![self.arrayOfFonts containsObject:fontName]) {
[self.arrayOfFonts addObject:fontName];
[self.arrayOfFonts sortUsingSelector:@selector(localizedCaseInsensitiveCompare:)];
[[NSNotificationCenter defaultCenter] postNotificationName:@"refreshFont" object:nil userInfo:@{@"font":fontName}];
}
}
}
}
-(void)startupLoadFontsInDocuments{
self.arrayOfFonts = [NSMutableArray new];
NSString *location = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)objectAtIndex:0] stringByAppendingPathComponent:@"fonts"];
NSArray *directoryContent = [[NSFileManager defaultManager] contentsOfDirectoryAtPath:location error:NULL];
for (NSInteger count = 0; count < [directoryContent count]; count++)
{
NSLog(@"File %ld: %@/%@", (count + 1), location,[directoryContent objectAtIndex:count]);
NSString *fontName = [self registerFont:[NSURL fileURLWithPath:[NSString stringWithFormat:@"%@/%@",location,[directoryContent objectAtIndex:count]] isDirectory:NO] checkIfNotify:NO];
if (fontName) {
if (![self.arrayOfFonts containsObject:fontName]) {
[self.arrayOfFonts addObject:fontName];
}
}
}
[self.arrayOfFonts sortUsingSelector:@selector(localizedCaseInsensitiveCompare:)];
}
-(NSString*)registerFont:(NSURL *)url checkIfNotify:(BOOL)checkIfNotify{
NSData *inData = [NSData dataWithContentsOfURL:url];
CFErrorRef registererror;
CGDataProviderRef provider = CGDataProviderCreateWithCFData((CFDataRef)inData);
CGFontRef fontRef = CGFontCreateWithDataProvider(provider);
NSString *fontName = (__bridge NSString *)CGFontCopyPostScriptName(fontRef);
BOOL registerFontStatus = CTFontManagerRegisterGraphicsFont(fontRef, &registererror);
if (!registerFontStatus) {
CFStringRef errorDescription = CFErrorCopyDescription(registererror);
NSError *registererr = (__bridge NSError*)registererror;
if ([registererr code]==kCTFontManagerErrorAlreadyRegistered) {
NSLog(@"Font is already registered!");
}
NSLog(@"Failed to load font: %@", registererror);
CFRelease(errorDescription);
/*CFErrorRef unregistererror;
BOOL unregisterFont = CTFontManagerUnregisterGraphicsFont(fontRef, &unregistererror);
NSLog(@"Font unregister status: %d",unregisterFont);
CFStringRef unregistererrorDescription = CFErrorCopyDescription(unregistererror);
NSError *unregistererr = (__bridge NSError*)unregistererror;
NSInteger code = [unregistererr code];
NSLog(@"Failed to unregister font: %@", unregistererr);
CFRelease(unregistererrorDescription);*/
if (checkIfNotify) {
UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"Already added" message:@"That font is already added to the app. Please select it from the fonts list." preferredStyle:UIAlertControllerStyleAlert];
[alert addAction:[UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action) {
}]];
[[self.window rootViewController] presentViewController:alert animated:YES completion:nil];
}
} else {
CFStringRef fontNameRef = CGFontCopyPostScriptName(fontRef);
fontName = (__bridge NSString*)fontNameRef;
CFRelease(fontNameRef);
NSLog(@"fontName: %@",fontName);
}
CFRelease(fontRef);
CFRelease(provider);
return fontName;
}

注意:正如您会注意到的,我已经注释掉了CTFontManagerUnregisterGraphicsFont。用于取消注册字体的CTFontManagerUnregisterGraphicsFont似乎对我不起作用,因为它给出了一个错误,指出字体正在使用中,因此无法取消注册。因此,当我需要删除字体时,我只需将其从self.arrayOfFonts数组和documents/fonts文件夹中删除即可。

你应该使用的东西是这样的:

- (void)getInstalledFonts {
NSDictionary *descriptorOptions = @{(id)kCTFontDownloadableAttribute : @YES};
CTFontDescriptorRef descriptor = CTFontDescriptorCreateWithAttributes((CFDictionaryRef)descriptorOptions);
CFArrayRef fontDescriptors = CTFontDescriptorCreateMatchingFontDescriptors(descriptor, NULL);
[self showExistingFonts:(NSArray *)CFBridgingRelease(fontDescriptors)];
CFRelease(descriptor);
}
- (void)showExistingFonts:(NSArray *)fontList {
NSMutableDictionary *fontFamilies = [NSMutableDictionary new];
for(UIFontDescriptor *descriptor in fontList) {
NSString *fontFamilyName = [descriptor objectForKey:UIFontDescriptorFamilyAttribute];
NSMutableArray *fontDescriptors = [fontFamilies objectForKey:fontFamilyName];
if(!fontDescriptors) {
fontDescriptors = [NSMutableArray new];
[fontFamilies setObject:fontDescriptors forKey:fontFamilyName];
}
[fontDescriptors addObject:descriptor];
}
}

取自: https://www.shinobicontrols.com/blog/ios7-day-by-day-day-22-downloadable-fonts

我发现,如果您曾经将UIFont(name:)与动态加载的字体一起使用。 CTFontManagerUnregisterFontsForURL无法如上所述工作。

所以解决方法是使用

let desc = UIFontDescriptor(name: <name>, size: <size>)
let font = UIFont(descriptor: desc, size: <size>)

如果您在任何时候使用 UIFont(名称:),则取消注册不会真正取消注册它。

我发现CTFontManager Register/Unregstier FontsForURL api和GraphicsFont apis就是这种情况。

最新更新