UICollectionView странная анимация ячеек, когда режим отклонения клавиатуры интерактивен - PullRequest
0 голосов
/ 30 мая 2019

Я пытаюсь создать систему чата, используя UICollectionView -Коллекция перевернута вверх дном, и поэтому клетки используют CGAffineTransform(scaleX: -1, y: 1)-, и пока все идет хорошо, я вычисляю размеры ячеек вручную, используя NSString.boundingRect и он работает, как и ожидалось, затем я включил интерактивное увольнение с клавиатуры в коллекционном представлении, с тех пор всякий раз, когда я немного перетаскиваю клавиатуру, я отпускаю ее, что приводит к действительно странной анимации, которую я не могу понять, что вызывает ее.

Я также использую Typist, который является небольшим служебным классом, который облегчает обработку клавиатуры https://github.com/totocaster/Typist, Я не подозреваю об этом, поскольку это простая оболочка UIKeyboard уведомлений.

Я пробовал несколько вещей, включая, но не ограничиваясь

1-Отключение анимации при показе ячейки с помощью UIView.setAnimationsEnabled(false) затем включить его снова перед возвратом неработающей ячейки.

2-Снятие перевернутого подхода в целом, и снова нет, даже когда он находится в обычном преобразовании, ошибка все еще происходит

3-Переписана вся реализация UICollectionView до реализации UITableView, что, что удивительно, дало лучший результат, странная анимация не так сильна, как на UICollectionView, но все же происходит в некоторой степени

4 - удаление Typist в целом и возврат к NotificationCenter и ручное управление и нет, не работает.

UICollectionView инициализации

    lazy var chatCollectionView: UICollectionView = {
        let collectionViewFlowLayout = UICollectionViewFlowLayout()
        collectionViewFlowLayout.scrollDirection = .vertical
        let collectionView = UICollectionView(frame: .zero, collectionViewLayout: collectionViewFlowLayout)
        collectionView.translatesAutoresizingMaskIntoConstraints = false
        collectionView.backgroundColor = .white
        collectionView.showsHorizontalScrollIndicator = false
        collectionView.showsVerticalScrollIndicator = false
        collectionView.register(OutgoingTextCell.self, forCellWithReuseIdentifier: OutgoingTextCell.className)
        collectionView.register(IncomingTextCell.self, forCellWithReuseIdentifier: IncomingTextCell.className)
        collectionView.register(OutgoingImageCell.self, forCellWithReuseIdentifier: OutgoingImageCell.className)
        collectionView.register(IncomingImageCell.self, forCellWithReuseIdentifier: IncomingImageCell.className)
        collectionView.transform = CGAffineTransform(scaleX: 1, y: -1)
        collectionView.semanticContentAttribute = .forceLeftToRight
        collectionView.keyboardDismissMode = .interactive
        return collectionView
    }()

UICollectionView DataSource / Удалить реализацию

extension ChatViewController: UICollectionViewDataSource, UICollectionViewDelegate, UICollectionViewDelegateFlowLayout {

    func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        return presenter.numberOfRows
    }

    func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
        if presenter.isSender(at: indexPath) {
            switch presenter.messageType(at: indexPath) {
            case .text:

                let cell = collectionView.dequeueReusableCell(withReuseIdentifier: OutgoingTextCell.className, for: indexPath) as! OutgoingTextCell
                presenter.configure(cell: AnyConfigurableCell(cell), at: indexPath)
                return cell

            case .photo:

                let cell = collectionView.dequeueReusableCell(withReuseIdentifier: OutgoingImageCell.className, for: indexPath) as! OutgoingImageCell
                presenter.configure(cell: AnyConfigurableCell(cell), at: indexPath)
                return cell
            }
        } else {
            switch presenter.messageType(at: indexPath) {
            case .text:

                let cell = collectionView.dequeueReusableCell(withReuseIdentifier: IncomingTextCell.className, for: indexPath) as! IncomingTextCell
                presenter.configure(cell: AnyConfigurableCell(cell), at: indexPath)
                return cell

            case .photo:

                let cell = collectionView.dequeueReusableCell(withReuseIdentifier: IncomingImageCell.className, for: indexPath) as! IncomingImageCell
                presenter.configure(cell: AnyConfigurableCell(cell), at: indexPath)
                return cell
            }
        }
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
        if presenter.messageType(at: indexPath) == .photo {
            return CGSize(width: view.frame.width, height: 250)
        } else {
            if let height = cachedSizes[presenter.uuidForMessage(at: indexPath)] {
                return CGSize(width: view.frame.width, height: height)
            } else {
                let height = calculateTextHeight(at: indexPath)
                cachedSizes[presenter.uuidForMessage(at: indexPath)] = height
                return CGSize(width: view.frame.width, height: height)
            }
        }
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, insetForSectionAt section: Int) -> UIEdgeInsets {
        return UIEdgeInsets(top: 8, left: 0, bottom: 8, right: 0)
    }

    func collectionView(_ collectionView: UICollectionView, willDisplay cell: UICollectionViewCell, forItemAt indexPath: IndexPath) {
        if indexPath.row == presenter.numberOfRows - 1 && !isPaginating {
            presenter.loadNextPage()
        }
    }

    private func calculateTextHeight(at indexPath: IndexPath) -> CGFloat {
        let approximateSize = CGSize(width: view.frame.width - 88, height: .greatestFiniteMagnitude)
        let estimatedHeight = NSString(string: presenter.contentForMessage(at: indexPath)).boundingRect(with: approximateSize, options: .usesLineFragmentOrigin, attributes: [.font: DinNextFont.regular.getFont(ofSize: 14)], context: nil).height
        return estimatedHeight + 40
    }

}

Ячейка реализации

class OutgoingTextCell: UICollectionViewCell, ConfigurableCell {

    lazy var containerView: UIView = {
        let view = UIView()
        view.translatesAutoresizingMaskIntoConstraints = false
        view.backgroundColor = .coral
        view.layer.cornerRadius = 10
        return view
    }()

    lazy var contentLabel: UILabel = {
        let label = UILabel()
        label.translatesAutoresizingMaskIntoConstraints = false
        label.textColor = .white
        label.numberOfLines = 0
        label.font = DinNextFont.regular.getFont(ofSize: 14)
        return label
    }()

    private lazy var stackView: UIStackView = {
        let stackView = UIStackView(arrangedSubviews: [timeLabel, checkMarkImageView])
        stackView.translatesAutoresizingMaskIntoConstraints = false
        stackView.axis = .horizontal
        stackView.distribution = .equalSpacing
        stackView.alignment = .center
        stackView.spacing = 4
        return stackView
    }()

    private lazy var checkMarkImageView: UIImageView = {
        let imageView = UIImageView(image: UIImage(named: "ic_check")?.withRenderingMode(.alwaysTemplate))
        imageView.translatesAutoresizingMaskIntoConstraints = false
        imageView.tintColor = .white
        imageView.heightAnchor.constraint(equalToConstant: 8).isActive = true
        imageView.widthAnchor.constraint(equalToConstant: 10).isActive = true
        return imageView
    }()

    lazy var timeLabel: UILabel = {
        let label = UILabel()
        label.translatesAutoresizingMaskIntoConstraints = false
        label.textColor = .white
        label.font = DinNextFont.regular.getFont(ofSize: 10)
        label.text = "12:14"
        return label
    }()

    override init(frame: CGRect) {
        super.init(frame: frame)
        layoutUI()
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    private func addSubviews() {
        addSubview(containerView)
        containerView.addSubview(contentLabel)
        containerView.addSubview(stackView)
    }

    func setupContainerViewConstraints() {
        NSLayoutConstraint.activate([
            containerView.topAnchor.constraint(equalTo: topAnchor, constant: 0),
            containerView.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -8),
            containerView.widthAnchor.constraint(greaterThanOrEqualToConstant: 100),
            containerView.heightAnchor.constraint(greaterThanOrEqualToConstant: 34),
            containerView.leadingAnchor.constraint(greaterThanOrEqualTo: leadingAnchor, constant: 64)
        ])
    }

    private func setupContentLabelConstraints() {
        NSLayoutConstraint.activate([
            contentLabel.topAnchor.constraint(equalTo: containerView.topAnchor, constant: 4),
            contentLabel.leadingAnchor.constraint(equalTo: containerView.leadingAnchor, constant: 8),
            contentLabel.trailingAnchor.constraint(equalTo: containerView.trailingAnchor, constant: -8),
        ])
    }

    private func setupStackViewConstraints() {
        NSLayoutConstraint.activate([
            stackView.topAnchor.constraint(equalTo: contentLabel.bottomAnchor, constant: 0),
            stackView.trailingAnchor.constraint(equalTo: containerView.trailingAnchor, constant: -6),
            stackView.bottomAnchor.constraint(equalTo: containerView.bottomAnchor, constant: -4),
            stackView.leadingAnchor.constraint(greaterThanOrEqualTo: containerView.leadingAnchor, constant: 6),
            stackView.heightAnchor.constraint(equalToConstant: 20),
        ])
    }

    private func layoutUI() {
        addSubviews()
        setupContainerViewConstraints()
        setupContentLabelConstraints()
        setupStackViewConstraints()
    }

    func configure(model: MessageViewModel) {
        containerView.transform = CGAffineTransform(scaleX: 1, y: -1)
        contentLabel.text = model.content
        timeLabel.text = model.time
        checkMarkImageView.isHidden = !model.isSent
    }
}

Вывод UICollectionView выглядит следующим образом

https://www.youtube.com/watch?v=RajfZk5lGCQ

Вывод UITableview выглядит следующим образом

https://www.youtube.com/watch?v=6ayG7WhYKXo

Это может быть не совсем понятно, но если вы посмотрите внимательно, вы увидите, что ячейки "кратковременно" анимируются, а при более длинных сообщениях это выглядит как "скользящая анимация", которая совсем не выглядит привлекательной.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...