forked from mirrors/linphone-iphone
137 lines
4.8 KiB
Swift
137 lines
4.8 KiB
Swift
/*
|
|
* Copyright (c) 2010-2024 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import Foundation
|
|
import linphonesw
|
|
import Combine
|
|
|
|
class MeetingsListViewModel: ObservableObject {
|
|
static let TAG = "[Meetings ListViewModel]"
|
|
static let ScrollToTodayNotification = Notification.Name("ScrollToToday")
|
|
|
|
private var coreContext = CoreContext.shared
|
|
private var mCoreSuscriptions = Set<AnyCancellable?>()
|
|
var selectedMeetingToDelete: MeetingModel?
|
|
|
|
@Published var meetingsList: [MeetingsListItemModel] = []
|
|
@Published var currentFilter = ""
|
|
@Published var todayIdx = 0
|
|
|
|
init() {
|
|
coreContext.doOnCoreQueue { core in
|
|
self.mCoreSuscriptions.insert(core.publisher?.onConferenceInfoReceived?.postOnCoreQueue { (cbVal: (core: Core, conferenceInfo: ConferenceInfo)) in
|
|
Log.info("\(MeetingsListViewModel.TAG) Conference info received [\(cbVal.conferenceInfo.uri?.asStringUriOnly() ?? "NIL")")
|
|
self.computeMeetingsList()
|
|
})
|
|
}
|
|
computeMeetingsList()
|
|
}
|
|
|
|
func computeMeetingsList() {
|
|
let filter = self.currentFilter.uppercased()
|
|
let isFiltering = !filter.isEmpty
|
|
|
|
coreContext.doOnCoreQueue { core in
|
|
var confInfoList: [ConferenceInfo] = []
|
|
|
|
if let account = core.defaultAccount {
|
|
confInfoList = account.conferenceInformationList
|
|
}
|
|
if confInfoList.isEmpty {
|
|
confInfoList = core.conferenceInformationList
|
|
}
|
|
|
|
var meetingsListTmp: [MeetingsListItemModel] = []
|
|
var meetingForTodayFound = false
|
|
var currentIdx = 0
|
|
var todayIdx = 0
|
|
for confInfo in confInfoList {
|
|
if confInfo.duration == 0 { continue }// This isn't a scheduled conference, don't display it
|
|
var add = true
|
|
if !filter.isEmpty {
|
|
let organizerCheck = confInfo.organizer?.asStringUriOnly().range(of: filter, options: .caseInsensitive) != nil
|
|
let subjectCheck = confInfo.subject?.range(of: filter, options: .caseInsensitive) != nil
|
|
let descriptionCheck = confInfo.description?.range(of: filter, options: .caseInsensitive) != nil
|
|
let participantsCheck = confInfo.participantInfos.first(
|
|
where: {$0.address?.asStringUriOnly().range(of: filter, options: .caseInsensitive) != nil}
|
|
) != nil
|
|
|
|
add = organizerCheck || subjectCheck || descriptionCheck || participantsCheck
|
|
}
|
|
|
|
if add {
|
|
let model = MeetingModel(conferenceInfo: confInfo)
|
|
|
|
if !meetingForTodayFound && !isFiltering {
|
|
if model.isToday {
|
|
meetingForTodayFound = true
|
|
todayIdx = currentIdx
|
|
} else if model.isAfterToday {
|
|
// If no meeting was found for today, insert "Today" fake model before the next meeting to come
|
|
meetingsListTmp.append(MeetingsListItemModel(meetingModel: nil))
|
|
meetingForTodayFound = true
|
|
todayIdx = currentIdx
|
|
}
|
|
}
|
|
|
|
var matchFilter = !isFiltering
|
|
if isFiltering {
|
|
matchFilter = matchFilter || confInfo.subject?.uppercased().contains(filter) ?? false
|
|
matchFilter = matchFilter || confInfo.description?.uppercased().contains(filter) ?? false
|
|
matchFilter = matchFilter || confInfo.organizer?.asStringUriOnly().uppercased().contains(filter) ?? false
|
|
for pInfo in confInfo.participantInfos {
|
|
matchFilter = matchFilter || pInfo.address?.asStringUriOnly().uppercased().contains(filter) ?? false
|
|
}
|
|
}
|
|
|
|
if matchFilter {
|
|
meetingsListTmp.append(MeetingsListItemModel(meetingModel: model))
|
|
currentIdx += 1
|
|
}
|
|
}
|
|
}
|
|
|
|
DispatchQueue.main.sync {
|
|
self.todayIdx = todayIdx
|
|
self.meetingsList = meetingsListTmp
|
|
}
|
|
}
|
|
}
|
|
|
|
func deleteSelectedMeeting() {
|
|
guard let meetingToDelete = selectedMeetingToDelete else {
|
|
Log.error("\(MeetingsListViewModel.TAG) Could not delete meeting because none was selected")
|
|
return
|
|
}
|
|
|
|
coreContext.doOnCoreQueue { core in
|
|
core.deleteConferenceInformation(conferenceInfo: meetingToDelete.confInfo)
|
|
}
|
|
|
|
if let index = self.meetingsList.firstIndex(where: { $0.model?.address == meetingToDelete.address }) {
|
|
if self.todayIdx > index {
|
|
// bump todayIdx one place up
|
|
self.todayIdx -= 1
|
|
}
|
|
self.meetingsList.remove(at: index)
|
|
ToastViewModel.shared.toastMessage = "Success_toast_meeting_deleted"
|
|
ToastViewModel.shared.displayToast = true
|
|
}
|
|
}
|
|
}
|