linphone-ios/Linphone/UI/Main/Meetings/ViewModel/MeetingsListViewModel.swift
2025-01-21 17:16:28 +01:00

148 lines
5.3 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 mMeetingsListCoreDelegate: CoreDelegate?
var selectedMeetingToDelete: MeetingModel?
@Published var meetingsList: [MeetingsListItemModel] = []
@Published var currentFilter = ""
@Published var todayIdx = 0
init() {
coreContext.doOnCoreQueue { core in
self.mMeetingsListCoreDelegate = CoreDelegateStub(onConferenceInfoReceived: { (_: Core, conferenceInfo: ConferenceInfo) in
Log.info("\(MeetingsListViewModel.TAG) Conference info received [\(conferenceInfo.uri?.asStringUriOnly() ?? "NIL")")
self.computeMeetingsList()
})
core.addDelegate(delegate: self.mMeetingsListCoreDelegate!)
}
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
}
}
}
if !meetingForTodayFound && !meetingsListTmp.isEmpty {
// All meetings in the list happened in the past, add "Today" fake model at the end
meetingsListTmp.append(MeetingsListItemModel(meetingModel: nil))
todayIdx = currentIdx
}
DispatchQueue.main.async {
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)
if self.meetingsList.count == 1 && self.meetingsList[0].model == nil {
// Only remaining meeting is the fake TodayMeeting, remove it too
meetingsList.removeAll()
}
ToastViewModel.shared.toastMessage = "Success_toast_meeting_deleted"
ToastViewModel.shared.displayToast = true
}
}
}