Sunset MathTable, created MTFontV2, MTFontMathTableV2 and MathImage, about to sunset MTMathImage
This commit is contained in:
@@ -26,8 +26,8 @@ internal struct MathTable {
|
||||
let kConstants = "constants"
|
||||
|
||||
let font: MathFont
|
||||
private let unitsPerEm: UInt
|
||||
private let fontSize: CGFloat
|
||||
let unitsPerEm: UInt
|
||||
let fontSize: CGFloat
|
||||
weak var fontMathTable: NSDictionary?
|
||||
|
||||
init(withFont font: MathFont, fontSize: CGFloat, unitsPerEm: UInt) {
|
||||
@@ -176,13 +176,15 @@ internal struct MathTable {
|
||||
let glyphName = font.get(nameForGlyph: glyph)
|
||||
let variantGlyphs = variants[glyphName] as? NSArray
|
||||
var glyphArray = [NSNumber]()
|
||||
if variantGlyphs == nil || variantGlyphs?.count == 0, let glyph = font.get(glyphWithName: glyphName) {
|
||||
if variantGlyphs == nil || variantGlyphs?.count == 0 {
|
||||
// There are no extra variants, so just add the current glyph to it.
|
||||
let glyph = font.get(glyphWithName: glyphName)
|
||||
glyphArray.append(NSNumber(value:glyph))
|
||||
return glyphArray
|
||||
} else if let variantGlyphs = variantGlyphs {
|
||||
for gvn in variantGlyphs {
|
||||
if let glyphVariantName = gvn as? String, let variantGlyph = font.get(glyphWithName: glyphVariantName) {
|
||||
if let glyphVariantName = gvn as? String {
|
||||
let variantGlyph = font.get(glyphWithName: glyphVariantName)
|
||||
glyphArray.append(NSNumber(value:variantGlyph))
|
||||
}
|
||||
}
|
||||
@@ -204,9 +206,8 @@ internal struct MathTable {
|
||||
// Find the first variant with a different name.
|
||||
for gvn in variantGlyphs! {
|
||||
if let glyphVariantName = gvn as? String,
|
||||
glyphVariantName != glyphName,
|
||||
let variantGlyph = font.get(glyphWithName: glyphVariantName) {
|
||||
return variantGlyph
|
||||
glyphVariantName != glyphName {
|
||||
return font.get(glyphWithName: glyphVariantName)
|
||||
}
|
||||
}
|
||||
// We did not find any variants of this glyph so return it.
|
||||
@@ -243,9 +244,7 @@ internal struct MathTable {
|
||||
} else {
|
||||
// If no top accent is defined then it is the center of the advance width.
|
||||
var advances = CGSize.zero
|
||||
guard let ctFont = font.ctFont(withSize: fontSize) else {
|
||||
fatalError("\(#function) unable to obtain ctFont resource name: \(font.rawValue) with size \(fontSize)")
|
||||
}
|
||||
let ctFont = font.ctFont(withSize: fontSize)
|
||||
CTFontGetAdvancesForGlyphs(ctFont, .horizontal, &glyph, &advances, 1)
|
||||
return advances.width/2
|
||||
}
|
||||
@@ -272,7 +271,8 @@ internal struct MathTable {
|
||||
}
|
||||
var rv = [GlyphPart]()
|
||||
for part in parts {
|
||||
guard let partInfo = part as? NSDictionary, let glyph = font.get(glyphWithName: glyphName) else { continue }
|
||||
guard let partInfo = part as? NSDictionary else { continue }
|
||||
let glyph = font.get(glyphWithName: glyphName)
|
||||
var part = GlyphPart(glyph: glyph)
|
||||
if let adv = partInfo["advance"] as? NSNumber,
|
||||
let end = partInfo["endConnector"] as? NSNumber,
|
||||
51
Sources/SwiftMath/MathBundle/MTFontMathTableV2.swift
Normal file
51
Sources/SwiftMath/MathBundle/MTFontMathTableV2.swift
Normal file
@@ -0,0 +1,51 @@
|
||||
//
|
||||
// MTFontMathTableV2.swift
|
||||
//
|
||||
//
|
||||
// Created by Peter Tang on 15/9/2023.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
// extension MathTable {
|
||||
// public func fontMathTableV2() -> MTFontMathTableV2 {
|
||||
// MTFontMathTableV2(mathFont: font, size: fontSize)
|
||||
// }
|
||||
// }
|
||||
internal class MTFontMathTableV2: MTFontMathTable {
|
||||
private let mathFont: MathFont
|
||||
private let fontSize: CGFloat
|
||||
private let unitsPerEm: UInt
|
||||
private let mTable: NSDictionary
|
||||
init(mathFont: MathFont, size: CGFloat) {
|
||||
self.mathFont = mathFont
|
||||
self.fontSize = size
|
||||
mTable = mathFont.mathTable()
|
||||
unitsPerEm = mathFont.ctFont(withSize: fontSize).unitsPerEm
|
||||
super.init(withFont: mathFont.mtfont(size: fontSize), mathTable: mTable)
|
||||
super._mathTable = nil
|
||||
// disable all possible access to _mathTable in superclass!
|
||||
}
|
||||
override var _mathTable: NSDictionary? {
|
||||
set { fatalError("\(#function) change to _mathTable \(mathFont.rawValue) not allowed.") }
|
||||
get { mTable }
|
||||
}
|
||||
override var muUnit: CGFloat { fontSize/18 }
|
||||
|
||||
override func fontUnitsToPt(_ fontUnits:Int) -> CGFloat {
|
||||
CGFloat(fontUnits) * fontSize / CGFloat(unitsPerEm)
|
||||
}
|
||||
override func constantFromTable(_ constName: String) -> CGFloat {
|
||||
guard let consts = mTable[kConstants] as? NSDictionary, let val = consts[constName] as? NSNumber else {
|
||||
return .zero
|
||||
}
|
||||
return fontUnitsToPt(val.intValue)
|
||||
}
|
||||
override func percentFromTable(_ percentName: String) -> CGFloat {
|
||||
guard let consts = mTable[kConstants] as? NSDictionary, let val = consts[percentName] as? NSNumber else {
|
||||
return .zero
|
||||
}
|
||||
return CGFloat(val.floatValue) / 100
|
||||
}
|
||||
|
||||
}
|
||||
57
Sources/SwiftMath/MathBundle/MTFontV2.swift
Normal file
57
Sources/SwiftMath/MathBundle/MTFontV2.swift
Normal file
@@ -0,0 +1,57 @@
|
||||
//
|
||||
// File.swift
|
||||
//
|
||||
//
|
||||
// Created by Peter Tang on 15/9/2023.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import CoreGraphics
|
||||
import CoreText
|
||||
|
||||
extension MathFont {
|
||||
public func mtfont(size: CGFloat) -> MTFontV2 {
|
||||
MTFontV2(font: self, size: size)
|
||||
}
|
||||
}
|
||||
public final class MTFontV2: MTFont {
|
||||
let font: MathFont
|
||||
let size: CGFloat
|
||||
private lazy var _cgFont: CGFont = {
|
||||
font.cgFont()
|
||||
}()
|
||||
private lazy var _ctFont: CTFont = {
|
||||
font.ctFont(withSize: size)
|
||||
}()
|
||||
private lazy var _mathTab = MTFontMathTableV2(mathFont: font, size: size)
|
||||
init(font: MathFont = .latinModernFont, size: CGFloat) {
|
||||
self.font = font
|
||||
self.size = size
|
||||
|
||||
super.init()
|
||||
|
||||
super.defaultCGFont = nil
|
||||
super.ctFont = nil
|
||||
super.mathTable = nil
|
||||
super.rawMathTable = nil
|
||||
}
|
||||
override var defaultCGFont: CGFont! {
|
||||
set { fatalError("\(#function): change to \(font.fontName) not allowed.") }
|
||||
get { _cgFont }
|
||||
}
|
||||
override var ctFont: CTFont! {
|
||||
set { fatalError("\(#function): change to \(font.fontName) not allowed.") }
|
||||
get { _ctFont }
|
||||
}
|
||||
override var mathTable: MTFontMathTable? {
|
||||
set { fatalError("\(#function): change to \(font.rawValue) not allowed.") }
|
||||
get { _mathTab }
|
||||
}
|
||||
override var rawMathTable: NSDictionary? {
|
||||
set { fatalError("\(#function): change to \(font.rawValue) not allowed.") }
|
||||
get { fatalError("\(#function): access to \(font.rawValue) not allowed.") }
|
||||
}
|
||||
public override func copy(withSize size: CGFloat) -> MTFont {
|
||||
MTFontV2(font: font, size: size)
|
||||
}
|
||||
}
|
||||
@@ -39,10 +39,10 @@ public enum MathFont: String, CaseIterable {
|
||||
case .termesFont: return "TeXGyreTermesMath-Regular"
|
||||
}
|
||||
}
|
||||
public func cgFont() -> CGFont? {
|
||||
public func cgFont() -> CGFont {
|
||||
BundleManager.manager.obtainCGFont(font: self)
|
||||
}
|
||||
public func ctFont(withSize size: CGFloat) -> CTFont? {
|
||||
public func ctFont(withSize size: CGFloat) -> CTFont {
|
||||
BundleManager.manager.obtainCTFont(font: self, withSize: size)
|
||||
}
|
||||
#if os(iOS)
|
||||
@@ -55,16 +55,16 @@ public enum MathFont: String, CaseIterable {
|
||||
NSFont(name: fontName, size: size)
|
||||
}
|
||||
#endif
|
||||
internal func mathTable() -> NSDictionary? {
|
||||
internal func mathTable() -> NSDictionary {
|
||||
BundleManager.manager.obtainMathTable(font: self)
|
||||
}
|
||||
internal func get(nameForGlyph glyph: CGGlyph) -> String {
|
||||
let name = cgFont()?.name(for: glyph) as? String
|
||||
return name ?? ""
|
||||
}
|
||||
internal func get(glyphWithName name: String) -> CGGlyph? {
|
||||
cgFont()?.getGlyphWithGlyphName(name: name as CFString)
|
||||
}
|
||||
// internal func get(nameForGlyph glyph: CGGlyph) -> String {
|
||||
// let name = cgFont().name(for: glyph) as? String
|
||||
// return name ?? ""
|
||||
// }
|
||||
// internal func get(glyphWithName name: String) -> CGGlyph {
|
||||
// cgFont().getGlyphWithGlyphName(name: name as CFString)
|
||||
// }
|
||||
}
|
||||
internal extension CTFont {
|
||||
/** The size of this font in points. */
|
||||
@@ -117,7 +117,6 @@ private class BundleManager {
|
||||
version == "1.3" else {
|
||||
throw FontError.invalidMathTable
|
||||
}
|
||||
//FIXME: mathTable = MTFontMathTable(withFont:self, mathTable:rawMathTable)
|
||||
mathTables[mathFont] = rawMathTable
|
||||
print("mathFonts bundle resource: \(mathFont.rawValue).plist registered.")
|
||||
}
|
||||
@@ -135,12 +134,15 @@ private class BundleManager {
|
||||
initializedOnceAlready.toggle()
|
||||
}
|
||||
|
||||
fileprivate func obtainCGFont(font: MathFont) -> CGFont? {
|
||||
fileprivate func obtainCGFont(font: MathFont) -> CGFont {
|
||||
if !initializedOnceAlready { registerAllBundleResources() }
|
||||
return cgFonts[font]
|
||||
guard let cfFont = cgFonts[font] else {
|
||||
fatalError("\(#function) unable to locate CTFont \(font.fontName)")
|
||||
}
|
||||
return cfFont
|
||||
}
|
||||
|
||||
fileprivate func obtainCTFont(font: MathFont, withSize size: CGFloat) -> CTFont? {
|
||||
fileprivate func obtainCTFont(font: MathFont, withSize size: CGFloat) -> CTFont {
|
||||
if !initializedOnceAlready { registerAllBundleResources() }
|
||||
let fontPair = CTFontPair(font: font, size: size)
|
||||
guard let ctFont = ctFonts[fontPair] else {
|
||||
@@ -149,13 +151,16 @@ private class BundleManager {
|
||||
ctFonts[fontPair] = ctFont
|
||||
return ctFont
|
||||
}
|
||||
return nil
|
||||
fatalError("\(#function) unable to locate CTFont \(font.fontName)")
|
||||
}
|
||||
return ctFont
|
||||
}
|
||||
fileprivate func obtainMathTable(font: MathFont) -> NSDictionary? {
|
||||
fileprivate func obtainMathTable(font: MathFont) -> NSDictionary {
|
||||
if !initializedOnceAlready { registerAllBundleResources() }
|
||||
return mathTables[font]
|
||||
guard let mathTable = mathTables[font] else {
|
||||
fatalError("\(#function) unable to locate mathTable: \(font.rawValue).plist")
|
||||
}
|
||||
return mathTable
|
||||
}
|
||||
deinit {
|
||||
ctFonts.removeAll()
|
||||
|
||||
112
Sources/SwiftMath/MathBundle/MathImage.swift
Normal file
112
Sources/SwiftMath/MathBundle/MathImage.swift
Normal file
@@ -0,0 +1,112 @@
|
||||
//
|
||||
// MTMathImageV2.swift
|
||||
//
|
||||
//
|
||||
// Created by Peter Tang on 15/9/2023.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
import Foundation
|
||||
|
||||
#if os(iOS)
|
||||
import UIKit
|
||||
#endif
|
||||
|
||||
#if os(macOS)
|
||||
import AppKit
|
||||
#endif
|
||||
|
||||
public struct MathImage {
|
||||
public var font: MathFont = .latinModernFont
|
||||
public var fontSize: CGFloat
|
||||
public var textColor: MTColor
|
||||
|
||||
public var labelMode: MTMathUILabelMode
|
||||
public var textAlignment: MTTextAlignment
|
||||
|
||||
public var contentInsets: MTEdgeInsets = MTEdgeInsetsZero
|
||||
|
||||
public let latex: String
|
||||
|
||||
private(set) var intrinsicContentSize = CGSize.zero
|
||||
|
||||
public init(latex: String, fontSize: CGFloat, textColor: MTColor, labelMode: MTMathUILabelMode = .display, textAlignment: MTTextAlignment = .center) {
|
||||
self.latex = latex
|
||||
self.fontSize = fontSize
|
||||
self.textColor = textColor
|
||||
self.labelMode = labelMode
|
||||
self.textAlignment = textAlignment
|
||||
}
|
||||
}
|
||||
extension MathImage {
|
||||
public var currentStyle: MTLineStyle {
|
||||
switch labelMode {
|
||||
case .display: return .display
|
||||
case .text: return .text
|
||||
}
|
||||
}
|
||||
private func intrinsicContentSize(_ displayList: MTMathListDisplay) -> CGSize {
|
||||
CGSize(width: displayList.width + contentInsets.left + contentInsets.right,
|
||||
height: displayList.ascent + displayList.descent + contentInsets.top + contentInsets.bottom)
|
||||
}
|
||||
public mutating func asImage() -> (NSError?, MTImage?) {
|
||||
func layoutImage(size: CGSize, displayList: MTMathListDisplay) {
|
||||
var textX = CGFloat(0)
|
||||
switch self.textAlignment {
|
||||
case .left: textX = contentInsets.left
|
||||
case .center: textX = (size.width - contentInsets.left - contentInsets.right - displayList.width) / 2 + contentInsets.left
|
||||
case .right: textX = size.width - displayList.width - contentInsets.right
|
||||
}
|
||||
let availableHeight = size.height - contentInsets.bottom - contentInsets.top
|
||||
|
||||
// center things vertically
|
||||
var height = displayList.ascent + displayList.descent
|
||||
if height < fontSize/2 {
|
||||
height = fontSize/2 // set height to half the font size
|
||||
}
|
||||
let textY = (availableHeight - height) / 2 + displayList.descent + contentInsets.bottom
|
||||
displayList.position = CGPoint(x: textX, y: textY)
|
||||
}
|
||||
var error: NSError?
|
||||
let mtfont: MTFont? = font.mtfont(size: fontSize)
|
||||
guard let mathList = MTMathListBuilder.build(fromString: latex, error: &error), error == nil,
|
||||
let displayList = MTTypesetter.createLineForMathList(mathList, font: mtfont, style: currentStyle) else {
|
||||
return (error, nil)
|
||||
}
|
||||
|
||||
intrinsicContentSize = intrinsicContentSize(displayList)
|
||||
displayList.textColor = textColor
|
||||
|
||||
let size = intrinsicContentSize
|
||||
layoutImage(size: size, displayList: displayList)
|
||||
|
||||
#if os(iOS)
|
||||
let renderer = UIGraphicsImageRenderer(size: size)
|
||||
let image = renderer.image { rendererContext in
|
||||
rendererContext.cgContext.saveGState()
|
||||
rendererContext.cgContext.concatenate(.flippedVertically(size.height))
|
||||
displayList.draw(rendererContext.cgContext)
|
||||
rendererContext.cgContext.restoreGState()
|
||||
}
|
||||
return (nil, image)
|
||||
#endif
|
||||
#if os(macOS)
|
||||
let image = NSImage(size: size, flipped: false) { bounds in
|
||||
guard let context = NSGraphicsContext.current?.cgContext else { return false }
|
||||
context.saveGState()
|
||||
displayList.draw(context)
|
||||
context.restoreGState()
|
||||
return true
|
||||
}
|
||||
return (nil, image)
|
||||
#endif
|
||||
}
|
||||
}
|
||||
private extension CGAffineTransform {
|
||||
static func flippedVertically(_ height: CGFloat) -> CGAffineTransform {
|
||||
var transform = CGAffineTransform(scaleX: 1, y: -1)
|
||||
transform = transform.translatedBy(x: 0, y: -height)
|
||||
return transform
|
||||
}
|
||||
}
|
||||
35
Tests/Obsolete/MathTableTests.swift
Normal file
35
Tests/Obsolete/MathTableTests.swift
Normal file
@@ -0,0 +1,35 @@
|
||||
import XCTest
|
||||
@testable import SwiftMath
|
||||
|
||||
//
|
||||
// MathTableTests.swift
|
||||
//
|
||||
//
|
||||
// Created by Peter Tang on 12/9/2023.
|
||||
//
|
||||
|
||||
final class MathTableTests: XCTestCase {
|
||||
func testMathFontScript() throws {
|
||||
// let size = Int.random(in: 20 ... 40)
|
||||
// MathFont.allCases.forEach {
|
||||
// // print("\(#function) cgfont \($0.cgFont())")
|
||||
// // print("\(#function) ctfont \($0.ctFont(withSize: CGFloat(size)))")
|
||||
// // XCTAssertNotNil($0.cgFont())
|
||||
// // XCTAssertNotNil($0.ctFont(withSize: CGFloat(size)))
|
||||
// // XCTAssertEqual($0.ctFont(withSize: CGFloat(size))?.fontSize, CGFloat(size), "ctFont fontSize test")
|
||||
// let ctFont = $0.ctFont(withSize: CGFloat(size))
|
||||
// let unitsPerEm = ctFont.unitsPerEm
|
||||
// let mathTable = MathTable(withFont: $0, fontSize: CGFloat(size), unitsPerEm: unitsPerEm)
|
||||
//
|
||||
// let values = [
|
||||
// mathTable.fractionNumeratorDisplayStyleShiftUp,
|
||||
// mathTable.fractionNumeratorShiftUp,
|
||||
// mathTable.fractionDenominatorDisplayStyleShiftDown,
|
||||
// mathTable.fractionDenominatorShiftDown,
|
||||
// mathTable.fractionNumeratorDisplayStyleGapMin,
|
||||
// mathTable.fractionNumeratorGapMin,
|
||||
// ]
|
||||
// print("\(ctFont) -> \(values)")
|
||||
// }
|
||||
}
|
||||
}
|
||||
28
Tests/SwiftMathTests/MTFontMathTableV2Tests.swift
Normal file
28
Tests/SwiftMathTests/MTFontMathTableV2Tests.swift
Normal file
@@ -0,0 +1,28 @@
|
||||
//
|
||||
// MTFontMathTableV2Tests.swift
|
||||
//
|
||||
//
|
||||
// Created by Peter Tang on 15/9/2023.
|
||||
//
|
||||
|
||||
import XCTest
|
||||
@testable import SwiftMath
|
||||
|
||||
final class MTFontMathTableV2Tests: XCTestCase {
|
||||
func testMTFontV2Script() throws {
|
||||
let size = CGFloat(Int.random(in: 20 ... 40))
|
||||
MathFont.allCases.forEach {
|
||||
let mTable = $0.mtfont(size: size).mathTable
|
||||
XCTAssertNotNil(mTable)
|
||||
let values = [
|
||||
mTable?.fractionNumeratorDisplayStyleShiftUp,
|
||||
mTable?.fractionNumeratorShiftUp,
|
||||
mTable?.fractionDenominatorDisplayStyleShiftDown,
|
||||
mTable?.fractionDenominatorShiftDown,
|
||||
mTable?.fractionNumeratorDisplayStyleGapMin,
|
||||
mTable?.fractionNumeratorGapMin,
|
||||
].compactMap{$0}
|
||||
print("\($0.rawValue).plist: \(values)")
|
||||
}
|
||||
}
|
||||
}
|
||||
21
Tests/SwiftMathTests/MTFontV2Tests.swift
Normal file
21
Tests/SwiftMathTests/MTFontV2Tests.swift
Normal file
@@ -0,0 +1,21 @@
|
||||
//
|
||||
// MTFontV2Tests.swift
|
||||
//
|
||||
//
|
||||
// Created by Peter Tang on 15/9/2023.
|
||||
//
|
||||
|
||||
import XCTest
|
||||
@testable import SwiftMath
|
||||
|
||||
final class MTFontV2Tests: XCTestCase {
|
||||
func testMTFontV2Script() throws {
|
||||
let size = CGFloat(Int.random(in: 20 ... 40))
|
||||
MathFont.allCases.forEach {
|
||||
let mtfont = $0.mtfont(size: size)
|
||||
let mTable = mtfont.mathTable?._mathTable
|
||||
XCTAssertNotNil(mtfont)
|
||||
XCTAssertNotNil(mTable)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -16,7 +16,7 @@ final class MathFontTests: XCTestCase {
|
||||
// print("\(#function) ctfont \($0.ctFont(withSize: CGFloat(size)))")
|
||||
XCTAssertNotNil($0.cgFont())
|
||||
XCTAssertNotNil($0.ctFont(withSize: CGFloat(size)))
|
||||
XCTAssertEqual($0.ctFont(withSize: CGFloat(size))?.fontSize, CGFloat(size), "ctFont fontSize test")
|
||||
XCTAssertEqual($0.ctFont(withSize: CGFloat(size)).fontSize, CGFloat(size), "ctFont fontSize test")
|
||||
}
|
||||
#if os(iOS)
|
||||
// for family in UIFont.familyNames.sorted() {
|
||||
|
||||
@@ -1,36 +0,0 @@
|
||||
import XCTest
|
||||
@testable import SwiftMath
|
||||
|
||||
//
|
||||
// MathTableTests.swift
|
||||
//
|
||||
//
|
||||
// Created by Peter Tang on 12/9/2023.
|
||||
//
|
||||
|
||||
final class MathTableTests: XCTestCase {
|
||||
func testMathFontScript() throws {
|
||||
let size = Int.random(in: 20 ... 40)
|
||||
MathFont.allCases.forEach {
|
||||
// print("\(#function) cgfont \($0.cgFont())")
|
||||
// print("\(#function) ctfont \($0.ctFont(withSize: CGFloat(size)))")
|
||||
// XCTAssertNotNil($0.cgFont())
|
||||
// XCTAssertNotNil($0.ctFont(withSize: CGFloat(size)))
|
||||
// XCTAssertEqual($0.ctFont(withSize: CGFloat(size))?.fontSize, CGFloat(size), "ctFont fontSize test")
|
||||
let ctFont = $0.ctFont(withSize: CGFloat(size))
|
||||
if let unitsPerEm = ctFont?.unitsPerEm {
|
||||
let mathTable = MathTable(withFont: $0, fontSize: CGFloat(size), unitsPerEm: unitsPerEm)
|
||||
|
||||
let values = [
|
||||
mathTable.fractionNumeratorDisplayStyleShiftUp,
|
||||
mathTable.fractionNumeratorShiftUp,
|
||||
mathTable.fractionDenominatorDisplayStyleShiftDown,
|
||||
mathTable.fractionDenominatorShiftDown,
|
||||
mathTable.fractionNumeratorDisplayStyleGapMin,
|
||||
mathTable.fractionNumeratorGapMin,
|
||||
]
|
||||
print("\(ctFont) -> \(values)")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user