GroceryListTableViewController

GroceryListTableViewController

我有一个杂货清单应用
当您将项目添加到类别列表时,它不应该添加到整个类别列表中!

https://github.com/mrbryankmiller/Grocery-TableView-.git

class GroceryItemsTableViewController: UITableViewController {

    //var groceryItem = ["Item1", "Item2", "Item3"]

    //var groceryList  = ["Breakfast","Lunch", "Dinner"]


    @IBOutlet var groceryItemTableView: UITableView!


    @IBAction func addGroceryItemButtonPressed(sender: UIBarButtonItem) {

        ///new way///

        let alertController: UIAlertController = UIAlertController(title: "Add Grocery Item", message: "", preferredStyle: .Alert)

        //Cancel Button

        let cancelAction: UIAlertAction = UIAlertAction(title: "Cancel", style: .Cancel) { action -> Void in
            //cancel code
        }
        alertController.addAction(cancelAction)
        let saveAction: UIAlertAction = UIAlertAction(title: "Save", style: .Default) { action -> Void in

            let textField = alertController.textFields![0]
            groceryItem.items.append(textField.text!)
            self.tableView.reloadData()

        }

        alertController.addAction(saveAction)


        //Add text field

        //        alertController.addTextFieldWithConfigurationHandler { (textField) -> Void in
        //        textField.textColor = UIColor.blackColor()

        alertController.addTextFieldWithConfigurationHandler { (textField : UITextField!) -> Void in
            textField.placeholder = "Enter an Item"
            //alertController.textFields
        }

        //Present the AlertController
        self.presentViewController(alertController, animated: true, completion: nil)
    }

    override func viewDidLoad() {
        super.viewDidLoad()

        //self.navigationItem.leftBarButtonItem = self.editButtonItem()

    }

    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }


    // MARK: - Table view data source

    override func numberOfSectionsInTableView(tableView: UITableView) -> Int {
        // #warning Incomplete implementation, return the number of sections
        return 1
    }

    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        // #warning Incomplete implementation, return the number of rows

        return groceryItem.items.count

    }


    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
            let cell = tableView.dequeueReusableCellWithIdentifier("groceryItem1", forIndexPath: indexPath)
        cell.textLabel!.text = groceryItem.items [indexPath.row]
        return cell

    }
}

最佳答案

如果您仔细查看类groceryItem的声明,则您的杂货清单中的每个项目都有一个static元素数组,因此,每次添加新元素时,所有杂货项目之间都会共享该元素。

取而代之的是,您应该为每个杂货项目提供与其每个项目相关联的列表。

您可以定义一个新的struct,以便为每个杂货项目保存其关联的项目列表,如下所示:

struct GroceryItem {
   var name: String
   var items: [String]
}


我们将对您的GroceryListTableViewController中的代码进行一些更改以根据您的新模型重构代码,因此应类似于以下内容:


  GroceryListTableViewController


class GroceryListTableViewController: UITableViewController, GroceryItemsTableViewControllerProtocol {


    var groceryList = [GroceryItem]()

    @IBAction func addButton(sender: UIBarButtonItem) {

       let alertController: UIAlertController = UIAlertController(title: "Add Grocery Category", message: "", preferredStyle: .Alert)

       //Cancel Button

       let cancelAction: UIAlertAction = UIAlertAction(title: "Cancel", style: .Cancel) { action -> Void in
        //cancel code
       }
       alertController.addAction(cancelAction)

       let saveAction: UIAlertAction = UIAlertAction(title: "Save", style: .Default) { action -> Void in

          let textField = alertController.textFields![0]
          self.groceryList.append(GroceryItem(name: textField.text!, items: [String]()))
          self.tableView.reloadData()
       }
       alertController.addAction(saveAction)

       alertController.addTextFieldWithConfigurationHandler { (textField : UITextField!) -> Void in
          textField.placeholder = "Enter an Item"
          //alertController.textFields
        }

       //Present the AlertController
       self.presentViewController(alertController, animated: true, completion: nil)
    }

    override func viewDidLoad() {

       super.viewDidLoad()

       //edit button
       self.navigationItem.leftBarButtonItem = self.editButtonItem()

       groceryList.append(GroceryItem(name: "Breakfast", items: ["Item1", "Item2",  "Item3"]))
       groceryList.append(GroceryItem(name: "Lunch", items: ["Item1", "Item2",  "Item3"]))
       groceryList.append(GroceryItem(name: "Dinner", items: ["Item1", "Item2",  "Item3"]))

    }

    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {

       return groceryList.count
    }

    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
       let cell = tableView.dequeueReusableCellWithIdentifier("prototype1", forIndexPath: indexPath) as UITableViewCell

       cell.textLabel!.text = groceryList [indexPath.row].name

       return cell
    }

   // pass a tableview cell value to navigationBar title in swift//

   override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
      let destinationVC = segue.destinationViewController as! GroceryItemsTableViewController
      let cell = sender as! UITableViewCell

      let idx = self.tableView.indexPathForSelectedRow?.row

      destinationVC.delegate = self
      destinationVC.itemList = groceryList[idx!].items
      destinationVC.navigationItem.title = cell.textLabel?.text
   }

   func didAddGroceryItem(itemName: String) {
      let idx = self.tableView.indexPathForSelectedRow?.row
      groceryList[idx!].items.append(itemName)
   }

   func didRemoveGroceryItem(index: Int) {
    let idx = self.tableView.indexPathForSelectedRow?.row
    groceryList[idx!].items.removeAtIndex(index)
   }
}


在上面,我重构了有关新模型的所有代码,仅将代码更改的地方放到了其余地方。

您需要将与所选单元格关联的项目传递给另一个UIViewController,您可以在prepareForSegue中轻松完成此操作。为此,我们需要获取所选单元格的索引,然后将元素传递到另一个UIViewController,在此处创建一个新的[String]数组作为数据源以显示项目。

该代码中的另一个重要点是GroceryListTableViewController现在实现了一个称为GroceryItemsTableViewControllerProtocol的新协议。此协议是每次将新项目添加到列表时从GroceryListTableViewController通知GroceryItemsTableViewController的一种方式,这称为委托模式。


  GroceryItemsTableViewController


protocol GroceryItemsTableViewControllerProtocol: class {

   func didAddGroceryItem(itemName: String)

   func didRemoveGroceryItem(index: Int)
}

class GroceryItemsTableViewController: UITableViewController {

     weak var delegate: GroceryItemsTableViewControllerProtocol?

     var itemList: [String]!

     @IBAction func addGroceryItemButtonPressed(sender: UIBarButtonItem) {

        ///new way///

       let alertController: UIAlertController = UIAlertController(title: "Add Grocery Item", message: "", preferredStyle: .Alert)

       //Cancel Button

       let cancelAction: UIAlertAction = UIAlertAction(title: "Cancel", style: .Cancel) { action -> Void in
          //cancel code
       }
       alertController.addAction(cancelAction)


       let saveAction: UIAlertAction = UIAlertAction(title: "Save", style: .Default) { [weak self] action -> Void in

          guard let s = self else { return }

          let textField = alertController.textFields![0]

          s.itemList.append(textField.text!)
          s.delegate?.didAddGroceryItem(textField.text!)
          s.tableView.reloadData()

       }

      alertController.addAction(saveAction)

      alertController.addTextFieldWithConfigurationHandler { (textField : UITextField!) -> Void in
         textField.placeholder = "Enter an Item"
        //alertController.textFields
      }

      //Present the AlertController
      self.presentViewController(alertController, animated: true, completion: nil)
  }

  override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
     return itemList.count
   }


  override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
      let cell = tableView.dequeueReusableCellWithIdentifier("groceryItem1", forIndexPath: indexPath)


      cell.textLabel!.text = itemList[indexPath.row]
      return cell

  }

  override func tableView(tableView: UITableView, commitEditingStyle editingStyle: UITableViewCellEditingStyle, forRowAtIndexPath indexPath: NSIndexPath) {
    if editingStyle == .Delete {
        // Delete the row from the data source
        itemList.removeAtIndex(indexPath.row)
        delegate?.didRemoveGroceryItem(indexPath.row)
        tableView.deleteRowsAtIndexPaths([indexPath], withRowAnimation: .Fade)
    } else if editingStyle == .Insert {
        // Create a new instance of the appropriate class, insert it into the array, and add a new row to the table view
    }
  }
}


编辑:
为了正确处理删除,您应该创建一个新的委托方法,而不必通知GroceryListTableViewController某项已被删除,然后正确地删除它,您可以在上面的更新代码中看到。

希望对您有所帮助。

10-08 06:14