Reworked the SessionCell.Accessory (same interface, builds much quicker)

pull/941/head
Morgan Pretty 1 year ago
parent ea45951f72
commit db633e69de

@ -132,249 +132,250 @@ class EditGroupViewModel: SessionTableViewModel, NavigatableStateHolder, Editabl
isValid: true
)
}
.map { [weak self, dependencies, threadId, userSessionId, selectedIdsSubject] (state: State) -> [SectionModel] in
guard state.isValid else {
return [
SectionModel(
model: .groupInfo,
elements: [
SessionCell.Info(
id: .groupName,
title: SessionCell.TextInfo(
"ERROR_UNABLE_TO_FIND_DATA".localized(),
font: .subtitle,
alignment: .center
),
styling: SessionCell.StyleInfo(
tintColor: .textSecondary,
alignment: .centerHugging,
customPadding: SessionCell.Padding(top: Values.smallSpacing),
backgroundStyle: .noBackground
)
)
]
)
]
}
let isUpdatedGroup: Bool = (((try? SessionId.Prefix(from: threadId)) ?? .group) == .group)
let threadVariant: SessionThread.Variant = (isUpdatedGroup ? .group : .legacyGroup)
let editIcon: UIImage? = UIImage(systemName: "pencil")
let sortedMembers: [WithProfile<GroupMember>] = {
guard !isUpdatedGroup else { return state.members }
// FIXME: Remove this once legacy groups are deprecated
/// In legacy groups there would be both `standard` and `admin` `GroupMember` entries for admins so
/// pre-process the members in order to remove the duplicates
return Array(state.members
.sorted(by: { lhs, rhs in lhs.value.role.rawValue < rhs.value.role.rawValue })
.reduce(into: [:]) { result, next in result[next.profileId] = next }
.values)
}()
.sorted()
.compactMap { [weak self] state -> [SectionModel]? in self?.content(state) }
private func content(_ state: State) -> [SectionModel] {
guard state.isValid else {
return [
SectionModel(
model: .groupInfo,
elements: [
SessionCell.Info(
id: .avatar,
accessory: .profile(
id: threadId,
size: .hero,
threadVariant: (isUpdatedGroup ? .group : .legacyGroup),
displayPictureFilename: state.group.displayPictureFilename,
profile: state.profileFront,
profileIcon: {
guard isUpdatedGroup && dependencies[feature: .updatedGroupsAllowDisplayPicture] else {
return .none
}
// If we already have a display picture then the main profile gets the icon
return (state.group.displayPictureFilename != nil ? .rightPlus : .none)
}(),
additionalProfile: state.profileBack,
additionalProfileIcon: {
guard isUpdatedGroup && dependencies[feature: .updatedGroupsAllowDisplayPicture] else {
return .none
}
// No display picture means the dual-profile so the additionalProfile gets the icon
return .rightPlus
}(),
accessibility: nil
id: .groupName,
title: SessionCell.TextInfo(
"ERROR_UNABLE_TO_FIND_DATA".localized(),
font: .subtitle,
alignment: .center
),
styling: SessionCell.StyleInfo(
tintColor: .textSecondary,
alignment: .centerHugging,
customPadding: SessionCell.Padding(bottom: Values.smallSpacing),
customPadding: SessionCell.Padding(top: Values.smallSpacing),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
label: "Profile picture"
),
onTap: {
)
)
]
)
]
}
let isUpdatedGroup: Bool = (((try? SessionId.Prefix(from: threadId)) ?? .group) == .group)
let editIcon: UIImage? = UIImage(systemName: "pencil")
let sortedMembers: [WithProfile<GroupMember>] = {
guard !isUpdatedGroup else { return state.members }
// FIXME: Remove this once legacy groups are deprecated
/// In legacy groups there would be both `standard` and `admin` `GroupMember` entries for admins so
/// pre-process the members in order to remove the duplicates
return Array(state.members
.sorted(by: { lhs, rhs in lhs.value.role.rawValue < rhs.value.role.rawValue })
.reduce(into: [:]) { result, next in result[next.profileId] = next }
.values)
}()
.sorted()
return [
SectionModel(
model: .groupInfo,
elements: [
SessionCell.Info(
id: .avatar,
accessory: .profile(
id: threadId,
size: .hero,
threadVariant: (isUpdatedGroup ? .group : .legacyGroup),
displayPictureFilename: state.group.displayPictureFilename,
profile: state.profileFront,
profileIcon: {
guard isUpdatedGroup && dependencies[feature: .updatedGroupsAllowDisplayPicture] else {
return .none
}
// If we already have a display picture then the main profile gets the icon
return (state.group.displayPictureFilename != nil ? .rightPlus : .none)
}(),
additionalProfile: state.profileBack,
additionalProfileIcon: {
guard isUpdatedGroup && dependencies[feature: .updatedGroupsAllowDisplayPicture] else {
return
return .none
}
self?.updateDisplayPicture(currentFileName: state.group.displayPictureFilename)
// No display picture means the dual-profile so the additionalProfile gets the icon
return .rightPlus
}(),
accessibility: nil
),
styling: SessionCell.StyleInfo(
alignment: .centerHugging,
customPadding: SessionCell.Padding(bottom: Values.smallSpacing),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
label: "Profile picture"
),
onTap: { [weak self, dependencies] in
guard isUpdatedGroup && dependencies[feature: .updatedGroupsAllowDisplayPicture] else {
return
}
self?.updateDisplayPicture(currentFileName: state.group.displayPictureFilename)
}
),
SessionCell.Info(
id: .groupName,
leadingAccessory: .icon(
editIcon?.withRenderingMode(.alwaysTemplate),
size: .medium,
customTint: .textSecondary
),
SessionCell.Info(
id: .groupName,
leadingAccessory: .icon(
editIcon?.withRenderingMode(.alwaysTemplate),
size: .medium,
customTint: .textSecondary
title: SessionCell.TextInfo(
state.group.name,
font: .titleLarge,
alignment: .center,
editingPlaceholder: "EDIT_GROUP_NAME_PLACEHOLDER".localized()
),
styling: SessionCell.StyleInfo(
alignment: .centerHugging,
customPadding: SessionCell.Padding(
top: Values.smallSpacing,
leading: -((IconSize.medium.size + (Values.smallSpacing * 2)) / 2),
bottom: Values.smallSpacing,
interItem: 0
),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
identifier: "Group name",
label: state.group.name
),
onTap: { [weak self] in
self?.updateGroupNameAndDescription(
isUpdatedGroup: isUpdatedGroup,
currentName: state.group.name,
currentDescription: state.group.groupDescription
)
}
),
((state.group.groupDescription ?? "").isEmpty ? nil :
SessionCell.Info(
id: .groupDescription,
title: SessionCell.TextInfo(
state.group.name,
font: .titleLarge,
(state.group.groupDescription ?? ""),
font: .subtitle,
alignment: .center,
editingPlaceholder: "EDIT_GROUP_NAME_PLACEHOLDER".localized()
editingPlaceholder: "EDIT_GROUP_DESCRIPTION_PLACEHOLDER".localized()
),
styling: SessionCell.StyleInfo(
tintColor: .textSecondary,
alignment: .centerHugging,
customPadding: SessionCell.Padding(
top: Values.smallSpacing,
leading: -((IconSize.medium.size + (Values.smallSpacing * 2)) / 2),
bottom: Values.smallSpacing,
interItem: 0
top: 0,
bottom: Values.smallSpacing
),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
identifier: "Group name",
label: state.group.name
identifier: "Group description",
label: (state.group.groupDescription ?? "")
),
onTap: {
self?.updateGroupNameAndDescription(
isUpdatedGroup: isUpdatedGroup,
currentName: state.group.name,
currentDescription: state.group.groupDescription
)
}
onTap: { [weak self] in
self?.updateGroupNameAndDescription(
isUpdatedGroup: isUpdatedGroup,
currentName: state.group.name,
currentDescription: state.group.groupDescription
)
}
)
)
].compactMap { $0 }
),
SectionModel(
model: .invite,
elements: [
SessionCell.Info(
id: .invite,
leadingAccessory: .icon(UIImage(named: "icon_invite")?.withRenderingMode(.alwaysTemplate)),
title: "GROUP_ACTION_INVITE_CONTACTS".localized(),
accessibility: Accessibility(
identifier: "Invite Contacts",
label: "Invite Contacts"
),
((state.group.groupDescription ?? "").isEmpty ? nil :
SessionCell.Info(
id: .groupDescription,
title: SessionCell.TextInfo(
(state.group.groupDescription ?? ""),
font: .subtitle,
alignment: .center,
editingPlaceholder: "EDIT_GROUP_DESCRIPTION_PLACEHOLDER".localized()
),
styling: SessionCell.StyleInfo(
tintColor: .textSecondary,
alignment: .centerHugging,
customPadding: SessionCell.Padding(
top: 0,
bottom: Values.smallSpacing
),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
identifier: "Group description",
label: (state.group.groupDescription ?? "")
),
onTap: {
self?.updateGroupNameAndDescription(
isUpdatedGroup: isUpdatedGroup,
currentName: state.group.name,
currentDescription: state.group.groupDescription
)
}
)
)
].compactMap { $0 }
),
SectionModel(
model: .invite,
elements: [
onTap: { [weak self] in self?.inviteContacts(currentGroupName: state.group.name) }
)
]
),
SectionModel(
model: .members,
elements: sortedMembers
.map { memberInfo -> SessionCell.Info in
SessionCell.Info(
id: .invite,
leadingAccessory: .icon(UIImage(named: "icon_invite")?.withRenderingMode(.alwaysTemplate)),
title: "GROUP_ACTION_INVITE_CONTACTS".localized(),
accessibility: Accessibility(
identifier: "Invite Contacts",
label: "Invite Contacts"
id: .member(memberInfo.profileId),
leadingAccessory: .profile(
id: memberInfo.profileId,
profile: memberInfo.profile,
profileIcon: memberInfo.value.profileIcon
),
onTap: { self?.inviteContacts(currentGroupName: state.group.name) }
)
]
),
SectionModel(
model: .members,
elements: sortedMembers
.map { memberInfo -> SessionCell.Info in
SessionCell.Info(
id: .member(memberInfo.profileId),
leadingAccessory: .profile(
id: memberInfo.profileId,
profile: memberInfo.profile,
profileIcon: memberInfo.value.profileIcon
),
title: (
memberInfo.profile?.displayName() ??
Profile.truncated(id: memberInfo.profileId, truncating: .middle)
title: (
memberInfo.profile?.displayName() ??
Profile.truncated(id: memberInfo.profileId, truncating: .middle)
),
subtitle: (isUpdatedGroup ? memberInfo.value.statusDescription : nil),
trailingAccessory: {
switch (memberInfo.value.role, memberInfo.value.roleStatus) {
case (.admin, _), (.moderator, _): return nil
case (.standard, .failed), (.standard, .sending):
return .highlightingBackgroundLabel(
title: "context_menu_resend".localized()
)
// Intentionally including the 'pending' state in here as we want admins to
// be able to remove pending members - to resend the admin will have to remove
// and re-add the member
case (.standard, .pending), (.standard, .accepted), (.zombie, _):
return .radio(
isSelected: selectedIdsSubject.value.contains(memberInfo.profileId)
)
}
}(),
styling: SessionCell.StyleInfo(
subtitleTintColor: (isUpdatedGroup ? memberInfo.value.statusDescriptionColor : nil),
allowedSeparators: [],
customPadding: SessionCell.Padding(
top: Values.smallSpacing,
bottom: Values.smallSpacing
),
subtitle: (isUpdatedGroup ? memberInfo.value.statusDescription : nil),
trailingAccessory: {
switch (memberInfo.value.role, memberInfo.value.roleStatus) {
case (.admin, _), (.moderator, _): return nil
case (.standard, .failed), (.standard, .sending):
return .highlightingBackgroundLabel(
title: "context_menu_resend".localized()
)
backgroundStyle: .noBackgroundEdgeToEdge
),
onTap: { [weak self, selectedIdsSubject] in
switch (memberInfo.value.role, memberInfo.value.roleStatus) {
case (.moderator, _): return
case (.admin, _):
self?.showToast(
text: "EDIT_GROUP_MEMBERS_ERROR_REMOVE_ADMIN".localized(),
backgroundColor: .backgroundSecondary
)
// Intentionally including the 'pending' state in here as we want admins to
// be able to remove pending members - to resend the admin will have to remove
// and re-add the member
case (.standard, .pending), (.standard, .accepted), (.zombie, _):
return .radio(
isSelected: selectedIdsSubject.value.contains(memberInfo.profileId)
)
}
}(),
styling: SessionCell.StyleInfo(
subtitleTintColor: (isUpdatedGroup ? memberInfo.value.statusDescriptionColor : nil),
allowedSeparators: [],
customPadding: SessionCell.Padding(
top: Values.smallSpacing,
bottom: Values.smallSpacing
),
backgroundStyle: .noBackgroundEdgeToEdge
),
onTap: {
switch (memberInfo.value.role, memberInfo.value.roleStatus) {
case (.moderator, _): return
case (.admin, _):
self?.showToast(
text: "EDIT_GROUP_MEMBERS_ERROR_REMOVE_ADMIN".localized(),
backgroundColor: .backgroundSecondary
)
case (.standard, .failed), (.standard, .sending):
self?.resendInvitation(memberId: memberInfo.profileId)
case (.standard, .pending), (.standard, .accepted), (.zombie, _):
if !selectedIdsSubject.value.contains(memberInfo.profileId) {
selectedIdsSubject.send(selectedIdsSubject.value.inserting(memberInfo.profileId))
}
else {
selectedIdsSubject.send(selectedIdsSubject.value.removing(memberInfo.profileId))
}
// Force the table data to be refreshed (the database wouldn't
// have been changed)
self?.forceRefresh(type: .postDatabaseQuery)
}
case (.standard, .failed), (.standard, .sending):
self?.resendInvitation(memberId: memberInfo.profileId)
case (.standard, .pending), (.standard, .accepted), (.zombie, _):
if !selectedIdsSubject.value.contains(memberInfo.profileId) {
selectedIdsSubject.send(selectedIdsSubject.value.inserting(memberInfo.profileId))
}
else {
selectedIdsSubject.send(selectedIdsSubject.value.removing(memberInfo.profileId))
}
// Force the table data to be refreshed (the database wouldn't
// have been changed)
self?.forceRefresh(type: .postDatabaseQuery)
}
)
}
)
]
}
}
)
}
)
]
}
lazy var footerButtonInfo: AnyPublisher<SessionButton.Info?, Never> = selectedIdsSubject
.prepend([])

@ -70,11 +70,8 @@ class ConversationSettingsViewModel: SessionTableViewModel, NavigatableStateHold
title: "CONVERSATION_SETTINGS_MESSAGE_TRIMMING_TITLE".localized(),
subtitle: "CONVERSATION_SETTINGS_MESSAGE_TRIMMING_DESCRIPTION".localized(),
trailingAccessory: .toggle(
.boolValue(
key: .trimOpenGroupMessagesOlderThanSixMonths,
value: current.trimOpenGroupMessagesOlderThanSixMonths,
oldValue: (previous ?? current).trimOpenGroupMessagesOlderThanSixMonths
)
current.trimOpenGroupMessagesOlderThanSixMonths,
oldValue: previous?.trimOpenGroupMessagesOlderThanSixMonths
),
onTap: {
dependencies[singleton: .storage].write { db in
@ -92,11 +89,8 @@ class ConversationSettingsViewModel: SessionTableViewModel, NavigatableStateHold
title: "CONVERSATION_SETTINGS_AUDIO_MESSAGES_AUTOPLAY_TITLE".localized(),
subtitle: "CONVERSATION_SETTINGS_AUDIO_MESSAGES_AUTOPLAY_DESCRIPTION".localized(),
trailingAccessory: .toggle(
.boolValue(
key: .shouldAutoPlayConsecutiveAudioMessages,
value: current.shouldAutoPlayConsecutiveAudioMessages,
oldValue: (previous ?? current).shouldAutoPlayConsecutiveAudioMessages
)
current.shouldAutoPlayConsecutiveAudioMessages,
oldValue: previous?.shouldAutoPlayConsecutiveAudioMessages
),
onTap: {
dependencies[singleton: .storage].write { db in

@ -112,302 +112,291 @@ class DeveloperSettingsViewModel: SessionTableViewModel, NavigatableStateHolder,
updatedGroupsAllowPromotions: dependencies[feature: .updatedGroupsAllowPromotions]
)
}
.mapWithPrevious { [weak self, dependencies] previous, current -> [SectionModel] in
return [
SectionModel(
model: .developerMode,
elements: [
SessionCell.Info(
id: .developerMode,
title: "Developer Mode",
subtitle: """
Grants access to this screen.
Disabling this setting will:
Reset all the below settings to default (removing data as described below)
Revoke access to this screen unless Developer Mode is re-enabled
""",
trailingAccessory: .toggle(
.boolValue(current.developerMode, oldValue: (previous ?? current).developerMode)
),
onTap: {
guard current.developerMode else { return }
self?.disableDeveloperMode()
}
)
]
),
SectionModel(
model: .network,
elements: [
SessionCell.Info(
id: .serviceNetwork,
title: "Environment",
subtitle: """
The environment used for sending requests and storing messages.
<b>Warning:</b>
Changing this setting will result in all conversation and snode data being cleared and any pending network requests being cancelled.
""",
trailingAccessory: .dropDown(
.dynamicString { current.serviceNetwork.title }
),
onTap: {
self?.transitionToScreen(
SessionTableViewController(
viewModel: SessionListViewModel<ServiceNetwork>(
title: "Environment",
options: ServiceNetwork.allCases,
behaviour: .autoDismiss(
initialSelection: current.serviceNetwork,
onOptionSelected: self?.updateServiceNetwork
),
using: dependencies
)
)
)
}
.compactMapWithPrevious { [weak self] prev, current -> [SectionModel]? in self?.content(prev, current) }
private func content(_ previous: State?, _ current: State) -> [SectionModel] {
return [
SectionModel(
model: .developerMode,
elements: [
SessionCell.Info(
id: .developerMode,
title: "Developer Mode",
subtitle: """
Grants access to this screen.
Disabling this setting will:
Reset all the below settings to default (removing data as described below)
Revoke access to this screen unless Developer Mode is re-enabled
""",
trailingAccessory: .toggle(
current.developerMode,
oldValue: previous?.developerMode
),
SessionCell.Info(
id: .networkLayer,
title: "Routing",
subtitle: """
The network layer which all network traffic should be routed through.
We do support sending network traffic through multiple network layers, if multiple layers are selected then requests will wait for a response from all layers before completing with the first successful response.
onTap: { [weak self] in
guard current.developerMode else { return }
<b>Warning:</b>
Different network layers offer different levels of privacy, make sure to read the description of the network layers before making a selection.
""",
trailingAccessory: .dropDown(
.dynamicString { current.networkLayer.title }
),
onTap: {
self?.transitionToScreen(
SessionTableViewController(
viewModel: SessionListViewModel<Network.Layers>(
title: "Routing",
options: Network.Layers.allCases,
behaviour: .singleSelect(
initialSelection: current.networkLayer,
onSaved: self?.updateNetworkLayers
),
using: dependencies
)
self?.disableDeveloperMode()
}
)
]
),
SectionModel(
model: .network,
elements: [
SessionCell.Info(
id: .serviceNetwork,
title: "Environment",
subtitle: """
The environment used for sending requests and storing messages.
<b>Warning:</b>
Changing this setting will result in all conversation and snode data being cleared and any pending network requests being cancelled.
""",
trailingAccessory: .dropDown { current.serviceNetwork.title },
onTap: { [weak self, dependencies] in
self?.transitionToScreen(
SessionTableViewController(
viewModel: SessionListViewModel<ServiceNetwork>(
title: "Environment",
options: ServiceNetwork.allCases,
behaviour: .autoDismiss(
initialSelection: current.serviceNetwork,
onOptionSelected: self?.updateServiceNetwork
),
using: dependencies
)
)
}
)
]
),
SectionModel(
model: .disappearingMessages,
elements: [
SessionCell.Info(
id: .debugDisappearingMessageDurations,
title: "Debug Durations",
subtitle: """
Adds 10 and 60 second durations for Disappearing Message settings.
These should only be used for debugging purposes and can result in odd behaviours.
""",
trailingAccessory: .toggle(
.boolValue(
current.debugDisappearingMessageDurations,
oldValue: (previous ?? current).debugDisappearingMessageDurations
)
),
onTap: {
self?.updateFlag(
for: .debugDisappearingMessageDurations,
to: !current.debugDisappearingMessageDurations
)
}
),
SessionCell.Info(
id: .networkLayer,
title: "Routing",
subtitle: """
The network layer which all network traffic should be routed through.
We do support sending network traffic through multiple network layers, if multiple layers are selected then requests will wait for a response from all layers before completing with the first successful response.
<b>Warning:</b>
Different network layers offer different levels of privacy, make sure to read the description of the network layers before making a selection.
""",
trailingAccessory: .dropDown { current.networkLayer.title },
onTap: { [weak self, dependencies] in
self?.transitionToScreen(
SessionTableViewController(
viewModel: SessionListViewModel<Network.Layers>(
title: "Routing",
options: Network.Layers.allCases,
behaviour: .singleSelect(
initialSelection: current.networkLayer,
onSaved: self?.updateNetworkLayers
),
using: dependencies
)
)
}
)
}
)
]
),
SectionModel(
model: .disappearingMessages,
elements: [
SessionCell.Info(
id: .debugDisappearingMessageDurations,
title: "Debug Durations",
subtitle: """
Adds 10 and 60 second durations for Disappearing Message settings.
These should only be used for debugging purposes and can result in odd behaviours.
""",
trailingAccessory: .toggle(
current.debugDisappearingMessageDurations,
oldValue: previous?.debugDisappearingMessageDurations
),
SessionCell.Info(
id: .updatedDisappearingMessages,
title: "Use Updated Disappearing Messages",
subtitle: """
Controls whether legacy or updated disappearing messages should be used.
""",
trailingAccessory: .toggle(
.boolValue(
current.updatedDisappearingMessages,
oldValue: (previous ?? current).updatedDisappearingMessages
)
),
onTap: {
self?.updateFlag(
for: .updatedDisappearingMessages,
to: !current.updatedDisappearingMessages
)
}
)
]
),
SectionModel(
model: .groups,
elements: [
SessionCell.Info(
id: .updatedGroups,
title: "Create Updated Groups",
subtitle: """
Controls whether newly created groups are updated or legacy groups.
""",
trailingAccessory: .toggle(
.boolValue(current.updatedGroups, oldValue: (previous ?? current).updatedGroups)
),
onTap: { self?.updateFlag(for: .updatedGroups, to: !current.updatedGroups) }
onTap: { [weak self] in
self?.updateFlag(
for: .debugDisappearingMessageDurations,
to: !current.debugDisappearingMessageDurations
)
}
),
SessionCell.Info(
id: .updatedDisappearingMessages,
title: "Use Updated Disappearing Messages",
subtitle: """
Controls whether legacy or updated disappearing messages should be used.
""",
trailingAccessory: .toggle(
current.updatedDisappearingMessages,
oldValue: previous?.updatedDisappearingMessages
),
SessionCell.Info(
id: .updatedGroupsDisableAutoApprove,
title: "Disable Auto Approve",
subtitle: """
Prevents a group from automatically getting approved if the admin is already approved.
<b>Note:</b> The default behaviour is to automatically approve new groups if the admin that sent the invitation is an approved contact.
""",
trailingAccessory: .toggle(
.boolValue(
current.updatedGroupsDisableAutoApprove,
oldValue: (previous ?? current).updatedGroupsDisableAutoApprove
)
),
onTap: {
self?.updateFlag(
for: .updatedGroupsDisableAutoApprove,
to: !current.updatedGroupsDisableAutoApprove
)
}
onTap: { [weak self] in
self?.updateFlag(
for: .updatedDisappearingMessages,
to: !current.updatedDisappearingMessages
)
}
)
]
),
SectionModel(
model: .groups,
elements: [
SessionCell.Info(
id: .updatedGroups,
title: "Create Updated Groups",
subtitle: """
Controls whether newly created groups are updated or legacy groups.
""",
trailingAccessory: .toggle(
current.updatedGroups,
oldValue: previous?.updatedGroups
),
SessionCell.Info(
id: .updatedGroupsRemoveMessagesOnKick,
title: "Remove Messages on Kick",
subtitle: """
Controls whether a group members messages should be removed when they are kicked from an updated group.
<b>Note:</b> In a future release we will offer this as an option when removing members but for the initial release it can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
.boolValue(
current.updatedGroupsRemoveMessagesOnKick,
oldValue: (previous ?? current).updatedGroupsRemoveMessagesOnKick
)
),
onTap: {
self?.updateFlag(
for: .updatedGroupsRemoveMessagesOnKick,
to: !current.updatedGroupsRemoveMessagesOnKick
)
}
onTap: { [weak self] in
self?.updateFlag(
for: .updatedGroups,
to: !current.updatedGroups
)
}
),
SessionCell.Info(
id: .updatedGroupsDisableAutoApprove,
title: "Disable Auto Approve",
subtitle: """
Prevents a group from automatically getting approved if the admin is already approved.
<b>Note:</b> The default behaviour is to automatically approve new groups if the admin that sent the invitation is an approved contact.
""",
trailingAccessory: .toggle(
current.updatedGroupsDisableAutoApprove,
oldValue: previous?.updatedGroupsDisableAutoApprove
),
SessionCell.Info(
id: .updatedGroupsAllowHistoricAccessOnInvite,
title: "Allow Historic Message Access",
subtitle: """
Controls whether members should be granted access to historic messages when invited to an updated group.
<b>Note:</b> In a future release we will offer this as an option when inviting members but for the initial release it can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
.boolValue(
current.updatedGroupsAllowHistoricAccessOnInvite,
oldValue: (previous ?? current).updatedGroupsAllowHistoricAccessOnInvite
)
),
onTap: {
self?.updateFlag(
for: .updatedGroupsAllowHistoricAccessOnInvite,
to: !current.updatedGroupsAllowHistoricAccessOnInvite
)
}
onTap: { [weak self] in
self?.updateFlag(
for: .updatedGroupsDisableAutoApprove,
to: !current.updatedGroupsDisableAutoApprove
)
}
),
SessionCell.Info(
id: .updatedGroupsRemoveMessagesOnKick,
title: "Remove Messages on Kick",
subtitle: """
Controls whether a group members messages should be removed when they are kicked from an updated group.
<b>Note:</b> In a future release we will offer this as an option when removing members but for the initial release it can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
current.updatedGroupsRemoveMessagesOnKick,
oldValue: previous?.updatedGroupsRemoveMessagesOnKick
),
SessionCell.Info(
id: .updatedGroupsAllowDisplayPicture,
title: "Custom Display Pictures",
subtitle: """
Controls whether the UI allows group admins to set a custom display picture for a group.
<b>Note:</b> In a future release we will offer this functionality but for the initial release it may not be fully supported across platforms so can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
.boolValue(
current.updatedGroupsAllowDisplayPicture,
oldValue: (previous ?? current).updatedGroupsAllowDisplayPicture
)
),
onTap: {
self?.updateFlag(
for: .updatedGroupsAllowDisplayPicture,
to: !current.updatedGroupsAllowDisplayPicture
)
}
onTap: { [weak self] in
self?.updateFlag(
for: .updatedGroupsRemoveMessagesOnKick,
to: !current.updatedGroupsRemoveMessagesOnKick
)
}
),
SessionCell.Info(
id: .updatedGroupsAllowHistoricAccessOnInvite,
title: "Allow Historic Message Access",
subtitle: """
Controls whether members should be granted access to historic messages when invited to an updated group.
<b>Note:</b> In a future release we will offer this as an option when inviting members but for the initial release it can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
current.updatedGroupsAllowHistoricAccessOnInvite,
oldValue: previous?.updatedGroupsAllowHistoricAccessOnInvite
),
SessionCell.Info(
id: .updatedGroupsAllowDescriptionEditing,
title: "Edit Group Descriptions",
subtitle: """
Controls whether the UI allows group admins to modify the descriptions of updated groups.
<b>Note:</b> In a future release we will offer this functionality but for the initial release it may not be fully supported across platforms so can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
.boolValue(
current.updatedGroupsAllowDescriptionEditing,
oldValue: (previous ?? current).updatedGroupsAllowDescriptionEditing
)
),
onTap: {
self?.updateFlag(
for: .updatedGroupsAllowDescriptionEditing,
to: !current.updatedGroupsAllowDescriptionEditing
)
}
onTap: { [weak self] in
self?.updateFlag(
for: .updatedGroupsAllowHistoricAccessOnInvite,
to: !current.updatedGroupsAllowHistoricAccessOnInvite
)
}
),
SessionCell.Info(
id: .updatedGroupsAllowDisplayPicture,
title: "Custom Display Pictures",
subtitle: """
Controls whether the UI allows group admins to set a custom display picture for a group.
<b>Note:</b> In a future release we will offer this functionality but for the initial release it may not be fully supported across platforms so can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
current.updatedGroupsAllowDisplayPicture,
oldValue: previous?.updatedGroupsAllowDisplayPicture
),
SessionCell.Info(
id: .updatedGroupsAllowPromotions,
title: "Allow Group Promotions",
subtitle: """
Controls whether the UI allows group admins promote other group members to admin within an updated group.
<b>Note:</b> In a future release we will offer this functionality but for the initial release it may not be fully supported across platforms so can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
.boolValue(
current.updatedGroupsAllowPromotions,
oldValue: (previous ?? current).updatedGroupsAllowPromotions
)
),
onTap: {
self?.updateFlag(
for: .updatedGroupsAllowPromotions,
to: !current.updatedGroupsAllowPromotions
)
}
)
]
),
SectionModel(
model: .database,
elements: [
SessionCell.Info(
id: .exportDatabase,
title: "Export Database",
trailingAccessory: .icon(
UIImage(systemName: "square.and.arrow.up.trianglebadge.exclamationmark")?
.withRenderingMode(.alwaysTemplate),
size: .small
),
styling: SessionCell.StyleInfo(
tintColor: .danger
),
onTapView: { [weak self] view in self?.exportDatabase(view) }
)
]
)
]
}
onTap: { [weak self] in
self?.updateFlag(
for: .updatedGroupsAllowDisplayPicture,
to: !current.updatedGroupsAllowDisplayPicture
)
}
),
SessionCell.Info(
id: .updatedGroupsAllowDescriptionEditing,
title: "Edit Group Descriptions",
subtitle: """
Controls whether the UI allows group admins to modify the descriptions of updated groups.
<b>Note:</b> In a future release we will offer this functionality but for the initial release it may not be fully supported across platforms so can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
current.updatedGroupsAllowDescriptionEditing,
oldValue: previous?.updatedGroupsAllowDescriptionEditing
),
onTap: { [weak self] in
self?.updateFlag(
for: .updatedGroupsAllowDescriptionEditing,
to: !current.updatedGroupsAllowDescriptionEditing
)
}
),
SessionCell.Info(
id: .updatedGroupsAllowPromotions,
title: "Allow Group Promotions",
subtitle: """
Controls whether the UI allows group admins promote other group members to admin within an updated group.
<b>Note:</b> In a future release we will offer this functionality but for the initial release it may not be fully supported across platforms so can be controlled via this flag for testing purposes.
""",
trailingAccessory: .toggle(
current.updatedGroupsAllowPromotions,
oldValue: previous?.updatedGroupsAllowPromotions
),
onTap: { [weak self] in
self?.updateFlag(
for: .updatedGroupsAllowPromotions,
to: !current.updatedGroupsAllowPromotions
)
}
)
]
),
SectionModel(
model: .database,
elements: [
SessionCell.Info(
id: .exportDatabase,
title: "Export Database",
trailingAccessory: .icon(
UIImage(systemName: "square.and.arrow.up.trianglebadge.exclamationmark")?
.withRenderingMode(.alwaysTemplate),
size: .small
),
styling: SessionCell.StyleInfo(
tintColor: .danger
),
onTapView: { [weak self] view in self?.exportDatabase(view) }
)
]
)
]
}
// MARK: - Functions

@ -90,10 +90,8 @@ class NotificationSettingsViewModel: SessionTableViewModel, NavigatableStateHold
title: "NOTIFICATIONS_STRATEGY_FAST_MODE_TITLE".localized(),
subtitle: "NOTIFICATIONS_STRATEGY_FAST_MODE_DESCRIPTION".localized(),
trailingAccessory: .toggle(
.boolValue(
current.isUsingFullAPNs,
oldValue: (previous ?? current).isUsingFullAPNs
)
current.isUsingFullAPNs,
oldValue: previous?.isUsingFullAPNs
),
styling: SessionCell.StyleInfo(
allowedSeparators: [.top],
@ -128,9 +126,7 @@ class NotificationSettingsViewModel: SessionTableViewModel, NavigatableStateHold
SessionCell.Info(
id: .styleSound,
title: "NOTIFICATIONS_STYLE_SOUND_TITLE".localized(),
trailingAccessory: .dropDown(
.dynamicString { current.notificationSound.displayName }
),
trailingAccessory: .dropDown { current.notificationSound.displayName },
onTap: { [weak self] in
self?.transitionToScreen(
SessionTableViewController(viewModel: NotificationSoundViewModel())
@ -141,11 +137,8 @@ class NotificationSettingsViewModel: SessionTableViewModel, NavigatableStateHold
id: .styleSoundWhenAppIsOpen,
title: "NOTIFICATIONS_STYLE_SOUND_WHEN_OPEN_TITLE".localized(),
trailingAccessory: .toggle(
.boolValue(
key: .playNotificationSoundInForeground,
value: current.playNotificationSoundInForeground,
oldValue: (previous ?? current).playNotificationSoundInForeground
)
current.playNotificationSoundInForeground,
oldValue: previous?.playNotificationSoundInForeground
),
onTap: {
dependencies[singleton: .storage].write { db in
@ -162,9 +155,7 @@ class NotificationSettingsViewModel: SessionTableViewModel, NavigatableStateHold
id: .content,
title: "NOTIFICATIONS_STYLE_CONTENT_TITLE".localized(),
subtitle: "NOTIFICATIONS_STYLE_CONTENT_DESCRIPTION".localized(),
trailingAccessory: .dropDown(
.dynamicString { current.previewType.name }
),
trailingAccessory: .dropDown { current.previewType.name },
onTap: { [weak self] in
self?.transitionToScreen(
SessionTableViewController(viewModel: NotificationContentViewModel())

@ -110,11 +110,8 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
title: "PRIVACY_SCREEN_SECURITY_LOCK_SESSION_TITLE".localized(),
subtitle: "PRIVACY_SCREEN_SECURITY_LOCK_SESSION_DESCRIPTION".localized(),
trailingAccessory: .toggle(
.boolValue(
key: .isScreenLockEnabled,
value: current.isScreenLockEnabled,
oldValue: (previous ?? current).isScreenLockEnabled
)
current.isScreenLockEnabled,
oldValue: previous?.isScreenLockEnabled
),
onTap: { [weak self] in
// Make sure the device has a passcode set before allowing screen lock to
@ -152,11 +149,8 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
title: "PRIVACY_SCREEN_MESSAGE_REQUESTS_COMMUNITY_TITLE".localized(),
subtitle: "PRIVACY_SCREEN_MESSAGE_REQUESTS_COMMUNITY_DESCRIPTION".localized(),
trailingAccessory: .toggle(
.boolValue(
key: .checkForCommunityMessageRequests,
value: current.checkForCommunityMessageRequests,
oldValue: (previous ?? current).checkForCommunityMessageRequests
)
current.checkForCommunityMessageRequests,
oldValue: previous?.checkForCommunityMessageRequests
),
onTap: { [weak self] in
dependencies[singleton: .storage].write { db in
@ -178,11 +172,8 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
title: "PRIVACY_READ_RECEIPTS_TITLE".localized(),
subtitle: "PRIVACY_READ_RECEIPTS_DESCRIPTION".localized(),
trailingAccessory: .toggle(
.boolValue(
key: .areReadReceiptsEnabled,
value: current.areReadReceiptsEnabled,
oldValue: (previous ?? current).areReadReceiptsEnabled
)
current.areReadReceiptsEnabled,
oldValue: previous?.areReadReceiptsEnabled
),
onTap: {
dependencies[singleton: .storage].write { db in
@ -236,11 +227,8 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
}
),
trailingAccessory: .toggle(
.boolValue(
key: .typingIndicatorsEnabled,
value: current.typingIndicatorsEnabled,
oldValue: (previous ?? current).typingIndicatorsEnabled
)
current.typingIndicatorsEnabled,
oldValue: previous?.typingIndicatorsEnabled
),
onTap: {
dependencies[singleton: .storage].write { db in
@ -262,11 +250,8 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
title: "PRIVACY_LINK_PREVIEWS_TITLE".localized(),
subtitle: "PRIVACY_LINK_PREVIEWS_DESCRIPTION".localized(),
trailingAccessory: .toggle(
.boolValue(
key: .areLinkPreviewsEnabled,
value: current.areLinkPreviewsEnabled,
oldValue: (previous ?? current).areLinkPreviewsEnabled
)
current.areLinkPreviewsEnabled,
oldValue: previous?.areLinkPreviewsEnabled
),
onTap: {
dependencies[singleton: .storage].write { db in
@ -288,11 +273,8 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
title: "PRIVACY_CALLS_TITLE".localized(),
subtitle: "PRIVACY_CALLS_DESCRIPTION".localized(),
trailingAccessory: .toggle(
.boolValue(
key: .areCallsEnabled,
value: current.areCallsEnabled,
oldValue: (previous ?? current).areCallsEnabled
)
current.areCallsEnabled,
oldValue: previous?.areCallsEnabled
),
accessibility: Accessibility(
label: "Allow voice and video calls"

@ -230,275 +230,277 @@ class SettingsViewModel: SessionTableViewModel, NavigationItemSource, Navigatabl
developerModeEnabled: db[.developerModeEnabled]
)
}
.map { [weak self, dependencies] state -> [SectionModel] in
return [
SectionModel(
model: .profileInfo,
elements: [
SessionCell.Info(
id: .avatar,
accessory: .profile(
id: state.profile.id,
size: .hero,
profile: state.profile,
profileIcon: (dependencies[feature: .serviceNetwork] == .mainnet ? .none :
.letter("T") // stringlint:disable
)
),
styling: SessionCell.StyleInfo(
alignment: .centerHugging,
customPadding: SessionCell.Padding(bottom: Values.smallSpacing),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
label: "Profile picture"
),
onTap: {
self?.updateProfilePicture(currentFileName: state.profile.profilePictureFileName)
}
),
SessionCell.Info(
id: .profileName,
title: SessionCell.TextInfo(
state.profile.displayName(),
font: .titleLarge,
alignment: .center,
interaction: .editable
),
styling: SessionCell.StyleInfo(
alignment: .centerHugging,
customPadding: SessionCell.Padding(top: Values.smallSpacing),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
identifier: "Username",
label: state.profile.displayName()
),
onTap: { self?.setIsEditing(true) }
)
]
),
SectionModel(
model: .sessionId,
elements: [
SessionCell.Info(
id: .sessionId,
title: SessionCell.TextInfo(
state.profile.id,
font: .monoLarge,
alignment: .center,
interaction: .copy
),
styling: SessionCell.StyleInfo(
customPadding: SessionCell.Padding(bottom: Values.smallSpacing),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
identifier: "Session ID",
label: state.profile.id
.compactMap { [weak self] state -> [SectionModel]? in self?.content(state) }
private func content(_ state: State) -> [SectionModel] {
return [
SectionModel(
model: .profileInfo,
elements: [
SessionCell.Info(
id: .avatar,
accessory: .profile(
id: state.profile.id,
size: .hero,
profile: state.profile,
profileIcon: (dependencies[feature: .serviceNetwork] == .mainnet ? .none :
.letter("T") // stringlint:disable
)
),
SessionCell.Info(
id: .idActions,
leadingAccessory: .button(
style: .bordered,
title: "copy".localized(),
run: { button in
self?.copySessionId(state.profile.id, button: button)
}
),
trailingAccessory: .button(
style: .bordered,
title: "share".localized(),
run: { _ in
self?.shareSessionId(state.profile.id)
}
),
styling: SessionCell.StyleInfo(
customPadding: SessionCell.Padding(
top: Values.smallSpacing,
leading: 0,
trailing: 0
),
backgroundStyle: .noBackground
)
)
]
),
SectionModel(
model: .menus,
elements: [
SessionCell.Info(
id: .path,
leadingAccessory: .customView(hashValue: "PathStatusView") { // stringlint:disable
// Need to ensure this view is the same size as the icons so
// wrap it in a larger view
let result: UIView = UIView()
let pathView: PathStatusView = PathStatusView(size: .large)
result.addSubview(pathView)
result.set(.width, to: IconSize.medium.size)
result.set(.height, to: IconSize.medium.size)
pathView.center(in: result)
return result
},
title: "vc_path_title".localized(),
onTap: { self?.transitionToScreen(PathVC(using: dependencies)) }
styling: SessionCell.StyleInfo(
alignment: .centerHugging,
customPadding: SessionCell.Padding(bottom: Values.smallSpacing),
backgroundStyle: .noBackground
),
SessionCell.Info(
id: .privacy,
leadingAccessory: .icon(
UIImage(named: "icon_privacy")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_privacy_button_title".localized(),
onTap: {
self?.transitionToScreen(
SessionTableViewController(viewModel: PrivacySettingsViewModel(using: dependencies))
)
}
accessibility: Accessibility(
label: "Profile picture"
),
SessionCell.Info(
id: .notifications,
leadingAccessory: .icon(
UIImage(named: "icon_speaker")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_notifications_button_title".localized(),
onTap: {
self?.transitionToScreen(
SessionTableViewController(viewModel: NotificationSettingsViewModel(using: dependencies))
)
}
onTap: { [weak self] in
self?.updateProfilePicture(currentFileName: state.profile.profilePictureFileName)
}
),
SessionCell.Info(
id: .profileName,
title: SessionCell.TextInfo(
state.profile.displayName(),
font: .titleLarge,
alignment: .center,
interaction: .editable
),
SessionCell.Info(
id: .conversations,
leadingAccessory: .icon(
UIImage(named: "icon_msg")?
.withRenderingMode(.alwaysTemplate)
),
title: "CONVERSATION_SETTINGS_TITLE".localized(),
onTap: {
self?.transitionToScreen(
SessionTableViewController(viewModel: ConversationSettingsViewModel(using: dependencies))
)
}
styling: SessionCell.StyleInfo(
alignment: .centerHugging,
customPadding: SessionCell.Padding(top: Values.smallSpacing),
backgroundStyle: .noBackground
),
SessionCell.Info(
id: .messageRequests,
leadingAccessory: .icon(
UIImage(named: "icon_msg_req")?
.withRenderingMode(.alwaysTemplate)
),
title: "MESSAGE_REQUESTS_TITLE".localized(),
onTap: {
self?.transitionToScreen(
SessionTableViewController(viewModel: MessageRequestsViewModel(using: dependencies))
)
}
accessibility: Accessibility(
identifier: "Username",
label: state.profile.displayName()
),
SessionCell.Info(
id: .appearance,
leadingAccessory: .icon(
UIImage(named: "icon_apperance")?
.withRenderingMode(.alwaysTemplate)
),
title: "APPEARANCE_TITLE".localized(),
onTap: {
self?.transitionToScreen(AppearanceViewController(using: dependencies))
}
onTap: { [weak self] in self?.setIsEditing(true) }
)
]
),
SectionModel(
model: .sessionId,
elements: [
SessionCell.Info(
id: .sessionId,
title: SessionCell.TextInfo(
state.profile.id,
font: .monoLarge,
alignment: .center,
interaction: .copy
),
SessionCell.Info(
id: .inviteAFriend,
leadingAccessory: .icon(
UIImage(named: "icon_invite")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_invite_a_friend_button_title".localized(),
onTap: {
let invitation: String = "Hey, I've been using Session to chat with complete privacy and security. Come join me! Download it at https://getsession.org/. My Session ID is \(state.profile.id) !"
self?.transitionToScreen(
UIActivityViewController(
activityItems: [ invitation ],
applicationActivities: nil
),
transitionType: .present
)
styling: SessionCell.StyleInfo(
customPadding: SessionCell.Padding(bottom: Values.smallSpacing),
backgroundStyle: .noBackground
),
accessibility: Accessibility(
identifier: "Session ID",
label: state.profile.id
)
),
SessionCell.Info(
id: .idActions,
leadingAccessory: .button(
style: .bordered,
title: "copy".localized(),
run: { [weak self] button in
self?.copySessionId(state.profile.id, button: button)
}
),
SessionCell.Info(
id: .recoveryPhrase,
leadingAccessory: .icon(
UIImage(named: "icon_recovery")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_recovery_phrase_button_title".localized(),
onTap: {
let targetViewController: UIViewController = {
if let modal: SeedModal = try? SeedModal() {
return modal
}
return ConfirmationModal(
info: ConfirmationModal.Info(
title: "ALERT_ERROR_TITLE".localized(),
body: .text("LOAD_RECOVERY_PASSWORD_ERROR".localized()),
cancelTitle: "BUTTON_OK".localized(),
cancelStyle: .alert_text
)
)
}()
self?.transitionToScreen(targetViewController, transitionType: .present)
trailingAccessory: .button(
style: .bordered,
title: "share".localized(),
run: { [weak self] _ in
self?.shareSessionId(state.profile.id)
}
),
SessionCell.Info(
id: .help,
leadingAccessory: .icon(
UIImage(named: "icon_help")?
.withRenderingMode(.alwaysTemplate)
styling: SessionCell.StyleInfo(
customPadding: SessionCell.Padding(
top: Values.smallSpacing,
leading: 0,
trailing: 0
),
title: "HELP_TITLE".localized(),
onTap: {
self?.transitionToScreen(
SessionTableViewController(viewModel: HelpViewModel(using: dependencies))
)
}
backgroundStyle: .noBackground
)
)
]
),
SectionModel(
model: .menus,
elements: [
SessionCell.Info(
id: .path,
leadingAccessory: .customView(uniqueId: "PathStatusView") { // stringlint:disable
// Need to ensure this view is the same size as the icons so
// wrap it in a larger view
let result: UIView = UIView()
let pathView: PathStatusView = PathStatusView(size: .large)
result.addSubview(pathView)
result.set(.width, to: IconSize.medium.size)
result.set(.height, to: IconSize.medium.size)
pathView.center(in: result)
return result
},
title: "vc_path_title".localized(),
onTap: { [weak self, dependencies] in self?.transitionToScreen(PathVC(using: dependencies)) }
),
SessionCell.Info(
id: .privacy,
leadingAccessory: .icon(
UIImage(named: "icon_privacy")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_privacy_button_title".localized(),
onTap: { [weak self, dependencies] in
self?.transitionToScreen(
SessionTableViewController(viewModel: PrivacySettingsViewModel(using: dependencies))
)
}
),
SessionCell.Info(
id: .notifications,
leadingAccessory: .icon(
UIImage(named: "icon_speaker")?
.withRenderingMode(.alwaysTemplate)
),
(!state.developerModeEnabled ? nil :
SessionCell.Info(
id: .developerSettings,
leadingAccessory: .icon(
UIImage(systemName: "wrench.and.screwdriver")?
.withRenderingMode(.alwaysTemplate)
title: "vc_settings_notifications_button_title".localized(),
onTap: { [weak self, dependencies] in
self?.transitionToScreen(
SessionTableViewController(viewModel: NotificationSettingsViewModel(using: dependencies))
)
}
),
SessionCell.Info(
id: .conversations,
leadingAccessory: .icon(
UIImage(named: "icon_msg")?
.withRenderingMode(.alwaysTemplate)
),
title: "CONVERSATION_SETTINGS_TITLE".localized(),
onTap: { [weak self, dependencies] in
self?.transitionToScreen(
SessionTableViewController(viewModel: ConversationSettingsViewModel(using: dependencies))
)
}
),
SessionCell.Info(
id: .messageRequests,
leadingAccessory: .icon(
UIImage(named: "icon_msg_req")?
.withRenderingMode(.alwaysTemplate)
),
title: "MESSAGE_REQUESTS_TITLE".localized(),
onTap: { [weak self, dependencies] in
self?.transitionToScreen(
SessionTableViewController(viewModel: MessageRequestsViewModel(using: dependencies))
)
}
),
SessionCell.Info(
id: .appearance,
leadingAccessory: .icon(
UIImage(named: "icon_apperance")?
.withRenderingMode(.alwaysTemplate)
),
title: "APPEARANCE_TITLE".localized(),
onTap: { [weak self, dependencies] in
self?.transitionToScreen(AppearanceViewController(using: dependencies))
}
),
SessionCell.Info(
id: .inviteAFriend,
leadingAccessory: .icon(
UIImage(named: "icon_invite")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_invite_a_friend_button_title".localized(),
onTap: { [weak self] in
let invitation: String = "Hey, I've been using Session to chat with complete privacy and security. Come join me! Download it at https://getsession.org/. My Session ID is \(state.profile.id) !"
self?.transitionToScreen(
UIActivityViewController(
activityItems: [ invitation ],
applicationActivities: nil
),
title: "Developer Settings", // stringlint:disable
styling: SessionCell.StyleInfo(tintColor: .warning),
onTap: {
self?.transitionToScreen(
SessionTableViewController(viewModel: DeveloperSettingsViewModel(using: dependencies))
)
}
transitionType: .present
)
}
),
SessionCell.Info(
id: .recoveryPhrase,
leadingAccessory: .icon(
UIImage(named: "icon_recovery")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_recovery_phrase_button_title".localized(),
onTap: { [weak self] in
let targetViewController: UIViewController = {
if let modal: SeedModal = try? SeedModal() {
return modal
}
return ConfirmationModal(
info: ConfirmationModal.Info(
title: "ALERT_ERROR_TITLE".localized(),
body: .text("LOAD_RECOVERY_PASSWORD_ERROR".localized()),
cancelTitle: "BUTTON_OK".localized(),
cancelStyle: .alert_text
)
)
}()
self?.transitionToScreen(targetViewController, transitionType: .present)
}
),
SessionCell.Info(
id: .help,
leadingAccessory: .icon(
UIImage(named: "icon_help")?
.withRenderingMode(.alwaysTemplate)
),
title: "HELP_TITLE".localized(),
onTap: { [weak self, dependencies] in
self?.transitionToScreen(
SessionTableViewController(viewModel: HelpViewModel(using: dependencies))
)
}
),
(!state.developerModeEnabled ? nil :
SessionCell.Info(
id: .clearData,
id: .developerSettings,
leadingAccessory: .icon(
UIImage(named: "icon_bin")?
UIImage(systemName: "wrench.and.screwdriver")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_clear_all_data_button_title".localized(),
styling: SessionCell.StyleInfo(tintColor: .danger),
onTap: {
self?.transitionToScreen(NukeDataModal(), transitionType: .present)
title: "Developer Settings", // stringlint:disable
styling: SessionCell.StyleInfo(tintColor: .warning),
onTap: { [weak self, dependencies] in
self?.transitionToScreen(
SessionTableViewController(viewModel: DeveloperSettingsViewModel(using: dependencies))
)
}
)
].compactMap { $0 }
)
]
}
),
SessionCell.Info(
id: .clearData,
leadingAccessory: .icon(
UIImage(named: "icon_bin")?
.withRenderingMode(.alwaysTemplate)
),
title: "vc_settings_clear_all_data_button_title".localized(),
styling: SessionCell.StyleInfo(tintColor: .danger),
onTap: { [weak self] in
self?.transitionToScreen(NukeDataModal(), transitionType: .present)
}
)
].compactMap { $0 }
)
]
}
public lazy var footerView: AnyPublisher<UIView?, Never> = Just(VersionFooterView(numTaps: 9) { [weak self, dependencies] in
/// Do nothing if developer mode is already enabled

@ -565,10 +565,10 @@ class SessionTableViewController<ViewModel>: BaseVC, UITableViewDataSource, UITa
}
switch (info.leadingAccessory, info.trailingAccessory) {
case (_, .highlightingBackgroundLabel(_, _)):
case (_, is SessionCell.AccessoryConfig.HighlightingBackgroundLabel):
return (!cell.trailingAccessoryView.isHidden ? cell.trailingAccessoryView : cell)
case (.highlightingBackgroundLabel(_, _), _):
case (is SessionCell.AccessoryConfig.HighlightingBackgroundLabel, _):
return (!cell.leadingAccessoryView.isHidden ? cell.leadingAccessoryView : cell)
default:
@ -579,8 +579,8 @@ class SessionTableViewController<ViewModel>: BaseVC, UITableViewDataSource, UITa
.enumerated()
.first(where: { index, info in
switch (info.leadingAccessory, info.trailingAccessory) {
case (_, .radio(_, _, let liveIsSelected, _, _)): return liveIsSelected()
case (.radio(_, _, let liveIsSelected, _, _), _): return liveIsSelected()
case (_, let accessory as SessionCell.AccessoryConfig.Radio): return accessory.liveIsSelected()
case (let accessory as SessionCell.AccessoryConfig.Radio, _): return accessory.liveIsSelected()
default: return false
}
})

@ -321,6 +321,12 @@ public extension TableObservation {
}
}
func compactMap<R>(transform: @escaping (T) -> R?) -> TableObservation<R> {
return TableObservation<R> { viewModel, dependencies in
self.generatePublisher(viewModel, dependencies).compactMap(transform).eraseToAnyPublisher()
}
}
func mapWithPrevious<R>(transform: @escaping (T?, T) -> R) -> TableObservation<R> {
return TableObservation<R> { viewModel, dependencies in
self.generatePublisher(viewModel, dependencies)
@ -329,6 +335,15 @@ public extension TableObservation {
.eraseToAnyPublisher()
}
}
func compactMapWithPrevious<R>(transform: @escaping (T?, T) -> R?) -> TableObservation<R> {
return TableObservation<R> { viewModel, dependencies in
self.generatePublisher(viewModel, dependencies)
.withPrevious()
.compactMap(transform)
.eraseToAnyPublisher()
}
}
}
public extension Array {

File diff suppressed because it is too large Load Diff

@ -151,7 +151,7 @@ class UserListViewModel<T: ProfileAssociated & FetchableRecord>: SessionTableVie
}
// Only update the selection if the accessory is a 'radio'
guard case .radio = trailingAccessory else { return }
guard trailingAccessory is SessionCell.AccessoryConfig.Radio else { return }
// Toggle the selection
if !selectedUsersSubject.value.contains(userInfo) {

@ -286,74 +286,74 @@ extension SessionCell {
self.isHidden = false
switch accessory {
// MARK: -- .icon
case .icon(let image, let iconSize, let customTint, let shouldFill, let accessibility):
imageView.accessibilityIdentifier = accessibility?.identifier
imageView.accessibilityLabel = accessibility?.label
imageView.image = image
imageView.themeTintColor = (customTint ?? tintColor)
imageView.contentMode = (shouldFill ? .scaleAspectFill : .scaleAspectFit)
// MARK: -- Icon
case let accessory as SessionCell.AccessoryConfig.Icon:
imageView.accessibilityIdentifier = accessory.accessibility?.identifier
imageView.accessibilityLabel = accessory.accessibility?.label
imageView.image = accessory.image
imageView.themeTintColor = (accessory.customTint ?? tintColor)
imageView.contentMode = (accessory.shouldFill ? .scaleAspectFill : .scaleAspectFit)
imageView.isHidden = false
switch iconSize {
switch accessory.iconSize {
case .fit:
imageView.sizeToFit()
fixedWidthConstraint.constant = (imageView.bounds.width + (shouldFill ? 0 : (Values.smallSpacing * 2)))
fixedWidthConstraint.constant = (imageView.bounds.width + (accessory.shouldFill ? 0 : (Values.smallSpacing * 2)))
fixedWidthConstraint.isActive = true
imageViewWidthConstraint.constant = imageView.bounds.width
imageViewHeightConstraint.constant = imageView.bounds.height
default:
fixedWidthConstraint.isActive = (iconSize.size <= fixedWidthConstraint.constant)
imageViewWidthConstraint.constant = iconSize.size
imageViewHeightConstraint.constant = iconSize.size
fixedWidthConstraint.isActive = (accessory.iconSize.size <= fixedWidthConstraint.constant)
imageViewWidthConstraint.constant = accessory.iconSize.size
imageViewHeightConstraint.constant = accessory.iconSize.size
}
minWidthConstraint.isActive = !fixedWidthConstraint.isActive
imageViewLeadingConstraint.constant = (shouldFill ? 0 : Values.smallSpacing)
imageViewTrailingConstraint.constant = (shouldFill ? 0 : -Values.smallSpacing)
imageViewLeadingConstraint.constant = (accessory.shouldFill ? 0 : Values.smallSpacing)
imageViewTrailingConstraint.constant = (accessory.shouldFill ? 0 : -Values.smallSpacing)
imageViewLeadingConstraint.isActive = true
imageViewTrailingConstraint.isActive = true
imageViewWidthConstraint.isActive = true
imageViewHeightConstraint.isActive = true
imageViewConstraints.forEach { $0.isActive = true }
// MARK: -- .iconAsync
case .iconAsync(let iconSize, let customTint, let shouldFill, let accessibility, let setter):
setter(imageView)
imageView.accessibilityIdentifier = accessibility?.identifier
imageView.accessibilityLabel = accessibility?.label
imageView.themeTintColor = (customTint ?? tintColor)
imageView.contentMode = (shouldFill ? .scaleAspectFill : .scaleAspectFit)
// MARK: -- IconAsync
case let accessory as SessionCell.AccessoryConfig.IconAsync:
accessory.setter(imageView)
imageView.accessibilityIdentifier = accessory.accessibility?.identifier
imageView.accessibilityLabel = accessory.accessibility?.label
imageView.themeTintColor = (accessory.customTint ?? tintColor)
imageView.contentMode = (accessory.shouldFill ? .scaleAspectFill : .scaleAspectFit)
imageView.isHidden = false
switch iconSize {
switch accessory.iconSize {
case .fit:
imageView.sizeToFit()
fixedWidthConstraint.constant = (imageView.bounds.width + (shouldFill ? 0 : (Values.smallSpacing * 2)))
fixedWidthConstraint.constant = (imageView.bounds.width + (accessory.shouldFill ? 0 : (Values.smallSpacing * 2)))
fixedWidthConstraint.isActive = true
imageViewWidthConstraint.constant = imageView.bounds.width
imageViewHeightConstraint.constant = imageView.bounds.height
default:
fixedWidthConstraint.isActive = (iconSize.size <= fixedWidthConstraint.constant)
imageViewWidthConstraint.constant = iconSize.size
imageViewHeightConstraint.constant = iconSize.size
fixedWidthConstraint.isActive = (accessory.iconSize.size <= fixedWidthConstraint.constant)
imageViewWidthConstraint.constant = accessory.iconSize.size
imageViewHeightConstraint.constant = accessory.iconSize.size
}
minWidthConstraint.isActive = !fixedWidthConstraint.isActive
imageViewLeadingConstraint.constant = (shouldFill ? 0 : Values.smallSpacing)
imageViewTrailingConstraint.constant = (shouldFill ? 0 : -Values.smallSpacing)
imageViewLeadingConstraint.constant = (accessory.shouldFill ? 0 : Values.smallSpacing)
imageViewTrailingConstraint.constant = (accessory.shouldFill ? 0 : -Values.smallSpacing)
imageViewLeadingConstraint.isActive = true
imageViewTrailingConstraint.isActive = true
imageViewWidthConstraint.isActive = true
imageViewHeightConstraint.isActive = true
imageViewConstraints.forEach { $0.isActive = true }
// MARK: -- .toggle
case .toggle(let dataSource, let accessibility):
toggleSwitch.accessibilityIdentifier = accessibility?.identifier
toggleSwitch.accessibilityLabel = accessibility?.label
// MARK: -- Toggle
case let accessory as SessionCell.AccessoryConfig.Toggle:
toggleSwitch.accessibilityIdentifier = accessory.accessibility?.identifier
toggleSwitch.accessibilityLabel = accessory.accessibility?.label
toggleSwitch.isHidden = false
toggleSwitch.isEnabled = isEnabled
@ -361,29 +361,29 @@ extension SessionCell {
toggleSwitchConstraints.forEach { $0.isActive = true }
if !isManualReload {
toggleSwitch.setOn(dataSource.oldBoolValue, animated: false)
toggleSwitch.setOn(accessory.oldValue, animated: false)
// Dispatch so the cell reload doesn't conflict with the setting change animation
if dataSource.oldBoolValue != dataSource.currentBoolValue {
if accessory.oldValue != accessory.value {
DispatchQueue.main.asyncAfter(deadline: .now() + .milliseconds(10)) { [weak toggleSwitch] in
toggleSwitch?.setOn(dataSource.currentBoolValue, animated: true)
toggleSwitch?.setOn(accessory.value, animated: true)
}
}
}
// MARK: -- .dropDown
case .dropDown(let dataSource, let accessibility):
dropDownLabel.accessibilityIdentifier = accessibility?.identifier
dropDownLabel.accessibilityLabel = accessibility?.label
dropDownLabel.text = dataSource.currentStringValue
// MARK: -- DropDown
case let accessory as SessionCell.AccessoryConfig.DropDown:
dropDownLabel.accessibilityIdentifier = accessory.accessibility?.identifier
dropDownLabel.accessibilityLabel = accessory.accessibility?.label
dropDownLabel.text = accessory.dynamicString()
dropDownStackView.isHidden = false
dropDownStackViewConstraints.forEach { $0.isActive = true }
minWidthConstraint.isActive = true
// MARK: -- .radio
case .radio(let size, _, let isSelectedRetriever, let storedSelection, let accessibility):
let isSelected: Bool = isSelectedRetriever()
let wasOldSelection: Bool = (!isSelected && storedSelection)
// MARK: -- Radio
case let accessory as SessionCell.AccessoryConfig.Radio:
let isSelected: Bool = accessory.liveIsSelected()
let wasOldSelection: Bool = (!isSelected && accessory.wasSavedSelection)
radioView.isAccessibilityElement = true
@ -405,12 +405,12 @@ extension SessionCell {
)
}()
radioBorderView.layer.cornerRadius = (size.borderSize / 2)
radioBorderView.layer.cornerRadius = (accessory.size.borderSize / 2)
radioView.accessibilityIdentifier = accessibility?.identifier
radioView.accessibilityLabel = accessibility?.label
radioView.accessibilityIdentifier = accessory.accessibility?.identifier
radioView.accessibilityLabel = accessory.accessibility?.label
radioView.alpha = (wasOldSelection ? 0.3 : 1)
radioView.isHidden = (!isSelected && !storedSelection)
radioView.isHidden = (!isSelected && !accessory.wasSavedSelection)
radioView.themeBackgroundColor = {
guard isEnabled else {
return (isSelected || wasOldSelection ?
@ -424,12 +424,12 @@ extension SessionCell {
.radioButton_unselectedBackground
)
}()
radioView.layer.cornerRadius = (size.selectionSize / 2)
radioView.layer.cornerRadius = (accessory.size.selectionSize / 2)
radioViewWidthConstraint.constant = size.selectionSize
radioViewHeightConstraint.constant = size.selectionSize
radioBorderViewWidthConstraint.constant = size.borderSize
radioBorderViewHeightConstraint.constant = size.borderSize
radioViewWidthConstraint.constant = accessory.size.selectionSize
radioViewHeightConstraint.constant = accessory.size.selectionSize
radioBorderViewWidthConstraint.constant = accessory.size.borderSize
radioBorderViewHeightConstraint.constant = accessory.size.borderSize
fixedWidthConstraint.isActive = true
radioViewWidthConstraint.isActive = true
@ -438,72 +438,64 @@ extension SessionCell {
radioBorderViewHeightConstraint.isActive = true
radioBorderViewConstraints.forEach { $0.isActive = true }
// MARK: -- .highlightingBackgroundLabel
case .highlightingBackgroundLabel(let title, let accessibility):
highlightingBackgroundLabel.accessibilityIdentifier = accessibility?.identifier
highlightingBackgroundLabel.accessibilityLabel = accessibility?.label
highlightingBackgroundLabel.text = title
// MARK: -- HighlightingBackgroundLabel
case let accessory as SessionCell.AccessoryConfig.HighlightingBackgroundLabel:
highlightingBackgroundLabel.accessibilityIdentifier = accessory.accessibility?.identifier
highlightingBackgroundLabel.accessibilityLabel = accessory.accessibility?.label
highlightingBackgroundLabel.text = accessory.title
highlightingBackgroundLabel.themeTextColor = tintColor
highlightingBackgroundLabel.isHidden = false
highlightingBackgroundLabelConstraints.forEach { $0.isActive = true }
minWidthConstraint.isActive = true
// MARK: -- .profile
case .profile(
let profileId,
let profileSize,
let threadVariant,
let displayPictureFilename,
let profile,
let profileIcon,
let additionalProfile,
let additionalProfileIcon,
let accessibility
):
// MARK: -- DisplayPicture
case let accessory as SessionCell.AccessoryConfig.DisplayPicture:
// Note: We MUST set the 'size' property before triggering the 'update'
// function or the profile picture won't layout correctly
profilePictureView.accessibilityIdentifier = accessibility?.identifier
profilePictureView.accessibilityLabel = accessibility?.label
profilePictureView.isAccessibilityElement = (accessibility != nil)
profilePictureView.size = profileSize
profilePictureView.accessibilityIdentifier = accessory.accessibility?.identifier
profilePictureView.accessibilityLabel = accessory.accessibility?.label
profilePictureView.isAccessibilityElement = (accessory.accessibility != nil)
profilePictureView.size = accessory.size
profilePictureView.update(
publicKey: profileId,
threadVariant: threadVariant,
displayPictureFilename: displayPictureFilename,
profile: profile,
profileIcon: profileIcon,
additionalProfile: additionalProfile,
additionalProfileIcon: additionalProfileIcon
publicKey: accessory.id,
threadVariant: accessory.threadVariant,
displayPictureFilename: accessory.displayPictureFilename,
profile: accessory.profile,
profileIcon: accessory.profileIcon,
additionalProfile: accessory.additionalProfile,
additionalProfileIcon: accessory.additionalProfileIcon
)
profilePictureView.isHidden = false
fixedWidthConstraint.constant = profileSize.viewSize
fixedWidthConstraint.constant = accessory.size.viewSize
fixedWidthConstraint.isActive = true
profilePictureViewConstraints.forEach { $0.isActive = true }
// MARK: -- .search
case .search(let placeholder, let accessibility, let searchTermChanged):
self.searchTermChanged = searchTermChanged
searchBar.accessibilityIdentifier = accessibility?.identifier
searchBar.accessibilityLabel = accessibility?.label
searchBar.placeholder = placeholder
// MARK: -- Search
case let accessory as SessionCell.AccessoryConfig.Search:
self.searchTermChanged = accessory.searchTermChanged
searchBar.accessibilityIdentifier = accessory.accessibility?.identifier
searchBar.accessibilityLabel = accessory.accessibility?.label
searchBar.placeholder = accessory.placeholder
searchBar.isHidden = false
searchBarConstraints.forEach { $0.isActive = true }
// MARK: -- .button
case .button(let style, let title, let accessibility, let onTap):
self.onTap = onTap
button.accessibilityIdentifier = accessibility?.identifier
button.accessibilityLabel = accessibility?.label
button.setTitle(title, for: .normal)
button.style = style
// MARK: -- Button
case let accessory as SessionCell.AccessoryConfig.Button:
self.onTap = accessory.run
button.accessibilityIdentifier = accessory.accessibility?.identifier
button.accessibilityLabel = accessory.accessibility?.label
button.setTitle(accessory.title, for: .normal)
button.style = accessory.style
button.isHidden = false
minWidthConstraint.isActive = true
buttonConstraints.forEach { $0.isActive = true }
// MARK: -- .customView
case .customView(_, let viewGenerator):
let generatedView: UIView = viewGenerator()
// MARK: -- CustomView
case let accessory as SessionCell.AccessoryConfig.CustomView:
let generatedView: UIView = accessory.viewGenerator()
generatedView.accessibilityIdentifier = accessory.accessibility?.identifier
generatedView.accessibilityLabel = accessory.accessibility?.label
addSubview(generatedView)
generatedView.pin(.top, to: .top, of: self)
@ -514,6 +506,9 @@ extension SessionCell {
customView?.removeFromSuperview() // Just in case
customView = generatedView
minWidthConstraint.isActive = true
// If we get an unknown case then just hide again
default: self.isHidden = true
}
}

@ -372,7 +372,7 @@ public class SessionCell: UITableViewCell {
trailingAccessoryFillConstraint.isActive = trailingFitToEdge
accessoryWidthMatchConstraint.isActive = {
switch (info.leadingAccessory, info.trailingAccessory) {
case (.button, .button): return true
case is (SessionCell.AccessoryConfig.Button, SessionCell.AccessoryConfig.Button): return true
default: return false
}
}()
@ -465,9 +465,8 @@ public class SessionCell: UITableViewCell {
let fittedEdgePadding: CGFloat = {
func targetSize(accessory: Accessory?) -> CGFloat {
switch accessory {
case .icon(_, let iconSize, _, _, _), .iconAsync(let iconSize, _, _, _, _):
return iconSize.size
case let accessory as SessionCell.AccessoryConfig.Icon: return accessory.iconSize.size
case let accessory as SessionCell.AccessoryConfig.IconAsync: return accessory.iconSize.size
default: return defaultEdgePadding
}
}

@ -81,7 +81,7 @@ public protocol ThemedNavigation {
// MARK: - ThemeValue
public indirect enum ThemeValue: Hashable {
public indirect enum ThemeValue: Hashable, Equatable {
case value(ThemeValue, alpha: CGFloat)
// The 'highlighted' state of a colour will automatically lighten/darken a ThemeValue

Loading…
Cancel
Save