mirror of
https://github.com/MessageKit/MessageKit.git
synced 2026-02-06 19:03:19 +00:00
bff35fda61
* build: Swiftlint plugin * build: Swiftformat plugin * build: Swiftformat plugin * build: Swiftformat bash command * style: Swiftformat rules * style: Swiftformat applied to codebase * style: Ignore Tests for Swiftlint * Update bundler * Update changelog and migration guide * style: Ignore Example for Swiftlint * chore: Changelog * Update Xcode version for ci_pr_tests.yml * Update ci_pr_framework.yml * Update ci_pr_example.yml * chore: Changelog Co-authored-by: Jakub Kaspar <kaspikk@gmail.com>
433 lines
16 KiB
Swift
433 lines
16 KiB
Swift
// MIT License
|
|
//
|
|
// Copyright (c) 2017-2020 MessageKit
|
|
//
|
|
// 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.
|
|
|
|
import InputBarAccessoryView
|
|
import Kingfisher
|
|
import MessageKit
|
|
import UIKit
|
|
|
|
// MARK: - AutocompleteExampleViewController
|
|
|
|
final class AutocompleteExampleViewController: ChatViewController {
|
|
// MARK: Internal
|
|
|
|
lazy var joinChatButton: UIButton = {
|
|
let button = UIButton()
|
|
button.layer.cornerRadius = 16
|
|
button.backgroundColor = .primaryColor
|
|
button.setTitle("JOIN CHAT", for: .normal)
|
|
button.setTitleColor(.white, for: .normal)
|
|
button.setTitleColor(UIColor(white: 1, alpha: 0.3), for: .highlighted)
|
|
button.addTarget(self, action: #selector(joinChat), for: .touchUpInside)
|
|
return button
|
|
}()
|
|
|
|
/// The object that manages autocomplete, from InputBarAccessoryView
|
|
lazy var autocompleteManager: AutocompleteManager = { [unowned self] in
|
|
let manager = AutocompleteManager(for: self.messageInputBar.inputTextView)
|
|
manager.delegate = self
|
|
manager.dataSource = self
|
|
return manager
|
|
}()
|
|
|
|
var hashtagAutocompletes: [AutocompleteCompletion] = {
|
|
var array: [AutocompleteCompletion] = []
|
|
for _ in 1 ... 100 {
|
|
array.append(AutocompleteCompletion(text: Lorem.word(), context: nil))
|
|
}
|
|
return array
|
|
}()
|
|
|
|
// Completions loaded async that get appended to local cached completions
|
|
var asyncCompletions: [AutocompleteCompletion] = []
|
|
|
|
override func viewDidAppear(_ animated: Bool) {
|
|
super.viewDidAppear(animated)
|
|
|
|
MockSocket.shared.connect(with: [SampleData.shared.nathan, SampleData.shared.wu])
|
|
.onTypingStatus { [weak self] in
|
|
self?.setTypingIndicatorViewHidden(false)
|
|
}.onNewMessage { [weak self] message in
|
|
self?.setTypingIndicatorViewHidden(true, performUpdates: {
|
|
self?.insertMessage(message)
|
|
})
|
|
}
|
|
}
|
|
|
|
override func viewDidLoad() {
|
|
super.viewDidLoad()
|
|
|
|
messageInputBar.inputTextView.keyboardType = .twitter
|
|
|
|
// Configure AutocompleteManager
|
|
autocompleteManager.register(
|
|
prefix: "@",
|
|
with: [
|
|
.font: UIFont.preferredFont(forTextStyle: .body),
|
|
.foregroundColor: UIColor.primaryColor,
|
|
.backgroundColor: UIColor.primaryColor.withAlphaComponent(0.3),
|
|
])
|
|
autocompleteManager.register(prefix: "#")
|
|
autocompleteManager.maxSpaceCountDuringCompletion = 1 // Allow for autocompletes with a space
|
|
|
|
// Set plugins
|
|
messageInputBar.inputPlugins = [autocompleteManager]
|
|
}
|
|
|
|
override func configureMessageCollectionView() {
|
|
super.configureMessageCollectionView()
|
|
|
|
let layout = messagesCollectionView.collectionViewLayout as? MessagesCollectionViewFlowLayout
|
|
layout?.sectionInset = UIEdgeInsets(top: 1, left: 8, bottom: 1, right: 8)
|
|
layout?.setMessageOutgoingCellBottomLabelAlignment(.init(textAlignment: .right, textInsets: .zero))
|
|
layout?.setMessageOutgoingAvatarSize(.zero)
|
|
layout?
|
|
.setMessageOutgoingMessageTopLabelAlignment(LabelAlignment(
|
|
textAlignment: .right,
|
|
textInsets: UIEdgeInsets(top: 0, left: 0, bottom: 0, right: 12)))
|
|
layout?
|
|
.setMessageOutgoingMessageBottomLabelAlignment(LabelAlignment(
|
|
textAlignment: .right,
|
|
textInsets: UIEdgeInsets(top: 0, left: 0, bottom: 0, right: 12)))
|
|
|
|
messagesCollectionView.messagesLayoutDelegate = self
|
|
messagesCollectionView.messagesDisplayDelegate = self
|
|
|
|
additionalBottomInset = 30
|
|
}
|
|
|
|
override func configureMessageInputBar() {
|
|
super.configureMessageInputBar()
|
|
messageInputBar.layer.shadowColor = UIColor.black.cgColor
|
|
messageInputBar.layer.shadowRadius = 4
|
|
messageInputBar.layer.shadowOpacity = 0.3
|
|
messageInputBar.layer.shadowOffset = CGSize(width: 0, height: 0)
|
|
messageInputBar.separatorLine.isHidden = true
|
|
messageInputBar.setRightStackViewWidthConstant(to: 0, animated: false)
|
|
messageInputBar.setMiddleContentView(joinChatButton, animated: false)
|
|
}
|
|
|
|
@objc
|
|
func joinChat() {
|
|
configureMessageInputBarForChat()
|
|
}
|
|
|
|
// MARK: - Helpers
|
|
|
|
func isTimeLabelVisible(at indexPath: IndexPath) -> Bool {
|
|
indexPath.section % 3 == 0 && !isPreviousMessageSameSender(at: indexPath)
|
|
}
|
|
|
|
func isPreviousMessageSameSender(at indexPath: IndexPath) -> Bool {
|
|
guard indexPath.section - 1 >= 0 else { return false }
|
|
return messageList[indexPath.section].user == messageList[indexPath.section - 1].user
|
|
}
|
|
|
|
func isNextMessageSameSender(at indexPath: IndexPath) -> Bool {
|
|
guard indexPath.section + 1 < messageList.count else { return false }
|
|
return messageList[indexPath.section].user == messageList[indexPath.section + 1].user
|
|
}
|
|
|
|
func setTypingIndicatorViewHidden(_ isHidden: Bool, performUpdates updates: (() -> Void)? = nil) {
|
|
setTypingIndicatorViewHidden(isHidden, animated: true, whilePerforming: updates) { [weak self] success in
|
|
if success, self?.isLastSectionVisible() == true {
|
|
self?.messagesCollectionView.scrollToLastItem(animated: true)
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - MessagesDataSource
|
|
|
|
override func cellTopLabelAttributedText(for message: MessageType, at indexPath: IndexPath) -> NSAttributedString? {
|
|
if isTimeLabelVisible(at: indexPath) {
|
|
return NSAttributedString(
|
|
string: MessageKitDateFormatter.shared.string(from: message.sentDate),
|
|
attributes: [
|
|
NSAttributedString.Key.font: UIFont.boldSystemFont(ofSize: 10),
|
|
NSAttributedString.Key.foregroundColor: UIColor.darkGray,
|
|
])
|
|
}
|
|
return nil
|
|
}
|
|
|
|
override func messageTopLabelAttributedText(for message: MessageType, at indexPath: IndexPath) -> NSAttributedString? {
|
|
if !isPreviousMessageSameSender(at: indexPath) {
|
|
let name = message.sender.displayName
|
|
return NSAttributedString(
|
|
string: name,
|
|
attributes: [NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .caption1)])
|
|
}
|
|
return nil
|
|
}
|
|
|
|
override func messageBottomLabelAttributedText(for message: MessageType, at indexPath: IndexPath) -> NSAttributedString? {
|
|
if !isNextMessageSameSender(at: indexPath), isFromCurrentSender(message: message) {
|
|
return NSAttributedString(
|
|
string: "Delivered",
|
|
attributes: [NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .caption1)])
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Async autocomplete requires the manager to reload
|
|
func inputBar(_: InputBarAccessoryView, textViewTextDidChangeTo _: String) {
|
|
guard autocompleteManager.currentSession != nil, autocompleteManager.currentSession?.prefix == "#" else { return }
|
|
// Load some data asyncronously for the given session.prefix
|
|
DispatchQueue.global(qos: .default).async {
|
|
// fake background loading task
|
|
var array: [AutocompleteCompletion] = []
|
|
for _ in 1 ... 10 {
|
|
array.append(AutocompleteCompletion(text: Lorem.word()))
|
|
}
|
|
sleep(1)
|
|
DispatchQueue.main.async { [weak self] in
|
|
self?.asyncCompletions = array
|
|
self?.autocompleteManager.reloadData()
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: Private
|
|
|
|
private func configureMessageInputBarForChat() {
|
|
messageInputBar.setMiddleContentView(messageInputBar.inputTextView, animated: false)
|
|
messageInputBar.setRightStackViewWidthConstant(to: 52, animated: false)
|
|
let bottomItems = [makeButton(named: "ic_at"), makeButton(named: "ic_hashtag"), .flexibleSpace]
|
|
messageInputBar.setStackViewItems(bottomItems, forStack: .bottom, animated: false)
|
|
|
|
messageInputBar.sendButton.activityViewColor = .white
|
|
messageInputBar.sendButton.backgroundColor = .primaryColor
|
|
messageInputBar.sendButton.layer.cornerRadius = 10
|
|
messageInputBar.sendButton.setTitleColor(.white, for: .normal)
|
|
messageInputBar.sendButton.setTitleColor(UIColor(white: 1, alpha: 0.3), for: .highlighted)
|
|
messageInputBar.sendButton.setTitleColor(UIColor(white: 1, alpha: 0.3), for: .disabled)
|
|
messageInputBar.sendButton
|
|
.onSelected { item in
|
|
item.transform = CGAffineTransform(scaleX: 1.05, y: 1.05)
|
|
}.onDeselected { item in
|
|
item.transform = .identity
|
|
}
|
|
}
|
|
|
|
private func makeButton(named: String) -> InputBarButtonItem {
|
|
InputBarButtonItem()
|
|
.configure {
|
|
$0.spacing = .fixed(10)
|
|
$0.image = UIImage(named: named)?.withRenderingMode(.alwaysTemplate)
|
|
$0.setSize(CGSize(width: 25, height: 25), animated: false)
|
|
$0.tintColor = UIColor(white: 0.8, alpha: 1)
|
|
}.onSelected {
|
|
$0.tintColor = .primaryColor
|
|
}.onDeselected {
|
|
$0.tintColor = UIColor(white: 0.8, alpha: 1)
|
|
}.onTouchUpInside { _ in
|
|
print("Item Tapped")
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: AutocompleteManagerDelegate, AutocompleteManagerDataSource
|
|
|
|
extension AutocompleteExampleViewController: AutocompleteManagerDelegate, AutocompleteManagerDataSource {
|
|
// MARK: - AutocompleteManagerDataSource
|
|
|
|
func autocompleteManager(_: AutocompleteManager, autocompleteSourceFor prefix: String) -> [AutocompleteCompletion] {
|
|
if prefix == "@" {
|
|
return SampleData.shared.senders
|
|
.map { user in
|
|
AutocompleteCompletion(
|
|
text: user.displayName,
|
|
context: ["id": user.senderId])
|
|
}
|
|
} else if prefix == "#" {
|
|
return hashtagAutocompletes + asyncCompletions
|
|
}
|
|
return []
|
|
}
|
|
|
|
func autocompleteManager(
|
|
_ manager: AutocompleteManager,
|
|
tableView: UITableView,
|
|
cellForRowAt indexPath: IndexPath,
|
|
for session: AutocompleteSession)
|
|
-> UITableViewCell
|
|
{
|
|
guard
|
|
let cell = tableView
|
|
.dequeueReusableCell(withIdentifier: AutocompleteCell.reuseIdentifier, for: indexPath) as? AutocompleteCell else
|
|
{
|
|
fatalError("Oops, some unknown error occurred")
|
|
}
|
|
let users = SampleData.shared.senders
|
|
let id = session.completion?.context?["id"] as? String
|
|
let user = users.filter { $0.senderId == id }.first
|
|
if let sender = user {
|
|
cell.imageView?.image = SampleData.shared.getAvatarFor(sender: sender).image
|
|
}
|
|
cell.imageViewEdgeInsets = UIEdgeInsets(top: 8, left: 8, bottom: 8, right: 8)
|
|
cell.imageView?.layer.cornerRadius = 14
|
|
cell.imageView?.layer.borderColor = UIColor.primaryColor.cgColor
|
|
cell.imageView?.layer.borderWidth = 1
|
|
cell.imageView?.clipsToBounds = true
|
|
cell.textLabel?.attributedText = manager.attributedText(matching: session, fontSize: 15)
|
|
return cell
|
|
}
|
|
|
|
// MARK: - AutocompleteManagerDelegate
|
|
|
|
func autocompleteManager(_: AutocompleteManager, shouldBecomeVisible: Bool) {
|
|
setAutocompleteManager(active: shouldBecomeVisible)
|
|
}
|
|
|
|
// Optional
|
|
func autocompleteManager(_: AutocompleteManager, shouldRegister _: String, at _: NSRange) -> Bool {
|
|
true
|
|
}
|
|
|
|
// Optional
|
|
func autocompleteManager(_: AutocompleteManager, shouldUnregister _: String) -> Bool {
|
|
true
|
|
}
|
|
|
|
// Optional
|
|
func autocompleteManager(_: AutocompleteManager, shouldComplete _: String, with _: String) -> Bool {
|
|
true
|
|
}
|
|
|
|
// MARK: - AutocompleteManagerDelegate Helper
|
|
|
|
func setAutocompleteManager(active: Bool) {
|
|
let topStackView = messageInputBar.topStackView
|
|
if active, !topStackView.arrangedSubviews.contains(autocompleteManager.tableView) {
|
|
topStackView.insertArrangedSubview(autocompleteManager.tableView, at: topStackView.arrangedSubviews.count)
|
|
topStackView.layoutIfNeeded()
|
|
} else if !active, topStackView.arrangedSubviews.contains(autocompleteManager.tableView) {
|
|
topStackView.removeArrangedSubview(autocompleteManager.tableView)
|
|
topStackView.layoutIfNeeded()
|
|
}
|
|
messageInputBar.invalidateIntrinsicContentSize()
|
|
}
|
|
}
|
|
|
|
// MARK: MessagesDisplayDelegate
|
|
|
|
extension AutocompleteExampleViewController: MessagesDisplayDelegate {
|
|
// MARK: - Text Messages
|
|
|
|
func textColor(for message: MessageType, at _: IndexPath, in _: MessagesCollectionView) -> UIColor {
|
|
isFromCurrentSender(message: message) ? .white : .darkText
|
|
}
|
|
|
|
func detectorAttributes(
|
|
for detector: DetectorType,
|
|
and message: MessageType,
|
|
at _: IndexPath) -> [NSAttributedString.Key: Any]
|
|
{
|
|
switch detector {
|
|
case .hashtag, .mention:
|
|
if isFromCurrentSender(message: message) {
|
|
return [.foregroundColor: UIColor.white]
|
|
} else {
|
|
return [.foregroundColor: UIColor.primaryColor]
|
|
}
|
|
default: return MessageLabel.defaultAttributes
|
|
}
|
|
}
|
|
|
|
func enabledDetectors(for _: MessageType, at _: IndexPath, in _: MessagesCollectionView) -> [DetectorType] {
|
|
[.url, .address, .phoneNumber, .date, .transitInformation, .mention, .hashtag]
|
|
}
|
|
|
|
// MARK: - All Messages
|
|
|
|
func backgroundColor(for message: MessageType, at _: IndexPath, in _: MessagesCollectionView) -> UIColor {
|
|
isFromCurrentSender(message: message) ? .primaryColor : UIColor(red: 230 / 255, green: 230 / 255, blue: 230 / 255, alpha: 1)
|
|
}
|
|
|
|
func messageStyle(for _: MessageType, at _: IndexPath, in _: MessagesCollectionView) -> MessageStyle {
|
|
.bubble
|
|
}
|
|
|
|
func configureAvatarView(
|
|
_ avatarView: AvatarView,
|
|
for message: MessageType,
|
|
at indexPath: IndexPath,
|
|
in _: MessagesCollectionView)
|
|
{
|
|
let avatar = SampleData.shared.getAvatarFor(sender: message.sender)
|
|
avatarView.set(avatar: avatar)
|
|
avatarView.isHidden = isNextMessageSameSender(at: indexPath)
|
|
avatarView.layer.borderWidth = 2
|
|
avatarView.layer.borderColor = UIColor.primaryColor.cgColor
|
|
}
|
|
|
|
func configureAccessoryView(_ accessoryView: UIView, for _: MessageType, at _: IndexPath, in _: MessagesCollectionView) {
|
|
// Cells are reused, so only add a button here once. For real use you would need to
|
|
// ensure any subviews are removed if not needed
|
|
accessoryView.subviews.forEach { $0.removeFromSuperview() }
|
|
|
|
let button = UIButton(type: .infoLight)
|
|
button.tintColor = .primaryColor
|
|
accessoryView.addSubview(button)
|
|
button.frame = accessoryView.bounds
|
|
button.isUserInteractionEnabled = false // respond to accessoryView tap through `MessageCellDelegate`
|
|
accessoryView.layer.cornerRadius = accessoryView.frame.height / 2
|
|
accessoryView.backgroundColor = UIColor.primaryColor.withAlphaComponent(0.3)
|
|
}
|
|
|
|
func configureMediaMessageImageView(
|
|
_ imageView: UIImageView,
|
|
for message: MessageType,
|
|
at _: IndexPath,
|
|
in _: MessagesCollectionView)
|
|
{
|
|
if case MessageKind.photo(let media) = message.kind, let imageURL = media.url {
|
|
imageView.kf.setImage(with: imageURL)
|
|
} else {
|
|
imageView.kf.cancelDownloadTask()
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: MessagesLayoutDelegate
|
|
|
|
extension AutocompleteExampleViewController: MessagesLayoutDelegate {
|
|
func cellTopLabelHeight(for _: MessageType, at indexPath: IndexPath, in _: MessagesCollectionView) -> CGFloat {
|
|
if isTimeLabelVisible(at: indexPath) {
|
|
return 18
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func messageTopLabelHeight(for message: MessageType, at indexPath: IndexPath, in _: MessagesCollectionView) -> CGFloat {
|
|
if isFromCurrentSender(message: message) {
|
|
return !isPreviousMessageSameSender(at: indexPath) ? 20 : 0
|
|
} else {
|
|
return !isPreviousMessageSameSender(at: indexPath) ? 20 : 0
|
|
}
|
|
}
|
|
|
|
func messageBottomLabelHeight(for message: MessageType, at indexPath: IndexPath, in _: MessagesCollectionView) -> CGFloat {
|
|
(!isNextMessageSameSender(at: indexPath) && isFromCurrentSender(message: message)) ? 16 : 0
|
|
}
|
|
}
|