Fixed a few more QA issues

• Added some localised copy
• Added the helpdesk article URL for the deprecated groups banner
• Updated mentions to include the current user in groups, 1-1 and NTS conversations (also made the current user appear as "you")
• Fixed a layout issue with the deleted message view
• Fixed an issue where the "mark as unread" action would incorrectly appear for the Note to Self conversation
• Fixed an issue where we were incorrectly sending an ExpirationTimerUpdate in some cases
pull/894/head
Morgan Pretty 2 months ago
parent c5cccad08f
commit 81242de035

@ -7916,7 +7916,7 @@
CLANG_WARN__ARC_BRIDGE_CAST_NONARC = YES; CLANG_WARN__ARC_BRIDGE_CAST_NONARC = YES;
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
CODE_SIGN_IDENTITY = "iPhone Developer"; CODE_SIGN_IDENTITY = "iPhone Developer";
CURRENT_PROJECT_VERSION = 528; CURRENT_PROJECT_VERSION = 529;
ENABLE_BITCODE = NO; ENABLE_BITCODE = NO;
ENABLE_STRICT_OBJC_MSGSEND = YES; ENABLE_STRICT_OBJC_MSGSEND = YES;
ENABLE_TESTABILITY = YES; ENABLE_TESTABILITY = YES;
@ -7992,7 +7992,7 @@
CLANG_WARN__ARC_BRIDGE_CAST_NONARC = YES; CLANG_WARN__ARC_BRIDGE_CAST_NONARC = YES;
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
CODE_SIGN_IDENTITY = "iPhone Distribution"; CODE_SIGN_IDENTITY = "iPhone Distribution";
CURRENT_PROJECT_VERSION = 528; CURRENT_PROJECT_VERSION = 529;
ENABLE_BITCODE = NO; ENABLE_BITCODE = NO;
ENABLE_MODULE_VERIFIER = YES; ENABLE_MODULE_VERIFIER = YES;
ENABLE_STRICT_OBJC_MSGSEND = YES; ENABLE_STRICT_OBJC_MSGSEND = YES;

@ -626,10 +626,14 @@ class EditGroupViewModel: SessionTableViewModel, NavigatableStateHolder, Editabl
self?.transitionToScreen( self?.transitionToScreen(
ConfirmationModal( ConfirmationModal(
info: ConfirmationModal.Info( info: ConfirmationModal.Info(
title: "Invitation Failed",//.localized(), title: "inviteFailed"
body: .text("An error occurred and the invitations were not successfully sent, would you like to try again?"),//.localized()), .putNumber(memberInfo.count)
confirmTitle: "retry".localized(), .localized(),
cancelTitle: "dismiss".localized(), body: .text("inviteFailedDescription"
.putNumber(memberInfo.count)
.localized()),
confirmTitle: "yes".localized(),
cancelTitle: "cancel".localized(),
cancelStyle: .alert_text, cancelStyle: .alert_text,
dismissOnConfirm: false, dismissOnConfirm: false,
onConfirm: { modal in onConfirm: { modal in
@ -692,10 +696,14 @@ class EditGroupViewModel: SessionTableViewModel, NavigatableStateHolder, Editabl
self?.transitionToScreen( self?.transitionToScreen(
ConfirmationModal( ConfirmationModal(
info: ConfirmationModal.Info( info: ConfirmationModal.Info(
title: "Invitation Failed",//.localized(), title: "inviteFailed"
body: .text("An error occurred and the invitations were not successfully sent, would you like to try again?"),//.localized()), .putNumber(memberIds.count)
confirmTitle: "retry".localized(), .localized(),
cancelTitle: "dismiss".localized(), body: .text("inviteFailedDescription"
.putNumber(memberIds.count)
.localized()),
confirmTitle: "yes".localized(),
cancelTitle: "cancel".localized(),
cancelStyle: .alert_text, cancelStyle: .alert_text,
dismissOnConfirm: false, dismissOnConfirm: false,
onConfirm: { modal in onConfirm: { modal in

@ -821,7 +821,12 @@ extension ConversationVC:
// stringlint:ignore_start // stringlint:ignore_start
if lastCharacter == "@" && isCharacterBeforeLastWhiteSpaceOrStartOfLine { if lastCharacter == "@" && isCharacterBeforeLastWhiteSpaceOrStartOfLine {
currentMentionStartIndex = lastCharacterIndex currentMentionStartIndex = lastCharacterIndex
snInputView.showMentionsUI(for: self.viewModel.mentions()) snInputView.showMentionsUI(
for: self.viewModel.mentions(),
currentUserSessionId: self.viewModel.threadData.currentUserSessionId,
currentUserBlinded15SessionId: self.viewModel.threadData.currentUserBlinded15SessionId,
currentUserBlinded25SessionId: self.viewModel.threadData.currentUserBlinded25SessionId
)
} }
else if lastCharacter.isWhitespace || lastCharacter == "@" { // the lastCharacter == "@" is to check for @@ else if lastCharacter.isWhitespace || lastCharacter == "@" { // the lastCharacter == "@" is to check for @@
currentMentionStartIndex = nil currentMentionStartIndex = nil
@ -830,7 +835,12 @@ extension ConversationVC:
else { else {
if let currentMentionStartIndex = currentMentionStartIndex { if let currentMentionStartIndex = currentMentionStartIndex {
let query = String(newText[newText.index(after: currentMentionStartIndex)...]) // + 1 to get rid of the @ let query = String(newText[newText.index(after: currentMentionStartIndex)...]) // + 1 to get rid of the @
snInputView.showMentionsUI(for: self.viewModel.mentions(for: query)) snInputView.showMentionsUI(
for: self.viewModel.mentions(for: query),
currentUserSessionId: self.viewModel.threadData.currentUserSessionId,
currentUserBlinded15SessionId: self.viewModel.threadData.currentUserBlinded15SessionId,
currentUserBlinded25SessionId: self.viewModel.threadData.currentUserBlinded25SessionId
)
} }
} }
// stringlint:ignore_stop // stringlint:ignore_stop

@ -850,7 +850,7 @@ public class ConversationViewModel: OWSAudioPlayerDelegate, NavigatableStateHold
let threadData: SessionThreadViewModel = self.internalThreadData let threadData: SessionThreadViewModel = self.internalThreadData
return dependencies[singleton: .storage] return dependencies[singleton: .storage]
.read { [dependencies] db -> [MentionInfo] in .read { [weak self, dependencies] db -> [MentionInfo] in
let userSessionId: SessionId = dependencies[cache: .general].sessionId let userSessionId: SessionId = dependencies[cache: .general].sessionId
let pattern: FTS5Pattern? = try? SessionThreadViewModel.pattern(db, searchTerm: query, forTable: Profile.self) let pattern: FTS5Pattern? = try? SessionThreadViewModel.pattern(db, searchTerm: query, forTable: Profile.self)
let capabilities: Set<Capability.Variant> = (threadData.threadVariant != .community ? let capabilities: Set<Capability.Variant> = (threadData.threadVariant != .community ?
@ -873,6 +873,8 @@ public class ConversationViewModel: OWSAudioPlayerDelegate, NavigatableStateHold
threadId: threadData.threadId, threadId: threadData.threadId,
threadVariant: threadData.threadVariant, threadVariant: threadData.threadVariant,
targetPrefixes: targetPrefixes, targetPrefixes: targetPrefixes,
currentUserBlinded15SessionId: self?.threadData.currentUserBlinded15SessionId,
currentUserBlinded25SessionId: self?.threadData.currentUserBlinded25SessionId,
pattern: pattern pattern: pattern
)? )?
.fetchAll(db)) .fetchAll(db))

@ -518,7 +518,15 @@ final class InputView: UIView, InputViewButtonDelegate, InputTextViewDelegate, M
) )
} }
func showMentionsUI(for candidates: [MentionInfo]) { func showMentionsUI(
for candidates: [MentionInfo],
currentUserSessionId: String,
currentUserBlinded15SessionId: String?,
currentUserBlinded25SessionId: String?
) {
mentionsView.currentUserSessionId = currentUserSessionId
mentionsView.currentUserBlinded15SessionId = currentUserBlinded15SessionId
mentionsView.currentUserBlinded25SessionId = currentUserBlinded25SessionId
mentionsView.candidates = candidates mentionsView.candidates = candidates
let mentionCellHeight = (ProfilePictureView.Size.message.viewSize + 2 * Values.smallSpacing) let mentionCellHeight = (ProfilePictureView.Size.message.viewSize + 2 * Values.smallSpacing)

@ -8,6 +8,9 @@ import SignalUtilitiesKit
final class MentionSelectionView: UIView, UITableViewDataSource, UITableViewDelegate { final class MentionSelectionView: UIView, UITableViewDataSource, UITableViewDelegate {
private let dependencies: Dependencies private let dependencies: Dependencies
var currentUserSessionId: String?
var currentUserBlinded15SessionId: String?
var currentUserBlinded25SessionId: String?
var candidates: [MentionInfo] = [] { var candidates: [MentionInfo] = [] {
didSet { didSet {
tableView.isScrollEnabled = (candidates.count > 4) tableView.isScrollEnabled = (candidates.count > 4)
@ -92,6 +95,9 @@ final class MentionSelectionView: UIView, UITableViewDataSource, UITableViewDele
for: candidates[indexPath.row].openGroupRoomToken, for: candidates[indexPath.row].openGroupRoomToken,
on: candidates[indexPath.row].openGroupServer on: candidates[indexPath.row].openGroupServer
), ),
currentUserSessionId: currentUserSessionId,
currentUserBlinded15SessionId: currentUserBlinded15SessionId,
currentUserBlinded25SessionId: currentUserBlinded25SessionId,
isLast: (indexPath.row == (candidates.count - 1)), isLast: (indexPath.row == (candidates.count - 1)),
using: dependencies using: dependencies
) )
@ -187,10 +193,21 @@ private extension MentionSelectionView {
with profile: Profile, with profile: Profile,
threadVariant: SessionThread.Variant, threadVariant: SessionThread.Variant,
isUserModeratorOrAdmin: Bool, isUserModeratorOrAdmin: Bool,
currentUserSessionId: String?,
currentUserBlinded15SessionId: String?,
currentUserBlinded25SessionId: String?,
isLast: Bool, isLast: Bool,
using dependencies: Dependencies using dependencies: Dependencies
) { ) {
displayNameLabel.text = profile.displayName(for: threadVariant) let currentUserSessionIds: Set<String> = [
currentUserSessionId,
currentUserBlinded15SessionId,
currentUserBlinded25SessionId
].compactMap { $0 }.asSet()
displayNameLabel.text = (currentUserSessionIds.contains(profile.id) ?
"you".localized() :
profile.displayName(for: threadVariant)
)
profilePictureView.update( profilePictureView.update(
publicKey: profile.id, publicKey: profile.id,
threadVariant: .contact, // Always show the display picture in 'contact' mode threadVariant: .contact, // Always show the display picture in 'contact' mode

@ -13,11 +13,11 @@ final class DeletedMessageView: UIView {
// MARK: - Lifecycle // MARK: - Lifecycle
init(textColor: ThemeValue, variant: Interaction.Variant) { init(textColor: ThemeValue, variant: Interaction.Variant, maxWidth: CGFloat) {
super.init(frame: CGRect.zero) super.init(frame: CGRect.zero)
accessibilityIdentifier = "Deleted message" accessibilityIdentifier = "Deleted message"
isAccessibilityElement = true isAccessibilityElement = true
setUpViewHierarchy(textColor: textColor, variant: variant) setUpViewHierarchy(textColor: textColor, variant: variant, maxWidth: maxWidth)
} }
override init(frame: CGRect) { override init(frame: CGRect) {
@ -28,7 +28,7 @@ final class DeletedMessageView: UIView {
preconditionFailure("Use init(textColor:) instead.") preconditionFailure("Use init(textColor:) instead.")
} }
private func setUpViewHierarchy(textColor: ThemeValue, variant: Interaction.Variant) { private func setUpViewHierarchy(textColor: ThemeValue, variant: Interaction.Variant, maxWidth: CGFloat) {
// Image view // Image view
let imageContainerView: UIView = UIView() let imageContainerView: UIView = UIView()
imageContainerView.set(.width, to: DeletedMessageView.iconImageViewSize) imageContainerView.set(.width, to: DeletedMessageView.iconImageViewSize)
@ -45,6 +45,7 @@ final class DeletedMessageView: UIView {
// Body label // Body label
let titleLabel = UILabel() let titleLabel = UILabel()
titleLabel.setContentHuggingPriority(.required, for: .vertical) titleLabel.setContentHuggingPriority(.required, for: .vertical)
titleLabel.preferredMaxLayoutWidth = maxWidth - 6 // `6` for the `stackView.layoutMargins`
titleLabel.font = .systemFont(ofSize: Values.smallFontSize) titleLabel.font = .systemFont(ofSize: Values.smallFontSize)
titleLabel.text = { titleLabel.text = {
switch variant { switch variant {
@ -66,8 +67,8 @@ final class DeletedMessageView: UIView {
stackView.layoutMargins = UIEdgeInsets(top: 0, leading: 0, bottom: 0, trailing: 6) stackView.layoutMargins = UIEdgeInsets(top: 0, leading: 0, bottom: 0, trailing: 6)
addSubview(stackView) addSubview(stackView)
let calculatedSize: CGSize = stackView.systemLayoutSizeFitting(CGSize(width: maxWidth, height: 999))
stackView.pin(to: self, withInset: Values.smallSpacing) stackView.pin(to: self, withInset: Values.smallSpacing)
stackView.set(.height, to: .height, of: titleLabel) stackView.set(.height, to: calculatedSize.height)
titleLabel.set(.height, greaterThanOrEqualTo: .height, of: imageView)
} }
} }

@ -451,13 +451,11 @@ final class VisibleMessageCell: MessageCell, TappableLabelDelegate {
lastSearchText: String?, lastSearchText: String?,
using dependencies: Dependencies using dependencies: Dependencies
) { ) {
let isOutgoing: Bool = ( let bodyLabelTextColor: ThemeValue = (cellViewModel.variant.isOutgoing ?
cellViewModel.variant == .standardOutgoing || .messageBubble_outgoingText :
cellViewModel.variant == .standardOutgoingDeleted || .messageBubble_incomingText
cellViewModel.variant == .standardOutgoingDeletedLocally
) )
let bodyLabelTextColor: ThemeValue = (isOutgoing ? .messageBubble_outgoingText : .messageBubble_incomingText) snContentView.alignment = (cellViewModel.variant.isOutgoing ? .trailing : .leading)
snContentView.alignment = (isOutgoing ? .trailing : .leading)
for subview in snContentView.arrangedSubviews { for subview in snContentView.arrangedSubviews {
snContentView.removeArrangedSubview(subview) snContentView.removeArrangedSubview(subview)
@ -474,9 +472,11 @@ final class VisibleMessageCell: MessageCell, TappableLabelDelegate {
// Handle the deleted state first (it's much simpler than the others) // Handle the deleted state first (it's much simpler than the others)
guard !cellViewModel.variant.isDeletedMessage else { guard !cellViewModel.variant.isDeletedMessage else {
let inset: CGFloat = 12
let deletedMessageView: DeletedMessageView = DeletedMessageView( let deletedMessageView: DeletedMessageView = DeletedMessageView(
textColor: bodyLabelTextColor, textColor: bodyLabelTextColor,
variant: cellViewModel.variant variant: cellViewModel.variant,
maxWidth: (VisibleMessageCell.getMaxWidth(for: cellViewModel) - 2 * inset)
) )
bubbleView.addSubview(deletedMessageView) bubbleView.addSubview(deletedMessageView)
deletedMessageView.pin(to: bubbleView) deletedMessageView.pin(to: bubbleView)
@ -510,7 +510,7 @@ final class VisibleMessageCell: MessageCell, TappableLabelDelegate {
imageAttachment: cellViewModel.linkPreviewAttachment, imageAttachment: cellViewModel.linkPreviewAttachment,
using: dependencies using: dependencies
), ),
isOutgoing: isOutgoing, isOutgoing: cellViewModel.variant.isOutgoing,
delegate: self, delegate: self,
cellViewModel: cellViewModel, cellViewModel: cellViewModel,
bodyLabelTextColor: bodyLabelTextColor, bodyLabelTextColor: bodyLabelTextColor,
@ -528,7 +528,7 @@ final class VisibleMessageCell: MessageCell, TappableLabelDelegate {
name: (linkPreview.title ?? ""), name: (linkPreview.title ?? ""),
url: linkPreview.url, url: linkPreview.url,
textColor: bodyLabelTextColor, textColor: bodyLabelTextColor,
isOutgoing: isOutgoing isOutgoing: cellViewModel.variant.isOutgoing
) )
openGroupInvitationView.isAccessibilityElement = true openGroupInvitationView.isAccessibilityElement = true
openGroupInvitationView.accessibilityIdentifier = "Community invitation" openGroupInvitationView.accessibilityIdentifier = "Community invitation"
@ -555,7 +555,7 @@ final class VisibleMessageCell: MessageCell, TappableLabelDelegate {
currentUserSessionId: cellViewModel.currentUserSessionId, currentUserSessionId: cellViewModel.currentUserSessionId,
currentUserBlinded15SessionId: cellViewModel.currentUserBlinded15SessionId, currentUserBlinded15SessionId: cellViewModel.currentUserBlinded15SessionId,
currentUserBlinded25SessionId: cellViewModel.currentUserBlinded25SessionId, currentUserBlinded25SessionId: cellViewModel.currentUserBlinded25SessionId,
direction: (isOutgoing ? .outgoing : .incoming), direction: (cellViewModel.variant.isOutgoing ? .outgoing : .incoming),
attachment: cellViewModel.quoteAttachment, attachment: cellViewModel.quoteAttachment,
using: dependencies using: dependencies
) )
@ -610,7 +610,7 @@ final class VisibleMessageCell: MessageCell, TappableLabelDelegate {
items: (cellViewModel.attachments? items: (cellViewModel.attachments?
.filter { $0.isVisualMedia }) .filter { $0.isVisualMedia })
.defaulting(to: []), .defaulting(to: []),
isOutgoing: isOutgoing, isOutgoing: cellViewModel.variant.isOutgoing,
maxMessageWidth: maxMessageWidth, maxMessageWidth: maxMessageWidth,
using: dependencies using: dependencies
) )

@ -359,20 +359,6 @@ class ThreadDisappearingMessagesSettingsViewModel: SessionTableViewModel, Naviga
using: dependencies using: dependencies
) )
// Send the ExpirationTimerUpdate message
let expirationTimerUpdateMessage: ExpirationTimerUpdate = ExpirationTimerUpdate()
.with(sentTimestampMs: UInt64(currentOffsetTimestampMs))
.with(updatedConfig)
try MessageSender.send(
db,
message: expirationTimerUpdateMessage,
interactionId: interactionId,
threadId: threadId,
threadVariant: threadVariant,
using: dependencies
)
// Update libSession // Update libSession
switch threadVariant { switch threadVariant {
case .contact: case .contact:

@ -1002,12 +1002,15 @@ class ThreadSettingsViewModel: SessionTableViewModel, NavigatableStateHolder, Ob
case .failure: case .failure:
viewModel?.transitionToScreen( viewModel?.transitionToScreen(
ConfirmationModal( ConfirmationModal(
// FIXME: Localise these
info: ConfirmationModal.Info( info: ConfirmationModal.Info(
title: "Promotion Failed", title: "promotionFailed"
body: .text("An error occurred and the promotions were not successfully sent, would you like to try again?"), .putNumber(memberInfo.count)
confirmTitle: "retry".localized(), .localized(),
cancelTitle: "dismiss".localized(), body: .text("promotionFailedDescription"
.putNumber(memberInfo.count)
.localized()),
confirmTitle: "yes".localized(),
cancelTitle: "cancel".localized(),
cancelStyle: .alert_text, cancelStyle: .alert_text,
dismissOnConfirm: false, dismissOnConfirm: false,
onConfirm: { modal in onConfirm: { modal in

@ -780,10 +780,14 @@ public final class HomeVC: BaseVC, LibSessionRespondingViewController, UITableVi
switch section.model { switch section.model {
case .threads: case .threads:
// Cannot properly sync outgoing blinded message requests so don't provide the option // Cannot properly sync outgoing blinded message requests so don't provide the option,
// the 'Note to Self' conversation also doesn't support 'mark as unread' so don't
// provide it there either
guard guard
threadViewModel.threadVariant != .contact || threadViewModel.threadId != threadViewModel.currentUserSessionId && (
(try? SessionId(from: section.elements[indexPath.row].threadId))?.prefix == .standard threadViewModel.threadVariant != .contact ||
(try? SessionId(from: section.elements[indexPath.row].threadId))?.prefix == .standard
)
else { return nil } else { return nil }
return UIContextualAction.configuration( return UIContextualAction.configuration(

File diff suppressed because one or more lines are too long

@ -498,7 +498,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "اِقبَل" "value" : "قبول"
} }
}, },
"az" : { "az" : {
@ -540,7 +540,7 @@
"cs" : { "cs" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "Přijmou" "value" : "Přijmout"
} }
}, },
"cy" : { "cy" : {
@ -6767,7 +6767,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "لايمكن ازاله الادمن." "value" : "لا يمكن إزاله المشرف."
} }
}, },
"az" : { "az" : {
@ -10635,7 +10635,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "فشل الترقيه كمسئول" "value" : "فشل الترقيه كمشرف"
} }
}, },
"az" : { "az" : {
@ -12586,7 +12586,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "تم إرسال الترقيه كمسئول" "value" : "تم إرسال الترقية كمشرف"
} }
}, },
"az" : { "az" : {
@ -14040,7 +14040,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "لا يوجد مسؤولين في هذا المجتمع." "value" : "لا يوجد مشرفين في هذا المجتمع."
} }
}, },
"az" : { "az" : {
@ -17452,7 +17452,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "اعدادات المسؤول" "value" : "إعدادات المشرف"
} }
}, },
"az" : { "az" : {
@ -30942,7 +30942,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد مسح كافة المرفقات؟ سيتم أيضًا حذف الرسائل ذات المرفقات." "value" : "هل أنت متأكد من حذف كافة المرفقات؟ سيتم أيضًا حذف الرسائل ذات المرفقات."
} }
}, },
"az" : { "az" : {
@ -36247,7 +36247,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "غير قادر على فتح الملف." "value" : "تعذر فتح الملف."
} }
}, },
"az" : { "az" : {
@ -48246,7 +48246,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "انضغط لتنزيل {file_type}" "value" : "انقر لتنزيل {file_type}"
} }
}, },
"az" : { "az" : {
@ -60287,7 +60287,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من حظر <b>{name}؟</b> المستخدمين المحظورين لايمكنهم إرسال طلبات الرسائل، دعوات المجموعات أو الإتصال بك." "value" : "هل أنت متأكد من حظر <b>{name}؟</b> المستخدمين المحظورين لا يمكنهم إرسال طلبات الرسائل، دعوات المجموعات أو الاتصال بك."
} }
}, },
"az" : { "az" : {
@ -72087,7 +72087,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "{app_name} Call" "value" : "مكالمة {app_name}"
} }
}, },
"az" : { "az" : {
@ -77841,7 +77841,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "{app_name} يحتاج إذن الوصول إلى الكاميرا لالتقاط الصور ومقاطع الفيديو، أو لمسح رموز الاستجابة السريعة." "value" : "{app_name} يحتاج إذن الوصول إلى الكاميرا لالتقاط الصور ومقاطع الفيديو، أو لمسح رموز QR."
} }
}, },
"az" : { "az" : {
@ -78326,7 +78326,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "{app_name} يحتاج إذن الوصول إلى الكاميرا لمسح رموز الاستجابة السريعة" "value" : "{app_name} يحتاج إذن الوصول إلى الكاميرا لمسح رموز QR"
} }
}, },
"az" : { "az" : {
@ -82203,6 +82203,48 @@
} }
} }
}, },
"ar" : {
"variations" : {
"plural" : {
"few" : {
"stringUnit" : {
"state" : "translated",
"value" : "لم يتم حذف البيانات بواسطة %lld من عقد الخدمة Service Nodes. معرفات عقد الخدمة Service Node IDs: {service_node_id}."
}
},
"many" : {
"stringUnit" : {
"state" : "translated",
"value" : "لم يتم حذف البيانات بواسطة %lld من عقد الخدمة Service Nodes. معرفات عقد الخدمة Service Node IDs: {service_node_id}."
}
},
"one" : {
"stringUnit" : {
"state" : "translated",
"value" : "لم يتم حذف البيانات بواسطة عقدة الخدمة (Service Node) %lld . معرف عقدة الخدمة (Service Node ID): {service_node_id}."
}
},
"other" : {
"stringUnit" : {
"state" : "translated",
"value" : "لم يتم حذف البيانات بواسطة %lld من عقد الخدمة Service Nodes. معرفات عقد الخدمة Service Node IDs: {service_node_id}."
}
},
"two" : {
"stringUnit" : {
"state" : "translated",
"value" : "لم يتم حذف البيانات بواسطة %lld من عقد الخدمة Service Nodes. معرفات عقد الخدمة Service Node IDs: {service_node_id}."
}
},
"zero" : {
"stringUnit" : {
"state" : "translated",
"value" : "لم يتم حذف البيانات بواسطة %lld من عقد الخدمة Service Nodes. معرفات عقد الخدمة Service Node IDs: {service_node_id}."
}
}
}
}
},
"az" : { "az" : {
"variations" : { "variations" : {
"plural" : { "plural" : {
@ -85657,7 +85699,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل متأكد من رغبتك لمسح جهازك؟" "value" : "هل متأكد من رغبتك بمسح جهازك؟"
} }
}, },
"az" : { "az" : {
@ -87112,7 +87154,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من مسح جميع الرسائل من محادثتك مع <b>{name}</b> من جهازك؟" "value" : "هل أنت متأكد من مسح جميع الرسائل من محادثتك مع <b>{name}</b> من جهازك؟"
} }
}, },
"az" : { "az" : {
@ -87597,7 +87639,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد حذف كافة الرسائل <b>{community_name}؟</b> من جهازك." "value" : "هل أنت متأكد من حذف كافة الرسائل <b>{community_name}؟</b> من جهازك."
} }
}, },
"az" : { "az" : {
@ -89052,7 +89094,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد حذف كافة الرسائل <b>{group_name}؟</b>" "value" : "هل أنت متأكد من حذف كافة الرسائل <b>{group_name}؟</b>"
} }
}, },
"az" : { "az" : {
@ -89537,7 +89579,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد حذف كافة الرسائل <b>{group_name}؟</b> من جهازك." "value" : "هل أنت متأكد من حذف كافة الرسائل <b>{group_name}؟</b> من جهازك."
} }
}, },
"az" : { "az" : {
@ -90022,7 +90064,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد حذف كافة رسائل ملاحظة لنفسي من جهازك؟" "value" : "هل أنت متأكد من مسح كافة رسائل \"ملاحظة لنفسي\" من جهازك؟"
} }
}, },
"az" : { "az" : {
@ -94806,7 +94848,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "عذراً، حدث خطأ. حاول مرة أخرى لاحقاً." "value" : "عفواً، حدث خطأ. الرجاء المحاولة مرة أخرى لاحقاً."
} }
}, },
"az" : { "az" : {
@ -102506,7 +102548,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد مسح <b>{name}؟</b> من قائمة جهات إتصالك؟ ستصل أي رسائل جديدة من <b>{name}</b> كطلب رسالة." "value" : "هل أنت متأكد من حذف <b>{name}؟</b> من قائمة جهات إتصالك؟ ستصل أي رسائل جديدة من <b>{name}</b> كطلب رسالة."
} }
}, },
"az" : { "az" : {
@ -103470,7 +103512,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "اختر اتصالات" "value" : "تحديد جهات الاتصال"
} }
}, },
"az" : { "az" : {
@ -116499,7 +116541,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "مراسلة جديدة" "value" : "محادثة جديدة"
} }
}, },
"az" : { "az" : {
@ -128055,7 +128097,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "عند التمكين، سيتم إشعارك فقط بالرسائل التي تذكرك." "value" : "عند التمكين، سيتم إشعارك بالرسائل التي تشير إليك فقط."
} }
}, },
"az" : { "az" : {
@ -135105,6 +135147,58 @@
"deleteMessageConfirm" : { "deleteMessageConfirm" : {
"extractionState" : "manual", "extractionState" : "manual",
"localizations" : { "localizations" : {
"ar" : {
"stringUnit" : {
"state" : "translated",
"value" : "%#@arg1@"
},
"substitutions" : {
"arg1" : {
"argNum" : 1,
"formatSpecifier" : "lld",
"variations" : {
"plural" : {
"few" : {
"stringUnit" : {
"state" : "translated",
"value" : "هل أنت متأكد من حذف الرسائل؟"
}
},
"many" : {
"stringUnit" : {
"state" : "translated",
"value" : "هل أنت متأكد من حذف الرسائل؟"
}
},
"one" : {
"stringUnit" : {
"state" : "translated",
"value" : "هل أنت متأكد من حذف الرسالة؟"
}
},
"other" : {
"stringUnit" : {
"state" : "translated",
"value" : "هل أنت متأكد من حذف الرسائل؟"
}
},
"two" : {
"stringUnit" : {
"state" : "translated",
"value" : "هل أنت متأكد من حذف الرسائل؟"
}
},
"zero" : {
"stringUnit" : {
"state" : "translated",
"value" : "هل أنت متأكد من حذف الرسائل؟"
}
}
}
}
}
}
},
"cs" : { "cs" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
@ -140766,6 +140860,58 @@
} }
} }
}, },
"ar" : {
"stringUnit" : {
"state" : "translated",
"value" : "%#@arg1@"
},
"substitutions" : {
"arg1" : {
"argNum" : 1,
"formatSpecifier" : "lld",
"variations" : {
"plural" : {
"few" : {
"stringUnit" : {
"state" : "translated",
"value" : "فشل حذف الرسائل"
}
},
"many" : {
"stringUnit" : {
"state" : "translated",
"value" : "فشل حذف الرسائل"
}
},
"one" : {
"stringUnit" : {
"state" : "translated",
"value" : "فشل حذف الرسالة"
}
},
"other" : {
"stringUnit" : {
"state" : "translated",
"value" : "فشل حذف الرسائل"
}
},
"two" : {
"stringUnit" : {
"state" : "translated",
"value" : "فشل حذف الرسائل"
}
},
"zero" : {
"stringUnit" : {
"state" : "translated",
"value" : "فشل حذف الرسائل"
}
}
}
}
}
}
},
"az" : { "az" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
@ -143175,7 +143321,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد مسح هذه الرسائل لدى الجميع؟" "value" : "هل أنت متأكد من حذف هذه الرسائل لدى الجميع؟"
} }
}, },
"az" : { "az" : {
@ -149088,7 +149234,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "ينطبق هذا الإعداد على الجميع في هذه المحادثة.<br/>يمكن لمسؤولي المجموعة فقط تغيير هذا الإعداد." "value" : "ينطبق هذا الإعداد على الجميع في هذه المحادثة.<br/>يمكن لمشرفي المجموعة فقط تغيير هذا الإعداد."
} }
}, },
"az" : { "az" : {
@ -169283,7 +169429,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "حيوانات & و طبيعة" "value" : "حيوانات و طبيعة"
} }
}, },
"az" : { "az" : {
@ -173121,7 +173267,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد مسح كافة {emoji}؟" "value" : "هل أنت متأكد من مسح كافة {emoji}؟"
} }
}, },
"az" : { "az" : {
@ -187595,7 +187741,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متيقِّن من أنك تريد مسح <b>{group_name}؟</b> سيؤدي ذلك إلى إزالة جميع الأعضاء وحذف كافة محتويات المجموعة." "value" : "هل أنت متأكد من حذف <b>{group_name}؟</b> سيؤدي ذلك إلى إزالة جميع الأعضاء وحذف كافة محتويات المجموعة."
} }
}, },
"az" : { "az" : {
@ -188082,6 +188228,12 @@
"groupDeletedMemberDescription" : { "groupDeletedMemberDescription" : {
"extractionState" : "manual", "extractionState" : "manual",
"localizations" : { "localizations" : {
"ar" : {
"stringUnit" : {
"state" : "translated",
"value" : "{group_name} تم حذفه بواسطة مشرف المجموعة. لن تتمكن من إرسال أي رسائل أخرى."
}
},
"cs" : { "cs" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
@ -191497,7 +191649,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متأكد من أنك تريد حذف دعوة المجموعة هذه؟" "value" : "هل أنت متأكد من حذف دعوة المجموعة؟"
} }
}, },
"az" : { "az" : {
@ -209021,6 +209173,17 @@
} }
} }
}, },
"groupNotUpdatedWarning" : {
"extractionState" : "manual",
"localizations" : {
"en" : {
"stringUnit" : {
"state" : "translated",
"value" : "Group has not been updated in over 30 days. You may experience issues sending messages or viewing Group information."
}
}
}
},
"groupOnlyAdmin" : { "groupOnlyAdmin" : {
"extractionState" : "manual", "extractionState" : "manual",
"localizations" : { "localizations" : {
@ -209030,6 +209193,12 @@
"value" : "Jy is die enigste administrateur in <b>{group_name}</b>.<br/><br/>Groepslede en instellings kan nie verander word sonder 'n administrateur nie." "value" : "Jy is die enigste administrateur in <b>{group_name}</b>.<br/><br/>Groepslede en instellings kan nie verander word sonder 'n administrateur nie."
} }
}, },
"ar" : {
"stringUnit" : {
"state" : "translated",
"value" : "أنت المشرف الوحيد في\n<b>{group_name}</b>.<br/><br/>لا يمكن تغيير أعضاء المجموعة والإعدادات بدون المشرف."
}
},
"az" : { "az" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
@ -209497,11 +209666,23 @@
"groupPendingRemoval" : { "groupPendingRemoval" : {
"extractionState" : "manual", "extractionState" : "manual",
"localizations" : { "localizations" : {
"cs" : {
"stringUnit" : {
"state" : "translated",
"value" : "Čeká na odebrání"
}
},
"en" : { "en" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "Pending removal" "value" : "Pending removal"
} }
},
"nl" : {
"stringUnit" : {
"state" : "translated",
"value" : "In afwachting van verwijdering"
}
} }
} }
}, },
@ -229919,6 +230100,72 @@
} }
} }
}, },
"inviteFailed" : {
"extractionState" : "manual",
"localizations" : {
"en" : {
"stringUnit" : {
"state" : "translated",
"value" : "%#@arg1@"
},
"substitutions" : {
"arg1" : {
"argNum" : 1,
"formatSpecifier" : "lld",
"variations" : {
"plural" : {
"one" : {
"stringUnit" : {
"state" : "translated",
"value" : "Invite Failed"
}
},
"other" : {
"stringUnit" : {
"state" : "translated",
"value" : "Invites Failed"
}
}
}
}
}
}
}
}
},
"inviteFailedDescription" : {
"extractionState" : "manual",
"localizations" : {
"en" : {
"stringUnit" : {
"state" : "translated",
"value" : "%#@arg1@"
},
"substitutions" : {
"arg1" : {
"argNum" : 1,
"formatSpecifier" : "lld",
"variations" : {
"plural" : {
"one" : {
"stringUnit" : {
"state" : "translated",
"value" : "The invite could not be sent. Would you like to try again?"
}
},
"other" : {
"stringUnit" : {
"state" : "translated",
"value" : "The invites could not be sent. Would you like to try again?"
}
}
}
}
}
}
}
}
},
"join" : { "join" : {
"extractionState" : "manual", "extractionState" : "manual",
"localizations" : { "localizations" : {
@ -236613,7 +236860,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "غير قادر على تحميل معاينة الرابط" "value" : "تعذر تحميل معاينة الرابط"
} }
}, },
"az" : { "az" : {
@ -268443,7 +268690,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متأكد من أنك تريد مسح كافة طلبات الرسائل ودعوات المجموعات؟" "value" : "هل أنت متأكد من مسح كافة طلبات الرسائل ودعوات المجموعات؟"
} }
}, },
"az" : { "az" : {
@ -280417,7 +280664,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "تعيين الاسم" "value" : "تعيين الاسم المستعار"
} }
}, },
"az" : { "az" : {
@ -283770,7 +284017,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "هل أنت متأكد من أنك تريد إخفاء الملاحظة لنفسي؟" "value" : "هل أنت متأكد من إخفاء \"الملاحظة لنفسي\"؟"
} }
}, },
"az" : { "az" : {
@ -305307,7 +305554,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "{app_name} مصمم لحماية خصوصيتك." "value" : "{app_name} مُصمم لحماية خصوصيتك."
} }
}, },
"az" : { "az" : {
@ -312516,7 +312763,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "افتح" "value" : "فتح"
} }
}, },
"az" : { "az" : {
@ -330724,7 +330971,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "منح إحدى الأذونات مطلوب" "value" : "الإذن مطلوب"
} }
}, },
"az" : { "az" : {
@ -333592,7 +333839,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "ًًًُُثَبت" "value" : "تثبيت"
} }
}, },
"az" : { "az" : {
@ -334550,7 +334797,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "الغ التثبيت" "value" : "إلغاء التثبيت"
} }
}, },
"az" : { "az" : {
@ -339334,6 +339581,72 @@
} }
} }
}, },
"promotionFailed" : {
"extractionState" : "manual",
"localizations" : {
"en" : {
"stringUnit" : {
"state" : "translated",
"value" : "%#@arg1@"
},
"substitutions" : {
"arg1" : {
"argNum" : 1,
"formatSpecifier" : "lld",
"variations" : {
"plural" : {
"one" : {
"stringUnit" : {
"state" : "translated",
"value" : "Promotion Failed"
}
},
"other" : {
"stringUnit" : {
"state" : "translated",
"value" : "Promotions Failed"
}
}
}
}
}
}
}
}
},
"promotionFailedDescription" : {
"extractionState" : "manual",
"localizations" : {
"en" : {
"stringUnit" : {
"state" : "translated",
"value" : "%#@arg1@"
},
"substitutions" : {
"arg1" : {
"argNum" : 1,
"formatSpecifier" : "lld",
"variations" : {
"plural" : {
"one" : {
"stringUnit" : {
"state" : "translated",
"value" : "The promotion could not be applied. Would you like to try again?"
}
},
"other" : {
"stringUnit" : {
"state" : "translated",
"value" : "The promotions could not be applied. Would you like to try again?"
}
}
}
}
}
}
}
}
},
"qrCode" : { "qrCode" : {
"extractionState" : "manual", "extractionState" : "manual",
"localizations" : { "localizations" : {
@ -341262,7 +341575,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "عرض QR" "value" : "عرض رمز QR"
} }
}, },
"az" : { "az" : {
@ -347962,7 +348275,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "الرجاء التحقق من كلمة المرور الخاصة بالاسترداد وحاول مرة أخرى." "value" : "يرجى التحقق من كلمة مرور الاسترداد الخاصة بك وحاول مرة أخرى."
} }
}, },
"az" : { "az" : {
@ -352279,7 +352592,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "إخفاء كلمة المرور الخاصة بالاسترداد على هذا الجهاز بشكل دائم." "value" : "قم بإخفاء كلمة مرور الاسترداد بشكل دائم على هذا الجهاز."
} }
}, },
"az" : { "az" : {
@ -366286,7 +366599,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "لم يتم العثور على أي نتيجة." "value" : "لا توجد نتائج."
} }
}, },
"az" : { "az" : {
@ -368202,7 +368515,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "حدد" "value" : "تحديد"
} }
}, },
"az" : { "az" : {
@ -368681,7 +368994,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "اختر الكل" "value" : "تحديد الكل"
} }
}, },
"az" : { "az" : {
@ -370597,7 +370910,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "مظهر" "value" : "المظهر"
} }
}, },
"az" : { "az" : {
@ -376842,7 +377155,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "شارك" "value" : "مشاركة"
} }
}, },
"az" : { "az" : {
@ -379225,7 +379538,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "أظهر" "value" : "إظهار"
} }
}, },
"az" : { "az" : {
@ -383548,7 +383861,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "حاول مرة أخرى" "value" : "حاول مجدداً"
} }
}, },
"az" : { "az" : {
@ -391721,7 +392034,7 @@
"ar" : { "ar" : {
"stringUnit" : { "stringUnit" : {
"state" : "translated", "state" : "translated",
"value" : "افتح الرابط" "value" : "فتح الرابط"
} }
}, },
"az" : { "az" : {

@ -418,8 +418,10 @@ public final class FullConversationCell: UITableViewCell, SwipeActionOptimisticC
public func update(with cellViewModel: SessionThreadViewModel, using dependencies: Dependencies) { public func update(with cellViewModel: SessionThreadViewModel, using dependencies: Dependencies) {
let unreadCount: UInt = (cellViewModel.threadUnreadCount ?? 0) let unreadCount: UInt = (cellViewModel.threadUnreadCount ?? 0)
let threadIsUnread: Bool = ( let threadIsUnread: Bool = (
unreadCount > 0 || unreadCount > 0 || (
cellViewModel.threadWasMarkedUnread == true cellViewModel.threadId != cellViewModel.currentUserSessionId &&
cellViewModel.threadWasMarkedUnread == true
)
) )
let themeBackgroundColor: ThemeValue = (threadIsUnread ? let themeBackgroundColor: ThemeValue = (threadIsUnread ?
.conversationButton_unreadBackground : .conversationButton_unreadBackground :

@ -33,11 +33,16 @@ public enum ConfigurationSyncJob: JobExecutor {
return success(job, true) return success(job, true)
} }
// It's possible for multiple ConfigSyncJob's with the same target (user/group) to try to run at the /// It's possible for multiple ConfigSyncJob's with the same target (user/group) to try to run at the same time since as soon as
// same time since as soon as one is started we will enqueue a second one, rather than adding dependencies /// one is started we will enqueue a second one, rather than adding dependencies between the jobs we just continue to defer
// between the jobs we just continue to defer the subsequent job while the first one is running in /// the subsequent job while the first one is running in order to prevent multiple configurationSync jobs with the same target
// order to prevent multiple configurationSync jobs with the same target from running at the same time /// from running at the same time
///
/// **Note:** The one exception to this rule is when the job has `AdditionalSequenceRequests` because if we don't
/// run it immediately then the `AdditionalSequenceRequests` may not get run at all
guard guard
(job.transientData as? AdditionalSequenceRequests)?.beforeSequenceRequests.isEmpty == false ||
(job.transientData as? AdditionalSequenceRequests)?.afterSequenceRequests.isEmpty == false ||
dependencies[singleton: .jobRunner] dependencies[singleton: .jobRunner]
.jobInfoFor(state: .running, variant: .configurationSync) .jobInfoFor(state: .running, variant: .configurationSync)
.filter({ key, info in .filter({ key, info in
@ -73,9 +78,14 @@ public enum ConfigurationSyncJob: JobExecutor {
return failure(job, StorageError.generic, false) return failure(job, StorageError.generic, false)
} }
// If there are no pending changes then the job can just complete (next time something /// If there is no `pushData`, `obsoleteHashes` or additional sequence requests then the job can just complete (next time
// is updated we want to try and run immediately so don't scuedule another run in this case) /// something is updated we want to try and run immediately so don't scuedule another run in this case)
guard !pendingChanges.pushData.isEmpty || !pendingChanges.obsoleteHashes.isEmpty else { guard
!pendingChanges.pushData.isEmpty ||
!pendingChanges.obsoleteHashes.isEmpty ||
(job.transientData as? AdditionalSequenceRequests)?.beforeSequenceRequests.isEmpty == false ||
(job.transientData as? AdditionalSequenceRequests)?.afterSequenceRequests.isEmpty == false
else {
Log.info(.cat, "For \(swarmPublicKey) completed with no pending changes") Log.info(.cat, "For \(swarmPublicKey) completed with no pending changes")
return success(job, true) return success(job, true)
} }

@ -18,6 +18,7 @@ extension MessageReceiver {
guard proto.hasExpirationType || proto.hasExpirationTimer else { return } guard proto.hasExpirationType || proto.hasExpirationTimer else { return }
guard guard
threadVariant != .community, threadVariant != .community,
threadVariant != .group, // Handled via the GROUP_INFO config instead
let sender: String = message.sender, let sender: String = message.sender,
let timestampMs: UInt64 = message.sentTimestampMs let timestampMs: UInt64 = message.sentTimestampMs
else { return } else { return }

@ -27,10 +27,10 @@ public extension MentionInfo {
threadId: String, threadId: String,
threadVariant: SessionThread.Variant, threadVariant: SessionThread.Variant,
targetPrefixes: [SessionId.Prefix], targetPrefixes: [SessionId.Prefix],
currentUserBlinded15SessionId: String?,
currentUserBlinded25SessionId: String?,
pattern: FTS5Pattern? pattern: FTS5Pattern?
) -> AdaptedFetchRequest<SQLRequest<MentionInfo>>? { ) -> AdaptedFetchRequest<SQLRequest<MentionInfo>>? {
guard threadVariant != .contact || userPublicKey != threadId else { return nil }
let profile: TypedTableAlias<Profile> = TypedTableAlias() let profile: TypedTableAlias<Profile> = TypedTableAlias()
let interaction: TypedTableAlias<Interaction> = TypedTableAlias() let interaction: TypedTableAlias<Interaction> = TypedTableAlias()
let openGroup: TypedTableAlias<OpenGroup> = TypedTableAlias() let openGroup: TypedTableAlias<OpenGroup> = TypedTableAlias()
@ -40,6 +40,11 @@ public extension MentionInfo {
.map { SQL("\(profile[.id]) LIKE '\(SQL(stringLiteral: "\($0.rawValue)%"))'") } .map { SQL("\(profile[.id]) LIKE '\(SQL(stringLiteral: "\($0.rawValue)%"))'") }
.joined(operator: .or) .joined(operator: .or)
let profileFullTextSearch: SQL = SQL(stringLiteral: Profile.fullTextSearchTableName) let profileFullTextSearch: SQL = SQL(stringLiteral: Profile.fullTextSearchTableName)
let currentUserIds: Set<String> = [
userPublicKey,
currentUserBlinded15SessionId,
currentUserBlinded25SessionId
].compactMap { $0 }.asSet()
/// The query needs to differ depending on the thread variant because the behaviour should be different: /// The query needs to differ depending on the thread variant because the behaviour should be different:
/// ///
@ -55,10 +60,8 @@ public extension MentionInfo {
FROM \(profileFullTextSearch) FROM \(profileFullTextSearch)
JOIN \(Profile.self) ON ( JOIN \(Profile.self) ON (
\(Profile.self).rowid = \(profileFullTextSearch).rowid AND \(Profile.self).rowid = \(profileFullTextSearch).rowid AND
\(SQL("\(profile[.id]) != \(userPublicKey)")) AND ( \(SQL("\(threadVariant) != \(SessionThread.Variant.community)")) OR
\(SQL("\(threadVariant) != \(SessionThread.Variant.community)")) OR \(prefixesLiteral)
\(prefixesLiteral)
)
) )
""" """
}() }()
@ -66,10 +69,8 @@ public extension MentionInfo {
guard let pattern: FTS5Pattern = pattern, pattern.rawPattern != "\"\"*" else { guard let pattern: FTS5Pattern = pattern, pattern.rawPattern != "\"\"*" else {
return """ return """
WHERE ( WHERE (
\(SQL("\(profile[.id]) != \(userPublicKey)")) AND ( \(SQL("\(threadVariant) != \(SessionThread.Variant.community)")) OR
\(SQL("\(threadVariant) != \(SessionThread.Variant.community)")) OR \(prefixesLiteral)
\(prefixesLiteral)
)
) )
""" """
} }
@ -87,7 +88,11 @@ public extension MentionInfo {
\(SQL("\(threadVariant) AS \(MentionInfo.Columns.threadVariant)")) \(SQL("\(threadVariant) AS \(MentionInfo.Columns.threadVariant)"))
\(targetJoin) \(targetJoin)
\(targetWhere) AND \(SQL("\(profile[.id]) = \(threadId)")) \(targetWhere) AND (
\(SQL("\(profile[.id]) = \(threadId)")) OR
\(SQL("\(profile[.id]) IN \(currentUserIds)"))
)
ORDER BY \(SQL("\(profile[.id]) IN \(currentUserIds)")) DESC
""") """)
case .legacyGroup, .group: case .legacyGroup, .group:
@ -104,7 +109,9 @@ public extension MentionInfo {
) )
\(targetWhere) \(targetWhere)
GROUP BY \(profile[.id]) GROUP BY \(profile[.id])
ORDER BY IFNULL(\(profile[.nickname]), \(profile[.name])) ASC ORDER BY
\(SQL("\(profile[.id]) IN \(currentUserIds)")) DESC,
IFNULL(\(profile[.nickname]), \(profile[.name])) ASC
""") """)
case .community: case .community:
@ -124,7 +131,9 @@ public extension MentionInfo {
JOIN \(OpenGroup.self) ON \(SQL("\(openGroup[.threadId]) = \(threadId)")) JOIN \(OpenGroup.self) ON \(SQL("\(openGroup[.threadId]) = \(threadId)"))
\(targetWhere) \(targetWhere)
GROUP BY \(profile[.id]) GROUP BY \(profile[.id])
ORDER BY \(interaction[.timestampMs].desc) ORDER BY
\(SQL("\(profile[.id]) IN \(currentUserIds)")) DESC,
\(interaction[.timestampMs].desc)
LIMIT 20 LIMIT 20
""") """)
} }

@ -402,39 +402,26 @@ public struct MessageViewModel: FetchableRecordWithRowId, Decodable, Equatable,
) )
}() }()
let (positionInCluster, isOnlyMessageInCluster): (Position, Bool) = { let (positionInCluster, isOnlyMessageInCluster): (Position, Bool) = {
let isSenderVariantConsistent: Bool = (
(
(
self.variant == .standardOutgoing ||
self.variant == .standardOutgoingDeleted ||
self.variant == .standardOutgoingDeletedLocally
) && (
prevModel?.variant != .standardOutgoing ||
prevModel?.variant != .standardOutgoingDeleted ||
prevModel?.variant != .standardOutgoingDeletedLocally
)
) || (
(
self.variant == .standardIncoming ||
self.variant == .standardIncomingDeleted ||
self.variant == .standardIncomingDeletedLocally
) && (
prevModel?.variant != .standardIncoming &&
prevModel?.variant != .standardIncomingDeleted &&
prevModel?.variant != .standardIncomingDeletedLocally
)
)
)
let isFirstInCluster: Bool = ( let isFirstInCluster: Bool = (
prevModel == nil || prevModel == nil ||
shouldShowDateBeforeThisModel || shouldShowDateBeforeThisModel || (
isSenderVariantConsistent || self.variant.isOutgoing &&
prevModel?.variant.isOutgoing != true
) || (
self.variant.isIncoming &&
prevModel?.variant.isIncoming != true
) ||
self.authorId != prevModel?.authorId self.authorId != prevModel?.authorId
) )
let isLastInCluster: Bool = ( let isLastInCluster: Bool = (
nextModel == nil || nextModel == nil ||
shouldShowDateBeforeNextModel || shouldShowDateBeforeNextModel || (
isSenderVariantConsistent || self.variant.isOutgoing &&
prevModel?.variant.isOutgoing != true
) || (
self.variant.isIncoming &&
prevModel?.variant.isIncoming != true
) ||
self.authorId != nextModel?.authorId self.authorId != nextModel?.authorId
) )

@ -6,7 +6,7 @@ import Foundation
public final class Features { public final class Features {
public static let legacyGroupDepricationDate: Date = Date.distantFuture // TODO: [GROUPS REBUILD] Set this date public static let legacyGroupDepricationDate: Date = Date.distantFuture // TODO: [GROUPS REBUILD] Set this date
public static let legacyGroupDepricationUrl: String = "https://getsession.org/faq" // TODO: [GROUPS REBUILD] Set this url public static let legacyGroupDepricationUrl: String = "https://getsession.org/groups"
} }
public extension FeatureStorage { public extension FeatureStorage {

Loading…
Cancel
Save