initial
This commit is contained in:
21
Pods/SwiftEntryKit/LICENSE
generated
Normal file
21
Pods/SwiftEntryKit/LICENSE
generated
Normal file
@@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2018 Daniel Huri, huri000@gmail.com
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
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 THE
|
||||
SOFTWARE.
|
||||
926
Pods/SwiftEntryKit/README.md
generated
Normal file
926
Pods/SwiftEntryKit/README.md
generated
Normal file
@@ -0,0 +1,926 @@
|
||||
# SwiftEntryKit <img align="left" height=42 src="https://github.com/huri000/assets/blob/master/swift-entrykit/project-icon.png">
|
||||
|
||||
[](https://developer.apple.com/iphone/index.action)
|
||||
[](https://developer.apple.com/swift)
|
||||
[](http://cocoapods.org/pods/SwiftEntryKit)
|
||||
[](https://github.com/Carthage/Carthage)
|
||||
[](https://github.com/JamitLabs/Accio)
|
||||
[](http://mit-license.org)
|
||||

|
||||
|
||||
🤗 Donations can be made [here](#donations).
|
||||
|
||||
## Table of Contents
|
||||
|
||||
* [Overview](#overview)
|
||||
* [Features](#features)
|
||||
* [Example Project](#example-project)
|
||||
* [Example Project Installation](#example-project-installation)
|
||||
* [Presets](#presets)
|
||||
* [Playground](#playground)
|
||||
* [Requirements](#requirements)
|
||||
* [Installation](#installation)
|
||||
* [Usage](#usage)
|
||||
* [Quick Usage](#quick-usage)
|
||||
* [Entry Attributes](#entry-attributes)
|
||||
* [Entry Name](#entry-name)
|
||||
* [Window Level](#window-level)
|
||||
* [Display Position](#display-position)
|
||||
* [Precedence](#precedence)
|
||||
* [Override](#override)
|
||||
* [Enqueue](#enqueue)
|
||||
* [Heuristics](#heuristics)
|
||||
* [Display Priority](#display-priority)
|
||||
* [Display Duration](#display-duration)
|
||||
* [Position Constraints](#position-constraints)
|
||||
* [User Interaction](#user-interaction)
|
||||
* [Scroll Behavior](#scroll-behavior)
|
||||
* [Haptic Feedback](#haptic-feedback)
|
||||
* [Lifecycle Events](#lifecycle-events)
|
||||
* [Display Mode](#display-mode)
|
||||
* [Background Style](#background-style)
|
||||
* [Shadow](#shadow)
|
||||
* [Round Corners](#round-corners)
|
||||
* [Border](#border)
|
||||
* [Animations](#animations)
|
||||
* [Pop Behavior](#pop-behavior)
|
||||
* [Status Bar](#status-bar)
|
||||
* [Presets Usage Example](#presets-usage-example)
|
||||
* [Custom View Usage Example](#custom-view-usage-example)
|
||||
* [Displaying a View Controller](#displaying-a-view-controller)
|
||||
* [Alternative Rollback Window](#alternative-rollback-window)
|
||||
* [Dismissing an Entry](#dismissing-an-entry)
|
||||
* [Swiping and Rubber Banding](#swiping-and-rubber-banding)
|
||||
* [Dealing With Safe Area](#dealing-with-safe-area)
|
||||
* [Dealing With Orientation Change](#dealing-with-orientation-change)
|
||||
* [Swift and Objective-C Interoperability](#swift-and-objective-c-interoperability)
|
||||
* [Author](#author)
|
||||
* [Donations](#donations)
|
||||
* [License](#license)
|
||||
|
||||
## Overview
|
||||
|
||||
SwiftEntryKit is a simple yet versatile content presenter written in Swift.
|
||||
|
||||
### Features
|
||||
|
||||
Banners or pop-ups are called *Entries*.
|
||||
|
||||
- Entries are displayed inside a separate UIWindow (of type EKWindow), so users are able to navigate the app freely while entries are being displayed in a non intrusive manner.
|
||||
- The kit offers beautiful [presets](#presets) that can be themed with your own colors and fonts.
|
||||
- **Customization**: Entries are highly customizable
|
||||
- [x] Can be [positioned](#display-position) either at the top, center, or the bottom of the screen.
|
||||
- [x] Can be displayed within or outside the screen safe area.
|
||||
- [x] Can be stylized: have a [border](#border), [drop-shadow](#shadow) and [round corners](#round-corners).
|
||||
- [x] Their content and the surrounding background can be blurred, dimmed, colored or have a gradient [style](#background-style).
|
||||
- [x] Transition [animations](#animations) are customizable - entrance, exit and pop (by another entry).
|
||||
- [x] The [user interaction](#user-interaction) with the entry or the screen can be intercepted.
|
||||
- [x] Entries can be enqueued or override previous entries using the [precedence](#precedence) attribute.
|
||||
- [x] Each entry has a [display priority](#display-priority) attribute. That means that it can be dismissed only by other entry with an equal or higher priority.
|
||||
- [x] Presets support accessibility.
|
||||
- [x] Entries have an optional rubber banding effect while panning.
|
||||
- [x] Entries can be optionally dismissed using a simple [swipe gesture](#swiping-and-rubber-banding).
|
||||
- [x] Entries can be optionally injected with [lifecycle events](#lifecycle-events): *will* and *did* appear/disappear.
|
||||
- [x] The [status bar style](#status-bar) is settable for the display duration of the entry.
|
||||
- [x] Supports [navigation controllers](#presets) & [custom views](#custom-view-usage-example) as well!
|
||||
|
||||
## Example Project
|
||||
|
||||
The example project contains various presets and examples you can use and modify as your like.
|
||||
|
||||
### Example Project Installation
|
||||
|
||||
You can either use the terminal or git client such as Source Tree.
|
||||
|
||||
#### Terminal Users
|
||||
|
||||
```bash
|
||||
$ git clone https://github.com/huri000/SwiftEntryKit.git
|
||||
```
|
||||
|
||||
#### Git Client (Source Tree)
|
||||
Clone https://github.com/huri000/SwiftEntryKit.git
|
||||
|
||||
### Presets
|
||||
|
||||
| Toasts | Notes | Floats | Popups |
|
||||
| --- | --- | --- | --- |
|
||||
|  |  |  |  |
|
||||
|
||||
| Alerts | Forms | Rating | More... |
|
||||
| --- | --- | --- | --- |
|
||||
|  |  |  |  |
|
||||
|
||||
### Playground
|
||||
|
||||
**noun: a place where people can play 🏈**
|
||||
|
||||
The example app contains a playground screen, an interface that allows you to customize your preferable entries.
|
||||
The playground screen has some limitations (allows to select constant values) but you can easily modify the code to suit your needs. Check it out!
|
||||
|
||||
The Playground Screen | Top Toast Sample
|
||||
--- | ---
|
||||
 | 
|
||||
|
||||
## Requirements
|
||||
|
||||
- iOS 9 or any higher version.
|
||||
- Xcode 9 or any higher version.
|
||||
- Swift 4.0 or any higher version.
|
||||
- The library has not been tested with iOS 8.x.y or a lower version.
|
||||
|
||||
## Installation
|
||||
|
||||
- SwiftEntryKit is compatible with Swift 5 as of release *1.0.0*.
|
||||
- SwiftEntryKit is compatible with Swift 4.2 as of release *0.8.1*.
|
||||
- Developers who use lower Swift version should install release *0.7.2*.
|
||||
|
||||
### CocoaPods
|
||||
|
||||
[CocoaPods](http://cocoapods.org) is a dependency manager for Cocoa projects. You can install it with the following command:
|
||||
|
||||
```bash
|
||||
$ gem install cocoapods
|
||||
```
|
||||
|
||||
To integrate SwiftEntryKit into your Xcode project using CocoaPods, specify it in your `Podfile`:
|
||||
|
||||
```ruby
|
||||
source 'https://github.com/cocoapods/specs.git'
|
||||
platform :ios, '9.0'
|
||||
use_frameworks!
|
||||
|
||||
pod 'SwiftEntryKit', '2.0.0'
|
||||
```
|
||||
|
||||
Then, run the following command:
|
||||
|
||||
```bash
|
||||
$ pod install
|
||||
```
|
||||
|
||||
### Carthage
|
||||
|
||||
[Carthage](https://github.com/Carthage/Carthage) is a decentralized dependency manager that builds your dependencies and provides you with binary frameworks.
|
||||
|
||||
You can install Carthage with [Homebrew](http://brew.sh/) using the following command:
|
||||
|
||||
```bash
|
||||
$ brew update
|
||||
$ brew install carthage
|
||||
```
|
||||
|
||||
To integrate SwiftEntryKit into your Xcode project using Carthage, specify the following in your `Cartfile`:
|
||||
|
||||
```ogdl
|
||||
github "huri000/SwiftEntryKit" == 2.0.0
|
||||
```
|
||||
|
||||
### Accio
|
||||
|
||||
[Accio](https://github.com/JamitLabs/Accio) is a decentralized dependency manager driven by SwiftPM that works for iOS/tvOS/watchOS/macOS projects.
|
||||
|
||||
You can install Accio with [Homebrew](http://brew.sh/) using the following command:
|
||||
|
||||
```bash
|
||||
$ brew tap JamitLabs/Accio https://github.com/JamitLabs/Accio.git
|
||||
$ brew install accio
|
||||
```
|
||||
|
||||
To integrate SwiftEntryKit into your Xcode project using Accio, specify the following in your `Package.swift` manifest:
|
||||
|
||||
```swift
|
||||
.package(url: "https://github.com/huri000/SwiftEntryKit", .exact("2.0.0"))
|
||||
```
|
||||
|
||||
After specifying `"SwiftEntryKit"` as a dependency of the target in which you want to use it, run `accio install`.
|
||||
|
||||
## Usage
|
||||
|
||||
### Quick Usage
|
||||
|
||||
No setup is needed! Each time you wish to display an entry, just create your view and initialize an EKAttributes struct.
|
||||
See also the [preset usage example](#presets-usage-example), and the example project.
|
||||
likewise:
|
||||
```Swift
|
||||
// Customized view
|
||||
let customView = SomeCustomView()
|
||||
/*
|
||||
Do some customization on customView
|
||||
*/
|
||||
|
||||
// Attributes struct that describes the display, style, user interaction and animations of customView.
|
||||
var attributes = EKAttributes()
|
||||
/*
|
||||
Adjust preferable attributes
|
||||
*/
|
||||
```
|
||||
And then, just call:
|
||||
```Swift
|
||||
SwiftEntryKit.display(entry: customView, using: attributes)
|
||||
```
|
||||
The kit will replace the application main window with the EKWindow instance and display the entry.
|
||||
|
||||
### Entry Attributes
|
||||
|
||||
*EKAttributes* is the entry's descriptor. Each time an entry is displayed, an EKAttributes struct is necessary to describe the entry's presentation, position inside the screen, the display duration, its frame constraints (if needed), its styling (corners, border and shadow), the user interaction events, the animations (in / out) and more.
|
||||
|
||||
Create a mutable EKAttributes structure likewise:
|
||||
```Swift
|
||||
var attributes = EKAttributes()
|
||||
```
|
||||
|
||||
Below are the properties that can be modified in the *EKAttributes*:
|
||||
|
||||
#### Entry Name
|
||||
Entries can have names.
|
||||
When an EKAttributes struct is instantiated, it is nameless, meaning, the `name` property is `nil`.
|
||||
It is recommended to set a meaningful name for an entry.
|
||||
|
||||
```Swift
|
||||
attributes.name = "Top Note"
|
||||
```
|
||||
|
||||
Entries with names can be specifically referred to later, for example, you can inquire whether a **specific** entry is currently displayed:
|
||||
```Swift
|
||||
if SwiftEntryKit.isCurrentlyDisplaying(entryNamed: "Top Note") {
|
||||
/* Do your things */
|
||||
}
|
||||
```
|
||||
|
||||
#### Window Level
|
||||
Entries can be displayed above the application main window, above the status bar, above the alerts window or even have a custom level (UIWindowLevel).
|
||||
|
||||
For example, set the window level to *normal*, likewise:
|
||||
```Swift
|
||||
attributes.windowLevel = .normal
|
||||
```
|
||||
This causes the entry to appear above the application key window and below the status bar.
|
||||
|
||||
The default value of `windowLevel` is `.statusBar`.
|
||||
|
||||
#### Display Position
|
||||
The entry can be displayed either at the *top*, *center*, or the *bottom* of the screen.
|
||||
|
||||
For example, set the display position to *bottom*, likewise:
|
||||
```Swift
|
||||
attributes.position = .bottom
|
||||
```
|
||||
|
||||
The default value of `position` is `.top`.
|
||||
|
||||
#### Precedence
|
||||
The precedence attribute of an entry describes the manner in which entries are pushed in. It offers 2 approaches for managing the presentation priority of multiple simultaneous entries.
|
||||
|
||||
##### Override
|
||||
If the [display priority](#display-priority) is equal or higher than the currently displayed entry, override it.
|
||||
|
||||
Example for setting `.override` precedence with `.max` display priority while ignoring entries that are already enqueued (leaving them to display after the new entry is dismissed).
|
||||
|
||||
```Swift
|
||||
attributes.precedence = .override(priority: .max, dropEnqueuedEntries: false)
|
||||
```
|
||||
|
||||
You can optionally flush the entries that are inside the queue.
|
||||
|
||||
In case `dropEnqueuedEntries` is `false`, enqueued entries remain in the queue. The first enqueued entry will show right after the new entry pops out.
|
||||
In case `dropEnqueuedEntries` is `true`, the entry-queue is flushed as the new entry is being displayed.
|
||||
|
||||
##### Enqueue
|
||||
If the queue is empty, display the entry immediately, otherwise, insert the entry into the queue until its turn to show arrives.
|
||||
|
||||
Example for setting `.enqueue` precedence with `.normal` display priority:
|
||||
|
||||
```Swift
|
||||
attributes.precedence = .enqueue(priority: .normal)
|
||||
```
|
||||
|
||||
###### Heuristics
|
||||
|
||||
There are 2 possible heuristics for entries prioritization in the queue:
|
||||
|
||||
- Display Priority Queue: The entries are sorted by their [display priority](#display-priority), then by chronological order.
|
||||
- Chronological Queue: The entries are sorted only by their chronological order (standard queue).
|
||||
|
||||
Select the heuristic that suits you best by doing the following, only once, before using `SwiftEntryKit` to display entries.
|
||||
|
||||
```Swift
|
||||
EKAttributes.Precedence.QueueingHeuristic.value = .priority
|
||||
```
|
||||
|
||||
Or:
|
||||
|
||||
```Swift
|
||||
EKAttributes.Precedence.QueueingHeuristic.value = .chronological
|
||||
```
|
||||
|
||||
The default value of `EKAttributes.Precedence.QueueingHeuristic.value` is `.priority`.
|
||||
|
||||
The default value of precedence is `.override(priority: .normal, dropEnqueuedEntries: false)`.
|
||||
|
||||
##### Display Priority
|
||||
The display priority of the entry determines whether it dismisses other entries or is dismissed by them.
|
||||
An entry can be dismissed only by an entry with an equal or a higher display priority.
|
||||
|
||||
```Swift
|
||||
let highPriorityAttributes = EKAttributes()
|
||||
highPriorityAttributes.precedence.priority = .high
|
||||
|
||||
let normalPriorityAttributes = EKAttributes()
|
||||
normalPriorityAttributes.precedence.priority = .normal
|
||||
|
||||
// Display high priority entry
|
||||
SwiftEntryKit.display(entry: view1, using: highPriorityAttributes)
|
||||
|
||||
// Display normal priority entry (ignored!)
|
||||
SwiftEntryKit.display(entry: view2, using: normalPriorityAttributes)
|
||||
```
|
||||
|
||||
*view2* won't be displayed!
|
||||
|
||||
#### Display Duration
|
||||
The display duration of the entry (Counted from the moment the entry has finished its entrance animation and until the exit animation begins).
|
||||
|
||||
Display for 4 seconds:
|
||||
```Swift
|
||||
attributes.displayDuration = 4
|
||||
```
|
||||
|
||||
Display for an infinite duration
|
||||
```Swift
|
||||
attributes.displayDuration = .infinity
|
||||
```
|
||||
|
||||
The default value of `displayDuration` is `2`.
|
||||
|
||||
#### Position Constraints
|
||||
Constraints that tie the entry tightly to the screen context, for example: Height, Width, Max Width, Max Height, Additional Vertical Offset & Safe Area related info.
|
||||
|
||||
- Entries that support Auto Layout - Their height is inferred from the constraints that applied to them.
|
||||
- Entries that don't support Auto Layout - Their exact size must be explicitly set using `positionConstraints`'s `size` property.
|
||||
|
||||
For example:
|
||||
|
||||
Ratio edge - signifies that the ratio of the width edge has a ratio of 0.9 of the screen's width.
|
||||
```Swift
|
||||
let widthConstraint = EKAttributes.PositionConstraints.Edge.ratio(value: 0.9)
|
||||
```
|
||||
|
||||
Intrinsic edge - signifies that the wanted height value is the content height - Decided by the entries vertical constraints
|
||||
```Swift
|
||||
let heightConstraint = EKAttributes.PositionConstraints.Edge.intrinsic
|
||||
```
|
||||
|
||||
Create the entry size constraints likewise:
|
||||
```Swift
|
||||
attributes.positionConstraints.size = .init(width: widthConstraint, height: heightConstraint)
|
||||
```
|
||||
You can also set *attributes.positionConstraints.maxSize* in order to make sure the entry does not exceeds predefined limitations. This is useful on [device orientation change](#how-to-deal-with-orientation-change).
|
||||
|
||||
Safe Area - can be used to override the safe area or to color it (More examples are in the example project)
|
||||
That snippet implies that the safe area insets should be kept and not be a part of the entry.
|
||||
```Swift
|
||||
attributes.positionConstraints.safeArea = .empty(fillSafeArea: false)
|
||||
```
|
||||
|
||||
Vertical Offset - an additional offset that can be applied to the entry (Other than the safe area).
|
||||
```Swift
|
||||
attributes.positionConstraints.verticalOffset = 10
|
||||
```
|
||||
|
||||
Autorotation - whether the entry autorotates along with the orientation of the device. Defaults to `true`.
|
||||
```Swift
|
||||
attributes.positionConstraints.rotation.isEnabled = false
|
||||
```
|
||||
|
||||
Keyboard Releation - used to bind an entry to the keyboard once the keyboard is displayed.
|
||||
|
||||
```Swift
|
||||
let offset = EKAttributes.PositionConstraints.KeyboardRelation.Offset(bottom: 10, screenEdgeResistance: 20)
|
||||
let keyboardRelation = EKAttributes.PositionConstraints.KeyboardRelation.bind(offset: offset)
|
||||
attributes.positionConstraints.keyboardRelation = keyboardRelation
|
||||
```
|
||||
In the example above the entry's bottom is tuned to have a 10pts offset from the top of the keyboard (while it shows)
|
||||
Because the entry's frame might exceed the screen bounds, the user might not see all the entry - we wouldn't want that. Therefore, an additional associated value has been added - `screenEdgeResistance` with value of 20pts. That is, to make sure that the entry remains within the bounds of the screen, and always visible to the user.
|
||||
The extreme situation might occur as the device orientation is landscape and the keyboard shows up (See example project form presets for more information).
|
||||
|
||||
#### User Interaction
|
||||
The entry and the screen can be interacted by the user. User interaction be can intercepted in various ways:
|
||||
|
||||
An interaction (Any touch whatsoever) with the entry delays its exit by 3s:
|
||||
```Swift
|
||||
attributes.entryInteraction = .delayExit(by: 3)
|
||||
```
|
||||
|
||||
A tap on the entry / screen dismisses it immediately:
|
||||
```Swift
|
||||
attributes.entryInteraction = .dismiss
|
||||
attributes.screenInteraction = .dismiss
|
||||
```
|
||||
|
||||
A tap on the entry is swallowed (ignored):
|
||||
```Swift
|
||||
attributes.entryInteraction = .absorbTouches
|
||||
```
|
||||
|
||||
A tap on the screen is forwarded to the lower level window, in most cases the receiver will be the application window.
|
||||
This is very useful when you want to display an unintrusive content like banners and push notification entries.
|
||||
```Swift
|
||||
attributes.screenInteraction = .forward
|
||||
```
|
||||
|
||||
Pass additional actions that are invoked when the user taps the entry:
|
||||
```Swift
|
||||
let action = {
|
||||
// Do something useful
|
||||
}
|
||||
attributes.entryInteraction.customTapActions.append(action)
|
||||
```
|
||||
|
||||
The default value of `screenInteraction` is `.forward`.
|
||||
|
||||
The default value of `entryInteraction` is `.dismiss`.
|
||||
|
||||
#### Scroll Behavior
|
||||
Describes the entry behavior when it's being scrolled, that is, dismissal by a swipe gesture and a rubber band effect much similar to a UIScrollView.
|
||||
|
||||
Disable the pan and swipe gestures on the entry:
|
||||
```Swift
|
||||
attributes.scroll = .disabled
|
||||
```
|
||||
|
||||
Enable swipe and stretch and pullback with jolt effect:
|
||||
```Swift
|
||||
attributes.scroll = .enabled(swipeable: true, pullbackAnimation: .jolt)
|
||||
```
|
||||
|
||||
Enable swipe and stretch and pullback with an ease-out effect:
|
||||
```Swift
|
||||
attributes.scroll = .enabled(swipeable: true, pullbackAnimation: .easeOut)
|
||||
```
|
||||
|
||||
Enable swipe but disable stretch:
|
||||
```Swift
|
||||
attributes.scroll = .edgeCrossingDisabled(swipeable: true)
|
||||
```
|
||||
|
||||
The default value of `scroll` is `.enabled(swipeable: true, pullbackAnimation: .jolt)`.
|
||||
|
||||
#### [Haptic Feedback](https://developer.apple.com/ios/human-interface-guidelines/user-interaction/feedback/)
|
||||
The device can produce a haptic feedback, thus adding an additional sensory depth to each entry.
|
||||
|
||||
The default value of `hapticFeedbackType` is `.none`.
|
||||
|
||||
#### Lifecycle Events
|
||||
Events can be injected to the entry so that they are to be called during its lifecycle.
|
||||
|
||||
```Swift
|
||||
attributes.lifecycleEvents.willAppear = {
|
||||
// Executed before the entry animates inside
|
||||
}
|
||||
|
||||
attributes.lifecycleEvents.didAppear = {
|
||||
// Executed after the entry animates inside
|
||||
}
|
||||
|
||||
attributes.lifecycleEvents.willDisappear = {
|
||||
// Executed before the entry animates outside
|
||||
}
|
||||
|
||||
attributes.lifecycleEvents.didDisappear = {
|
||||
// Executed after the entry animates outside
|
||||
}
|
||||
```
|
||||
|
||||
#### Display Mode
|
||||
To allow you to fully support any user interface style, `SwiftEntryKit` introduces two specialized types:
|
||||
- `EKColor` describes a color under light and dark modes.
|
||||
- `EKAttributes.BackgroundStyle.BlurStyle` describes a blur effect under light and dark modes.
|
||||
|
||||
The following forces `SwiftEntryKit` to display the entry on dark mode.
|
||||
```Swift
|
||||
attributes.displayMode = .dark
|
||||
```
|
||||
|
||||
The possible values are: `.light`, `.dark`, `.inferred`.
|
||||
The default value is `.inferred`, which means that the entry will be displayed with the current user interface style.
|
||||
|
||||
#### Background Style
|
||||
The entry and the screen can have various background styles, such as blur, color, gradient and even an image.
|
||||
|
||||
The following example implies clear background for both the entry and the screen:
|
||||
```Swift
|
||||
attributes.entryBackground = .clear
|
||||
attributes.screenBackground = .clear
|
||||
```
|
||||
|
||||
Colored entry background and dimmed screen background:
|
||||
```Swift
|
||||
attributes.entryBackground = .color(color: .standardContent)
|
||||
attributes.screenBackground = .color(color: EKColor(UIColor(white: 0.5, alpha: 0.5)))
|
||||
```
|
||||
|
||||
Gradient entry background (diagonal vector):
|
||||
```Swift
|
||||
let colors: [EKColor] = ...
|
||||
attributes.entryBackground = .gradient(gradient: .init(colors: colors, startPoint: .zero, endPoint: CGPoint(x: 1, y: 1)))
|
||||
```
|
||||
|
||||
Visual Effect entry background:
|
||||
```Swift
|
||||
attributes.entryBackground = .visualEffect(style: .dark)
|
||||
```
|
||||
|
||||
The default value of `entryBackground` and `screenBackground` is `.clear`.
|
||||
|
||||
#### Shadow
|
||||
The shadow that surrounds the entry.
|
||||
|
||||
Enable shadow around the entry:
|
||||
```Swift
|
||||
attributes.shadow = .active(with: .init(color: .black, opacity: 0.3, radius: 10, offset: .zero))
|
||||
```
|
||||
|
||||
Disable shadow around the entry:
|
||||
```Swift
|
||||
attributes.shadow = .none
|
||||
```
|
||||
|
||||
The default value of `shadow` is `.none`.
|
||||
|
||||
#### Round Corners
|
||||
Round corners around the entry.
|
||||
|
||||
Only top left and right corners with radius of 10:
|
||||
```Swift
|
||||
attributes.roundCorners = .top(radius: 10)
|
||||
```
|
||||
|
||||
Only bottom left and right corners with radius of 10:
|
||||
```Swift
|
||||
attributes.roundCorners = .bottom(radius: 10)
|
||||
```
|
||||
|
||||
All corners with radius of 10:
|
||||
```Swift
|
||||
attributes.roundCorners = .all(radius: 10)
|
||||
```
|
||||
|
||||
No round corners:
|
||||
```Swift
|
||||
attributes.roundCorners = .none
|
||||
```
|
||||
|
||||
The default value of `roundCorners` is `.none`.
|
||||
|
||||
#### Border
|
||||
The border around the entry.
|
||||
|
||||
Add a black border with thickness of 0.5pts:
|
||||
```Swift
|
||||
attributes.border = .value(color: .black, width: 0.5)
|
||||
```
|
||||
|
||||
No border:
|
||||
```Swift
|
||||
attributes.border = .none
|
||||
```
|
||||
|
||||
The default value of `border` is `.none`.
|
||||
|
||||
#### Animations
|
||||
Describes how the entry animates into and out of the screen.
|
||||
|
||||
* Each animation descriptor can have up to 3 types of animations at the same time. Those can be combined to a single complex one!
|
||||
* Translation animation anchor can be explicitly set but it receives a default value according to position of the entry.
|
||||
|
||||
Example for _translation_ from top with spring, _scale_ in and even _fade in_ as a single entrance animation:
|
||||
```Swift
|
||||
attributes.entranceAnimation = .init(
|
||||
translate: .init(duration: 0.7, anchorPosition: .top, spring: .init(damping: 1, initialVelocity: 0)),
|
||||
scale: .init(from: 0.6, to: 1, duration: 0.7),
|
||||
fade: .init(from: 0.8, to: 1, duration: 0.3))
|
||||
```
|
||||
|
||||
The default value of `entranceAnimation` and `exitAnimation` is `.translation` - The entry translates in or out, respectively, with duration of 0.3 seconds.
|
||||
|
||||
#### Pop Behavior
|
||||
Describes the entry behavior when it's being popped (dismissed by an entry with equal / higher display-priority.
|
||||
|
||||
The entry is being popped animatedly:
|
||||
```Swift
|
||||
attributes.popBehavior = .animated(animation: .init(translate: .init(duration: 0.2)))
|
||||
```
|
||||
|
||||
The entry is being overridden (Disappears promptly):
|
||||
```Swift
|
||||
attributes.popBehavior = .overridden
|
||||
```
|
||||
|
||||
The default value of `popBehavior` is `.animated(animation: .translation)` - It translates out with duration of 0.3 seconds.
|
||||
|
||||
#### Status Bar
|
||||
The status bar appearance can be modified during the display of the entry.
|
||||
SwiftEntryKit supports both *View controller-based status bar appearance* and manual setting.
|
||||
|
||||
Setting the status bar style is fairly simple -
|
||||
|
||||
Status bar becomes visible and gets a light style:
|
||||
```Swift
|
||||
attributes.statusBar = .light
|
||||
```
|
||||
|
||||
The status bar becomes hidden:
|
||||
```Swift
|
||||
attributes.statusBar = .hidden
|
||||
```
|
||||
|
||||
The status bar appearance is inferred from the previous context (won't be changed):
|
||||
```Swift
|
||||
attributes.statusBar = .inferred
|
||||
```
|
||||
|
||||
In case there is an already presenting entry with lower/equal display priority, the status bar will change its style.
|
||||
When the entry is removed, the status bar gets its initial style back.
|
||||
|
||||
The default value of `statusBar` is `.inferred`.
|
||||
|
||||
#### EKAttributes' interface is as follows:
|
||||
|
||||
```Swift
|
||||
public struct EKAttributes
|
||||
|
||||
// Identification
|
||||
public var name: String?
|
||||
|
||||
// Display
|
||||
public var windowLevel: WindowLevel
|
||||
public var position: Position
|
||||
public var precedence: Precedence
|
||||
public var displayDuration: DisplayDuration
|
||||
public var positionConstraints: PositionConstraints
|
||||
|
||||
// User Interaction
|
||||
public var screenInteraction: UserInteraction
|
||||
public var entryInteraction: UserInteraction
|
||||
public var scroll: Scroll
|
||||
public var hapticFeedbackType: NotificationHapticFeedback
|
||||
public var lifecycleEvents: LifecycleEvents
|
||||
|
||||
// Theme & Style
|
||||
public var displayMode = DisplayMode.inferred
|
||||
public var entryBackground: BackgroundStyle
|
||||
public var screenBackground: BackgroundStyle
|
||||
public var shadow: Shadow
|
||||
public var roundCorners: RoundCorners
|
||||
public var border: Border
|
||||
public var statusBar: StatusBar
|
||||
|
||||
// Animations
|
||||
public var entranceAnimation: Animation
|
||||
public var exitAnimation: Animation
|
||||
public var popBehavior: PopBehavior
|
||||
}
|
||||
```
|
||||
|
||||
### Presets Usage Example:
|
||||
|
||||
You can use one of the presets that come with SwiftEntryKit, doing these 4 simple steps:
|
||||
|
||||
1. Create your *EKAttributes* struct and set your preferable properties.
|
||||
2. Create *EKNotificationMessage* struct (The Content) and set the content.
|
||||
3. Create *EKNotificationMessageView* (The View) and inject *EKNotificationMessage* struct to it.
|
||||
4. Display the entry using *SwiftEntryKit* class method.
|
||||
|
||||
#### EKNotificationMessageView preset example:
|
||||
|
||||
```Swift
|
||||
// Generate top floating entry and set some properties
|
||||
var attributes = EKAttributes.topFloat
|
||||
attributes.entryBackground = .gradient(gradient: .init(colors: [EKColor(.red), EKColor(.green)], startPoint: .zero, endPoint: CGPoint(x: 1, y: 1)))
|
||||
attributes.popBehavior = .animated(animation: .init(translate: .init(duration: 0.3), scale: .init(from: 1, to: 0.7, duration: 0.7)))
|
||||
attributes.shadow = .active(with: .init(color: .black, opacity: 0.5, radius: 10, offset: .zero))
|
||||
attributes.statusBar = .dark
|
||||
attributes.scroll = .enabled(swipeable: true, pullbackAnimation: .jolt)
|
||||
attributes.positionConstraints.maxSize = .init(width: .constant(value: UIScreen.main.minEdge), height: .intrinsic)
|
||||
|
||||
let title = EKProperty.LabelContent(text: titleText, style: .init(font: titleFont, color: textColor))
|
||||
let description = EKProperty.LabelContent(text: descText, style: .init(font: descFont, color: textColor))
|
||||
let image = EKProperty.ImageContent(image: UIImage(named: imageName)!, size: CGSize(width: 35, height: 35))
|
||||
let simpleMessage = EKSimpleMessage(image: image, title: title, description: description)
|
||||
let notificationMessage = EKNotificationMessage(simpleMessage: simpleMessage)
|
||||
|
||||
let contentView = EKNotificationMessageView(with: notificationMessage)
|
||||
SwiftEntryKit.display(entry: contentView, using: attributes)
|
||||
```
|
||||
|
||||
### Custom View Usage Example:
|
||||
|
||||
```Swift
|
||||
// Create a basic toast that appears at the top
|
||||
var attributes = EKAttributes.topToast
|
||||
|
||||
// Set its background to white
|
||||
attributes.entryBackground = .color(color: .white)
|
||||
|
||||
// Animate in and out using default translation
|
||||
attributes.entranceAnimation = .translation
|
||||
attributes.exitAnimation = .translation
|
||||
|
||||
let customView = UIView()
|
||||
/*
|
||||
... Customize the view as you like ...
|
||||
*/
|
||||
|
||||
// Display the view with the configuration
|
||||
SwiftEntryKit.display(entry: customView, using: attributes)
|
||||
```
|
||||
|
||||
### Displaying a View Controller
|
||||
As from version 0.4.0, view controllers are supported as well.
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.display(entry: customViewController, using: attributes)
|
||||
```
|
||||
|
||||
### Alternative Rollback Window
|
||||
By default, the window held by the application delegate becomes the key again right after SwiftEntryKit has finished displaying the entry.
|
||||
This behavior can be changed using `rollbackWindow` parameter.
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.display(entry: view, using: attributes, rollbackWindow: .custom(window: alternativeWindow))
|
||||
```
|
||||
After the entry has been dismissed, the given window `alternativeWindow` would become the key instead of the window that is held by the application delegate.
|
||||
|
||||
### Dismissing an Entry
|
||||
You can dismiss the currently displayed entry by simply invoke *dismiss* in the SwiftEntryKit class, likewise:
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.dismiss()
|
||||
```
|
||||
Or:
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.dismiss(.displayed)
|
||||
```
|
||||
|
||||
This dismisses the entry animatedly using its *exitAnimation* attribute and on completion, the window would be removed as well.
|
||||
|
||||
You can dismiss the currently displayed entry and flush the queue as well, likewise:
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.dismiss(.all)
|
||||
```
|
||||
|
||||
Only flush the queue, leaving any currently displayed entry to its natural lifecycle:
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.dismiss(.queue)
|
||||
```
|
||||
|
||||
Dismiss a specific entry by name - either currently displayed or enqueued. All the entries with the given name are dismissed.
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.dismiss(.specific(entryName: "Entry Name"))
|
||||
```
|
||||
|
||||
Dismiss any entry with a lower or equal display priority of `.normal`.
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.dismiss(.prioritizedLowerOrEqualTo(priority: .normal))
|
||||
```
|
||||
|
||||
#### Using a completion handler
|
||||
|
||||
Inject a trailing closure to be executed after the entry dismissal.
|
||||
|
||||
```Swift
|
||||
SwiftEntryKit.dismiss {
|
||||
// Executed right after the entry has been dismissed
|
||||
}
|
||||
```
|
||||
|
||||
### Is Currently Displaying
|
||||
Inquire whether an entry is currently displayed:
|
||||
```Swift
|
||||
if SwiftEntryKit.isCurrentlyDisplaying {
|
||||
/* Do your things */
|
||||
}
|
||||
```
|
||||
|
||||
Inquire whether a **specific** entry is currently displayed using the `name` property inside `EKAttributes`.
|
||||
```Swift
|
||||
if SwiftEntryKit.isCurrentlyDisplaying(entryNamed: "Top Note") {
|
||||
/* Do your things */
|
||||
}
|
||||
```
|
||||
|
||||
### Queue Contains
|
||||
Inquire whether the queue of entries is not empty:
|
||||
```Swift
|
||||
if SwiftEntryKit.isQueueEmpty {
|
||||
/* Do your things */
|
||||
}
|
||||
```
|
||||
|
||||
Inquire whether the queue of entries contains an entry with name:
|
||||
```Swift
|
||||
if SwiftEntryKit.queueContains(entryNamed: "Custom-Name") {
|
||||
/* Do your things */
|
||||
}
|
||||
```
|
||||
|
||||
### Swiping and Rubber Banding
|
||||
Entries can be panned vertically (This ability can be enabled using the *scroll* attributes).
|
||||
Thefore it's only natural that an entry can be dismissed using a swipe-like gesture.
|
||||
|
||||
Enable swipe gesture. When the swipe gesture fails (doesn't pass the velocity threshold) ease it back.
|
||||
```Swift
|
||||
attributes.scroll = .enabled(swipeable: true, pullbackAnimation: .easeOut)
|
||||
```
|
||||
|
||||
Enable swipe gesture. When the swipe gesture fails throw it back out with a jolt.
|
||||
```Swift
|
||||
attributes.scroll = .enabled(swipeable: true, pullbackAnimation: .jolt)
|
||||
```
|
||||
|
||||
The *PullbackAnimation* values (duration, damping & initialSpringVelocity) can be customized as well.
|
||||
|
||||
Swipe | Jolt |
|
||||
--- | --- |
|
||||
 |  |
|
||||
|
||||
### Dealing with safe area:
|
||||
*EKAttributes.PositionConstraints.SafeArea* may be used to override the safe area with the entry's content, or to fill the safe area with a background color (like [Toasts](https://github.com/huri000/assets/blob/master/swift-entrykit/toasts.gif) do), or even leave the safe area empty (Like [Floats](https://github.com/huri000/assets/blob/master/swift-entrykit/floats.gif) do).
|
||||
|
||||
SwiftEntryKit supports iOS 11.x.y and is backward compatible to iOS 9.x.y, so the status bar area is treated as same as the safe area in earlier iOS versions.
|
||||
|
||||
### Dealing with orientation change:
|
||||
SwiftEntryKit identifies orientation changes and adjust the entry's layout to those changes.
|
||||
Therefore, if you wish to limit the entries's width, you are able to do so by giving it a maximum value, likewise:
|
||||
|
||||
```Swift
|
||||
var attributes = EKAttributes.topFloat
|
||||
|
||||
// Give the entry the width of the screen minus 20pts from each side, the height is decided by the content's contraint's
|
||||
attributes.positionConstraints.size = .init(width: .offset(value: 20), height: .intrinsic)
|
||||
|
||||
// Give the entry maximum width of the screen minimum edge - thus the entry won't grow much when the device orientation changes from portrait to landscape mode.
|
||||
let edgeWidth = min(UIScreen.main.bounds.width, UIScreen.main.bounds.height)
|
||||
attributes.positionConstraints.maxSize = .init(width: .constant(value: edgeWidth), height: .intrinsic)
|
||||
|
||||
let customView = UIView()
|
||||
/*
|
||||
... Customize the view as you like ...
|
||||
*/
|
||||
|
||||
// Use class method of SwiftEntryKit to display the view using the desired attributes
|
||||
SwiftEntryKit.display(entry: customView, using: attributes)
|
||||
```
|
||||
|
||||
Orientation Change Demonstration |
|
||||
--- |
|
||||

|
||||
|
||||
### Dark Mode in the Example Project
|
||||
|
||||
You can tinker with the display mode using a segmented control on presets screen, forcing light and dark modes.
|
||||
All the presets are dark mode ready, but only some in the example project demonstrate dark mode capabilities.
|
||||
|
||||

|
||||
|
||||
### Swift and Objective-C Interoperability
|
||||
SwiftEntryKit's APIs use the Swift language exclusive syntax (enums, associated values, and more).
|
||||
Therefore, `SwiftEntryKit` cannot be referenced directly from an Objective-C file (*.m*, *.h* or *.mm*).
|
||||
|
||||
Yet, it is pretty easy to integrate SwiftEntryKit into an Objective-C project using a simple *.swift* class that is a sort of adapter between `SwiftEntryKit` and your Objective-C code.
|
||||
|
||||
[This project](https://github.com/huri000/ObjcEntryKitExample) demonstrates that using Carthage and CocoaPods.
|
||||
|
||||
## Author
|
||||
|
||||
Daniel Huri, huri000@gmail.com
|
||||
|
||||
## Donations
|
||||
|
||||
Donations can be made by sending either Bitcoin or Ether to the following addresses.
|
||||
|
||||
| BTC | ETH |
|
||||
| :---: | :---: |
|
||||
| 134TiBiUvVNt7Na5KXEFBSChLdgVDw1Hnr | 0xAe6616181FCdde4793AE749Ce21Cd5Af9333A3E2 |
|
||||
|  |  |
|
||||
|
||||
## Thank You
|
||||
|
||||
Thanks Lily Azar, lilushkaa@gmail.com for those awesome preset icons.
|
||||
|
||||
## Credits
|
||||
|
||||
[**Icons Credits**](/CREDITS.md)
|
||||
|
||||
## License
|
||||
|
||||
SwiftEntryKit is available under the MIT license. See the [LICENSE](/LICENSE) file for more info.
|
||||
|
||||
### Exceptions
|
||||
Please be aware that any use of the icons inside the project requires attribution to the creator. See [credits](/CREDITS.md) for the creators list.
|
||||
14
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/QLCompatibility.swift
generated
Normal file
14
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/QLCompatibility.swift
generated
Normal file
@@ -0,0 +1,14 @@
|
||||
//
|
||||
// QLCompatibility.swift
|
||||
// Pods
|
||||
//
|
||||
// Created by Daniel Huri on 5/12/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
public typealias QLAttribute = NSLayoutConstraint.Attribute
|
||||
public typealias QLRelation = NSLayoutConstraint.Relation
|
||||
public typealias QLView = UIView
|
||||
public typealias QLPriority = UILayoutPriority
|
||||
107
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/QLUtils.swift
generated
Normal file
107
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/QLUtils.swift
generated
Normal file
@@ -0,0 +1,107 @@
|
||||
//
|
||||
// QLUtils.swift
|
||||
// QuickLayout
|
||||
//
|
||||
// Created by Daniel Huri on 11/21/17.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
/**
|
||||
Typealias for dictionary that contains multiple constraints
|
||||
*/
|
||||
public typealias QLMultipleConstraints = [QLAttribute: NSLayoutConstraint]
|
||||
|
||||
/**
|
||||
Extends layout priority to other readable types
|
||||
*/
|
||||
public extension QLPriority {
|
||||
static let must = QLPriority(rawValue: 999)
|
||||
static let zero = QLPriority(rawValue: 0)
|
||||
}
|
||||
|
||||
/**
|
||||
Represents pair of attributes
|
||||
*/
|
||||
public struct QLAttributePair {
|
||||
public let first: QLAttribute
|
||||
public let second: QLAttribute
|
||||
}
|
||||
|
||||
/**
|
||||
Represents size constraints
|
||||
*/
|
||||
public struct QLSizeConstraints {
|
||||
public let width: NSLayoutConstraint
|
||||
public let height: NSLayoutConstraint
|
||||
}
|
||||
|
||||
/**
|
||||
Represents center constraints
|
||||
*/
|
||||
public struct QLCenterConstraints {
|
||||
public let x: NSLayoutConstraint
|
||||
public let y: NSLayoutConstraint
|
||||
}
|
||||
|
||||
/**
|
||||
Represents axis constraints (might be .top and .bottom, .left and .right, .leading and .trailing)
|
||||
*/
|
||||
public struct QLAxisConstraints {
|
||||
public let first: NSLayoutConstraint
|
||||
public let second: NSLayoutConstraint
|
||||
}
|
||||
|
||||
/**
|
||||
Represents center and size constraints
|
||||
*/
|
||||
public struct QLFillConstraints {
|
||||
public let center: QLCenterConstraints
|
||||
public let size: QLSizeConstraints
|
||||
}
|
||||
|
||||
/**
|
||||
Represents pair of priorities
|
||||
*/
|
||||
public struct QLPriorityPair {
|
||||
|
||||
public let horizontal: QLPriority
|
||||
public let vertical: QLPriority
|
||||
public static var required: QLPriorityPair {
|
||||
return QLPriorityPair(.required, .required)
|
||||
}
|
||||
|
||||
public static var must: QLPriorityPair {
|
||||
return QLPriorityPair(.must, .must)
|
||||
}
|
||||
|
||||
public init(_ horizontal: QLPriority, _ vertical: QLPriority) {
|
||||
self.horizontal = horizontal
|
||||
self.vertical = vertical
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Represents axis description
|
||||
*/
|
||||
public enum QLAxis {
|
||||
case horizontally
|
||||
case vertically
|
||||
|
||||
public var attributes: QLAttributePair {
|
||||
|
||||
let first: QLAttribute
|
||||
let second: QLAttribute
|
||||
|
||||
switch self {
|
||||
case .horizontally:
|
||||
first = .left
|
||||
second = .right
|
||||
case .vertically:
|
||||
first = .top
|
||||
second = .bottom
|
||||
}
|
||||
return QLAttributePair(first: first, second: second)
|
||||
}
|
||||
}
|
||||
111
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/UIView+QLContentWrap.swift
generated
Normal file
111
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/UIView+QLContentWrap.swift
generated
Normal file
@@ -0,0 +1,111 @@
|
||||
//
|
||||
// QLView+QLContentWrap.swift
|
||||
// QuickLayout
|
||||
//
|
||||
// Created by Daniel Huri on 11/21/17.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
|
||||
// MARK: Content Compression Resistance & Content Hugging Priority
|
||||
public extension QLView {
|
||||
|
||||
/**
|
||||
Force hugging and compression resistance for the given axes, using variadic parameter.
|
||||
- parameter axes: The axes
|
||||
*/
|
||||
func forceContentWrap(_ axes: QLAxis...) {
|
||||
if axes.contains(.vertically) {
|
||||
verticalHuggingPriority = .required
|
||||
verticalCompressionResistancePriority = .required
|
||||
}
|
||||
if axes.contains(.horizontally) {
|
||||
horizontalHuggingPriority = .required
|
||||
horizontalCompressionResistancePriority = .required
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Force hugging and compression resistance vertically and horizontally.
|
||||
*/
|
||||
func forceContentWrap() {
|
||||
contentHuggingPriority = .required
|
||||
contentCompressionResistancePriority = .required
|
||||
}
|
||||
|
||||
/**
|
||||
Vertical hugging priority
|
||||
*/
|
||||
var verticalHuggingPriority: QLPriority {
|
||||
set {
|
||||
setContentHuggingPriority(newValue, for: .vertical)
|
||||
}
|
||||
get {
|
||||
return contentHuggingPriority(for: .vertical)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Horizontal hugging priority
|
||||
*/
|
||||
var horizontalHuggingPriority: QLPriority {
|
||||
set {
|
||||
setContentHuggingPriority(newValue, for: .horizontal)
|
||||
}
|
||||
get {
|
||||
return contentHuggingPriority(for: .horizontal)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Content hugging priority (Vertical & Horizontal)
|
||||
*/
|
||||
var contentHuggingPriority: QLPriorityPair {
|
||||
set {
|
||||
horizontalHuggingPriority = newValue.horizontal
|
||||
verticalHuggingPriority = newValue.vertical
|
||||
}
|
||||
get {
|
||||
return QLPriorityPair(horizontalHuggingPriority, verticalHuggingPriority)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Vertical content compression resistance priority
|
||||
*/
|
||||
var verticalCompressionResistancePriority: QLPriority {
|
||||
set {
|
||||
setContentCompressionResistancePriority(newValue, for: .vertical)
|
||||
}
|
||||
get {
|
||||
return contentCompressionResistancePriority(for: .vertical)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Horizontal content compression resistance priority
|
||||
*/
|
||||
var horizontalCompressionResistancePriority: QLPriority {
|
||||
set {
|
||||
setContentCompressionResistancePriority(newValue, for: .horizontal)
|
||||
}
|
||||
get {
|
||||
return contentCompressionResistancePriority(for: .horizontal)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Content compression resistance priority (Vertical & Horizontal)
|
||||
*/
|
||||
var contentCompressionResistancePriority: QLPriorityPair {
|
||||
set {
|
||||
horizontalCompressionResistancePriority = newValue.horizontal
|
||||
verticalCompressionResistancePriority = newValue.vertical
|
||||
}
|
||||
get {
|
||||
return QLPriorityPair(horizontalCompressionResistancePriority, verticalCompressionResistancePriority)
|
||||
}
|
||||
}
|
||||
}
|
||||
267
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/UIView+QuickLayout.swift
generated
Normal file
267
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/UIView+QuickLayout.swift
generated
Normal file
@@ -0,0 +1,267 @@
|
||||
//
|
||||
// QLView+QuickLayout.swift
|
||||
// QuickLayout
|
||||
//
|
||||
// Created by Daniel Huri on 11/19/17.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
public extension QLView {
|
||||
|
||||
/**
|
||||
Set constant value of an edge.
|
||||
Should be used with *width* or *height*
|
||||
- parameter edge: Edge type.
|
||||
- parameter value: Edge size.
|
||||
- parameter relation: Relation to the given constant value (default is *.equal*).
|
||||
- parameter ratio: Ratio of the cconstant constraint to actual given value (default is *1*)
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: The applied constraint (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func set(_ edge: QLAttribute, of value: CGFloat, relation: QLRelation = .equal,
|
||||
ratio: CGFloat = 1.0, priority: QLPriority = .required) -> NSLayoutConstraint {
|
||||
if translatesAutoresizingMaskIntoConstraints {
|
||||
translatesAutoresizingMaskIntoConstraints = false
|
||||
}
|
||||
let constraint = NSLayoutConstraint(item: self, attribute: edge, relatedBy: relation, toItem: nil, attribute: .notAnAttribute, multiplier: ratio, constant: value)
|
||||
constraint.priority = priority
|
||||
addConstraint(constraint)
|
||||
return constraint
|
||||
}
|
||||
|
||||
/**
|
||||
Set constant value for multiple edges simultaniously, using variadic parameter.
|
||||
Should be used with *width* or *height*
|
||||
- parameter edges: Edge types.
|
||||
- parameter value: Edges size.
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: The applied constraints in QLMultipleConstraints - see definition (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func set(_ edges: QLAttribute..., of value: CGFloat, relation: QLRelation = .equal,
|
||||
ratio: CGFloat = 1.0, priority: QLPriority = .required) -> QLMultipleConstraints {
|
||||
return set(edges, to: value, relation: relation, ratio: ratio, priority: priority)
|
||||
}
|
||||
|
||||
/** **PRIVATELY USED** AS A REPLACEMENT for the variadic version for the method*/
|
||||
@discardableResult
|
||||
func set(_ edges: [QLAttribute], to value: CGFloat, relation: QLRelation = .equal,
|
||||
ratio: CGFloat = 1.0, priority: QLPriority = .required) -> QLMultipleConstraints {
|
||||
var constraints: QLMultipleConstraints = [:]
|
||||
let uniqueEdges = Set(edges)
|
||||
for edge in uniqueEdges {
|
||||
let constraint = set(edge, of: value, priority: priority)
|
||||
constraints[edge] = constraint
|
||||
}
|
||||
return constraints
|
||||
}
|
||||
|
||||
/**
|
||||
Layout edge to another view's edge.
|
||||
- You can optionally define relation, ratio, constant and priority (each gets a default value)
|
||||
- For example - Can be used to align self *left* edge to the *right* of another view.
|
||||
- *self* and *view* must be directly connected (siblings / child-parent) in the view hierarchy.
|
||||
- *superview* must not be *nil*.
|
||||
- parameter edge: The edge of the first view. If not sent or *nil* - The function automatically assumes *edge* to be *otherEdge*
|
||||
- parameter otherEdge: The edge of the second view.
|
||||
- parameter view: The second view that self must be aligned with.
|
||||
- parameter relation: The relation of the first edge to the second edge (default is .equal)
|
||||
- parameter ratio: The ratio of the edge in relative to the superview edge (default is 1).
|
||||
- parameter offset: Additional offset which is applied to the constraint (default is 0).
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: The instance of the constraint that was applied (discardable). nil if method failed to apply the constraint.
|
||||
*/
|
||||
@discardableResult
|
||||
func layout(_ edge: QLAttribute? = nil, to otherEdge: QLAttribute, of view: QLView,
|
||||
relation: QLRelation = .equal, ratio: CGFloat = 1.0, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> NSLayoutConstraint? {
|
||||
guard isValidForQuickLayout else {
|
||||
print("\(String(describing: self)) Error in func: \(#function)")
|
||||
return nil
|
||||
}
|
||||
let constraint = NSLayoutConstraint(item: self, attribute: edge ?? otherEdge, relatedBy: relation, toItem: view, attribute: otherEdge, multiplier: ratio, constant: offset)
|
||||
constraint.priority = priority
|
||||
superview!.addConstraint(constraint)
|
||||
return constraint
|
||||
}
|
||||
|
||||
/**
|
||||
Layout multiple edges of the view to the corresonding edges of another given view.
|
||||
- You can optionally define relation, ratio, constant and priority (each gets a default value)
|
||||
- For example - Can be used to align self *left* and *right* edges the same edge of another given view.
|
||||
- *self* and *view* must be directly connected (siblings / child-parent) in the view hierarchy.
|
||||
- *superview* must not be *nil*.
|
||||
- parameter edges: The view edges
|
||||
- parameter view: Another view that self must be aligned with.
|
||||
- parameter relation: The relation of the edges. Can be applied to *.width* or *height* for example. (default is *.equal*).
|
||||
- parameter ratio: The ratio of the edges to the other view edges (default is 1).
|
||||
- parameter offset: Additional offset which is applied to each of the constraints (default is 0).
|
||||
- parameter priority: Constraints' priority (default is *.required*).
|
||||
- returns: The instance of the constraint that was applied (discardable). *nil* if the method failed to apply the constraint.
|
||||
*/
|
||||
@discardableResult
|
||||
func layout(_ edges: QLAttribute..., to view: QLView, relation: QLRelation = .equal,
|
||||
ratio: CGFloat = 1.0, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> QLMultipleConstraints {
|
||||
var constraints: QLMultipleConstraints = [:]
|
||||
guard isValidForQuickLayout else {
|
||||
print("\(String(describing: self)) Error in func: \(#function)")
|
||||
return constraints
|
||||
}
|
||||
let uniqueEdges = Set(edges)
|
||||
for edge in uniqueEdges {
|
||||
let constraint = NSLayoutConstraint(item: self, attribute: edge, relatedBy: relation, toItem: view, attribute: edge, multiplier: ratio, constant: offset)
|
||||
constraint.priority = priority
|
||||
superview!.addConstraint(constraint)
|
||||
constraints[edge] = constraint
|
||||
}
|
||||
return constraints
|
||||
}
|
||||
|
||||
/**
|
||||
Layout edge to the same edge of superview.
|
||||
- Example of usage: *view.layoutToSuperview(.top)* makes *view* cling to the *top* of it's *superview*.
|
||||
- You can optionally define ratio, constant and priority (each gets a default value)
|
||||
- *superview* must not be *nil*.
|
||||
- parameter edge: The edge (.width, .height, .left, .right, .leading, .trailing, etc...)
|
||||
- parameter relation: The relation of the edge to the superview's corresponding edge (default is *.equal*)
|
||||
- parameter ratio: The ratio of the edge in relative to the superview edge (default is 1).
|
||||
- parameter offset: Additional offset from that can be applied to the constraint (default is 0).
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: The instance of the constraint that was applied (discardable). Nil if method failed to apply constraint.
|
||||
*/
|
||||
@discardableResult
|
||||
func layoutToSuperview(_ edge: QLAttribute, relation: QLRelation = .equal,
|
||||
ratio: CGFloat = 1, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> NSLayoutConstraint? {
|
||||
guard isValidForQuickLayout else {
|
||||
print("\(String(describing: self)) Error in func: \(#function)")
|
||||
return nil
|
||||
}
|
||||
let constraint = NSLayoutConstraint(item: self, attribute: edge, relatedBy: relation, toItem: superview, attribute: edge, multiplier: ratio, constant: offset)
|
||||
constraint.priority = priority
|
||||
superview!.addConstraint(constraint)
|
||||
return constraint
|
||||
}
|
||||
|
||||
/**
|
||||
Layout multiple edges to the same edges as superview, using variadic parameter.
|
||||
Example for edges value:
|
||||
- You can optionally define ratio, constant and priority (each gets a default value)
|
||||
- *superview* must not be *nil*.
|
||||
- parameter edges: The edges (.width, .height, .left, .right, .leading, .trailing, etc...)
|
||||
- parameter relation: The relation of the edges to the superview's corresponding edges (default is *.equal*)
|
||||
- parameter ratio: The ratio of the edges in relative to the superview edge (default is 1).
|
||||
- parameter offset: Additional offset from that can be applied to the constraints (default is 0).
|
||||
- parameter priority: Constraints' priority (default is *.required*).
|
||||
- returns: The instance of QLMultipleConstraints - see type definition (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func layoutToSuperview(_ edges: QLAttribute..., relation: QLRelation = .equal,
|
||||
ratio: CGFloat = 1, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> QLMultipleConstraints {
|
||||
var constraints: QLMultipleConstraints = [:]
|
||||
guard !edges.isEmpty && isValidForQuickLayout else {
|
||||
return constraints
|
||||
}
|
||||
let uniqueEdges = Set(edges)
|
||||
for edge in uniqueEdges {
|
||||
let constraint = NSLayoutConstraint(item: self, attribute: edge, relatedBy: relation, toItem: superview, attribute: edge, multiplier: ratio, constant: offset)
|
||||
constraint.priority = priority
|
||||
superview!.addConstraint(constraint)
|
||||
constraints[edge] = constraint
|
||||
}
|
||||
return constraints
|
||||
}
|
||||
|
||||
/**
|
||||
Layout to one of the superview's axes.
|
||||
- You can optionally define ratio, constant and priority (each gets a default value)
|
||||
- *superview* must not be *nil*.
|
||||
- parameter axis: The axis to which the view must be stretched (horizontally or vertically)
|
||||
- parameter offset: Represents an additional edge offset from that can be applied to the constraints (default is 0)
|
||||
- parameter priority: Represents constraint's priority (default is *.required*)
|
||||
- returns: The instance of the constraint that was applied (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func layoutToSuperview(axis: QLAxis, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> QLAxisConstraints? {
|
||||
let attributes = axis.attributes
|
||||
guard let first = layoutToSuperview(attributes.first, offset: offset, priority: priority) else {
|
||||
return nil
|
||||
}
|
||||
guard let second = layoutToSuperview(attributes.second, offset: -offset, priority: priority) else {
|
||||
return nil
|
||||
}
|
||||
return QLAxisConstraints(first: first, second: second)
|
||||
}
|
||||
|
||||
/**
|
||||
Size to superview with a given ratio and constant
|
||||
- *superview* must not be *nil*.
|
||||
- parameter ratio: The ratio of view to the size of superview.
|
||||
- parameter offset: Represents an additional edge offset from that can be applied to the size (default is 0)
|
||||
- parameter priority: Represents constraint's priority (default is *.required*)
|
||||
- returns: The instance of QLSizeConstraints - see definition (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func sizeToSuperview(withRatio ratio: CGFloat = 1, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> QLSizeConstraints? {
|
||||
let size = layoutToSuperview(.width, .height, ratio: ratio, offset: offset, priority: priority)
|
||||
guard !size.isEmpty else {
|
||||
return nil
|
||||
}
|
||||
return QLSizeConstraints(width: size[.width]!, height: size[.height]!)
|
||||
}
|
||||
|
||||
/**
|
||||
Center in superview with an optional offset
|
||||
- *superview* must not be *nil*.
|
||||
- parameter offset: Represents an additional offset from the center (default is 0)
|
||||
- parameter priority: Represents constraint's priority (default is *.required*)
|
||||
- returns: The instance of QLCenterConstraints - see definition (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func centerInSuperview(offset: CGFloat = 0, priority: QLPriority = .required) -> QLCenterConstraints? {
|
||||
let center = layoutToSuperview(.centerX, .centerY, offset: offset)
|
||||
guard !center.isEmpty else {
|
||||
return nil
|
||||
}
|
||||
return QLCenterConstraints(x: center[.centerX]!, y: center[.centerY]!)
|
||||
}
|
||||
|
||||
/**
|
||||
Fill superview totally (center and size to superview)
|
||||
- *superview* must not be *nil*.
|
||||
- parameter ratio: Ratio to the superview's size (default is 1)
|
||||
- parameter offset: Offset from center (default is 0)
|
||||
- parameter priority: Represents constraint's priority (default is *.required*)
|
||||
- returns: The instance of QLFillConstraints - see definition (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func fillSuperview(withSizeRatio ratio: CGFloat = 1, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> QLFillConstraints? {
|
||||
guard let center = centerInSuperview(priority: priority) else {
|
||||
return nil
|
||||
}
|
||||
guard let size = sizeToSuperview(withRatio: ratio, offset: offset, priority: priority) else {
|
||||
return nil
|
||||
}
|
||||
return QLFillConstraints(center: center, size: size)
|
||||
}
|
||||
|
||||
/** **PRIVATELY USED** to test for validation*/
|
||||
var isValidForQuickLayout: Bool {
|
||||
guard superview != nil else {
|
||||
print("\(String(describing: self)):\(#function) - superview is unexpectedly nullified")
|
||||
return false
|
||||
}
|
||||
if translatesAutoresizingMaskIntoConstraints {
|
||||
translatesAutoresizingMaskIntoConstraints = false
|
||||
}
|
||||
return true
|
||||
}
|
||||
}
|
||||
214
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/UIViewArray+QuickLayout.swift
generated
Normal file
214
Pods/SwiftEntryKit/Source/Extensions/QuickLayout/UIViewArray+QuickLayout.swift
generated
Normal file
@@ -0,0 +1,214 @@
|
||||
//
|
||||
// QLViewArray+QuickLayout.swift
|
||||
// QuickLayout
|
||||
//
|
||||
// Created by Daniel Huri on 11/20/17.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
// MARK: Multiple Views in Array
|
||||
public extension Array where Element: QLView {
|
||||
|
||||
/**
|
||||
All elements in the collection recieve constant value for the given edge.
|
||||
- parameter edge: Should be used with *.width* or *.height*.
|
||||
- parameter value: The size of the edge.
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: The instance of the constraint that was applied (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func set(_ edge: QLAttribute, of value: CGFloat,
|
||||
priority: QLPriority = .required) -> [NSLayoutConstraint] {
|
||||
var constraints: [NSLayoutConstraint] = []
|
||||
for view in self {
|
||||
let constraint = view.set(edge, of: value)
|
||||
constraints.append(constraint)
|
||||
}
|
||||
return constraints
|
||||
}
|
||||
|
||||
/**
|
||||
All elements in the collection recieve constant value for the given edges, using variadic parameter.
|
||||
- parameter edges: Should be used with *.width* or *.height*.
|
||||
- parameter value: The size of the edge.
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: The instance of the constraint that was applied (discardable).
|
||||
*/
|
||||
@discardableResult
|
||||
func set(_ edges: QLAttribute..., of value: CGFloat,
|
||||
priority: QLPriority = .required) -> [QLMultipleConstraints] {
|
||||
var constraintsArray: [QLMultipleConstraints] = []
|
||||
for view in self {
|
||||
let constraints = view.set(edges, to: value, priority: priority)
|
||||
constraintsArray.append(constraints)
|
||||
}
|
||||
return constraintsArray
|
||||
}
|
||||
|
||||
/**
|
||||
Spread elements consecutively according to the given axis.
|
||||
- parameter axis: The axis: *.vertically*, *horizontally*
|
||||
- parameter stretchEdgesToSuperview: Decides whether the first and last items in the array must be clipped to their parent edges.
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: Array of constraints that were applied (discardable)
|
||||
*/
|
||||
@discardableResult
|
||||
func spread(_ axis: QLAxis, stretchEdgesToSuperview: Bool = false, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> [NSLayoutConstraint] {
|
||||
guard isValidForQuickLayout else {
|
||||
return []
|
||||
}
|
||||
let attributes = axis.attributes
|
||||
var constraints: [NSLayoutConstraint] = []
|
||||
|
||||
if stretchEdgesToSuperview {
|
||||
let constraint = first!.layoutToSuperview(attributes.first, offset: offset)!
|
||||
constraints.append(constraint)
|
||||
}
|
||||
|
||||
for (index, view) in enumerated() {
|
||||
guard index > 0 else {
|
||||
continue
|
||||
}
|
||||
let previousView = self[index - 1]
|
||||
let constraint = view.layout(attributes.first, to: attributes.second, of: previousView, offset: offset, priority: priority)!
|
||||
constraints.append(constraint)
|
||||
}
|
||||
|
||||
if stretchEdgesToSuperview {
|
||||
let constraint = last!.layoutToSuperview(attributes.second, offset: -offset)!
|
||||
constraints.append(constraint)
|
||||
}
|
||||
|
||||
return constraints
|
||||
}
|
||||
|
||||
/**
|
||||
Layout elements to superview's axis
|
||||
- parameter axis: The axis: *.vertically*, *horizontally*
|
||||
- parameter offset: Additional side offset that must be applied (identical spacing from each side)
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: Array of QLAxisConstraints - see definition (discardable)
|
||||
*/
|
||||
@discardableResult
|
||||
func layoutToSuperview(axis: QLAxis, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> [QLAxisConstraints] {
|
||||
|
||||
let attributes = axis.attributes
|
||||
|
||||
let firstConstraints = layoutToSuperview(attributes.first, offset: offset, priority: priority)
|
||||
guard !firstConstraints.isEmpty else {
|
||||
return []
|
||||
}
|
||||
|
||||
let secondConstraints = layoutToSuperview(attributes.second, offset: -offset, priority: priority)
|
||||
guard !secondConstraints.isEmpty else {
|
||||
return []
|
||||
}
|
||||
|
||||
var constraints: [QLAxisConstraints] = []
|
||||
for (first, second) in zip(firstConstraints, secondConstraints) {
|
||||
constraints.append(QLAxisConstraints(first: first, second: second))
|
||||
}
|
||||
|
||||
return constraints
|
||||
}
|
||||
|
||||
/**
|
||||
Layout elements' edges to superview's edge (The same edge - top to top, bottom to bottom, etc...)
|
||||
- parameter edge: The edge of the view / superview
|
||||
- parameter ratio: The ratio of the edge in relation to the superview's (default is 1).
|
||||
- parameter offset: Additional offset from that must be applied to the constraint (default is 0).
|
||||
- parameter priority: Constraint's priority (default is *.required*).
|
||||
- returns: Array of applied constraints - see definition (discardable)
|
||||
*/
|
||||
@discardableResult
|
||||
func layoutToSuperview(_ edge: QLAttribute, ratio: CGFloat = 1, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> [NSLayoutConstraint] {
|
||||
guard isValidForQuickLayout else {
|
||||
return []
|
||||
}
|
||||
return layout(to: edge, of: first!.superview!, ratio: ratio, offset: offset, priority: priority)
|
||||
}
|
||||
|
||||
/**
|
||||
Layout elements' edges to to anchorView edge
|
||||
- parameter firstEdge: The edge of the elements in the array
|
||||
- parameter anchorEdge: The edge of the anchor view
|
||||
- parameter anchorView: The anchor view
|
||||
- parameter ratio: The ratio of the edge in relative to the superview edge (default is 1).
|
||||
- parameter offset: Additional offset from that can be applied to the constraints (default is 0).
|
||||
- parameter priority: Constraints' priority (default is *.required*).
|
||||
- returns: Array of applied constraints - see definition (discardable)
|
||||
*/
|
||||
@discardableResult
|
||||
func layout(_ firstEdge: QLAttribute? = nil, to anchorEdge: QLAttribute,
|
||||
of anchorView: QLView, ratio: CGFloat = 1, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> [NSLayoutConstraint] {
|
||||
guard isValidForQuickLayout else {
|
||||
return []
|
||||
}
|
||||
|
||||
let edge: QLAttribute
|
||||
if let firstEdge = firstEdge {
|
||||
edge = firstEdge
|
||||
} else {
|
||||
edge = anchorEdge
|
||||
}
|
||||
|
||||
var result: [NSLayoutConstraint] = []
|
||||
for view in self {
|
||||
let constraint = view.layout(edge, to: anchorEdge, of: anchorView, ratio: ratio, offset: offset, priority: priority)!
|
||||
result.append(constraint)
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
/**
|
||||
Layout elements' multiple edges to to anchorView's same edges (top to top, bottom to bottom, etc...)
|
||||
- parameter edges: The edges of the view - variadic parameter
|
||||
- parameter anchorView: The anchor view
|
||||
- parameter ratio: The ratio of the edge in relative to the superview edge (default is 1).
|
||||
- parameter offset: Additional offset from that can be applied to the constraints (default is 0).
|
||||
- parameter priority: Constraints' priority (default is *.required*).
|
||||
- returns: Array of applied constraints, each element is of type QLMultipleConstraints - see definition (discardable)
|
||||
*/
|
||||
@discardableResult
|
||||
func layout(_ edges: QLAttribute..., to anchorView: QLView,
|
||||
ratio: CGFloat = 1, offset: CGFloat = 0,
|
||||
priority: QLPriority = .required) -> [QLMultipleConstraints] {
|
||||
guard !edges.isEmpty && isValidForQuickLayout else {
|
||||
return []
|
||||
}
|
||||
// Avoid duplicities
|
||||
let uniqueEdges = Set(edges)
|
||||
var result: [QLMultipleConstraints] = []
|
||||
for view in self {
|
||||
var multipleConstraints: QLMultipleConstraints = [:]
|
||||
for edge in uniqueEdges {
|
||||
let constraint = view.layout(to: edge, of: anchorView, ratio: ratio, offset: offset, priority: priority)!
|
||||
multipleConstraints[edge] = constraint
|
||||
}
|
||||
result.append(multipleConstraints)
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
/** **PRIVATELY USED** to test for validation*/
|
||||
var isValidForQuickLayout: Bool {
|
||||
guard !isEmpty else {
|
||||
print("\(String(describing: self)) Error in func: \(#function), Views collection is empty!")
|
||||
return false
|
||||
}
|
||||
|
||||
for view in self {
|
||||
guard view.isValidForQuickLayout else {
|
||||
print("\(String(describing: self)) Error in func: \(#function)")
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}
|
||||
}
|
||||
18
Pods/SwiftEntryKit/Source/Extensions/UIApplication+EKAppearance.swift
generated
Normal file
18
Pods/SwiftEntryKit/Source/Extensions/UIApplication+EKAppearance.swift
generated
Normal file
@@ -0,0 +1,18 @@
|
||||
//
|
||||
// UIApplication+EKAppearance.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/25/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
extension UIApplication {
|
||||
|
||||
func set(statusBarStyle: EKAttributes.StatusBar) {
|
||||
let appearance = statusBarStyle.appearance
|
||||
UIApplication.shared.isStatusBarHidden = !appearance.visible
|
||||
UIApplication.shared.statusBarStyle = appearance.style
|
||||
}
|
||||
}
|
||||
28
Pods/SwiftEntryKit/Source/Extensions/UIColor+Utils.swift
generated
Normal file
28
Pods/SwiftEntryKit/Source/Extensions/UIColor+Utils.swift
generated
Normal file
@@ -0,0 +1,28 @@
|
||||
//
|
||||
// UIColor+Utils.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel on 21/07/2019.
|
||||
// Copyright © 2019 CocoaPods. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
extension UIColor {
|
||||
convenience init(red: Int, green: Int, blue: Int) {
|
||||
assert(red >= 0 && red <= 255, "Invalid red component")
|
||||
assert(green >= 0 && green <= 255, "Invalid green component")
|
||||
assert(blue >= 0 && blue <= 255, "Invalid blue component")
|
||||
|
||||
self.init(red: CGFloat(red) / 255.0, green: CGFloat(green) / 255.0, blue: CGFloat(blue) / 255.0, alpha: 1.0)
|
||||
}
|
||||
|
||||
convenience init(rgb: Int) {
|
||||
self.init(
|
||||
red: (rgb >> 16) & 0xFF,
|
||||
green: (rgb >> 8) & 0xFF,
|
||||
blue: rgb & 0xFF
|
||||
)
|
||||
}
|
||||
}
|
||||
14
Pods/SwiftEntryKit/Source/Extensions/UIEdgeInsets+Utils.swift
generated
Normal file
14
Pods/SwiftEntryKit/Source/Extensions/UIEdgeInsets+Utils.swift
generated
Normal file
@@ -0,0 +1,14 @@
|
||||
//
|
||||
// UIEdgeInsets.swift
|
||||
// FBSnapshotTestCase
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
extension UIEdgeInsets {
|
||||
var hasVerticalInsets: Bool {
|
||||
return top > 0 || bottom > 0
|
||||
}
|
||||
}
|
||||
15
Pods/SwiftEntryKit/Source/Extensions/UIRectCorner+Short.swift
generated
Normal file
15
Pods/SwiftEntryKit/Source/Extensions/UIRectCorner+Short.swift
generated
Normal file
@@ -0,0 +1,15 @@
|
||||
//
|
||||
// UIView+FrameStyle.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/24/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
extension UIRectCorner {
|
||||
static let top: UIRectCorner = [.topLeft, .topRight]
|
||||
static let bottom: UIRectCorner = [.bottomLeft, .bottomRight]
|
||||
static let none: UIRectCorner = []
|
||||
}
|
||||
46
Pods/SwiftEntryKit/Source/Extensions/UIView+Shadow.swift
generated
Normal file
46
Pods/SwiftEntryKit/Source/Extensions/UIView+Shadow.swift
generated
Normal file
@@ -0,0 +1,46 @@
|
||||
//
|
||||
// UIView+Shadow.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/25/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
extension UIView {
|
||||
func applyDropShadow(withOffset offset: CGSize,
|
||||
opacity: Float,
|
||||
radius: CGFloat,
|
||||
color: UIColor) {
|
||||
layer.applyDropShadow(withOffset: offset,
|
||||
opacity: opacity,
|
||||
radius: radius,
|
||||
color: color)
|
||||
}
|
||||
|
||||
func removeDropShadow() {
|
||||
layer.removeDropShadow()
|
||||
}
|
||||
}
|
||||
|
||||
extension CALayer {
|
||||
func applyDropShadow(withOffset offset: CGSize,
|
||||
opacity: Float,
|
||||
radius: CGFloat,
|
||||
color: UIColor) {
|
||||
shadowOffset = offset
|
||||
shadowOpacity = opacity
|
||||
shadowRadius = radius
|
||||
shadowColor = color.cgColor
|
||||
shouldRasterize = true
|
||||
rasterizationScale = UIScreen.main.scale
|
||||
}
|
||||
|
||||
func removeDropShadow() {
|
||||
shadowOffset = .zero
|
||||
shadowOpacity = 0
|
||||
shadowRadius = 0
|
||||
shadowColor = UIColor.clear.cgColor
|
||||
shouldRasterize = false
|
||||
}
|
||||
}
|
||||
149
Pods/SwiftEntryKit/Source/Extensions/UIView+Utils.swift
generated
Normal file
149
Pods/SwiftEntryKit/Source/Extensions/UIView+Utils.swift
generated
Normal file
@@ -0,0 +1,149 @@
|
||||
//
|
||||
// UILabel+Message.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 04/14/2018.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
extension UILabel {
|
||||
var style: EKProperty.LabelStyle {
|
||||
set {
|
||||
font = newValue.font
|
||||
textColor = newValue.color(for: traitCollection)
|
||||
textAlignment = newValue.alignment
|
||||
numberOfLines = newValue.numberOfLines
|
||||
}
|
||||
get {
|
||||
return EKProperty.LabelStyle(font: font,
|
||||
color: EKColor(textColor),
|
||||
alignment: textAlignment,
|
||||
numberOfLines: numberOfLines)
|
||||
}
|
||||
}
|
||||
|
||||
var content: EKProperty.LabelContent {
|
||||
set {
|
||||
text = newValue.text
|
||||
accessibilityIdentifier = newValue.accessibilityIdentifier
|
||||
style = newValue.style
|
||||
}
|
||||
get {
|
||||
return EKProperty.LabelContent(text: text ?? "", style: style)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
extension UIButton {
|
||||
var buttonContent: EKProperty.ButtonContent {
|
||||
set {
|
||||
setTitle(newValue.label.text, for: .normal)
|
||||
setTitleColor(newValue.label.style.color(for: traitCollection), for: .normal)
|
||||
titleLabel?.font = newValue.label.style.font
|
||||
accessibilityIdentifier = newValue.accessibilityIdentifier
|
||||
backgroundColor = newValue.backgroundColor.color(
|
||||
for: traitCollection,
|
||||
mode: newValue.displayMode
|
||||
)
|
||||
}
|
||||
get {
|
||||
fatalError("buttonContent doesn't have a getter")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
extension UIImageView {
|
||||
var imageContent: EKProperty.ImageContent {
|
||||
set {
|
||||
stopAnimating()
|
||||
if newValue.images.count == 1 {
|
||||
image = newValue.images.first
|
||||
} else {
|
||||
animationImages = newValue.images.map {
|
||||
$0.withRenderingMode(.alwaysTemplate)
|
||||
}
|
||||
animationDuration = newValue.imageSequenceAnimationDuration
|
||||
}
|
||||
|
||||
contentMode = newValue.contentMode
|
||||
tintColor = newValue.tint?.color(for: traitCollection,
|
||||
mode: newValue.displayMode)
|
||||
accessibilityIdentifier = newValue.accessibilityIdentifier
|
||||
|
||||
if let size = newValue.size {
|
||||
set(.width, of: size.width)
|
||||
set(.height, of: size.height)
|
||||
} else {
|
||||
forceContentWrap()
|
||||
}
|
||||
|
||||
if newValue.makesRound {
|
||||
clipsToBounds = true
|
||||
if let size = newValue.size {
|
||||
layer.cornerRadius = min(size.width, size.height) * 0.5
|
||||
} else {
|
||||
layoutIfNeeded()
|
||||
layer.cornerRadius = min(bounds.width, bounds.height) * 0.5
|
||||
}
|
||||
}
|
||||
|
||||
startAnimating()
|
||||
|
||||
if case .animate(duration: let duration,
|
||||
options: let options,
|
||||
transform: let transform) = newValue.animation {
|
||||
let options: UIView.AnimationOptions = [.repeat, .autoreverse, options]
|
||||
// A hack that forces the animation to run on the main thread,
|
||||
// on one of the next run loops
|
||||
DispatchQueue.main.async {
|
||||
UIView.animate(withDuration: duration,
|
||||
delay: 0,
|
||||
options: options,
|
||||
animations: {
|
||||
self.transform = transform
|
||||
}, completion: nil)
|
||||
}
|
||||
}
|
||||
}
|
||||
get {
|
||||
fatalError("imageContent doesn't have a getter")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
extension UITextField {
|
||||
|
||||
var placeholder: EKProperty.LabelContent {
|
||||
set {
|
||||
attributedPlaceholder = NSAttributedString(
|
||||
string: newValue.text,
|
||||
attributes: [
|
||||
.font: newValue.style.font,
|
||||
.foregroundColor: newValue.style.color(for: traitCollection)
|
||||
]
|
||||
)
|
||||
}
|
||||
get {
|
||||
fatalError("placeholder doesn't have a getter")
|
||||
}
|
||||
}
|
||||
|
||||
var textFieldContent: EKProperty.TextFieldContent {
|
||||
set {
|
||||
placeholder = newValue.placeholder
|
||||
keyboardType = newValue.keyboardType
|
||||
textColor = newValue.textStyle.color(for: traitCollection)
|
||||
font = newValue.textStyle.font
|
||||
textAlignment = newValue.textStyle.alignment
|
||||
isSecureTextEntry = newValue.isSecure
|
||||
text = newValue.textContent
|
||||
tintColor = newValue.tintColor(for: traitCollection)
|
||||
accessibilityIdentifier = newValue.accessibilityIdentifier
|
||||
}
|
||||
get {
|
||||
fatalError("textFieldContent doesn't have a getter")
|
||||
}
|
||||
}
|
||||
}
|
||||
92
Pods/SwiftEntryKit/Source/Infra/EKBackgroundView.swift
generated
Normal file
92
Pods/SwiftEntryKit/Source/Infra/EKBackgroundView.swift
generated
Normal file
@@ -0,0 +1,92 @@
|
||||
//
|
||||
// EKBackgroundView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/20/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final class EKBackgroundView: EKStyleView {
|
||||
|
||||
struct Style {
|
||||
let background: EKAttributes.BackgroundStyle
|
||||
let displayMode: EKAttributes.DisplayMode
|
||||
}
|
||||
|
||||
// MARK: Props
|
||||
private let visualEffectView: UIVisualEffectView
|
||||
private let imageView: UIImageView
|
||||
private let gradientView: GradientView
|
||||
|
||||
// MARK: Setup
|
||||
init() {
|
||||
imageView = UIImageView()
|
||||
visualEffectView = UIVisualEffectView(effect: nil)
|
||||
gradientView = GradientView()
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
|
||||
addSubview(imageView)
|
||||
imageView.contentMode = .scaleAspectFill
|
||||
imageView.fillSuperview()
|
||||
|
||||
addSubview(visualEffectView)
|
||||
visualEffectView.fillSuperview()
|
||||
|
||||
addSubview(gradientView)
|
||||
gradientView.fillSuperview()
|
||||
}
|
||||
|
||||
required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
// Background setter
|
||||
var style: Style! {
|
||||
didSet {
|
||||
guard let style = style else {
|
||||
return
|
||||
}
|
||||
var gradient: EKAttributes.BackgroundStyle.Gradient?
|
||||
var backgroundEffect: UIBlurEffect?
|
||||
var backgroundColor: UIColor = .clear
|
||||
var backgroundImage: UIImage?
|
||||
|
||||
switch style.background {
|
||||
case .color(color: let color):
|
||||
backgroundColor = color.color(for: traitCollection,
|
||||
mode: style.displayMode)
|
||||
case .gradient(gradient: let value):
|
||||
gradient = value
|
||||
case .image(image: let image):
|
||||
backgroundImage = image
|
||||
case .visualEffect(style: let value):
|
||||
backgroundEffect = value.blurEffect(for: traitCollection,
|
||||
mode: style.displayMode)
|
||||
case .clear:
|
||||
break
|
||||
}
|
||||
|
||||
gradientView.style = GradientView.Style(gradient: gradient,
|
||||
displayMode: style.displayMode)
|
||||
visualEffectView.effect = backgroundEffect
|
||||
layer.backgroundColor = backgroundColor.cgColor
|
||||
imageView.image = backgroundImage
|
||||
}
|
||||
}
|
||||
|
||||
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
guard let style = style else { return }
|
||||
switch style.background {
|
||||
case .color(color: let color):
|
||||
layer.backgroundColor = color.color(for: traitCollection,
|
||||
mode: style.displayMode).cgColor
|
||||
case .visualEffect(style: let value):
|
||||
visualEffectView.effect = value.blurEffect(for: traitCollection,
|
||||
mode: style.displayMode)
|
||||
default:
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
752
Pods/SwiftEntryKit/Source/Infra/EKContentView.swift
generated
Normal file
752
Pods/SwiftEntryKit/Source/Infra/EKContentView.swift
generated
Normal file
@@ -0,0 +1,752 @@
|
||||
//
|
||||
// EKScrollView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
protocol EntryContentViewDelegate: AnyObject {
|
||||
func changeToActive(withAttributes attributes: EKAttributes)
|
||||
func changeToInactive(withAttributes attributes: EKAttributes, pushOut: Bool)
|
||||
func didFinishDisplaying(entry: EKEntryView, keepWindowActive: Bool, dismissCompletionHandler: SwiftEntryKit.DismissCompletionHandler?)
|
||||
}
|
||||
|
||||
class EKContentView: UIView {
|
||||
|
||||
enum OutTranslation {
|
||||
case exit
|
||||
case pop
|
||||
case swipeDown
|
||||
case swipeUp
|
||||
}
|
||||
|
||||
struct OutTranslationAnchor {
|
||||
var messageOut: QLAttribute
|
||||
var screenOut: QLAttribute
|
||||
|
||||
init(_ messageOut: QLAttribute, to screenOut: QLAttribute) {
|
||||
self.messageOut = messageOut
|
||||
self.screenOut = screenOut
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Props
|
||||
|
||||
// Entry delegate
|
||||
private weak var entryDelegate: EntryContentViewDelegate!
|
||||
|
||||
// Constraints and Offsets
|
||||
private var entranceOutConstraint: NSLayoutConstraint!
|
||||
private var exitOutConstraint: NSLayoutConstraint!
|
||||
private var swipeDownOutConstraint: NSLayoutConstraint!
|
||||
private var swipeUpOutConstraint: NSLayoutConstraint!
|
||||
private var popOutConstraint: NSLayoutConstraint!
|
||||
private var inConstraint: NSLayoutConstraint!
|
||||
private var resistanceConstraint: NSLayoutConstraint!
|
||||
private var inKeyboardConstraint: NSLayoutConstraint!
|
||||
|
||||
private var inOffset: CGFloat = 0
|
||||
private var totalTranslation: CGFloat = 0
|
||||
private var verticalLimit: CGFloat = 0
|
||||
private let swipeMinVelocity: CGFloat = 60
|
||||
|
||||
private var outDispatchWorkItem: DispatchWorkItem!
|
||||
|
||||
private var keyboardState = KeyboardState.hidden
|
||||
|
||||
// Dismissal handler
|
||||
var dismissHandler: SwiftEntryKit.DismissCompletionHandler?
|
||||
|
||||
// Data source
|
||||
private var attributes: EKAttributes {
|
||||
return contentView.attributes
|
||||
}
|
||||
|
||||
// Content
|
||||
private var contentView: EKEntryView!
|
||||
|
||||
// MARK: Setup
|
||||
required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
init(withEntryDelegate entryDelegate: EntryContentViewDelegate) {
|
||||
self.entryDelegate = entryDelegate
|
||||
super.init(frame: .zero)
|
||||
}
|
||||
|
||||
// Called from outer scope with a presentable view and attributes
|
||||
func setup(with contentView: EKEntryView) {
|
||||
|
||||
self.contentView = contentView
|
||||
|
||||
// Execute willAppear lifecycle action if needed
|
||||
contentView.attributes.lifecycleEvents.willAppear?()
|
||||
|
||||
// Setup attributes
|
||||
setupAttributes()
|
||||
|
||||
// Setup initial position
|
||||
setupInitialPosition()
|
||||
|
||||
// Setup width, height and maximum width
|
||||
setupLayoutConstraints()
|
||||
|
||||
// Animate in
|
||||
animateIn()
|
||||
|
||||
// Setup tap gesture
|
||||
setupTapGestureRecognizer()
|
||||
|
||||
// Generate haptic feedback
|
||||
generateHapticFeedback()
|
||||
|
||||
setupKeyboardChangeIfNeeded()
|
||||
}
|
||||
|
||||
// Setup the scrollView initial position
|
||||
private func setupInitialPosition() {
|
||||
|
||||
// Determine the layout entrance type according to the entry type
|
||||
let messageInAnchor: NSLayoutConstraint.Attribute
|
||||
|
||||
inOffset = 0
|
||||
|
||||
var totalEntryHeight: CGFloat = 0
|
||||
|
||||
// Define a spacer to catch top / bottom offsets
|
||||
var spacerView: UIView!
|
||||
let safeAreaInsets = EKWindowProvider.safeAreaInsets
|
||||
let overrideSafeArea = attributes.positionConstraints.safeArea.isOverridden
|
||||
|
||||
if !overrideSafeArea && safeAreaInsets.hasVerticalInsets && !attributes.position.isCenter {
|
||||
spacerView = UIView()
|
||||
addSubview(spacerView)
|
||||
spacerView.set(.height, of: safeAreaInsets.top)
|
||||
spacerView.layoutToSuperview(.width, .centerX)
|
||||
|
||||
totalEntryHeight += safeAreaInsets.top
|
||||
}
|
||||
|
||||
switch attributes.position {
|
||||
case .top:
|
||||
messageInAnchor = .top
|
||||
inOffset = overrideSafeArea ? 0 : safeAreaInsets.top
|
||||
inOffset += attributes.positionConstraints.verticalOffset
|
||||
spacerView?.layout(.bottom, to: .top, of: self)
|
||||
case .bottom:
|
||||
messageInAnchor = .bottom
|
||||
inOffset = overrideSafeArea ? 0 : -safeAreaInsets.bottom
|
||||
inOffset -= attributes.positionConstraints.verticalOffset
|
||||
spacerView?.layout(.top, to: .bottom, of: self)
|
||||
case .center:
|
||||
messageInAnchor = .centerY
|
||||
}
|
||||
|
||||
// Layout the content view inside the scroll view
|
||||
addSubview(contentView)
|
||||
contentView.layoutToSuperview(.left, .right, .top, .bottom)
|
||||
contentView.layoutToSuperview(.width, .height)
|
||||
|
||||
inConstraint = layout(to: messageInAnchor, of: superview!, offset: inOffset, priority: .defaultLow)
|
||||
|
||||
// Set position constraints
|
||||
setupOutConstraints(messageInAnchor: messageInAnchor)
|
||||
|
||||
totalTranslation = inOffset
|
||||
switch attributes.position {
|
||||
case .top:
|
||||
verticalLimit = inOffset
|
||||
case .bottom, .center:
|
||||
verticalLimit = UIScreen.main.bounds.height + inOffset
|
||||
}
|
||||
|
||||
// Setup keyboard constraints
|
||||
switch attributes.positionConstraints.keyboardRelation {
|
||||
case .bind(offset: let offset):
|
||||
if let screenEdgeResistance = offset.screenEdgeResistance {
|
||||
resistanceConstraint = layoutToSuperview(.top, relation: .greaterThanOrEqual, offset: screenEdgeResistance, priority: .defaultLow)
|
||||
}
|
||||
inKeyboardConstraint = layoutToSuperview(.bottom, priority: .defaultLow)
|
||||
default:
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
private func setupOutConstraint(animation: EKAttributes.Animation?, messageInAnchor: QLAttribute, priority: QLPriority) -> NSLayoutConstraint {
|
||||
let constraint: NSLayoutConstraint
|
||||
if let translation = animation?.translate {
|
||||
var anchor: OutTranslationAnchor
|
||||
switch translation.anchorPosition {
|
||||
case .top:
|
||||
anchor = OutTranslationAnchor(.bottom, to: .top)
|
||||
case .bottom:
|
||||
anchor = OutTranslationAnchor(.top, to: .bottom)
|
||||
case .automatic where attributes.position.isTop:
|
||||
anchor = OutTranslationAnchor(.bottom, to: .top)
|
||||
case .automatic: // attributes.position.isBottom:
|
||||
anchor = OutTranslationAnchor(.top, to: .bottom)
|
||||
}
|
||||
constraint = layout(anchor.messageOut, to: anchor.screenOut, of: superview!, priority: priority)!
|
||||
} else {
|
||||
constraint = layout(to: messageInAnchor, of: superview!, offset: inOffset, priority: priority)!
|
||||
}
|
||||
return constraint
|
||||
}
|
||||
|
||||
// Setup out constraints - taking into account the full picture and all the possible use-cases
|
||||
private func setupOutConstraints(messageInAnchor: QLAttribute) {
|
||||
|
||||
// Setup entrance and exit out constraints
|
||||
entranceOutConstraint = setupOutConstraint(animation: attributes.entranceAnimation, messageInAnchor: messageInAnchor, priority: .must)
|
||||
exitOutConstraint = setupOutConstraint(animation: attributes.exitAnimation, messageInAnchor: messageInAnchor, priority: .defaultLow)
|
||||
swipeDownOutConstraint = layout(.top, to: .bottom, of: superview!, priority: .defaultLow)!
|
||||
swipeUpOutConstraint = layout(.bottom, to: .top, of: superview!, priority: .defaultLow)!
|
||||
|
||||
// Setup pop out constraint
|
||||
var popAnimation: EKAttributes.Animation?
|
||||
if case .animated(animation: let animation) = attributes.popBehavior {
|
||||
popAnimation = animation
|
||||
}
|
||||
popOutConstraint = setupOutConstraint(animation: popAnimation, messageInAnchor: messageInAnchor, priority: .defaultLow)
|
||||
}
|
||||
|
||||
|
||||
private func setupSize() {
|
||||
|
||||
// Layout the scroll view horizontally inside the screen
|
||||
switch attributes.positionConstraints.size.width {
|
||||
case .offset(value: let offset):
|
||||
layoutToSuperview(axis: .horizontally, offset: offset, priority: .must)
|
||||
case .ratio(value: let ratio):
|
||||
layoutToSuperview(.width, ratio: ratio, priority: .must)
|
||||
case .constant(value: let constant):
|
||||
set(.width, of: constant, priority: .must)
|
||||
case .intrinsic:
|
||||
break
|
||||
}
|
||||
|
||||
// Layout the scroll view vertically inside the screen
|
||||
switch attributes.positionConstraints.size.height {
|
||||
case .offset(value: let offset):
|
||||
layoutToSuperview(.height, offset: -offset * 2, priority: .must)
|
||||
case .ratio(value: let ratio):
|
||||
layoutToSuperview(.height, ratio: ratio, priority: .must)
|
||||
case .constant(value: let constant):
|
||||
set(.height, of: constant, priority: .must)
|
||||
case .intrinsic:
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
private func setupMaxSize() {
|
||||
|
||||
// Layout the scroll view according to the maximum width (if given any)
|
||||
switch attributes.positionConstraints.maxSize.width {
|
||||
case .offset(value: let offset):
|
||||
layout(to: .left, of: superview!, relation: .greaterThanOrEqual, offset: offset)
|
||||
layout(to: .right, of: superview!, relation: .lessThanOrEqual, offset: -offset)
|
||||
case .ratio(value: let ratio):
|
||||
layoutToSuperview(.centerX)
|
||||
layout(to: .width, of: superview!, relation: .lessThanOrEqual, ratio: ratio)
|
||||
case .constant(value: let constant):
|
||||
set(.width, of: constant, relation: .lessThanOrEqual)
|
||||
break
|
||||
case .intrinsic:
|
||||
break
|
||||
}
|
||||
|
||||
// Layout the scroll view according to the maximum width (if given any)
|
||||
switch attributes.positionConstraints.maxSize.height {
|
||||
case .offset(value: let offset):
|
||||
layout(to: .height, of: superview!, relation: .lessThanOrEqual, offset: -offset * 2)
|
||||
case .ratio(value: let ratio):
|
||||
layout(to: .height, of: superview!, relation: .lessThanOrEqual, ratio: ratio)
|
||||
case .constant(value: let constant):
|
||||
set(.height, of: constant, relation: .lessThanOrEqual)
|
||||
break
|
||||
case .intrinsic:
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
// Setup layout constraints according to EKAttributes.PositionConstraints
|
||||
private func setupLayoutConstraints() {
|
||||
layoutToSuperview(.centerX)
|
||||
setupSize()
|
||||
setupMaxSize()
|
||||
}
|
||||
|
||||
// Setup general attributes
|
||||
private func setupAttributes() {
|
||||
clipsToBounds = false
|
||||
let panGestureRecognizer = UIPanGestureRecognizer(target: self, action: #selector(panGestureRecognized(gr:)))
|
||||
panGestureRecognizer.isEnabled = attributes.scroll.isEnabled
|
||||
addGestureRecognizer(panGestureRecognizer)
|
||||
}
|
||||
|
||||
// Setup tap gesture
|
||||
private func setupTapGestureRecognizer() {
|
||||
switch attributes.entryInteraction.defaultAction {
|
||||
case .forward:
|
||||
return
|
||||
default:
|
||||
let tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(tapGestureRecognized))
|
||||
tapGestureRecognizer.numberOfTapsRequired = 1
|
||||
tapGestureRecognizer.cancelsTouchesInView = false
|
||||
addGestureRecognizer(tapGestureRecognizer)
|
||||
}
|
||||
}
|
||||
|
||||
// Generate a haptic feedback if needed
|
||||
private func generateHapticFeedback() {
|
||||
guard #available(iOS 10.0, *) else {
|
||||
return
|
||||
}
|
||||
HapticFeedbackGenerator.notification(type: attributes.hapticFeedbackType)
|
||||
}
|
||||
|
||||
// MARK: Animations
|
||||
|
||||
// Schedule out animation
|
||||
private func scheduleAnimateOut(withDelay delay: TimeInterval? = nil) {
|
||||
outDispatchWorkItem?.cancel()
|
||||
outDispatchWorkItem = DispatchWorkItem { [weak self] in
|
||||
self?.animateOut(pushOut: false)
|
||||
}
|
||||
let delay = attributes.entranceAnimation.totalDuration + (delay ?? attributes.displayDuration)
|
||||
DispatchQueue.main.asyncAfter(deadline: .now() + delay, execute: outDispatchWorkItem)
|
||||
}
|
||||
|
||||
// Animate out
|
||||
func animateOut(pushOut: Bool) {
|
||||
|
||||
// Execute willDisappear action if needed
|
||||
contentView.attributes.lifecycleEvents.willDisappear?()
|
||||
|
||||
if attributes.positionConstraints.keyboardRelation.isBound {
|
||||
endEditing(true)
|
||||
}
|
||||
|
||||
outDispatchWorkItem?.cancel()
|
||||
entryDelegate?.changeToInactive(withAttributes: attributes, pushOut: pushOut)
|
||||
|
||||
if case .animated(animation: let animation) = attributes.popBehavior, pushOut {
|
||||
animateOut(with: animation, outTranslationType: .pop)
|
||||
} else {
|
||||
animateOut(with: attributes.exitAnimation, outTranslationType: .exit)
|
||||
}
|
||||
}
|
||||
|
||||
// Animate out
|
||||
private func animateOut(with animation: EKAttributes.Animation, outTranslationType: OutTranslation) {
|
||||
|
||||
superview?.layoutIfNeeded()
|
||||
|
||||
if let translation = animation.translate {
|
||||
performAnimation(out: true, with: translation) { [weak self] in
|
||||
self?.translateOut(withType: outTranslationType)
|
||||
}
|
||||
}
|
||||
|
||||
if let fade = animation.fade {
|
||||
performAnimation(out: true, with: fade, preAction: { self.alpha = fade.start }) {
|
||||
self.alpha = fade.end
|
||||
}
|
||||
}
|
||||
|
||||
if let scale = animation.scale {
|
||||
performAnimation(out: true, with: scale, preAction: { self.transform = CGAffineTransform(scaleX: scale.start, y: scale.start) }) {
|
||||
self.transform = CGAffineTransform(scaleX: scale.end, y: scale.end)
|
||||
}
|
||||
}
|
||||
|
||||
if animation.containsAnimation {
|
||||
DispatchQueue.main.asyncAfter(deadline: .now() + animation.maxDuration) {
|
||||
self.removeFromSuperview(keepWindow: false)
|
||||
}
|
||||
} else {
|
||||
translateOut(withType: outTranslationType)
|
||||
removeFromSuperview(keepWindow: false)
|
||||
}
|
||||
}
|
||||
|
||||
// Animate in
|
||||
private func animateIn() {
|
||||
|
||||
let animation = attributes.entranceAnimation
|
||||
|
||||
superview?.layoutIfNeeded()
|
||||
|
||||
if let translation = animation.translate {
|
||||
performAnimation(out: false, with: translation, action: translateIn)
|
||||
} else {
|
||||
translateIn()
|
||||
}
|
||||
|
||||
if let fade = animation.fade {
|
||||
performAnimation(out: false, with: fade, preAction: { self.alpha = fade.start }) {
|
||||
self.alpha = fade.end
|
||||
}
|
||||
}
|
||||
|
||||
if let scale = animation.scale {
|
||||
performAnimation(out: false, with: scale, preAction: { self.transform = CGAffineTransform(scaleX: scale.start, y: scale.start) }) {
|
||||
self.transform = CGAffineTransform(scaleX: scale.end, y: scale.end)
|
||||
}
|
||||
}
|
||||
|
||||
entryDelegate?.changeToActive(withAttributes: attributes)
|
||||
|
||||
// Execute didAppear action if needed
|
||||
if animation.containsAnimation {
|
||||
DispatchQueue.main.asyncAfter(deadline: .now() + animation.maxDuration) {
|
||||
self.contentView.attributes.lifecycleEvents.didAppear?()
|
||||
}
|
||||
} else {
|
||||
contentView.attributes.lifecycleEvents.didAppear?()
|
||||
}
|
||||
|
||||
scheduleAnimateOut()
|
||||
}
|
||||
|
||||
// Translate in
|
||||
private func translateIn() {
|
||||
entranceOutConstraint.priority = .defaultLow
|
||||
exitOutConstraint.priority = .defaultLow
|
||||
popOutConstraint.priority = .defaultLow
|
||||
inConstraint.priority = .must
|
||||
superview?.layoutIfNeeded()
|
||||
}
|
||||
|
||||
// Translate out
|
||||
private func translateOut(withType type: OutTranslation) {
|
||||
inConstraint.priority = .defaultLow
|
||||
entranceOutConstraint.priority = .defaultLow
|
||||
switch type {
|
||||
case .exit:
|
||||
exitOutConstraint.priority = .must
|
||||
case .pop:
|
||||
popOutConstraint.priority = .must
|
||||
case .swipeUp:
|
||||
swipeUpOutConstraint.priority = .must
|
||||
case .swipeDown:
|
||||
swipeDownOutConstraint.priority = .must
|
||||
}
|
||||
superview?.layoutIfNeeded()
|
||||
}
|
||||
|
||||
// Perform animation - translate / scale / fade
|
||||
private func performAnimation(out: Bool, with animation: EKAnimation, preAction: @escaping () -> () = {}, action: @escaping () -> ()) {
|
||||
let curve: UIView.AnimationOptions = out ? .curveEaseIn : .curveEaseOut
|
||||
let options: UIView.AnimationOptions = [curve, .beginFromCurrentState]
|
||||
preAction()
|
||||
if let spring = animation.spring {
|
||||
UIView.animate(withDuration: animation.duration, delay: animation.delay, usingSpringWithDamping: spring.damping, initialSpringVelocity: spring.initialVelocity, options: options, animations: {
|
||||
action()
|
||||
}, completion: nil)
|
||||
} else {
|
||||
UIView.animate(withDuration: animation.duration, delay: animation.delay, options: options, animations: {
|
||||
action()
|
||||
}, completion: nil)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Remvoe entry
|
||||
|
||||
// Removes the view promptly - DOES NOT animate out
|
||||
func removePromptly(keepWindow: Bool = true) {
|
||||
outDispatchWorkItem?.cancel()
|
||||
entryDelegate?.changeToInactive(withAttributes: attributes, pushOut: false)
|
||||
contentView.content.attributes.lifecycleEvents.willDisappear?()
|
||||
removeFromSuperview(keepWindow: keepWindow)
|
||||
}
|
||||
|
||||
// Remove self from superview
|
||||
func removeFromSuperview(keepWindow: Bool) {
|
||||
guard superview != nil else {
|
||||
return
|
||||
}
|
||||
|
||||
// Execute didDisappear action if needed
|
||||
let didDisappear = contentView.content.attributes.lifecycleEvents.didDisappear
|
||||
|
||||
// Remove the view from its superview and in a case of a view controller, from its parent controller.
|
||||
super.removeFromSuperview()
|
||||
contentView.content.viewController?.removeFromParent()
|
||||
|
||||
entryDelegate.didFinishDisplaying(entry: contentView, keepWindowActive: keepWindow, dismissCompletionHandler: dismissHandler)
|
||||
|
||||
// Lastly, perform the Dismiss Completion Handler as the entry is no longer displayed
|
||||
didDisappear?()
|
||||
}
|
||||
|
||||
deinit {
|
||||
NotificationCenter.default.removeObserver(self)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// MARK: Keyboard Logic
|
||||
extension EKContentView {
|
||||
|
||||
private enum KeyboardState {
|
||||
case visible
|
||||
case hidden
|
||||
|
||||
var isVisible: Bool {
|
||||
return self == .visible
|
||||
}
|
||||
|
||||
var isHidden: Bool {
|
||||
return self == .hidden
|
||||
}
|
||||
}
|
||||
|
||||
private struct KeyboardAttributes {
|
||||
let duration: TimeInterval
|
||||
let curve: UIView.AnimationOptions
|
||||
let begin: CGRect
|
||||
let end: CGRect
|
||||
|
||||
init?(withRawValue rawValue: [AnyHashable: Any]?) {
|
||||
guard let rawValue = rawValue else {
|
||||
return nil
|
||||
}
|
||||
duration = rawValue[UIResponder.keyboardAnimationDurationUserInfoKey] as! TimeInterval
|
||||
curve = .init(rawValue: rawValue[UIResponder.keyboardAnimationCurveUserInfoKey] as! UInt)
|
||||
begin = (rawValue[UIResponder.keyboardFrameBeginUserInfoKey] as! NSValue).cgRectValue
|
||||
end = (rawValue[UIResponder.keyboardFrameEndUserInfoKey] as! NSValue).cgRectValue
|
||||
}
|
||||
|
||||
var height: CGFloat {
|
||||
return end.maxY - end.minY
|
||||
}
|
||||
}
|
||||
|
||||
private func setupKeyboardChangeIfNeeded() {
|
||||
guard attributes.positionConstraints.keyboardRelation.isBound else {
|
||||
return
|
||||
}
|
||||
|
||||
let notificationCenter = NotificationCenter.default
|
||||
notificationCenter.addObserver(self, selector: #selector(keyboardWillShow(_:)), name: UIResponder.keyboardWillShowNotification, object: nil)
|
||||
notificationCenter.addObserver(self, selector: #selector(keyboardWillHide(_:)), name: UIResponder.keyboardWillHideNotification, object: nil)
|
||||
notificationCenter.addObserver(self, selector: #selector(keyboardDidHide(_:)), name: UIResponder.keyboardDidHideNotification, object: nil)
|
||||
notificationCenter.addObserver(self, selector: #selector(keyboardWillChangeFrame(_:)), name: UIResponder.keyboardWillChangeFrameNotification, object: nil)
|
||||
}
|
||||
|
||||
private func animate(by userInfo: [AnyHashable: Any]?, entrance: Bool) {
|
||||
|
||||
// Guard that the entry is bound to the keyboard
|
||||
guard case .bind(offset: let offset) = attributes.positionConstraints.keyboardRelation else {
|
||||
return
|
||||
}
|
||||
|
||||
// Convert the user info into keyboard attributes
|
||||
guard let keyboardAtts = KeyboardAttributes(withRawValue: userInfo) else {
|
||||
return
|
||||
}
|
||||
|
||||
if entrance {
|
||||
inKeyboardConstraint.constant = -(keyboardAtts.height + offset.bottom)
|
||||
inKeyboardConstraint.priority = .must
|
||||
resistanceConstraint?.priority = .must
|
||||
inConstraint.priority = .defaultLow
|
||||
} else {
|
||||
inKeyboardConstraint.priority = .defaultLow
|
||||
resistanceConstraint?.priority = .defaultLow
|
||||
inConstraint.priority = .must
|
||||
}
|
||||
|
||||
UIView.animate(withDuration: keyboardAtts.duration, delay: 0, options: keyboardAtts.curve, animations: {
|
||||
self.superview?.layoutIfNeeded()
|
||||
}, completion: nil)
|
||||
}
|
||||
|
||||
@objc func keyboardWillShow(_ notification: Notification) {
|
||||
guard containsFirstResponder else {
|
||||
return
|
||||
}
|
||||
keyboardState = .visible
|
||||
animate(by: notification.userInfo, entrance: true)
|
||||
}
|
||||
|
||||
@objc func keyboardWillHide(_ notification: Notification) {
|
||||
animate(by: notification.userInfo, entrance: false)
|
||||
}
|
||||
|
||||
@objc func keyboardDidHide(_ notification: Notification) {
|
||||
keyboardState = .hidden
|
||||
}
|
||||
|
||||
@objc func keyboardWillChangeFrame(_ notification: Notification) {
|
||||
guard containsFirstResponder else {
|
||||
return
|
||||
}
|
||||
animate(by: notification.userInfo, entrance: true)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Responds to user interactions (tap / pan / swipe / touches)
|
||||
extension EKContentView {
|
||||
|
||||
// Tap gesture handler
|
||||
@objc func tapGestureRecognized() {
|
||||
switch attributes.entryInteraction.defaultAction {
|
||||
case .delayExit(by: _) where attributes.displayDuration.isFinite:
|
||||
scheduleAnimateOut()
|
||||
case .dismissEntry:
|
||||
animateOut(pushOut: false)
|
||||
default:
|
||||
break
|
||||
}
|
||||
attributes.entryInteraction.customTapActions.forEach { $0() }
|
||||
}
|
||||
|
||||
// Pan gesture handler
|
||||
@objc func panGestureRecognized(gr: UIPanGestureRecognizer) {
|
||||
guard keyboardState.isHidden else {
|
||||
return
|
||||
}
|
||||
|
||||
// Delay the exit of the entry if needed
|
||||
handleExitDelayIfNeeded(byPanState: gr.state)
|
||||
|
||||
let translation = gr.translation(in: superview!).y
|
||||
|
||||
if shouldStretch(with: translation) {
|
||||
if attributes.scroll.isEdgeCrossingEnabled {
|
||||
totalTranslation += translation
|
||||
calculateLogarithmicOffset(forOffset: totalTranslation, currentTranslation: translation)
|
||||
|
||||
switch gr.state {
|
||||
case .ended, .failed, .cancelled:
|
||||
animateRubberBandPullback()
|
||||
default:
|
||||
break
|
||||
}
|
||||
}
|
||||
} else {
|
||||
switch gr.state {
|
||||
case .ended, .failed, .cancelled:
|
||||
let velocity = gr.velocity(in: superview!).y
|
||||
swipeEnded(withVelocity: velocity)
|
||||
case .changed:
|
||||
inConstraint.constant += translation
|
||||
default:
|
||||
break
|
||||
}
|
||||
}
|
||||
gr.setTranslation(.zero, in: superview!)
|
||||
}
|
||||
|
||||
private func swipeEnded(withVelocity velocity: CGFloat) {
|
||||
let distance = Swift.abs(inOffset - inConstraint.constant)
|
||||
var duration = max(0.3, TimeInterval(distance / Swift.abs(velocity)))
|
||||
duration = min(0.7, duration)
|
||||
|
||||
if attributes.scroll.isSwipeable && testSwipeVelocity(with: velocity) && testSwipeInConstraint() {
|
||||
stretchOut(usingSwipe: velocity > 0 ? .swipeDown : .swipeUp, duration: duration)
|
||||
} else {
|
||||
animateRubberBandPullback()
|
||||
}
|
||||
}
|
||||
|
||||
private func stretchOut(usingSwipe type: OutTranslation, duration: TimeInterval) {
|
||||
outDispatchWorkItem?.cancel()
|
||||
entryDelegate?.changeToInactive(withAttributes: attributes, pushOut: false)
|
||||
contentView.content.attributes.lifecycleEvents.willDisappear?()
|
||||
UIView.animate(withDuration: duration, delay: 0, usingSpringWithDamping: 1, initialSpringVelocity: 4, options: [.allowUserInteraction, .beginFromCurrentState], animations: {
|
||||
self.translateOut(withType: type)
|
||||
}, completion: { finished in
|
||||
self.removeFromSuperview(keepWindow: false)
|
||||
})
|
||||
}
|
||||
|
||||
private func calculateLogarithmicOffset(forOffset offset: CGFloat, currentTranslation: CGFloat) {
|
||||
guard verticalLimit != 0 else { return }
|
||||
if attributes.position.isTop {
|
||||
inConstraint.constant = verticalLimit * (1 + log10(offset / verticalLimit))
|
||||
} else {
|
||||
let offset = Swift.abs(offset) + verticalLimit
|
||||
let addition: CGFloat = abs(currentTranslation) < 2 ? 0 : 1
|
||||
inConstraint.constant -= (addition + log10(offset / verticalLimit))
|
||||
}
|
||||
}
|
||||
|
||||
private func shouldStretch(with translation: CGFloat) -> Bool {
|
||||
if attributes.position.isTop {
|
||||
return translation > 0 && inConstraint.constant >= inOffset
|
||||
} else {
|
||||
return translation < 0 && inConstraint.constant <= inOffset
|
||||
}
|
||||
}
|
||||
|
||||
private func animateRubberBandPullback() {
|
||||
totalTranslation = verticalLimit
|
||||
|
||||
let animation: EKAttributes.Scroll.PullbackAnimation
|
||||
if case .enabled(swipeable: _, pullbackAnimation: let pullbackAnimation) = attributes.scroll {
|
||||
animation = pullbackAnimation
|
||||
} else {
|
||||
animation = .easeOut
|
||||
}
|
||||
|
||||
UIView.animate(withDuration: animation.duration, delay: 0, usingSpringWithDamping: animation.damping, initialSpringVelocity: animation.initialSpringVelocity, options: [.allowUserInteraction, .beginFromCurrentState], animations: {
|
||||
self.inConstraint?.constant = self.inOffset
|
||||
self.superview?.layoutIfNeeded()
|
||||
}, completion: nil)
|
||||
}
|
||||
|
||||
private func testSwipeInConstraint() -> Bool {
|
||||
if attributes.position.isTop {
|
||||
return inConstraint.constant < inOffset
|
||||
} else {
|
||||
return inConstraint.constant > inOffset
|
||||
}
|
||||
}
|
||||
|
||||
private func testSwipeVelocity(with velocity: CGFloat) -> Bool {
|
||||
if attributes.position.isTop {
|
||||
return velocity < -swipeMinVelocity
|
||||
} else {
|
||||
return velocity > swipeMinVelocity
|
||||
}
|
||||
}
|
||||
|
||||
private func handleExitDelayIfNeeded(byPanState state: UIGestureRecognizer.State) {
|
||||
guard attributes.entryInteraction.isDelayExit && attributes.displayDuration.isFinite else {
|
||||
return
|
||||
}
|
||||
switch state {
|
||||
case .began:
|
||||
outDispatchWorkItem?.cancel()
|
||||
case .ended, .failed, .cancelled:
|
||||
scheduleAnimateOut()
|
||||
default:
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: UIResponder
|
||||
|
||||
override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
|
||||
if attributes.entryInteraction.isDelayExit && attributes.displayDuration.isFinite {
|
||||
outDispatchWorkItem?.cancel()
|
||||
}
|
||||
}
|
||||
|
||||
override func touchesEnded(_ touches: Set<UITouch>, with event: UIEvent?) {
|
||||
if attributes.entryInteraction.isDelayExit && attributes.displayDuration.isFinite {
|
||||
scheduleAnimateOut()
|
||||
}
|
||||
}
|
||||
|
||||
override func touchesCancelled(_ touches: Set<UITouch>, with event: UIEvent?) {
|
||||
touchesEnded(touches, with: event)
|
||||
}
|
||||
}
|
||||
182
Pods/SwiftEntryKit/Source/Infra/EKEntryView.swift
generated
Normal file
182
Pods/SwiftEntryKit/Source/Infra/EKEntryView.swift
generated
Normal file
@@ -0,0 +1,182 @@
|
||||
//
|
||||
// EKEntryView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/15/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
class EKEntryView: EKStyleView {
|
||||
|
||||
struct Content {
|
||||
var viewController: UIViewController!
|
||||
var view: UIView!
|
||||
var attributes: EKAttributes
|
||||
|
||||
init(viewController: UIViewController, attributes: EKAttributes) {
|
||||
self.viewController = viewController
|
||||
self.view = viewController.view
|
||||
self.attributes = attributes
|
||||
}
|
||||
|
||||
init(view: UIView, attributes: EKAttributes) {
|
||||
self.view = view
|
||||
self.attributes = attributes
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Props
|
||||
|
||||
/** Background view */
|
||||
private var backgroundView: EKBackgroundView!
|
||||
|
||||
/** The content - contains the view, view controller, attributes */
|
||||
var content: Content
|
||||
|
||||
private lazy var contentView: UIView = {
|
||||
return UIView()
|
||||
}()
|
||||
|
||||
var attributes: EKAttributes {
|
||||
return content.attributes
|
||||
}
|
||||
|
||||
private lazy var contentContainerView: EKStyleView = {
|
||||
let contentContainerView = EKStyleView()
|
||||
self.addSubview(contentContainerView)
|
||||
contentContainerView.layoutToSuperview(axis: .vertically)
|
||||
contentContainerView.layoutToSuperview(axis: .horizontally)
|
||||
contentContainerView.clipsToBounds = true
|
||||
return contentContainerView
|
||||
}()
|
||||
|
||||
// MARK: Setup
|
||||
init(newEntry content: Content) {
|
||||
self.content = content
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setupContentView()
|
||||
applyDropShadow()
|
||||
applyBackgroundToContentView()
|
||||
applyFrameStyle()
|
||||
adjustInnerContentAppearanceIfNeeded()
|
||||
}
|
||||
|
||||
required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
override func layoutSubviews() {
|
||||
super.layoutSubviews()
|
||||
applyFrameStyle()
|
||||
}
|
||||
|
||||
func transform(to view: UIView) {
|
||||
|
||||
let previousView = content.view
|
||||
content.view = view
|
||||
view.layoutIfNeeded()
|
||||
|
||||
let previousHeight = set(.height, of: frame.height, priority: .must)
|
||||
let nextHeight = set(.height, of: view.frame.height, priority: .defaultLow)
|
||||
|
||||
SwiftEntryKit.layoutIfNeeded()
|
||||
|
||||
UIView.animate(withDuration: 0.4, delay: 0, usingSpringWithDamping: 1, initialSpringVelocity: 0, options: [.beginFromCurrentState, .layoutSubviews], animations: {
|
||||
|
||||
previousHeight.priority = .defaultLow
|
||||
nextHeight.priority = .must
|
||||
|
||||
previousView!.alpha = 0
|
||||
|
||||
SwiftEntryKit.layoutIfNeeded()
|
||||
|
||||
}, completion: { (finished) in
|
||||
|
||||
view.alpha = 0
|
||||
|
||||
previousView!.removeFromSuperview()
|
||||
self.removeConstraints([previousHeight, nextHeight])
|
||||
|
||||
self.setupContentView()
|
||||
|
||||
UIView.animate(withDuration: 0.4, delay: 0, usingSpringWithDamping: 1, initialSpringVelocity: 0, options: [.curveEaseOut], animations: {
|
||||
view.alpha = 1
|
||||
}, completion: nil)
|
||||
})
|
||||
}
|
||||
|
||||
private func setupContentView() {
|
||||
contentView.addSubview(content.view)
|
||||
content.view.layoutToSuperview(axis: .horizontally)
|
||||
content.view.layoutToSuperview(axis: .vertically)
|
||||
|
||||
contentContainerView.addSubview(contentView)
|
||||
contentView.fillSuperview()
|
||||
contentView.layoutToSuperview(axis: .vertically)
|
||||
contentView.layoutToSuperview(axis: .horizontally)
|
||||
}
|
||||
|
||||
// Complementary logic for issue #117
|
||||
private func adjustInnerContentAppearanceIfNeeded() {
|
||||
guard let view = content.view as? EntryAppearanceDescriptor else {
|
||||
return
|
||||
}
|
||||
view.bottomCornerRadius = attributes.roundCorners.cornerValues?.radius ?? 0
|
||||
}
|
||||
|
||||
// Apply round corners
|
||||
private func applyFrameStyle() {
|
||||
backgroundView.applyFrameStyle(roundCorners: attributes.roundCorners, border: attributes.border)
|
||||
}
|
||||
|
||||
// Apply drop shadow
|
||||
private func applyDropShadow() {
|
||||
switch attributes.shadow {
|
||||
case .active(with: let value):
|
||||
applyDropShadow(withOffset: value.offset,
|
||||
opacity: value.opacity,
|
||||
radius: value.radius,
|
||||
color: value.color.color(for: traitCollection, mode: attributes.displayMode))
|
||||
case .none:
|
||||
removeDropShadow()
|
||||
}
|
||||
}
|
||||
|
||||
// Apply background
|
||||
private func applyBackgroundToContentView() {
|
||||
let attributes = content.attributes
|
||||
|
||||
let backgroundView = EKBackgroundView()
|
||||
backgroundView.style = .init(background: attributes.entryBackground,
|
||||
displayMode: attributes.displayMode)
|
||||
|
||||
switch attributes.positionConstraints.safeArea {
|
||||
case .empty(fillSafeArea: let fillSafeArea) where fillSafeArea: // Safe area filled with color
|
||||
insertSubview(backgroundView, at: 0)
|
||||
backgroundView.layoutToSuperview(axis: .horizontally)
|
||||
|
||||
var topInset: CGFloat = 0
|
||||
var bottomInset: CGFloat = 0
|
||||
switch attributes.position {
|
||||
case .top:
|
||||
topInset = -EKWindowProvider.safeAreaInsets.top
|
||||
case .bottom, .center:
|
||||
bottomInset = EKWindowProvider.safeAreaInsets.bottom
|
||||
}
|
||||
|
||||
backgroundView.layoutToSuperview(.top, offset: topInset)
|
||||
backgroundView.layoutToSuperview(.bottom, offset: bottomInset)
|
||||
default: // Float case or a Toast with unfilled safe area
|
||||
contentView.insertSubview(backgroundView, at: 0)
|
||||
backgroundView.fillSuperview()
|
||||
}
|
||||
|
||||
self.backgroundView = backgroundView
|
||||
}
|
||||
|
||||
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
applyDropShadow()
|
||||
}
|
||||
}
|
||||
268
Pods/SwiftEntryKit/Source/Infra/EKRootViewController.swift
generated
Normal file
268
Pods/SwiftEntryKit/Source/Infra/EKRootViewController.swift
generated
Normal file
@@ -0,0 +1,268 @@
|
||||
//
|
||||
// EntryViewController.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
protocol EntryPresenterDelegate: AnyObject {
|
||||
var isResponsiveToTouches: Bool { set get }
|
||||
func displayPendingEntryOrRollbackWindow(dismissCompletionHandler: SwiftEntryKit.DismissCompletionHandler?)
|
||||
}
|
||||
|
||||
class EKRootViewController: UIViewController {
|
||||
|
||||
// MARK: - Props
|
||||
|
||||
private unowned let delegate: EntryPresenterDelegate
|
||||
|
||||
private var lastAttributes: EKAttributes!
|
||||
|
||||
private let backgroundView = EKBackgroundView()
|
||||
|
||||
private lazy var wrapperView: EKWrapperView = {
|
||||
return EKWrapperView()
|
||||
}()
|
||||
|
||||
/*
|
||||
Count the total amount of currently displaying entries,
|
||||
meaning, total subviews less one - the backgorund of the entry
|
||||
*/
|
||||
fileprivate var displayingEntryCount: Int {
|
||||
return view.subviews.count - 1
|
||||
}
|
||||
|
||||
fileprivate var isDisplaying: Bool {
|
||||
return lastEntry != nil
|
||||
}
|
||||
|
||||
private var lastEntry: EKContentView? {
|
||||
return view.subviews.last as? EKContentView
|
||||
}
|
||||
|
||||
private var isResponsive = false {
|
||||
didSet {
|
||||
wrapperView.isAbleToReceiveTouches = isResponsive
|
||||
delegate.isResponsiveToTouches = isResponsive
|
||||
}
|
||||
}
|
||||
|
||||
override var shouldAutorotate: Bool {
|
||||
if lastAttributes == nil {
|
||||
return true
|
||||
}
|
||||
return lastAttributes.positionConstraints.rotation.isEnabled
|
||||
}
|
||||
|
||||
override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
|
||||
guard let lastAttributes = lastAttributes else {
|
||||
return super.supportedInterfaceOrientations
|
||||
}
|
||||
switch lastAttributes.positionConstraints.rotation.supportedInterfaceOrientations {
|
||||
case .standard:
|
||||
return super.supportedInterfaceOrientations
|
||||
case .all:
|
||||
return .all
|
||||
}
|
||||
}
|
||||
|
||||
// Previous status bar style
|
||||
private let previousStatusBar: EKAttributes.StatusBar
|
||||
|
||||
private var statusBar: EKAttributes.StatusBar? = nil {
|
||||
didSet {
|
||||
if let statusBar = statusBar, ![statusBar, oldValue].contains(.ignored) {
|
||||
UIApplication.shared.set(statusBarStyle: statusBar)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
override var preferredStatusBarStyle: UIStatusBarStyle {
|
||||
if [previousStatusBar, statusBar].contains(.ignored) {
|
||||
return super.preferredStatusBarStyle
|
||||
}
|
||||
return statusBar?.appearance.style ?? previousStatusBar.appearance.style
|
||||
}
|
||||
|
||||
override var prefersStatusBarHidden: Bool {
|
||||
if [previousStatusBar, statusBar].contains(.ignored) {
|
||||
return super.prefersStatusBarHidden
|
||||
}
|
||||
return !(statusBar?.appearance.visible ?? previousStatusBar.appearance.visible)
|
||||
}
|
||||
|
||||
// MARK: - Lifecycle
|
||||
|
||||
required public init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
public init(with delegate: EntryPresenterDelegate) {
|
||||
self.delegate = delegate
|
||||
previousStatusBar = .currentStatusBar
|
||||
super.init(nibName: nil, bundle: nil)
|
||||
}
|
||||
|
||||
override public func loadView() {
|
||||
view = wrapperView
|
||||
view.insertSubview(backgroundView, at: 0)
|
||||
backgroundView.isUserInteractionEnabled = false
|
||||
backgroundView.fillSuperview()
|
||||
}
|
||||
|
||||
override public func viewWillDisappear(_ animated: Bool) {
|
||||
super.viewWillDisappear(animated)
|
||||
statusBar = previousStatusBar
|
||||
}
|
||||
|
||||
// Set status bar
|
||||
func setStatusBarStyle(for attributes: EKAttributes) {
|
||||
statusBar = attributes.statusBar
|
||||
}
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
func configure(entryView: EKEntryView) {
|
||||
|
||||
// In case the entry is a view controller, add the entry as child of root
|
||||
if let viewController = entryView.content.viewController {
|
||||
addChild(viewController)
|
||||
}
|
||||
|
||||
// Extract the attributes struct
|
||||
let attributes = entryView.attributes
|
||||
|
||||
// Assign attributes
|
||||
let previousAttributes = lastAttributes
|
||||
|
||||
// Remove the last entry
|
||||
removeLastEntry(lastAttributes: previousAttributes, keepWindow: true)
|
||||
|
||||
lastAttributes = attributes
|
||||
|
||||
let entryContentView = EKContentView(withEntryDelegate: self)
|
||||
view.addSubview(entryContentView)
|
||||
entryContentView.setup(with: entryView)
|
||||
|
||||
switch attributes.screenInteraction.defaultAction {
|
||||
case .forward:
|
||||
isResponsive = false
|
||||
default:
|
||||
isResponsive = true
|
||||
}
|
||||
|
||||
if previousAttributes?.statusBar != attributes.statusBar {
|
||||
setNeedsStatusBarAppearanceUpdate()
|
||||
}
|
||||
|
||||
if shouldAutorotate {
|
||||
UIViewController.attemptRotationToDeviceOrientation()
|
||||
}
|
||||
}
|
||||
|
||||
// Check priority precedence for a given entry
|
||||
func canDisplay(attributes: EKAttributes) -> Bool {
|
||||
guard let lastAttributes = lastAttributes else {
|
||||
return true
|
||||
}
|
||||
return attributes.precedence.priority >= lastAttributes.precedence.priority
|
||||
}
|
||||
|
||||
// Removes last entry - can keep the window 'ON' if necessary
|
||||
private func removeLastEntry(lastAttributes: EKAttributes?, keepWindow: Bool) {
|
||||
guard let attributes = lastAttributes else {
|
||||
return
|
||||
}
|
||||
if attributes.popBehavior.isOverriden {
|
||||
lastEntry?.removePromptly()
|
||||
} else {
|
||||
popLastEntry()
|
||||
}
|
||||
}
|
||||
|
||||
// Make last entry exit using exitAnimation - animatedly
|
||||
func animateOutLastEntry(completionHandler: SwiftEntryKit.DismissCompletionHandler? = nil) {
|
||||
lastEntry?.dismissHandler = completionHandler
|
||||
lastEntry?.animateOut(pushOut: false)
|
||||
}
|
||||
|
||||
// Pops last entry (using pop animation) - animatedly
|
||||
func popLastEntry() {
|
||||
lastEntry?.animateOut(pushOut: true)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - UIResponder
|
||||
|
||||
extension EKRootViewController {
|
||||
|
||||
override func touchesEnded(_ touches: Set<UITouch>, with event: UIEvent?) {
|
||||
switch lastAttributes.screenInteraction.defaultAction {
|
||||
case .dismissEntry:
|
||||
lastEntry?.animateOut(pushOut: false)
|
||||
fallthrough
|
||||
default:
|
||||
lastAttributes.screenInteraction.customTapActions.forEach { $0() }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - EntryScrollViewDelegate
|
||||
|
||||
extension EKRootViewController: EntryContentViewDelegate {
|
||||
|
||||
func didFinishDisplaying(entry: EKEntryView, keepWindowActive: Bool, dismissCompletionHandler: SwiftEntryKit.DismissCompletionHandler?) {
|
||||
guard !isDisplaying else {
|
||||
return
|
||||
}
|
||||
|
||||
guard !keepWindowActive else {
|
||||
return
|
||||
}
|
||||
|
||||
delegate.displayPendingEntryOrRollbackWindow(dismissCompletionHandler: dismissCompletionHandler)
|
||||
}
|
||||
|
||||
func changeToInactive(withAttributes attributes: EKAttributes, pushOut: Bool) {
|
||||
guard displayingEntryCount <= 1 else {
|
||||
return
|
||||
}
|
||||
|
||||
let clear = {
|
||||
let style = EKBackgroundView.Style(background: .clear, displayMode: attributes.displayMode)
|
||||
self.changeBackground(to: style, duration: attributes.exitAnimation.totalDuration)
|
||||
}
|
||||
|
||||
guard pushOut else {
|
||||
clear()
|
||||
return
|
||||
}
|
||||
|
||||
guard let lastBackroundStyle = lastAttributes?.screenBackground else {
|
||||
clear()
|
||||
return
|
||||
}
|
||||
|
||||
if lastBackroundStyle != attributes.screenBackground {
|
||||
clear()
|
||||
}
|
||||
}
|
||||
|
||||
func changeToActive(withAttributes attributes: EKAttributes) {
|
||||
let style = EKBackgroundView.Style(background: attributes.screenBackground,
|
||||
displayMode: attributes.displayMode)
|
||||
changeBackground(to: style, duration: attributes.entranceAnimation.totalDuration)
|
||||
}
|
||||
|
||||
private func changeBackground(to style: EKBackgroundView.Style, duration: TimeInterval) {
|
||||
DispatchQueue.main.async {
|
||||
UIView.animate(withDuration: duration, delay: 0, options: [], animations: {
|
||||
self.backgroundView.style = style
|
||||
}, completion: nil)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
57
Pods/SwiftEntryKit/Source/Infra/EKStyleView.swift
generated
Normal file
57
Pods/SwiftEntryKit/Source/Infra/EKStyleView.swift
generated
Normal file
@@ -0,0 +1,57 @@
|
||||
//
|
||||
// EKStyleView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/28/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
class EKStyleView: UIView {
|
||||
|
||||
private lazy var borderLayer: CAShapeLayer = {
|
||||
return CAShapeLayer()
|
||||
}()
|
||||
|
||||
private var roundCorners: EKAttributes.RoundCorners!
|
||||
private var border: EKAttributes.Border!
|
||||
|
||||
var appliedStyle = false
|
||||
|
||||
func applyFrameStyle(roundCorners: EKAttributes.RoundCorners, border: EKAttributes.Border) {
|
||||
self.roundCorners = roundCorners
|
||||
self.border = border
|
||||
|
||||
var cornerRadius: CGFloat = 0
|
||||
var corners: UIRectCorner = []
|
||||
(corners, cornerRadius) = roundCorners.cornerValues ?? ([], 0)
|
||||
|
||||
let size = CGSize(width: cornerRadius, height: cornerRadius)
|
||||
let path = UIBezierPath(roundedRect: bounds, byRoundingCorners: corners, cornerRadii: size)
|
||||
|
||||
if !corners.isEmpty && cornerRadius > 0 {
|
||||
let maskLayer = CAShapeLayer()
|
||||
maskLayer.path = path.cgPath
|
||||
layer.mask = maskLayer
|
||||
}
|
||||
|
||||
if let borderValues = border.borderValues {
|
||||
borderLayer.path = path.cgPath
|
||||
borderLayer.fillColor = UIColor.clear.cgColor
|
||||
borderLayer.strokeColor = borderValues.color.cgColor
|
||||
borderLayer.lineWidth = borderValues.width
|
||||
borderLayer.frame = bounds
|
||||
layer.addSublayer(borderLayer)
|
||||
}
|
||||
|
||||
appliedStyle = true
|
||||
}
|
||||
|
||||
override func layoutSubviews() {
|
||||
super.layoutSubviews()
|
||||
guard let roundCorners = roundCorners, let border = border else {
|
||||
return
|
||||
}
|
||||
applyFrameStyle(roundCorners: roundCorners, border: border)
|
||||
}
|
||||
}
|
||||
50
Pods/SwiftEntryKit/Source/Infra/EKWindow.swift
generated
Normal file
50
Pods/SwiftEntryKit/Source/Infra/EKWindow.swift
generated
Normal file
@@ -0,0 +1,50 @@
|
||||
//
|
||||
// EKWindow.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
class EKWindow: UIWindow {
|
||||
|
||||
var isAbleToReceiveTouches = false
|
||||
|
||||
init(with rootVC: UIViewController) {
|
||||
if #available(iOS 13.0, *) {
|
||||
// TODO: Patched to support SwiftUI out of the box but should require attendance
|
||||
if let scene = UIApplication.shared.connectedScenes.filter({$0.activationState == .foregroundActive}).first as? UIWindowScene {
|
||||
super.init(windowScene: scene)
|
||||
} else {
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
}
|
||||
} else {
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
}
|
||||
backgroundColor = .clear
|
||||
rootViewController = rootVC
|
||||
accessibilityViewIsModal = true
|
||||
}
|
||||
|
||||
required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? {
|
||||
if isAbleToReceiveTouches {
|
||||
return super.hitTest(point, with: event)
|
||||
}
|
||||
|
||||
guard let rootVC = EKWindowProvider.shared.rootVC else {
|
||||
return nil
|
||||
}
|
||||
|
||||
if let view = rootVC.view.hitTest(point, with: event) {
|
||||
return view
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
}
|
||||
231
Pods/SwiftEntryKit/Source/Infra/EKWindowProvider.swift
generated
Normal file
231
Pods/SwiftEntryKit/Source/Infra/EKWindowProvider.swift
generated
Normal file
@@ -0,0 +1,231 @@
|
||||
//
|
||||
// EKWindowProvider.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final class EKWindowProvider: EntryPresenterDelegate {
|
||||
|
||||
/** The artificial safe area insets */
|
||||
static var safeAreaInsets: UIEdgeInsets {
|
||||
if #available(iOS 11.0, *) {
|
||||
return EKWindowProvider.shared.entryWindow?.rootViewController?.view?.safeAreaInsets ?? UIApplication.shared.keyWindow?.rootViewController?.view.safeAreaInsets ?? .zero
|
||||
} else {
|
||||
let statusBarMaxY = UIApplication.shared.statusBarFrame.maxY
|
||||
return UIEdgeInsets(top: statusBarMaxY, left: 0, bottom: 10, right: 0)
|
||||
}
|
||||
}
|
||||
|
||||
/** Single access point */
|
||||
static let shared = EKWindowProvider()
|
||||
|
||||
/** Current entry window */
|
||||
var entryWindow: EKWindow!
|
||||
|
||||
/** Returns the root view controller if it is instantiated */
|
||||
var rootVC: EKRootViewController? {
|
||||
return entryWindow?.rootViewController as? EKRootViewController
|
||||
}
|
||||
|
||||
/** A window to go back to when the last entry has been dismissed */
|
||||
private var rollbackWindow: SwiftEntryKit.RollbackWindow!
|
||||
|
||||
/** The main rollback window to be used internally in case `rollbackWindow`'s value is `.main` */
|
||||
private weak var mainRollbackWindow: UIWindow?
|
||||
|
||||
/** Entry queueing heuristic */
|
||||
private let entryQueue = EKAttributes.Precedence.QueueingHeuristic.value.heuristic
|
||||
|
||||
private weak var entryView: EKEntryView!
|
||||
|
||||
/** Cannot be instantiated, customized, inherited */
|
||||
private init() {}
|
||||
|
||||
var isResponsiveToTouches: Bool {
|
||||
set {
|
||||
entryWindow.isAbleToReceiveTouches = newValue
|
||||
}
|
||||
get {
|
||||
return entryWindow.isAbleToReceiveTouches
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Setup and Teardown methods
|
||||
|
||||
// Prepare the window and the host view controller
|
||||
private func prepare(for attributes: EKAttributes, presentInsideKeyWindow: Bool) -> EKRootViewController? {
|
||||
let entryVC = setupWindowAndRootVC()
|
||||
guard entryVC.canDisplay(attributes: attributes) || attributes.precedence.isEnqueue else {
|
||||
return nil
|
||||
}
|
||||
entryVC.setStatusBarStyle(for: attributes)
|
||||
|
||||
entryWindow.windowLevel = attributes.windowLevel.value
|
||||
if presentInsideKeyWindow {
|
||||
entryWindow.makeKeyAndVisible()
|
||||
} else {
|
||||
entryWindow.isHidden = false
|
||||
}
|
||||
|
||||
return entryVC
|
||||
}
|
||||
|
||||
/** Boilerplate generic setup for entry-window and root-view-controller */
|
||||
private func setupWindowAndRootVC() -> EKRootViewController {
|
||||
let entryVC: EKRootViewController
|
||||
if entryWindow == nil {
|
||||
entryVC = EKRootViewController(with: self)
|
||||
entryWindow = EKWindow(with: entryVC)
|
||||
mainRollbackWindow = UIApplication.shared.keyWindow
|
||||
} else {
|
||||
entryVC = rootVC!
|
||||
}
|
||||
return entryVC
|
||||
}
|
||||
|
||||
/**
|
||||
Privately used to display an entry
|
||||
*/
|
||||
private func display(entryView: EKEntryView, using attributes: EKAttributes, presentInsideKeyWindow: Bool, rollbackWindow: SwiftEntryKit.RollbackWindow) {
|
||||
switch entryView.attributes.precedence {
|
||||
case .override(priority: _, dropEnqueuedEntries: let dropEnqueuedEntries):
|
||||
if dropEnqueuedEntries {
|
||||
entryQueue.removeAll()
|
||||
}
|
||||
show(entryView: entryView, presentInsideKeyWindow: presentInsideKeyWindow, rollbackWindow: rollbackWindow)
|
||||
case .enqueue where isCurrentlyDisplaying():
|
||||
entryQueue.enqueue(entry: .init(view: entryView, presentInsideKeyWindow: presentInsideKeyWindow, rollbackWindow: rollbackWindow))
|
||||
case .enqueue:
|
||||
show(entryView: entryView, presentInsideKeyWindow: presentInsideKeyWindow, rollbackWindow: rollbackWindow)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Exposed Actions
|
||||
|
||||
func queueContains(entryNamed name: String? = nil) -> Bool {
|
||||
if name == nil && !entryQueue.isEmpty {
|
||||
return true
|
||||
}
|
||||
if let name = name {
|
||||
return entryQueue.contains(entryNamed: name)
|
||||
} else {
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Returns *true* if the currently displayed entry has the given name.
|
||||
In case *name* has the value of *nil*, the result is *true* if any entry is currently displayed.
|
||||
*/
|
||||
func isCurrentlyDisplaying(entryNamed name: String? = nil) -> Bool {
|
||||
guard let entryView = entryView else {
|
||||
return false
|
||||
}
|
||||
if let name = name { // Test for names equality
|
||||
return entryView.content.attributes.name == name
|
||||
} else { // Return true by default if the name is *nil*
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
/** Transform current entry to view */
|
||||
func transform(to view: UIView) {
|
||||
entryView?.transform(to: view)
|
||||
}
|
||||
|
||||
/** Display a view using attributes */
|
||||
func display(view: UIView, using attributes: EKAttributes, presentInsideKeyWindow: Bool, rollbackWindow: SwiftEntryKit.RollbackWindow) {
|
||||
let entryView = EKEntryView(newEntry: .init(view: view, attributes: attributes))
|
||||
display(entryView: entryView, using: attributes, presentInsideKeyWindow: presentInsideKeyWindow, rollbackWindow: rollbackWindow)
|
||||
}
|
||||
|
||||
/** Display a view controller using attributes */
|
||||
func display(viewController: UIViewController, using attributes: EKAttributes, presentInsideKeyWindow: Bool, rollbackWindow: SwiftEntryKit.RollbackWindow) {
|
||||
let entryView = EKEntryView(newEntry: .init(viewController: viewController, attributes: attributes))
|
||||
display(entryView: entryView, using: attributes, presentInsideKeyWindow: presentInsideKeyWindow, rollbackWindow: rollbackWindow)
|
||||
}
|
||||
|
||||
/** Clear all entries immediately and display to the rollback window */
|
||||
func displayRollbackWindow() {
|
||||
if #available(iOS 13.0, *) {
|
||||
entryWindow.windowScene = nil
|
||||
}
|
||||
entryWindow = nil
|
||||
entryView = nil
|
||||
switch rollbackWindow! {
|
||||
case .main:
|
||||
if let mainRollbackWindow = mainRollbackWindow {
|
||||
mainRollbackWindow.makeKeyAndVisible()
|
||||
} else {
|
||||
UIApplication.shared.keyWindow?.makeKeyAndVisible()
|
||||
}
|
||||
case .custom(window: let window):
|
||||
window.makeKeyAndVisible()
|
||||
}
|
||||
}
|
||||
|
||||
/** Display a pending entry if there is any inside the queue */
|
||||
func displayPendingEntryOrRollbackWindow(dismissCompletionHandler: SwiftEntryKit.DismissCompletionHandler?) {
|
||||
if let next = entryQueue.dequeue() {
|
||||
|
||||
// Execute dismiss handler if needed before dequeuing (potentially) another entry
|
||||
dismissCompletionHandler?()
|
||||
|
||||
// Show the next entry in queue
|
||||
show(entryView: next.view, presentInsideKeyWindow: next.presentInsideKeyWindow, rollbackWindow: next.rollbackWindow)
|
||||
} else {
|
||||
|
||||
// Display the rollback window
|
||||
displayRollbackWindow()
|
||||
|
||||
// As a last step, invoke the dismissal method
|
||||
dismissCompletionHandler?()
|
||||
}
|
||||
}
|
||||
|
||||
/** Dismiss entries according to a given descriptor */
|
||||
func dismiss(_ descriptor: SwiftEntryKit.EntryDismissalDescriptor, with completion: SwiftEntryKit.DismissCompletionHandler? = nil) {
|
||||
guard let rootVC = rootVC else {
|
||||
return
|
||||
}
|
||||
|
||||
switch descriptor {
|
||||
case .displayed:
|
||||
rootVC.animateOutLastEntry(completionHandler: completion)
|
||||
case .specific(entryName: let name):
|
||||
entryQueue.removeEntries(by: name)
|
||||
if entryView?.attributes.name == name {
|
||||
rootVC.animateOutLastEntry(completionHandler: completion)
|
||||
}
|
||||
case .prioritizedLowerOrEqualTo(priority: let priorityThreshold):
|
||||
entryQueue.removeEntries(withPriorityLowerOrEqualTo: priorityThreshold)
|
||||
if let currentPriority = entryView?.attributes.precedence.priority, currentPriority <= priorityThreshold {
|
||||
rootVC.animateOutLastEntry(completionHandler: completion)
|
||||
}
|
||||
case .enqueued:
|
||||
entryQueue.removeAll()
|
||||
case .all:
|
||||
entryQueue.removeAll()
|
||||
rootVC.animateOutLastEntry(completionHandler: completion)
|
||||
}
|
||||
}
|
||||
|
||||
/** Layout the view-hierarchy rooted in the window */
|
||||
func layoutIfNeeded() {
|
||||
entryWindow?.layoutIfNeeded()
|
||||
}
|
||||
|
||||
/** Privately used to prepare the root view controller and show the entry immediately */
|
||||
private func show(entryView: EKEntryView, presentInsideKeyWindow: Bool, rollbackWindow: SwiftEntryKit.RollbackWindow) {
|
||||
guard let entryVC = prepare(for: entryView.attributes, presentInsideKeyWindow: presentInsideKeyWindow) else {
|
||||
return
|
||||
}
|
||||
entryVC.configure(entryView: entryView)
|
||||
self.entryView = entryView
|
||||
self.rollbackWindow = rollbackWindow
|
||||
}
|
||||
}
|
||||
23
Pods/SwiftEntryKit/Source/Infra/EKWrapperView.swift
generated
Normal file
23
Pods/SwiftEntryKit/Source/Infra/EKWrapperView.swift
generated
Normal file
@@ -0,0 +1,23 @@
|
||||
//
|
||||
// EKWrapperView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
class EKWrapperView: UIView {
|
||||
var isAbleToReceiveTouches = false
|
||||
override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? {
|
||||
if isAbleToReceiveTouches {
|
||||
return super.hitTest(point, with: event)
|
||||
}
|
||||
if let view = super.hitTest(point, with: event), view != self {
|
||||
return view
|
||||
}
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
98
Pods/SwiftEntryKit/Source/Infra/EntryCachingHeuristic.swift
generated
Normal file
98
Pods/SwiftEntryKit/Source/Infra/EntryCachingHeuristic.swift
generated
Normal file
@@ -0,0 +1,98 @@
|
||||
//
|
||||
// EKEntryCacher.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 9/1/18.
|
||||
// Copyright © 2018 CocoaPods. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
struct CachedEntry {
|
||||
let view: EKEntryView
|
||||
let presentInsideKeyWindow: Bool
|
||||
let rollbackWindow: SwiftEntryKit.RollbackWindow
|
||||
}
|
||||
|
||||
protocol EntryCachingHeuristic: AnyObject {
|
||||
var entries: [CachedEntry] { set get }
|
||||
var isEmpty: Bool { get }
|
||||
|
||||
func dequeue() -> CachedEntry?
|
||||
func enqueue(entry: CachedEntry)
|
||||
|
||||
func removeEntries(by name: String)
|
||||
func removeEntries(withPriorityLowerOrEqualTo priority: EKAttributes.Precedence.Priority)
|
||||
func remove(entry: CachedEntry)
|
||||
func removeAll()
|
||||
|
||||
func contains(entryNamed name: String) -> Bool
|
||||
}
|
||||
|
||||
extension EntryCachingHeuristic {
|
||||
|
||||
var isEmpty: Bool {
|
||||
return entries.isEmpty
|
||||
}
|
||||
|
||||
func contains(entryNamed name: String) -> Bool {
|
||||
return entries.contains { $0.view.attributes.name == name }
|
||||
}
|
||||
|
||||
func dequeue() -> CachedEntry? {
|
||||
guard let first = entries.first else {
|
||||
return nil
|
||||
}
|
||||
entries.removeFirst()
|
||||
return first
|
||||
}
|
||||
|
||||
func removeEntries(withPriorityLowerOrEqualTo priority: EKAttributes.Precedence.Priority) {
|
||||
while let index = (entries.firstIndex { $0.view.attributes.precedence.priority <= priority }) {
|
||||
entries.remove(at: index)
|
||||
}
|
||||
}
|
||||
|
||||
func removeEntries(by name: String) {
|
||||
while let index = (entries.firstIndex { $0.view.attributes.name == name }) {
|
||||
entries.remove(at: index)
|
||||
}
|
||||
}
|
||||
|
||||
func remove(entry: CachedEntry) {
|
||||
guard let index = (entries.firstIndex { $0.view == entry.view }) else {
|
||||
return
|
||||
}
|
||||
entries.remove(at: index)
|
||||
}
|
||||
|
||||
func removeAll() {
|
||||
entries.removeAll()
|
||||
}
|
||||
}
|
||||
|
||||
class EKEntryChronologicalQueue: EntryCachingHeuristic {
|
||||
|
||||
var entries: [CachedEntry] = []
|
||||
|
||||
func enqueue(entry: CachedEntry) {
|
||||
entries.append(entry)
|
||||
}
|
||||
}
|
||||
|
||||
class EKEntryPriorityQueue: EntryCachingHeuristic {
|
||||
|
||||
var entries: [CachedEntry] = []
|
||||
|
||||
func enqueue(entry: CachedEntry) {
|
||||
let entryPriority = entry.view.attributes.precedence.priority
|
||||
let index = entries.firstIndex {
|
||||
return entryPriority > $0.view.attributes.precedence.priority
|
||||
}
|
||||
if let index = index {
|
||||
entries.insert(entry, at: index)
|
||||
} else {
|
||||
entries.append(entry)
|
||||
}
|
||||
}
|
||||
}
|
||||
90
Pods/SwiftEntryKit/Source/MessageViews/EKAlertMessageView.swift
generated
Normal file
90
Pods/SwiftEntryKit/Source/MessageViews/EKAlertMessageView.swift
generated
Normal file
@@ -0,0 +1,90 @@
|
||||
//
|
||||
// EKAlertMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/5/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final public class EKAlertMessageView: EKSimpleMessageView, EntryAppearanceDescriptor {
|
||||
|
||||
// MARK: Props
|
||||
var buttonBarView: EKButtonBarView!
|
||||
private var buttonsBarCompressedConstraint: NSLayoutConstraint!
|
||||
private var buttonsBarExpandedConstraint: NSLayoutConstraint!
|
||||
|
||||
// MARK: EntryAppearenceDescriptor
|
||||
|
||||
var bottomCornerRadius: CGFloat = 0 {
|
||||
didSet {
|
||||
buttonBarView.bottomCornerRadius = bottomCornerRadius
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Setup
|
||||
public init(with message: EKAlertMessage) {
|
||||
super.init(with: message.simpleMessage)
|
||||
setupButtonBarView(with: message.buttonBarContent)
|
||||
layoutContent(with: message)
|
||||
}
|
||||
|
||||
public required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupButtonBarView(with content: EKProperty.ButtonBarContent) {
|
||||
buttonBarView = EKButtonBarView(with: content)
|
||||
buttonBarView.clipsToBounds = true
|
||||
addSubview(buttonBarView)
|
||||
}
|
||||
|
||||
func layoutContent(with message: EKAlertMessage) {
|
||||
switch message.imagePosition {
|
||||
case .top:
|
||||
messageContentView.verticalMargins = 16
|
||||
messageContentView.horizontalMargins = 16
|
||||
messageContentView.labelsOffset = 5
|
||||
|
||||
if let thumbImageView = thumbImageView {
|
||||
thumbImageView.layoutToSuperview(.top, offset: 20)
|
||||
thumbImageView.layoutToSuperview(.centerX)
|
||||
messageContentView.layout(.top, to: .bottom, of: thumbImageView)
|
||||
} else {
|
||||
messageContentView.layoutToSuperview(.top)
|
||||
}
|
||||
|
||||
messageContentView.layoutToSuperview(axis: .horizontally)
|
||||
buttonBarView.layout(.top, to: .bottom, of: messageContentView)
|
||||
case .left:
|
||||
messageContentView.verticalMargins = 0
|
||||
messageContentView.horizontalMargins = 0
|
||||
messageContentView.labelsOffset = 5
|
||||
|
||||
if let thumbImageView = thumbImageView {
|
||||
thumbImageView.layoutToSuperview(.top, .left, offset: 16)
|
||||
messageContentView.layout(.left, to: .right, of: thumbImageView, offset: 12)
|
||||
messageContentView.layout(to: .top, of: thumbImageView, offset: 2)
|
||||
} else {
|
||||
messageContentView.layoutToSuperview(.left, .top, offset: 16)
|
||||
}
|
||||
|
||||
messageContentView.layoutToSuperview(.right, offset: -16)
|
||||
buttonBarView.layout(.top, to: .bottom, of: messageContentView, offset: 10)
|
||||
}
|
||||
|
||||
buttonBarView.layoutToSuperview(axis: .horizontally)
|
||||
buttonBarView.layoutToSuperview(.bottom)
|
||||
buttonBarView.alpha = 0
|
||||
|
||||
if !message.buttonBarContent.content.isEmpty {
|
||||
if message.buttonBarContent.expandAnimatedly {
|
||||
DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) {
|
||||
self.buttonBarView.expand()
|
||||
}
|
||||
} else {
|
||||
buttonBarView.expand()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
130
Pods/SwiftEntryKit/Source/MessageViews/EKFormMessageView.swift
generated
Normal file
130
Pods/SwiftEntryKit/Source/MessageViews/EKFormMessageView.swift
generated
Normal file
@@ -0,0 +1,130 @@
|
||||
//
|
||||
// EKFormMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/15/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final public class EKFormMessageView: UIView {
|
||||
|
||||
private let scrollViewVerticalOffset: CGFloat = 20
|
||||
|
||||
// MARK: Props
|
||||
|
||||
private let titleLabel = UILabel()
|
||||
private let scrollView = UIScrollView()
|
||||
private let textFieldsContent: [EKProperty.TextFieldContent]
|
||||
private var textFieldViews: [EKTextField] = []
|
||||
private var buttonBarView: EKButtonBarView!
|
||||
|
||||
private let titleContent: EKProperty.LabelContent
|
||||
|
||||
// MARK: Setup
|
||||
|
||||
public init(with title: EKProperty.LabelContent,
|
||||
textFieldsContent: [EKProperty.TextFieldContent],
|
||||
buttonContent: EKProperty.ButtonContent) {
|
||||
self.titleContent = title
|
||||
self.textFieldsContent = textFieldsContent
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setupScrollView()
|
||||
setupTitleLabel()
|
||||
setupTextFields(with: textFieldsContent)
|
||||
setupButton(with: buttonContent)
|
||||
setupTapGestureRecognizer()
|
||||
scrollView.layoutIfNeeded()
|
||||
set(.height,
|
||||
of: scrollView.contentSize.height + scrollViewVerticalOffset * 2,
|
||||
priority: .defaultHigh)
|
||||
}
|
||||
|
||||
required public init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupTextFields(with textFieldsContent: [EKProperty.TextFieldContent]) {
|
||||
var textFieldIndex = 0
|
||||
textFieldViews = textFieldsContent.map { content -> EKTextField in
|
||||
let textField = EKTextField(with: content)
|
||||
scrollView.addSubview(textField)
|
||||
textField.tag = textFieldIndex
|
||||
textFieldIndex += 1
|
||||
return textField
|
||||
}
|
||||
textFieldViews.first!.layout(.top, to: .bottom, of: titleLabel, offset: 20)
|
||||
textFieldViews.spread(.vertically, offset: 5)
|
||||
textFieldViews.layoutToSuperview(axis: .horizontally)
|
||||
}
|
||||
|
||||
// Setup tap gesture
|
||||
private func setupTapGestureRecognizer() {
|
||||
let tapGestureRecognizer = UITapGestureRecognizer(
|
||||
target: self,
|
||||
action: #selector(tapGestureRecognized)
|
||||
)
|
||||
tapGestureRecognizer.numberOfTapsRequired = 1
|
||||
addGestureRecognizer(tapGestureRecognizer)
|
||||
}
|
||||
|
||||
private func setupScrollView() {
|
||||
addSubview(scrollView)
|
||||
scrollView.layoutToSuperview(axis: .horizontally, offset: 20)
|
||||
scrollView.layoutToSuperview(axis: .vertically, offset: scrollViewVerticalOffset)
|
||||
scrollView.layoutToSuperview(.width, .height, offset: -scrollViewVerticalOffset * 2)
|
||||
}
|
||||
|
||||
private func setupTitleLabel() {
|
||||
scrollView.addSubview(titleLabel)
|
||||
titleLabel.layoutToSuperview(.top, .width)
|
||||
titleLabel.layoutToSuperview(axis: .horizontally)
|
||||
titleLabel.forceContentWrap(.vertically)
|
||||
titleLabel.content = titleContent
|
||||
}
|
||||
|
||||
private func setupButton(with buttonContent: EKProperty.ButtonContent) {
|
||||
var buttonContent = buttonContent
|
||||
let action = buttonContent.action
|
||||
buttonContent.action = { [weak self] in
|
||||
self?.extractTextFieldsContent()
|
||||
action?()
|
||||
}
|
||||
let buttonsBarContent = EKProperty.ButtonBarContent(
|
||||
with: buttonContent,
|
||||
separatorColor: .clear,
|
||||
expandAnimatedly: true
|
||||
)
|
||||
buttonBarView = EKButtonBarView(with: buttonsBarContent)
|
||||
buttonBarView.clipsToBounds = true
|
||||
scrollView.addSubview(buttonBarView)
|
||||
buttonBarView.expand()
|
||||
buttonBarView.layout(.top, to: .bottom, of: textFieldViews.last!, offset: 20)
|
||||
buttonBarView.layoutToSuperview(.centerX)
|
||||
buttonBarView.layoutToSuperview(.width, offset: -40)
|
||||
buttonBarView.layoutToSuperview(.bottom)
|
||||
buttonBarView.layer.cornerRadius = 5
|
||||
}
|
||||
|
||||
private func extractTextFieldsContent() {
|
||||
for (content, textField) in zip(textFieldsContent, textFieldViews) {
|
||||
content.contentWrapper.text = textField.text
|
||||
}
|
||||
}
|
||||
|
||||
/** Makes a specific text field the first responder */
|
||||
public func becomeFirstResponder(with textFieldIndex: Int) {
|
||||
textFieldViews[textFieldIndex].makeFirstResponder()
|
||||
}
|
||||
|
||||
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
titleLabel.textColor = titleContent.style.color(for: traitCollection)
|
||||
}
|
||||
|
||||
// MARK: User Intractions
|
||||
|
||||
// Tap Gesture
|
||||
@objc func tapGestureRecognized() {
|
||||
endEditing(true)
|
||||
}
|
||||
}
|
||||
119
Pods/SwiftEntryKit/Source/MessageViews/EKMessageContentView.swift
generated
Normal file
119
Pods/SwiftEntryKit/Source/MessageViews/EKMessageContentView.swift
generated
Normal file
@@ -0,0 +1,119 @@
|
||||
//
|
||||
// EKMessageContentView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public class EKMessageContentView: UIView {
|
||||
|
||||
// MARK: Properties
|
||||
|
||||
private let titleLabel = UILabel()
|
||||
private let subtitleLabel = UILabel()
|
||||
|
||||
private var horizontalConstraints: QLAxisConstraints!
|
||||
private var topConstraint: NSLayoutConstraint!
|
||||
private var bottomConstraint: NSLayoutConstraint!
|
||||
private var labelsOffsetConstraint: NSLayoutConstraint!
|
||||
|
||||
public var titleContent: EKProperty.LabelContent! {
|
||||
didSet {
|
||||
titleLabel.content = titleContent
|
||||
}
|
||||
}
|
||||
|
||||
public var subtitleContent: EKProperty.LabelContent! {
|
||||
didSet {
|
||||
subtitleLabel.content = subtitleContent
|
||||
}
|
||||
}
|
||||
|
||||
public var titleAttributes: EKProperty.LabelStyle! {
|
||||
didSet {
|
||||
titleLabel.style = titleAttributes
|
||||
}
|
||||
}
|
||||
|
||||
public var subtitleAttributes: EKProperty.LabelStyle! {
|
||||
didSet {
|
||||
subtitleLabel.style = subtitleAttributes
|
||||
}
|
||||
}
|
||||
|
||||
public var title: String! {
|
||||
didSet {
|
||||
titleLabel.text = title
|
||||
}
|
||||
}
|
||||
|
||||
public var subtitle: String! {
|
||||
didSet {
|
||||
subtitleLabel.text = subtitle
|
||||
}
|
||||
}
|
||||
|
||||
public var verticalMargins: CGFloat = 20 {
|
||||
didSet {
|
||||
topConstraint.constant = verticalMargins
|
||||
bottomConstraint.constant = -verticalMargins
|
||||
layoutIfNeeded()
|
||||
}
|
||||
}
|
||||
|
||||
public var horizontalMargins: CGFloat = 20 {
|
||||
didSet {
|
||||
horizontalConstraints.first.constant = horizontalMargins
|
||||
horizontalConstraints.second.constant = -horizontalMargins
|
||||
layoutIfNeeded()
|
||||
}
|
||||
}
|
||||
|
||||
public var labelsOffset: CGFloat = 8 {
|
||||
didSet {
|
||||
labelsOffsetConstraint.constant = labelsOffset
|
||||
layoutIfNeeded()
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Setup
|
||||
|
||||
public init() {
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
clipsToBounds = true
|
||||
setupTitleLabel()
|
||||
setupSubtitleLabel()
|
||||
}
|
||||
|
||||
public required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupTitleLabel() {
|
||||
addSubview(titleLabel)
|
||||
topConstraint = titleLabel.layoutToSuperview(.top, offset: verticalMargins)
|
||||
horizontalConstraints = titleLabel.layoutToSuperview(axis: .horizontally, offset: horizontalMargins)
|
||||
titleLabel.forceContentWrap(.vertically)
|
||||
}
|
||||
|
||||
private func setupSubtitleLabel() {
|
||||
addSubview(subtitleLabel)
|
||||
labelsOffsetConstraint = subtitleLabel.layout(.top, to: .bottom, of: titleLabel, offset: labelsOffset)
|
||||
subtitleLabel.layout(to: .left, of: titleLabel)
|
||||
subtitleLabel.layout(to: .right, of: titleLabel)
|
||||
bottomConstraint = subtitleLabel.layoutToSuperview(.bottom, offset: -verticalMargins, priority: .must)
|
||||
subtitleLabel.forceContentWrap(.vertically)
|
||||
}
|
||||
|
||||
private func setupInterfaceStyle() {
|
||||
titleLabel.textColor = titleContent?.style.color(for: traitCollection)
|
||||
subtitleLabel.textColor = subtitleContent?.style.color(for: traitCollection)
|
||||
}
|
||||
|
||||
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
setupInterfaceStyle()
|
||||
}
|
||||
}
|
||||
71
Pods/SwiftEntryKit/Source/MessageViews/EKNotificationMessageView.swift
generated
Normal file
71
Pods/SwiftEntryKit/Source/MessageViews/EKNotificationMessageView.swift
generated
Normal file
@@ -0,0 +1,71 @@
|
||||
//
|
||||
// EKNotificationMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final public class EKNotificationMessageView: EKSimpleMessageView {
|
||||
|
||||
// MARK: Props
|
||||
private var auxLabel: UILabel!
|
||||
private var auxiliaryContent: EKProperty.LabelContent!
|
||||
|
||||
private let message: EKNotificationMessage
|
||||
|
||||
// MARK: Setup
|
||||
public init(with message: EKNotificationMessage) {
|
||||
self.message = message
|
||||
super.init(with: message.simpleMessage)
|
||||
setupAuxLabel(with: message.auxiliary)
|
||||
layoutContent(with: message.insets)
|
||||
}
|
||||
|
||||
public required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupAuxLabel(with content: EKProperty.LabelContent?) {
|
||||
auxiliaryContent = content
|
||||
guard let content = content else {
|
||||
return
|
||||
}
|
||||
auxLabel = UILabel()
|
||||
auxLabel.content = content
|
||||
addSubview(auxLabel)
|
||||
}
|
||||
|
||||
private func layoutContent(with insets: EKNotificationMessage.Insets) {
|
||||
messageContentView.verticalMargins = 0
|
||||
messageContentView.horizontalMargins = 0
|
||||
messageContentView.labelsOffset = insets.titleToDescription
|
||||
|
||||
if let thumbImageView = thumbImageView {
|
||||
thumbImageView.layoutToSuperview(.left, offset: insets.contentInsets.left)
|
||||
thumbImageView.layoutToSuperview(.top, offset: insets.contentInsets.top)
|
||||
messageContentView.layout(.left, to: .right, of: thumbImageView, offset: 12)
|
||||
messageContentView.layout(to: .top, of: thumbImageView, offset: 4)
|
||||
} else {
|
||||
messageContentView.layoutToSuperview(.left, offset: insets.contentInsets.left)
|
||||
messageContentView.layoutToSuperview(.top, offset: insets.contentInsets.top)
|
||||
}
|
||||
|
||||
if let auxLabel = auxLabel {
|
||||
auxLabel.layoutToSuperview(.right, offset: -insets.contentInsets.right)
|
||||
auxLabel.layoutToSuperview(.top, offset: insets.contentInsets.top + 2)
|
||||
auxLabel.forceContentWrap()
|
||||
messageContentView.layout(.right, to: .left, of: auxLabel)
|
||||
} else {
|
||||
messageContentView.layoutToSuperview(.right, offset: -insets.contentInsets.right)
|
||||
}
|
||||
messageContentView.layoutToSuperview(.bottom, offset: -insets.contentInsets.bottom)
|
||||
}
|
||||
|
||||
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
super.traitCollectionDidChange(previousTraitCollection)
|
||||
auxLabel?.textColor = auxiliaryContent?.style.color(for: traitCollection)
|
||||
}
|
||||
}
|
||||
106
Pods/SwiftEntryKit/Source/MessageViews/EKPopUpMessageView.swift
generated
Normal file
106
Pods/SwiftEntryKit/Source/MessageViews/EKPopUpMessageView.swift
generated
Normal file
@@ -0,0 +1,106 @@
|
||||
//
|
||||
// EKPopUpMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/20/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final public class EKPopUpMessageView: UIView {
|
||||
|
||||
// MARK: - Properties
|
||||
|
||||
private var imageView: UIImageView!
|
||||
private let titleLabel = UILabel()
|
||||
private let descriptionLabel = UILabel()
|
||||
private let actionButton = UIButton()
|
||||
|
||||
private let message: EKPopUpMessage
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
public init(with message: EKPopUpMessage) {
|
||||
self.message = message
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setupImageView()
|
||||
setupTitleLabel()
|
||||
setupDescriptionLabel()
|
||||
setupActionButton()
|
||||
setupInterfaceStyle()
|
||||
}
|
||||
|
||||
public required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupImageView() {
|
||||
guard let themeImage = message.themeImage else {
|
||||
return
|
||||
}
|
||||
imageView = UIImageView()
|
||||
addSubview(imageView)
|
||||
imageView.layoutToSuperview(.centerX)
|
||||
switch themeImage.position {
|
||||
case .centerToTop(offset: let value):
|
||||
imageView.layout(.centerY, to: .top, of: self, offset: value)
|
||||
case .topToTop(offset: let value):
|
||||
imageView.layoutToSuperview(.top, offset: value)
|
||||
}
|
||||
imageView.imageContent = themeImage.image
|
||||
}
|
||||
|
||||
private func setupTitleLabel() {
|
||||
addSubview(titleLabel)
|
||||
titleLabel.content = message.title
|
||||
titleLabel.layoutToSuperview(axis: .horizontally, offset: 30)
|
||||
if let imageView = imageView {
|
||||
titleLabel.layout(.top, to: .bottom, of: imageView, offset: 20)
|
||||
} else {
|
||||
titleLabel.layoutToSuperview(.top, offset: 20)
|
||||
}
|
||||
titleLabel.forceContentWrap(.vertically)
|
||||
}
|
||||
|
||||
private func setupDescriptionLabel() {
|
||||
addSubview(descriptionLabel)
|
||||
descriptionLabel.content = message.description
|
||||
descriptionLabel.layoutToSuperview(axis: .horizontally, offset: 30)
|
||||
descriptionLabel.layout(.top, to: .bottom, of: titleLabel, offset: 16)
|
||||
descriptionLabel.forceContentWrap(.vertically)
|
||||
}
|
||||
|
||||
private func setupActionButton() {
|
||||
addSubview(actionButton)
|
||||
let height: CGFloat = 45
|
||||
actionButton.set(.height, of: height)
|
||||
actionButton.layout(.top, to: .bottom, of: descriptionLabel, offset: 30)
|
||||
actionButton.layoutToSuperview(.bottom, offset: -30)
|
||||
actionButton.layoutToSuperview(.centerX)
|
||||
|
||||
let buttonAttributes = message.button
|
||||
actionButton.buttonContent = buttonAttributes
|
||||
actionButton.contentEdgeInsets = UIEdgeInsets(top: 0, left: 30, bottom: 0, right: 30)
|
||||
actionButton.layer.cornerRadius = height * 0.5
|
||||
actionButton.addTarget(self, action: #selector(actionButtonPressed), for: .touchUpInside)
|
||||
}
|
||||
|
||||
private func setupInterfaceStyle() {
|
||||
titleLabel.textColor = message.title.style.color(for: traitCollection)
|
||||
imageView?.tintColor = message.themeImage?.image.tintColor(for: traitCollection)
|
||||
let tapColor = message.button.highlighedLabelColor(for: traitCollection)
|
||||
actionButton.setTitleColor(tapColor, for: .highlighted)
|
||||
actionButton.setTitleColor(tapColor, for: .selected)
|
||||
}
|
||||
|
||||
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
setupInterfaceStyle()
|
||||
}
|
||||
|
||||
// MARK: - User Interaction
|
||||
|
||||
@objc func actionButtonPressed() {
|
||||
message.action()
|
||||
}
|
||||
}
|
||||
110
Pods/SwiftEntryKit/Source/MessageViews/EKRatingMessageView.swift
generated
Normal file
110
Pods/SwiftEntryKit/Source/MessageViews/EKRatingMessageView.swift
generated
Normal file
@@ -0,0 +1,110 @@
|
||||
//
|
||||
// EKRatingMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 6/1/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final public class EKRatingMessageView: UIView, EntryAppearanceDescriptor {
|
||||
|
||||
// MARK: Properties
|
||||
|
||||
private var message: EKRatingMessage
|
||||
|
||||
// MARK: EntryAppearenceDescriptor
|
||||
|
||||
var bottomCornerRadius: CGFloat = 0 {
|
||||
didSet {
|
||||
buttonBarView.bottomCornerRadius = bottomCornerRadius
|
||||
}
|
||||
}
|
||||
|
||||
private var selectedIndex: Int! {
|
||||
didSet {
|
||||
message.selectedIndex = selectedIndex
|
||||
let item = message.ratingItems[selectedIndex]
|
||||
set(title: item.title,
|
||||
description: item.description)
|
||||
}
|
||||
}
|
||||
|
||||
private let messageContentView = EKMessageContentView()
|
||||
private let symbolsView = EKRatingSymbolsContainerView()
|
||||
private var buttonBarView: EKButtonBarView!
|
||||
|
||||
public init(with message: EKRatingMessage) {
|
||||
self.message = message
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setupMessageContentView()
|
||||
setupSymbolsView()
|
||||
setupButtonBarView()
|
||||
set(title: message.initialTitle,
|
||||
description: message.initialDescription)
|
||||
}
|
||||
|
||||
required public init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func set(title: EKProperty.LabelContent,
|
||||
description: EKProperty.LabelContent) {
|
||||
messageContentView.titleContent = title
|
||||
messageContentView.subtitleContent = description
|
||||
UIView.animate(withDuration: 0.4,
|
||||
delay: 0,
|
||||
usingSpringWithDamping: 1,
|
||||
initialSpringVelocity: 0,
|
||||
options: [.transitionCrossDissolve],
|
||||
animations: {
|
||||
SwiftEntryKit.layoutIfNeeded()
|
||||
}, completion: nil)
|
||||
}
|
||||
|
||||
private func setupMessageContentView() {
|
||||
addSubview(messageContentView)
|
||||
messageContentView.verticalMargins = 20
|
||||
messageContentView.horizontalMargins = 30
|
||||
messageContentView.layoutToSuperview(axis: .horizontally,
|
||||
priority: .must)
|
||||
messageContentView.layoutToSuperview(.top, offset: 10)
|
||||
}
|
||||
|
||||
private func setupSymbolsView() {
|
||||
addSubview(symbolsView)
|
||||
symbolsView.setup(with: message) { [unowned self] (index: Int) in
|
||||
self.message.selectedIndex = index
|
||||
self.message.selection?(index)
|
||||
self.selectedIndex = index
|
||||
self.animateIn()
|
||||
}
|
||||
symbolsView.layoutToSuperview(.centerX)
|
||||
symbolsView.layout(.top,
|
||||
to: .bottom,
|
||||
of: messageContentView,
|
||||
offset: 10,
|
||||
priority: .must)
|
||||
}
|
||||
|
||||
private func setupButtonBarView() {
|
||||
buttonBarView = EKButtonBarView(with: message.buttonBarContent)
|
||||
buttonBarView.clipsToBounds = true
|
||||
buttonBarView.alpha = 0
|
||||
addSubview(buttonBarView)
|
||||
buttonBarView.layout(.top,
|
||||
to: .bottom,
|
||||
of: symbolsView,
|
||||
offset: 30)
|
||||
buttonBarView.layoutToSuperview(.bottom)
|
||||
buttonBarView.layoutToSuperview(axis: .horizontally)
|
||||
}
|
||||
|
||||
// MARK: - Internal Animation
|
||||
|
||||
private func animateIn() {
|
||||
layoutIfNeeded()
|
||||
buttonBarView.expand()
|
||||
}
|
||||
}
|
||||
58
Pods/SwiftEntryKit/Source/MessageViews/EKSimpleMessageView.swift
generated
Normal file
58
Pods/SwiftEntryKit/Source/MessageViews/EKSimpleMessageView.swift
generated
Normal file
@@ -0,0 +1,58 @@
|
||||
//
|
||||
// EKMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/5/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public class EKSimpleMessageView: UIView {
|
||||
|
||||
// MARK: Props
|
||||
var thumbImageView: UIImageView!
|
||||
let messageContentView = EKMessageContentView()
|
||||
private let message: EKSimpleMessage
|
||||
|
||||
// MARK: Setup
|
||||
init(with message: EKSimpleMessage) {
|
||||
self.message = message
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setupThumbImageView(with: message.image)
|
||||
setupMessageContentView(with: message.title,
|
||||
description: message.description)
|
||||
}
|
||||
|
||||
required public init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupThumbImageView(with content: EKProperty.ImageContent?) {
|
||||
guard let content = content else {
|
||||
return
|
||||
}
|
||||
thumbImageView = UIImageView()
|
||||
addSubview(thumbImageView)
|
||||
thumbImageView.imageContent = content
|
||||
}
|
||||
|
||||
private func setupMessageContentView(with title: EKProperty.LabelContent,
|
||||
description: EKProperty.LabelContent) {
|
||||
messageContentView.titleContent = title
|
||||
messageContentView.subtitleContent = description
|
||||
addSubview(messageContentView)
|
||||
}
|
||||
|
||||
private func setupInterfaceStyle() {
|
||||
if let image = message.image {
|
||||
thumbImageView?.tintColor = image.tint?.color(
|
||||
for: traitCollection,
|
||||
mode: image.displayMode
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
setupInterfaceStyle()
|
||||
}
|
||||
}
|
||||
191
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKButtonBarView.swift
generated
Normal file
191
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKButtonBarView.swift
generated
Normal file
@@ -0,0 +1,191 @@
|
||||
//
|
||||
// ButtonsBarView.swift
|
||||
// SwiftEntryKit_Example
|
||||
//
|
||||
// Created by Daniel Huri on 4/28/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
/**
|
||||
Dynamic button bar view
|
||||
Buttons are set according to the received content.
|
||||
1-2 buttons spread horizontally
|
||||
3 or more buttons spread vertically
|
||||
*/
|
||||
final public class EKButtonBarView: UIView {
|
||||
|
||||
// MARK: - Properties
|
||||
|
||||
private var buttonViews: [EKButtonView] = []
|
||||
private var separatorViews: [UIView] = []
|
||||
|
||||
private let buttonBarContent: EKProperty.ButtonBarContent
|
||||
private let spreadAxis: QLAxis
|
||||
private let oppositeAxis: QLAxis
|
||||
private let relativeEdge: NSLayoutConstraint.Attribute
|
||||
|
||||
var bottomCornerRadius: CGFloat = 0 {
|
||||
didSet {
|
||||
adjustRoundCornersIfNecessary()
|
||||
}
|
||||
}
|
||||
|
||||
private lazy var buttonEdgeRatio: CGFloat = {
|
||||
return 1.0 / CGFloat(self.buttonBarContent.content.count)
|
||||
}()
|
||||
|
||||
private(set) lazy var intrinsicHeight: CGFloat = {
|
||||
var height: CGFloat = 0
|
||||
switch buttonBarContent.content.count {
|
||||
case 0:
|
||||
height += 1
|
||||
case 1...buttonBarContent.horizontalDistributionThreshold:
|
||||
height += buttonBarContent.buttonHeight
|
||||
default:
|
||||
for _ in 1...buttonBarContent.content.count {
|
||||
height += buttonBarContent.buttonHeight
|
||||
}
|
||||
}
|
||||
return height
|
||||
}()
|
||||
|
||||
private var compressedConstraint: NSLayoutConstraint!
|
||||
private lazy var expandedConstraint: NSLayoutConstraint = {
|
||||
return set(.height, of: intrinsicHeight, priority: .defaultLow)
|
||||
}()
|
||||
|
||||
// MARK: Setup
|
||||
required public init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
public init(with buttonBarContent: EKProperty.ButtonBarContent) {
|
||||
self.buttonBarContent = buttonBarContent
|
||||
if buttonBarContent.content.count <= buttonBarContent.horizontalDistributionThreshold {
|
||||
spreadAxis = .horizontally
|
||||
oppositeAxis = .vertically
|
||||
relativeEdge = .width
|
||||
} else {
|
||||
spreadAxis = .vertically
|
||||
oppositeAxis = .horizontally
|
||||
relativeEdge = .height
|
||||
}
|
||||
super.init(frame: .zero)
|
||||
setupButtonBarContent()
|
||||
setupSeparatorViews()
|
||||
|
||||
compressedConstraint = set(.height, of: 1, priority: .must)
|
||||
}
|
||||
|
||||
public override func layoutSubviews() {
|
||||
super.layoutSubviews()
|
||||
adjustRoundCornersIfNecessary()
|
||||
}
|
||||
|
||||
private func setupButtonBarContent() {
|
||||
for content in buttonBarContent.content {
|
||||
let buttonView = EKButtonView(content: content)
|
||||
addSubview(buttonView)
|
||||
buttonViews.append(buttonView)
|
||||
}
|
||||
layoutButtons()
|
||||
}
|
||||
|
||||
private func layoutButtons() {
|
||||
guard !buttonViews.isEmpty else {
|
||||
return
|
||||
}
|
||||
let suffix = Array(buttonViews.dropFirst())
|
||||
if !suffix.isEmpty {
|
||||
suffix.layout(.height, to: buttonViews.first!)
|
||||
}
|
||||
buttonViews.layoutToSuperview(axis: oppositeAxis)
|
||||
buttonViews.spread(spreadAxis, stretchEdgesToSuperview: true)
|
||||
buttonViews.layout(relativeEdge, to: self, ratio: buttonEdgeRatio, priority: .must)
|
||||
}
|
||||
|
||||
private func setupTopSeperatorView() {
|
||||
let topSeparatorView = UIView()
|
||||
addSubview(topSeparatorView)
|
||||
topSeparatorView.set(.height, of: 1)
|
||||
topSeparatorView.layoutToSuperview(.left, .right, .top)
|
||||
separatorViews.append(topSeparatorView)
|
||||
}
|
||||
|
||||
private func setupSeperatorView(after view: UIView) {
|
||||
let midSepView = UIView()
|
||||
addSubview(midSepView)
|
||||
let sepAttribute: NSLayoutConstraint.Attribute
|
||||
let buttonAttribute: NSLayoutConstraint.Attribute
|
||||
switch oppositeAxis {
|
||||
case .vertically:
|
||||
sepAttribute = .centerX
|
||||
buttonAttribute = .right
|
||||
case .horizontally:
|
||||
sepAttribute = .centerY
|
||||
buttonAttribute = .bottom
|
||||
}
|
||||
midSepView.layout(sepAttribute, to: buttonAttribute, of: view)
|
||||
midSepView.set(relativeEdge, of: 1)
|
||||
midSepView.layoutToSuperview(axis: oppositeAxis)
|
||||
separatorViews.append(midSepView)
|
||||
}
|
||||
|
||||
private func setupSeparatorViews() {
|
||||
setupTopSeperatorView()
|
||||
for button in buttonViews.dropLast() {
|
||||
setupSeperatorView(after: button)
|
||||
}
|
||||
setupInterfaceStyle()
|
||||
}
|
||||
|
||||
// Amination
|
||||
public func expand() {
|
||||
let expansion = {
|
||||
self.compressedConstraint.priority = .defaultLow
|
||||
self.expandedConstraint.priority = .must
|
||||
|
||||
/* NOTE: Calling layoutIfNeeded for the whole view hierarchy.
|
||||
Sometimes it's easier to just use frames instead of AutoLayout for
|
||||
hierarch complexity considerations. Here the animation influences almost the
|
||||
entire view hierarchy. */
|
||||
SwiftEntryKit.layoutIfNeeded()
|
||||
}
|
||||
|
||||
alpha = 1
|
||||
if buttonBarContent.expandAnimatedly {
|
||||
let damping: CGFloat = buttonBarContent.content.count <= 2 ? 0.4 : 0.8
|
||||
SwiftEntryKit.layoutIfNeeded()
|
||||
UIView.animate(withDuration: 0.8, delay: 0, usingSpringWithDamping: damping, initialSpringVelocity: 0, options: [.beginFromCurrentState, .allowUserInteraction, .layoutSubviews, .allowAnimatedContent], animations: {
|
||||
expansion()
|
||||
}, completion: nil)
|
||||
} else {
|
||||
expansion()
|
||||
}
|
||||
}
|
||||
|
||||
public func compress() {
|
||||
compressedConstraint.priority = .must
|
||||
expandedConstraint.priority = .defaultLow
|
||||
}
|
||||
|
||||
private func adjustRoundCornersIfNecessary() {
|
||||
let size = CGSize(width: bottomCornerRadius, height: bottomCornerRadius)
|
||||
let path = UIBezierPath(roundedRect: bounds, byRoundingCorners: .bottom, cornerRadii: size)
|
||||
let maskLayer = CAShapeLayer()
|
||||
maskLayer.path = path.cgPath
|
||||
layer.mask = maskLayer
|
||||
}
|
||||
|
||||
private func setupInterfaceStyle() {
|
||||
for view in separatorViews {
|
||||
view.backgroundColor = buttonBarContent.separatorColor(for: traitCollection)
|
||||
}
|
||||
}
|
||||
|
||||
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
setupInterfaceStyle()
|
||||
}
|
||||
}
|
||||
97
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKButtonView.swift
generated
Normal file
97
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKButtonView.swift
generated
Normal file
@@ -0,0 +1,97 @@
|
||||
//
|
||||
// EKButtonView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 12/8/18.
|
||||
// Copyright © 2018 CocoaPods. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final class EKButtonView: UIView {
|
||||
|
||||
// MARK: - Properties
|
||||
|
||||
private let button = UIButton()
|
||||
private let titleLabel = UILabel()
|
||||
|
||||
private let content: EKProperty.ButtonContent
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
init(content: EKProperty.ButtonContent) {
|
||||
self.content = content
|
||||
super.init(frame: .zero)
|
||||
setupTitleLabel()
|
||||
setupButton()
|
||||
setupAcceessibility()
|
||||
setupInterfaceStyle()
|
||||
}
|
||||
|
||||
required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupAcceessibility() {
|
||||
isAccessibilityElement = false
|
||||
button.isAccessibilityElement = true
|
||||
button.accessibilityIdentifier = content.accessibilityIdentifier
|
||||
button.accessibilityLabel = content.label.text
|
||||
}
|
||||
|
||||
private func setupButton() {
|
||||
addSubview(button)
|
||||
button.fillSuperview()
|
||||
button.addTarget(self, action: #selector(buttonTouchUp),
|
||||
for: [.touchUpInside, .touchUpOutside, .touchCancel])
|
||||
button.addTarget(self, action: #selector(buttonTouchDown),
|
||||
for: .touchDown)
|
||||
button.addTarget(self, action: #selector(buttonTouchUpInside),
|
||||
for: .touchUpInside)
|
||||
}
|
||||
|
||||
private func setupTitleLabel() {
|
||||
titleLabel.numberOfLines = content.label.style.numberOfLines
|
||||
titleLabel.font = content.label.style.font
|
||||
titleLabel.text = content.label.text
|
||||
titleLabel.textAlignment = .center
|
||||
titleLabel.lineBreakMode = .byWordWrapping
|
||||
addSubview(titleLabel)
|
||||
titleLabel.layoutToSuperview(axis: .horizontally,
|
||||
offset: content.contentEdgeInset)
|
||||
titleLabel.layoutToSuperview(axis: .vertically,
|
||||
offset: content.contentEdgeInset)
|
||||
}
|
||||
|
||||
private func setBackground(by content: EKProperty.ButtonContent,
|
||||
isHighlighted: Bool) {
|
||||
if isHighlighted {
|
||||
backgroundColor = content.highlightedBackgroundColor(for: traitCollection)
|
||||
} else {
|
||||
backgroundColor = content.backgroundColor(for: traitCollection)
|
||||
}
|
||||
}
|
||||
|
||||
private func setupInterfaceStyle() {
|
||||
backgroundColor = content.backgroundColor(for: traitCollection)
|
||||
titleLabel.textColor = content.label.style.color(for: traitCollection)
|
||||
}
|
||||
|
||||
// MARK: - Selectors
|
||||
|
||||
@objc func buttonTouchUpInside() {
|
||||
content.action?()
|
||||
}
|
||||
|
||||
@objc func buttonTouchDown() {
|
||||
setBackground(by: content, isHighlighted: true)
|
||||
}
|
||||
|
||||
@objc func buttonTouchUp() {
|
||||
setBackground(by: content, isHighlighted: false)
|
||||
}
|
||||
|
||||
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
setupInterfaceStyle()
|
||||
}
|
||||
}
|
||||
69
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKRatingSymbolView.swift
generated
Normal file
69
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKRatingSymbolView.swift
generated
Normal file
@@ -0,0 +1,69 @@
|
||||
//
|
||||
// EKRatingSymbolView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 6/1/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final public class EKRatingSymbolView: UIView {
|
||||
|
||||
private let button = UIButton()
|
||||
private let imageView = UIImageView()
|
||||
|
||||
private let unselectedImage: EKProperty.ImageContent
|
||||
private let selectedImage: EKProperty.ImageContent
|
||||
|
||||
var selection: EKRatingMessage.Selection
|
||||
|
||||
public var isSelected: Bool {
|
||||
set {
|
||||
imageView.imageContent = newValue ? selectedImage : unselectedImage
|
||||
}
|
||||
get {
|
||||
return imageView.image == selectedImage.images.first
|
||||
}
|
||||
}
|
||||
|
||||
public init(unselectedImage: EKProperty.ImageContent, selectedImage: EKProperty.ImageContent, selection: @escaping EKRatingMessage.Selection) {
|
||||
self.unselectedImage = unselectedImage
|
||||
self.selectedImage = selectedImage
|
||||
self.selection = selection
|
||||
super.init(frame: .zero)
|
||||
setupImageView()
|
||||
setupButton()
|
||||
}
|
||||
|
||||
required public init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupButton() {
|
||||
addSubview(button)
|
||||
button.fillSuperview()
|
||||
button.addTarget(self, action: #selector(touchUpInside), for: .touchUpInside)
|
||||
button.addTarget(self, action: #selector(touchDown), for: [.touchDown])
|
||||
button.addTarget(self, action: #selector(touchUp), for: [.touchUpInside, .touchUpOutside, .touchCancel])
|
||||
}
|
||||
|
||||
private func setupImageView() {
|
||||
addSubview(imageView)
|
||||
imageView.imageContent = unselectedImage
|
||||
imageView.centerInSuperview()
|
||||
imageView.sizeToSuperview(withRatio: 0.7)
|
||||
}
|
||||
|
||||
@objc func touchUpInside() {
|
||||
selection(tag)
|
||||
}
|
||||
|
||||
@objc func touchDown() {
|
||||
transform = CGAffineTransform(scaleX: 1.15, y: 1.15)
|
||||
}
|
||||
|
||||
@objc func touchUp() {
|
||||
transform = CGAffineTransform(scaleX: 1, y: 1)
|
||||
}
|
||||
}
|
||||
62
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKRatingSymbolsContainerView.swift
generated
Normal file
62
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKRatingSymbolsContainerView.swift
generated
Normal file
@@ -0,0 +1,62 @@
|
||||
//
|
||||
// EKRatingSymbolsContainerView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 6/1/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
final public class EKRatingSymbolsContainerView: UIView {
|
||||
|
||||
private var message: EKRatingMessage!
|
||||
private var symbolsArray: [EKRatingSymbolView] = []
|
||||
|
||||
public func setup(with message: EKRatingMessage,
|
||||
externalSelection: @escaping EKRatingMessage.Selection) {
|
||||
self.message = message
|
||||
let internalSelection = { [unowned self] (index: Int) in
|
||||
self.select(index: index)
|
||||
externalSelection(index)
|
||||
}
|
||||
|
||||
for (index, item) in message.ratingItems.enumerated() {
|
||||
let itemView = EKRatingSymbolView(unselectedImage: item.unselectedImage,
|
||||
selectedImage: item.selectedImage,
|
||||
selection: internalSelection)
|
||||
itemView.tag = index
|
||||
addSubview(itemView)
|
||||
itemView.set(.height, of: item.size.height)
|
||||
itemView.set(.width, of: item.size.width)
|
||||
symbolsArray.append(itemView)
|
||||
}
|
||||
symbolsArray.layoutToSuperview(axis: .vertically, priority: .must)
|
||||
symbolsArray.spread(.horizontally, stretchEdgesToSuperview: true)
|
||||
|
||||
select(index: message.selectedIndex)
|
||||
}
|
||||
|
||||
private func select(index: Int? = nil) {
|
||||
var delay: TimeInterval = 0
|
||||
for (i, view) in symbolsArray.enumerated() {
|
||||
DispatchQueue.main.asyncAfter(deadline: .now() + delay) {
|
||||
if let index = index, i <= index {
|
||||
view.isSelected = true
|
||||
view.transform = CGAffineTransform(scaleX: 1.5, y: 1.5)
|
||||
} else if view.isSelected || index == nil {
|
||||
view.isSelected = false
|
||||
view.transform = CGAffineTransform(scaleX: 0.5, y: 0.5)
|
||||
}
|
||||
UIView.animate(withDuration: 0.6,
|
||||
delay: 0,
|
||||
usingSpringWithDamping: 0.5,
|
||||
initialSpringVelocity: 0,
|
||||
options: [.allowUserInteraction], animations: {
|
||||
view.transform = .identity
|
||||
}, completion: nil)
|
||||
}
|
||||
delay += 0.05
|
||||
}
|
||||
}
|
||||
}
|
||||
88
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKTextField.swift
generated
Normal file
88
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EKTextField.swift
generated
Normal file
@@ -0,0 +1,88 @@
|
||||
//
|
||||
// EKTextField.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/16/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
final public class EKTextField: UIView {
|
||||
|
||||
// MARK: - Properties
|
||||
|
||||
static let totalHeight: CGFloat = 45
|
||||
|
||||
private let content: EKProperty.TextFieldContent
|
||||
|
||||
private let imageView = UIImageView()
|
||||
private let textField = UITextField()
|
||||
private let separatorView = UIView()
|
||||
|
||||
public var text: String {
|
||||
set {
|
||||
textField.text = newValue
|
||||
}
|
||||
get {
|
||||
return textField.text ?? ""
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
public init(with content: EKProperty.TextFieldContent) {
|
||||
self.content = content
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setupImageView()
|
||||
setupTextField()
|
||||
setupSeparatorView()
|
||||
textField.accessibilityIdentifier = content.accessibilityIdentifier
|
||||
}
|
||||
|
||||
required public init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setupImageView() {
|
||||
addSubview(imageView)
|
||||
imageView.contentMode = .center
|
||||
imageView.set(.width, .height, of: EKTextField.totalHeight)
|
||||
imageView.layoutToSuperview(.leading)
|
||||
imageView.image = content.leadingImage
|
||||
imageView.tintColor = content.tintColor(for: traitCollection)
|
||||
}
|
||||
|
||||
private func setupTextField() {
|
||||
addSubview(textField)
|
||||
textField.textFieldContent = content
|
||||
textField.delegate = content.delegate
|
||||
textField.set(.height, of: EKTextField.totalHeight)
|
||||
textField.layout(.leading, to: .trailing, of: imageView)
|
||||
textField.layoutToSuperview(.top, .trailing)
|
||||
imageView.layout(to: .centerY, of: textField)
|
||||
}
|
||||
|
||||
private func setupSeparatorView() {
|
||||
addSubview(separatorView)
|
||||
separatorView.layout(.top, to: .bottom, of: textField)
|
||||
separatorView.set(.height, of: 1)
|
||||
separatorView.layoutToSuperview(.bottom)
|
||||
separatorView.layoutToSuperview(axis: .horizontally, offset: 10)
|
||||
separatorView.backgroundColor = content.bottomBorderColor.color(
|
||||
for: traitCollection,
|
||||
mode: content.displayMode
|
||||
)
|
||||
}
|
||||
|
||||
public func makeFirstResponder() {
|
||||
textField.becomeFirstResponder()
|
||||
}
|
||||
|
||||
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
separatorView.backgroundColor = content.bottomBorderColor(for: traitCollection)
|
||||
imageView.tintColor = content.tintColor(for: traitCollection)
|
||||
textField.textColor = content.textStyle.color(for: traitCollection)
|
||||
textField.placeholder = content.placeholder
|
||||
}
|
||||
}
|
||||
18
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EntryAppearanceDescriptor.swift
generated
Normal file
18
Pods/SwiftEntryKit/Source/MessageViews/MessagesUtils/EntryAppearanceDescriptor.swift
generated
Normal file
@@ -0,0 +1,18 @@
|
||||
//
|
||||
// EntryAppearanceDescriptor.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 1/5/19.
|
||||
// Copyright © 2019 CocoaPods. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
/**
|
||||
An anti-pattern for SwiftEntryKit views to know more about their appearence,
|
||||
if necessary, since views don't have access to EKAttributes.
|
||||
This is a solution to bug #117 (round buttons in alert)
|
||||
*/
|
||||
protocol EntryAppearanceDescriptor: AnyObject {
|
||||
var bottomCornerRadius: CGFloat { get set }
|
||||
}
|
||||
35
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKAccessoryNoteMessageView.swift
generated
Normal file
35
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKAccessoryNoteMessageView.swift
generated
Normal file
@@ -0,0 +1,35 @@
|
||||
//
|
||||
// EKAccessoryNoteMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/4/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public class EKAccessoryNoteMessageView: UIView {
|
||||
|
||||
// MARK: Props
|
||||
private let contentView = UIView()
|
||||
private var noteMessageView: EKNoteMessageView!
|
||||
var accessoryView: UIView!
|
||||
|
||||
func setup(with content: EKProperty.LabelContent) {
|
||||
clipsToBounds = true
|
||||
|
||||
addSubview(contentView)
|
||||
contentView.layoutToSuperview(.centerX, .top, .bottom)
|
||||
contentView.layoutToSuperview(.left, relation: .greaterThanOrEqual, offset: 16)
|
||||
contentView.layoutToSuperview(.right, relation: .lessThanOrEqual, offset: -16)
|
||||
|
||||
noteMessageView = EKNoteMessageView(with: content)
|
||||
noteMessageView.horizontalOffset = 8
|
||||
noteMessageView.verticalOffset = 7
|
||||
contentView.addSubview(noteMessageView)
|
||||
noteMessageView.layoutToSuperview(.top, .bottom, .trailing)
|
||||
|
||||
contentView.addSubview(accessoryView)
|
||||
accessoryView.layoutToSuperview(.leading, .centerY)
|
||||
accessoryView.layout(.trailing, to: .leading, of: noteMessageView)
|
||||
}
|
||||
}
|
||||
28
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKImageNoteMessageView.swift
generated
Normal file
28
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKImageNoteMessageView.swift
generated
Normal file
@@ -0,0 +1,28 @@
|
||||
//
|
||||
// EKImageNoteMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/4/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public class EKImageNoteMessageView: EKAccessoryNoteMessageView {
|
||||
|
||||
// MARK: Setup
|
||||
public required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
public init(with content: EKProperty.LabelContent, imageContent: EKProperty.ImageContent) {
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setup(with: content, imageContent: imageContent)
|
||||
}
|
||||
|
||||
private func setup(with content: EKProperty.LabelContent, imageContent: EKProperty.ImageContent) {
|
||||
let imageView = UIImageView()
|
||||
imageView.imageContent = imageContent
|
||||
accessoryView = imageView
|
||||
super.setup(with: content)
|
||||
}
|
||||
}
|
||||
52
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKNoteMessageView.swift
generated
Normal file
52
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKNoteMessageView.swift
generated
Normal file
@@ -0,0 +1,52 @@
|
||||
//
|
||||
// EKNoteMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public class EKNoteMessageView: UIView {
|
||||
|
||||
// MARK: Props
|
||||
private let label = UILabel()
|
||||
|
||||
private var horizontalConstrainsts: QLAxisConstraints!
|
||||
private var verticalConstrainsts: QLAxisConstraints!
|
||||
|
||||
public var horizontalOffset: CGFloat = 10 {
|
||||
didSet {
|
||||
horizontalConstrainsts.first.constant = horizontalOffset
|
||||
horizontalConstrainsts.second.constant = -horizontalOffset
|
||||
layoutIfNeeded()
|
||||
}
|
||||
}
|
||||
|
||||
public var verticalOffset: CGFloat = 5 {
|
||||
didSet {
|
||||
verticalConstrainsts.first.constant = verticalOffset
|
||||
verticalConstrainsts.second.constant = -verticalOffset
|
||||
layoutIfNeeded()
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Setup
|
||||
public init(with content: EKProperty.LabelContent) {
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setup(with: content)
|
||||
}
|
||||
|
||||
public required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setup(with content: EKProperty.LabelContent) {
|
||||
clipsToBounds = true
|
||||
addSubview(label)
|
||||
label.content = content
|
||||
horizontalConstrainsts = label.layoutToSuperview(axis: .horizontally, offset: horizontalOffset, priority: .must)
|
||||
verticalConstrainsts = label.layoutToSuperview(axis: .vertically, offset: verticalOffset, priority: .must)
|
||||
}
|
||||
}
|
||||
45
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKProcessingNoteMessageView.swift
generated
Normal file
45
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKProcessingNoteMessageView.swift
generated
Normal file
@@ -0,0 +1,45 @@
|
||||
//
|
||||
// EKProcessingNoteMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/20/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public class EKProcessingNoteMessageView: EKAccessoryNoteMessageView {
|
||||
|
||||
// MARK: Props
|
||||
private var activityIndicatorView: UIActivityIndicatorView!
|
||||
private var noteMessageView: EKNoteMessageView!
|
||||
|
||||
/** Activity indication can be turned off / on */
|
||||
public var isProcessing: Bool = true {
|
||||
didSet {
|
||||
if isProcessing {
|
||||
activityIndicatorView.startAnimating()
|
||||
} else {
|
||||
activityIndicatorView.stopAnimating()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Setup
|
||||
public required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
public init(with content: EKProperty.LabelContent, activityIndicator: UIActivityIndicatorView.Style) {
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setup(with: content, activityIndicator: activityIndicator)
|
||||
}
|
||||
|
||||
private func setup(with content: EKProperty.LabelContent, activityIndicator: UIActivityIndicatorView.Style, setProcessing: Bool = true) {
|
||||
activityIndicatorView = UIActivityIndicatorView()
|
||||
activityIndicatorView.style = activityIndicator
|
||||
isProcessing = setProcessing
|
||||
accessoryView = activityIndicatorView
|
||||
super.setup(with: content)
|
||||
}
|
||||
}
|
||||
46
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKXStatusBarMessageView.swift
generated
Normal file
46
Pods/SwiftEntryKit/Source/MessageViews/Notes/EKXStatusBarMessageView.swift
generated
Normal file
@@ -0,0 +1,46 @@
|
||||
//
|
||||
// EKXStatusBarMessageView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public class EKXStatusBarMessageView: UIView {
|
||||
|
||||
// MARK: Props
|
||||
private let leadingLabel = UILabel()
|
||||
private let trailingLabel = UILabel()
|
||||
|
||||
// MARK: Setup
|
||||
public init(leading: EKProperty.LabelContent, trailing: EKProperty.LabelContent) {
|
||||
super.init(frame: UIScreen.main.bounds)
|
||||
setup(leading: leading, trailing: trailing)
|
||||
}
|
||||
|
||||
public required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
private func setup(leading: EKProperty.LabelContent, trailing: EKProperty.LabelContent) {
|
||||
clipsToBounds = true
|
||||
|
||||
set(.height, of: UIApplication.shared.statusBarFrame.maxY)
|
||||
|
||||
addSubview(leadingLabel)
|
||||
leadingLabel.content = leading
|
||||
|
||||
leadingLabel.layoutToSuperview(axis: .vertically)
|
||||
leadingLabel.layoutToSuperview(.leading)
|
||||
leadingLabel.layoutToSuperview(.width, ratio: 0.26)
|
||||
|
||||
addSubview(trailingLabel)
|
||||
trailingLabel.content = trailing
|
||||
|
||||
trailingLabel.layoutToSuperview(axis: .vertically)
|
||||
trailingLabel.layoutToSuperview(.trailing)
|
||||
trailingLabel.layoutToSuperview(.width, ratio: 0.26)
|
||||
}
|
||||
}
|
||||
32
Pods/SwiftEntryKit/Source/Model/EKAlertMessage.swift
generated
Normal file
32
Pods/SwiftEntryKit/Source/Model/EKAlertMessage.swift
generated
Normal file
@@ -0,0 +1,32 @@
|
||||
//
|
||||
// EKAlertMessage.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 6/1/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
public struct EKAlertMessage {
|
||||
|
||||
public enum ImagePosition {
|
||||
case top
|
||||
case left
|
||||
}
|
||||
|
||||
/** The position of the image inside the alert */
|
||||
public let imagePosition: ImagePosition
|
||||
|
||||
/** Image, Title, Description */
|
||||
public let simpleMessage: EKSimpleMessage
|
||||
|
||||
/** Contents of button bar */
|
||||
public let buttonBarContent: EKProperty.ButtonBarContent
|
||||
|
||||
public init(simpleMessage: EKSimpleMessage,
|
||||
imagePosition: ImagePosition = .top,
|
||||
buttonBarContent: EKProperty.ButtonBarContent) {
|
||||
self.simpleMessage = simpleMessage
|
||||
self.imagePosition = imagePosition
|
||||
self.buttonBarContent = buttonBarContent
|
||||
}
|
||||
}
|
||||
110
Pods/SwiftEntryKit/Source/Model/EKColor.swift
generated
Normal file
110
Pods/SwiftEntryKit/Source/Model/EKColor.swift
generated
Normal file
@@ -0,0 +1,110 @@
|
||||
//
|
||||
// EKColor.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel on 21/07/2019.
|
||||
// Copyright © 2019 CocoaPods. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
/** A color representation attribute as per user interface style */
|
||||
public struct EKColor: Equatable {
|
||||
|
||||
// MARK: - Properties
|
||||
|
||||
public private(set) var dark: UIColor
|
||||
public private(set) var light: UIColor
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
public init(light: UIColor, dark: UIColor) {
|
||||
self.light = light
|
||||
self.dark = dark
|
||||
}
|
||||
|
||||
public init(_ unified: UIColor) {
|
||||
self.light = unified
|
||||
self.dark = unified
|
||||
}
|
||||
|
||||
public init(rgb: Int) {
|
||||
dark = UIColor(rgb: rgb)
|
||||
light = UIColor(rgb: rgb)
|
||||
}
|
||||
|
||||
public init(red: Int, green: Int, blue: Int) {
|
||||
assert(red >= 0 && red <= 255, "Invalid red component")
|
||||
assert(green >= 0 && green <= 255, "Invalid green component")
|
||||
assert(blue >= 0 && blue <= 255, "Invalid blue component")
|
||||
let color = UIColor(red: CGFloat(red) / 255.0,
|
||||
green: CGFloat(green) / 255.0,
|
||||
blue: CGFloat(blue) / 255.0,
|
||||
alpha: 1.0)
|
||||
light = color
|
||||
dark = color
|
||||
}
|
||||
|
||||
/** Computes the proper UIColor */
|
||||
public func color(for traits: UITraitCollection,
|
||||
mode: EKAttributes.DisplayMode) -> UIColor {
|
||||
switch mode {
|
||||
case .inferred:
|
||||
if #available(iOS 13, *) {
|
||||
switch traits.userInterfaceStyle {
|
||||
case .light, .unspecified:
|
||||
return light
|
||||
case .dark:
|
||||
return dark
|
||||
@unknown default:
|
||||
return light
|
||||
}
|
||||
} else {
|
||||
return light
|
||||
}
|
||||
case .light:
|
||||
return light
|
||||
case .dark:
|
||||
return dark
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public extension EKColor {
|
||||
|
||||
/// Returns the inverse of `self` (light and dark swapped)
|
||||
var inverted: EKColor {
|
||||
return EKColor(light: dark, dark: light)
|
||||
}
|
||||
|
||||
/** Returns an `EKColor` with the specified alpha component */
|
||||
func with(alpha: CGFloat) -> EKColor {
|
||||
return EKColor(light: light.withAlphaComponent(alpha),
|
||||
dark: dark.withAlphaComponent(alpha))
|
||||
}
|
||||
|
||||
/** White color for all user interface styles */
|
||||
static var white: EKColor {
|
||||
return EKColor(.white)
|
||||
}
|
||||
|
||||
/** Black color for all user interface styles */
|
||||
static var black: EKColor {
|
||||
return EKColor(.black)
|
||||
}
|
||||
|
||||
/** Clear color for all user interface styles */
|
||||
static var clear: EKColor {
|
||||
return EKColor(.clear)
|
||||
}
|
||||
|
||||
/** Color that represents standard background. White for light mode, black for dark mode */
|
||||
static var standardBackground: EKColor {
|
||||
return EKColor(light: .white, dark: .black)
|
||||
}
|
||||
|
||||
/** Color that represents standard content. black for light mode, white for dark mode */
|
||||
static var standardContent: EKColor {
|
||||
return EKColor(light: .black, dark: .white)
|
||||
}
|
||||
}
|
||||
41
Pods/SwiftEntryKit/Source/Model/EKNotificationMessage.swift
generated
Normal file
41
Pods/SwiftEntryKit/Source/Model/EKNotificationMessage.swift
generated
Normal file
@@ -0,0 +1,41 @@
|
||||
//
|
||||
// EKNotificationMessage.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/20/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public struct EKNotificationMessage {
|
||||
|
||||
/** Insets of the content of the message */
|
||||
public struct Insets {
|
||||
|
||||
/** The insets of the content of the message, from the top, bottom, left, right */
|
||||
public var contentInsets = UIEdgeInsets(top: 16, left: 16, bottom: 16, right: 16)
|
||||
|
||||
/** The distance between the title and the description */
|
||||
public var titleToDescription: CGFloat = 5
|
||||
|
||||
public static var `default` = Insets()
|
||||
}
|
||||
|
||||
/** Image, Title, Description */
|
||||
public let simpleMessage: EKSimpleMessage
|
||||
|
||||
/** Optional auxiliary label descriptor (For instance, it be used to display time of message) */
|
||||
public let auxiliary: EKProperty.LabelContent?
|
||||
|
||||
/** Defines the vertical and horizontal margins */
|
||||
public let insets: Insets
|
||||
|
||||
public init(simpleMessage: EKSimpleMessage,
|
||||
auxiliary: EKProperty.LabelContent? = nil,
|
||||
insets: Insets = .default) {
|
||||
self.simpleMessage = simpleMessage
|
||||
self.auxiliary = auxiliary
|
||||
self.insets = insets
|
||||
}
|
||||
}
|
||||
60
Pods/SwiftEntryKit/Source/Model/EKPopUpMessage.swift
generated
Normal file
60
Pods/SwiftEntryKit/Source/Model/EKPopUpMessage.swift
generated
Normal file
@@ -0,0 +1,60 @@
|
||||
//
|
||||
// EKPopUpMessage.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public struct EKPopUpMessage {
|
||||
|
||||
/** Code block that is executed as the user taps the popup button */
|
||||
public typealias EKPopUpMessageAction = () -> ()
|
||||
|
||||
/** Popup theme image */
|
||||
public struct ThemeImage {
|
||||
|
||||
/** Position of the theme image */
|
||||
public enum Position {
|
||||
case topToTop(offset: CGFloat)
|
||||
case centerToTop(offset: CGFloat)
|
||||
}
|
||||
|
||||
/** The content of the image */
|
||||
public var image: EKProperty.ImageContent
|
||||
|
||||
/** The psotion of the image */
|
||||
public var position: Position
|
||||
|
||||
/** Initializer */
|
||||
public init(image: EKProperty.ImageContent,
|
||||
position: Position = .topToTop(offset: 40)) {
|
||||
self.image = image
|
||||
self.position = position
|
||||
}
|
||||
}
|
||||
|
||||
public var themeImage: ThemeImage?
|
||||
public var title: EKProperty.LabelContent
|
||||
public var description: EKProperty.LabelContent
|
||||
public var button: EKProperty.ButtonContent
|
||||
public var action: EKPopUpMessageAction
|
||||
|
||||
var containsImage: Bool {
|
||||
return themeImage != nil
|
||||
}
|
||||
|
||||
public init(themeImage: ThemeImage? = nil,
|
||||
title: EKProperty.LabelContent,
|
||||
description: EKProperty.LabelContent,
|
||||
button: EKProperty.ButtonContent,
|
||||
action: @escaping EKPopUpMessageAction) {
|
||||
self.themeImage = themeImage
|
||||
self.title = title
|
||||
self.description = description
|
||||
self.button = button
|
||||
self.action = action
|
||||
}
|
||||
}
|
||||
399
Pods/SwiftEntryKit/Source/Model/EKProperty.swift
generated
Normal file
399
Pods/SwiftEntryKit/Source/Model/EKProperty.swift
generated
Normal file
@@ -0,0 +1,399 @@
|
||||
//
|
||||
// EKProperty.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public struct EKProperty {
|
||||
|
||||
/** Button content descriptor */
|
||||
public struct ButtonContent {
|
||||
|
||||
public typealias Action = () -> ()
|
||||
|
||||
/** Button title label content descriptor */
|
||||
public var label: LabelContent
|
||||
|
||||
/** Button background color */
|
||||
public var backgroundColor: EKColor
|
||||
public var highlightedBackgroundColor: EKColor
|
||||
|
||||
/** Content edge inset */
|
||||
public var contentEdgeInset: CGFloat
|
||||
|
||||
/** The display mode of the button */
|
||||
public var displayMode: EKAttributes.DisplayMode
|
||||
|
||||
/** Accessibility identifier that identifies the button */
|
||||
public var accessibilityIdentifier: String?
|
||||
|
||||
/** Action */
|
||||
public var action: Action?
|
||||
|
||||
public init(label: LabelContent,
|
||||
backgroundColor: EKColor,
|
||||
highlightedBackgroundColor: EKColor,
|
||||
contentEdgeInset: CGFloat = 5,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
accessibilityIdentifier: String? = nil,
|
||||
action: @escaping Action = {}) {
|
||||
self.label = label
|
||||
self.backgroundColor = backgroundColor
|
||||
self.highlightedBackgroundColor = highlightedBackgroundColor
|
||||
self.contentEdgeInset = contentEdgeInset
|
||||
self.displayMode = displayMode
|
||||
self.accessibilityIdentifier = accessibilityIdentifier
|
||||
self.action = action
|
||||
}
|
||||
|
||||
public func backgroundColor(for traitCollection: UITraitCollection) -> UIColor {
|
||||
return backgroundColor.color(for: traitCollection, mode: displayMode)
|
||||
}
|
||||
|
||||
public func highlightedBackgroundColor(for traitCollection: UITraitCollection) -> UIColor {
|
||||
return highlightedBackgroundColor.color(for: traitCollection, mode: displayMode)
|
||||
}
|
||||
|
||||
public func highlighedLabelColor(for traitCollection: UITraitCollection) -> UIColor {
|
||||
return label.style.color.with(alpha: 0.8).color(
|
||||
for: traitCollection,
|
||||
mode: label.style.displayMode
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
/** Label content descriptor */
|
||||
public struct LabelContent {
|
||||
|
||||
/** The text */
|
||||
public var text: String
|
||||
|
||||
/** The label's style */
|
||||
public var style: LabelStyle
|
||||
|
||||
/** The label's accessibility ideentifier */
|
||||
public var accessibilityIdentifier: String?
|
||||
|
||||
public init(text: String,
|
||||
style: LabelStyle,
|
||||
accessibilityIdentifier: String? = nil) {
|
||||
self.text = text
|
||||
self.style = style
|
||||
self.accessibilityIdentifier = accessibilityIdentifier
|
||||
}
|
||||
}
|
||||
|
||||
/** Label style descriptor */
|
||||
public struct LabelStyle {
|
||||
|
||||
/** Font of the text */
|
||||
public var font: UIFont
|
||||
|
||||
/** Color of the text */
|
||||
public var color: EKColor
|
||||
|
||||
/** Text Alignment */
|
||||
public var alignment: NSTextAlignment
|
||||
|
||||
/** Number of lines */
|
||||
public var numberOfLines: Int
|
||||
|
||||
/** Display mode for the label */
|
||||
public var displayMode: EKAttributes.DisplayMode
|
||||
|
||||
public init(font: UIFont,
|
||||
color: EKColor,
|
||||
alignment: NSTextAlignment = .left,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
numberOfLines: Int = 0) {
|
||||
self.font = font
|
||||
self.color = color
|
||||
self.alignment = alignment
|
||||
self.displayMode = displayMode
|
||||
self.numberOfLines = numberOfLines
|
||||
}
|
||||
|
||||
public func color(for traitCollection: UITraitCollection) -> UIColor {
|
||||
return color.color(for: traitCollection, mode: displayMode)
|
||||
}
|
||||
}
|
||||
|
||||
/** Image View style descriptor */
|
||||
public struct ImageContent {
|
||||
|
||||
/** Repeated-reversed animation throughout the presentation of an image */
|
||||
public enum TransformAnimation {
|
||||
case animate(duration: TimeInterval, options: UIView.AnimationOptions, transform: CGAffineTransform)
|
||||
case none
|
||||
}
|
||||
|
||||
/** Tint color for the image/s */
|
||||
public var tint: EKColor?
|
||||
|
||||
/** The images */
|
||||
public var images: [UIImage]
|
||||
|
||||
/** Image sequence duration, if any */
|
||||
public var imageSequenceAnimationDuration: TimeInterval
|
||||
|
||||
/** Image View size - can be forced.
|
||||
If nil, then the image view hugs content and resists compression */
|
||||
public var size: CGSize?
|
||||
|
||||
/** Content mode */
|
||||
public var contentMode: UIView.ContentMode
|
||||
|
||||
/** Should the image be rounded */
|
||||
public var makesRound: Bool
|
||||
|
||||
/** Repeated-Reversed animation */
|
||||
public var animation: TransformAnimation
|
||||
|
||||
/** The display mode of the image */
|
||||
public var displayMode: EKAttributes.DisplayMode
|
||||
|
||||
/** Image accessibility identifier */
|
||||
public var accessibilityIdentifier: String?
|
||||
|
||||
public init(imageName: String,
|
||||
animation: TransformAnimation = .none,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
size: CGSize? = nil,
|
||||
contentMode: UIView.ContentMode = .scaleToFill,
|
||||
tint: EKColor? = nil,
|
||||
makesRound: Bool = false,
|
||||
accessibilityIdentifier: String? = nil) {
|
||||
let image = UIImage(named: imageName)!
|
||||
self.init(image: image,
|
||||
displayMode: displayMode,
|
||||
size: size,
|
||||
tint: tint,
|
||||
contentMode: contentMode,
|
||||
makesRound: makesRound,
|
||||
accessibilityIdentifier: accessibilityIdentifier)
|
||||
}
|
||||
|
||||
public init(image: UIImage,
|
||||
animation: TransformAnimation = .none,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
size: CGSize? = nil,
|
||||
tint: EKColor? = nil,
|
||||
contentMode: UIView.ContentMode = .scaleToFill,
|
||||
makesRound: Bool = false,
|
||||
accessibilityIdentifier: String? = nil) {
|
||||
self.images = [image]
|
||||
self.size = size
|
||||
self.tint = tint
|
||||
self.displayMode = displayMode
|
||||
self.contentMode = contentMode
|
||||
self.makesRound = makesRound
|
||||
self.animation = animation
|
||||
self.imageSequenceAnimationDuration = 0
|
||||
self.accessibilityIdentifier = accessibilityIdentifier
|
||||
}
|
||||
|
||||
public init(images: [UIImage],
|
||||
imageSequenceAnimationDuration: TimeInterval = 1,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
animation: TransformAnimation = .none,
|
||||
size: CGSize? = nil,
|
||||
tint: EKColor? = nil,
|
||||
contentMode: UIView.ContentMode = .scaleToFill,
|
||||
makesRound: Bool = false,
|
||||
accessibilityIdentifier: String? = nil) {
|
||||
self.images = images
|
||||
self.size = size
|
||||
self.displayMode = displayMode
|
||||
self.tint = tint
|
||||
self.contentMode = contentMode
|
||||
self.makesRound = makesRound
|
||||
self.animation = animation
|
||||
self.imageSequenceAnimationDuration = imageSequenceAnimationDuration
|
||||
self.accessibilityIdentifier = accessibilityIdentifier
|
||||
}
|
||||
|
||||
public init(imagesNames: [String],
|
||||
imageSequenceAnimationDuration: TimeInterval = 1,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
animation: TransformAnimation = .none,
|
||||
size: CGSize? = nil,
|
||||
tint: EKColor? = nil,
|
||||
contentMode: UIView.ContentMode = .scaleToFill,
|
||||
makesRound: Bool = false,
|
||||
accessibilityIdentifier: String? = nil) {
|
||||
let images = imagesNames.map { return UIImage(named: $0)! }
|
||||
self.init(images: images,
|
||||
imageSequenceAnimationDuration: imageSequenceAnimationDuration,
|
||||
displayMode: displayMode,
|
||||
animation: animation,
|
||||
size: size,
|
||||
tint: tint,
|
||||
contentMode: contentMode,
|
||||
makesRound: makesRound,
|
||||
accessibilityIdentifier: accessibilityIdentifier)
|
||||
}
|
||||
|
||||
/** Quick thumbail property generator */
|
||||
public static func thumb(with image: UIImage,
|
||||
edgeSize: CGFloat) -> ImageContent {
|
||||
return ImageContent(images: [image],
|
||||
size: CGSize(width: edgeSize, height: edgeSize),
|
||||
contentMode: .scaleAspectFill,
|
||||
makesRound: true)
|
||||
}
|
||||
|
||||
/** Quick thumbail property generator */
|
||||
public static func thumb(with imageName: String,
|
||||
edgeSize: CGFloat) -> ImageContent {
|
||||
return ImageContent(imagesNames: [imageName],
|
||||
size: CGSize(width: edgeSize, height: edgeSize),
|
||||
contentMode: .scaleAspectFill,
|
||||
makesRound: true)
|
||||
}
|
||||
|
||||
public func tintColor(for traitCollection: UITraitCollection) -> UIColor? {
|
||||
return tint?.color(for: traitCollection, mode: displayMode)
|
||||
}
|
||||
}
|
||||
|
||||
/** Text field content **/
|
||||
public struct TextFieldContent {
|
||||
|
||||
// NOTE: Intentionally a reference type
|
||||
class ContentWrapper {
|
||||
var text = ""
|
||||
}
|
||||
|
||||
public weak var delegate: UITextFieldDelegate?
|
||||
public var keyboardType: UIKeyboardType
|
||||
public var isSecure: Bool
|
||||
public var leadingImage: UIImage!
|
||||
public var placeholder: LabelContent
|
||||
public var textStyle: LabelStyle
|
||||
public var tintColor: EKColor!
|
||||
public var displayMode: EKAttributes.DisplayMode
|
||||
public var bottomBorderColor: EKColor
|
||||
public var accessibilityIdentifier: String?
|
||||
let contentWrapper = ContentWrapper()
|
||||
public var textContent: String {
|
||||
set {
|
||||
contentWrapper.text = newValue
|
||||
}
|
||||
get {
|
||||
return contentWrapper.text
|
||||
}
|
||||
}
|
||||
|
||||
public init(delegate: UITextFieldDelegate? = nil,
|
||||
keyboardType: UIKeyboardType = .default,
|
||||
placeholder: LabelContent,
|
||||
tintColor: EKColor? = nil,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
textStyle: LabelStyle,
|
||||
isSecure: Bool = false,
|
||||
leadingImage: UIImage? = nil,
|
||||
bottomBorderColor: EKColor = .clear,
|
||||
accessibilityIdentifier: String? = nil) {
|
||||
self.delegate = delegate
|
||||
self.keyboardType = keyboardType
|
||||
self.placeholder = placeholder
|
||||
self.textStyle = textStyle
|
||||
self.tintColor = tintColor
|
||||
self.displayMode = displayMode
|
||||
self.isSecure = isSecure
|
||||
self.leadingImage = leadingImage
|
||||
self.bottomBorderColor = bottomBorderColor
|
||||
self.accessibilityIdentifier = accessibilityIdentifier
|
||||
}
|
||||
|
||||
public func tintColor(for traitCollection: UITraitCollection) -> UIColor? {
|
||||
return tintColor?.color(for: traitCollection, mode: displayMode)
|
||||
}
|
||||
|
||||
public func bottomBorderColor(for traitCollection: UITraitCollection) -> UIColor? {
|
||||
return bottomBorderColor.color(for: traitCollection, mode: displayMode)
|
||||
}
|
||||
}
|
||||
|
||||
/** Button bar content */
|
||||
public struct ButtonBarContent {
|
||||
|
||||
/** Button content array */
|
||||
public var content: [ButtonContent] = []
|
||||
|
||||
/** The color of the separator */
|
||||
public var separatorColor: EKColor
|
||||
|
||||
/** Upper threshold for the number of buttons (*ButtonContent*) for horizontal distribution. Must be a positive value */
|
||||
public var horizontalDistributionThreshold: Int
|
||||
|
||||
/** Determines whether the buttons expands animately */
|
||||
public var expandAnimatedly: Bool
|
||||
|
||||
/** The height of each button. All are equally distributed in their axis */
|
||||
public var buttonHeight: CGFloat
|
||||
|
||||
/** The display mode of the button bar */
|
||||
public var displayMode: EKAttributes.DisplayMode
|
||||
|
||||
public init(with buttonContents: ButtonContent...,
|
||||
separatorColor: EKColor,
|
||||
horizontalDistributionThreshold: Int = 2,
|
||||
buttonHeight: CGFloat = 50,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
expandAnimatedly: Bool) {
|
||||
self.init(with: buttonContents,
|
||||
separatorColor: separatorColor,
|
||||
horizontalDistributionThreshold: horizontalDistributionThreshold,
|
||||
buttonHeight: buttonHeight,
|
||||
displayMode: displayMode,
|
||||
expandAnimatedly: expandAnimatedly)
|
||||
}
|
||||
|
||||
public init(with buttonContents: [ButtonContent],
|
||||
separatorColor: EKColor,
|
||||
horizontalDistributionThreshold: Int = 2,
|
||||
buttonHeight: CGFloat = 50,
|
||||
displayMode: EKAttributes.DisplayMode = .inferred,
|
||||
expandAnimatedly: Bool) {
|
||||
guard horizontalDistributionThreshold > 0 else {
|
||||
fatalError("horizontalDistributionThreshold Must have a positive value!")
|
||||
}
|
||||
self.separatorColor = separatorColor
|
||||
self.horizontalDistributionThreshold = horizontalDistributionThreshold
|
||||
self.buttonHeight = buttonHeight
|
||||
self.displayMode = displayMode
|
||||
self.expandAnimatedly = expandAnimatedly
|
||||
content.append(contentsOf: buttonContents)
|
||||
}
|
||||
|
||||
public func separatorColor(for traitCollection: UITraitCollection) -> UIColor {
|
||||
return separatorColor.color(for: traitCollection, mode: displayMode)
|
||||
}
|
||||
}
|
||||
|
||||
/** Rating item content */
|
||||
public struct EKRatingItemContent {
|
||||
public var title: EKProperty.LabelContent
|
||||
public var description: EKProperty.LabelContent
|
||||
public var unselectedImage: EKProperty.ImageContent
|
||||
public var selectedImage: EKProperty.ImageContent
|
||||
public var size: CGSize
|
||||
|
||||
public init(title: EKProperty.LabelContent,
|
||||
description: EKProperty.LabelContent,
|
||||
unselectedImage: EKProperty.ImageContent,
|
||||
selectedImage: EKProperty.ImageContent,
|
||||
size: CGSize = CGSize(width: 50, height: 50)) {
|
||||
self.title = title
|
||||
self.description = description
|
||||
self.unselectedImage = unselectedImage
|
||||
self.selectedImage = selectedImage
|
||||
self.size = size
|
||||
}
|
||||
}
|
||||
}
|
||||
60
Pods/SwiftEntryKit/Source/Model/EKRatingMessage.swift
generated
Normal file
60
Pods/SwiftEntryKit/Source/Model/EKRatingMessage.swift
generated
Normal file
@@ -0,0 +1,60 @@
|
||||
//
|
||||
// EKRatingMessage.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 6/1/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public struct EKRatingMessage {
|
||||
|
||||
// NOTE: Intentionally a reference type
|
||||
class SelectedIndex {
|
||||
var selectedIndex: Int!
|
||||
}
|
||||
|
||||
/** Selection */
|
||||
public typealias Selection = (Int) -> Void
|
||||
|
||||
/** Initial title */
|
||||
public var initialTitle: EKProperty.LabelContent
|
||||
|
||||
/** Initial description */
|
||||
public var initialDescription: EKProperty.LabelContent
|
||||
|
||||
/** Rating items */
|
||||
public var ratingItems: [EKProperty.EKRatingItemContent]
|
||||
|
||||
/** Button bar content appears after selection */
|
||||
public var buttonBarContent: EKProperty.ButtonBarContent
|
||||
|
||||
/** Selection event - Each time the user interacts a rating star */
|
||||
public var selection: Selection!
|
||||
|
||||
let selectedIndexRef = SelectedIndex()
|
||||
|
||||
/** Selected index (if there is one) */
|
||||
public var selectedIndex: Int? {
|
||||
get {
|
||||
return selectedIndexRef.selectedIndex
|
||||
}
|
||||
set {
|
||||
selectedIndexRef.selectedIndex = newValue
|
||||
}
|
||||
}
|
||||
|
||||
/** Initializer */
|
||||
public init(initialTitle: EKProperty.LabelContent,
|
||||
initialDescription: EKProperty.LabelContent,
|
||||
ratingItems: [EKProperty.EKRatingItemContent],
|
||||
buttonBarContent: EKProperty.ButtonBarContent,
|
||||
selection: Selection? = nil) {
|
||||
self.initialTitle = initialTitle
|
||||
self.initialDescription = initialDescription
|
||||
self.ratingItems = ratingItems
|
||||
self.buttonBarContent = buttonBarContent
|
||||
self.selection = selection
|
||||
}
|
||||
}
|
||||
29
Pods/SwiftEntryKit/Source/Model/EKSimpleMessage.swift
generated
Normal file
29
Pods/SwiftEntryKit/Source/Model/EKSimpleMessage.swift
generated
Normal file
@@ -0,0 +1,29 @@
|
||||
//
|
||||
// EKSimpleMessage.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 6/1/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public struct EKSimpleMessage {
|
||||
|
||||
/** The image view descriptor */
|
||||
public let image: EKProperty.ImageContent?
|
||||
|
||||
/** The title label descriptor */
|
||||
public let title: EKProperty.LabelContent
|
||||
|
||||
/** The description label descriptor */
|
||||
public let description: EKProperty.LabelContent
|
||||
|
||||
public init(image: EKProperty.ImageContent? = nil,
|
||||
title: EKProperty.LabelContent,
|
||||
description: EKProperty.LabelContent) {
|
||||
self.image = image
|
||||
self.title = title
|
||||
self.description = description
|
||||
}
|
||||
}
|
||||
171
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Animation.swift
generated
Normal file
171
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Animation.swift
generated
Normal file
@@ -0,0 +1,171 @@
|
||||
//
|
||||
// EKAttributes+Animation.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
// A protocol that describes an animation
|
||||
protocol EKAnimation {
|
||||
var delay: TimeInterval { get set }
|
||||
var duration: TimeInterval { get set }
|
||||
var spring: EKAttributes.Animation.Spring? { get set }
|
||||
}
|
||||
|
||||
// A protocol that describes a range animation
|
||||
protocol EKRangeAnimation: EKAnimation {
|
||||
var start: CGFloat { get set }
|
||||
var end: CGFloat { get set }
|
||||
}
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Describes an animation that can be performed on the entry */
|
||||
struct Animation: Equatable {
|
||||
|
||||
/** Describes properties for a spring animation that can be performed on the entry */
|
||||
public struct Spring: Equatable {
|
||||
|
||||
/** The dampic of the spring animation */
|
||||
public var damping: CGFloat
|
||||
|
||||
/** The initial velocity of the spring animation */
|
||||
public var initialVelocity: CGFloat
|
||||
|
||||
/** Initializer */
|
||||
public init(damping: CGFloat, initialVelocity: CGFloat) {
|
||||
self.damping = damping
|
||||
self.initialVelocity = initialVelocity
|
||||
}
|
||||
}
|
||||
|
||||
/** Describes an animation with range */
|
||||
public struct RangeAnimation: EKRangeAnimation, Equatable {
|
||||
|
||||
/** The duration of the range animation */
|
||||
public var duration: TimeInterval
|
||||
|
||||
/** The delay of the range animation */
|
||||
public var delay: TimeInterval
|
||||
|
||||
/** The start value of the range animation (e.g. alpha, scale) */
|
||||
public var start: CGFloat
|
||||
|
||||
/** The end value of the range animation (e.g. alpha, scale) */
|
||||
public var end: CGFloat
|
||||
|
||||
/** The spring of the animation */
|
||||
public var spring: Spring?
|
||||
|
||||
/** Initializer */
|
||||
public init(from start: CGFloat, to end: CGFloat, duration: TimeInterval, delay: TimeInterval = 0, spring: Spring? = nil) {
|
||||
self.start = start
|
||||
self.end = end
|
||||
self.delay = delay
|
||||
self.duration = duration
|
||||
self.spring = spring
|
||||
}
|
||||
}
|
||||
|
||||
/** Describes translation animation */
|
||||
public struct Translate: EKAnimation, Equatable {
|
||||
|
||||
/** Describes the anchor position */
|
||||
public enum AnchorPosition: Equatable {
|
||||
|
||||
/** Top position - the entry shows from top or exits towards the top */
|
||||
case top
|
||||
|
||||
/** Bottom position - the entry shows from bottom or exits towards the bottom */
|
||||
case bottom
|
||||
|
||||
/** Automatic position - the entry shows and exits according to EKAttributes.Position value. If the position of the entry is top, bottom, the entry's translation anchor is top, bottom - respectively.*/
|
||||
case automatic
|
||||
}
|
||||
|
||||
/** Animation duration */
|
||||
public var duration: TimeInterval
|
||||
|
||||
/** Animation delay */
|
||||
public var delay: TimeInterval
|
||||
|
||||
/** To where OR from the entry is animated */
|
||||
public var anchorPosition: AnchorPosition
|
||||
|
||||
/** Optional translation spring */
|
||||
public var spring: Spring?
|
||||
|
||||
/** Initializer */
|
||||
public init(duration: TimeInterval, anchorPosition: AnchorPosition = .automatic, delay: TimeInterval = 0, spring: Spring? = nil) {
|
||||
self.anchorPosition = anchorPosition
|
||||
self.duration = duration
|
||||
self.delay = delay
|
||||
self.spring = spring
|
||||
}
|
||||
}
|
||||
|
||||
/** Translation animation prop */
|
||||
public var translate: Translate?
|
||||
|
||||
/** Scale animation prop */
|
||||
public var scale: RangeAnimation?
|
||||
|
||||
/** Fade animation prop */
|
||||
public var fade: RangeAnimation?
|
||||
|
||||
/** Does the animation contains translation */
|
||||
public var containsTranslation: Bool {
|
||||
return translate != nil
|
||||
}
|
||||
|
||||
/** Does the animation contains scale */
|
||||
public var containsScale: Bool {
|
||||
return scale != nil
|
||||
}
|
||||
|
||||
/** Does the animation contains fade */
|
||||
public var containsFade: Bool {
|
||||
return fade != nil
|
||||
}
|
||||
|
||||
/** Does the animation contains any animation whatsoever */
|
||||
public var containsAnimation: Bool {
|
||||
return containsTranslation || containsScale || containsFade
|
||||
}
|
||||
|
||||
/** Returns the maximum delay amongst all animations */
|
||||
public var maxDelay: TimeInterval {
|
||||
return max(translate?.delay ?? 0, max(scale?.delay ?? 0, fade?.delay ?? 0))
|
||||
}
|
||||
|
||||
/** Returns the maximum duration amongst all animations */
|
||||
public var maxDuration: TimeInterval {
|
||||
return max(translate?.duration ?? 0, max(scale?.duration ?? 0, fade?.duration ?? 0))
|
||||
}
|
||||
|
||||
/** Returns the maximum (duration+delay) amongst all animations */
|
||||
public var totalDuration: TimeInterval {
|
||||
return maxDelay + maxDuration
|
||||
}
|
||||
|
||||
/** Returns the maximum (duration+delay) amongst all animations */
|
||||
public static var translation: Animation {
|
||||
return Animation(translate: .init(duration: 0.3))
|
||||
}
|
||||
|
||||
/** No animation at all */
|
||||
public static var none: Animation {
|
||||
return Animation()
|
||||
}
|
||||
|
||||
/** Initializer */
|
||||
public init(translate: Translate? = nil, scale: RangeAnimation? = nil, fade: RangeAnimation? = nil) {
|
||||
self.translate = translate
|
||||
self.scale = scale
|
||||
self.fade = fade
|
||||
}
|
||||
}
|
||||
}
|
||||
138
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+BackgroundStyle.swift
generated
Normal file
138
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+BackgroundStyle.swift
generated
Normal file
@@ -0,0 +1,138 @@
|
||||
//
|
||||
// EKAttributes+BackgroundStyle.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** The background style property */
|
||||
enum BackgroundStyle: Equatable {
|
||||
|
||||
/** Blur style for light and dark modes */
|
||||
public struct BlurStyle: Equatable {
|
||||
|
||||
public static var extra: BlurStyle {
|
||||
return BlurStyle(light: .extraLight, dark: .dark)
|
||||
}
|
||||
|
||||
public static var standard: BlurStyle {
|
||||
return BlurStyle(light: .light, dark: .dark)
|
||||
}
|
||||
|
||||
@available(iOS 10.0, *)
|
||||
public static var prominent: BlurStyle {
|
||||
return BlurStyle(light: .prominent, dark: .prominent)
|
||||
}
|
||||
|
||||
public static var dark: BlurStyle {
|
||||
return BlurStyle(light: .dark, dark: .dark)
|
||||
}
|
||||
|
||||
let light: UIBlurEffect.Style
|
||||
let dark: UIBlurEffect.Style
|
||||
|
||||
public init(style: UIBlurEffect.Style) {
|
||||
self.light = style
|
||||
self.dark = style
|
||||
}
|
||||
|
||||
public init(light: UIBlurEffect.Style, dark: UIBlurEffect.Style) {
|
||||
self.light = light
|
||||
self.dark = dark
|
||||
}
|
||||
|
||||
/** Computes a proper `UIBlurEffect.Style` instance */
|
||||
public func blurStyle(for traits: UITraitCollection,
|
||||
mode: EKAttributes.DisplayMode) -> UIBlurEffect.Style {
|
||||
switch mode {
|
||||
case .inferred:
|
||||
if #available(iOS 13, *) {
|
||||
switch traits.userInterfaceStyle {
|
||||
case .light, .unspecified:
|
||||
return light
|
||||
case .dark:
|
||||
return dark
|
||||
@unknown default:
|
||||
return light
|
||||
}
|
||||
} else {
|
||||
return light
|
||||
}
|
||||
case .light:
|
||||
return light
|
||||
case .dark:
|
||||
return dark
|
||||
}
|
||||
}
|
||||
|
||||
public func blurEffect(for traits: UITraitCollection,
|
||||
mode: EKAttributes.DisplayMode) -> UIBlurEffect {
|
||||
return UIBlurEffect(style: blurStyle(for: traits, mode: mode))
|
||||
}
|
||||
}
|
||||
|
||||
/** Gradient background style */
|
||||
public struct Gradient {
|
||||
public var colors: [EKColor]
|
||||
public var startPoint: CGPoint
|
||||
public var endPoint: CGPoint
|
||||
|
||||
public init(colors: [EKColor],
|
||||
startPoint: CGPoint,
|
||||
endPoint: CGPoint) {
|
||||
self.colors = colors
|
||||
self.startPoint = startPoint
|
||||
self.endPoint = endPoint
|
||||
}
|
||||
}
|
||||
|
||||
/** Visual Effect (Blurred) background style */
|
||||
case visualEffect(style: BlurStyle)
|
||||
|
||||
/** Color background style */
|
||||
case color(color: EKColor)
|
||||
|
||||
/** Gradient background style */
|
||||
case gradient(gradient: Gradient)
|
||||
|
||||
/** Image background style */
|
||||
case image(image: UIImage)
|
||||
|
||||
/** Clear background style */
|
||||
case clear
|
||||
|
||||
/** == operator overload */
|
||||
public static func == (lhs: EKAttributes.BackgroundStyle,
|
||||
rhs: EKAttributes.BackgroundStyle) -> Bool {
|
||||
switch (lhs, rhs) {
|
||||
case (visualEffect(style: let leftStyle),
|
||||
visualEffect(style: let rightStyle)):
|
||||
return leftStyle == rightStyle
|
||||
case (color(color: let leftColor),
|
||||
color(color: let rightColor)):
|
||||
return leftColor == rightColor
|
||||
case (image(image: let leftImage),
|
||||
image(image: let rightImage)):
|
||||
return leftImage == rightImage
|
||||
case (gradient(gradient: let leftGradient),
|
||||
gradient(gradient: let rightGradient)):
|
||||
for (leftColor, rightColor) in zip(leftGradient.colors, rightGradient.colors) {
|
||||
guard leftColor == rightColor else {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return leftGradient.startPoint == rightGradient.startPoint &&
|
||||
leftGradient.endPoint == rightGradient.endPoint
|
||||
case (clear, clear):
|
||||
return true
|
||||
default:
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
25
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+DisplayMode.swift
generated
Normal file
25
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+DisplayMode.swift
generated
Normal file
@@ -0,0 +1,25 @@
|
||||
//
|
||||
// EKAttributes+DisplayMode.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel on 26/07/2019.
|
||||
// Copyright © 2019 CocoaPods. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Display mode for the entry */
|
||||
enum DisplayMode {
|
||||
|
||||
/** The display mode is inferred from the current user interface style */
|
||||
case inferred
|
||||
|
||||
/** The display mode is light */
|
||||
case light
|
||||
|
||||
/** The display mode is dark */
|
||||
case dark
|
||||
}
|
||||
}
|
||||
12
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Duration.swift
generated
Normal file
12
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Duration.swift
generated
Normal file
@@ -0,0 +1,12 @@
|
||||
//
|
||||
// EKAttributes+Duration.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/4/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public extension EKAttributes {
|
||||
typealias DisplayDuration = TimeInterval
|
||||
}
|
||||
79
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+FrameStyle.swift
generated
Normal file
79
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+FrameStyle.swift
generated
Normal file
@@ -0,0 +1,79 @@
|
||||
//
|
||||
// EKAttributes+FrameStyle.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/28/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import CoreGraphics
|
||||
import UIKit
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Corner radius of the entry - Specifies the corners */
|
||||
enum RoundCorners {
|
||||
|
||||
/** *None* of the corners will be round */
|
||||
case none
|
||||
|
||||
/** *All* of the corners will be round */
|
||||
case all(radius: CGFloat)
|
||||
|
||||
/** Only the *top* left and right corners will be round */
|
||||
case top(radius: CGFloat)
|
||||
|
||||
/** Only the *bottom* left and right corners will be round */
|
||||
case bottom(radius: CGFloat)
|
||||
|
||||
var hasRoundCorners: Bool {
|
||||
switch self {
|
||||
case .none:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
var cornerValues: (value: UIRectCorner, radius: CGFloat)? {
|
||||
switch self {
|
||||
case .all(radius: let radius):
|
||||
return (value: .allCorners, radius: radius)
|
||||
case .top(radius: let radius):
|
||||
return (value: .top, radius: radius)
|
||||
case .bottom(radius: let radius):
|
||||
return (value: .bottom, radius: radius)
|
||||
case .none:
|
||||
return nil
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/** The border around the entry */
|
||||
enum Border {
|
||||
|
||||
/** No border */
|
||||
case none
|
||||
|
||||
/** Border wirh color and width */
|
||||
case value(color: UIColor, width: CGFloat)
|
||||
|
||||
var hasBorder: Bool {
|
||||
switch self {
|
||||
case .none:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
var borderValues: (color: UIColor, width: CGFloat)? {
|
||||
switch self {
|
||||
case .value(color: let color, width: let width):
|
||||
return(color: color, width: width)
|
||||
case .none:
|
||||
return nil
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
38
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+HapticFeedback.swift
generated
Normal file
38
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+HapticFeedback.swift
generated
Normal file
@@ -0,0 +1,38 @@
|
||||
//
|
||||
// EKAttributes+HapticFeedback.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/1/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Notification haptic feedback type. Adds an additional sensuous layer. Read more at UINotificationFeedbackType. Available from iOS 10, but you are not required to check the iOS version before using it. It's automatically handled by the kit.
|
||||
*/
|
||||
enum NotificationHapticFeedback {
|
||||
case success
|
||||
case warning
|
||||
case error
|
||||
case none
|
||||
|
||||
@available(iOS 10.0, *)
|
||||
var value: UINotificationFeedbackGenerator.FeedbackType? {
|
||||
switch self {
|
||||
case .success:
|
||||
return .success
|
||||
case .warning:
|
||||
return .warning
|
||||
case .error:
|
||||
return .error
|
||||
case .none:
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
var isValid: Bool {
|
||||
return self != .none
|
||||
}
|
||||
}
|
||||
}
|
||||
41
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+LifecycleActions.swift
generated
Normal file
41
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+LifecycleActions.swift
generated
Normal file
@@ -0,0 +1,41 @@
|
||||
//
|
||||
// EKAttributes+LifecycleActions.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 6/16/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Contains optionally injected events that take place during the entry lifecycle */
|
||||
struct LifecycleEvents {
|
||||
|
||||
public typealias Event = () -> Void
|
||||
|
||||
/** Executed before the entry appears - before the animation starts.
|
||||
Might not get called in case another entry with a higher display priority is displayed.
|
||||
*/
|
||||
public var willAppear: Event?
|
||||
|
||||
/** Executed after the animation ends.
|
||||
Might not get called in case another entry with a higher display priority is displayed.
|
||||
*/
|
||||
public var didAppear: Event?
|
||||
|
||||
/** Executed before the entry disappears (Before the animation starts) */
|
||||
public var willDisappear: Event?
|
||||
|
||||
/** Executed after the entry disappears (After the animation ends) */
|
||||
public var didDisappear: Event?
|
||||
|
||||
public init(willAppear: Event? = nil, didAppear: Event? = nil, willDisappear: Event? = nil, didDisappear: Event? = nil) {
|
||||
self.willAppear = willAppear
|
||||
self.didAppear = didAppear
|
||||
self.willDisappear = willDisappear
|
||||
self.didDisappear = didDisappear
|
||||
}
|
||||
}
|
||||
}
|
||||
51
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+PopBehavior.swift
generated
Normal file
51
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+PopBehavior.swift
generated
Normal file
@@ -0,0 +1,51 @@
|
||||
//
|
||||
// EKAttributes+PopBehavior.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/26/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Describes the entry behavior when a new entry shows (with equal or higher display-priority) */
|
||||
enum PopBehavior {
|
||||
|
||||
/** The entry disappears promptly (Does not animates out) when a new one shows */
|
||||
case overridden
|
||||
|
||||
/** Animate the entry out - The entry rolls out when a new one shows */
|
||||
case animated(animation: Animation)
|
||||
|
||||
public var isOverriden: Bool {
|
||||
switch self {
|
||||
case .overridden:
|
||||
return true
|
||||
case .animated:
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
var animation: Animation? {
|
||||
switch self {
|
||||
case .animated(animation: let animation):
|
||||
return animation
|
||||
case .overridden:
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
func validate() {
|
||||
#if DEBUG
|
||||
guard let animation = animation else { return }
|
||||
guard animation == .none else { return }
|
||||
print("""
|
||||
SwiftEntryKit warning: cannot associate value `EKAttributes.Animation()`
|
||||
with `EKAttributes.PopBehavior.animated`. This may result in undefined behavior.
|
||||
Please use `PopBehavior.overridden` instead.
|
||||
""")
|
||||
#endif
|
||||
}
|
||||
}
|
||||
}
|
||||
37
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Position.swift
generated
Normal file
37
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Position.swift
generated
Normal file
@@ -0,0 +1,37 @@
|
||||
//
|
||||
// EKAttributes+Position.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** The position of the entry. */
|
||||
enum Position {
|
||||
|
||||
/** The entry appears at the top of the screen. */
|
||||
case top
|
||||
|
||||
/** The entry appears at the bottom of the screen. */
|
||||
case bottom
|
||||
|
||||
/** The entry appears at the center of the screen. */
|
||||
case center
|
||||
|
||||
public var isTop: Bool {
|
||||
return self == .top
|
||||
}
|
||||
|
||||
public var isCenter: Bool {
|
||||
return self == .center
|
||||
}
|
||||
|
||||
public var isBottom: Bool {
|
||||
return self == .bottom
|
||||
}
|
||||
}
|
||||
}
|
||||
196
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+PositionConstraints.swift
generated
Normal file
196
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+PositionConstraints.swift
generated
Normal file
@@ -0,0 +1,196 @@
|
||||
//
|
||||
// EKAttributes+Frame.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Describes the frame of the entry. It's limitations, width and offset from the anchor (top / bottom of the screen) */
|
||||
struct PositionConstraints {
|
||||
|
||||
/** Describes safe area relation */
|
||||
public enum SafeArea {
|
||||
|
||||
/** Entry overrides safe area */
|
||||
case overridden
|
||||
|
||||
/** The entry shows outs. But can optionally be colored */
|
||||
case empty(fillSafeArea: Bool)
|
||||
|
||||
public var isOverridden: Bool {
|
||||
switch self {
|
||||
case .overridden:
|
||||
return true
|
||||
default:
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/** Describes an edge constraint of the entry */
|
||||
public enum Edge {
|
||||
|
||||
/** Ratio constraint to screen edge */
|
||||
case ratio(value: CGFloat)
|
||||
|
||||
/** Offset from each edge of the screen */
|
||||
case offset(value: CGFloat)
|
||||
|
||||
/** Constant edge length */
|
||||
case constant(value: CGFloat)
|
||||
|
||||
/** Unspecified edge length */
|
||||
case intrinsic
|
||||
|
||||
/** Edge totally filled */
|
||||
public static var fill: Edge {
|
||||
return .offset(value: 0)
|
||||
}
|
||||
}
|
||||
|
||||
/** Describes the size of the entry */
|
||||
public struct Size {
|
||||
|
||||
/** Describes a width constraint */
|
||||
public var width: Edge
|
||||
|
||||
/** Describes a height constraint */
|
||||
public var height: Edge
|
||||
|
||||
/** Initializer */
|
||||
public init(width: Edge, height: Edge) {
|
||||
self.width = width
|
||||
self.height = height
|
||||
}
|
||||
|
||||
/** The content's size. Entry's content view must have tight constraints */
|
||||
public static var intrinsic: Size {
|
||||
return Size(width: .intrinsic, height: .intrinsic)
|
||||
}
|
||||
|
||||
/** The content's size. Entry's content view must have tight constraints */
|
||||
public static var sizeToWidth: Size {
|
||||
return Size(width: .offset(value: 0), height: .intrinsic)
|
||||
}
|
||||
|
||||
/** Screen size, without horizontal or vertical offset */
|
||||
public static var screen: Size {
|
||||
return Size(width: .fill, height: .fill)
|
||||
}
|
||||
}
|
||||
|
||||
/** The relation to the keyboard's top and the screen's top while it is opened */
|
||||
public enum KeyboardRelation {
|
||||
|
||||
/** Describes the offset when the keyboard is opened */
|
||||
public struct Offset {
|
||||
|
||||
/** Describes top keyboard offset to the entry's bottom */
|
||||
public var bottom: CGFloat
|
||||
|
||||
/** Describes top screen offset to the entry's top, useful to prevent the entry from exceeding the screen top bounds */
|
||||
public var screenEdgeResistance: CGFloat?
|
||||
|
||||
public init(bottom: CGFloat = 0, screenEdgeResistance: CGFloat? = nil) {
|
||||
self.bottom = bottom
|
||||
self.screenEdgeResistance = screenEdgeResistance
|
||||
}
|
||||
|
||||
/** None offset */
|
||||
public static var none: Offset {
|
||||
return Offset()
|
||||
}
|
||||
}
|
||||
|
||||
/** Bind the entry's bottom to the keyboard's top with an offset.
|
||||
Additionally, the top edge of the screen can have a resistance offset which the entry isn't able to cross.
|
||||
The resistance is mostly used when the device orientation changes and the entry's frame crosses the screen bounds.
|
||||
Current isn't supported with center entry position.*/
|
||||
case bind(offset: Offset)
|
||||
|
||||
/** Entry is unbound to the keyboard. It's location doesn't change. */
|
||||
case unbind
|
||||
|
||||
/** Returns true if the entry is bound to the keyboard */
|
||||
public var isBound: Bool {
|
||||
switch self {
|
||||
case .bind(offset: _):
|
||||
return true
|
||||
case .unbind:
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/** Rotation related position constraints */
|
||||
public struct Rotation {
|
||||
|
||||
/** Attributes of supported interface orientations */
|
||||
public enum SupportedInterfaceOrientation {
|
||||
|
||||
/** Uses standard supported interface orientation (target specification in general settings) */
|
||||
case standard
|
||||
|
||||
/** Supports all orinetations */
|
||||
case all
|
||||
}
|
||||
|
||||
/** Autorotate the entry along with the device orientation */
|
||||
public var isEnabled = true
|
||||
|
||||
/** The screen autorotates with accordance to this option */
|
||||
public var supportedInterfaceOrientations = SupportedInterfaceOrientation.standard
|
||||
|
||||
public init() {}
|
||||
}
|
||||
|
||||
/** The rotation attributes of the entry */
|
||||
public var rotation = Rotation()
|
||||
|
||||
/** The entry can be bound to keyboard in case of appearance */
|
||||
public var keyboardRelation = KeyboardRelation.unbind
|
||||
|
||||
/** The size of the entry */
|
||||
public var size: Size
|
||||
|
||||
/** The maximum size of the entry */
|
||||
public var maxSize: Size
|
||||
|
||||
/** The vertical offset from the top or bottom anchor */
|
||||
public var verticalOffset: CGFloat
|
||||
|
||||
/** Can be used to display the content outside the safe area margins such as on the notch of the iPhone X or the status bar itself. */
|
||||
public var safeArea = SafeArea.empty(fillSafeArea: false)
|
||||
|
||||
public var hasVerticalOffset: Bool {
|
||||
return verticalOffset > 0
|
||||
}
|
||||
|
||||
/** Returns a floating entry (float-like) */
|
||||
public static var float: PositionConstraints {
|
||||
return PositionConstraints(verticalOffset: 10, size: .init(width: .offset(value: 20), height: .intrinsic))
|
||||
}
|
||||
|
||||
/** A full width entry (toast-like) */
|
||||
public static var fullWidth: PositionConstraints {
|
||||
return PositionConstraints(verticalOffset: 0, size: .sizeToWidth)
|
||||
}
|
||||
|
||||
/** A full screen entry - fills the entire screen, modal-like */
|
||||
public static var fullScreen: PositionConstraints {
|
||||
return PositionConstraints(verticalOffset: 0, size: .screen)
|
||||
}
|
||||
|
||||
/** Initialize with default parameters */
|
||||
public init(verticalOffset: CGFloat = 0, size: Size = .sizeToWidth, maxSize: Size = .intrinsic) {
|
||||
self.verticalOffset = verticalOffset
|
||||
self.size = size
|
||||
self.maxSize = maxSize
|
||||
}
|
||||
}
|
||||
}
|
||||
142
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Precedence.swift
generated
Normal file
142
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Precedence.swift
generated
Normal file
@@ -0,0 +1,142 @@
|
||||
//
|
||||
// EKAttributes+Precedence.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/29/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
fileprivate extension Int {
|
||||
var isValidDisplayPriority: Bool {
|
||||
return self >= EKAttributes.Precedence.Priority.minRawValue && self <= EKAttributes.Precedence.Priority.maxRawValue
|
||||
}
|
||||
}
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/**
|
||||
Describes the manner on which the entry is pushed and displayed.
|
||||
See the various values of more explanation.
|
||||
*/
|
||||
enum Precedence {
|
||||
|
||||
/**
|
||||
The display priority of the entry - Determines whether is can be overriden by other entries.
|
||||
Must be in range [0...1000]
|
||||
*/
|
||||
public struct Priority: Hashable, Equatable, RawRepresentable, Comparable {
|
||||
public var rawValue: Int
|
||||
|
||||
public var hashValue: Int {
|
||||
return rawValue
|
||||
}
|
||||
|
||||
public init(_ rawValue: Int) {
|
||||
assert(rawValue.isValidDisplayPriority, "Display Priority must be in range [\(Priority.minRawValue)...\(Priority.maxRawValue)]")
|
||||
self.rawValue = rawValue
|
||||
}
|
||||
|
||||
public init(rawValue: Int) {
|
||||
assert(rawValue.isValidDisplayPriority, "Display Priority must be in range [\(Priority.minRawValue)...\(Priority.maxRawValue)]")
|
||||
self.rawValue = rawValue
|
||||
}
|
||||
|
||||
public static func == (lhs: Priority, rhs: Priority) -> Bool {
|
||||
return lhs.rawValue == rhs.rawValue
|
||||
}
|
||||
|
||||
public static func < (lhs: Priority, rhs: Priority) -> Bool {
|
||||
return lhs.rawValue < rhs.rawValue
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Describes the queueing heoristic of entries.
|
||||
*/
|
||||
public enum QueueingHeuristic {
|
||||
|
||||
/** Determines the heuristic which the entry-queue is based on */
|
||||
public static var value = QueueingHeuristic.priority
|
||||
|
||||
/** Chronological - FIFO */
|
||||
case chronological
|
||||
|
||||
/** Ordered by priority */
|
||||
case priority
|
||||
|
||||
/** Returns the caching heuristics mechanism that determines the priority in queue */
|
||||
var heuristic: EntryCachingHeuristic {
|
||||
switch self {
|
||||
case .chronological:
|
||||
return EKEntryChronologicalQueue()
|
||||
case .priority:
|
||||
return EKEntryPriorityQueue()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Describes an *overriding* behavior for a new entry.
|
||||
- In case no previous entry is currently presented, display the new entry.
|
||||
- In case there is an entry that is currently presented - override it using the new entry. Also optionally drop all previously enqueued entries.
|
||||
*/
|
||||
case override(priority: Priority, dropEnqueuedEntries: Bool)
|
||||
|
||||
/**
|
||||
Describes a FIFO behavior for an entry presentation.
|
||||
- In case no previous entry is currently presented, display the new entry.
|
||||
- In case there is an entry that is currently presented - enqueue the new entry, an present it just after the previous one is dismissed.
|
||||
*/
|
||||
case enqueue(priority: Priority)
|
||||
|
||||
var isEnqueue: Bool {
|
||||
switch self {
|
||||
case .enqueue:
|
||||
return true
|
||||
default:
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
/** Setter / Getter for the display priority */
|
||||
public var priority: Priority {
|
||||
set {
|
||||
switch self {
|
||||
case .enqueue(priority: _):
|
||||
self = .enqueue(priority: newValue)
|
||||
case .override(priority: _, dropEnqueuedEntries: let dropEnqueuedEntries):
|
||||
self = .override(priority: newValue, dropEnqueuedEntries: dropEnqueuedEntries)
|
||||
}
|
||||
}
|
||||
get {
|
||||
switch self {
|
||||
case .enqueue(priority: let priority):
|
||||
return priority
|
||||
case .override(priority: let priority, dropEnqueuedEntries: _):
|
||||
return priority
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/** High priority entries can be overriden by other equal or higher priority entries only.
|
||||
Entries are ignored as a higher priority entry is being displayed.
|
||||
High priority entry overrides any other entry including another equal priority one.
|
||||
You can you on of the values (.max, high, normal, low, min) and also set your own values. */
|
||||
public extension EKAttributes.Precedence.Priority {
|
||||
static let maxRawValue = 1000
|
||||
static let highRawValue = 750
|
||||
static let normalRawValue = 500
|
||||
static let lowRawValue = 250
|
||||
static let minRawValue = 0
|
||||
|
||||
/** Max - the highest possible priority of an entry. Can override only entries with *max* priority */
|
||||
static let max = EKAttributes.Precedence.Priority(rawValue: maxRawValue)
|
||||
static let high = EKAttributes.Precedence.Priority(rawValue: highRawValue)
|
||||
static let normal = EKAttributes.Precedence.Priority(rawValue: normalRawValue)
|
||||
static let low = EKAttributes.Precedence.Priority(rawValue: lowRawValue)
|
||||
static let min = EKAttributes.Precedence.Priority(rawValue: minRawValue)
|
||||
}
|
||||
|
||||
97
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Presets.swift
generated
Normal file
97
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Presets.swift
generated
Normal file
@@ -0,0 +1,97 @@
|
||||
//
|
||||
// EKAttributes+Presets.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/23/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Default attributes - Can be mutated according to the hosting application theme */
|
||||
static var `default` = EKAttributes()
|
||||
|
||||
/** Toast preset - The frame fills margins and safe area is filled with background view */
|
||||
static var toast: EKAttributes {
|
||||
var attributes = EKAttributes()
|
||||
attributes.positionConstraints = .fullWidth
|
||||
attributes.positionConstraints.safeArea = .empty(fillSafeArea: true)
|
||||
attributes.windowLevel = .statusBar
|
||||
attributes.scroll = .edgeCrossingDisabled(swipeable: true)
|
||||
attributes.popBehavior = .animated(animation: .translation)
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Float preset - The frame is margined and the safe area is left cleared */
|
||||
static var float: EKAttributes {
|
||||
var attributes = EKAttributes()
|
||||
attributes.positionConstraints = .float
|
||||
attributes.roundCorners = .all(radius: 10)
|
||||
attributes.positionConstraints.safeArea = .empty(fillSafeArea: false)
|
||||
attributes.windowLevel = .statusBar
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Preset for top float entry */
|
||||
static var topFloat: EKAttributes {
|
||||
var attributes = float
|
||||
attributes.position = .top
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Preset for a bottom float entry */
|
||||
static var bottomFloat: EKAttributes {
|
||||
var attributes = float
|
||||
attributes.position = .bottom
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Preset for a center float entry */
|
||||
static var centerFloat: EKAttributes {
|
||||
var attributes = float
|
||||
attributes.position = .center
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Preset for a bottom toast entry */
|
||||
static var bottomToast: EKAttributes {
|
||||
var attributes = toast
|
||||
attributes.position = .bottom
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Preset for a top toast entry */
|
||||
static var topToast: EKAttributes {
|
||||
var attributes = toast
|
||||
attributes.position = .top
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Preset for a top note entry */
|
||||
static var topNote: EKAttributes {
|
||||
var attributes = topToast
|
||||
attributes.scroll = .disabled
|
||||
attributes.windowLevel = .normal
|
||||
attributes.entryInteraction = .absorbTouches
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Preset for a bottom note entry */
|
||||
static var bottomNote: EKAttributes {
|
||||
var attributes = bottomToast
|
||||
attributes.scroll = .disabled
|
||||
attributes.windowLevel = .normal
|
||||
attributes.entryInteraction = .absorbTouches
|
||||
return attributes
|
||||
}
|
||||
|
||||
/** Preset for a status bar entry - appears on top of the status bar */
|
||||
static var statusBar: EKAttributes {
|
||||
var attributes = topToast
|
||||
attributes.windowLevel = .statusBar
|
||||
attributes.entryInteraction = .absorbTouches
|
||||
attributes.positionConstraints.safeArea = .overridden
|
||||
return attributes
|
||||
}
|
||||
}
|
||||
75
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Scroll.swift
generated
Normal file
75
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Scroll.swift
generated
Normal file
@@ -0,0 +1,75 @@
|
||||
//
|
||||
// EKAttributes+Scroll.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/30/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import CoreGraphics
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Describes the event of scroll user interaction */
|
||||
enum Scroll {
|
||||
|
||||
/** Describes the event when the user leaves the entry after rubber-banding it - How the entry behaves */
|
||||
public struct PullbackAnimation {
|
||||
public var duration: TimeInterval
|
||||
public var damping: CGFloat
|
||||
public var initialSpringVelocity: CGFloat
|
||||
|
||||
public init(duration: TimeInterval, damping: CGFloat, initialSpringVelocity: CGFloat) {
|
||||
self.duration = duration
|
||||
self.damping = damping
|
||||
self.initialSpringVelocity = initialSpringVelocity
|
||||
}
|
||||
|
||||
/** The entry is jolted when it's pulled back into the original position */
|
||||
public static var jolt: PullbackAnimation {
|
||||
return PullbackAnimation(duration: 0.5, damping: 0.3, initialSpringVelocity: 10)
|
||||
}
|
||||
|
||||
/** The view eases out when it's pulled back into the original position */
|
||||
public static var easeOut: PullbackAnimation {
|
||||
return PullbackAnimation(duration: 0.3, damping: 1, initialSpringVelocity: 10)
|
||||
}
|
||||
}
|
||||
|
||||
/** The scroll ability is totally disabled */
|
||||
case disabled
|
||||
|
||||
/** The scroll in the opposite direction to the edge is disabled */
|
||||
case edgeCrossingDisabled(swipeable: Bool)
|
||||
|
||||
/** The scroll abiliby is enabled */
|
||||
case enabled(swipeable: Bool, pullbackAnimation: PullbackAnimation)
|
||||
|
||||
var isEnabled: Bool {
|
||||
switch self {
|
||||
case .disabled:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
var isSwipeable: Bool {
|
||||
switch self {
|
||||
case .edgeCrossingDisabled(swipeable: let swipeable), .enabled(swipeable: let swipeable, pullbackAnimation: _):
|
||||
return swipeable
|
||||
default:
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
var isEdgeCrossingEnabled: Bool {
|
||||
switch self {
|
||||
case .edgeCrossingDisabled:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
43
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Shadow.swift
generated
Normal file
43
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Shadow.swift
generated
Normal file
@@ -0,0 +1,43 @@
|
||||
//
|
||||
// EKAttributes+Shadow.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** The shadow around the entry */
|
||||
enum Shadow {
|
||||
|
||||
/** No shadow */
|
||||
case none
|
||||
|
||||
/** Shadow with value */
|
||||
case active(with: Value)
|
||||
|
||||
/** The shadow properties */
|
||||
public struct Value {
|
||||
public let radius: CGFloat
|
||||
public let opacity: Float
|
||||
public let color: EKColor
|
||||
public let offset: CGSize
|
||||
|
||||
public init(color: EKColor = .black,
|
||||
opacity: Float,
|
||||
radius: CGFloat,
|
||||
offset: CGSize = .zero) {
|
||||
self.color = color
|
||||
self.radius = radius
|
||||
self.offset = offset
|
||||
self.opacity = opacity
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
91
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+StatusBar.swift
generated
Normal file
91
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+StatusBar.swift
generated
Normal file
@@ -0,0 +1,91 @@
|
||||
//
|
||||
// EKAttributes+StatusBar.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/25/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Status bar appearance */
|
||||
enum StatusBar {
|
||||
|
||||
/** The appearance of the status bar */
|
||||
public typealias Appearance = (visible: Bool, style: UIStatusBarStyle)
|
||||
|
||||
/** Ignored. Status bar is ignored by entries with this apperance value*/
|
||||
case ignored
|
||||
|
||||
/** Hidden. Doesn't apply to iPhone X */
|
||||
case hidden
|
||||
|
||||
/** Visible with explicit dark style */
|
||||
case dark
|
||||
|
||||
/** Visible with explicit light style */
|
||||
case light
|
||||
|
||||
/** Keep previous state of status bar.
|
||||
In case there is an already displayed entry, keep its status bar appearance.
|
||||
In case the app is already displaying a status bar, keep its appearance */
|
||||
case inferred
|
||||
|
||||
/** Returns the status bar appearance.
|
||||
Note: See *Appearance* */
|
||||
public var appearance: Appearance {
|
||||
switch self {
|
||||
case .dark:
|
||||
if #available(iOS 13, *) {
|
||||
return (true, .darkContent)
|
||||
} else {
|
||||
return (true, .default)
|
||||
}
|
||||
case .light:
|
||||
return (true, .lightContent)
|
||||
case .inferred:
|
||||
return StatusBar.currentAppearance
|
||||
case .hidden:
|
||||
return (false, StatusBar.currentStyle)
|
||||
case .ignored:
|
||||
fatalError("There is no defined appearance for an ignored status bar")
|
||||
}
|
||||
}
|
||||
|
||||
/** Returns the status bar according to a given appearance */
|
||||
public static func statusBar(by appearance: Appearance) -> StatusBar {
|
||||
guard appearance.visible else {
|
||||
return .hidden
|
||||
}
|
||||
switch appearance.style {
|
||||
case .lightContent:
|
||||
return .light
|
||||
default:
|
||||
return .dark
|
||||
}
|
||||
}
|
||||
|
||||
/** Returns the current appearance */
|
||||
public static var currentAppearance: Appearance {
|
||||
return (StatusBar.isCurrentVisible, StatusBar.currentStyle)
|
||||
}
|
||||
|
||||
/** Returns the current status bar */
|
||||
public static var currentStatusBar: StatusBar {
|
||||
return statusBar(by: currentAppearance)
|
||||
}
|
||||
|
||||
// Accessors
|
||||
// TODO: Use `statusBarManager` of the window scene on iOS 13
|
||||
private static var currentStyle: UIStatusBarStyle {
|
||||
return UIApplication.shared.statusBarStyle
|
||||
}
|
||||
|
||||
// TODO: Use `statusBarManager` of the window scene on iOS 13
|
||||
private static var isCurrentVisible: Bool {
|
||||
return !UIApplication.shared.isStatusBarHidden
|
||||
}
|
||||
}
|
||||
}
|
||||
84
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+UserInteraction.swift
generated
Normal file
84
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+UserInteraction.swift
generated
Normal file
@@ -0,0 +1,84 @@
|
||||
//
|
||||
// EKAttributes+UserInteraction.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Describes the user interaction events that are triggered as the user taps the entry / screen */
|
||||
struct UserInteraction {
|
||||
|
||||
/** Code that is executed when the user taps the entry / screen */
|
||||
public typealias Action = () -> ()
|
||||
|
||||
/** The default event that happens as the user interacts */
|
||||
public enum Default {
|
||||
|
||||
/** Absorbs touches. The entry / screen does nothing (Swallows the touch) */
|
||||
case absorbTouches
|
||||
|
||||
/** Touches delay the exit of the entry */
|
||||
case delayExit(by: TimeInterval)
|
||||
|
||||
/** Taps dismiss the entry immediately */
|
||||
case dismissEntry
|
||||
|
||||
/** Touches are forwarded to the lower window (In most cases it would be the application main window that will handle it */
|
||||
case forward
|
||||
}
|
||||
|
||||
var isResponsive: Bool {
|
||||
switch defaultAction {
|
||||
case .forward:
|
||||
return false
|
||||
default:
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
var isDelayExit: Bool {
|
||||
switch defaultAction {
|
||||
case .delayExit:
|
||||
return true
|
||||
default:
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
/** A default action that is executed when the entry or the screen are interacted by the user */
|
||||
public var defaultAction: Default
|
||||
|
||||
/** Additional actions that can be customized by the user */
|
||||
public var customTapActions: [Action]
|
||||
|
||||
public init(defaultAction: Default = .absorbTouches, customTapActions: [Action] = []) {
|
||||
self.defaultAction = defaultAction
|
||||
self.customTapActions = customTapActions
|
||||
}
|
||||
|
||||
/** Dismiss action */
|
||||
public static var dismiss: UserInteraction {
|
||||
return UserInteraction(defaultAction: .dismissEntry)
|
||||
}
|
||||
|
||||
/** Forward action */
|
||||
public static var forward: UserInteraction {
|
||||
return UserInteraction(defaultAction: .forward)
|
||||
}
|
||||
|
||||
/** Absorb touches action */
|
||||
public static var absorbTouches: UserInteraction {
|
||||
return UserInteraction(defaultAction: .absorbTouches)
|
||||
}
|
||||
|
||||
/** Delay exit action */
|
||||
public static func delayExit(by delay: TimeInterval) -> UserInteraction {
|
||||
return UserInteraction(defaultAction: .delayExit(by: delay))
|
||||
}
|
||||
}
|
||||
}
|
||||
30
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Validations.swift
generated
Normal file
30
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+Validations.swift
generated
Normal file
@@ -0,0 +1,30 @@
|
||||
//
|
||||
// EKAttributes+Validations.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/18/18.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
extension EKAttributes {
|
||||
|
||||
private static var minDisplayDuration: DisplayDuration {
|
||||
return 0
|
||||
}
|
||||
|
||||
var validateDisplayDuration: Bool {
|
||||
guard displayDuration >= EKAttributes.minDisplayDuration else {
|
||||
return false
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
var validateWindowLevel: Bool {
|
||||
return windowLevel.value >= .normal
|
||||
}
|
||||
|
||||
var isValid: Bool {
|
||||
return validateDisplayDuration && validateWindowLevel
|
||||
}
|
||||
}
|
||||
42
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+WindowLevel.swift
generated
Normal file
42
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes+WindowLevel.swift
generated
Normal file
@@ -0,0 +1,42 @@
|
||||
//
|
||||
// EKAttributes+WindowLevel.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/21/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
public extension EKAttributes {
|
||||
|
||||
/** Describes the window level in which the entry would be displayed */
|
||||
enum WindowLevel {
|
||||
|
||||
/** Above the alerts */
|
||||
case alerts
|
||||
|
||||
/** Above the status bar */
|
||||
case statusBar
|
||||
|
||||
/** Above the application window */
|
||||
case normal
|
||||
|
||||
/** Custom level */
|
||||
case custom(level: UIWindow.Level)
|
||||
|
||||
/** Returns the raw value - the window level itself */
|
||||
public var value: UIWindow.Level {
|
||||
switch self {
|
||||
case .alerts:
|
||||
return .alert
|
||||
case .statusBar:
|
||||
return .statusBar
|
||||
case .normal:
|
||||
return .normal
|
||||
case .custom(level: let level):
|
||||
return level
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
99
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes.swift
generated
Normal file
99
Pods/SwiftEntryKit/Source/Model/EntryAttributes/EKAttributes.swift
generated
Normal file
@@ -0,0 +1,99 @@
|
||||
//
|
||||
// EKAttributes.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/19/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
public struct EKAttributes {
|
||||
|
||||
// MARK: Identification
|
||||
|
||||
/**
|
||||
A settable **optional** name that matches the entry-attributes.
|
||||
- Nameless entries cannot be inquired using *SwiftEntryKit.isCurrentlyDisplaying(entryNamed: _) -> Bool*
|
||||
*/
|
||||
public var name: String?
|
||||
|
||||
// MARK: Display Attributes
|
||||
|
||||
/** Entry presentation window level */
|
||||
public var windowLevel = WindowLevel.statusBar
|
||||
|
||||
/** The position of the entry inside the screen */
|
||||
public var position = Position.top
|
||||
|
||||
/** The display manner of the entry. */
|
||||
public var precedence = Precedence.override(priority: .normal, dropEnqueuedEntries: false)
|
||||
|
||||
/** Describes how long the entry is displayed before it is dismissed */
|
||||
public var displayDuration: DisplayDuration = 2 // Use .infinity for infinite duration
|
||||
|
||||
/** The frame attributes of the entry */
|
||||
public var positionConstraints = PositionConstraints()
|
||||
|
||||
// MARK: User Interaction Attributes
|
||||
|
||||
/** Describes what happens when the user interacts the screen,
|
||||
forwards the touch to the application window by default */
|
||||
public var screenInteraction = UserInteraction.forward
|
||||
|
||||
/** Describes what happens when the user interacts the entry,
|
||||
dismisses the content by default */
|
||||
public var entryInteraction = UserInteraction.dismiss
|
||||
|
||||
/** Describes the scrolling behaviour of the entry.
|
||||
The entry can be swiped out and in with an ability to spring back with a jolt */
|
||||
public var scroll = Scroll.enabled(swipeable: true, pullbackAnimation: .jolt)
|
||||
|
||||
/** Generate haptic feedback once the entry is displayed */
|
||||
public var hapticFeedbackType = NotificationHapticFeedback.none
|
||||
|
||||
/** Describes the actions that take place when the entry appears or is being dismissed */
|
||||
public var lifecycleEvents = LifecycleEvents()
|
||||
|
||||
// MARK: Theme & Style Attributes
|
||||
|
||||
/** The display mode of the entry */
|
||||
public var displayMode = DisplayMode.inferred
|
||||
|
||||
/** Describes the entry's background appearance while it shows */
|
||||
public var entryBackground = BackgroundStyle.clear
|
||||
|
||||
/** Describes the background appearance while the entry shows */
|
||||
public var screenBackground = BackgroundStyle.clear
|
||||
|
||||
/** The shadow around the entry */
|
||||
public var shadow = Shadow.none
|
||||
|
||||
/** The corner attributes */
|
||||
public var roundCorners = RoundCorners.none
|
||||
|
||||
/** The border around the entry */
|
||||
public var border = Border.none
|
||||
|
||||
/** Preferred status bar style while the entry shows */
|
||||
public var statusBar = StatusBar.inferred
|
||||
|
||||
// MARK: Animation Attributes
|
||||
|
||||
/** Describes how the entry animates in */
|
||||
public var entranceAnimation = Animation.translation
|
||||
|
||||
/** Describes how the entry animates out */
|
||||
public var exitAnimation = Animation.translation
|
||||
|
||||
/** Describes the previous entry behaviour when a new entry with higher display-priority shows */
|
||||
public var popBehavior = PopBehavior.animated(animation: .translation) {
|
||||
didSet {
|
||||
popBehavior.validate()
|
||||
}
|
||||
}
|
||||
|
||||
/** Init with default attributes */
|
||||
public init() {}
|
||||
}
|
||||
170
Pods/SwiftEntryKit/Source/SwiftEntryKit.swift
generated
Normal file
170
Pods/SwiftEntryKit/Source/SwiftEntryKit.swift
generated
Normal file
@@ -0,0 +1,170 @@
|
||||
//
|
||||
// SwiftEntryKit.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/29/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
/**
|
||||
A stateless, threadsafe (unless described otherwise) entry point that contains the display and the dismissal logic of entries.
|
||||
*/
|
||||
public final class SwiftEntryKit {
|
||||
|
||||
/** Describes the a single or multiple entries for possible dismissal states */
|
||||
public enum EntryDismissalDescriptor {
|
||||
|
||||
/** Describes specific entry / entries with name */
|
||||
case specific(entryName: String)
|
||||
|
||||
/** Describes a group of entries with lower or equal display priority */
|
||||
case prioritizedLowerOrEqualTo(priority: EKAttributes.Precedence.Priority)
|
||||
|
||||
/** Describes all the entries that are currently in the queue and pending presentation */
|
||||
case enqueued
|
||||
|
||||
/** Describes all the entries */
|
||||
case all
|
||||
|
||||
/** Describes the currently displayed entry */
|
||||
case displayed
|
||||
}
|
||||
|
||||
/** The window to rollback to after dismissal */
|
||||
public enum RollbackWindow {
|
||||
|
||||
/** The main window */
|
||||
case main
|
||||
|
||||
/** A given custom window */
|
||||
case custom(window: UIWindow)
|
||||
}
|
||||
|
||||
/** Completion handler for the dismissal method */
|
||||
public typealias DismissCompletionHandler = () -> Void
|
||||
|
||||
/** Cannot be instantiated, customized, inherited. */
|
||||
private init() {}
|
||||
|
||||
/**
|
||||
Returns the window that displays the entry.
|
||||
**Warning**: the returned `UIWindow` instance is `nil` in case
|
||||
no entry is currently displayed.
|
||||
This can be used
|
||||
*/
|
||||
public class var window: UIWindow? {
|
||||
return EKWindowProvider.shared.entryWindow
|
||||
}
|
||||
|
||||
/**
|
||||
Returns true if **any** entry is currently displayed.
|
||||
- Not thread safe - should be called from the main queue only in order to receive a reliable result.
|
||||
- Convenience computed variable. Using it is the same as invoking **isCurrentlyDisplaying() -> Bool** (witohut the name of the entry).
|
||||
*/
|
||||
public class var isCurrentlyDisplaying: Bool {
|
||||
return isCurrentlyDisplaying()
|
||||
}
|
||||
|
||||
/**
|
||||
Returns true if an entry with a given name is currently displayed.
|
||||
- Not thread safe - should be called from the main queue only in order to receive a reliable result.
|
||||
- If invoked with *name* = *nil* or without the parameter value, it will return *true* if **any** entry is currently displayed.
|
||||
- Returns a *false* value for currently enqueued entries.
|
||||
- parameter name: The name of the entry. Its default value is *nil*.
|
||||
*/
|
||||
public class func isCurrentlyDisplaying(entryNamed name: String? = nil) -> Bool {
|
||||
return EKWindowProvider.shared.isCurrentlyDisplaying(entryNamed: name)
|
||||
}
|
||||
|
||||
/**
|
||||
Returns true if **any** entry is currently enqueued and waiting to be displayed.
|
||||
- Not thread safe - should be called from the main queue only in order to receive a reliable result.
|
||||
- Convenience computed variable. Using it is the same as invoking **~queueContains() -> Bool** (witohut the name of the entry)
|
||||
*/
|
||||
public class var isQueueEmpty: Bool {
|
||||
return !queueContains()
|
||||
}
|
||||
|
||||
/**
|
||||
Returns true if an entry with a given name is currently enqueued and waiting to be displayed.
|
||||
- Not thread safe - should be called from the main queue only in order to receive a reliable result.
|
||||
- If invoked with *name* = *nil* or without the parameter value, it will return *true* if **any** entry is currently displayed, meaning, the queue is not currently empty.
|
||||
- parameter name: The name of the entry. Its default value is *nil*.
|
||||
*/
|
||||
public class func queueContains(entryNamed name: String? = nil) -> Bool {
|
||||
return EKWindowProvider.shared.queueContains(entryNamed: name)
|
||||
}
|
||||
|
||||
/**
|
||||
Displays a given entry view using an attributes struct.
|
||||
- A thread-safe method - Can be invokes from any thread
|
||||
- A class method - Should be called on the class
|
||||
- parameter view: Custom view that is to be displayed
|
||||
- parameter attributes: Display properties
|
||||
- parameter presentInsideKeyWindow: Indicates whether the entry window should become the key window.
|
||||
- parameter rollbackWindow: After the entry has been dismissed, SwiftEntryKit rolls back to the given window. By default it is *.main* which is the app main window
|
||||
*/
|
||||
public class func display(entry view: UIView, using attributes: EKAttributes, presentInsideKeyWindow: Bool = false, rollbackWindow: RollbackWindow = .main) {
|
||||
DispatchQueue.main.async {
|
||||
EKWindowProvider.shared.display(view: view, using: attributes, presentInsideKeyWindow: presentInsideKeyWindow, rollbackWindow: rollbackWindow)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Displays a given entry view controller using an attributes struct.
|
||||
- A thread-safe method - Can be invokes from any thread
|
||||
- A class method - Should be called on the class
|
||||
- parameter view: Custom view that is to be displayed
|
||||
- parameter attributes: Display properties
|
||||
- parameter presentInsideKeyWindow: Indicates whether the entry window should become the key window.
|
||||
- parameter rollbackWindow: After the entry has been dismissed, SwiftEntryKit rolls back to the given window. By default it is *.main* - which is the app main window
|
||||
*/
|
||||
public class func display(entry viewController: UIViewController, using attributes: EKAttributes, presentInsideKeyWindow: Bool = false, rollbackWindow: RollbackWindow = .main) {
|
||||
DispatchQueue.main.async {
|
||||
EKWindowProvider.shared.display(viewController: viewController, using: attributes, presentInsideKeyWindow: presentInsideKeyWindow, rollbackWindow: rollbackWindow)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
ALPHA FEATURE: Transform the previous entry to the current one using the previous attributes struct.
|
||||
- A thread-safe method - Can be invoked from any thread.
|
||||
- A class method - Should be called on the class.
|
||||
- This feature hasn't been fully tested. Use with caution.
|
||||
- parameter view: Custom view that is to be displayed instead of the currently displayed entry
|
||||
*/
|
||||
public class func transform(to view: UIView) {
|
||||
DispatchQueue.main.async {
|
||||
EKWindowProvider.shared.transform(to: view)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Dismisses the currently presented entry and removes the presented window instance after the exit animation is concluded.
|
||||
- A thread-safe method - Can be invoked from any thread.
|
||||
- A class method - Should be called on the class.
|
||||
- parameter descriptor: A descriptor for the entries that are to be dismissed. The default value is *.displayed*.
|
||||
- parameter completion: A completion handler that is to be called right after the entry is dismissed (After the animation is concluded).
|
||||
*/
|
||||
public class func dismiss(_ descriptor: EntryDismissalDescriptor = .displayed, with completion: DismissCompletionHandler? = nil) {
|
||||
DispatchQueue.main.async {
|
||||
EKWindowProvider.shared.dismiss(descriptor, with: completion)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Layout the view hierarchy that is rooted in the window.
|
||||
- In case you use complex animations, you can call it to refresh the AutoLayout mechanism on the entire view hierarchy.
|
||||
- A thread-safe method - Can be invoked from any thread.
|
||||
- A class method - Should be called on the class.
|
||||
*/
|
||||
public class func layoutIfNeeded() {
|
||||
if Thread.isMainThread {
|
||||
EKWindowProvider.shared.layoutIfNeeded()
|
||||
} else {
|
||||
DispatchQueue.main.async {
|
||||
EKWindowProvider.shared.layoutIfNeeded()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
63
Pods/SwiftEntryKit/Source/Utils/GradientView.swift
generated
Normal file
63
Pods/SwiftEntryKit/Source/Utils/GradientView.swift
generated
Normal file
@@ -0,0 +1,63 @@
|
||||
//
|
||||
// GradientView.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/20/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
class GradientView: UIView {
|
||||
|
||||
struct Style {
|
||||
let gradient: EKAttributes.BackgroundStyle.Gradient
|
||||
let displayMode: EKAttributes.DisplayMode
|
||||
|
||||
init?(gradient: EKAttributes.BackgroundStyle.Gradient?,
|
||||
displayMode: EKAttributes.DisplayMode) {
|
||||
guard let gradient = gradient else {
|
||||
return nil
|
||||
}
|
||||
self.gradient = gradient
|
||||
self.displayMode = displayMode
|
||||
}
|
||||
}
|
||||
|
||||
private let gradientLayer = CAGradientLayer()
|
||||
|
||||
var style: Style? {
|
||||
didSet {
|
||||
setupColor()
|
||||
}
|
||||
}
|
||||
|
||||
init() {
|
||||
super.init(frame: .zero)
|
||||
layer.addSublayer(gradientLayer)
|
||||
}
|
||||
|
||||
required init?(coder aDecoder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
override func layoutSubviews() {
|
||||
super.layoutSubviews()
|
||||
gradientLayer.frame = bounds
|
||||
}
|
||||
|
||||
private func setupColor() {
|
||||
guard let style = style else {
|
||||
return
|
||||
}
|
||||
gradientLayer.colors = style.gradient.colors.map {
|
||||
$0.color(for: traitCollection, mode: style.displayMode).cgColor
|
||||
}
|
||||
gradientLayer.startPoint = style.gradient.startPoint
|
||||
gradientLayer.endPoint = style.gradient.endPoint
|
||||
}
|
||||
|
||||
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
||||
setupColor()
|
||||
}
|
||||
}
|
||||
20
Pods/SwiftEntryKit/Source/Utils/HapticFeedbackGenerator.swift
generated
Normal file
20
Pods/SwiftEntryKit/Source/Utils/HapticFeedbackGenerator.swift
generated
Normal file
@@ -0,0 +1,20 @@
|
||||
//
|
||||
// HapticFeedbackGenerator.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 4/20/18.
|
||||
// Copyright (c) 2018 huri000@gmail.com. All rights reserved.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
struct HapticFeedbackGenerator {
|
||||
@available(iOS 10.0, *)
|
||||
static func notification(type: EKAttributes.NotificationHapticFeedback) {
|
||||
guard let value = type.value else {
|
||||
return
|
||||
}
|
||||
let generator = UINotificationFeedbackGenerator()
|
||||
generator.notificationOccurred(value)
|
||||
}
|
||||
}
|
||||
22
Pods/SwiftEntryKit/Source/Utils/UIView+Responder.swift
generated
Normal file
22
Pods/SwiftEntryKit/Source/Utils/UIView+Responder.swift
generated
Normal file
@@ -0,0 +1,22 @@
|
||||
//
|
||||
// UIView+Responder.swift
|
||||
// SwiftEntryKit
|
||||
//
|
||||
// Created by Daniel Huri on 5/17/18.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
|
||||
extension UIView {
|
||||
var containsFirstResponder: Bool {
|
||||
var contains = false
|
||||
for subview in subviews.reversed() where !contains {
|
||||
if subview.isFirstResponder {
|
||||
contains = true
|
||||
} else {
|
||||
contains = subview.containsFirstResponder
|
||||
}
|
||||
}
|
||||
return contains
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user