mirror of
https://gitlab.opencode.de/bwi/bundesmessenger/clients/bundesmessenger-ios.git
synced 2026-04-26 11:30:50 +02:00
Create KeyboardAvoider that help to avoid keyboard overlap with scroll view content.
This commit is contained in:
@@ -0,0 +1,130 @@
|
||||
/*
|
||||
Copyright 2018 New Vector Ltd
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
*/
|
||||
|
||||
import Foundation
|
||||
|
||||
/// Avoid keyboard overlap with scroll view content
|
||||
final class KeyboardAvoider {
|
||||
|
||||
// MARK: - Constants
|
||||
|
||||
private enum KeyboardAnimation {
|
||||
static let defaultDuration: TimeInterval = 0.25
|
||||
static let defaultAnimationCurveRawValue: Int = UIViewAnimationCurve.easeInOut.rawValue
|
||||
}
|
||||
|
||||
// MARK: - Properties
|
||||
|
||||
weak var scrollViewContainerView: UIView?
|
||||
weak var scrollView: UIScrollView?
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
/// Designated initializer.
|
||||
///
|
||||
/// - Parameter scrollViewContainerView: The view that wrap the scroll view.
|
||||
/// - Parameter scrollView: The scroll view containing keyboard inputs and where content view overlap with keyboard should be avoided.
|
||||
init(scrollViewContainerView: UIView, scrollView: UIScrollView) {
|
||||
self.scrollViewContainerView = scrollViewContainerView
|
||||
self.scrollView = scrollView
|
||||
}
|
||||
|
||||
// MARK: - Public
|
||||
|
||||
/// Start keyboard avoiding
|
||||
func startAvoiding() {
|
||||
self.registerKeyboardNotifications()
|
||||
}
|
||||
|
||||
/// Stop keyboard avoiding
|
||||
func stopAvoiding() {
|
||||
self.unregisterKeyboardNotifications()
|
||||
}
|
||||
|
||||
// MARK: - Private
|
||||
|
||||
private func registerKeyboardNotifications() {
|
||||
let notificationCenter = NotificationCenter.default
|
||||
notificationCenter.addObserver(
|
||||
self,
|
||||
selector: #selector(keyboardWillShow(notification:)),
|
||||
name: .UIKeyboardWillShow,
|
||||
object: nil)
|
||||
notificationCenter.addObserver(
|
||||
self,
|
||||
selector: #selector(keyboardWillHide(notification:)),
|
||||
name: .UIKeyboardWillHide,
|
||||
object: nil)
|
||||
}
|
||||
|
||||
private func unregisterKeyboardNotifications() {
|
||||
NotificationCenter.default.removeObserver(self, name: .UIKeyboardWillShow, object: nil)
|
||||
NotificationCenter.default.removeObserver(self, name: .UIKeyboardWillHide, object: nil)
|
||||
}
|
||||
|
||||
@objc private func keyboardWillShow(notification: Notification) {
|
||||
guard let view = self.scrollViewContainerView, let scrollView = self.scrollView else {
|
||||
return
|
||||
}
|
||||
|
||||
guard let keyboardNotification = KeyboardNotification(notification: notification),
|
||||
let keyboardFrame = keyboardNotification.keyboardFrameEnd else {
|
||||
return
|
||||
}
|
||||
|
||||
let animationDuration = keyboardNotification.animationDuration ?? KeyboardAnimation.defaultDuration
|
||||
|
||||
let animationOptions = keyboardNotification.animationOptions(fallbackAnimationCurveValue: KeyboardAnimation.defaultAnimationCurveRawValue)
|
||||
|
||||
// Transform the keyboard's frame into our view's coordinate system
|
||||
let keyboardFrameInView = view.convert(keyboardFrame, from: nil)
|
||||
|
||||
// Find how much the keyboard overlaps the scroll view
|
||||
let scrollViewBottomInset = scrollView.frame.maxY - keyboardFrameInView.origin.y
|
||||
|
||||
UIView.animate(withDuration: animationDuration,
|
||||
delay: 0.0,
|
||||
options: animationOptions, animations: {
|
||||
|
||||
scrollView.contentInset.bottom = scrollViewBottomInset
|
||||
scrollView.scrollIndicatorInsets.bottom = scrollViewBottomInset
|
||||
}, completion: nil)
|
||||
}
|
||||
|
||||
@objc private func keyboardWillHide(notification: Notification) {
|
||||
guard let scrollView = self.scrollView else {
|
||||
return
|
||||
}
|
||||
|
||||
guard let keyboardNotification = KeyboardNotification(notification: notification) else {
|
||||
return
|
||||
}
|
||||
|
||||
let animationDuration = keyboardNotification.animationDuration ?? KeyboardAnimation.defaultDuration
|
||||
|
||||
let animationOptions = keyboardNotification.animationOptions(fallbackAnimationCurveValue: KeyboardAnimation.defaultAnimationCurveRawValue)
|
||||
|
||||
// Reset scroll view bottom inset to zero
|
||||
let scrollViewBottomInset: CGFloat = 0.0
|
||||
|
||||
UIView.animate(withDuration: animationDuration,
|
||||
delay: 0.0,
|
||||
options: animationOptions, animations: {
|
||||
scrollView.contentInset.bottom = scrollViewBottomInset
|
||||
scrollView.scrollIndicatorInsets.bottom = scrollViewBottomInset
|
||||
}, completion: nil)
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,65 @@
|
||||
/*
|
||||
Copyright 2018 New Vector Ltd
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
*/
|
||||
|
||||
import UIKit
|
||||
|
||||
/// Wrapper for the Notification userInfo values associated with a keyboard notification.
|
||||
public struct KeyboardNotification {
|
||||
|
||||
let userInfo: [AnyHashable: Any]
|
||||
|
||||
public init?(notification: Notification) {
|
||||
guard let userInfo = notification.userInfo else {
|
||||
return nil
|
||||
}
|
||||
self.userInfo = userInfo
|
||||
}
|
||||
|
||||
public var keyboardFrameBegin: CGRect? {
|
||||
guard let value = userInfo[UIKeyboardFrameBeginUserInfoKey] as? NSValue else {
|
||||
return nil
|
||||
}
|
||||
return value.cgRectValue
|
||||
}
|
||||
|
||||
public var keyboardFrameEnd: CGRect? {
|
||||
guard let value = userInfo[UIKeyboardFrameEndUserInfoKey] as? NSValue else {
|
||||
return nil
|
||||
}
|
||||
return value.cgRectValue
|
||||
}
|
||||
|
||||
public var animationDuration: TimeInterval? {
|
||||
guard let number = userInfo[UIKeyboardAnimationDurationUserInfoKey] as? NSNumber else {
|
||||
return nil
|
||||
}
|
||||
return number.doubleValue
|
||||
}
|
||||
|
||||
/// Keyboard UIViewAnimationCurve enum raw value
|
||||
public var animationCurveRawValue: Int? {
|
||||
guard let number = userInfo[UIKeyboardAnimationCurveUserInfoKey] as? NSNumber else {
|
||||
return nil
|
||||
}
|
||||
return number.intValue
|
||||
}
|
||||
|
||||
/// Convert UIViewAnimationCurve raw value to UIViewAnimationOptions
|
||||
public func animationOptions(fallbackAnimationCurveValue: Int = UIViewAnimationCurve.easeInOut.rawValue) -> UIViewAnimationOptions {
|
||||
let animationCurveRawValue = self.animationCurveRawValue ?? fallbackAnimationCurveValue
|
||||
return UIViewAnimationOptions(rawValue: UInt(animationCurveRawValue << 16))
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user