mirror of
https://github.com/element-hq/element-x-ios.git
synced 2025-03-10 13:37:11 +00:00
Show Encryption Authenticity warnings on messages in the timeline. (#3051)
* Initial implementation. * Add developer option for showing timeline item authenticity. * Refactor code to use new SendInfo.Status. --------- Co-authored-by: Doug <douglase@element.io>
This commit is contained in:
parent
cdaa88efcc
commit
a11faeb131
@ -31,6 +31,7 @@
|
|||||||
02D8DF8EB7537EB4E9019DDB /* EventBasedTimelineItemProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 218AB05B4E3889731959C5F1 /* EventBasedTimelineItemProtocol.swift */; };
|
02D8DF8EB7537EB4E9019DDB /* EventBasedTimelineItemProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 218AB05B4E3889731959C5F1 /* EventBasedTimelineItemProtocol.swift */; };
|
||||||
02F4FAE40AF63A1941FD3BBA /* NotificationCenterProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 10B7F8EE25775DE2A305CBB5 /* NotificationCenterProtocol.swift */; };
|
02F4FAE40AF63A1941FD3BBA /* NotificationCenterProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 10B7F8EE25775DE2A305CBB5 /* NotificationCenterProtocol.swift */; };
|
||||||
037006FB6DF1374F94E4058D /* Dictionary.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFDCAC6CAAD65A2C24EA9C4B /* Dictionary.swift */; };
|
037006FB6DF1374F94E4058D /* Dictionary.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFDCAC6CAAD65A2C24EA9C4B /* Dictionary.swift */; };
|
||||||
|
03CDCA6243F89B194E3FAD17 /* EncryptionAuthenticity.swift in Sources */ = {isa = PBXBuildFile; fileRef = 955336CBD5ED73C792D1F580 /* EncryptionAuthenticity.swift */; };
|
||||||
0437765FF480249486893CC7 /* ScreenTrackerViewModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 196004E7695FBA292A7944AF /* ScreenTrackerViewModifier.swift */; };
|
0437765FF480249486893CC7 /* ScreenTrackerViewModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 196004E7695FBA292A7944AF /* ScreenTrackerViewModifier.swift */; };
|
||||||
044DD8F80231BC30570F7965 /* UserDiscoveryService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65AAD845E53B0C8B5E0812C2 /* UserDiscoveryService.swift */; };
|
044DD8F80231BC30570F7965 /* UserDiscoveryService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 65AAD845E53B0C8B5E0812C2 /* UserDiscoveryService.swift */; };
|
||||||
04A16B45228F7678A027C079 /* RoomHeaderView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 422724361B6555364C43281E /* RoomHeaderView.swift */; };
|
04A16B45228F7678A027C079 /* RoomHeaderView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 422724361B6555364C43281E /* RoomHeaderView.swift */; };
|
||||||
@ -1771,6 +1772,7 @@
|
|||||||
94028A227645FA880B966211 /* WaveformSource.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WaveformSource.swift; sourceTree = "<group>"; };
|
94028A227645FA880B966211 /* WaveformSource.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WaveformSource.swift; sourceTree = "<group>"; };
|
||||||
94D670124FC3E84F23A62CCF /* APNSPayload.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = APNSPayload.swift; sourceTree = "<group>"; };
|
94D670124FC3E84F23A62CCF /* APNSPayload.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = APNSPayload.swift; sourceTree = "<group>"; };
|
||||||
9501D11B4258DFA33BA3B40F /* ServerSelectionScreenModels.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ServerSelectionScreenModels.swift; sourceTree = "<group>"; };
|
9501D11B4258DFA33BA3B40F /* ServerSelectionScreenModels.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ServerSelectionScreenModels.swift; sourceTree = "<group>"; };
|
||||||
|
955336CBD5ED73C792D1F580 /* EncryptionAuthenticity.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EncryptionAuthenticity.swift; sourceTree = "<group>"; };
|
||||||
95BAC0F6C9644336E9567EE6 /* NSRegularExpresion.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NSRegularExpresion.swift; sourceTree = "<group>"; };
|
95BAC0F6C9644336E9567EE6 /* NSRegularExpresion.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NSRegularExpresion.swift; sourceTree = "<group>"; };
|
||||||
969694F67E844FCA51F7E051 /* sv */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = sv; path = sv.lproj/InfoPlist.strings; sourceTree = "<group>"; };
|
969694F67E844FCA51F7E051 /* sv */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = sv; path = sv.lproj/InfoPlist.strings; sourceTree = "<group>"; };
|
||||||
96C4762F8D6112E43117DB2F /* CustomStringConvertible.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CustomStringConvertible.swift; sourceTree = "<group>"; };
|
96C4762F8D6112E43117DB2F /* CustomStringConvertible.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CustomStringConvertible.swift; sourceTree = "<group>"; };
|
||||||
@ -3516,6 +3518,7 @@
|
|||||||
children = (
|
children = (
|
||||||
B858A61F2A570DFB8DE570A7 /* AggregratedReaction.swift */,
|
B858A61F2A570DFB8DE570A7 /* AggregratedReaction.swift */,
|
||||||
96C4762F8D6112E43117DB2F /* CustomStringConvertible.swift */,
|
96C4762F8D6112E43117DB2F /* CustomStringConvertible.swift */,
|
||||||
|
955336CBD5ED73C792D1F580 /* EncryptionAuthenticity.swift */,
|
||||||
314F1C79850BE46E8ABEAFCB /* ReadReceipt.swift */,
|
314F1C79850BE46E8ABEAFCB /* ReadReceipt.swift */,
|
||||||
5DE8D25D6A91030175D52A20 /* RoomTimelineItemProperties.swift */,
|
5DE8D25D6A91030175D52A20 /* RoomTimelineItemProperties.swift */,
|
||||||
BE89A8BD65CCE3FCC925CA14 /* TimelineItemReplyDetails.swift */,
|
BE89A8BD65CCE3FCC925CA14 /* TimelineItemReplyDetails.swift */,
|
||||||
@ -6198,6 +6201,7 @@
|
|||||||
8B1D5CE017EEC734CF5FE130 /* Encodable.swift in Sources */,
|
8B1D5CE017EEC734CF5FE130 /* Encodable.swift in Sources */,
|
||||||
4C5A638DAA8AF64565BA4866 /* EncryptedRoomTimelineItem.swift in Sources */,
|
4C5A638DAA8AF64565BA4866 /* EncryptedRoomTimelineItem.swift in Sources */,
|
||||||
B5903E48CF43259836BF2DBF /* EncryptedRoomTimelineView.swift in Sources */,
|
B5903E48CF43259836BF2DBF /* EncryptedRoomTimelineView.swift in Sources */,
|
||||||
|
03CDCA6243F89B194E3FAD17 /* EncryptionAuthenticity.swift in Sources */,
|
||||||
FBD402E3170EB1ED0D1AA672 /* EncryptionKeyProvider.swift in Sources */,
|
FBD402E3170EB1ED0D1AA672 /* EncryptionKeyProvider.swift in Sources */,
|
||||||
46A6DB0F78FB399BD59E2D41 /* EncryptionKeyProviderProtocol.swift in Sources */,
|
46A6DB0F78FB399BD59E2D41 /* EncryptionKeyProviderProtocol.swift in Sources */,
|
||||||
0C6DF318E9C8F6461E6ABDE7 /* EncryptionResetPasswordScreen.swift in Sources */,
|
0C6DF318E9C8F6461E6ABDE7 /* EncryptionResetPasswordScreen.swift in Sources */,
|
||||||
@ -7553,7 +7557,7 @@
|
|||||||
repositoryURL = "https://github.com/element-hq/matrix-rust-components-swift";
|
repositoryURL = "https://github.com/element-hq/matrix-rust-components-swift";
|
||||||
requirement = {
|
requirement = {
|
||||||
kind = exactVersion;
|
kind = exactVersion;
|
||||||
version = 1.0.30;
|
version = 1.0.31;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
701C7BEF8F70F7A83E852DCC /* XCRemoteSwiftPackageReference "GZIP" */ = {
|
701C7BEF8F70F7A83E852DCC /* XCRemoteSwiftPackageReference "GZIP" */ = {
|
||||||
|
@ -149,8 +149,8 @@
|
|||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
"location" : "https://github.com/element-hq/matrix-rust-components-swift",
|
"location" : "https://github.com/element-hq/matrix-rust-components-swift",
|
||||||
"state" : {
|
"state" : {
|
||||||
"revision" : "bc534e15fa0749d668b201b923ee57204afb868a",
|
"revision" : "8e2b4049fb492dcf5b0c796784b7aa7a3c099943",
|
||||||
"version" : "1.0.30"
|
"version" : "1.0.31"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -256,6 +256,7 @@
|
|||||||
"error_some_messages_have_not_been_sent" = "Some messages have not been sent";
|
"error_some_messages_have_not_been_sent" = "Some messages have not been sent";
|
||||||
"error_unknown" = "Sorry, an error occurred";
|
"error_unknown" = "Sorry, an error occurred";
|
||||||
"event_shield_reason_authenticity_not_guaranteed" = "The authenticity of this encrypted message can't be guaranteed on this device.";
|
"event_shield_reason_authenticity_not_guaranteed" = "The authenticity of this encrypted message can't be guaranteed on this device.";
|
||||||
|
"event_shield_reason_sent_in_clear" = "Sent in clear.";
|
||||||
"event_shield_reason_unknown_device" = "Encrypted by an unknown or deleted device.";
|
"event_shield_reason_unknown_device" = "Encrypted by an unknown or deleted device.";
|
||||||
"event_shield_reason_unsigned_device" = "Encrypted by a device not verified by its owner.";
|
"event_shield_reason_unsigned_device" = "Encrypted by a device not verified by its owner.";
|
||||||
"event_shield_reason_unverified_identity" = "Encrypted by an unverified user.";
|
"event_shield_reason_unverified_identity" = "Encrypted by an unverified user.";
|
||||||
|
@ -4,10 +4,6 @@
|
|||||||
/* Used for testing */
|
/* Used for testing */
|
||||||
"untranslated" = "Untranslated";
|
"untranslated" = "Untranslated";
|
||||||
|
|
||||||
// MARK: - Shields
|
|
||||||
|
|
||||||
"send_info_not_encrypted" = "Not encrypted";
|
|
||||||
|
|
||||||
// MARK: - Soft logout
|
// MARK: - Soft logout
|
||||||
|
|
||||||
"soft_logout_signin_title" = "Sign in";
|
"soft_logout_signin_title" = "Sign in";
|
||||||
|
@ -47,6 +47,7 @@ final class AppSettings {
|
|||||||
case publicSearchEnabled
|
case publicSearchEnabled
|
||||||
case fuzzyRoomListSearchEnabled
|
case fuzzyRoomListSearchEnabled
|
||||||
case pinningEnabled
|
case pinningEnabled
|
||||||
|
case timelineItemAuthenticityEnabled
|
||||||
}
|
}
|
||||||
|
|
||||||
private static var suiteName: String = InfoPlistReader.main.appGroupIdentifier
|
private static var suiteName: String = InfoPlistReader.main.appGroupIdentifier
|
||||||
@ -284,6 +285,9 @@ final class AppSettings {
|
|||||||
|
|
||||||
@UserPreference(key: UserDefaultsKeys.pinningEnabled, defaultValue: false, storageType: .userDefaults(store))
|
@UserPreference(key: UserDefaultsKeys.pinningEnabled, defaultValue: false, storageType: .userDefaults(store))
|
||||||
var pinningEnabled
|
var pinningEnabled
|
||||||
|
|
||||||
|
@UserPreference(key: UserDefaultsKeys.timelineItemAuthenticityEnabled, defaultValue: false, storageType: .userDefaults(store))
|
||||||
|
var timelineItemAuthenticityEnabled
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -540,6 +540,7 @@ class RoomFlowCoordinator: FlowCoordinatorProtocol {
|
|||||||
let userID = userSession.clientProxy.userID
|
let userID = userSession.clientProxy.userID
|
||||||
|
|
||||||
let timelineItemFactory = RoomTimelineItemFactory(userID: userID,
|
let timelineItemFactory = RoomTimelineItemFactory(userID: userID,
|
||||||
|
encryptionAuthenticityEnabled: appSettings.timelineItemAuthenticityEnabled,
|
||||||
attributedStringBuilder: AttributedStringBuilder(mentionBuilder: MentionBuilder()),
|
attributedStringBuilder: AttributedStringBuilder(mentionBuilder: MentionBuilder()),
|
||||||
stateEventStringBuilder: RoomStateEventStringBuilder(userID: userID))
|
stateEventStringBuilder: RoomStateEventStringBuilder(userID: userID))
|
||||||
|
|
||||||
@ -1033,6 +1034,7 @@ class RoomFlowCoordinator: FlowCoordinatorProtocol {
|
|||||||
let userID = userSession.clientProxy.userID
|
let userID = userSession.clientProxy.userID
|
||||||
|
|
||||||
let timelineItemFactory = RoomTimelineItemFactory(userID: userID,
|
let timelineItemFactory = RoomTimelineItemFactory(userID: userID,
|
||||||
|
encryptionAuthenticityEnabled: appSettings.timelineItemAuthenticityEnabled,
|
||||||
attributedStringBuilder: AttributedStringBuilder(mentionBuilder: MentionBuilder()),
|
attributedStringBuilder: AttributedStringBuilder(mentionBuilder: MentionBuilder()),
|
||||||
stateEventStringBuilder: RoomStateEventStringBuilder(userID: userID))
|
stateEventStringBuilder: RoomStateEventStringBuilder(userID: userID))
|
||||||
|
|
||||||
|
@ -10,8 +10,6 @@ import Foundation
|
|||||||
// swiftlint:disable explicit_type_interface function_parameter_count identifier_name line_length
|
// swiftlint:disable explicit_type_interface function_parameter_count identifier_name line_length
|
||||||
// swiftlint:disable nesting type_body_length type_name vertical_whitespace_opening_braces
|
// swiftlint:disable nesting type_body_length type_name vertical_whitespace_opening_braces
|
||||||
internal enum UntranslatedL10n {
|
internal enum UntranslatedL10n {
|
||||||
/// Not encrypted
|
|
||||||
internal static var sendInfoNotEncrypted: String { return UntranslatedL10n.tr("Untranslated", "send_info_not_encrypted") }
|
|
||||||
/// Clear all data currently stored on this device?
|
/// Clear all data currently stored on this device?
|
||||||
/// Sign in again to access your account data and messages.
|
/// Sign in again to access your account data and messages.
|
||||||
internal static var softLogoutClearDataDialogContent: String { return UntranslatedL10n.tr("Untranslated", "soft_logout_clear_data_dialog_content") }
|
internal static var softLogoutClearDataDialogContent: String { return UntranslatedL10n.tr("Untranslated", "soft_logout_clear_data_dialog_content") }
|
||||||
|
@ -570,6 +570,8 @@ internal enum L10n {
|
|||||||
internal static var errorUnknown: String { return L10n.tr("Localizable", "error_unknown") }
|
internal static var errorUnknown: String { return L10n.tr("Localizable", "error_unknown") }
|
||||||
/// The authenticity of this encrypted message can't be guaranteed on this device.
|
/// The authenticity of this encrypted message can't be guaranteed on this device.
|
||||||
internal static var eventShieldReasonAuthenticityNotGuaranteed: String { return L10n.tr("Localizable", "event_shield_reason_authenticity_not_guaranteed") }
|
internal static var eventShieldReasonAuthenticityNotGuaranteed: String { return L10n.tr("Localizable", "event_shield_reason_authenticity_not_guaranteed") }
|
||||||
|
/// Sent in clear.
|
||||||
|
internal static var eventShieldReasonSentInClear: String { return L10n.tr("Localizable", "event_shield_reason_sent_in_clear") }
|
||||||
/// Encrypted by an unknown or deleted device.
|
/// Encrypted by an unknown or deleted device.
|
||||||
internal static var eventShieldReasonUnknownDevice: String { return L10n.tr("Localizable", "event_shield_reason_unknown_device") }
|
internal static var eventShieldReasonUnknownDevice: String { return L10n.tr("Localizable", "event_shield_reason_unknown_device") }
|
||||||
/// Encrypted by a device not verified by its owner.
|
/// Encrypted by a device not verified by its owner.
|
||||||
|
@ -330,6 +330,8 @@ struct TimelineItemBubbledStylerView_Previews: PreviewProvider, TestablePreview
|
|||||||
.previewDisplayName("Replies")
|
.previewDisplayName("Replies")
|
||||||
threads
|
threads
|
||||||
.previewDisplayName("Thread decorator")
|
.previewDisplayName("Thread decorator")
|
||||||
|
encryptionAuthenticity
|
||||||
|
.previewDisplayName("Encryption Indicators")
|
||||||
}
|
}
|
||||||
|
|
||||||
// These akwats include a reply
|
// These akwats include a reply
|
||||||
@ -477,4 +479,80 @@ struct TimelineItemBubbledStylerView_Previews: PreviewProvider, TestablePreview
|
|||||||
}
|
}
|
||||||
.environmentObject(viewModel.context)
|
.environmentObject(viewModel.context)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static var encryptionAuthenticity: some View {
|
||||||
|
VStack(spacing: 0) {
|
||||||
|
RoomTimelineItemView(viewState: .init(item: TextRoomTimelineItem(id: .init(timelineID: ""),
|
||||||
|
timestamp: "10:42",
|
||||||
|
isOutgoing: true,
|
||||||
|
isEditable: false,
|
||||||
|
canBeRepliedTo: true,
|
||||||
|
isThreaded: false,
|
||||||
|
sender: .init(id: "whoever"),
|
||||||
|
content: .init(body: "A long message that should be on multiple lines."),
|
||||||
|
properties: RoomTimelineItemProperties(encryptionAuthenticity: .unsignedDevice(color: .red))),
|
||||||
|
groupStyle: .single))
|
||||||
|
|
||||||
|
RoomTimelineItemView(viewState: .init(item: TextRoomTimelineItem(id: .init(timelineID: ""),
|
||||||
|
timestamp: "10:42",
|
||||||
|
isOutgoing: true,
|
||||||
|
isEditable: false,
|
||||||
|
canBeRepliedTo: true,
|
||||||
|
isThreaded: false,
|
||||||
|
sender: .init(id: "whoever"),
|
||||||
|
content: .init(body: "A long message that should be on multiple lines."),
|
||||||
|
properties: RoomTimelineItemProperties(isEdited: true,
|
||||||
|
encryptionAuthenticity: .unsignedDevice(color: .red))),
|
||||||
|
groupStyle: .single))
|
||||||
|
|
||||||
|
RoomTimelineItemView(viewState: .init(item: TextRoomTimelineItem(id: .init(timelineID: ""),
|
||||||
|
timestamp: "10:42",
|
||||||
|
isOutgoing: false,
|
||||||
|
isEditable: false,
|
||||||
|
canBeRepliedTo: true,
|
||||||
|
isThreaded: false,
|
||||||
|
sender: .init(id: "whoever"),
|
||||||
|
content: .init(body: "Short message"),
|
||||||
|
properties: RoomTimelineItemProperties(encryptionAuthenticity: .unknownDevice(color: .red))),
|
||||||
|
groupStyle: .first))
|
||||||
|
|
||||||
|
RoomTimelineItemView(viewState: .init(item: TextRoomTimelineItem(id: .init(timelineID: ""),
|
||||||
|
timestamp: "10:42",
|
||||||
|
isOutgoing: false,
|
||||||
|
isEditable: false,
|
||||||
|
canBeRepliedTo: true,
|
||||||
|
isThreaded: false,
|
||||||
|
sender: .init(id: "whoever"),
|
||||||
|
content: .init(body: "Message goes Here"),
|
||||||
|
properties: RoomTimelineItemProperties(encryptionAuthenticity: .notGuaranteed(color: .gray))),
|
||||||
|
groupStyle: .last))
|
||||||
|
|
||||||
|
ImageRoomTimelineView(timelineItem: ImageRoomTimelineItem(id: .random,
|
||||||
|
timestamp: "Now",
|
||||||
|
isOutgoing: false,
|
||||||
|
isEditable: false,
|
||||||
|
canBeRepliedTo: true,
|
||||||
|
isThreaded: false,
|
||||||
|
sender: .init(id: "Bob"),
|
||||||
|
content: .init(body: "Some other image", source: MediaSourceProxy(url: .picturesDirectory, mimeType: "image/png"), thumbnailSource: nil),
|
||||||
|
|
||||||
|
properties: RoomTimelineItemProperties(encryptionAuthenticity: .notGuaranteed(color: .gray))))
|
||||||
|
|
||||||
|
VoiceMessageRoomTimelineView(timelineItem: .init(id: .init(timelineID: ""),
|
||||||
|
timestamp: "10:42",
|
||||||
|
isOutgoing: true,
|
||||||
|
isEditable: false,
|
||||||
|
canBeRepliedTo: true,
|
||||||
|
isThreaded: true,
|
||||||
|
sender: .init(id: ""),
|
||||||
|
content: .init(body: "audio.ogg",
|
||||||
|
duration: 100,
|
||||||
|
waveform: EstimatedWaveform.mockWaveform,
|
||||||
|
source: nil,
|
||||||
|
contentType: nil),
|
||||||
|
properties: RoomTimelineItemProperties(encryptionAuthenticity: .notGuaranteed(color: .gray))),
|
||||||
|
playerState: AudioPlayerState(id: .timelineItemIdentifier(.random), duration: 10, waveform: EstimatedWaveform.mockWaveform))
|
||||||
|
}
|
||||||
|
.environmentObject(viewModel.context)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -55,10 +55,17 @@ private struct TimelineItemSendInfoLabel: View {
|
|||||||
|
|
||||||
var statusIcon: KeyPath<CompoundIcons, Image>? {
|
var statusIcon: KeyPath<CompoundIcons, Image>? {
|
||||||
switch sendInfo.status {
|
switch sendInfo.status {
|
||||||
case .sendingFailed: \.error
|
case .sendingFailed:
|
||||||
case .unverifiedSession, .authenticityUnknown: \.admin
|
\.error
|
||||||
case .unencrypted: \.keyOff
|
case .encryptionAuthenticity(.notGuaranteed):
|
||||||
case .none: nil
|
\.infoSolid
|
||||||
|
case .encryptionAuthenticity(.unknownDevice),
|
||||||
|
.encryptionAuthenticity(.unsignedDevice),
|
||||||
|
.encryptionAuthenticity(.unverifiedIdentity),
|
||||||
|
.encryptionAuthenticity(.sentInClear):
|
||||||
|
\.lockOff
|
||||||
|
case .none:
|
||||||
|
nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,9 +74,7 @@ private struct TimelineItemSendInfoLabel: View {
|
|||||||
case .sendingFailed: L10n.commonSendingFailed
|
case .sendingFailed: L10n.commonSendingFailed
|
||||||
case .none: nil
|
case .none: nil
|
||||||
// Temporary testing strings.
|
// Temporary testing strings.
|
||||||
case .unverifiedSession: L10n.eventShieldReasonUnsignedDevice
|
case .encryptionAuthenticity(let authenticity): authenticity.message
|
||||||
case .authenticityUnknown: L10n.eventShieldReasonAuthenticityNotGuaranteed
|
|
||||||
case .unencrypted: UntranslatedL10n.sendInfoNotEncrypted
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,7 +116,7 @@ private struct TimelineItemSendInfoLabel: View {
|
|||||||
|
|
||||||
/// All the data needed to render a timeline item's send info label.
|
/// All the data needed to render a timeline item's send info label.
|
||||||
private struct TimelineItemSendInfo {
|
private struct TimelineItemSendInfo {
|
||||||
enum Status { case sendingFailed, unverifiedSession, authenticityUnknown, unencrypted }
|
enum Status { case sendingFailed, encryptionAuthenticity(EncryptionAuthenticity) }
|
||||||
|
|
||||||
/// Describes how the content and the send info should be arranged inside a bubble
|
/// Describes how the content and the send info should be arranged inside a bubble
|
||||||
enum LayoutType {
|
enum LayoutType {
|
||||||
@ -126,9 +131,11 @@ private struct TimelineItemSendInfo {
|
|||||||
|
|
||||||
var foregroundStyle: Color {
|
var foregroundStyle: Color {
|
||||||
switch status {
|
switch status {
|
||||||
case .sendingFailed, .unverifiedSession:
|
case .sendingFailed:
|
||||||
.compound.textCriticalPrimary
|
.compound.textCriticalPrimary
|
||||||
case .authenticityUnknown, .unencrypted, .none:
|
case .encryptionAuthenticity(let authenticity):
|
||||||
|
authenticity.foregroundStyle
|
||||||
|
case .none:
|
||||||
.compound.textSecondary
|
.compound.textSecondary
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -140,6 +147,8 @@ private extension TimelineItemSendInfo {
|
|||||||
|
|
||||||
status = if adjustedDeliveryStatus == .sendingFailed {
|
status = if adjustedDeliveryStatus == .sendingFailed {
|
||||||
.sendingFailed
|
.sendingFailed
|
||||||
|
} else if let authenticity = timelineItem.properties.encryptionAuthenticity {
|
||||||
|
.encryptionAuthenticity(authenticity)
|
||||||
} else {
|
} else {
|
||||||
nil
|
nil
|
||||||
}
|
}
|
||||||
@ -161,6 +170,24 @@ private extension TimelineItemSendInfo {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private extension EncryptionAuthenticity {
|
||||||
|
var foregroundStyle: SwiftUI.Color {
|
||||||
|
switch self {
|
||||||
|
case .notGuaranteed(let color),
|
||||||
|
.unknownDevice(let color),
|
||||||
|
.unsignedDevice(let color),
|
||||||
|
.unverifiedIdentity(let color),
|
||||||
|
.sentInClear(let color):
|
||||||
|
switch color {
|
||||||
|
case .red:
|
||||||
|
.compound.textCriticalPrimary
|
||||||
|
case .gray:
|
||||||
|
.compound.textSecondary
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// MARK: - Previews
|
// MARK: - Previews
|
||||||
|
|
||||||
struct TimelineItemSendInfoLabel_Previews: PreviewProvider, TestablePreview {
|
struct TimelineItemSendInfoLabel_Previews: PreviewProvider, TestablePreview {
|
||||||
@ -172,14 +199,14 @@ struct TimelineItemSendInfoLabel_Previews: PreviewProvider, TestablePreview {
|
|||||||
status: .sendingFailed,
|
status: .sendingFailed,
|
||||||
layoutType: .horizontal()))
|
layoutType: .horizontal()))
|
||||||
TimelineItemSendInfoLabel(sendInfo: .init(localizedString: "09:47 AM",
|
TimelineItemSendInfoLabel(sendInfo: .init(localizedString: "09:47 AM",
|
||||||
status: .unverifiedSession,
|
status: .encryptionAuthenticity(.unsignedDevice(color: .red)),
|
||||||
layoutType: .horizontal()))
|
layoutType: .horizontal()))
|
||||||
TimelineItemSendInfoLabel(sendInfo: .init(localizedString: "09:47 AM",
|
TimelineItemSendInfoLabel(sendInfo: .init(localizedString: "09:47 AM",
|
||||||
status: .authenticityUnknown,
|
status: .encryptionAuthenticity(.notGuaranteed(color: .gray)),
|
||||||
layoutType: .horizontal()))
|
|
||||||
TimelineItemSendInfoLabel(sendInfo: .init(localizedString: "09:47 AM",
|
|
||||||
status: .unencrypted,
|
|
||||||
layoutType: .horizontal()))
|
layoutType: .horizontal()))
|
||||||
|
// TimelineItemSendInfoLabel(sendInfo: .init(localizedString: "09:47 AM",
|
||||||
|
// status: .unencrypted,
|
||||||
|
// layoutType: .horizontal()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -53,6 +53,7 @@ protocol DeveloperOptionsProtocol: AnyObject {
|
|||||||
var elementCallBaseURLOverride: URL? { get set }
|
var elementCallBaseURLOverride: URL? { get set }
|
||||||
var fuzzyRoomListSearchEnabled: Bool { get set }
|
var fuzzyRoomListSearchEnabled: Bool { get set }
|
||||||
var pinningEnabled: Bool { get set }
|
var pinningEnabled: Bool { get set }
|
||||||
|
var timelineItemAuthenticityEnabled: Bool { get set }
|
||||||
}
|
}
|
||||||
|
|
||||||
extension AppSettings: DeveloperOptionsProtocol { }
|
extension AppSettings: DeveloperOptionsProtocol { }
|
||||||
|
@ -50,6 +50,13 @@ struct DeveloperOptionsScreen: View {
|
|||||||
Text("Fuzzy searching")
|
Text("Fuzzy searching")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Section("Encryption") {
|
||||||
|
Toggle(isOn: $context.timelineItemAuthenticityEnabled) {
|
||||||
|
Text("Message authenticity warnings")
|
||||||
|
Text("Requires app reboot")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Section("Element Call") {
|
Section("Element Call") {
|
||||||
TextField(context.viewState.elementCallBaseURL.absoluteString, text: $elementCallBaseURLString)
|
TextField(context.viewState.elementCallBaseURL.absoluteString, text: $elementCallBaseURLString)
|
||||||
|
@ -0,0 +1,74 @@
|
|||||||
|
//
|
||||||
|
// Copyright 2024 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 Foundation
|
||||||
|
import MatrixRustSDK
|
||||||
|
|
||||||
|
/// Represents and issue with a timeline item's authenticity such as coming from an
|
||||||
|
/// unsigned session or being sent unencrypted in an encrypted room. See Rust's
|
||||||
|
/// `ShieldStateCode` for more information about the meaning of the cases.
|
||||||
|
enum EncryptionAuthenticity: Hashable {
|
||||||
|
enum Color { case red, gray }
|
||||||
|
|
||||||
|
case notGuaranteed(color: Color)
|
||||||
|
case unknownDevice(color: Color)
|
||||||
|
case unsignedDevice(color: Color)
|
||||||
|
case unverifiedIdentity(color: Color)
|
||||||
|
case sentInClear(color: Color)
|
||||||
|
|
||||||
|
var message: String {
|
||||||
|
switch self {
|
||||||
|
case .notGuaranteed:
|
||||||
|
L10n.eventShieldReasonAuthenticityNotGuaranteed
|
||||||
|
case .unknownDevice:
|
||||||
|
L10n.eventShieldReasonUnknownDevice
|
||||||
|
case .unsignedDevice:
|
||||||
|
L10n.eventShieldReasonUnsignedDevice
|
||||||
|
case .unverifiedIdentity:
|
||||||
|
L10n.eventShieldReasonUnverifiedIdentity
|
||||||
|
case .sentInClear:
|
||||||
|
L10n.eventShieldReasonSentInClear
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
extension EncryptionAuthenticity {
|
||||||
|
init?(shieldState: ShieldState) {
|
||||||
|
switch shieldState {
|
||||||
|
case .red(let code, _):
|
||||||
|
self.init(shieldStateCode: code, color: .red)
|
||||||
|
case .grey(let code, _):
|
||||||
|
self.init(shieldStateCode: code, color: .gray)
|
||||||
|
case .none:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
init(shieldStateCode: ShieldStateCode, color: EncryptionAuthenticity.Color) {
|
||||||
|
switch shieldStateCode {
|
||||||
|
case .authenticityNotGuaranteed:
|
||||||
|
self = .notGuaranteed(color: color)
|
||||||
|
case .unknownDevice:
|
||||||
|
self = .unknownDevice(color: color)
|
||||||
|
case .unsignedDevice:
|
||||||
|
self = .unsignedDevice(color: color)
|
||||||
|
case .unverifiedIdentity:
|
||||||
|
self = .unverifiedIdentity(color: color)
|
||||||
|
case .sentInClear:
|
||||||
|
self = .sentInClear(color: color)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -24,6 +24,8 @@ struct RoomTimelineItemProperties: Hashable {
|
|||||||
var reactions: [AggregatedReaction] = []
|
var reactions: [AggregatedReaction] = []
|
||||||
/// The delivery status for this item. If a sent message is echoed the value is nil.
|
/// The delivery status for this item. If a sent message is echoed the value is nil.
|
||||||
var deliveryStatus: TimelineItemDeliveryStatus?
|
var deliveryStatus: TimelineItemDeliveryStatus?
|
||||||
/// The read receipts of the item, ordered from newest to oldest
|
/// The read receipts of the item, ordered from newest to oldest.
|
||||||
var orderedReadReceipts: [ReadReceipt] = []
|
var orderedReadReceipts: [ReadReceipt] = []
|
||||||
|
/// Authenticity warnings for item's sent in encrypted rooms.
|
||||||
|
var encryptionAuthenticity: EncryptionAuthenticity?
|
||||||
}
|
}
|
||||||
|
@ -120,7 +120,9 @@ class EventTimelineItemProxy {
|
|||||||
let debugInfo = item.debugInfo()
|
let debugInfo = item.debugInfo()
|
||||||
return TimelineItemDebugInfo(model: debugInfo.model, originalJSON: debugInfo.originalJson, latestEditJSON: debugInfo.latestEditJson)
|
return TimelineItemDebugInfo(model: debugInfo.model, originalJSON: debugInfo.originalJson, latestEditJSON: debugInfo.latestEditJson)
|
||||||
}()
|
}()
|
||||||
|
|
||||||
|
lazy var shieldState = item.getShield(strict: false)
|
||||||
|
|
||||||
lazy var readReceipts = item.readReceipts()
|
lazy var readReceipts = item.readReceipts()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,7 +50,11 @@ extension EventBasedTimelineItemProtocol {
|
|||||||
var pollIfAvailable: Poll? {
|
var pollIfAvailable: Poll? {
|
||||||
(self as? PollRoomTimelineItem)?.poll
|
(self as? PollRoomTimelineItem)?.poll
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var hasStatusIcon: Bool {
|
||||||
|
hasFailedToSend || properties.encryptionAuthenticity != nil
|
||||||
|
}
|
||||||
|
|
||||||
var hasFailedToSend: Bool {
|
var hasFailedToSend: Bool {
|
||||||
properties.deliveryStatus == .sendingFailed
|
properties.deliveryStatus == .sendingFailed
|
||||||
}
|
}
|
||||||
@ -74,8 +78,8 @@ extension EventBasedTimelineItemProtocol {
|
|||||||
whiteSpaces += 1
|
whiteSpaces += 1
|
||||||
}
|
}
|
||||||
|
|
||||||
// To account for the extra spacing created by the alert icon
|
// To account for the extra spacing created by the status icon
|
||||||
if hasFailedToSend {
|
if hasStatusIcon {
|
||||||
whiteSpaces += 3
|
whiteSpaces += 3
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,11 +24,14 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
|
|
||||||
/// The Matrix ID of the current user.
|
/// The Matrix ID of the current user.
|
||||||
private let userID: String
|
private let userID: String
|
||||||
|
private let encryptionAuthenticityEnabled: Bool
|
||||||
|
|
||||||
init(userID: String,
|
init(userID: String,
|
||||||
|
encryptionAuthenticityEnabled: Bool,
|
||||||
attributedStringBuilder: AttributedStringBuilderProtocol,
|
attributedStringBuilder: AttributedStringBuilderProtocol,
|
||||||
stateEventStringBuilder: RoomStateEventStringBuilder) {
|
stateEventStringBuilder: RoomStateEventStringBuilder) {
|
||||||
self.userID = userID
|
self.userID = userID
|
||||||
|
self.encryptionAuthenticityEnabled = encryptionAuthenticityEnabled
|
||||||
self.attributedStringBuilder = attributedStringBuilder
|
self.attributedStringBuilder = attributedStringBuilder
|
||||||
self.stateEventStringBuilder = stateEventStringBuilder
|
self.stateEventStringBuilder = stateEventStringBuilder
|
||||||
}
|
}
|
||||||
@ -156,7 +159,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
blurhash: imageInfo.blurhash,
|
blurhash: imageInfo.blurhash,
|
||||||
properties: RoomTimelineItemProperties(reactions: aggregateReactions(eventItemProxy.reactions),
|
properties: RoomTimelineItemProperties(reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildEncryptedTimelineItem(_ eventItemProxy: EventTimelineItemProxy,
|
private func buildEncryptedTimelineItem(_ eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -220,7 +224,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildImageTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
private func buildImageTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -240,7 +245,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildVideoTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
private func buildVideoTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -260,7 +266,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildAudioTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
private func buildAudioTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -280,7 +287,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildVoiceTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
private func buildVoiceTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -300,7 +308,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildFileTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
private func buildFileTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -320,7 +329,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildNoticeTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
private func buildNoticeTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -340,7 +350,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildEmoteTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
private func buildEmoteTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -360,7 +371,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildLocationTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
private func buildLocationTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
||||||
@ -380,7 +392,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// swiftlint:disable:next function_parameter_count
|
// swiftlint:disable:next function_parameter_count
|
||||||
@ -429,7 +442,8 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
properties: RoomTimelineItemProperties(isEdited: edited,
|
properties: RoomTimelineItemProperties(isEdited: edited,
|
||||||
reactions: aggregateReactions(eventItemProxy.reactions),
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
||||||
deliveryStatus: eventItemProxy.deliveryStatus,
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
||||||
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts)))
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
||||||
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
||||||
}
|
}
|
||||||
|
|
||||||
private func buildCallInviteTimelineItem(for eventItemProxy: EventTimelineItemProxy) -> RoomTimelineItemProtocol {
|
private func buildCallInviteTimelineItem(for eventItemProxy: EventTimelineItemProxy) -> RoomTimelineItemProtocol {
|
||||||
@ -488,6 +502,11 @@ struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private func authenticity(_ shieldState: ShieldState?) -> EncryptionAuthenticity? {
|
||||||
|
guard encryptionAuthenticityEnabled else { return nil }
|
||||||
|
return shieldState.flatMap(EncryptionAuthenticity.init)
|
||||||
|
}
|
||||||
|
|
||||||
// MARK: - Message events content
|
// MARK: - Message events content
|
||||||
|
|
||||||
private func buildTextTimelineItemContent(_ messageContent: TextMessageContent) -> TextRoomTimelineItemContent {
|
private func buildTextTimelineItemContent(_ messageContent: TextMessageContent) -> TextRoomTimelineItemContent {
|
||||||
|
@ -636,6 +636,7 @@ class MockScreen: Identifiable {
|
|||||||
let timelineController = RoomTimelineController(roomProxy: roomProxy,
|
let timelineController = RoomTimelineController(roomProxy: roomProxy,
|
||||||
initialFocussedEventID: nil,
|
initialFocussedEventID: nil,
|
||||||
timelineItemFactory: RoomTimelineItemFactory(userID: "@alice:matrix.org",
|
timelineItemFactory: RoomTimelineItemFactory(userID: "@alice:matrix.org",
|
||||||
|
encryptionAuthenticityEnabled: true,
|
||||||
attributedStringBuilder: AttributedStringBuilder(mentionBuilder: MentionBuilder()),
|
attributedStringBuilder: AttributedStringBuilder(mentionBuilder: MentionBuilder()),
|
||||||
stateEventStringBuilder: RoomStateEventStringBuilder(userID: "@alice:matrix.org")),
|
stateEventStringBuilder: RoomStateEventStringBuilder(userID: "@alice:matrix.org")),
|
||||||
appSettings: ServiceLocator.shared.settings)
|
appSettings: ServiceLocator.shared.settings)
|
||||||
|
@ -5,8 +5,8 @@
|
|||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
"location" : "https://github.com/apple/swift-argument-parser",
|
"location" : "https://github.com/apple/swift-argument-parser",
|
||||||
"state" : {
|
"state" : {
|
||||||
"revision" : "0fbc8848e389af3bb55c182bc19ca9d5dc2f255b",
|
"revision" : "41982a3656a71c768319979febd796c6fd111d5c",
|
||||||
"version" : "1.4.0"
|
"version" : "1.5.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -22,8 +22,8 @@
|
|||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
"location" : "https://github.com/jpsim/Yams",
|
"location" : "https://github.com/jpsim/Yams",
|
||||||
"state" : {
|
"state" : {
|
||||||
"revision" : "9234124cff5e22e178988c18d8b95a8ae8007f76",
|
"revision" : "3036ba9d69cf1fd04d433527bc339dc0dc75433d",
|
||||||
"version" : "5.1.2"
|
"version" : "5.1.3"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemBubbledStylerView-iPad-en-GB.Encryption-Indicators.png
(Stored with Git LFS)
Normal file
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemBubbledStylerView-iPad-en-GB.Encryption-Indicators.png
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemBubbledStylerView-iPad-pseudo.Encryption-Indicators.png
(Stored with Git LFS)
Normal file
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemBubbledStylerView-iPad-pseudo.Encryption-Indicators.png
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemBubbledStylerView-iPhone-15-en-GB.Encryption-Indicators.png
(Stored with Git LFS)
Normal file
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemBubbledStylerView-iPhone-15-en-GB.Encryption-Indicators.png
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemBubbledStylerView-iPhone-15-pseudo.Encryption-Indicators.png
(Stored with Git LFS)
Normal file
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemBubbledStylerView-iPhone-15-pseudo.Encryption-Indicators.png
(Stored with Git LFS)
Normal file
Binary file not shown.
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemSendInfoLabel-iPad-en-GB.1.png
(Stored with Git LFS)
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemSendInfoLabel-iPad-en-GB.1.png
(Stored with Git LFS)
Binary file not shown.
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemSendInfoLabel-iPad-pseudo.1.png
(Stored with Git LFS)
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemSendInfoLabel-iPad-pseudo.1.png
(Stored with Git LFS)
Binary file not shown.
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemSendInfoLabel-iPhone-15-en-GB.1.png
(Stored with Git LFS)
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemSendInfoLabel-iPhone-15-en-GB.1.png
(Stored with Git LFS)
Binary file not shown.
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemSendInfoLabel-iPhone-15-pseudo.1.png
(Stored with Git LFS)
BIN
PreviewTests/__Snapshots__/PreviewTests/test_timelineItemSendInfoLabel-iPhone-15-pseudo.1.png
(Stored with Git LFS)
Binary file not shown.
@ -24,6 +24,7 @@ class TimelineItemFactoryTests: XCTestCase {
|
|||||||
let senderUserID = "@bob:matrix.org"
|
let senderUserID = "@bob:matrix.org"
|
||||||
|
|
||||||
let factory = RoomTimelineItemFactory(userID: ownUserID,
|
let factory = RoomTimelineItemFactory(userID: ownUserID,
|
||||||
|
encryptionAuthenticityEnabled: true,
|
||||||
attributedStringBuilder: AttributedStringBuilder(mentionBuilder: MentionBuilder()),
|
attributedStringBuilder: AttributedStringBuilder(mentionBuilder: MentionBuilder()),
|
||||||
stateEventStringBuilder: RoomStateEventStringBuilder(userID: ownUserID))
|
stateEventStringBuilder: RoomStateEventStringBuilder(userID: ownUserID))
|
||||||
|
|
||||||
|
@ -60,7 +60,7 @@ packages:
|
|||||||
# Element/Matrix dependencies
|
# Element/Matrix dependencies
|
||||||
MatrixRustSDK:
|
MatrixRustSDK:
|
||||||
url: https://github.com/element-hq/matrix-rust-components-swift
|
url: https://github.com/element-hq/matrix-rust-components-swift
|
||||||
exactVersion: 1.0.30
|
exactVersion: 1.0.31
|
||||||
# path: ../matrix-rust-sdk
|
# path: ../matrix-rust-sdk
|
||||||
Compound:
|
Compound:
|
||||||
url: https://github.com/element-hq/compound-ios
|
url: https://github.com/element-hq/compound-ios
|
||||||
|
Loading…
x
Reference in New Issue
Block a user