mirror of
https://github.com/simplex-chat/simplex-chat.git
synced 2026-04-08 01:55:59 +00:00
* 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>
175 lines
5.9 KiB
Swift
175 lines
5.9 KiB
Swift
//
|
|
// MsgContentView.swift
|
|
// SimpleX
|
|
//
|
|
// Created by Evgeny on 13/03/2022.
|
|
// Copyright © 2022 SimpleX Chat. All rights reserved.
|
|
//
|
|
|
|
import SwiftUI
|
|
import SimpleXChat
|
|
|
|
let uiLinkColor = UIColor(red: 0, green: 0.533, blue: 1, alpha: 1)
|
|
|
|
private let noTyping = Text(" ")
|
|
|
|
private let typingIndicators: [Text] = [
|
|
(typing(.black) + typing() + typing()),
|
|
(typing(.bold) + typing(.black) + typing()),
|
|
(typing() + typing(.bold) + typing(.black)),
|
|
(typing() + typing() + typing(.bold))
|
|
]
|
|
|
|
private func typing(_ w: Font.Weight = .light) -> Text {
|
|
Text(".").fontWeight(w)
|
|
}
|
|
|
|
struct MsgContentView: View {
|
|
@ObservedObject var chat: Chat
|
|
@EnvironmentObject var theme: AppTheme
|
|
var text: String
|
|
var formattedText: [FormattedText]? = nil
|
|
var sender: String? = nil
|
|
var meta: CIMeta? = nil
|
|
var rightToLeft = false
|
|
var showSecrets: Bool
|
|
@State private var typingIdx = 0
|
|
@State private var timer: Timer?
|
|
|
|
@AppStorage(DEFAULT_SHOW_SENT_VIA_RPOXY) private var showSentViaProxy = false
|
|
|
|
var body: some View {
|
|
if meta?.isLive == true {
|
|
msgContentView()
|
|
.onAppear { switchTyping() }
|
|
.onDisappear(perform: stopTyping)
|
|
.onChange(of: meta?.isLive, perform: switchTyping)
|
|
.onChange(of: meta?.recent, perform: switchTyping)
|
|
} else {
|
|
msgContentView()
|
|
}
|
|
}
|
|
|
|
private func switchTyping(_: Bool? = nil) {
|
|
if let meta = meta, meta.isLive && meta.recent {
|
|
timer = timer ?? Timer.scheduledTimer(withTimeInterval: 0.25, repeats: true) { _ in
|
|
typingIdx = (typingIdx + 1) % typingIndicators.count
|
|
}
|
|
} else {
|
|
stopTyping()
|
|
}
|
|
}
|
|
|
|
private func stopTyping() {
|
|
timer?.invalidate()
|
|
timer = nil
|
|
}
|
|
|
|
private func msgContentView() -> Text {
|
|
var v = messageText(text, formattedText, sender, showSecrets: showSecrets, secondaryColor: theme.colors.secondary)
|
|
if let mt = meta {
|
|
if mt.isLive {
|
|
v = v + typingIndicator(mt.recent)
|
|
}
|
|
v = v + reserveSpaceForMeta(mt)
|
|
}
|
|
return v
|
|
}
|
|
|
|
private func typingIndicator(_ recent: Bool) -> Text {
|
|
return (recent ? typingIndicators[typingIdx] : noTyping)
|
|
.font(.body.monospaced())
|
|
.kerning(-2)
|
|
.foregroundColor(theme.colors.secondary)
|
|
}
|
|
|
|
private func reserveSpaceForMeta(_ mt: CIMeta) -> Text {
|
|
(rightToLeft ? Text("\n") : Text(" ")) + ciMetaText(mt, chatTTL: chat.chatInfo.timedMessagesTTL, encrypted: nil, transparent: true, showViaProxy: showSentViaProxy)
|
|
}
|
|
}
|
|
|
|
func messageText(_ text: String, _ formattedText: [FormattedText]?, _ sender: String?, icon: String? = nil, preview: Bool = false, showSecrets: Bool, secondaryColor: Color) -> Text {
|
|
let s = text
|
|
var res: Text
|
|
if let ft = formattedText, ft.count > 0 && ft.count <= 200 {
|
|
res = formatText(ft[0], preview, showSecret: showSecrets)
|
|
var i = 1
|
|
while i < ft.count {
|
|
res = res + formatText(ft[i], preview, showSecret: showSecrets)
|
|
i = i + 1
|
|
}
|
|
} else {
|
|
res = Text(s)
|
|
}
|
|
|
|
if let i = icon {
|
|
res = Text(Image(systemName: i)).foregroundColor(secondaryColor) + Text(" ") + res
|
|
}
|
|
|
|
if let s = sender {
|
|
let t = Text(s)
|
|
return (preview ? t : t.fontWeight(.medium)) + Text(": ") + res
|
|
} else {
|
|
return res
|
|
}
|
|
}
|
|
|
|
private func formatText(_ ft: FormattedText, _ preview: Bool, showSecret: Bool) -> Text {
|
|
let t = ft.text
|
|
if let f = ft.format {
|
|
switch (f) {
|
|
case .bold: return Text(t).bold()
|
|
case .italic: return Text(t).italic()
|
|
case .strikeThrough: return Text(t).strikethrough()
|
|
case .snippet: return Text(t).font(.body.monospaced())
|
|
case .secret: return
|
|
showSecret
|
|
? Text(t)
|
|
: Text(AttributedString(t, attributes: AttributeContainer([
|
|
.foregroundColor: UIColor.clear as Any,
|
|
.backgroundColor: UIColor.secondarySystemFill as Any
|
|
])))
|
|
case let .colored(color): return Text(t).foregroundColor(color.uiColor)
|
|
case .uri: return linkText(t, t, preview, prefix: "")
|
|
case let .simplexLink(linkType, simplexUri, smpHosts):
|
|
switch privacySimplexLinkModeDefault.get() {
|
|
case .description: return linkText(simplexLinkText(linkType, smpHosts), simplexUri, preview, prefix: "")
|
|
case .full: return linkText(t, simplexUri, preview, prefix: "")
|
|
case .browser: return linkText(t, simplexUri, preview, prefix: "")
|
|
}
|
|
case .email: return linkText(t, t, preview, prefix: "mailto:")
|
|
case .phone: return linkText(t, t.replacingOccurrences(of: " ", with: ""), preview, prefix: "tel:")
|
|
}
|
|
} else {
|
|
return Text(t)
|
|
}
|
|
}
|
|
|
|
private func linkText(_ s: String, _ link: String, _ preview: Bool, prefix: String, color: Color = Color(uiColor: uiLinkColor), uiColor: UIColor = uiLinkColor) -> Text {
|
|
preview
|
|
? Text(s).foregroundColor(color).underline(color: color)
|
|
: Text(AttributedString(s, attributes: AttributeContainer([
|
|
.link: NSURL(string: prefix + link) as Any,
|
|
.foregroundColor: uiColor as Any
|
|
]))).underline()
|
|
}
|
|
|
|
func simplexLinkText(_ linkType: SimplexLinkType, _ smpHosts: [String]) -> String {
|
|
linkType.description + " " + "(via \(smpHosts.first ?? "?"))"
|
|
}
|
|
|
|
struct MsgContentView_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
let chatItem = ChatItem.getSample(1, .directSnd, .now, "hello")
|
|
return MsgContentView(
|
|
chat: Chat.sampleData,
|
|
text: chatItem.text,
|
|
formattedText: chatItem.formattedText,
|
|
sender: chatItem.memberDisplayName,
|
|
meta: chatItem.meta,
|
|
showSecrets: false
|
|
)
|
|
.environmentObject(Chat.sampleData)
|
|
}
|
|
}
|