Files
GetGoing/ProjectClock/Utils.swift
T
2021-01-22 16:34:40 -05:00

194 lines
5.2 KiB
Swift

//
// Utils.swift
// ProjectClock
//
// Created by Hykilpikonna on 1/17/21.
//
import Foundation
import CryptoKit
import UIKit
/// Date manipulations
extension Date
{
/// Add toString to Date
func str() -> String
{
let f = DateFormatter()
f.dateFormat = "yyyy-MM-dd hh:mm:ss"
return f.string(from: self)
}
/// Constructor from components
static func create(_ year: Int, _ month: Int, _ day: Int, _ hour: Int, _ minute: Int) -> Date
{
var c = DateComponents()
c.year = year
c.month = month
c.day = day
c.hour = hour
c.minute = minute
let cal = Calendar(identifier: .gregorian)
return cal.date(from: c)!
}
/// Get year, month, day
func getYMD() -> (y: Int, m: Int, d: Int)
{
let calendar = Calendar.current
let comp = calendar.dateComponents([.year, .month, .day], from: self)
return (comp.year!, comp.month!, comp.day!)
}
/// Get hour, minute, seconds
func getHMS() -> (h: Int, m: Int, s: Int)
{
let calendar = Calendar.current
let comp = calendar.dateComponents([.hour, .minute, .second], from: self)
return (comp.hour!, comp.minute!, comp.second!)
}
/// Get another component
func get(_ c: Calendar.Component) -> Int
{
let calendar = Calendar.current
let comp = calendar.dateComponents([c], from: self)
return comp.value(for: c)!
}
/// Return a new modified date
func added(_ c: Calendar.Component, _ v: Int) -> Date
{
return Calendar.current.date(byAdding: c, value: v, to: self)!
}
}
extension TimeInterval
{
var seconds: Int { return Int(self) % 60 }
var minutes: Int { return (Int(self) / 60) % 60 }
var hours: Int { return (Int(self) / 3600) % 24 }
var days: Int { return Int(self) / (3600 * 24) }
/// Add toString to time interval
func str() -> String
{
if days != 0 { return "\(days)d \(hours)h \(minutes)m \(seconds)s" }
else if hours != 0 { return "\(hours)h \(minutes)m \(seconds)s" }
else if minutes != 0 { return "\(minutes)m \(seconds)s" }
else { return "\(seconds)s" }
}
}
/// Apply like Kotlin
protocol HasApply {}
extension HasApply
{
@discardableResult
func apply(_ c: (Self) -> ()) -> Self
{
c(self)
return self
}
}
extension Alarm: HasApply {}
extension Alarms: HasApply {}
/// Hashing
extension Digest
{
var bytes: [UInt8] { Array(makeIterator()) }
var b64: String { Data(bytes).base64EncodedString() }
}
extension String
{
var sha256: String { SHA256.hash(data: self.data(using: .utf8)!).b64 }
}
/// UI Extensions
extension UIViewController
{
/**
Send an alert
- Parameter title: Title of the alert
- Parameter message: Body message of the alert
- Parameter okayable: Whether the alert can be okayed
*/
@discardableResult
func alert(_ title: String, _ message: String, okayable: Bool = false) -> UIAlertController
{
// Create alert
let alert = UIAlertController(title: title, message: message, preferredStyle: .alert)
// Add okay button if it's okayable
if okayable { alert.addAction(UIAlertAction(title: "OK", style: .default)) }
// Display alert
self.present(alert, animated: true, completion: nil)
return alert
}
/// A message is an okayable alert
@discardableResult
func msg(_ title: String, _ message: String) -> UIAlertController { alert(title, message, okayable: true) }
/// More convenient dismiss function
func dismiss(_ completion: (() -> Void)? = nil) { dismiss(animated: false, completion: completion) }
}
/// Regex Matching (Credit: https://www.hackingwithswift.com/articles/108/how-to-use-regular-expressions-in-swift)
extension NSRegularExpression
{
convenience init(_ pattern: String)
{
do { try self.init(pattern: pattern) }
catch { preconditionFailure("Illegal regular expression: \(pattern).") }
}
func matches(_ string: String) -> Bool
{
let range = NSRange(location: 0, length: string.utf16.count)
return firstMatch(in: string, options: [], range: range) != nil
}
}
extension String
{
static func ~= (lhs: String, rhs: String) -> Bool
{
guard let regex = try? NSRegularExpression(pattern: rhs) else { return false }
let range = NSRange(location: 0, length: lhs.utf16.count)
return regex.firstMatch(in: lhs, options: [], range: range) != nil
}
}
/// More convenient ui update closure
func ui(closure: @escaping () -> Void) { DispatchQueue.main.async { closure() } }
/// More convenient UserDefaults access (Credit: https://gist.github.com/Otbivnoe/04b8bd7984fba0cb58ca7f136fd95582)
extension UserDefaults
{
subscript<T>(key: String) -> T?
{
get { return value(forKey: key) as? T }
set { set(newValue, forKey: key) }
}
subscript<T: RawRepresentable>(key: String) -> T?
{
get
{
if let rawValue = value(forKey: key) as? T.RawValue { return T(rawValue: rawValue) }
return nil
}
set { self[key] = newValue?.rawValue }
}
}