回答編集履歴

1

質問者より2を実施するという質問の修正があったため、回答を修正します。

2020/01/28 03:13

投稿

nerianighthawk
nerianighthawk

スコア544

test CHANGED
@@ -49,3 +49,231 @@
49
49
  2つの方法を提示しましたが、冒頭に述べた通り、どちらも根本的な修正が必要な回答になってしまいました。
50
50
 
51
51
  あまりいい手段を提示できなくてすみません。
52
+
53
+
54
+
55
+ ----以下追記-----
56
+
57
+ 2の方法で実践したいとのことで、いくつか問題点を上げていただいたので、その回答として追記します。
58
+
59
+
60
+
61
+ まず、checkボタンもレンダリングしてしまう問題についてはもう一つcheck用の変数を用意することで、そのcheckの真偽値によって分岐する関数を親コンポーネントで作ることによって、ボタンを親コンポーネントに持つことができます。
62
+
63
+
64
+
65
+ また、2回レンダリングをする件についても同じようにchildのcheck用変数を用意することで解決できます。
66
+
67
+ そもそも2回レンダリングをしてしまう原因ですが、childをアンマウントするタイミングと、openの変化によってparentのテキストフィールドを空にするタイミングの2回でレンダリングしてしまうからです。
68
+
69
+ これをcheck用の変数が切り替わるタイミングのみでのレンダリングにすることによって、1回のレンダリングにまとめることができます。
70
+
71
+ 以下、ソースコードを載せます。雑に作ったので、関数内の変数名がbだったりしますが、ご了承ください。
72
+
73
+
74
+
75
+ ```
76
+
77
+ const App = () => {
78
+
79
+ parent++;
80
+
81
+ console.log("parent: " + parent);
82
+
83
+
84
+
85
+ const [open, setOpen] = useState(false);
86
+
87
+ const [checkOpen, setCheckOpen] = useState(false);
88
+
89
+ const [checkClose, setCheckClose] = useState(false);
90
+
91
+
92
+
93
+ const checkParrentText = b => {
94
+
95
+ if (b) {
96
+
97
+ setOpen(true);
98
+
99
+ return;
100
+
101
+ }
102
+
103
+ alert("The text is incorrect.");
104
+
105
+ };
106
+
107
+
108
+
109
+ return (
110
+
111
+ <>
112
+
113
+ <ParentHeader />
114
+
115
+ <ParentInput checkClose={checkClose} setCheckOpen={setCheckOpen} />
116
+
117
+ <button onClick={() => checkParrentText(checkOpen)}>check text</button>
118
+
119
+ <div>
120
+
121
+ {open && (
122
+
123
+ <Child
124
+
125
+ setOpen={setOpen}
126
+
127
+ check={checkClose}
128
+
129
+ setCheck={setCheckClose}
130
+
131
+ />
132
+
133
+ )}
134
+
135
+ </div>
136
+
137
+ </>
138
+
139
+ );
140
+
141
+ };
142
+
143
+
144
+
145
+ const ParentInput = ({ checkClose, setCheckOpen }) => {
146
+
147
+ parentInput++;
148
+
149
+ console.log("ParentInput: " + parentInput);
150
+
151
+
152
+
153
+ const [parentText, setParentText] = useState("");
154
+
155
+ useEffect(() => {
156
+
157
+ if (checkClose) setParentText("");
158
+
159
+ }, [checkClose]);
160
+
161
+ useEffect(() => {
162
+
163
+ if (parentText === "open") {
164
+
165
+ setCheckOpen(true);
166
+
167
+ }
168
+
169
+ }, [parentText, setCheckOpen]);
170
+
171
+
172
+
173
+ return (
174
+
175
+ <>
176
+
177
+ <input
178
+
179
+ type="text"
180
+
181
+ value={parentText}
182
+
183
+ onChange={e => setParentText(e.target.value)}
184
+
185
+ />
186
+
187
+ </>
188
+
189
+ );
190
+
191
+ };
192
+
193
+
194
+
195
+ const Child = memo(({ setOpen, check, setCheck }) => {
196
+
197
+ child++;
198
+
199
+ console.log("child: " + child);
200
+
201
+ const checkChildText = b => {
202
+
203
+ if (b) {
204
+
205
+ setOpen(false);
206
+
207
+ return;
208
+
209
+ }
210
+
211
+ alert("The text is incorrect.");
212
+
213
+ };
214
+
215
+ return (
216
+
217
+ <>
218
+
219
+ <ChildHeader />
220
+
221
+ <ChildInput setCheck={setCheck} />
222
+
223
+ <button onClick={() => checkChildText(check)}>check text</button>
224
+
225
+ </>
226
+
227
+ );
228
+
229
+ });
230
+
231
+
232
+
233
+ const ChildInput = ({ setCheck }) => {
234
+
235
+ childInput++;
236
+
237
+ console.log("ChildInput: " + childInput);
238
+
239
+
240
+
241
+ const [childText, setChildText] = useState("");
242
+
243
+ useEffect(() => {
244
+
245
+ if (childText === "close") {
246
+
247
+ setCheck(true);
248
+
249
+ }
250
+
251
+ }, [childText, setCheck]);
252
+
253
+
254
+
255
+ return (
256
+
257
+ <>
258
+
259
+ <input
260
+
261
+ type="text"
262
+
263
+ value={childText}
264
+
265
+ onChange={e => setChildText(e.target.value)}
266
+
267
+ />
268
+
269
+ </>
270
+
271
+ );
272
+
273
+ };
274
+
275
+
276
+
277
+ export default App;
278
+
279
+ ```