mirror of
https://github.com/simplex-chat/simplex-chat.git
synced 2026-04-26 19:35:48 +00:00
f5eea018d9
* ios: wallpapers (#4304) * ios: wallpapers * theme selection * applied theme colors and preset wallpaper * more places with background * one more * accent color * defaults * rename * background * no change to cell color * unneeded * changes * no global tint * defaults * removed unneeded class * for merging * ios: wallpapers types (#4325) * types and api * divided types per target * creating directory for wallpapers * creating wallpaper dir at launch * ios: wallpapers appearance (#4335) * appearance * changes * refactor * scale * lambda to function --------- Co-authored-by: Evgeny Poberezkin <2769109+epoberezkin@users.noreply.github.com> * ios: wallpapers user/chat overrides (#4345) * ios: wallpapers user/chat overrides * chat overrides * color picker updates colors correctly * fix state update * labels * background for light theme * small optimization * removed commented code * ios: enhancements to wallpapers (#4361) * ios: enhancements to wallpapers * colors for background * ios: wallpapers import/export (#4362) * ios: wallpapers import/export * comment * ios: wallpapers theme updates (#4365) * ios: wallpapers theme updates * group member background * colors * profile picture colors * unneeded * optimizations, images, state fixes * fixes * no editing of title color * rename Menus and alerts, refactor * tint applying fix * fixes * migration of accent and themes * fix updating system theme * migration changes * limiting color range * ios: wallpapers rename enum (#4384) * ios: wallpapers rename enum2 (#4385) * ios: wallpapers rename enum2 * change * colors were commented * fix build and look --------- Co-authored-by: Stanislav Dmitrenko <7953703+avently@users.noreply.github.com>
193 lines
7.0 KiB
Swift
193 lines
7.0 KiB
Swift
//
|
|
// ChatItemForwardingView.swift
|
|
// SimpleX (iOS)
|
|
//
|
|
// Created by spaced4ndy on 12.04.2024.
|
|
// Copyright © 2024 SimpleX Chat. All rights reserved.
|
|
//
|
|
|
|
import SwiftUI
|
|
import SimpleXChat
|
|
|
|
struct ChatItemForwardingView: View {
|
|
@EnvironmentObject var chatModel: ChatModel
|
|
@EnvironmentObject var theme: AppTheme
|
|
@Environment(\.dismiss) var dismiss
|
|
|
|
var ci: ChatItem
|
|
var fromChatInfo: ChatInfo
|
|
@Binding var composeState: ComposeState
|
|
|
|
@State private var searchText: String = ""
|
|
@FocusState private var searchFocused
|
|
@State private var alert: SomeAlert?
|
|
@State private var hasSimplexLink_: Bool?
|
|
private let chatsToForwardTo = filterChatsToForwardTo()
|
|
|
|
var body: some View {
|
|
NavigationView {
|
|
forwardListView()
|
|
.toolbar {
|
|
ToolbarItem(placement: .navigationBarLeading) {
|
|
Button("Cancel") {
|
|
dismiss()
|
|
}
|
|
}
|
|
ToolbarItem(placement: .principal) {
|
|
Text("Forward")
|
|
.bold()
|
|
}
|
|
}
|
|
}
|
|
.modifier(ThemedBackground())
|
|
.alert(item: $alert) { $0.alert }
|
|
}
|
|
|
|
@ViewBuilder private func forwardListView() -> some View {
|
|
VStack(alignment: .leading) {
|
|
if !chatsToForwardTo.isEmpty {
|
|
List {
|
|
searchFieldView(text: $searchText, focussed: $searchFocused, theme.colors.onBackground, theme.colors.secondary)
|
|
.padding(.leading, 2)
|
|
let s = searchText.trimmingCharacters(in: .whitespaces).localizedLowercase
|
|
let chats = s == "" ? chatsToForwardTo : chatsToForwardTo.filter { foundChat($0, s) }
|
|
ForEach(chats) { chat in
|
|
forwardListChatView(chat)
|
|
.disabled(chatModel.deletedChats.contains(chat.chatInfo.id))
|
|
}
|
|
}
|
|
.modifier(ThemedBackground(grouped: true))
|
|
} else {
|
|
ZStack {
|
|
emptyList()
|
|
}
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
|
.modifier(ThemedBackground())
|
|
}
|
|
}
|
|
}
|
|
|
|
private func foundChat(_ chat: Chat, _ searchStr: String) -> Bool {
|
|
let cInfo = chat.chatInfo
|
|
return switch cInfo {
|
|
case let .direct(contact):
|
|
viewNameContains(cInfo, searchStr) ||
|
|
contact.profile.displayName.localizedLowercase.contains(searchStr) ||
|
|
contact.fullName.localizedLowercase.contains(searchStr)
|
|
default:
|
|
viewNameContains(cInfo, searchStr)
|
|
}
|
|
|
|
func viewNameContains(_ cInfo: ChatInfo, _ s: String) -> Bool {
|
|
cInfo.chatViewName.localizedLowercase.contains(s)
|
|
}
|
|
}
|
|
|
|
private func prohibitedByPref(_ chat: Chat) -> Bool {
|
|
// preference checks should match checks in compose view
|
|
let simplexLinkProhibited = hasSimplexLink && !chat.groupFeatureEnabled(.simplexLinks)
|
|
let fileProhibited = (ci.content.msgContent?.isMediaOrFileAttachment ?? false) && !chat.groupFeatureEnabled(.files)
|
|
let voiceProhibited = (ci.content.msgContent?.isVoice ?? false) && !chat.chatInfo.featureEnabled(.voice)
|
|
return switch chat.chatInfo {
|
|
case .direct: voiceProhibited
|
|
case .group: simplexLinkProhibited || fileProhibited || voiceProhibited
|
|
case .local: false
|
|
case .contactRequest: false
|
|
case .contactConnection: false
|
|
case .invalidJSON: false
|
|
}
|
|
}
|
|
|
|
private var hasSimplexLink: Bool {
|
|
if let hasSimplexLink_ { return hasSimplexLink_ }
|
|
let r =
|
|
if let mcText = ci.content.msgContent?.text,
|
|
let parsedMsg = parseSimpleXMarkdown(mcText) {
|
|
parsedMsgHasSimplexLink(parsedMsg)
|
|
} else {
|
|
false
|
|
}
|
|
hasSimplexLink_ = r
|
|
return r
|
|
}
|
|
|
|
private func emptyList() -> some View {
|
|
Text("No filtered chats")
|
|
.foregroundColor(theme.colors.secondary)
|
|
.frame(maxWidth: .infinity)
|
|
}
|
|
|
|
@ViewBuilder private func forwardListChatView(_ chat: Chat) -> some View {
|
|
let prohibited = prohibitedByPref(chat)
|
|
Button {
|
|
if prohibited {
|
|
alert = SomeAlert(
|
|
alert: mkAlert(
|
|
title: "Cannot forward message",
|
|
message: "Selected chat preferences prohibit this message."
|
|
),
|
|
id: "forward prohibited by preferences"
|
|
)
|
|
} else {
|
|
dismiss()
|
|
if chat.id == fromChatInfo.id {
|
|
composeState = ComposeState(
|
|
message: composeState.message,
|
|
preview: composeState.linkPreview != nil ? composeState.preview : .noPreview,
|
|
contextItem: .forwardingItem(chatItem: ci, fromChatInfo: fromChatInfo)
|
|
)
|
|
} else {
|
|
composeState = ComposeState.init(forwardingItem: ci, fromChatInfo: fromChatInfo)
|
|
chatModel.chatId = chat.id
|
|
}
|
|
}
|
|
} label: {
|
|
HStack {
|
|
ChatInfoImage(chat: chat, size: 30)
|
|
.padding(.trailing, 2)
|
|
Text(chat.chatInfo.chatViewName)
|
|
.foregroundColor(prohibited ? theme.colors.secondary : theme.colors.onBackground)
|
|
.lineLimit(1)
|
|
if chat.chatInfo.incognito {
|
|
Spacer()
|
|
Image(systemName: "theatermasks")
|
|
.resizable()
|
|
.scaledToFit()
|
|
.frame(width: 22, height: 22)
|
|
.foregroundColor(theme.colors.secondary)
|
|
}
|
|
}
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func filterChatsToForwardTo() -> [Chat] {
|
|
var filteredChats = ChatModel.shared.chats.filter { c in
|
|
c.chatInfo.chatType != .local && canForwardToChat(c)
|
|
}
|
|
if let privateNotes = ChatModel.shared.chats.first(where: { $0.chatInfo.chatType == .local }) {
|
|
filteredChats.insert(privateNotes, at: 0)
|
|
}
|
|
return filteredChats
|
|
}
|
|
|
|
private func canForwardToChat(_ chat: Chat) -> Bool {
|
|
switch chat.chatInfo {
|
|
case let .direct(contact): contact.sendMsgEnabled && !contact.nextSendGrpInv
|
|
case let .group(groupInfo): groupInfo.sendMsgEnabled
|
|
case let .local(noteFolder): noteFolder.sendMsgEnabled
|
|
case .contactRequest: false
|
|
case .contactConnection: false
|
|
case .invalidJSON: false
|
|
}
|
|
}
|
|
|
|
#Preview {
|
|
ChatItemForwardingView(
|
|
ci: ChatItem.getSample(1, .directSnd, .now, "hello"),
|
|
fromChatInfo: .direct(contact: Contact.sampleData),
|
|
composeState: Binding.constant(ComposeState(message: "hello"))
|
|
).environmentObject(CurrentColors.toAppTheme())
|
|
}
|