[iOS] UITableView + UITableViewCell (4): ์ปค์คํ UITableViewCell
์ปค์คํ ์ ์ ๋ง๋ค์ด๋ณด์
์ง๋ ํฌ์คํธ๊น์ง ํ ์ด๋ธ ๋ทฐ๋ฅผ ๋ง๋ค์ด ์ ์ ์ผ๋ก ์ ์ ์ ๋ ฅํด ๋ฃ๋ ๋ฒ, ํ ์ด๋ธ ๋ทฐ ์ปจํธ๋กค๋ฌ ํด๋์ค๋ฅผ ์๋ก ์์ฑํด์ ๋ฐ์ดํฐ๋ฅผ ์๋์ผ๋ก ์ ๋ก ๋ฃ๋ ๊ณผ์ , ๊ทธ๋ฆฌ๊ณ ๊ทธ ์์์ ์ฌ์ฉ๋๋ data source์ ๊ฐ๋ ๊ณผ ์ ์ ์ฌ์ฌ์ฉ์ ๋ํด ์ ๋ฆฌํด๋ณด์์ต๋๋ค. ์ด์ ํ ์ด๋ธ ๋ทฐ์ ๋ง์ง๋ง ์๋ฆฌ์ฆ์ธ๋ฐ์, ์ค๋์ ๋ชฉํ๋ UITableViewCell์ ์ปค์คํ ํด๋์ค๋ฅผ ๋ง๋ค์ด์ ๋ ์ ์ฐํ๊ฒ ์ ์ ๊ตฌ์ฑํ๋ ๋ฐฉ๋ฒ์ ์ ๋ฆฌํด๋ณด๊ฒ ์ต๋๋ค.
์์ ๋ค์ด ์ด์ ํฌ์คํธ๋ค์์ ์ด์ด์ง๋๋ค! ํนํ ์ด ํฌ์คํธ๋ฅผ ์ฝ์ง ์์ผ๋ฉด ์์ ๊ฐ ์ดํด๊ฐ ์ ํ ์๋ ์๋ ์์ด์! ๊ทธ๋ผ ์์ํ๊ฒ ์ต๋๋ค~!
์ง๋ ํฌ์คํธ์์๋ ์ด๋ฐ ๋ชจ์์ผ๋ก ํ ์ด๋ธ ๋ทฐ๋ฅผ ๊ตฌ์ฑํ์๋๋ฐ์, ์ฌ์ค ํ ๊ฐ์ง ๋น ์ง ๊ฒ์ด ์์์ต๋๋ค.
์ฌ๊ธฐ์ ๋ณด์ด๋ ์/๋ฐ์ ์์ผ์ด ๋น ์ ธ์์์ต๋๋ค..! ์ ํฌ๋ ํ ์ด๋ธ ๋ทฐ์์ ํ ์คํธ๋ฅผ ๋ฃ์ด์ฃผ๊ธฐ ์ํด์ ContentConfigutration์ text์ secondary text๋ฅผ ์ฌ์ฉํ์์ต๋๋ค. ๊ทธ๋ฐ๋ฐ ์ด๋ ๊ฒ ํ๋ฉด ์์ผ ๋ ์ด๋ธ์ ์ถ๊ฐํ ์๊ฐ ์์ด์.. ๊ทธ๋์ ์ปค์คํ ์ ์ ๋ง๋ค์ด์ ๋ด๋ถ์ ๋ ์ด๋ธ ์ธ ๊ฐ๋ฅผ ํฌํจํ๋ ์ ์ ๋ง๋ค์ด๋ณด๊ฒ ์ต๋๋ค.
ํด๋์ค ์ถ๊ฐํ๊ธฐ
๋จผ์ ์ปค์คํ ์ ์ ์ํ ํด๋์ค๋ฅผ ์ถ๊ฐํด์ฃผ๊ฒ ์ต๋๋ค.
ํผ์ ๋ง๋ค๊ธฐ์์ Cocoa Touch Class๋ฅผ ์ ํํ๊ณ ์,
UITableViewCell์ ์์ํ๋ ํด๋์ค๋ฅผ ์ถ๊ฐํฉ๋๋ค!
์ด๋ ๊ฒ ์๋ก์ด ํด๋์ค๊ฐ ์ถ๊ฐ๋์๋๋ฐ์, ๋ ๊ฐ์ ๋ฉ์๋๊ฐ ๊ธฐ๋ณธ์ ์ผ๋ก ์์ฑ๋์ด ์์ต๋๋ค. setSelected๋ ์ ์ด ์ ํ๋์์ ๋์ ๋์์ ์ง์ ํ๋ ๊ฒ ๊ฐ์ฃ ? ๊ทธ๋ฐ๋ฐ awakeFromNib()์ ๋ญ๊น์? ์ฃผ์์ ๋ณด๋ ์ด๊ธฐํ ์ฝ๋์ธ ๊ฒ ๊ฐ์๋ฐ ๋ ์์ธํ ์์๋ณด๊ธฐ ์ํด์ ์ด๋ฒ์๋ ์ญ์ ๊ณต์๋ฌธ์๋ฅผ ์ฐพ์๋ดค์ต๋๋ค!
awakeFromNib ์์๋ณด๊ธฐ
https://developer.apple.com/documentation/objectivec/nsobject/1402907-awakefromnib
๐ก The nib-loading infrastructure sends an awakeFromNib message to each object recreated from a nib archive, but only after all the objects in the archive have been loaded and initialized. When an object receives an awakeFromNib message, it is guaranteed to have all its outlet and action connections already established.
nib๋ ํ๋ฉด์ ๊ตฌ์ฑํ๋ ํด๋์ค๋ค์ XML๋ก ์ ์ํ XIB๋ฅผ ์ปดํ์ผํ ํ์ผ์ธ๋ฐ์, nib๊ฐ ์์นด์ด๋ธ ๋์ด์๋ค๊ฐ ๊ฐ์ฒด๋ก ์์ฑ๋ ๋ ์์นด์ด๋ธ์ ํฌํจ๋ ๋ชจ๋ ๊ฐ์ฒด๊ฐ ์์ฑ๋๊ณ ์ด๊ธฐํ๋๋ฉด awakeFromNib ๋ฉ์์ง๋ฅผ ๋ด๋ถ์ ์ผ๋ก ๊ฐ์ฒด๋ค์๊ฒ ๋ณด๋ธ๋ค๊ณ ํฉ๋๋ค. ์ฌ๊ธฐ์ ์์นด์ด๋ธ๋ ์ด๋ค ํด๋์ค์ ๊ทธ ํด๋์ค์ ์ฐธ์กฐ ๊ด๊ณ์ ์๋ ํด๋์ค๊น์ง ๊ทธ๋ํ๋ก ๋ฌถ์ด์ ์ ์ฅ(์์นด์ด๋น)ํ๋ ๊ฒ์ ์๋ฏธํฉ๋๋ค. ๊ฐ์ฒด๋ค์ด ์ด ๋ฉ์์ง๋ฅผ ๋ฐ์ผ๋ฉด ์์ ์ด ๊ฐ์ง๊ณ ์๋ ์์๋ ๋ณ์์ ์ก์ ๋ฉ์๋๊ฐ ๋ชจ๋ ์ฐ๊ฒฐ๋์ด ์์์ ๋ณด์ฅํ๊ฒ ๋ฉ๋๋ค. ๊ฐ๋จํ ๋งํด์ awakeFromNib์ ํ๋ฉด์ ๋ณด์ด๋ ์ ๋ณด๋ค์ ๋ด์ ๊ฐ์ฒด๊ฐ ์์ฑ&์ด๊ธฐํ๋ฅผ ๋ชจ๋ ์๋ฃํ ๋ค์ ํธ์ถ๋๋ ๋ฉ์๋๋ผ๊ณ ํ ์ ์์ต๋๋ค!
๐ก You must call the super implementation of awakeFromNib to give parent classes the opportunity to perform any additional initialization they require. Although the default implementation of this method does nothing, many UIKit classes provide non-empty implementations. You may call the super implementation at any point during your own awakeFromNib method.
awakeFromNib์ ๋ฐ๋์ super๊ฐ ๊ฐ์ง๊ณ ์๋ ๋ฉ์๋๋ ํจ๊ป ํธ์ถํด์ผ ํฉ๋๋ค. ๋ง์ UIKit ํด๋์ค๋ค์ด ์ฌ๊ธฐ์ ์ฝ๋๋ฅผ ํฌํจํ๊ณ ์๋ค๊ณ ํ๋ค์. ์ปค์คํ ํด๋์ค๋ค์ ๋๋ถ๋ถ UIKit ํด๋์ค๋ค์ ์์ํ๊ธฐ ๋๋ฌธ์ ๋ฐ๋์ ํธ์ถํด์ผ๊ฒ ๋ค์.
๐ก During the instantiation process, each object in the archive is unarchived and then initialized with the method befitting its type. Objects that conform to the NSCoding protocol (including all subclasses of UIView and UIViewController) are initialized using their initWithCoder: method. All objects that do not conform to the NSCoding protocol are initialized using their init method. After all objects have been instantiated and initialized, the nib-loading code reestablishes the outlet and action connections for all of those objects. It then calls the awakeFromNib method of the objects. For more detailed information about the steps followed during the nib-loading process, see Nib Files in Resource Programming Guide.
๊ฐ์ฒด๋ค์ ์ธ์คํด์ค๊ฐ ์์ฑ๋๊ณ awakeFromNib์ด ๋ถ๋ฆฌ๊ธฐ๊น์ง์ ๊ณผ์ ์ ๋ํ ๋ ์์ธํ ์ค๋ช ์ ๋๋ค. ๊ฐ ์ธ์คํด์ค๊ฐ ์์ฑ๋๋ ๊ณผ์ ์์ ์์นด์ด๋ธ ๋์๋ ๊ฐ ๊ฐ์ฒด๋ค์ ์ธ์์นด์ด๋ธ๊ฐ ๋๋ฉด์ ์์ฑ์๋ฅผ ํธ์ถํฉ๋๋ค. ์ด๋ NSCoding ํ๋กํ ์ฝ์ ์ฑํํ๊ณ ์๋ ๊ฐ์ฒด๋ค์ ๋ชจ๋ initWithCoder ๋ฉ์๋๋ฅผ ํธ์ถํ๊ณ ๊ทธ ์ธ์ ๊ฐ์ฒด๋ค์ ์์ ์ ์์ฑ์๋ฅผ ํธ์ถํฉ๋๋ค. ๋ชจ๋ ๊ฐ์ฒด์ ์ธ์คํด์ค๊ฐ ์์ฑ๋๋ฉด nib-loading ์ฝ๋๊ฐ ๊ฐ์ฒด๋ค์ด ๊ฐ์ง๊ณ ์๋ outlet๊ณผ action์ ๋ค์ ์ฐ๊ฒฐํ๋ ์์ ์ ํฉ๋๋ค. ์ด ์์ ์ด ์๋ฃ๋๋ฉด ๋ง์ง๋ง์ผ๋ก awakeFromNib ๋ฉ์๋๋ฅผ ํธ์ถํฉ๋๋ค.
๐ก Typically, you implement awakeFromNib for objects that require additional set up that cannot be done at design time. For example, you might use this method to customize the default configuration of any controls to match user preferences or the values in other controls. You might also use it to restore individual controls to some previous state of your application.
awakeFromNib์ ์ธ์คํด์ค๊ฐ ์์ฑ๋๋ ์๊ฐ์ ํ์ง ๋ชปํ๋ ์ผ๋ค ๊ตฌํํด๋ ์ ์์ต๋๋ค. Control์ ๋ํ ์ค์ ์ ์ถ๊ฐํ๊ฑฐ๋ ์ด์ ์ํ๋ฅผ ๋ณต์ํ๋ ์ฝ๋๋ฅผ ์ถ๊ฐํ ์ ์์ฃ !
์ปค์คํ ์ ๋ง๋ค๊ธฐ
๊ทธ๋ผ ์ด์ ์ปค์คํ ์ ์ ๋ง๋ค์ด์ ๋ฐฉ๊ธ ์์ฑํ๋ ํด๋์ค์ ์ฐ๊ฒฐํด์ฃผ๊ฒ ์ต๋๋ค. ์ฌ์ค ์ปค์คํ ์ ์ ๋ง๋๋ ค๋ฉด XIB๋ฅผ ๋ง๋๋ ๊ฒ์ด ๋ง๊ฒ ์ง๋ง ์ฌ๊ธฐ์๋ ์๋ตํ๊ณ ์ด์ ํฌ์คํธ์์ ์ฌ์ฉํ๋ ์ฌ์ฌ์ฉ ์ ์ ํด๋์ค๋ง ์ ํ์ฃผ๋๋ก ํ ๊ฒ์.
๋จผ์ ์ง๋ ํฌ์คํธ์์ ๋ง๋ค์ด๋์๋ ์ฌ์ฌ์ฉ ์ ์ ํด๋์ค๋ฅผ ์์ฑํ ์ปค์คํ ํด๋์ค๋ฅผ ์ถ๊ฐํด์ค๋๋ค.
์ธ ์ข ๋ฅ์ ๋ ์ด๋ธ์ด ํ์ํ๊ฒ ์ฃ ?
๊ทธ๋ฆฌ๊ณ ์ด ๋ ์ด๋ธ๋ค์ ์ ๊ทผํด์ ๊ฐ์ ์ง์ ํด์ฃผ์ด์ผ ํ๋, Outlet ๋ณ์๋ฅผ ๋ง๋ค์ด์ผํฉ๋๋ค.
์ด๋ ๊ฒ 3๊ฐ์ UILabel์ ๋ํ Outlet ๋ณ์๋ฅผ ๋ง๋ค๊ณ ,
์คํ ๋ฆฌ๋ณด๋๋ก ์ด๋ํด์ ์ด๋ ๊ฒ Outlet๊ณผ ๋ง๋ค์ด๋ ๋ ์ด๋ธ์ ์ฐ๊ฒฐํฉ๋๋ค.
๊ทธ๋ผ ์ด์ ์ค๋น๋ ๋ ๊ฒ ๊ฐ๋ค์. ์ด์ ๋ทฐ ์ปจํธ๋กค๋ฌ๋ก ์ด๋ํด์ ์ ์ธ์คํด์ค๋ฅผ ๋ง๋ค์ด ํ ์ด๋ธ ๋ทฐ์๊ฒ ๋ฐํํด์ฃผ๋ cellForRowAt ๋ฉ์๋๋ฅผ ๊ตฌํํด๋ณด๊ฒ ์ต๋๋ค.
์ ์ ๋ฐ์ดํฐ๋ฅผ ์ ํ ๋ฐํํ๊ธฐ
์ด์ ํฌ์คํธ์์ ๊ตฌํํ๋ ๋ฉ์๋์์ ์์ ํ ์ฌํญ๋ค์ด ์๊ฒผ์ต๋๋ค! ๋จผ์ dequeueReusableCell์์ ์ ์ ๊บผ๋ด์ ๋ฐ์ดํฐ๋ฅผ ์ ํ์ฃผ์ด์ผ ํ๋๋ฐ์,
์ด ๋ฉ์๋๋ ์ฐ๋ฆฌ๊ฐ ๋ง๋ ์ปค์คํ ํด๋์ค์ ์์ ํด๋์ค์ธ UITableViewCell ํด๋์ค๋ฅผ ๋ฐํํฉ๋๋ค. ๊ทธ๋์ ์ด ์ํ๋ก๋ ์ฐ๋ฆฌ๊ฐ ๋ง๋ ๋ ์ด๋ธ๋ค์ Outlet ๋ณ์์ ์ ๊ทผํ ์๊ฐ ์์ฃ . ์ด๋ฅผ ์ํด์ ๋ฐํ๋ฐ์ UITableViewCell์ ์ปค์คํ ํด๋์ค๋ก ๋ค์ด์บ์คํ ์ ํด์ฃผ์ด์ผํฉ๋๋ค.
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath) as! TableViewCell
.
.
.
์ด๋ ๊ฒ์!
๊ทธ๋ผ ์ด์ Outlet ๋ณ์๋ค์ ์ ๊ทผํ ์ ์๊ฒ ์ฃ ? ์ง๊ธ๋ถํฐ๋ ์ด์ ํฌ์คํธ์์ ๊ตฌํํ๋ ๋ด์ฉ์ ์กฐ๊ธ๋ง ๋ณ๊ฒฝํด์ฃผ๋ฉด ๋ฉ๋๋ค~!
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath) as! TableViewCell
cell.name.text = sections[indexPath.section][indexPath.row].name
cell.callType.text = sections[indexPath.section][indexPath.row].type
cell.day.text = sections[indexPath.section][indexPath.row].day
cell.accessoryType = .detailButton
return cell
}
์ฌ์ฌ์ฉ ์ ์ธ์คํด์ค๊ฐ ๊ฐ์ง Outlet ๋ณ์์ธ ๋ ์ด๋ธ๋ค์ ํ์ฌ ์ธ๋ฑ์ค์ ๋ง๋ ๋ฐ์ดํฐ๋ค์ ๋ถ๋ฌ์ ๋ฃ์ด์ฃผ๊ณ ๊ทธ๋๋ก ๋ฐํํ๋ฉด ๋์ ๋๋ค!
๊ทธ๋ผ ์คํํด๋ณด์ฃ !
์ด์ ์ ๋ฒ ์ฐ๋ฝ์ฒ ์ฑ์ ํ ์ด๋ธ๊ณผ ๋น์ทํด์ก๋ค์~!
์ฃผ๋ง์ ๋ชจ๋ ์ผ๋ ํ ์ด๋ธ ๋ทฐ ์ ๋ณตํ๊ธฐ ํ๋ก์ ํธ๊ฐ ๋๋ฌ์ต๋๋คใ ใ XIB๋ฅผ ๋ง๋ค์ด์ ์ปค์คํ ์ ์ ๋ง๋ค๊ณ ์ฝ๋๋ก ๋ฃ์ด์ฃผ๋ ๊ฒ๋ ํ๊ณ ์ถ์๋ฐ ์ด๋ฒ์ฃผ ์ค์ผ๋ก ์ด ํฌ์คํธ์ ์ ๋ฐ์ดํธ ํด๋ณด๋๋ก ํ๊ฒ ์ต๋๋ค. ์ฝ์ด์ฃผ์ ์ ๊ฐ์ฌํฉ๋๋ค!!
'๐ ์์ด-์ค-์์ค > ๐ฑ iOS ๊ฐ๋ฐ' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
๋๊ธ
์ด ๊ธ ๊ณต์ ํ๊ธฐ
-
๊ตฌ๋
ํ๊ธฐ
๊ตฌ๋ ํ๊ธฐ
-
์นด์นด์คํก
์นด์นด์คํก
-
๋ผ์ธ
๋ผ์ธ
-
ํธ์ํฐ
ํธ์ํฐ
-
Facebook
Facebook
-
์นด์นด์ค์คํ ๋ฆฌ
์นด์นด์ค์คํ ๋ฆฌ
-
๋ฐด๋
๋ฐด๋
-
๋ค์ด๋ฒ ๋ธ๋ก๊ทธ
๋ค์ด๋ฒ ๋ธ๋ก๊ทธ
-
Pocket
Pocket
-
Evernote
Evernote