Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for custom fonts #421

Merged
merged 11 commits into from
Jul 12, 2021
194 changes: 194 additions & 0 deletions Sources/TokamakCore/Tokens/Font/Font.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,194 @@
// Copyright 2018-2021 Tokamak contributors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

import Foundation

public struct Font: Hashable {
let provider: AnyFontBox

init(_ provider: AnyFontBox) {
self.provider = provider
}
}

public extension Font {
struct Weight: Hashable {
public let value: Int

public static let ultraLight: Self = .init(value: 100)
public static let thin: Self = .init(value: 200)
public static let light: Self = .init(value: 300)
public static let regular: Self = .init(value: 400)
public static let medium: Self = .init(value: 500)
public static let semibold: Self = .init(value: 600)
public static let bold: Self = .init(value: 700)
public static let heavy: Self = .init(value: 800)
public static let black: Self = .init(value: 900)
}
}

public extension Font {
enum Leading {
case standard
case tight
case loose
}
}

public enum _FontNames: Hashable {
case system
case custom(String)
}

public extension Font {
static func system(size: CGFloat, weight: Weight = .regular,
design: Design = .default) -> Self
{
.init(
_ConcreteFontBox(
.init(
name: .system,
size: size,
design: design,
weight: weight,
smallCaps: false,
italic: false,
bold: false,
monospaceDigit: false,
leading: .standard
)
)
)
}

enum Design: Hashable {
case `default`
case serif
case rounded
case monospaced
}
}

public extension Font {
static let largeTitle: Self = .init(_SystemFontBox(.largeTitle))
static let title: Self = .init(_SystemFontBox(.title))
static let title2: Self = .init(_SystemFontBox(.title2))
static let title3: Self = .init(_SystemFontBox(.title3))
static let headline: Font = .init(_SystemFontBox(.headline))
static let subheadline: Self = .init(_SystemFontBox(.subheadline))
static let body: Self = .init(_SystemFontBox(.body))
static let callout: Self = .init(_SystemFontBox(.callout))
static let footnote: Self = .init(_SystemFontBox(.footnote))
static let caption: Self = .init(_SystemFontBox(.caption))
static let caption2: Self = .init(_SystemFontBox(.caption2))

static func system(_ style: TextStyle, design: Design = .default) -> Self {
.init(_ModifiedFontBox(previously: style.font.provider) {
$0._design = design
})
}

enum TextStyle: Hashable, CaseIterable {
case largeTitle
case title
case title2
case title3
case headline
case subheadline
case body
case callout
case footnote
case caption
case caption2

var font: Font {
switch self {
case .largeTitle: return .largeTitle
case .title: return .title
case .title2: return .title2
case .title3: return .title3
case .headline: return .headline
case .subheadline: return .subheadline
case .body: return .body
case .callout: return .callout
case .footnote: return .footnote
case .caption: return .caption
case .caption2: return .caption2
}
}
}
}

public extension Font {
static func custom(_ name: String, size: CGFloat) -> Self {
.init(_CustomFontBox(name, size: .dynamic(size)))
}

static func custom(_ name: String, size: CGFloat, relativeTo textStyle: TextStyle) -> Self {
.init(_CustomFontBox(name, size: .dynamic(size), relativeTo: textStyle))
}

static func custom(_ name: String, fixedSize: CGFloat) -> Self {
.init(_CustomFontBox(name, size: .fixed(fixedSize)))
}
}

public struct _FontProxy {
let subject: Font
public init(_ subject: Font) { self.subject = subject }

public var provider: AnyFontBox { subject.provider }

public func resolve(in environment: EnvironmentValues) -> AnyFontBox.ResolvedValue {
if let deferred = subject.provider as? AnyFontBoxDeferredToRenderer {
return deferred.deferredResolve(in: environment)
} else {
return subject.provider.resolve(in: environment)
}
}
}

enum FontPathKey: EnvironmentKey {
static let defaultValue: [Font] = []
}

public extension EnvironmentValues {
var _fontPath: [Font] {
get {
self[FontPathKey.self]
}
set {
self[FontPathKey.self] = newValue
}
}

var font: Font? {
get {
_fontPath.first
}
set {
if let newFont = newValue {
_fontPath = [newFont] + _fontPath.filter { $0 != newFont }
} else {
_fontPath = []
}
}
}
}

public extension View {
func font(_ font: Font?) -> some View {
environment(\.font, font)
}
}
Loading