mirror of
https://github.com/simplex-chat/simplex-chat.git
synced 2026-04-19 13:05:54 +00:00
* light github image for dark mode * show message received status, remove chevrons in chat list * show unread message status * add message send error mark * refactor alerts to use AlertManager * show alert message on tapping undelivered message, simplify text-only alerts
150 lines
5.7 KiB
Swift
150 lines
5.7 KiB
Swift
//
|
|
// TextItemView.swift
|
|
// SimpleX
|
|
//
|
|
// Created by Evgeny Poberezkin on 04/02/2022.
|
|
// Copyright © 2022 SimpleX Chat. All rights reserved.
|
|
//
|
|
|
|
import SwiftUI
|
|
|
|
private let emailRegex = try! NSRegularExpression(pattern: "^[a-z0-9.!#$%&'*+/=?^_`{|}~-]+@[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?(?:\\.[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?)*$", options: .caseInsensitive)
|
|
|
|
private let phoneRegex = try! NSRegularExpression(pattern: "^\\+?[0-9\\.\\(\\)\\-]{7,20}$")
|
|
|
|
private let sentColorLight = Color(.sRGB, red: 0.27, green: 0.72, blue: 1, opacity: 0.12)
|
|
private let sentColorDark = Color(.sRGB, red: 0.27, green: 0.72, blue: 1, opacity: 0.17)
|
|
private let linkColor = UIColor(red: 0, green: 0.533, blue: 1, alpha: 1)
|
|
|
|
struct TextItemView: View {
|
|
@Environment(\.colorScheme) var colorScheme
|
|
var chatItem: ChatItem
|
|
var width: CGFloat
|
|
private let codeFont = Font.custom("Courier", size: UIFont.preferredFont(forTextStyle: .body).pointSize)
|
|
|
|
var body: some View {
|
|
let sent = chatItem.chatDir.sent
|
|
let maxWidth = width * 0.78
|
|
|
|
return ZStack(alignment: .bottomTrailing) {
|
|
(messageText(chatItem) + reserveSpaceForMeta(chatItem.timestampText))
|
|
.padding(.vertical, 6)
|
|
.padding(.horizontal, 12)
|
|
.frame(minWidth: 0, alignment: .leading)
|
|
.textSelection(.enabled)
|
|
|
|
CIMetaView(chatItem: chatItem)
|
|
.padding(.trailing, 12)
|
|
.padding(.bottom, 6)
|
|
}
|
|
.background(
|
|
sent
|
|
? (colorScheme == .light ? sentColorLight : sentColorDark)
|
|
: Color(uiColor: .tertiarySystemGroupedBackground)
|
|
)
|
|
.cornerRadius(18)
|
|
.padding(.horizontal)
|
|
.frame(
|
|
maxWidth: maxWidth,
|
|
maxHeight: .infinity,
|
|
alignment: sent ? .trailing : .leading
|
|
)
|
|
.onTapGesture {
|
|
switch chatItem.meta.itemStatus {
|
|
case .sndErrorAuth: msgDeliveryError("Most likely this contact has deleted the connection with you.")
|
|
case let .sndError(agentError): msgDeliveryError("Unexpected error: \(String(describing: agentError))")
|
|
default: return
|
|
}
|
|
}
|
|
}
|
|
|
|
private func messageText(_ chatItem: ChatItem) -> Text {
|
|
let s = chatItem.content.text
|
|
var res: Text
|
|
if s == "" {
|
|
res = Text("")
|
|
} else {
|
|
let parts = s.split(separator: " ")
|
|
res = wordToText(parts[0])
|
|
var i = 1
|
|
while i < parts.count {
|
|
res = res + Text(" ") + wordToText(parts[i])
|
|
i = i + 1
|
|
}
|
|
}
|
|
if case let .groupRcv(groupMember) = chatItem.chatDir {
|
|
let member = Text(groupMember.memberProfile.displayName).font(.headline)
|
|
return member + Text(": ") + res
|
|
} else {
|
|
return res
|
|
}
|
|
}
|
|
|
|
private func reserveSpaceForMeta(_ meta: String) -> Text {
|
|
Text(" \(meta)")
|
|
.font(.caption)
|
|
.foregroundColor(.clear)
|
|
}
|
|
|
|
private func wordToText(_ s: String.SubSequence) -> Text {
|
|
let str = String(s)
|
|
switch true {
|
|
case s.starts(with: "http://") || s.starts(with: "https://"):
|
|
return linkText(str, prefix: "")
|
|
case match(str, emailRegex):
|
|
return linkText(str, prefix: "mailto:")
|
|
case match(str, phoneRegex):
|
|
return linkText(str, prefix: "tel:")
|
|
default:
|
|
if (s.count > 1) {
|
|
switch true {
|
|
case s.first == "*" && s.last == "*": return mdText(s).bold()
|
|
case s.first == "_" && s.last == "_": return mdText(s).italic()
|
|
case s.first == "+" && s.last == "+": return mdText(s).underline()
|
|
case s.first == "~" && s.last == "~": return mdText(s).strikethrough()
|
|
default: return Text(s)
|
|
}
|
|
} else {
|
|
return Text(s)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func match(_ s: String, _ regex: NSRegularExpression) -> Bool {
|
|
regex.firstMatch(in: s, options: [], range: NSRange(location: 0, length: s.count)) != nil
|
|
}
|
|
|
|
private func linkText(_ s: String, prefix: String) -> Text {
|
|
Text(AttributedString(s, attributes: AttributeContainer([
|
|
.link: NSURL(string: prefix + s) as Any,
|
|
.foregroundColor: linkColor as Any
|
|
]))).underline()
|
|
}
|
|
|
|
private func mdText(_ s: String.SubSequence) -> Text {
|
|
Text(s[s.index(s.startIndex, offsetBy: 1)..<s.index(s.endIndex, offsetBy: -1)])
|
|
}
|
|
|
|
private func msgDeliveryError(_ err: String) {
|
|
AlertManager.shared.showAlertMsg(
|
|
title: "Message delivery error",
|
|
message: err
|
|
)
|
|
}
|
|
}
|
|
|
|
struct TextItemView_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
Group{
|
|
TextItemView(chatItem: ChatItem.getSample(1, .directSnd, .now, "hello"), width: 360)
|
|
TextItemView(chatItem: ChatItem.getSample(1, .groupRcv(groupMember: GroupMember.sampleData), .now, "hello"), width: 360)
|
|
TextItemView(chatItem: ChatItem.getSample(2, .directSnd, .now, "https://simplex.chat", .sndSent), width: 360)
|
|
TextItemView(chatItem: ChatItem.getSample(2, .directRcv, .now, "hello there too!!! this covers -"), width: 360)
|
|
TextItemView(chatItem: ChatItem.getSample(2, .directRcv, .now, "hello there too!!! this text has the time on the same line "), width: 360)
|
|
TextItemView(chatItem: ChatItem.getSample(2, .directRcv, .now, "https://simplex.chat"), width: 360)
|
|
TextItemView(chatItem: ChatItem.getSample(2, .directRcv, .now, "chaT@simplex.chat"), width: 360)
|
|
}
|
|
.previewLayout(.fixed(width: 360, height: 70))
|
|
}
|
|
}
|