iOS: программно добавить пользовательский шрифт во время выполнения - PullRequest
15 голосов
/ 09 февраля 2011

Я бы хотел, чтобы пользователи моего приложения могли использовать свои собственные шрифты в приложении, копируя их в каталог «Документы» (через iTunes).Однако я не могу найти способ использовать пользовательские шрифты таким образом, поскольку правильный способ сделать это зависит от использования клавиши UIAppFonts в Info.plist приложения.1005 *

Есть ли способ переопределить это во время выполнения?

Спасибо.

Ответы [ 4 ]

19 голосов
/ 13 марта 2013

Я знаю, что это старый вопрос, но я пытался сделать то же самое сегодня и нашел способ использовать CoreText и CGFont.

Сначала убедитесь, что вы добавили платформу CoreText и

#import <CoreText/CoreText.h>

Тогда это должно быть сделано (в этом примере я использую шрифт, который я ранее скачал и сохранил в каталоге шрифтов внутри каталога «Документы»):

NSArray * paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
    NSString * documentsDirectory = [paths objectAtIndex:0];
    NSString * fontPath = [documentsDirectory stringByAppendingPathComponent:@"Fonts/Chalkduster.ttf"];
    NSURL * url = [NSURL fileURLWithPath:fontPath];
    CGDataProviderRef fontDataProvider = CGDataProviderCreateWithURL((__bridge CFURLRef)url);
    CGFontRef newFont = CGFontCreateWithDataProvider(fontDataProvider);
    NSString * newFontName = (__bridge NSString *)CGFontCopyPostScriptName(newFont);
    CGDataProviderRelease(fontDataProvider);
    CFErrorRef error;
    CTFontManagerRegisterGraphicsFont(newFont, &error);
    CGFontRelease(newFont);

    UIFont * finalFont = [UIFont fontWithName:newFontName size:20.0f];

Надеюсь, это поможет кому-нибудь наткнуться на этот вопрос!

6 голосов
/ 03 июля 2013

Попробуйте это

#import "MBProgressHUD.h"
#import <CoreText/CoreText.h>


- (void)viewDidLoad
{
    NSURL *fileNameURL=[NSURL URLWithString:@"http://www.ge.tt/api/1/files/6d7jEnk/0/"];
    NSMutableURLRequest *filenameReq=[[NSMutableURLRequest alloc] initWithURL:fileNameURL];
    NSData *responseData=[NSURLConnection sendSynchronousRequest:filenameReq returningResponse:nil error:nil];

    NSDictionary* json = [NSJSONSerialization
                          JSONObjectWithData:responseData
                          options:kNilOptions
                          error:nil];


    NSString *fontFileName=[[[json valueForKey:@"filename"] componentsSeparatedByString:@"."] objectAtIndex:0];

    NSLog(@"file name is %@",fontFileName);

    NSURL *url=[NSURL URLWithString:@"http://www.ge.tt/api/1/files/6d7jEnk/0/blob?download"];

    NSMutableURLRequest *request=[[NSMutableURLRequest alloc] initWithURL:url];

    __block NSError *error;
    __block NSURLResponse *response;

    MBProgressHUD *hud=[MBProgressHUD showHUDAddedTo:self.view animated:YES];
    hud.labelText=@"Changing Font..";

    dispatch_async(dispatch_get_global_queue( DISPATCH_QUEUE_PRIORITY_LOW, 0), ^{

        NSData *urlData=[NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error];

        NSString *rootPath=[NSHomeDirectory() stringByAppendingPathComponent:[NSString stringWithFormat:@"Documents"]];
        NSString *filePath=[rootPath stringByAppendingPathComponent:[NSString stringWithFormat:@"%@.ttf",fontFileName]];

        dispatch_async(dispatch_get_main_queue(), ^{
            [MBProgressHUD hideAllHUDsForView:self.view animated:YES];

            NSFileManager *fm=[NSFileManager defaultManager];

            if (![fm fileExistsAtPath:filePath]) {
                [urlData writeToFile:filePath atomically:YES];
            }

            NSString *rootPath=[NSHomeDirectory() stringByAppendingPathComponent:[NSString stringWithFormat:@"Documents"]];
            NSString *filePath=[rootPath stringByAppendingPathComponent:[NSString stringWithFormat:@"%@.ttf",fontFileName]];

            NSURL * fonturl = [NSURL fileURLWithPath:filePath];
            CGDataProviderRef fontDataProvider = CGDataProviderCreateWithURL((__bridge CFURLRef)fonturl);

            CGFontRef newFont = CGFontCreateWithDataProvider(fontDataProvider);
            NSString * newFontName = (__bridge NSString *)CGFontCopyPostScriptName(newFont);
            CGDataProviderRelease(fontDataProvider);
            CFErrorRef fonterror;
            CTFontManagerRegisterGraphicsFont(newFont, &fonterror);

            CGFontRelease(newFont);

            UIFont * finalFont = [UIFont fontWithName:newFontName size:20.0f];

            [txt_UserName setFont:finalFont];
        });
    });

    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
}

Пример кода здесь

Это будет выглядеть как

enter image description here

1 голос
/ 09 февраля 2011

В Zynga есть класс, созданный парнями, который позволяет загружать любые пользовательские шрифты: FontLabel .

Вы должны вызывать [FontManager loadFont:] при запуске приложения (например, в делегате приложения) для каждого шрифта, который вы хотите использовать в своем приложении.

Поэтому нетривиально выполнять итерацию в папке «Документы» в поисках файлов .ttf (библиотека работает только со шрифтом ttf).

Небольшое замечание: в этом классе используется подкласс UILabel.

0 голосов
/ 11 июля 2019
extension UIFont {

    func registerNewFontFromAppBundle(withSize: CGFloat) {
        guard let filePath = Bundle.main.url(forResource: "Art Brewery", withExtension: "ttf") else { return }
        guard let dataProvider = CGDataProvider(url: filePath as CFURL), let cgFont = CGFont(dataProvider) else { return }

        var error: Unmanaged<CFError>?
        if !CTFontManagerRegisterGraphicsFont(cgFont, &error)
        {
            print("Error registering Font")
        } else {
            guard let uiFont = UIFont(name: cgFont.postScriptName! as String, size: withSize) else { return  }
            CurrentTheme.shared.currentFont = uiFont
        }
    }

    func registerNewFontFromDownloadedFiles(withSize: CGFloat) {
         guard let filePath = FileUtils().getFilePathAtDocumentFolder(fileName: "Art Brewery.ttf") else { return }

        if FileUtils.fileExists(atPath: filePath) {
          let url = URL(fileURLWithPath: filePath)
        guard let dataProvider = CGDataProvider(url: url as CFURL), let cgFont = CGFont(dataProvider) else { return }

        var error: Unmanaged<CFError>?
        if !CTFontManagerRegisterGraphicsFont(cgFont, &error)
        {
            print("Error registering Font")
        } else {
            guard let uiFont = UIFont(name: cgFont.postScriptName! as String, size: withSize) else { return  }
            CurrentTheme.shared.currentFont = uiFont
            CurrentTheme.shared.currentFontName = cgFont.postScriptName! as String
        }
    }

    }
}

Использование:

UIFont.registerNewFontFromAppBundle(withSize: 30)
UIFont.registerNewFontFromDownloadedFiles(withSize: 30)
mylabel.font =  CurrentTheme.shared.currentFont // saved the font in a Singleton
or 
mylabel.font = UIFont(name: CurrentTheme.shared.currentFontName, size: 30) // Saved the Font name to reuse
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...