Files
MessageKit/Example/Sources/View Controllers/ChatViewController.swift
T
Martin Púčik bff35fda61 Added Swiftlint and Swiftformat plugins (#1729)
* 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>
2022-07-25 08:46:14 +00:00

374 lines
12 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 MessageKit
import UIKit
// MARK: - ChatViewController
/// A base class for the example controllers
class ChatViewController: MessagesViewController, MessagesDataSource {
// MARK: Internal
// MARK: - Public properties
/// The `BasicAudioController` control the AVAudioPlayer state (play, pause, stop) and update audio cell UI accordingly.
lazy var audioController = BasicAudioController(messageCollectionView: messagesCollectionView)
lazy var messageList: [MockMessage] = []
private(set) lazy var refreshControl: UIRefreshControl = {
let control = UIRefreshControl()
control.addTarget(self, action: #selector(loadMoreMessages), for: .valueChanged)
return control
}()
// MARK: - MessagesDataSource
var currentSender: SenderType {
SampleData.shared.currentSender
}
override func viewDidLoad() {
super.viewDidLoad()
navigationItem.largeTitleDisplayMode = .never
navigationItem.title = "MessageKit"
configureMessageCollectionView()
configureMessageInputBar()
loadFirstMessages()
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
MockSocket.shared.connect(with: [SampleData.shared.nathan, SampleData.shared.wu])
.onNewMessage { [weak self] message in
self?.insertMessage(message)
}
}
override func viewDidDisappear(_ animated: Bool) {
super.viewDidDisappear(animated)
MockSocket.shared.disconnect()
audioController.stopAnyOngoingPlaying()
}
func loadFirstMessages() {
DispatchQueue.global(qos: .userInitiated).async {
let count = UserDefaults.standard.mockMessagesCount()
SampleData.shared.getMessages(count: count) { messages in
DispatchQueue.main.async {
self.messageList = messages
self.messagesCollectionView.reloadData()
self.messagesCollectionView.scrollToLastItem(animated: false)
}
}
}
}
@objc
func loadMoreMessages() {
DispatchQueue.global(qos: .userInitiated).asyncAfter(deadline: .now() + 1) {
SampleData.shared.getMessages(count: 20) { messages in
DispatchQueue.main.async {
self.messageList.insert(contentsOf: messages, at: 0)
self.messagesCollectionView.reloadDataAndKeepOffset()
self.refreshControl.endRefreshing()
}
}
}
}
func configureMessageCollectionView() {
messagesCollectionView.messagesDataSource = self
messagesCollectionView.messageCellDelegate = self
scrollsToLastItemOnKeyboardBeginsEditing = true // default false
maintainPositionOnInputBarHeightChanged = true // default false
showMessageTimestampOnSwipeLeft = true // default false
messagesCollectionView.refreshControl = refreshControl
}
func configureMessageInputBar() {
messageInputBar.delegate = self
messageInputBar.inputTextView.tintColor = .primaryColor
messageInputBar.sendButton.setTitleColor(.primaryColor, for: .normal)
messageInputBar.sendButton.setTitleColor(
UIColor.primaryColor.withAlphaComponent(0.3),
for: .highlighted)
}
// MARK: - Helpers
func insertMessage(_ message: MockMessage) {
messageList.append(message)
// Reload last section to update header/footer labels and insert a new one
messagesCollectionView.performBatchUpdates({
messagesCollectionView.insertSections([messageList.count - 1])
if messageList.count >= 2 {
messagesCollectionView.reloadSections([messageList.count - 2])
}
}, completion: { [weak self] _ in
if self?.isLastSectionVisible() == true {
self?.messagesCollectionView.scrollToLastItem(animated: true)
}
})
}
func isLastSectionVisible() -> Bool {
guard !messageList.isEmpty else { return false }
let lastIndexPath = IndexPath(item: 0, section: messageList.count - 1)
return messagesCollectionView.indexPathsForVisibleItems.contains(lastIndexPath)
}
func numberOfSections(in _: MessagesCollectionView) -> Int {
messageList.count
}
func messageForItem(at indexPath: IndexPath, in _: MessagesCollectionView) -> MessageType {
messageList[indexPath.section]
}
func cellTopLabelAttributedText(for message: MessageType, at indexPath: IndexPath) -> NSAttributedString? {
if indexPath.section % 3 == 0 {
return NSAttributedString(
string: MessageKitDateFormatter.shared.string(from: message.sentDate),
attributes: [
NSAttributedString.Key.font: UIFont.boldSystemFont(ofSize: 10),
NSAttributedString.Key.foregroundColor: UIColor.darkGray,
])
}
return nil
}
func cellBottomLabelAttributedText(for _: MessageType, at _: IndexPath) -> NSAttributedString? {
NSAttributedString(
string: "Read",
attributes: [
NSAttributedString.Key.font: UIFont.boldSystemFont(ofSize: 10),
NSAttributedString.Key.foregroundColor: UIColor.darkGray,
])
}
func messageTopLabelAttributedText(for message: MessageType, at _: IndexPath) -> NSAttributedString? {
let name = message.sender.displayName
return NSAttributedString(
string: name,
attributes: [NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .caption1)])
}
func messageBottomLabelAttributedText(for message: MessageType, at _: IndexPath) -> NSAttributedString? {
let dateString = formatter.string(from: message.sentDate)
return NSAttributedString(
string: dateString,
attributes: [NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .caption2)])
}
func textCell(for _: MessageType, at _: IndexPath, in _: MessagesCollectionView) -> UICollectionViewCell? {
nil
}
// MARK: Private
// MARK: - Private properties
private let formatter: DateFormatter = {
let formatter = DateFormatter()
formatter.dateStyle = .medium
return formatter
}()
}
// MARK: MessageCellDelegate
extension ChatViewController: MessageCellDelegate {
func didTapAvatar(in _: MessageCollectionViewCell) {
print("Avatar tapped")
}
func didTapMessage(in _: MessageCollectionViewCell) {
print("Message tapped")
}
func didTapImage(in _: MessageCollectionViewCell) {
print("Image tapped")
}
func didTapCellTopLabel(in _: MessageCollectionViewCell) {
print("Top cell label tapped")
}
func didTapCellBottomLabel(in _: MessageCollectionViewCell) {
print("Bottom cell label tapped")
}
func didTapMessageTopLabel(in _: MessageCollectionViewCell) {
print("Top message label tapped")
}
func didTapMessageBottomLabel(in _: MessageCollectionViewCell) {
print("Bottom label tapped")
}
func didTapPlayButton(in cell: AudioMessageCell) {
guard
let indexPath = messagesCollectionView.indexPath(for: cell),
let message = messagesCollectionView.messagesDataSource?.messageForItem(at: indexPath, in: messagesCollectionView)
else {
print("Failed to identify message when audio cell receive tap gesture")
return
}
guard audioController.state != .stopped else {
// There is no audio sound playing - prepare to start playing for given audio message
audioController.playSound(for: message, in: cell)
return
}
if audioController.playingMessage?.messageId == message.messageId {
// tap occur in the current cell that is playing audio sound
if audioController.state == .playing {
audioController.pauseSound(for: message, in: cell)
} else {
audioController.resumeSound()
}
} else {
// tap occur in a difference cell that the one is currently playing sound. First stop currently playing and start the sound for given message
audioController.stopAnyOngoingPlaying()
audioController.playSound(for: message, in: cell)
}
}
func didStartAudio(in _: AudioMessageCell) {
print("Did start playing audio sound")
}
func didPauseAudio(in _: AudioMessageCell) {
print("Did pause audio sound")
}
func didStopAudio(in _: AudioMessageCell) {
print("Did stop audio sound")
}
func didTapAccessoryView(in _: MessageCollectionViewCell) {
print("Accessory view tapped")
}
}
// MARK: MessageLabelDelegate
extension ChatViewController: MessageLabelDelegate {
func didSelectAddress(_ addressComponents: [String: String]) {
print("Address Selected: \(addressComponents)")
}
func didSelectDate(_ date: Date) {
print("Date Selected: \(date)")
}
func didSelectPhoneNumber(_ phoneNumber: String) {
print("Phone Number Selected: \(phoneNumber)")
}
func didSelectURL(_ url: URL) {
print("URL Selected: \(url)")
}
func didSelectTransitInformation(_ transitInformation: [String: String]) {
print("TransitInformation Selected: \(transitInformation)")
}
func didSelectHashtag(_ hashtag: String) {
print("Hashtag selected: \(hashtag)")
}
func didSelectMention(_ mention: String) {
print("Mention selected: \(mention)")
}
func didSelectCustom(_ pattern: String, match _: String?) {
print("Custom data detector patter selected: \(pattern)")
}
}
// MARK: InputBarAccessoryViewDelegate
extension ChatViewController: InputBarAccessoryViewDelegate {
// MARK: Internal
@objc
func inputBar(_: InputBarAccessoryView, didPressSendButtonWith _: String) {
processInputBar(messageInputBar)
}
func processInputBar(_ inputBar: InputBarAccessoryView) {
// Here we can parse for which substrings were autocompleted
let attributedText = inputBar.inputTextView.attributedText!
let range = NSRange(location: 0, length: attributedText.length)
attributedText.enumerateAttribute(.autocompleted, in: range, options: []) { _, range, _ in
let substring = attributedText.attributedSubstring(from: range)
let context = substring.attribute(.autocompletedContext, at: 0, effectiveRange: nil)
print("Autocompleted: `", substring, "` with context: ", context ?? [])
}
let components = inputBar.inputTextView.components
inputBar.inputTextView.text = String()
inputBar.invalidatePlugins()
// Send button activity animation
inputBar.sendButton.startAnimating()
inputBar.inputTextView.placeholder = "Sending..."
// Resign first responder for iPad split view
inputBar.inputTextView.resignFirstResponder()
DispatchQueue.global(qos: .default).async {
// fake send request task
sleep(1)
DispatchQueue.main.async { [weak self] in
inputBar.sendButton.stopAnimating()
inputBar.inputTextView.placeholder = "Aa"
self?.insertMessages(components)
self?.messagesCollectionView.scrollToLastItem(animated: true)
}
}
}
// MARK: Private
private func insertMessages(_ data: [Any]) {
for component in data {
let user = SampleData.shared.currentSender
if let str = component as? String {
let message = MockMessage(text: str, user: user, messageId: UUID().uuidString, date: Date())
insertMessage(message)
} else if let img = component as? UIImage {
let message = MockMessage(image: img, user: user, messageId: UUID().uuidString, date: Date())
insertMessage(message)
}
}
}
}