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

时间:2017-06-18 09:03:10

标签: ios fonts core-text uifont

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

- (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]列出的字体。

3 个答案:

答案 0 :(得分:3)

我用Apple DTS(开发者技术支持)记录了一张票,他们说:

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

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

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

目前,我的应用允许用户使用Action Sheet's&#34;复制到MYAPP&#34;添加字体。字体文件上的按钮。这个解决方案也适用于我从服务器下载的字体文件。

为了让我的应用列在.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中显示。因此,用户可以将字体文件放在dropbox,google驱动器或任何其他文件共享应用程序上。然后他们可以将字体从那里导入我的应用程序。导入后,我需要将字体文件从临时tmp inbox文件夹移动到我应用的Documents fonts目录。 fonts目录保留所有自定义字体。之后,我注册这个自定义字体并将名称添加到self.arrayOfFonts数组。这是包含所有字体列表的数组。

此外,CTFontManagerRegisterGraphicsFont似乎仅适用于会话。因此,当应用程序从App Switcher关闭并重新启动时,该字体不再被注册。因此,每次启动后,我都会浏览我的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:\n\n%@\n\nto:\n\n%@\n\n", 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文件夹中删除。

答案 1 :(得分:1)

你应该使用的是这样的:

- (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

答案 2 :(得分:0)

我发现如果你使用 UIFont(name:) 和你动态加载的字体。 CTFontManagerUnregisterFontsForURL 不能按上述方式工作。

所以解决方法是使用

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

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

如果您在任何时候使用 UIFont(name: ),注销并不会真正注销它。

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