29

我正在创建一个带有用于打包资源(笔尖、图像、字体)的包的 ios框架,并且我正在尝试在包中嵌入自定义字体,但我无法从框架中加载它,这可能吗?

1)我可以用这个本地化字体文件: objc NSString *fontPath = [[NSBundle frameworkBundle] pathForResource:@"MyCustomFont" ofType:@"ttf"]; 2)但我无法在我的字体列表中获取它: objc NSArray * array = [UIFont familyNames]; 我将我的字体名称包含在包的 plist 中,并带有“应用程序提供的字体”,但没有成功,也在app info plist,将其包含在框架资源中但没有成功。

我可以从包中加载笔尖和图像(通过以包名称为前缀),但不能加载字体。任何想法 ?

编辑:我看到以下帖子:我可以在 iPhone 应用程序中嵌入自定义字体吗?,但问题只是“我可以在 iPhone 应用程序中嵌入自定义字体吗?” 不是“我可以在外部框架/包中嵌入自定义字体吗?” 它还引用了一个有趣的动态加载,但它使用的是私有 api,这对于框架来说不是可用的解决方案。

谢谢

4

7 回答 7

22

斯威夫特 3:

首先,不要从 main 访问带有附加路径组件的框架包......从它的标识符实例化它。您可以像这样获取字体 URL:

static func fontsURLs() -> [URL] {
    let bundle = Bundle(identifier: "com.company.project.framework")!
    let fileNames = ["Roboto-Bold", "Roboto-Italic", "Roboto-Regular"]
    return fileNames.map({ bundle.url(forResource: $0, withExtension: "ttf")! })
}

而且我发现有UIFont注册字体的扩展名很好:

public extension UIFont {
    public static func register(from url: URL) throws {
        guard let fontDataProvider = CGDataProvider(url: url as CFURL) else {
            throw SVError.internal("Could not create font data provider for \(url).")
        }
        let font = CGFont(fontDataProvider)
        var error: Unmanaged<CFError>?
        guard CTFontManagerRegisterGraphicsFont(font, &error) else {
            throw error!.takeUnretainedValue()
        }
    }
}

现在享受注册:

do {
    try fontsURLs().forEach({ try UIFont.register(from: $0) })
} catch {
    print(error)
}
于 2016-11-22T18:47:08.340 回答
21

这是一种新方法,可让您动态加载字体而无需将它们放入 Info.plist: http: //www.marco.org/2012/12/21/ios-dynamic-font-loading

于 2013-03-19T20:56:10.460 回答
17

这是我根据“David M”提供的解决方案为我的 fmk 实现它的方式。此解决方案不需要在 plist 中添加对字体的引用。

1)加载字体的类

- (void) loadMyCustomFont{
    NSString *fontPath = [[NSBundle frameworkBundle] pathForResource:@"MyFontFileNameWithoutExtension" ofType:@"ttf"];
    NSData *inData = [NSData dataWithContentsOfFile:fontPath];
    CFErrorRef error;
    CGDataProviderRef provider = CGDataProviderCreateWithCFData((__bridge CFDataRef)inData);
    CGFontRef font = CGFontCreateWithDataProvider(provider);
    if (! CTFontManagerRegisterGraphicsFont(font, &error)) {
        CFStringRef errorDescription = CFErrorCopyDescription(error);
        NSLog(@"Failed to load font: %@", errorDescription);
        CFRelease(errorDescription);
    }
    CFRelease(font);
    CFRelease(provider);
}

2) NSBundle 上的类别以访问我的包

+ (NSBundle *)frameworkBundle {
    static NSBundle* frameworkBundle = nil;
    static dispatch_once_t predicate;
    dispatch_once(&predicate, ^{
        NSString* mainBundlePath = [[NSBundle mainBundle] resourcePath];
        NSString* frameworkBundlePath = [mainBundlePath stringByAppendingPathComponent:@"MyBundleName.bundle"];
        frameworkBundle = [NSBundle bundleWithPath:frameworkBundlePath];
    });
    return frameworkBundle;
}

注意:需要在你的项目中集成 CoreText

于 2013-03-20T10:35:11.260 回答
4

迅速,我使用下面的代码:

public class func loadMyCustomFont(name:String) -> Bool{
  let fontPath = self.frameworkBundle().pathForResource(name, ofType: "ttf")!
  let inData = NSData(contentsOfFile:fontPath)
  var error: Unmanaged<CFError>?
  let provider = CGDataProviderCreateWithCFData(inData)
  if let font = CGFontCreateWithDataProvider(provider) {
     CTFontManagerRegisterGraphicsFont(font, &error)
     if error != nil {
       print(error) //Or logged it
       return false
     }


      }
       return true
   }

frameworkBundle 方法:

class func frameworkBundle() -> NSBundle{
       var bundle = NSBundle()
       var predicate = dispatch_once_t()
       dispatch_once(&predicate) {
          let mainBundlePath = NSBundle.mainBundle().bundlePath
          let frameworkBundlePath = mainBundlePath.stringByAppendingString("/myFramework.framework/")
          bundle = NSBundle(path: frameworkBundlePath)!
       }
       return bundle
 }

调用示例:(在我的情况下,我在Fonts文件夹中添加了所有字体)

YouClassName.loadMyCustomFont("Fonts/Roboto-Regular")

欢迎您的指正和评论!

于 2015-10-16T16:53:21.853 回答
1

如果您在文件/包中有字体,则可以使用此扩展名。

public extension UIFont {

    static func register(from url: URL) throws {
        if !FileManager.default.fileExists(atPath: url.path) {
            throw VError.incorrectFont
        }

        var error: Unmanaged<CFError>?

        guard CTFontManagerRegisterFontsForURL(url as CFURL, .process, &error) else {
            throw error!.takeUnretainedValue()
        }
    }
}
于 2020-11-16T15:18:23.647 回答
1

为 Swift 4/5 更新并更改为抛出错误而不是返回 Bool。

enum FontLoadingError: Error {
    case fileNotFound
    case unreadableFontData
}

func loadCustomFont(name: String) throws {
    guard let fontURL = frameworkBundle.url(forResource: name, withExtension: "ttf") else {
        throw FontLoadingError.fileNotFound
    }

    guard
        let provider = CGDataProvider(url: fontURL as CFURL),
        let font = CGFont(provider)
    else {
        throw FontLoadingError.unreadableFontData
    }

    var cfError: Unmanaged<CFError>?
    CTFontManagerRegisterGraphicsFont(font, &cfError)

    if let error = cfError as? Error {
        throw error
    }
}
于 2020-06-18T13:48:36.887 回答
0

@Ali-ABBAS 答案的 Swift 3 版本,也更新为向上包装选项,而不是强制展开。

fileprivate func loadCustomFont(name:String) -> Bool{

    guard let fontPath = frameworkBundle.path(forResource: name, ofType: "ttf") else {
        return false
    }

    guard let inData = NSData(contentsOfFile:fontPath) else {
        return false
    }


    guard let provider = CGDataProvider(data: inData) else {
        return false
    }

    let font = CGFont(provider)
    var error: Unmanaged<CFError>?
    CTFontManagerRegisterGraphicsFont(font, &error)
    guard error == nil else {
        print(error) //Or logged it
        return false
    }

    return true
}
于 2017-03-17T18:05:20.097 回答