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

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

新規登録して質問してみよう
ただいま回答率
85.48%
React.js

Reactは、アプリケーションのインターフェースを構築するためのオープンソースJavaScriptライブラリです。

Q&A

解決済

1回答

411閲覧

React: toggleが正常に動かない

vankick

総合スコア22

React.js

Reactは、アプリケーションのインターフェースを構築するためのオープンソースJavaScriptライブラリです。

0グッド

0クリップ

投稿2019/06/06 09:31

Reactで、お気に入りに追加機能を作っています。

機能としては、名前横のアイコンをクリックし、お気に入りに追加するというものです。

もしもう一度アイコンをクリックすると、toggleされて、likesからは外されます。

その中で、いくつか上手くいかない点があります。


toggleが正常に動いていません。1番目をクリックし、2番目をクリックし、さらにもう1回2番目をクリックすると、なぜか1番目が消えてしまいます。


console.log("this.state.likes", this.state.likes); でstateのlikes arrayを見てみると、1個前にクリックしたelementが表示されます。どうしてこのようなことが起こるのでしょうか?

どなたかお力をお貸しいただければ幸いです。

import React from 'react'; import axios from 'axios'; import IcoMoon from 'react-icomoon'; export default class PersonList extends React.Component { state = { persons: [], likes: [] } componentDidMount() { axios.get(`https://jsonplaceholder.typicode.com/users`) .then(res => { const persons = res.data; this.setState({ persons }); }) } handleClick = person => { if (this.state.likes.includes(person)) { this.setState({ likes: this.state.likes.splice(this.state.likes.indexOf(person), 1) }); console.log("included", this.state.likes); return; } this.setState({ likes: [...this.state.likes, person] }); console.log("this.state.likes", this.state.likes); }; likesTemplate = item => <li key={item}>{item}</li>; renderLikes = () => { return this.state.likes.map(i => this.likesTemplate(i)); } render() { return ( <div> {this.state.persons.map(person => { return <li key={person.name}><IcoMoon icon="heart" onClick={() => {this.handleClick(person.name)}} />{person.name}</li>} )} <h2>Favorite Person</h2> <ul> {this.renderLikes()} </ul> </div> ) } }

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

Array.splice の戻り値は取り除かれた要素を含む配列ですので、そのような挙動になるのだと思います。Array.filter を使ってみてはいかがでしょうか?

if (this.state.likes.includes(person)) { this.setState({ likes: this.state.likes.filter(item => item !== person) }); console.log("included", this.state.likes); return; }

console.log("this.state.likes", this.state.likes); でstateのlikes arrayを見てみると、1個前にクリックしたelementが表示されます。どうしてこのようなことが起こるのでしょうか?

setStateコンポーネントのローカル state の更新をスケジュールするために使用するものです。
言い換えると、

this.setState({ likes: [...this.state.likes, person] });

の直後に this.state はまだ更新されていません。ですので this.setState 直後に
console.log を呼び出しても、直前の状態がコンソールログに表示されます。

何度も申し訳ありません。もう1つだけ、if (this.state.likes.includes(person)) {のところに書いた return; なのですが、こちらはどうして必要なのでしょうか? 実はこれは別の人にアドバイスをされて書いたのです。このreturnを削除すると、挙動がおかしくなってしまいます。ですが、何のための記述なのか理解できずにいました。

return 文は関数の実行を終了 する役割があります。そのため return 文を消してしまうと、personthis.state.likes に含まれている場合に、if 文の中が実行された後に

this.setState({ likes: [...this.state.likes, person] }); console.log("this.state.likes", this.state.likes);

までもが実行されてしまいます。

投稿2019/06/06 09:55

編集2019/06/06 12:14
YukiYamashina

総合スコア1011

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

vankick

2019/06/06 10:37

思うような動きになりました!.spliceと.filterの詳しい挙動が分かっていないので、勉強しようと思います。もう1つ質問させていただきたいのですが、上記コードの"console.log("this.state.likes", this.state.likes);" の部分なのですが、これを見ると、今クリックしたelementは追加されておらず、1個前にクリックしたもののみ表示されるのですが、これの原因なども分かったりはしますか?
vankick

2019/06/06 11:50

ご丁寧に教えていただき、ありがとうございます。 何度も申し訳ありません。もう1つだけ、if (this.state.likes.includes(person)) {のところに書いた return; なのですが、こちらはどうして必要なのでしょうか? 実はこれは別の人にアドバイスをされて書いたのです。このreturnを削除すると、挙動がおかしくなってしまいます。ですが、何のための記述なのか理解できずにいました。
vankick

2019/06/06 12:34

非常に分かりやすいです。不明点が解消されました。ご丁寧に説明していただきまして、ありがとうございました!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問