각각의 프로젝트들은 공통의 리소스를 사용하여 개발해야 하기 때문에, 리소스 Framework를 만들어 관리할 수 있다.
중복되는 이미지도 없고, 리소스 Framework를 import만 하면 된다.
Image, Color를 리소스 Framework으로 관리
먼저 리소스 Framework를 만든다. 그리고 이미지를 Images.assets에 추가한다.
해당 이미지들은 외부에서 코드로 불러와 사용하는 경우가 있기 때문에, 이미지를 외부에서 접근할 수 있는 코드를 만든다.
Resource Framework는 R 이라는 타입으로 접근하여 사용할 것이다.
먼저 R 타입을 만들자.
Framework로 만들었기 때문에, 이미지를 불러올 때 Resource Framework의 Bundle 위치를 알기 위해 내부에서 사용할 Bundle을 만들었다.
이제 이미지를 외부에서 접근할 수 있는 코드를 R.Image.[이미지이름] 형태를 따르도록 만든다.
이제 외부에서는 다음과 같이 이미지를 불러올 수 있다.
Color도 Image와 마찬가지로 만들 수 있다.
Storyboard, Xib를 리소스 Framework에서 관리
iOS 개발시, 항상 논쟁이 되는 주제가 있다. 바로 View를 Storyboard vs Xib로 작성하는 것이다.
Storyboard 파일을 갖기 위해서는 Dynamic Framework를 만들어야하는데, 각 기능마다 Framework로 만들게 되면 Framework 개수가 빠른 속도로 늘어날 뿐만 아니라, 기능을 더 작게 나눠 한 화면을 Framework로 만들면 어마어마할 것이다.
따라서 Storyboard, Xib를 리소스 Framework에서 관리하면 된다.
리소스 Framework에서 관리하면 화면 단위의 Framework를 Static으로 만들어도 Bundle의 위치가 리소스 Framework이기 때문에 문제가 없다.
그리고 Storyboard, Xib에서 이미지와 Color를 지정해도 Framework 자신의 내부에서 가져오기 때문에 문제가 없다.
그렇다면 다른 Framework에 있는 UIViewController Class와 ViewController View를 어떻게 연결할까?
위 사진처럼 Module을 선택하고 해당 UIViewController Class 선택하면 된다.
iOS를 개발하다보면 View의 Layout과 Content와 관련된 이슈를 자주 접하게 된다.
실제로 UIView가 언제 Update되는지 모르기 때문에 발생한다.
View가 언제 Update 되는지 알기 위해서는 iOS App's Run Loop 를 이해하고 그것이 UIView가 제공하는 메서드들과 어떤 관계를 갖고 있는지 파악해야 한다.
iOS App's Run Loop
iOS App의 Main Run Loop는 유저로부터 모든 input Event를 받고, 응답을 담당한다.
유저가 발생한 모든 상호작용은 Event Queue에 추가된다.
아래의 사진처럼 App 객체는 Event Queue로부터 Event를 하나씩 꺼내서 App의 다른 객체들에게 전달한다.
App 객체는 유저로부터 input Event를 해석하고 그에 상응되는 App의 Core 객체들 안에 있는 Handler를 호출해준다.
또한 이러한 Handler는 개발자들이 만들어놓은 코드를 실행한다.
이러한 메서드들이 반환되면 다시 Main Run Loop로 돌아가 Update Cycle이 다시 시작된다.
Update Cycle은 View들을 배치하고 다시 그리는 역할을 한다.
Update Cycle
Update Cycle은 App이 유저로부터의 모든 Event Handling Code를 수행하고 다시 Main Run Loop로 컨트롤을 반환하는 지점
바로 이 지점에서 시스템은 우리의 View들을 배치하고(layout), 보여주고(display), 제약(contraints)한다.
만약 우리가 Event Handler들을 처리하는 과정에서 어떤 UIView에 변화를 준다면, 해당 UIView는 다시 그려져야 한다고 표시됨
다음 Update Cycle에서 시스템은 이 UIView의 모든 변화를 수행한다.
유저가 상호작용하는 것과 Layout이 변하는 시간의 Gap은 유저가 인지하지 못한다.
iOS App은 60 FPS이고, Update Cycle은 이 중 1 프레임에 해당된다.
이렇게 빠르게 Update되기 때문에, 유저는 UI와 상호작용 간의 차이를 느끼지 못한다.
그러나 이벤트가 처리되는 시점과 실제로 View가 다시 그려지는 시점의 차이가 존재하기 때문에, View는 우리가 View를 Update하기 원하는 Run Loop의 특정 시점에 Update되지 않을 수도 있다.
이는 다음과 같은 위험을 초래한다.
만약 우리가 View의 마지막 Layout이나 Content에 대해 계산을 해야하는 시점이라면, 예전 정보를 갖고 View를 조작할 가능성이 생긴다.
Layout
UIView의 Layout은 화면에서 UIView의 크기와 위치를 의미한다.
모든 View의 frame을 갖고 있고, 이는 SuperView의 좌표계에서의 위치와 크기를 나타낸다.
UIView는 시스템에게 Layout이 변했다고 알려줄 수 있는 메서드
View의 Layout이 다시 계산되는 시점에 특정 작업을 실행할 수 있게 오버라이드 가능한 콜백 메서드를 제공
layoutSubviews()
View와 SubView들의 위치와 크기를 재조정한다.
현재 View와 모든 SubView들의 위치와 크기를 제공한다.
이 메서드는 재귀적으로 모든 SubView들의 layoutSubviews까지 호출되기 때문에, 부하가 크다.
이 메소드를 직접 호출하면 안된다
대신, layoutSubviews를 시스템이 호출하도록 유도하는 방식 존재
이 방식은 모두 run loop가 돌아가는 동안 layoutSubviews가 실행되는 시점이 모두 다르다.
setNeedsLayout() 호출 or 즉시 업데이트를 원하면 layoutIfNeeded() 메소드 호출
layoutSubviews가 완료될 때, View를 소유한 VC의 viewDidLayoutSubviews가 호출된다.
layoutSUbviews는 View의 layout이 변화했다는 callback이기 때문에, layout 관련 로직은 오래된 layout을 사용하는 것을 방지하기 위해 viewDidLoad/viewWillAppear가 아닌 viewDidLayoutSubviews에 호출해야 한다.
자동 refresh triggers
다음과 같은 이벤트들은 자동으로 View가 그것들의 layout에 변화가 생겼다는 것을 인지하여, 시스템에서 layoutSubviews가 다음 update Cycle에서 호출된다.
View를 Resizing
SubView 추가
UIScrollView 스크롤 시, UIScrollView와 그것의 부모뷰에 layoutSubviews가 호출됨
Device를 회전
View의 Constraint를 변경
위와 같은 방법들은 자동으로 시스템이 알아채어 layoutSubviews를 호출해준다.
그러나 layoutSubviews를 직접 호출할 수 있는 방법들도 존재한다.
setNeedsLayout()
layoutSubviews를 가장 적은 부하로 호출할 수 있는 메서드이다.
setNeedsLayouts는 시스템에게 이 View의 Layout이 재계산되어야 한다고 알린다.
setNeedsLayouts는 즉시 반환되고, 실제로 View Update를 해주는 것은 아니다.
대신, 시스템이 다음 Update Cycle에서 layoutSubviews를 View와 SubView들에게 호출하게 한다.
실제로 setNeedsLayouts가 호출되는 시점과 View가 다시 그려지는 시점은 정확하지는 않지만, 유저가 인지할 수는 없다.
Returns the farthest descendant of the receiver in the view hierarchy (including itself) that contains a specified point 지정된 점(point)을 포함하는 View 계층에서 리시버의 가장 먼 자손을 리턴한다.
라고 한다.
메소드이다.
point = 리시버의 bounds로 지점된 점
event = 메소드에 대한 호출을 보증하는 이벤트이다. 이벤트 처리코드 외부에서 이 메소드를 호출하는 경우에는 nil을 지정할 수 있다.
리턴 타입이 UIView?로 옵셔널이다.. nil이 반환될 수도 있다는 건데 -> 잠시 후에 알아보자
Discussion
이 메소드는 SubView의 point(inside:with:)를 호출하여 View 계층을 탐색하고, 어떤 하위 VIew가 터치 이벤트를 받아야 하는지 결정한다.
point 메소드의 파라미터는 hitTest와 같고, 반환타입만 다르다.
hitTest = 해당 point를 통해 리시버로부터 가장 먼 SubView를 반환
point = 해당 point가 리시버의 Bounds 내에 존재하는가?
point가 true를 반환하면, 하위 View 계층구조는 지정된 point를 포함하는 가장 앞에 있는 View를 찾을 때까지 재귀적으로 찾는다.
point가 false를 반환하면, View 계층 구조의 해당 분기가 무시된다.
(분기? -> hitTest는 Reverse DFS 방식으로 찾는다.)
ex)
해당 View 계층일 때,
MainView -> View C -> View C.2 -> View C.1 -> View B ...