質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.83%

  • Swift

    6313questions

    Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

  • Xcode

    3641questions

    Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

  • iOS

    3582questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

tableViewについてあれこれ

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 129

haruka-t

score 61

//ViewController.swift
import UIKit

class ViewController: UIViewController,UITableViewDataSource,UITableViewDelegate{

  @IBOutlet weak var mytableView: UITableView!

  //構造体
  let initialItems: [String] = []
  var kei: [Cell] = Array()

  var item: [String] = []
  var cnt  = 0
  var itemTotal: [[String]] = [[]]

  func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    if (itemTotal.count == 0
      || itemTotal.count > 0 && itemTotal[0].count == 0) {
      return 0
    } else {
      return itemTotal.count
    }
  }

  func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = UITableViewCell()
    let index = itemTotal.count - 1 - indexPath.row
    let item = itemTotal[index]

    guard item.count > 0 else {
      return cell
    }

    var labels: [UILabel] = []
    //labelについて
    for i in item {
      let label = UILabel()
      label.backgroundColor = UIColor.darkGray
      label.text = i


      // textLabel という名前の変数に格納された UILabel にフォントサイズの自動調整を設定します。
      label.adjustsFontSizeToFitWidth = true
      label.minimumScaleFactor = 10.0
      //丸みに対して
      label.layer.cornerRadius = 5
      label.layer.masksToBounds = true
      cell.contentView.addSubview(label)
      labels.append(label)
    }

    var preLabel: UILabel? = nil
    for l in labels {

      l.translatesAutoresizingMaskIntoConstraints = false
      if (preLabel == nil) {
        l.leftAnchor.constraint(equalTo: cell.contentView.leftAnchor, constant: 12).isActive = true
      } else {
        l.leftAnchor.constraint(equalTo: preLabel!.rightAnchor, constant: 20).isActive = true
      }
      cell.contentView.heightAnchor.constraint(equalTo: l.heightAnchor, multiplier: 1).isActive = true
      preLabel = l
    }


    return cell
  }


  @IBAction func addcellbtr(_ sender: Any) {

    alertnormal()
  }

  func alertnormal(){
    // テキストフィールド付きアラート表示

    let alert = UIAlertController(title: "タイトル", message: "メッセージ", preferredStyle: .alert)

    // OKボタンの設定
    let okAction = UIAlertAction(title: "OK", style: .default, handler: {
      (action:UIAlertAction!) -> Void in

      // OKを押した時入力されていたテキストを表示
      if let textFields = alert.textFields {

        // アラートに含まれるすべてのテキストフィールドを調べる
        for textField in textFields {
          self.item.insert(textField.text!, at: 0)
          self.mytableView.insertRows(at: [IndexPath(row: 0, section: 0)],with: UITableViewRowAnimation.automatic)
          print(textField.text!)
        }
        self.mytableView.reloadData()
      }
    })
    alert.addAction(okAction)

    // キャンセルボタンの設定
    let cancelAction = UIAlertAction(title: "Cancel", style: .cancel, handler: nil)
    alert.addAction(cancelAction)

    // テキストフィールドを追加
    alert.addTextField(configurationHandler: {(textField: UITextField!) -> Void in
      textField.placeholder = "テキスト"
    })

    alert.view.setNeedsLayout() // シミュレータの種類によっては、これがないと警告が発生

    // アラートを画面に表示
    self.present(alert, animated: true, completion: nil)

  }


  func tableView(_ tableView: UITableView, trailingSwipeActionsConfigurationForRowAt indexPath: IndexPath) -> UISwipeActionsConfiguration? {

    let edit = UIContextualAction(style: .normal,title: "追加", handler: { (action: UIContextualAction, view: UIView, success :(Bool) -> Void) in

      let alert = UIAlertController(title: "タイトル", message: "メッセージ", preferredStyle: .alert)

      // OKボタンの設定
      let okAction = UIAlertAction(title: "OK", style: .default, handler: {
        (action:UIAlertAction!) -> Void in

        // OKを押した時入力されていたテキストを表示
        if let textFields = alert.textFields {

          // アラートに含まれるすべてのテキストフィールドを調べる
          for textField in textFields {
            self.item.insert(textField.text!, at: 0)
            self.itemTotal[self.cnt] = self.item

            if self.item.count % 5 == 0, self.itemTotal[0].count > 1 {
              self.cnt += 1
              self.itemTotal.append([])
              self.item = []
              print(textField.text!)
              //self.Gesture()
              //self.doubleclic()
            }
          self.mytableView.reloadData()
          }
        }
      })
      alert.addAction(okAction)

      // キャンセルボタンの設定
      let cancelAction = UIAlertAction(title: "Cancel", style: .cancel, handler: nil)
      alert.addAction(cancelAction)

      // テキストフィールドを追加
      alert.addTextField(configurationHandler: {(textField: UITextField!) -> Void in
        textField.placeholder = "テキスト"
      })


      alert.view.setNeedsLayout() // シミュレータの種類によっては、これがないと警告が発生

      // アラートを画面に表示
      self.present(alert, animated: true, completion: nil)
      print("edit")

      success(true)
    })

    edit.backgroundColor = .blue

    return UISwipeActionsConfiguration(actions: [edit])
  }



  // trueを返すことでCellのアクションを許可しています
  func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool {

    if item.count > (indexPath.row + 1) {
      return true
    }else{
      return false
    }
  }


  //注意

  func alert(){
    // テキストフィールド付きアラート表示

    let alert = UIAlertController(title: "タイトル", message: "メッセージ", preferredStyle: .alert)

    // OKボタンの設定
    let okAction = UIAlertAction(title: "OK", style: .default, handler: {
      (action:UIAlertAction!) -> Void in

      // OKを押した時入力されていたテキストを表示
      if let textFields = alert.textFields {

        // アラートに含まれるすべてのテキストフィールドを調べる
        for textField in textFields {
          self.item.insert(textField.text!, at: 0)
          self.itemTotal[self.cnt] = self.item

          if self.item.count % 5 == 0, self.itemTotal[0].count > 1 {
            self.cnt += 1
            self.itemTotal.append([])
            self.item = []
            print(textField.text!)
          //self.Gesture()
          //self.doubleclic()
          }
           self.mytableView.reloadData()
        }
      }
    })
    alert.addAction(okAction)

    // キャンセルボタンの設定
    let cancelAction = UIAlertAction(title: "Cancel", style: .cancel, handler: nil)
    alert.addAction(cancelAction)

    // テキストフィールドを追加
    alert.addTextField(configurationHandler: {(textField: UITextField!) -> Void in
      textField.placeholder = "テキスト"
    })

    alert.view.setNeedsLayout() // シミュレータの種類によっては、これがないと警告が発生

    // アラートを画面に表示
    self.present(alert, animated: true, completion: nil)

  }

  func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
    return 40
  }

  override func viewDidLoad() {
    super.viewDidLoad()

    mytableView.delegate = self
    mytableView.dataSource = self
    mytableView.estimatedRowHeight = 60
    //mytableView.backgroundColor = UIColor(red: 0.0, green: 0.8, blue: 1.0, alpha: 0.1)

    for item in initialItems {
      kei.append(Cell(item: item, cellcolor: false))
    }
  }



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


}

 今やっている事

まずボタンをタップした時に題名となるcellとその中身となるcellを挿入しようとしています。
中身となるcellはスワイプによって編集できるようにしています。

困っている事
最初に題名となるセルを追加しようとコードを加えたのですが実行時にはエラーは起きないのですが実際にアラートを呼び出しOKとタップすると落ちてしまいます。
エラーはsignal SIGABRTによるものです。

//cell.swift
struct Cell {
  //var category: String
  //var name: String
  var title: String
  var item: String
  var cellcolor: Bool
  var cnt  = 0

  init(item: String,cellcolor: Bool) {
    self.init(title: item,item: item,cellcolor: cellcolor)
    self.item = item;
    self.cellcolor = cellcolor;
    //self.itemTotal = item;
  }

  init(title: String,item: String,cellcolor: Bool) {
    self.title = title;
    self.item = item;
    self.cellcolor = cellcolor;
    //self.itemTotal = item;
}

}

最新のコードです。
構造体でtitleセルとdetailセルをグループ化しようとしています。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 2

checkベストアンサー

+1

原因となっているのは、

func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    if (itemTotal.count == 0
        || itemTotal.count > 0 && itemTotal[0].count == 0) {
        return 0
    } else {
        return itemTotal.count
    }
}


でitemTotalのサイズを数えていますが、

func alertnormal(){

            ...(省略)

            // アラートに含まれるすべてのテキストフィールドを調べる
            for textField in textFields {
                self.item.insert(textField.text!, at: 0)
                self.mytableView.insertRows(at: [IndexPath(row: 0, section: 0)],with: UITableViewRowAnimation.automatic)
                print(textField.text!)
            }
            self.mytableView.reloadData()

ここではitemTotalにデータを追加していないため、
self.mytableView.insertRowsで行を追加しようとしていますが、
itemTotalの数と合わなくなっていることが原因と考えられます。

解決方法としては、alertnormalでもitemTotalにデータを追加する必要があります。

とりあえずの回避方法としては

func alertnormal(){

            ...(省略)
            // アラートに含まれるすべてのテキストフィールドを調べる
            for textField in textFields {
                self.item.insert(textField.text!, at: 0)

                if (self.cnt != 0) {
                    self.itemTotal.append([])
                }

                self.itemTotal[self.cnt].append(textField.text!)
                self.tableView.insertRows(at: [IndexPath(row: 0, section: 0)],with: UITableViewRowAnimation.automatic)

                self.cnt += 1

                print(textField.text!)
            }
            self.tableView.reloadData()

のような形にするとエラーは出なくなりました。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/05/14 14:03

    alertnormalにitemTotalのデータを追加してしまうとラベルとして追加されてしまうのでそれを避ける為にitemTotalを使わないようにしていたのですが、ご教示して頂いたコートですとラベルとして追加されてしまうのでただのテキストとしてセルに追加することは出来ないのででしょうか?
    私としてはtitleとしてのセルとdetailとなるセルを区別にする為にstructで使ってグルーピングしたいと思っているのですが配列の入ってる今回のようなコードだとどう構造体を使って書けばいいか分からないでいるのでご教示いただけないでしょうか?

    キャンセル

  • 2018/05/15 00:36 編集

    構造体を使ってtitleとなるセルとdetailとなるセルをグループにして表示させようとしているのですが下記のコードをどうstructの中に下記入れればいいか分からなくて困っています。
    var item: [String] = []
    var cnt = 0
    var itemTotal: [[String]] = [[]]
    イメージとしては
    apple商品         ←titleセル
    [iPhone] [iPad] [iMac] [Macbook] [Apple Watch] ←detailセル
    このような感じになります。以前、構造体でセルのグループ分けの仕方は知っているのですがラベルのあるセルでは作ったことがなかったので教えて頂けると幸いです。

    キャンセル

  • 2018/05/15 06:23

    やられたいことから考えると、セルで分けるのではなくセクションを使った方が良いのではないかと思います。以下の内容が参考になるかと思います。
    http://docs.fabo.io/swift/uikit/019_sectionuitableview.html

    構造体を使うとなると
    struct Product {
    let title: String
    let detail: [String]
    }
    として
    var products: [Product]という配列で管理するかなと思います。

    キャンセル

  • 2018/05/15 14:20

    懇切に教えていただき有難うございます。
    配列の部分でのエラーはなくなりました。
    それでなんですがitemの部分をproductsに置き換えているのですがitemTotalの部分は何に置き換えて行けばいいのでしょうか?それともitemTotalはそのまま残す感じでしょうか?
    var cnt = 0はカウントする為に残しましたがitemTotalはどうすればいいか分からないのですご教示願いたいです。

    キャンセル

  • 2018/05/16 06:12

    私のイメージですと、let detail: [String]がitemになり、var products: [Product]がitemTotalのイメージです。cntは配列の順番と行の順番が連動している(indexPathから取得できる)ので特に必要ないのかなと思いました。

    キャンセル

  • 2018/05/16 14:37

    https://teratail.com/questions/126260
    私なりに置き換えてみたのですが上手く動かなかったので見てもらえませんか?

    キャンセル

  • 2018/05/16 18:27 編集

    numberOfRowsInSectionのメソッドで多次元配列のitemTotalの部分いprodauctsを入れるとエラーが起こるのですがどう対処すれば良いのでしょうか?

    キャンセル

+1

まずは Xcode のデバッガを利用して、 OK をタップした後の処理をステップ実行し、どのコードが原因か絞り込むのが良いと思います。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/05/11 12:17

    デバッグを見てみたのですがあまり理解できなかったです。
    でも多分、cellForRowAtの中にラベルに関するコードのせいでエラーが起きてるんじゃないんかと思っています。

    キャンセル

  • 2018/05/11 13:01

    題名となるセルに中身となるラベルの処理を呼ばれないようにするにはどうしたらいいですか?
    多分、cellForRowAtの中の処理が呼ばれてエラーが起きたのではないかと思います。

    キャンセル

  • 2018/05/11 17:03

    signal SIGABRT以外に何かメッセージは出ていませんか?

    「多分、cellForRowAtの中の処理が呼ばれてエラーが起きたのではないかと思います。」
    これはブレークポイントを設定した場合に呼ばれていたということでしょうか?その場合、どこの行でエラーが発生しているか特定できていますでしょうか?

    キャンセル

  • 2018/05/13 16:22

    メッセージはsignal SIGABRTしか出ていないと思われます。

    私が思うには今回のアラートはただ入力したものをcellに反映させたいだけなのでcellForRowAt内でのlabelの処理で落ちたのかなと思います。
    実際に、旧アラート(挿入時にlabelとして挿入されるアラート)では問題はありませんでした。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.83%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る

  • Swift

    6313questions

    Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

  • Xcode

    3641questions

    Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

  • iOS

    3582questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。