Compare commits
5 Commits
1.0.5
...
57ce0dc87a
| Author | SHA1 | Date | |
|---|---|---|---|
| 57ce0dc87a | |||
|
|
d06a66226e | ||
| 336b47ace2 | |||
| b4a081c4d2 | |||
|
|
57696e3036 |
@@ -1,10 +1,10 @@
|
|||||||
{
|
{
|
||||||
"originHash" : "57d4d1724511f49da67a759e590ddf12887d65ffb0702ad7a2ff8f6c830c9b78",
|
"originHash" : "33e7d52ad13cf774717778548edb365d33ff62d766d0049165bc8970f19a23ef",
|
||||||
"pins" : [
|
"pins" : [
|
||||||
{
|
{
|
||||||
"identity" : "lcecryptokitbinary",
|
"identity" : "lcecryptokitbinary",
|
||||||
"kind" : "remoteSourceControl",
|
"kind" : "localSourceControl",
|
||||||
"location" : "https://60c260c85d3a2fe840411b0ff98f521b5eca3c56@git.loverde.com.br/Loverde-Company-LTDA/LCECryptoKitBinary.git",
|
"location" : "/Users/loverde_co/Documents/Loverde_JOBs/Producao/Loverde Co/GIT/XCODE/Repositorios/LCECryptoKit/PrivateLib/LCECryptoKitBinary",
|
||||||
"state" : {
|
"state" : {
|
||||||
"revision" : "2c5c47cebef40a8adc5557d071a35be405c05e30",
|
"revision" : "2c5c47cebef40a8adc5557d071a35be405c05e30",
|
||||||
"version" : "1.0.2"
|
"version" : "1.0.2"
|
||||||
|
|||||||
@@ -3,6 +3,23 @@ import PackageDescription
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
let isLocalDevelopment = FileManager.default.fileExists(atPath: "../LCECryptoKit/PrivateLib/LCECryptoKitBinary")
|
let isLocalDevelopment = FileManager.default.fileExists(atPath: "../LCECryptoKit/PrivateLib/LCECryptoKitBinary")
|
||||||
|
let enableCryptoBinary = ProcessInfo.processInfo.environment["LCE_ENABLE_CRYPTO_BINARY"] != "0"
|
||||||
|
|
||||||
|
let cryptoPackageURL = isLocalDevelopment
|
||||||
|
? "../LCECryptoKit/PrivateLib/LCECryptoKitBinary"
|
||||||
|
: "https://60c260c85d3a2fe840411b0ff98f521b5eca3c56@git.loverde.com.br/Loverde-Company-LTDA/LCECryptoKitBinary.git"
|
||||||
|
|
||||||
|
let packageDependencies: [Package.Dependency] = enableCryptoBinary
|
||||||
|
? [
|
||||||
|
.package(url: cryptoPackageURL, exact: "1.0.2")
|
||||||
|
]
|
||||||
|
: []
|
||||||
|
|
||||||
|
let targetDependencies: [Target.Dependency] = enableCryptoBinary
|
||||||
|
? [
|
||||||
|
.product(name: "LCECryptoKit", package: "lcecryptokitbinary")
|
||||||
|
]
|
||||||
|
: []
|
||||||
|
|
||||||
let package = Package(
|
let package = Package(
|
||||||
name: "LCEssentials",
|
name: "LCEssentials",
|
||||||
@@ -17,17 +34,10 @@ let package = Package(
|
|||||||
name: "LCEssentials",
|
name: "LCEssentials",
|
||||||
targets: ["LCEssentials"]),
|
targets: ["LCEssentials"]),
|
||||||
],
|
],
|
||||||
dependencies: [
|
dependencies: packageDependencies,
|
||||||
.package(
|
|
||||||
url: isLocalDevelopment ?
|
|
||||||
"../LCECryptoKit/PrivateLib/LCECryptoKitBinary" :
|
|
||||||
"https://60c260c85d3a2fe840411b0ff98f521b5eca3c56@git.loverde.com.br/Loverde-Company-LTDA/LCECryptoKitBinary.git",
|
|
||||||
exact: "1.0.2"
|
|
||||||
)
|
|
||||||
],
|
|
||||||
targets: [
|
targets: [
|
||||||
.target(
|
.target(
|
||||||
name: "LCEssentials",
|
name: "LCEssentials",
|
||||||
dependencies: [.product(name: "LCECryptoKit", package: "lcecryptokitbinary")]),
|
dependencies: targetDependencies),
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -5,8 +5,10 @@
|
|||||||
// Created by iOSDevCenters on 11/12/15.
|
// Created by iOSDevCenters on 11/12/15.
|
||||||
// Copyright © 2016 iOSDevCenters. All rights reserved.
|
// Copyright © 2016 iOSDevCenters. All rights reserved.
|
||||||
//
|
//
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
#if os(iOS) || os(macOS)
|
#endif
|
||||||
|
#if canImport(UIKit)
|
||||||
import ImageIO
|
import ImageIO
|
||||||
|
|
||||||
//let jeremyGif = UIImage.gifWithName("jeremy")
|
//let jeremyGif = UIImage.gifWithName("jeremy")
|
||||||
|
|||||||
@@ -21,6 +21,8 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
|
#if canImport(LCECryptoKit)
|
||||||
import LCECryptoKit
|
import LCECryptoKit
|
||||||
|
|
||||||
public final class LCECryptoKitManager {
|
public final class LCECryptoKitManager {
|
||||||
@@ -57,3 +59,38 @@ public final class LCECryptoKitManager {
|
|||||||
LCECryptoKit.decodeSeed(otpKey: otpHash, hashKey: self.hashKey)
|
LCECryptoKit.decodeSeed(otpKey: otpHash, hashKey: self.hashKey)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
|
||||||
|
public final class LCECryptoKitManager {
|
||||||
|
|
||||||
|
private let hashKey: String
|
||||||
|
|
||||||
|
public init() {
|
||||||
|
self.hashKey = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
public init(privateKey: String){
|
||||||
|
self.hashKey = privateKey
|
||||||
|
}
|
||||||
|
|
||||||
|
public static func generateKey() -> String {
|
||||||
|
""
|
||||||
|
}
|
||||||
|
|
||||||
|
public func encodeTP(email: String, password: String) -> String? {
|
||||||
|
nil
|
||||||
|
}
|
||||||
|
|
||||||
|
public func decodeOTP(_ otpHash: String) -> String? {
|
||||||
|
nil
|
||||||
|
}
|
||||||
|
|
||||||
|
public func encodeOTPWithKey(email: String, password: String) -> String? {
|
||||||
|
nil
|
||||||
|
}
|
||||||
|
|
||||||
|
public func decodeOTPWithKey(_ otpHash: String) -> Bool {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|||||||
@@ -26,8 +26,10 @@ import Foundation
|
|||||||
import Security
|
import Security
|
||||||
#endif
|
#endif
|
||||||
#if canImport(UIKit)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
#endif
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
/// A generic `Result` enumeration to represent either a success `Value` or a failure `Error`.
|
/// A generic `Result` enumeration to represent either a success `Value` or a failure `Error`.
|
||||||
public enum Result<Value, Error: Swift.Error> {
|
public enum Result<Value, Error: Swift.Error> {
|
||||||
|
|||||||
@@ -21,8 +21,12 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
#if canImport(AVFoundation)
|
||||||
import AVFoundation
|
import AVFoundation
|
||||||
|
#endif
|
||||||
#if os(watchOS)
|
#if os(watchOS)
|
||||||
import WatchKit
|
import WatchKit
|
||||||
#endif
|
#endif
|
||||||
@@ -155,7 +159,7 @@ public struct LCEssentials {
|
|||||||
diskPath: "file_cache"
|
diskPath: "file_cache"
|
||||||
)
|
)
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
/// Extract the file name from the file path
|
/// Extract the file name from the file path
|
||||||
///
|
///
|
||||||
/// - Parameter filePath: Full file path in bundle
|
/// - Parameter filePath: Full file path in bundle
|
||||||
@@ -266,7 +270,11 @@ public struct LCEssentials {
|
|||||||
/// - LoverdeCo: Check if app is running in debug mode.
|
/// - LoverdeCo: Check if app is running in debug mode.
|
||||||
@MainActor
|
@MainActor
|
||||||
public static var isInDebuggingMode: Bool {
|
public static var isInDebuggingMode: Bool {
|
||||||
|
#if canImport(UIKit)
|
||||||
return UIApplication.inferredEnvironment == .debug
|
return UIApplication.inferredEnvironment == .debug
|
||||||
|
#else
|
||||||
|
return false
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
#if !os(macOS)
|
#if !os(macOS)
|
||||||
@@ -416,7 +424,7 @@ public struct LCEssentials {
|
|||||||
// MARK: - Methods
|
// MARK: - Methods
|
||||||
public extension LCEssentials {
|
public extension LCEssentials {
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
/// - LoverdeCo: Share link with message
|
/// - LoverdeCo: Share link with message
|
||||||
///
|
///
|
||||||
/// - Parameters:
|
/// - Parameters:
|
||||||
|
|||||||
@@ -20,7 +20,9 @@
|
|||||||
// THE SOFTWARE.
|
// THE SOFTWARE.
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
@objc public protocol LCESingletonDelegate: AnyObject {
|
@objc public protocol LCESingletonDelegate: AnyObject {
|
||||||
@objc optional func singleton(object: Any?, withData: Any)
|
@objc optional func singleton(object: Any?, withData: Any)
|
||||||
|
|||||||
@@ -21,7 +21,9 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
// MARK: - Methods (Equatable)
|
// MARK: - Methods (Equatable)
|
||||||
public extension Array where Element: Equatable {
|
public extension Array where Element: Equatable {
|
||||||
@@ -148,6 +150,7 @@ public extension Array where Element: Equatable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if canImport(UIKit)
|
||||||
extension Array where Element == NSLayoutConstraint {
|
extension Array where Element == NSLayoutConstraint {
|
||||||
|
|
||||||
@MainActor
|
@MainActor
|
||||||
@@ -169,3 +172,5 @@ extension Array where Element == NSLayoutConstraint {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|||||||
@@ -21,8 +21,12 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(CryptoKit)
|
||||||
import CryptoKit
|
import CryptoKit
|
||||||
|
#endif
|
||||||
|
#if canImport(CommonCrypto)
|
||||||
import CommonCrypto
|
import CommonCrypto
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
public extension Data {
|
public extension Data {
|
||||||
@@ -71,17 +75,25 @@ public extension Data {
|
|||||||
|
|
||||||
@available(iOS 13.0, *)
|
@available(iOS 13.0, *)
|
||||||
func HMACSHA512(key: Data) -> Data {
|
func HMACSHA512(key: Data) -> Data {
|
||||||
|
#if canImport(CryptoKit)
|
||||||
var hmac = HMAC<SHA512>.init(key: SymmetricKey(data: key))
|
var hmac = HMAC<SHA512>.init(key: SymmetricKey(data: key))
|
||||||
hmac.update(data: self)
|
hmac.update(data: self)
|
||||||
return Data(hmac.finalize())
|
return Data(hmac.finalize())
|
||||||
|
#else
|
||||||
|
return Data()
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
func SHA512() -> Data {
|
func SHA512() -> Data {
|
||||||
|
#if canImport(CommonCrypto)
|
||||||
var digest = [UInt8](repeating: 0, count: Int(CC_SHA512_DIGEST_LENGTH))
|
var digest = [UInt8](repeating: 0, count: Int(CC_SHA512_DIGEST_LENGTH))
|
||||||
self.withUnsafeBytes {
|
self.withUnsafeBytes {
|
||||||
_ = CC_SHA512($0.baseAddress, CC_LONG(self.count), &digest)
|
_ = CC_SHA512($0.baseAddress, CC_LONG(self.count), &digest)
|
||||||
}
|
}
|
||||||
return Data(digest)
|
return Data(digest)
|
||||||
|
#else
|
||||||
|
return Data()
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
func XOR(with other: Data) -> Data {
|
func XOR(with other: Data) -> Data {
|
||||||
@@ -89,11 +101,15 @@ public extension Data {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func SHA256() -> Data {
|
func SHA256() -> Data {
|
||||||
|
#if canImport(CommonCrypto)
|
||||||
var digest = [UInt8](repeating: 0, count: Int(CC_SHA256_DIGEST_LENGTH))
|
var digest = [UInt8](repeating: 0, count: Int(CC_SHA256_DIGEST_LENGTH))
|
||||||
self.withUnsafeBytes {
|
self.withUnsafeBytes {
|
||||||
_ = CC_SHA256($0.baseAddress, CC_LONG(self.count), &digest)
|
_ = CC_SHA256($0.baseAddress, CC_LONG(self.count), &digest)
|
||||||
}
|
}
|
||||||
return Data(digest)
|
return Data(digest)
|
||||||
|
#else
|
||||||
|
return Data()
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
func object<T: Codable>() -> T? {
|
func object<T: Codable>() -> T? {
|
||||||
@@ -124,9 +140,13 @@ public extension Data {
|
|||||||
///print("md5Hex: \(md5Hex)")
|
///print("md5Hex: \(md5Hex)")
|
||||||
@available(iOS 13.0, *)
|
@available(iOS 13.0, *)
|
||||||
static func MD5(string: String) -> Data {
|
static func MD5(string: String) -> Data {
|
||||||
|
#if canImport(CryptoKit)
|
||||||
let messageData = string.data(using: .utf8)!
|
let messageData = string.data(using: .utf8)!
|
||||||
let digestData = Insecure.MD5.hash (data: messageData)
|
let digestData = Insecure.MD5.hash (data: messageData)
|
||||||
let digestHex = String(digestData.map { String(format: "%02hhx", $0) }.joined().prefix(32))
|
let digestHex = String(digestData.map { String(format: "%02hhx", $0) }.joined().prefix(32))
|
||||||
return Data(digestHex.utf8)
|
return Data(digestHex.utf8)
|
||||||
|
#else
|
||||||
|
return Data()
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,7 +21,9 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension Dictionary {
|
public extension Dictionary {
|
||||||
|
|
||||||
|
|||||||
@@ -21,7 +21,9 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension FileManager {
|
public extension FileManager {
|
||||||
|
|
||||||
@@ -54,7 +56,7 @@ public extension FileManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
func saveImageToDirectory( _ imageWithPath : String, imagem : UIImage ) -> Bool {
|
func saveImageToDirectory( _ imageWithPath : String, imagem : UIImage ) -> Bool {
|
||||||
|
|
||||||
let data = imagem.pngData()
|
let data = imagem.pngData()
|
||||||
|
|||||||
@@ -21,8 +21,11 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if canImport(UIKit)
|
||||||
extension NSLayoutConstraint {
|
extension NSLayoutConstraint {
|
||||||
|
|
||||||
func constraintWithMultiplier(_ multiplier: CGFloat) -> NSLayoutConstraint {
|
func constraintWithMultiplier(_ multiplier: CGFloat) -> NSLayoutConstraint {
|
||||||
@@ -72,3 +75,4 @@ extension NSLayoutConstraint {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|||||||
@@ -21,8 +21,10 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension NSMutableAttributedString {
|
public extension NSMutableAttributedString {
|
||||||
@discardableResult func customize(_ text: String,
|
@discardableResult func customize(_ text: String,
|
||||||
|
|||||||
@@ -20,7 +20,9 @@
|
|||||||
// THE SOFTWARE.
|
// THE SOFTWARE.
|
||||||
|
|
||||||
|
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension Optional {
|
public extension Optional {
|
||||||
/// Get self of default value (if self is nil).
|
/// Get self of default value (if self is nil).
|
||||||
|
|||||||
@@ -21,7 +21,9 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
#if canImport(CommonCrypto)
|
#if canImport(CommonCrypto)
|
||||||
import CommonCrypto
|
import CommonCrypto
|
||||||
#endif
|
#endif
|
||||||
@@ -48,10 +50,14 @@ public extension String {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var convertToHTML: NSAttributedString? {
|
var convertToHTML: NSAttributedString? {
|
||||||
|
#if canImport(UIKit)
|
||||||
return convertHtmlToAttributedStringWithCSS(font: nil,
|
return convertHtmlToAttributedStringWithCSS(font: nil,
|
||||||
csscolor: "",
|
csscolor: "",
|
||||||
lineheight: 0,
|
lineheight: 0,
|
||||||
csstextalign: "")
|
csstextalign: "")
|
||||||
|
#else
|
||||||
|
return convertHtmlToNSAttributedString
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Check if string is a valid URL.
|
/// Check if string is a valid URL.
|
||||||
@@ -646,7 +652,7 @@ public extension String {
|
|||||||
return strDate?.date(withCurrFormatt: newFormatt, localeIdentifier: localeIdentifier, timeZone: timeZone)
|
return strDate?.date(withCurrFormatt: newFormatt, localeIdentifier: localeIdentifier, timeZone: timeZone)
|
||||||
}
|
}
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
func height(withConstrainedWidth width: CGFloat, font: UIFont) -> CGFloat {
|
func height(withConstrainedWidth width: CGFloat, font: UIFont) -> CGFloat {
|
||||||
let constraintRect = CGSize(width: width, height: .greatestFiniteMagnitude)
|
let constraintRect = CGSize(width: width, height: .greatestFiniteMagnitude)
|
||||||
let boundingBox = self.boundingRect(with: constraintRect, options: .usesLineFragmentOrigin, attributes: [NSAttributedString.Key.font: font], context: nil)
|
let boundingBox = self.boundingRect(with: constraintRect, options: .usesLineFragmentOrigin, attributes: [NSAttributedString.Key.font: font], context: nil)
|
||||||
@@ -757,7 +763,7 @@ public extension String {
|
|||||||
}
|
}
|
||||||
return self
|
return self
|
||||||
}
|
}
|
||||||
|
#if canImport(UIKit)
|
||||||
/// Converte String para HTML com CSS.
|
/// Converte String para HTML com CSS.
|
||||||
///
|
///
|
||||||
/// - Parameters:
|
/// - Parameters:
|
||||||
@@ -801,6 +807,7 @@ public extension String {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/// Float value from string (if applicable).
|
/// Float value from string (if applicable).
|
||||||
///
|
///
|
||||||
|
|||||||
@@ -19,8 +19,10 @@
|
|||||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
// THE SOFTWARE.
|
// THE SOFTWARE.
|
||||||
|
|
||||||
#if canImport(UIKit) && os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension UIApplication {
|
public extension UIApplication {
|
||||||
/// Application running environment.
|
/// Application running environment.
|
||||||
|
|||||||
@@ -21,9 +21,11 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
// MARK: - Properties
|
// MARK: - Properties
|
||||||
public extension UIButton {
|
public extension UIButton {
|
||||||
|
|
||||||
|
|||||||
@@ -21,8 +21,10 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension UICollectionView {
|
public extension UICollectionView {
|
||||||
|
|
||||||
|
|||||||
@@ -21,9 +21,11 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
public extension UIColor {
|
public extension UIColor {
|
||||||
|
|
||||||
var redValue: CGFloat{ return CIColor(color: self).red }
|
var redValue: CGFloat{ return CIColor(color: self).red }
|
||||||
|
|||||||
@@ -21,9 +21,11 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
public extension UIDevice {
|
public extension UIDevice {
|
||||||
|
|
||||||
static var topNotch: CGFloat {
|
static var topNotch: CGFloat {
|
||||||
@@ -64,59 +66,104 @@ public extension UIDevice {
|
|||||||
|
|
||||||
#if os(iOS)
|
#if os(iOS)
|
||||||
switch identifier {
|
switch identifier {
|
||||||
case "iPod5,1": return "iPod touch (5th generation)"
|
case "iPod5,1": return "iPod touch (5th generation)"
|
||||||
case "iPod7,1": return "iPod touch (6th generation)"
|
case "iPod7,1": return "iPod touch (6th generation)"
|
||||||
case "iPod9,1": return "iPod touch (7th generation)"
|
case "iPod9,1": return "iPod touch (7th generation)"
|
||||||
case "iPhone3,1", "iPhone3,2", "iPhone3,3": return "iPhone 4"
|
case "iPhone3,1", "iPhone3,2", "iPhone3,3": return "iPhone 4"
|
||||||
case "iPhone4,1": return "iPhone 4s"
|
case "iPhone4,1": return "iPhone 4s"
|
||||||
case "iPhone5,1", "iPhone5,2": return "iPhone 5"
|
case "iPhone5,1", "iPhone5,2": return "iPhone 5"
|
||||||
case "iPhone5,3", "iPhone5,4": return "iPhone 5c"
|
case "iPhone5,3", "iPhone5,4": return "iPhone 5c"
|
||||||
case "iPhone6,1", "iPhone6,2": return "iPhone 5s"
|
case "iPhone6,1", "iPhone6,2": return "iPhone 5s"
|
||||||
case "iPhone7,2": return "iPhone 6"
|
case "iPhone7,2": return "iPhone 6"
|
||||||
case "iPhone7,1": return "iPhone 6 Plus"
|
case "iPhone7,1": return "iPhone 6 Plus"
|
||||||
case "iPhone8,1": return "iPhone 6s"
|
case "iPhone8,1": return "iPhone 6s"
|
||||||
case "iPhone8,2": return "iPhone 6s Plus"
|
case "iPhone8,2": return "iPhone 6s Plus"
|
||||||
case "iPhone9,1", "iPhone9,3": return "iPhone 7"
|
case "iPhone9,1", "iPhone9,3": return "iPhone 7"
|
||||||
case "iPhone9,2", "iPhone9,4": return "iPhone 7 Plus"
|
case "iPhone9,2", "iPhone9,4": return "iPhone 7 Plus"
|
||||||
case "iPhone8,4", "iPhone12,8": return "iPhone SE"
|
case "iPhone10,1", "iPhone10,4": return "iPhone 8"
|
||||||
case "iPhone10,1", "iPhone10,4": return "iPhone 8"
|
case "iPhone10,2", "iPhone10,5": return "iPhone 8 Plus"
|
||||||
case "iPhone10,2", "iPhone10,5": return "iPhone 8 Plus"
|
case "iPhone10,3", "iPhone10,6": return "iPhone X"
|
||||||
case "iPhone10,3", "iPhone10,6": return "iPhone X"
|
case "iPhone11,2": return "iPhone XS"
|
||||||
case "iPhone11,2": return "iPhone XS"
|
case "iPhone11,4", "iPhone11,6": return "iPhone XS Max"
|
||||||
case "iPhone11,4", "iPhone11,6": return "iPhone XS Max"
|
case "iPhone11,8": return "iPhone XR"
|
||||||
case "iPhone11,8": return "iPhone XR"
|
case "iPhone12,1": return "iPhone 11"
|
||||||
case "iPhone12,1": return "iPhone 11"
|
case "iPhone12,3": return "iPhone 11 Pro"
|
||||||
case "iPhone12,3": return "iPhone 11 Pro"
|
case "iPhone12,5": return "iPhone 11 Pro Max"
|
||||||
case "iPhone12,5": return "iPhone 11 Pro Max"
|
case "iPhone13,1": return "iPhone 12 mini"
|
||||||
case "iPhone13,1": return "iPhone 12 mini"
|
case "iPhone13,2": return "iPhone 12"
|
||||||
case "iPhone13,2": return "iPhone 12"
|
case "iPhone13,3": return "iPhone 12 Pro"
|
||||||
case "iPhone13,3": return "iPhone 12 Pro"
|
case "iPhone13,4": return "iPhone 12 Pro Max"
|
||||||
case "iPhone13,4": return "iPhone 12 Pro Max"
|
case "iPhone14,4": return "iPhone 13 mini"
|
||||||
case "iPad2,1", "iPad2,2", "iPad2,3", "iPad2,4":return "iPad 2"
|
case "iPhone14,5": return "iPhone 13"
|
||||||
case "iPad3,1", "iPad3,2", "iPad3,3": return "iPad 3"
|
case "iPhone14,2": return "iPhone 13 Pro"
|
||||||
case "iPad3,4", "iPad3,5", "iPad3,6": return "iPad 4"
|
case "iPhone14,3": return "iPhone 13 Pro Max"
|
||||||
case "iPad4,1", "iPad4,2", "iPad4,3": return "iPad Air"
|
case "iPhone14,7": return "iPhone 14"
|
||||||
case "iPad5,3", "iPad5,4": return "iPad Air 2"
|
case "iPhone14,8": return "iPhone 14 Plus"
|
||||||
case "iPad6,11", "iPad6,12": return "iPad 5"
|
case "iPhone15,2": return "iPhone 14 Pro"
|
||||||
case "iPad7,5", "iPad7,6": return "iPad 6"
|
case "iPhone15,3": return "iPhone 14 Pro Max"
|
||||||
case "iPad7,11", "iPad7,12": return "iPad 7"
|
case "iPhone15,4": return "iPhone 15"
|
||||||
case "iPad11,4", "iPad11,5": return "iPad Air (3rd generation)"
|
case "iPhone15,5": return "iPhone 15 Plus"
|
||||||
case "iPad2,5", "iPad2,6", "iPad2,7": return "iPad Mini"
|
case "iPhone16,1": return "iPhone 15 Pro"
|
||||||
case "iPad4,4", "iPad4,5", "iPad4,6": return "iPad Mini 2"
|
case "iPhone16,2": return "iPhone 15 Pro Max"
|
||||||
case "iPad4,7", "iPad4,8", "iPad4,9": return "iPad Mini 3"
|
case "iPhone17,3": return "iPhone 16"
|
||||||
case "iPad5,1", "iPad5,2": return "iPad Mini 4"
|
case "iPhone17,4": return "iPhone 16 Plus"
|
||||||
case "iPad11,1", "iPad11,2": return "iPad Mini 5"
|
case "iPhone17,1": return "iPhone 16 Pro"
|
||||||
case "iPad6,3", "iPad6,4": return "iPad Pro (9.7-inch)"
|
case "iPhone17,2": return "iPhone 16 Pro Max"
|
||||||
case "iPad6,7", "iPad6,8": return "iPad Pro (12.9-inch)"
|
case "iPhone17,5": return "iPhone 16e"
|
||||||
case "iPad7,1", "iPad7,2": return "iPad Pro (12.9-inch) (2nd generation)"
|
case "iPhone18,3": return "iPhone 17"
|
||||||
case "iPad7,3", "iPad7,4": return "iPad Pro (10.5-inch)"
|
case "iPhone18,4": return "iPhone Air"
|
||||||
case "iPad8,1", "iPad8,2", "iPad8,3", "iPad8,4":return "iPad Pro (11-inch)"
|
case "iPhone18,1": return "iPhone 17 Pro"
|
||||||
case "iPad8,5", "iPad8,6", "iPad8,7", "iPad8,8":return "iPad Pro (12.9-inch) (3rd generation)"
|
case "iPhone18,2": return "iPhone 17 Pro Max"
|
||||||
case "AppleTV5,3": return "Apple TV"
|
case "iPhone8,4": return "iPhone SE"
|
||||||
case "AppleTV6,2": return "Apple TV 4K"
|
case "iPhone12,8": return "iPhone SE (2nd generation)"
|
||||||
case "AudioAccessory1,1": return "HomePod"
|
case "iPhone14,6": return "iPhone SE (3rd generation)"
|
||||||
case "i386", "x86_64": return "Simulator \(identifier)"
|
case "iPad2,1", "iPad2,2", "iPad2,3", "iPad2,4": return "iPad 2"
|
||||||
default: return identifier
|
case "iPad3,1", "iPad3,2", "iPad3,3": return "iPad (3rd generation)"
|
||||||
|
case "iPad3,4", "iPad3,5", "iPad3,6": return "iPad (4th generation)"
|
||||||
|
case "iPad6,11", "iPad6,12": return "iPad (5th generation)"
|
||||||
|
case "iPad7,5", "iPad7,6": return "iPad (6th generation)"
|
||||||
|
case "iPad7,11", "iPad7,12": return "iPad (7th generation)"
|
||||||
|
case "iPad11,6", "iPad11,7": return "iPad (8th generation)"
|
||||||
|
case "iPad12,1", "iPad12,2": return "iPad (9th generation)"
|
||||||
|
case "iPad13,18", "iPad13,19": return "iPad (10th generation)"
|
||||||
|
case "iPad15,7", "iPad15,8": return "iPad (11th generation)"
|
||||||
|
case "iPad4,1", "iPad4,2", "iPad4,3": return "iPad Air"
|
||||||
|
case "iPad5,3", "iPad5,4": return "iPad Air 2"
|
||||||
|
case "iPad11,3", "iPad11,4": return "iPad Air (3rd generation)"
|
||||||
|
case "iPad13,1", "iPad13,2": return "iPad Air (4th generation)"
|
||||||
|
case "iPad13,16", "iPad13,17": return "iPad Air (5th generation)"
|
||||||
|
case "iPad14,8", "iPad14,9": return "iPad Air (11-inch) (M2)"
|
||||||
|
case "iPad14,10", "iPad14,11": return "iPad Air (13-inch) (M2)"
|
||||||
|
case "iPad15,3", "iPad15,4": return "iPad Air (11-inch) (M3)"
|
||||||
|
case "iPad15,5", "iPad15,6": return "iPad Air (13-inch) (M3)"
|
||||||
|
case "iPad2,5", "iPad2,6", "iPad2,7": return "iPad mini"
|
||||||
|
case "iPad4,4", "iPad4,5", "iPad4,6": return "iPad mini 2"
|
||||||
|
case "iPad4,7", "iPad4,8", "iPad4,9": return "iPad mini 3"
|
||||||
|
case "iPad5,1", "iPad5,2": return "iPad mini 4"
|
||||||
|
case "iPad11,1", "iPad11,2": return "iPad mini (5th generation)"
|
||||||
|
case "iPad14,1", "iPad14,2": return "iPad mini (6th generation)"
|
||||||
|
case "iPad16,1", "iPad16,2": return "iPad mini (A17 Pro)"
|
||||||
|
case "iPad6,3", "iPad6,4": return "iPad Pro (9.7-inch)"
|
||||||
|
case "iPad7,3", "iPad7,4": return "iPad Pro (10.5-inch)"
|
||||||
|
case "iPad8,1", "iPad8,2", "iPad8,3", "iPad8,4": return "iPad Pro (11-inch) (1st generation)"
|
||||||
|
case "iPad8,9", "iPad8,10": return "iPad Pro (11-inch) (2nd generation)"
|
||||||
|
case "iPad13,4", "iPad13,5", "iPad13,6", "iPad13,7": return "iPad Pro (11-inch) (3rd generation)"
|
||||||
|
case "iPad14,3", "iPad14,4": return "iPad Pro (11-inch) (4th generation)"
|
||||||
|
case "iPad16,3", "iPad16,4": return "iPad Pro (11-inch) (M4)"
|
||||||
|
case "iPad17,1", "iPad17,2": return "iPad Pro (11-inch) (M5)"
|
||||||
|
case "iPad6,7", "iPad6,8": return "iPad Pro (12.9-inch) (1st generation)"
|
||||||
|
case "iPad7,1", "iPad7,2": return "iPad Pro (12.9-inch) (2nd generation)"
|
||||||
|
case "iPad8,5", "iPad8,6", "iPad8,7", "iPad8,8": return "iPad Pro (12.9-inch) (3rd generation)"
|
||||||
|
case "iPad8,11", "iPad8,12": return "iPad Pro (12.9-inch) (4th generation)"
|
||||||
|
case "iPad13,8", "iPad13,9", "iPad13,10", "iPad13,11":return "iPad Pro (12.9-inch) (5th generation)"
|
||||||
|
case "iPad14,5", "iPad14,6": return "iPad Pro (12.9-inch) (6th generation)"
|
||||||
|
case "iPad16,5", "iPad16,6": return "iPad Pro (13-inch) (M4)"
|
||||||
|
case "iPad17,3", "iPad17,4": return "iPad Pro (13-inch) (M5)"
|
||||||
|
case "AppleTV5,3": return "Apple TV"
|
||||||
|
case "AppleTV6,2": return "Apple TV 4K"
|
||||||
|
case "AudioAccessory1,1": return "HomePod"
|
||||||
|
case "AudioAccessory5,1": return "HomePod mini"
|
||||||
|
default: return identifier
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,9 +21,11 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
public extension UIImage {
|
public extension UIImage {
|
||||||
//Extension Required by RoundedButton to create UIImage from UIColor
|
//Extension Required by RoundedButton to create UIImage from UIColor
|
||||||
func imageWithColor(color: UIColor) -> UIImage {
|
func imageWithColor(color: UIColor) -> UIImage {
|
||||||
|
|||||||
@@ -21,9 +21,11 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
@MainActor
|
@MainActor
|
||||||
public extension UIImageView {
|
public extension UIImageView {
|
||||||
|
|
||||||
|
|||||||
@@ -21,9 +21,11 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
public extension UILabel {
|
public extension UILabel {
|
||||||
|
|
||||||
func lineNumbers() -> Int{
|
func lineNumbers() -> Int{
|
||||||
|
|||||||
@@ -21,8 +21,10 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
import QuartzCore
|
import QuartzCore
|
||||||
|
|
||||||
public extension UINavigationController {
|
public extension UINavigationController {
|
||||||
|
|||||||
@@ -21,8 +21,10 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension UIResponder {
|
public extension UIResponder {
|
||||||
|
|
||||||
|
|||||||
@@ -21,7 +21,9 @@
|
|||||||
|
|
||||||
|
|
||||||
#if canImport(UIKit) && !os(watchOS)
|
#if canImport(UIKit) && !os(watchOS)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
// MARK: - Methods
|
// MARK: - Methods
|
||||||
public extension UIScrollView {
|
public extension UIScrollView {
|
||||||
|
|||||||
@@ -21,7 +21,9 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
#if os(iOS)
|
#if os(iOS)
|
||||||
|
|
||||||
// MARK: - Initializers
|
// MARK: - Initializers
|
||||||
|
|||||||
@@ -21,8 +21,10 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public class CustomTabBadge: UILabel {
|
public class CustomTabBadge: UILabel {
|
||||||
|
|
||||||
|
|||||||
@@ -21,9 +21,11 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
//MARK: - UITableView Animation Cell
|
//MARK: - UITableView Animation Cell
|
||||||
public typealias UITableViewCellAnimation = (UITableViewCell, IndexPath, UITableView) -> Void
|
public typealias UITableViewCellAnimation = (UITableViewCell, IndexPath, UITableView) -> Void
|
||||||
|
|
||||||
|
|||||||
@@ -21,8 +21,10 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
#if canImport(UIKit) && os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension UITapGestureRecognizer {
|
public extension UITapGestureRecognizer {
|
||||||
|
|
||||||
|
|||||||
@@ -21,8 +21,10 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
public extension UITextField {
|
public extension UITextField {
|
||||||
|
|
||||||
|
|||||||
@@ -21,8 +21,10 @@
|
|||||||
|
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
typealias GradientPoints = (startPoint: CGPoint, endPoint: CGPoint)
|
typealias GradientPoints = (startPoint: CGPoint, endPoint: CGPoint)
|
||||||
|
|
||||||
|
|||||||
@@ -22,8 +22,10 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
import QuartzCore
|
import QuartzCore
|
||||||
|
|
||||||
public enum ToastPosition {
|
public enum ToastPosition {
|
||||||
|
|||||||
@@ -20,8 +20,10 @@
|
|||||||
// THE SOFTWARE.
|
// THE SOFTWARE.
|
||||||
|
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
import AVFoundation
|
import AVFoundation
|
||||||
import Photos
|
import Photos
|
||||||
|
|
||||||
|
|||||||
@@ -19,9 +19,11 @@
|
|||||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
// THE SOFTWARE.
|
// THE SOFTWARE.
|
||||||
|
|
||||||
|
#if canImport(UIKit)
|
||||||
import UIKit
|
import UIKit
|
||||||
|
#endif
|
||||||
|
|
||||||
#if os(iOS) || os(macOS)
|
#if canImport(UIKit)
|
||||||
/// A protocol for delegates of `ImageZoomController` to provide callbacks for zoom and close events.
|
/// A protocol for delegates of `ImageZoomController` to provide callbacks for zoom and close events.
|
||||||
@objc public protocol ImageZoomControllerDelegate {
|
@objc public protocol ImageZoomControllerDelegate {
|
||||||
/// Called when the image in the controller is zoomed.
|
/// Called when the image in the controller is zoomed.
|
||||||
|
|||||||
@@ -20,6 +20,8 @@
|
|||||||
// THE SOFTWARE.
|
// THE SOFTWARE.
|
||||||
|
|
||||||
// MARK: - Framework headers
|
// MARK: - Framework headers
|
||||||
|
#if canImport(UIKit)
|
||||||
|
import Foundation
|
||||||
import UIKit
|
import UIKit
|
||||||
|
|
||||||
// MARK: - Protocols
|
// MARK: - Protocols
|
||||||
@@ -512,3 +514,5 @@ public extension LCSnackBarView {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|||||||
@@ -18,7 +18,7 @@
|
|||||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
// THE SOFTWARE.
|
// THE SOFTWARE.
|
||||||
#if canImport(SwiftUI)
|
#if canImport(SwiftUI) && os(iOS)
|
||||||
import SwiftUI
|
import SwiftUI
|
||||||
|
|
||||||
/// `LCENavigationState` is an `ObservableObject` that manages the state for `LCENavigationView`.
|
/// `LCENavigationState` is an `ObservableObject` that manages the state for `LCENavigationView`.
|
||||||
|
|||||||
Reference in New Issue
Block a user