Explore typing notifications inspired by web

- First iteration
This commit is contained in:
Gil Eluard
2021-03-22 19:55:41 +01:00
parent ea8add1953
commit 2340c575d0
7 changed files with 375 additions and 49 deletions
@@ -0,0 +1,105 @@
//
// Copyright 2021 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
@objcMembers
class RoomTypingBubbleCell: UITableViewCell {
// MARK: - Constants
static let cellIdentifier = "RoomTypingBubbleCell"
// MARK: - Outlets
@IBOutlet weak var additionalUsersLabel: UILabel?
@IBOutlet weak var additionalUsersLabelLeadingConstraint: NSLayoutConstraint?
@IBOutlet weak var dotsView: DotsView?
@IBOutlet weak var dotsViewLeadingConstraint: NSLayoutConstraint?
// MARK: - members
private var userPictureViews: Array<MXKImageView> = Array()
private var typingUsers: Array<MXRoomMember> = Array()
// MARK: - Lifecycle
override func awakeFromNib() {
super.awakeFromNib()
additionalUsersLabel?.textColor = ThemeService.shared().theme.textSecondaryColor
dotsView?.highlightedDotColor = ThemeService.shared().theme.textTertiaryColor
dotsView?.dotColor = ThemeService.shared().theme.tabBarUnselectedItemTintColor
}
override func prepareForReuse() {
super.prepareForReuse()
for pictureView in userPictureViews {
pictureView.removeFromSuperview()
}
}
override func layoutSubviews() {
super.layoutSubviews()
guard userPictureViews.count > 0 else {
return
}
additionalUsersLabel?.sizeToFit()
var pictureViewsMaxX: CGFloat = 0
var xOffset: CGFloat = 0
for pictureView in userPictureViews {
pictureView.center = CGPoint(x: 16 + xOffset + pictureView.bounds.midX, y: self.bounds.midY)
xOffset += round(pictureView.bounds.maxX * 2 / 3)
pictureViewsMaxX = pictureView.frame.maxX
}
let leftMagin: CGFloat = pictureViewsMaxX + (userPictureViews.count == 1 ? 16 : 8)
additionalUsersLabelLeadingConstraint?.constant = leftMagin
dotsViewLeadingConstraint?.constant = additionalUsersLabel?.text.isEmptyOrNil == true ? leftMagin : leftMagin + 8 + (additionalUsersLabel?.frame.width ?? 0)
}
// MARK: - Business methods
func updateTypingUsers(_ typingUsers: Array<MXRoomMember>, mediaManager: MXMediaManager) {
for pictureView in userPictureViews {
pictureView.removeFromSuperview()
}
userPictureViews = Array()
for user in typingUsers {
if userPictureViews.count >= 4 {
break
}
let pictureView = MXKImageView(frame: CGRect(x: 0, y: 0, width: 24, height: 24))
pictureView.layer.masksToBounds = true
pictureView.layer.cornerRadius = pictureView.bounds.midX
let defaultavatarImage = AvatarGenerator.generateAvatar(forMatrixItem: user.userId, withDisplayName: user.displayname)
pictureView.setImageURI(user.avatarUrl, withType: nil, andImageOrientation: .up, toFitViewSize: pictureView.bounds.size, with: MXThumbnailingMethodCrop, previewImage: defaultavatarImage, mediaManager: mediaManager)
userPictureViews.append(pictureView)
self.contentView.addSubview(pictureView)
}
additionalUsersLabel?.text = typingUsers.count <= 4 ? nil : "+\(typingUsers.count - 4)"
self.setNeedsLayout()
}
}