|
|
|
@ -1,10 +1,5 @@
|
|
|
|
|
// Grabbed from: https://github.com/cbpowell/MarqueeLabel-Swift/blob/cd331f3cfc3f9d7114ffa5aa4f243f1d5eda9d0d/Classes/MarqueeLabel.swift
|
|
|
|
|
// License: MIT License
|
|
|
|
|
//
|
|
|
|
|
// MarqueeLabel.swift
|
|
|
|
|
//
|
|
|
|
|
// Created by Charles Powell on 8/6/14.
|
|
|
|
|
// Copyright (c) 2015 Charles Powell. All rights reserved.
|
|
|
|
|
// Copyright (c) 2018 Open Whisper Systems. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
import UIKit
|
|
|
|
@ -332,13 +327,11 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
The length of delay in seconds that the label pauses at the completion of a scroll.
|
|
|
|
|
*/
|
|
|
|
|
@IBInspectable open var animationDelay: CGFloat = 1.0
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** The read-only duration of the scroll animation (not including delay).
|
|
|
|
|
|
|
|
|
|
The value of this property is calculated from the value set to the `speed` property. If a .duration value is
|
|
|
|
@ -419,7 +412,6 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
MarqueeLabel.notifyController(controller, message: .Animate)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// MARK: - Initialization
|
|
|
|
|
//
|
|
|
|
@ -492,8 +484,8 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(MarqueeLabel.labelizeForController(_:)), name: NSNotification.Name(rawValue: MarqueeKeys.Labelize.rawValue), object: nil)
|
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(MarqueeLabel.animateForController(_:)), name: NSNotification.Name(rawValue: MarqueeKeys.Animate.rawValue), object: nil)
|
|
|
|
|
// UIApplication state notifications
|
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(MarqueeLabel.restartLabel), name: NSNotification.Name.UIApplicationDidBecomeActive, object: nil)
|
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(MarqueeLabel.shutdownLabel), name: NSNotification.Name.UIApplicationDidEnterBackground, object: nil)
|
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(MarqueeLabel.restartLabel), name: NSNotification.Name.OWSApplicationDidBecomeActive, object: nil)
|
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(MarqueeLabel.shutdownLabel), name: NSNotification.Name.OWSApplicationDidEnterBackground, object: nil)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
override open func awakeFromNib() {
|
|
|
|
@ -596,7 +588,7 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
awayOffset = 0.0
|
|
|
|
|
|
|
|
|
|
// Remove an additional sublabels (for continuous types)
|
|
|
|
|
repliLayer?.instanceCount = 1;
|
|
|
|
|
repliLayer?.instanceCount = 1
|
|
|
|
|
|
|
|
|
|
// Set the sublabel frame to calculated labelFrame
|
|
|
|
|
sublabel.frame = labelFrame
|
|
|
|
@ -701,13 +693,11 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
ScrollStep(timeStep: animationDuration, timingFunction: animationCurve, // Away position, using animationCurve transition, with only leading edge faded in
|
|
|
|
|
position: .away, edgeFades: .leading),
|
|
|
|
|
ScrollStep(timeStep: 60*60*24*365.0, // "Delay" at away, for huge time to effectie stay at away permanently
|
|
|
|
|
position: .away, edgeFades: .leading),
|
|
|
|
|
position: .away, edgeFades: .leading)
|
|
|
|
|
]
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Configure gradient for current condition
|
|
|
|
|
applyGradientMask(fadeLength, animated: !self.labelize)
|
|
|
|
|
|
|
|
|
@ -842,10 +832,10 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
fader = nil
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
fader = nil;
|
|
|
|
|
fader = nil
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
scrollCompletionBlock = { [weak self] (finished: Bool) -> () in
|
|
|
|
|
scrollCompletionBlock = { [weak self] (finished: Bool) -> Void in
|
|
|
|
|
guard finished else {
|
|
|
|
|
// Do not continue into the next loop
|
|
|
|
|
return
|
|
|
|
@ -1003,7 +993,7 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
animation.keyTimes = fadeKeyTimes
|
|
|
|
|
animation.timingFunctions = fadeTimingFunctions
|
|
|
|
|
|
|
|
|
|
return (anim: animation, duration: max(totalTime,totalDuration))
|
|
|
|
|
return (anim: animation, duration: max(totalTime, totalDuration))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private func applyGradientMask(_ fadeLength: CGFloat, animated: Bool, firstStep: MarqueeStep? = nil) {
|
|
|
|
@ -1132,7 +1122,6 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// MARK: - Private details
|
|
|
|
|
//
|
|
|
|
@ -1162,8 +1151,8 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
|
|
|
|
|
// Draw only background color
|
|
|
|
|
if let bgColor = backgroundColor {
|
|
|
|
|
ctx.setFillColor(bgColor.cgColor);
|
|
|
|
|
ctx.fill(layer.bounds);
|
|
|
|
|
ctx.setFillColor(bgColor.cgColor)
|
|
|
|
|
ctx.fill(layer.bounds)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -1202,7 +1191,6 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// MARK: - Label Control
|
|
|
|
|
//
|
|
|
|
@ -1573,7 +1561,6 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// MARK: - Support
|
|
|
|
|
//
|
|
|
|
@ -1592,7 +1579,6 @@ open class MarqueeLabel: UILabel, CAAnimationDelegate {
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// MARK: - Support
|
|
|
|
|
//
|
|
|
|
@ -1602,7 +1588,6 @@ public protocol MarqueeStep {
|
|
|
|
|
var edgeFades: EdgeFade { get }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
`ScrollStep` types define the label position at a specified time delta since the last `ScrollStep` step, as well as
|
|
|
|
|
the animation curve to that position and edge fade state at the position
|
|
|
|
@ -1661,7 +1646,6 @@ public struct ScrollStep: MarqueeStep {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
`FadeStep` types allow additional edge fade state definitions, around the states defined by the `ScrollStep` steps of
|
|
|
|
|
a sequence. `FadeStep` steps are defined by the time delta to the preceding or subsequent `ScrollStep` step and the timing
|
|
|
|
@ -1706,21 +1690,21 @@ public struct FadeStep: MarqueeStep {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public struct EdgeFade : OptionSet {
|
|
|
|
|
public struct EdgeFade: OptionSet {
|
|
|
|
|
public let rawValue: Int
|
|
|
|
|
public static let leading = EdgeFade(rawValue: 1 << 0)
|
|
|
|
|
public static let trailing = EdgeFade(rawValue: 1 << 1)
|
|
|
|
|
|
|
|
|
|
public init(rawValue: Int) {
|
|
|
|
|
self.rawValue = rawValue;
|
|
|
|
|
self.rawValue = rawValue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Define helpful typealiases
|
|
|
|
|
fileprivate typealias MLAnimationCompletionBlock = (_ finished: Bool) -> ()
|
|
|
|
|
fileprivate typealias MLAnimationCompletionBlock = (_ finished: Bool) -> Void
|
|
|
|
|
fileprivate typealias MLAnimation = (anim: CAKeyframeAnimation, duration: CGFloat)
|
|
|
|
|
|
|
|
|
|
fileprivate class GradientSetupAnimation: CABasicAnimation {
|
|
|
|
|
private class GradientSetupAnimation: CABasicAnimation {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fileprivate extension UIResponder {
|
|
|
|
@ -1796,14 +1780,14 @@ fileprivate extension CAMediaTimingFunction {
|
|
|
|
|
return t0
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func YforCurveAt(_ t: CGFloat, controlPoints:[CGPoint]) -> CGFloat {
|
|
|
|
|
func YforCurveAt(_ t: CGFloat, controlPoints: [CGPoint]) -> CGFloat {
|
|
|
|
|
let P0 = controlPoints[0]
|
|
|
|
|
let P1 = controlPoints[1]
|
|
|
|
|
let P2 = controlPoints[2]
|
|
|
|
|
let P3 = controlPoints[3]
|
|
|
|
|
|
|
|
|
|
// Per http://en.wikipedia.org/wiki/Bezier_curve#Cubic_B.C3.A9zier_curves
|
|
|
|
|
let y0 = (pow((1.0 - t),3.0) * P0.y)
|
|
|
|
|
let y0 = (pow((1.0 - t), 3.0) * P0.y)
|
|
|
|
|
let y1 = (3.0 * pow(1.0 - t, 2.0) * t * P1.y)
|
|
|
|
|
let y2 = (3.0 * (1.0 - t) * pow(t, 2.0) * P2.y)
|
|
|
|
|
let y3 = (pow(t, 3.0) * P3.y)
|
|
|
|
@ -1819,7 +1803,7 @@ fileprivate extension CAMediaTimingFunction {
|
|
|
|
|
|
|
|
|
|
// Per http://en.wikipedia.org/wiki/Bezier_curve#Cubic_B.C3.A9zier_curves
|
|
|
|
|
|
|
|
|
|
let x0 = (pow((1.0 - t),3.0) * P0.x)
|
|
|
|
|
let x0 = (pow((1.0 - t), 3.0) * P0.x)
|
|
|
|
|
let x1 = (3.0 * pow(1.0 - t, 2.0) * t * P1.x)
|
|
|
|
|
let x2 = (3.0 * (1.0 - t) * pow(t, 2.0) * P2.x)
|
|
|
|
|
let x3 = (pow(t, 3.0) * P3.x)
|
|
|
|
@ -1852,4 +1836,3 @@ fileprivate extension CAMediaTimingFunction {
|
|
|
|
|
return pointArray
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|