์ค๋๋ถํฐ ๋ฏธ๋ผํด ๋ชจ๋ ์๊ฐ์ 1์ผ 1์ ์์ดํท ํฌ์คํ ์ ํด๋ณด๋ ค๊ณ ํ๋ค. ๊ตฌํ์์ฃผ์ ๊ณต๋ถ๋ฅผ ํ๋ค๋ณด๋, ๊ธฐ์ด์ง์์ด ๋๋ฌด ๋ถ์กฑํ๋ค๊ณ ๋๊ปด์ก๊ณ , UIKit์ ๊ตฌ์กฐ๋ฅผ ์ ํํ๊ฒ ํ์ ํ๊ณ ์ถ์ด์ ์์!
๐ต UIResponder
xcodeํ๋ก์ ํธ๋ฅด ๋ง๋ค๋ฉด AppDelegate์ SceneDelegate๋ฅผ ์ฒ์์ผ๋ก ์ ํ๊ฒ ๋๋๋ฐ, ์ด ๋๊ฐ์ ํด๋์ค๋ ๋์ผํ๊ฒ "UIResponder"๋ฅผ ์์๋ฐ๋๋ค.
์ฐ์ ๊ณต์๋ฌธ์๊ฐ ์ ์ํ ๊ฐ๋ ๋ถํฐ ์ฐจ๊ทผ์ฐจ๊ทผ ์ฝ์ด๋ณด์.
https://developer.apple.com/documentation/uikit/uiresponder
An abstract interface for responding to and handling events.
์ด๋ฒคํธ์ ์๋ตํ๊ณ ์ฒ๋ฆฌํ๊ธฐ ์ํ ์ถ์ ์ธํฐํ์ด์ค์ด๋ค.
Responder objects—that is, instances of UIResponder —constitute the event-handling backbone of a UIKit app. Many key objects are also responders, including the UIApplication object, UIViewController objects, and all UIView objects (which includes UIWindow). As events occur, UIKit dispatches them to your app's responder objects for handling.
Responder ๊ฐ์ฒด๋ UIResponder์ ์ธ์คํด์ค์ด๋ค. ๊ทธ๋ฆฌ๊ณ ์ด ์ธ์คํด์ค๊ฐ UIKit์์ ์ด๋ฒคํธ ์ฒ๋ฆฌ์ ์ค์ถ์ญํ ์ ํ๋ค. UIKit์ ๋ง์ ๊ฐ์ฒด๋ค, UIApplication, UIViewController, UIView๊ฐ์ ๊ฐ์ฒด๋ค์ด ๋ชจ๋ Responder์ด๊ธฐ๋ ํ๋ค. ์ด๋ฒคํธ๊ฐ ๋ฐ์ํ๋ฉด UIKit์ ์ด๋ฒคํธ Responder๊ฐ์ฒด์๊ฒ ์ ๋ฌํด์ ์ฒ๋ฆฌํ๋ค.
There are several kinds of events, including touch events, motion events, remote-control events, and press events. To handle a specific type of event, a responder must override the corresponding methods. For example, to handle touch events, a responder implements the touchesBegan(_:with:), touchesMoved(_:with:), touchesEnded(_:with:), and touchesCancelled(_:with:) methods. In the case of touches, the responder uses the event information provided by UIKit to track changes to those touches and to update the app's interface appropriately.
ํ๋ฉด์ ํฐ์นํ๊ฑฐ๋ ์ ์ค์ณ๋ฅผ ์ทจํ๋ ๋ฑ ์ฌ๋ฌ๊ฐ์ง ์ด๋ฒคํธ๊ฐ ์๋ค. ์ด๋ค ํน์ ํ ์ด๋ฒคํธ๋ฅผ ์ฒ๋ฆฌํ๋ ค๋ฉด ๋ฐ๋์ ํด๋น ์ด๋ฒคํธ์ ๋ํ ๋ฉ์๋๋ฅผ Responder๊ฐ์ฒด๊ฐ ์ค๋ฒ๋ผ์ด๋ ํด์ ๋ด์ฉ์ ๊ตฌํํ๋ค. touchesBegan, touchesMoved ๊ฐ์ ๋ฉ์๋๋ฅผ ๋ทฐ ์ปจํธ๋กค๋ฌ์ UIView์์ ์ ์ํ ์ ์์๋ ๊ฒ๋ ๋ชจ๋ ์ด ๊ฐ์ฒด๋ค์ด Responder์ด๊ธฐ ๋๋ฌธ์ด๋ค. ์๋ฅผ๋ค์ด ํฐ์น์ด๋ฒคํธ๊ฐ ๋ฐ์ํ๋ฉด responder๊ฐ์ฒด๋ UIKit์์ ์ ๊ณตํ๋ ์ ๋ณด๋ฅผ ๊ธฐ๋ฐ์ผ๋ก ์ฑ์ ์ธํฐํ์ด์ค๋ฅผ ์ ๋ฐ์ดํธํ๋ค.
In addition to handling events, UIKit responders also manage the forwarding of unhandled events to other parts of your app. If a given responder does not handle an event, it forwards that event to the next event in the responder chain. UIKit manages the responder chain dynamically, using predefined rules to determine which object should be next to receive an event. For example, a view forwards events to its superview, and the root view of a hierarchy forwards events to its view controller.
์ด๋ฒคํธ๋ฅผ ์ฒ๋ฆฌํ๋ ๊ฒ ๋ฟ๋ง์๋๋ผ UIKit ์ responder๋ค์ ์์ ์ด ์ฒ๋ฆฌํ์ง ์๋ ์ด๋ฒคํธ๋ฅผ ์ฑ์ ๋ค๋ฅธ ์์๋ค์๊ฒ ์ ๋ฌํ๋ ์ญํ ๋ ์ํํ๋ค. ๋ง์ฝ์ ์ด๋ค responder๊ฐ ์ด๋ฒคํธ๋ฅผ ์ฒ๋ฆฌํ์ง ์์ผ๋ฉด ์ด ๊ฐ์ฒด๊ฐ Responder Chain์ ๋ค์ responder์๊ฒ ์ด๋ฒคํธ๋ฅผ ์ ๋ฌํ๋ค. Responder chain์ UIkit์ ์ํด ๋ฏธ๋ฆฌ ์ ํด์ง ๊ท์น์ใท ๋ฐ๋ผ ๋์ ์ผ๋ก ๊ด๋ฆฌ๋๋ค. ์๋ฅผ๋ค์ด View๋ ์์ ์ super view์๊ฒ ์ด๋ฒคํธ๋ฅผ ์ ๋ฌํ๊ณ , ์ต์์์ ์๋ view๋ ViewController์๊ฒ ์ด๋ฒคํธ๋ฅผ ์ ๋ฌํ๋ค.
Responders process UIEvent objects but can also accept custom input through an input view. The system's keyboard is the most obvious example of an input view. When the user taps a UITextField and UITextView object onscreen, the view becomes the first responder and displays its input view, which is the system keyboard. Similarly, you can create custom input views and display them when other responders become active. To associate a custom input view with a responder, assign that view to the inputView property of the responder.
Responder ๊ฐ์ฒด๋ค์ UIEvent ๊ฐ์ฒด๋ค์ ์ฒ๋ฆฌํ์ง๋ง input view๋ฅผ ํตํด์ ์ปค์คํ ์ ๋ ฅ์ ๋ฐ์ ์๋ ์๋ค. ๊ฐ์ฅ ๋ํ์ ์ธ custom input ์์๋ ํค๋ณด๋์ด๋ค. ์ฌ์ฉ์๊ฐ ํ๋ฉด์ ์๋ UITextField์ UITextView์ ํญํ๋ฉด, ํด๋น ๋ทฐ๋ First Responder๊ฐ ๋์ด ์ ๋ ฅ๋ทฐ (ํค๋ณด๋)๋ฅผ ํ์ํ๋ค. ์ด์ฒ๋ผ ๊ฐ๋ฐ์๋ ์ปค์คํ ์ ๋ ฅ๋ทฐ๋ฅผ ๋ง๋ค์ด์ ์ด๋ค responder๊ฐ์ฒด๊ฐ ํ์ฑํ๋๋ฉด ํ๋ฉด์ ํ์๋๋๋ก ํ ์ ์๋ค.
๐น UIResponder ์ ์ญํ ์ ๋ฆฌ
๊ทธ๋์ ์์ฝํด๋ณด๋ฉด, ์๋์ ๊ฐ๋ค.
- UIResponder๋ ์ด๋ฒคํธ๊ฐ ๋ฐ์ํ๋ฉด, ์ฒ๋ฆฌํด์ ๋ด์ฉ์ ๊ตฌํํ๋ ์ผ์ ์ฃผ๋ก ๋งก์์ ํ๋ค.
- ์ด๋ฒคํธ๋ฅผ ์ฒ๋ฆฌํ๋ ์ผ ๋ฟ๋ง ์๋๋ผ, ์์ ์ด ์ฒ๋ฆฌํ์ง ์๋ ์ด๋ฒคํธ๋ Responder Chain ๊ท์น์ ์ํด ์ ํด์ง ๋ค์ responder์๊ฒ ์ด๋ฒคํธ๋ฅผ ์ ๋ฌํ๋ ์ญํ ๋ ํ๋ค.
- input view๋ฅผ ํตํด์ ์ปค์คํ ์ ๋ ฅ์ ๋ฐ์ ์๋ ์๋ค.
๊ณต์๋ฌธ์๋ฅผ ๋ณด๋ฉด์ ์ด๋ฒคํธ๋ฅผ ์ฒ๋ฆฌํด์ฃผ๋๊ฒ UIResponder์ธ๊ฑด ์๊ฒ ๋ค. ๊ทผ๋ฐ ๋ชจ๋ฅด๋๊ฐ๋ ์ด ์ธ๊ฐ์ง ๋์์ 'ํ๋์'๊ธ์จ๋ก ํ์ํด๋๋ค. ํด๋น ๊ฐ๋ ๋ค์ ์ฐจ๊ทผ์ฐจ๊ทผ ์์๋ณด๋ฉด์ ๋ ๋ช ํํ๊ฒ ์ดํดํด๋ณด์.
๐ต Responder ๊ฐ์ฒด
Responder๋ ์ด๋ฒคํธ๋ฅผ ํธ๋ค๋งํ๊ณ ๋ฐ์ํ ์ ์๋ ๊ฐ์ฒด์ด๋ค. ๋ชจ๋ Responder๊ฐ์ฒด๋ UIResponder์์ ์์๋ ํด๋์ค๋ค์ ์ธ์คํด์ค์ด๋ค. ๊ทธ๋๊น UIResponder ํด๋์ค์ ์ธ์คํด์ค๋ค์ Responder ๊ฐ์ฒด๋ผ๊ณ ํ๋ค. ๋ํ์ ์ผ๋ก UIView, UIView์ ์๋ธํด๋์ค์ธ UIWindow, UIViewController, UIApplication ๋ฑ์ด ์๋ค.
์์์ ์ค๋ช ํ๋ฏ, Responder๋ ์ด๋ฒคํธ๋ฅผ ์ ๋ฌ ๋ฐ์์ผ๋ฉด ๋ฐ๋์ ๊ทธ๊ฒ์ ์ฒ๋ฆฌํ๊ฑฐ๋ ๋ค๋ฅธ Resonder๊ฐ์ฒด๋ก ๋๊ฒจ์ฃผ์ด์ผํ๋ค. ์ฑ์ด ์ด๋ฒคํธ๋ฅผ ์ ๋ฌ๋ฐ์์ ๋, UIKit์ ์๋์ผ๋ก ์ด๋ฒคํธ๋ฅผ ์ฒ๋ฆฌํ๊ธฐ์ ๊ฐ์ฅ ์ ์ ํ Responder ๊ฐ์ฒด๋ก ์ด๋ฒคํธ๋ฅผ ์ ๋ฌํ๋ค.
์ฌ๊ธฐ์ ์ด๋ฒคํธ๋ฅผ ์ฒ๋ฆฌํ๊ธฐ์ ๊ฐ์ฅ ์ ์ ํ Resonder๊ฐ์ฒด๋ฅผ FirstResponder๋ผ๊ณ ํ๋ค!
๐ต Responder Chain
์ฒ๋ฆฌ๋์ง ์์ ์ด๋ฒคํธ๋ค์ ๋ฆฌ์คํฐ๋์์ ๋ค๋ฅธ ๋ฆฌ์คํฐ๋๋ก Active Responder Chain์ ๋ฐ๋ผ ์ด๋ํ๋ค. resopnder chain์ ์ฑ์ responder ๊ฐ์ฒด์ ๋ฐ๋ผ ๋์ ์ผ๋ก ๊ตฌ์ฑ๋๋ค. (๋งค๋ฒ ๊ฐ์ง๋ ์๋ค๋ ์๊ธฐ!)
์๋ ๊ทธ๋ฆผ์ UILabel, UITextField, UIButton ๊ทธ๋ฆฌ๊ณ ๋ ๊ฐ์ ๋ฐฑ๊ทธ๋ผ์ด๋ View๋ก UI๋ฅผ ๊ตฌ์ฑํ responder๋ค์ ๋ณด์ฌ์ฃผ๋ ๊ทธ๋ฆผ์ด๋ค. ๊ทธ๋ฆผ์์๋ ํ์ดํ๋ฅผ ํตํด responder ๊ฐ์ฒด๋ค์ด responder chain์ ๊ตฌ์ฑํ๋ค.
ํ์ดํ๋ฐฉํฅ์ผ๋ก ์ฒ๋ฆฌ๋์ง ์์ ์ด๋ฒคํธ๋ค์ด ์ ๋ฌ๋๋ค๊ณ ์ดํดํ๋ฉด ๋๋ค.
๋ง์ฝ UIButton์ด ์ ๋ฌ๋ฐ์ ์ด๋ฒคํธ๋ฅผ ์ฒ๋ฆฌํ์ง ์๋๋ค๋ฉด, UIKit์ Button์ super view์ธ UIView๋ก ์ฐ์ ์ ๋ฌํ๋ค. ์ด UIView์์๋ ์ด๋ฒคํธ ์ฒ๋ฆฌ๋ฅผ ํ์ง ์๋๋ค๋ฉด, UIView์ ๋ฃจํธ๋ทฐ์ธ UIView(๋๋ฒ์งธ ์ฒญ๋ก์!)์ผ๋ก ์ ๋ฌ๋๋ค. ์ฌ๊ธฐ์๋ ์ฒ๋ฆฌํ์ง ์๋๋ค๋ฉด, ์ด view์ ์ฐ๊ฒฐ๋ ViewController๋ฅผ ๊ฑฐ์ณ UIWindow๊น์ง ์ฐ์์ ์ผ๋ก ์ ๋ฌ๋๋ค. ์ฌ๊ธฐ์๋ ์ด๋ฒคํธ๊ฐ ์ฒ๋ฆฌ๋์ง ๋ชปํ๋ฉด, UIApplication๊น์ง ์ฌ๋ผ๊ฐ์ ์ฒ๋ฆฌ๋ ์๋ ์๋ค. ์ฌ๊ธฐ์๋ ์ด๋ฒคํธ๊ฐ ์ฒ๋ฆฌ๋์ง ์๋๋ค๋ฉด, ์ญ์ ๋๋ค.
(UIView์ ์ ์๋ฅผ ์ดํด๋ณด๋ฉด, UIResponder๋ฅผ ์์๋ฐ๊ณ ์๋ค. UITextField๋ UIControl์ ์์๋ฐ๊ณ ์๋๋ฐ, UIContrl์ด UIResponder๋ฅผ ์์๋ฐ๊ณ ์๊ธฐ ๋๋ฌธ์ ์์ Responder chain์ ํธ์ถํ ์ ์๋ค.)
์๋ Method๋ค์ responder chain์ ๊ด๋ฆฌํด์ฃผ๋ ๋ฉ์๋๋ค์ด๋ค. ๊ฐ๋จํ๊ฒ ์ค๋ช ํ์๋ฉด ์๋์ ๊ฐ์ง๋ง, ๋ ์์ธํ ๋ด์ฉ์ ๋์ค์ ๋ค๋ค๋ณด๋๋ก ํ์.
- becomeFirstResponder : window์ ์ฒซ๋ฒ์งธ responder๋ก ๋ง๋ค๊ธฐ์ํด ์ฌ์ฉ
- resignFirstResponder : window์ ์ต์ด responder๋ก์ ์ํ๋ฅผ ์๋ํ๋๋ก ์์ฒญ๋ฐ์์์ ์๋ฆฐ๋ค. (์ฆ, UITextField์์ resignFirstResponder()๋ฅผ ํธ์ถํ๋ฉด, ํค๋ณด๋๊ฐ ์ฌ๋ผ์ง๋๊ฒ ๊ทธ ์์์ด๋ค.)
๐น ์ด๋ฒคํธ์ First Responder ๊ฒฐ์ ํ๊ธฐ
๊ทธ๋ผ ์ด๋ป๊ฒ UIKit์ firstResponder๋ฅผ ์ค์ ํ ๊น? UIKit์ '๋ฐ์ํ ์ด๋ฒคํธ์ ํ์ '์ ๊ธฐ์ค์ผ๋ก firstResponder๋ฅผ ์ค์ ํ๋ค. ์ฌ๊ธฐ์ ๋งํ๋ event ๋ UIEvent์ ์ธ์คํด์ค์ธ๋ฐ, ์ฌ๋ฌ type์ ์ด๋ฒคํธ๊ฐ ์กด์ฌํ๋ค.
https://developer.apple.com/documentation/uikit/uievent/1613840-type
์ฌ๊ธฐ์ ๊ฐ์ฅ ์์ฃผ ๋ฐ์ํ๋ ์ด๋ฒคํธ๋ ํฐ์น ์ด๋ฒคํธ์ด๋ค. (press๋ ๋ฌผ๋ฆฌ ํค์์ ๋ฐ์ํ๋ ์ด๋ฒคํธ, remote-control์ ํค๋ํฐ ๋ฑ ์ธ๋ถ ๊ธฐ๊ธฐ์์ ๋๋ฅด๋ ๋ฒํผ์์ ๋ฐ์ํ๋ ์ด๋ฒคํธ, editing menu๋ ํธ๋ํฐ์์ text๋ฅผ ์งง๊ฒ ํน์ ๊ธธ๊ฒ ๋๋ฅผ ๋ ๋จ๋ ๋ฉ๋ด๋ฅผ ์๋ฏธํ๋ค. shake-motion์ ํ๋ค๋ ๋ฐ์ํ๋๋ฐ, ์ด๊ฒ์ responder chain์์ ์ฒ๋ฆฌํ์ง ์๊ณ core motion ๊ฐ์ฒด๋ก ์ ๋ฌ๋๋ค.)
๐น ์ด๋ค Responder๊ฐ ์ด๋ฒคํธ๋ฅผ ํฌํจํ๊ณ ์๋์ง ๊ฒฐ์ ํ๊ธฐ
UIKit์ ๋ทฐ ๊ธฐ๋ฐ์ hit-testing์ ์ฌ์ฉํ์ฌ ํฐ์น ์ด๋ฒคํธ๊ฐ ์ด๋์ ๋ฐ์ํ๋์ง ๊ฒฐ์ ํ๋ค. ๋ ๊ตฌ์ฒด์ ์ผ๋ก ๋งํ๋ฉด, UIKit์ ํฐ์น ์ด๋ฒคํธ๊ฐ ๋ฐ์ํ ์์น์ ๋ทฐ ๊ณ์ธต์ ๋ทฐ ์ค๋ธ์ ํธ์ bounds์ ๋น๊ตํ๋ค.
UIView์ hitTest(_:with: ) ๋ฉ์๋๋ view ๊ณ์ธต์ ๋์๋ค๋๋ฉด์ ์ด๋ฒคํธ๊ฐ ๋ฐ์ํ ํฐ์น ์ง์ ์ ํฌํจํ๊ณ ์๋ view์ ๊ณ์ธต ๊ฐ์ฅ ์ตํ๋จ subView๋ฅผ ์ฐพ๊ณ , ๊ทธ๊ฒ์ touch์ด๋ฒคํธ์ first responder๋ก ์ง์ ํ๋ค. ๋ง์ฝ ํฐ์น ์ด๋ฒคํธ๊ฐ ๋ฐ์ํ ์์น๊ฐ view์ bounds ๋ฐ์ ์๋ค๋ฉด, hitTest(_:with: )์ ํด๋น ๋ทฐ์ ์๋ธ๋ทฐ๋ฅผ ๋ชจ๋ ๋ฌด์ํ๋ค. (์๋ hitTest ํฌ์คํ
๋งํฌ๋ก ๋ค์ด๊ฐ์ hitTest์ ๋ํด์ ์ดํด๋ณด๋ฉด ๋ ๋ช
ํํ ์ดํด๊ฐ ๋ ๊ฒ์ด๋ค!)
๊ฒฐ๊ณผ์ ์ผ๋ก view์ ์์ฑ clipsToBounds๊ฐ false๋ก ์ค์ ๋์ด ์์ ๋, view์ bounds๋ฅผ ๋ฒ์ด๋ ์๋ธ๋ทฐ์์ ํฐ์น๊ฐ ๋ฐ์ํ๋๋ผ๋ hitTest๋ view๋ฅผ ๋ฆฌํดํ์ง ์๋๋ค.
์.. ๋ฌด์จ๋ง์ธ์ง ์ ๋ง ์ดํด๊ฐ ๋์ง ์์์? ๋ค์ ํฌ์คํ ์์ hit testing์ ๋ํด์ ์ดํด๋ณด์...... ๋๋์ฒด ์ด๊ฑด ๋ ๋ญ๊ฐ๋ ์ด์ผ!
https://zeddios.tistory.com/536
์ด๋ ๊ฒ ์ค๋์ UIResponder์ ๋ํด์ ์์๋ดค๋ค. ํ๋์ ๊ฐ๋ ์ ์๊ธฐ์ํด์ ๋ช๊ฐ์ ๊ฐ๋ ์ ์ฐพ์๋ดค๋์ง ๋ชจ๋ฅด๊ฒ ๋ค. ๋์ด์ ์๋ก์ด ๊ฐ๋ ์ ๋จธ๋ฆฌ์ ๋ฃ์ผ๋ฉด ์๋ฌด๊ฒ๋ ๋จ์ง ์์ ๊ฒ ๊ฐ๊ธฐ์, ์ค๋์ responder์ ๋ํ ์ ์์ ๊ฐ๋ ๋ง ์ ํํ๊ฒ ์ดํดํ ๊ฒ์ผ๋ก ๋ง์กฑํ๊ณ , hitTest์ ๋ํ ๊ฐ๋ ์ ๋ค์ ํฌ์คํ ์ผ๋ก ๋ง๋๋ณด์....
๐ hitTest์ ๋ํด์ ์์๋ณด๋ฌ๊ฐ๊ธฐ ๐
https://didu-story.tistory.com/292
๐ References
https://zeddios.tistory.com/538
https://jeonyeohun.tistory.com/257
https://woozzang.tistory.com/144