/* * Copyright (c) 2010-2023 Belledonne Communications SARL. * * This file is part of linphone-iphone * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ import SwiftUI import linphonesw // swiftlint:disable type_body_length struct StartConversationFragment: View { @ObservedObject var contactsManager = ContactsManager.shared @ObservedObject var magicSearch = MagicSearchSingleton.shared @StateObject private var startConversationViewModel = StartConversationViewModel() @EnvironmentObject var conversationsListViewModel: ConversationsListViewModel @Binding var isShowStartConversationFragment: Bool @FocusState var isSearchFieldFocused: Bool @State private var delayedColor = Color.white @State var operationInProgress: Bool = false var body: some View { NavigationView { ZStack { VStack(spacing: 1) { Rectangle() .foregroundColor(delayedColor) .edgesIgnoringSafeArea(.top) .frame(height: 0) .task(delayColor) HStack { Image("caret-left") .renderingMode(.template) .resizable() .foregroundStyle(Color.orangeMain500) .frame(width: 25, height: 25, alignment: .leading) .padding(.all, 10) .padding(.top, 2) .padding(.leading, -10) .onTapGesture { startConversationViewModel.searchField = "" magicSearch.currentFilter = "" magicSearch.searchForContacts() delayColorDismiss() withAnimation { isShowStartConversationFragment = false } } Text("new_conversation_title") .multilineTextAlignment(.leading) .default_text_style_orange_800(styleSize: 16) Spacer() } .frame(maxWidth: .infinity) .frame(height: 50) .padding(.horizontal) .padding(.bottom, 4) .background(.white) VStack(spacing: 0) { ZStack(alignment: .trailing) { TextField("history_call_start_search_bar_filter_hint", text: $startConversationViewModel.searchField) .default_text_style(styleSize: 15) .frame(height: 25) .focused($isSearchFieldFocused) .padding(.horizontal, 30) .onChange(of: startConversationViewModel.searchField) { newValue in magicSearch.currentFilter = newValue magicSearch.searchForContacts() } HStack { Button(action: { }, label: { Image("magnifying-glass") .renderingMode(.template) .resizable() .foregroundStyle(Color.grayMain2c500) .frame(width: 25, height: 25) }) Spacer() if !startConversationViewModel.searchField.isEmpty { Button(action: { startConversationViewModel.searchField = "" magicSearch.currentFilter = "" magicSearch.searchForContacts() }, label: { Image("x") .renderingMode(.template) .resizable() .foregroundStyle(Color.grayMain2c500) .frame(width: 25, height: 25) }) } } } .padding(.horizontal, 15) .padding(.vertical, 10) .cornerRadius(60) .overlay( RoundedRectangle(cornerRadius: 60) .inset(by: 0.5) .stroke(isSearchFieldFocused ? Color.orangeMain500 : Color.gray200, lineWidth: 1) ) .padding(.vertical) .padding(.horizontal) if !startConversationViewModel.hideGroupChatButton { NavigationLink(destination: { StartGroupConversationFragment(isShowStartConversationFragment: $isShowStartConversationFragment) .environmentObject(startConversationViewModel) }, label: { HStack { HStack(alignment: .center) { Image("users-three") .renderingMode(.template) .resizable() .foregroundStyle(.white) .frame(width: 28, height: 28) } .padding(10) .background(Color.orangeMain500) .cornerRadius(40) Text("new_conversation_create_group") .foregroundStyle(.black) .default_text_style_800(styleSize: 16) .lineLimit(1) Spacer() Image("caret-right") .renderingMode(.template) .resizable() .foregroundStyle(Color.grayMain2c500) .frame(width: 25, height: 25, alignment: .leading) } }) .padding(.vertical, 10) .padding(.horizontal, 20) .background( LinearGradient(gradient: Gradient(colors: [.grayMain2c100, .white]), startPoint: .leading, endPoint: .trailing) .padding(.vertical, 10) .padding(.horizontal, 40) ) } ZStack { ScrollView { if !ContactsManager.shared.lastSearch.isEmpty { HStack(alignment: .center) { Text("contacts_list_all_contacts_title") .default_text_style_800(styleSize: 16) Spacer() } .padding(.vertical, 10) .padding(.horizontal, 16) } ContactsListFragment(showingSheet: .constant(false), startCallFunc: { addr in startConversationViewModel.createOneToOneChatRoomWith(remote: addr) }) .padding(.horizontal, 16) if !contactsManager.lastSearchSuggestions.isEmpty { HStack(alignment: .center) { Text("generic_address_picker_suggestions_list_title") .default_text_style_800(styleSize: 16) Spacer() } .padding(.vertical, 10) .padding(.horizontal, 16) suggestionsList } } if magicSearch.isLoading { ProgressView() .controlSize(.large) .progressViewStyle(CircularProgressViewStyle(tint: .orangeMain500)) } } } .frame(maxWidth: .infinity) } .background(.white) if startConversationViewModel.operationOneToOneInProgress { PopupLoadingView() .background(.black.opacity(0.65)) .onDisappear { startConversationViewModel.searchField = "" MagicSearchSingleton.shared.currentFilter = "" MagicSearchSingleton.shared.searchForContacts() delayColorDismiss() isShowStartConversationFragment = false if let displayedConversation = startConversationViewModel.displayedConversation { self.conversationsListViewModel.changeDisplayedChatRoom(conversationModel: displayedConversation) startConversationViewModel.displayedConversation = nil } } } } .onAppear { if !magicSearch.currentFilter.isEmpty || (self.contactsManager.lastSearch.isEmpty && self.contactsManager.lastSearchSuggestions.isEmpty) { magicSearch.currentFilter = "" magicSearch.searchForContacts() } } .navigationTitle("") .navigationBarHidden(true) } .navigationViewStyle(StackNavigationViewStyle()) } @Sendable private func delayColor() async { try? await Task.sleep(nanoseconds: 250_000_000) delayedColor = Color.orangeMain500 } func delayColorDismiss() { Task { try? await Task.sleep(nanoseconds: 80_000_000) delayedColor = .white } } var suggestionsList: some View { ForEach(0..