iosswiftuitableviewmobile

Why does my UITableView display duplicate tasks after adding a new task in an iOS app using Core Data?


I am working on an iOS app using Core Data and UITableView to manage tasks. When I add a new task, the UITableView displays duplicate entries for existing tasks.

Here is the code for managing tasks and updating the table view:

here is gif of the problem after adding new task: gif of the problem

ViewController.Swift

class ViewController: UIViewController {
    let viewModel = TaskViewModel.shared
    lazy var tasksTableView: UITableView = {
        let tableView = UITableView()
        tableView.translatesAutoresizingMaskIntoConstraints = false
        tableView.dataSource = self
        tableView.delegate = self
        tableView.register(TaskTableViewCell.self, forCellReuseIdentifier: "TaskTableCell")
       
        tableView.estimatedRowHeight = 200
        tableView.rowHeight = UITableView.automaticDimension
        return tableView
    }()

    lazy var addNewButton: UIButton = {
        let v = UIButton()
        v.translatesAutoresizingMaskIntoConstraints = false
        v.setTitle("Add Task", for: .normal)
        v.backgroundColor = .systemBlue
        v.layer.cornerRadius = 10
        v.frame = CGRect(x: 0, y: 0, width: 100, height: 35)

        v.addTarget(self, action: #selector(addPressed(sender: )), for: .touchUpInside)
        return v
    }()

    override func viewDidAppear(_ animated: Bool) {

        tasksTableView.reloadData()
    }


    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view.
        view.backgroundColor = .systemBackground
        title = "Alisveris Listesi"

       navigationItem.rightBarButtonItem = UIBarButtonItem(image: UIImage(systemName: "plus.circle.fill"), style: .done, target: self, action: #selector(showAddVC))

        navigationItem.leftBarButtonItem = UIBarButtonItem(barButtonSystemItem: .close,  target: self, action: #selector(deleteAllPressed(sender: )))

        setupViews()
    }
    private func setupViews() {
        view.addSubview(tasksTableView)
        configureConstraints()
    }

    private func configureConstraints() {
        NSLayoutConstraint.activate([
            tasksTableView.topAnchor.constraint(equalTo: view.safeAreaLayoutGuide.topAnchor, constant: 16),
            tasksTableView.leadingAnchor.constraint(equalTo: view.leadingAnchor, constant: 16),
            tasksTableView.trailingAnchor.constraint(equalTo: view.trailingAnchor, constant: -16),
            tasksTableView.bottomAnchor.constraint(equalTo: view.bottomAnchor, constant: 16)
        ])
    }

    @objc func showAddVC() {
        let vc = AddNewTaskViewController()
        vc.delegate = self
        let navController = UINavigationController(rootViewController:  vc)

        navigationController?.present(navController, animated: true)
    }

    @objc func addPressed(sender: UIButton) {
        viewModel.addNewTask(name: "New Task")
    }

    @objc func getTaskPressed(sender: UIButton) {
        let tasks = CoreDataManager.shared.fetchAll()
        for task in tasks {
            print(task.name ?? "" )
        }
    }

    @objc func deleteAllPressed(sender: UIButton) {
        let tasks = CoreDataManager.shared.fetchAll()
        for task in tasks {
            CoreDataManager.shared.deleteItem(id: task.id ?? UUID() )
        }

        let fetchedTasks = CoreDataManager.shared.fetchAll()
        
        tasksTableView.reloadData()
        print(fetchedTasks.count)

    }

}


extension ViewController: UITableViewDelegate, UITableViewDataSource {
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return viewModel.numberOfRows(by: section)
    }
    
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
//        if indexPath.section == 0 {
//            guard let cell = tableView.dequeueReusableCell(withIdentifier: "TaskTableCell", for: indexPath) as? TaskTableViewCell else {
//                return UITableViewCell()
//            }
////            let taskSummary =
//            cell.configure(with: viewModel.tasks[indexPath.row])
//            return cell
//        }

        guard let cell = tableView.dequeueReusableCell(withIdentifier: "TaskTableCell", for: indexPath) as? TaskTableViewCell else {
            return UITableViewCell()
        }
        cell.configure(with: viewModel.tasks[indexPath.row])
        return cell
    }

    func numberOfSections(in tableView: UITableView) -> Int {
        return viewModel.numberOfTasks
    }


}

extension ViewController: ItemControllerDelegate {
    // Implement delegate methods
    func didItemAdded() {

        // Handle item added event
        tasksTableView.reloadData()
    }

    func didItemUpdated() {
        // Handle item updated event
    }
}


class TaskViewModel {
    static let shared = TaskViewModel()
    var tasks = [Task]()
    
    private init() {
        // Clear the existing tasks array
        tasks.removeAll()
        // Get all records from CoreData
        fetchAllTasks()
    }

    var numberOfTasks: Int {
        tasks.count
    }

    func fetchAllTasks() {
        // Fetch all data from core data
        tasks = CoreDataManager.shared.fetchAll().map(Task.init)
        print("Number of tasks after fetching: \(tasks.count)")
    }

    func numberOfRows(by section:Int) -> Int {
        if section == 0 {
            return 1
        }
        return numberOfTasks
    }

    func getTasksByType() -> (complete: Int, InComplete: Int) {
        let completedCount = tasks.lazy.filter({ $0.completed }).count
        let InCompletedCount = tasks.lazy.filter({ !$0.completed }).count
        
        return (completedCount,InCompletedCount)
    }

    func task(by index: Int) -> Task {
        return tasks[index]
    }

    func addNewTask(name: String) {
        let newItem = Item(context: CoreDataManager.shared.context)
        newItem.id = UUID()
        newItem.completed = false

        newItem.name = name
        newItem.createdAt = Date.now

        let newTask = Task(task: newItem)
        tasks.append(newTask)

        CoreDataManager.shared.addNewItem(item: newItem)

    }

    func toggleCompleted(task: Task) {
        CoreDataManager.shared.toggleCompleted(id: task.id)
        // call core data to toggle
        fetchAllTasks()
    }

    func deleteItem(task: Task) {
        CoreDataManager.shared.deleteItem(id: task.id)
        // call core data to delete the task
        fetchAllTasks()
    }

}

TaskViewModel.swift

class TaskViewModel {
    static let shared = TaskViewModel()
    var tasks = [Task]()
    
    private init() {
        // Clear the existing tasks array
        tasks.removeAll()
        // Get all records from CoreData
        fetchAllTasks()
    }

    var numberOfTasks: Int {
        tasks.count
    }

    func fetchAllTasks() {
        // Fetch all data from core data
        tasks = CoreDataManager.shared.fetchAll().map(Task.init)
        print("Number of tasks after fetching: \(tasks.count)")
    }

    func numberOfRows(by section:Int) -> Int {
        if section == 0 {
            return 1
        }
        return numberOfTasks
    }

    func getTasksByType() -> (complete: Int, InComplete: Int) {
        let completedCount = tasks.lazy.filter({ $0.completed }).count
        let InCompletedCount = tasks.lazy.filter({ !$0.completed }).count
        
        return (completedCount,InCompletedCount)
    }

    func task(by index: Int) -> Task {
        return tasks[index]
    }

    func addNewTask(name: String) {
        let newItem = Item(context: CoreDataManager.shared.context)
        newItem.id = UUID()
        newItem.completed = false

        newItem.name = name
        newItem.createdAt = Date.now

        let newTask = Task(task: newItem)
        tasks.append(newTask)

        CoreDataManager.shared.addNewItem(item: newItem)

    }

    func toggleCompleted(task: Task) {
        CoreDataManager.shared.toggleCompleted(id: task.id)
        // call core data to toggle
        fetchAllTasks()
    }

    func deleteItem(task: Task) {
        CoreDataManager.shared.deleteItem(id: task.id)
        // call core data to delete the task
        fetchAllTasks()
    }

}

CoreDataManager.swift

class CoreDataManager {
    static let shared = CoreDataManager()

    private init() {}

    // persistence CoreDataModel
    lazy var persistentContainer: NSPersistentContainer = {
        // Name of the CoreDataModel -- Items
        let container = NSPersistentContainer(name: "Items")
        container.loadPersistentStores { _ , error in
            if let error = error as NSError? {
                fatalError("Unresolved error \(error), \(error.userInfo)")
            }
        }
        return container
    }()
    
    // for saving -- save, delete etc.
    var context: NSManagedObjectContext {
        persistentContainer.viewContext
    }

    func saveContext() {
        if context.hasChanges {
            do {
                try context.save()
            } catch {
                let nserror = error as NSError
                print("Error - saveContext:", nserror.description, nserror.userInfo)
            }
        }
    }

    // Fetch all Items from CoreData
    func fetchAll() -> [Item] {
        var tasks = [Item]()
        
        // EntityName.fetchRequest()
        let fetchRequest: NSFetchRequest<Item> = Item.fetchRequest()
        let sortByCreatedDate = NSSortDescriptor(key: "createdAt", ascending: true)

        fetchRequest.sortDescriptors = [sortByCreatedDate]

        do {
            tasks = try context.fetch(fetchRequest)
        } catch {
            let nserror = error as NSError
            print("Error - fetchAll:", nserror.description, nserror.userInfo)
        }

        return tasks
    }

    // Add a item to CoreData
    func addNewItem(item: Item) {
        // create a new item if this item doesn't exist

        // save changes
        saveContext()
    }
    
    // Toggle Completed from CoreData
    func toggleCompleted(id: UUID) {
        let fetchRequest: NSFetchRequest<Item> = Item.fetchRequest()
        let predicate = NSPredicate(format: "id=%@", id.uuidString)
        fetchRequest.predicate = predicate

        do {
            if let fetchedTasks = try context.fetch(fetchRequest).first(where: { foundItem in
                foundItem.id == id
            }) {
                fetchedTasks.completed = !fetchedTasks.completed
            }

            // Save Core Data
            saveContext()

        } catch let error as NSError {
            print("toggleCompleted Error: \(error) \(error.userInfo)")
        }
    }

    // Delete item from CoreData
    func deleteItem(id: UUID) {
        let fetchRequest: NSFetchRequest<Item> = Item.fetchRequest()

        let predicate = NSPredicate(format: "id=%@", id.uuidString)
        fetchRequest.predicate = predicate

        do {
            let fetchedTasks = try context.fetch(fetchRequest)

            for task in fetchedTasks {
                context.delete(task)
            }

            // Save Core Data
            saveContext()

        } catch let error as NSError {
            print("deleteItem Error: \(error) \(error.userInfo)")
        }
    }

}

AddNewTaskViewController.swift

class AddNewTaskViewController: UIViewController {
    weak var delegate: ItemControllerDelegate?

    lazy var taskNameLabel: UILabel = {
        let label = UILabel()
        label.translatesAutoresizingMaskIntoConstraints = false
        label.text = "Task Name"
        return label
    }()

    lazy var taskNameTextField: UITextField = {
        let textField = UITextField()
        textField.translatesAutoresizingMaskIntoConstraints = false
        textField.placeholder = "Enter Task Name"
        textField.borderStyle = .roundedRect
        return textField
    }()

    let viewModel = TaskViewModel.shared

    override func viewDidLoad() {
        super.viewDidLoad()
        
        setupView()
    }

    private func setupView() {
        view.backgroundColor = .systemBackground
        title = "Add New Task"
        navigationItem.rightBarButtonItem = UIBarButtonItem(title: "Save", style: .done, target: self, action: #selector(savePressed))

        view.addSubview(taskNameLabel)
        view.addSubview(taskNameTextField)
        setupConstraints()
    }

    private func setupConstraints() {
        NSLayoutConstraint.activate([
            taskNameLabel.topAnchor.constraint(equalTo: view.safeAreaLayoutGuide.topAnchor, constant: 15),
            taskNameLabel.leadingAnchor.constraint(equalTo: view.leadingAnchor, constant: 15),
            taskNameLabel.trailingAnchor.constraint(equalTo: view.trailingAnchor, constant: -15),

            taskNameTextField.topAnchor.constraint(equalTo: taskNameLabel.bottomAnchor, constant: 15),
            taskNameTextField.leadingAnchor.constraint(equalTo: view.leadingAnchor, constant: 15),
            taskNameTextField.trailingAnchor.constraint(equalTo: view.trailingAnchor, constant: -15),
        ])
    }

    // Action Function
    @objc func savePressed() {
        guard let name = taskNameTextField.text else { 
            let alert = UIAlertController(title: "Error", message: "Name can't be empty", preferredStyle: .alert)
            alert.addAction(UIAlertAction(title: "OK", style: .default))
            present(alert, animated: true)
            return }

        viewModel.addNewTask(name: name)

        delegate?.didItemAdded()

        navigationController?.dismiss(animated: true)
    }
}

What could be causing tasks to be duplicated in the UITableView, and how can I resolve this issue?


Solution

  • In the delegate method numberOfSections(in:) you return the total number of rows which must be wrong

    func numberOfSections(in tableView: UITableView) -> Int {
        return viewModel.numberOfTasks
    }
    

    Looking at the function numberOfRows(by:) in your view model I assume the number of sections are two so change the delegate method to

    func numberOfSections(in tableView: UITableView) -> Int {
        return 2
    }