From d5c756caa3ee52c18ba61c7263f58edfd67af806 Mon Sep 17 00:00:00 2001 From: Morgan Pretty Date: Thu, 15 Aug 2024 10:50:57 +1000 Subject: [PATCH] Fixed a SignalAttachment hash uniqueness issue, fixed attachment UX issues MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit • Fixed an issue where you could select 0 attachments and go to the "send attachment" screen • Fixed an issue where returning from the "send attachment" screen wouldn't have the initially selected attachment selected • Added a uniqueId to the DataSource so instances can be better distinguished from each other (not ideal as the same file would have separate identifiers) --- .../ImagePickerController.swift | 17 +++++++++++++++++ .../SendMediaNavigationController.swift | 4 ++++ .../Attachments/SignalAttachment.swift | 4 ++-- SessionUtilitiesKit/Media/DataSource.h | 1 + SessionUtilitiesKit/Media/DataSource.m | 3 +++ 5 files changed, 27 insertions(+), 2 deletions(-) diff --git a/Session/Media Viewing & Editing/ImagePickerController.swift b/Session/Media Viewing & Editing/ImagePickerController.swift index 3e642fc24..957d59079 100644 --- a/Session/Media Viewing & Editing/ImagePickerController.swift +++ b/Session/Media Viewing & Editing/ImagePickerController.swift @@ -30,6 +30,7 @@ class ImagePickerGridController: UICollectionViewController, PhotoLibraryDelegat private var photoCollection: PhotoCollection private var photoCollectionContents: PhotoCollectionContents private let photoMediaSize = PhotoMediaSize() + private var firstSelectedIndexPath: IndexPath? var collectionViewFlowLayout: UICollectionViewFlowLayout var titleView: TitleView! @@ -212,6 +213,12 @@ class ImagePickerGridController: UICollectionViewController, PhotoLibraryDelegat let cellSize = collectionViewFlowLayout.itemSize photoMediaSize.thumbnailSize = CGSize(width: cellSize.width * scale, height: cellSize.height * scale) + // When we select the first item we immediately deselect it so it doesn't look odd when pushing to the + // next screen, but this in turn looks odd if the user returns and the item is deselected + if let firstSelectedIndexPath: IndexPath = firstSelectedIndexPath { + collectionView.cellForItem(at: firstSelectedIndexPath)?.isSelected = true + } + if !hasEverAppeared { scrollToBottom(animated: false) } @@ -249,6 +256,13 @@ class ImagePickerGridController: UICollectionViewController, PhotoLibraryDelegat DispatchQueue.main.async { // pre-layout collectionPicker for snappier response self.collectionPickerController.view.layoutIfNeeded() + + // We also need to actually inform the collectionView that the item should be selected (if we don't + // then the user won't be able to deselect it) + if let firstSelectedIndexPath: IndexPath = self.firstSelectedIndexPath { + self.collectionView.selectItem(at: firstSelectedIndexPath, animated: false, scrollPosition: .centeredHorizontally) + self.collectionView.cellForItem(at: firstSelectedIndexPath)?.isSelected = true + } } } @@ -489,9 +503,11 @@ class ImagePickerGridController: UICollectionViewController, PhotoLibraryDelegat didSelectAsset: asset, attachmentPublisher: photoCollectionContents.outgoingAttachment(for: asset) ) + firstSelectedIndexPath = nil if !delegate.isInBatchSelectMode { // Don't show "selected" badge unless we're in batch mode + firstSelectedIndexPath = indexPath collectionView.deselectItem(at: indexPath, animated: false) delegate.imagePickerDidCompleteSelection(self) } @@ -511,6 +527,7 @@ class ImagePickerGridController: UICollectionViewController, PhotoLibraryDelegat } delegate.imagePicker(self, didDeselectAsset: asset) + firstSelectedIndexPath = nil } override func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int { diff --git a/Session/Media Viewing & Editing/SendMediaNavigationController.swift b/Session/Media Viewing & Editing/SendMediaNavigationController.swift index 274bf0f85..b5434b26c 100644 --- a/Session/Media Viewing & Editing/SendMediaNavigationController.swift +++ b/Session/Media Viewing & Editing/SendMediaNavigationController.swift @@ -691,6 +691,7 @@ private class DoneButton: UIView { func updateCount() { guard let delegate = delegate else { return } + badge.themeBackgroundColor = (delegate.doneButtonCount > 0 ? .primary : .disabled) badgeLabel.text = numberFormatter.string(for: delegate.doneButtonCount) } @@ -729,11 +730,14 @@ private class DoneButton: UIView { } @objc func didTap(tapGesture: UITapGestureRecognizer) { + guard (delegate?.doneButtonCount ?? 0) > 0 else { return } + delegate?.doneButtonWasTapped(self) } override func touchesBegan(_ touches: Set, with event: UIEvent?) { guard + (delegate?.doneButtonCount ?? 0) > 0, isUserInteractionEnabled, let location: CGPoint = touches.first?.location(in: self), bounds.contains(location) diff --git a/SessionMessagingKit/Sending & Receiving/Attachments/SignalAttachment.swift b/SessionMessagingKit/Sending & Receiving/Attachments/SignalAttachment.swift index dec044d86..e7fe2b82e 100644 --- a/SessionMessagingKit/Sending & Receiving/Attachments/SignalAttachment.swift +++ b/SessionMessagingKit/Sending & Receiving/Attachments/SignalAttachment.swift @@ -1140,11 +1140,11 @@ public class SignalAttachment: Equatable, Hashable { isConvertibleToContactShare.hash(into: &hasher) isVoiceMessage.hash(into: &hasher) - /// There was a crash in `AttachmentApprovalViewController when trying to generate the hash + /// There was a crash in `AttachmentApprovalViewController` when trying to generate the hash /// value to store in a dictionary, I'm guessing it's due to either `dataSource`, `cachedImage` or `cachedVideoPreview` /// so, instead of trying to hash them directly which involves unknown behaviours due to `NSObject` & `UIImage` types, this /// has been reworked to use primitives - dataSource.sourceFilename.hash(into: &hasher) + dataSource.uniqueId.hash(into: &hasher) cachedImage?.size.width.hash(into: &hasher) cachedImage?.size.height.hash(into: &hasher) cachedVideoPreview?.size.width.hash(into: &hasher) diff --git a/SessionUtilitiesKit/Media/DataSource.h b/SessionUtilitiesKit/Media/DataSource.h index 544800031..59b0bc5cb 100755 --- a/SessionUtilitiesKit/Media/DataSource.h +++ b/SessionUtilitiesKit/Media/DataSource.h @@ -10,6 +10,7 @@ NS_ASSUME_NONNULL_BEGIN @interface DataSource : NSObject @property (nonatomic, nullable) NSString *sourceFilename; +@property (nonatomic) NSUUID *uniqueId; // Should not be called unless necessary as it can involve an expensive read. - (NSData *)data; diff --git a/SessionUtilitiesKit/Media/DataSource.m b/SessionUtilitiesKit/Media/DataSource.m index dad3ed801..f673528c1 100755 --- a/SessionUtilitiesKit/Media/DataSource.m +++ b/SessionUtilitiesKit/Media/DataSource.m @@ -124,6 +124,7 @@ NS_ASSUME_NONNULL_BEGIN instance.dataValue = data; instance.fileExtension = fileExtension; instance.shouldDeleteOnDeallocation = YES; + instance.uniqueId = [NSUUID UUID]; return instance; } @@ -246,6 +247,7 @@ NS_ASSUME_NONNULL_BEGIN DataSourcePath *instance = [DataSourcePath new]; instance.filePath = fileUrl.path; instance.shouldDeleteOnDeallocation = shouldDeleteOnDeallocation; + instance.uniqueId = [NSUUID UUID]; return instance; } @@ -259,6 +261,7 @@ NS_ASSUME_NONNULL_BEGIN DataSourcePath *instance = [DataSourcePath new]; instance.filePath = filePath; instance.shouldDeleteOnDeallocation = shouldDeleteOnDeallocation; + instance.uniqueId = [NSUUID UUID]; return instance; }