Есть ли способ получить доступ к элементу массива в KVC, используя значение (forKeyPath:)?
Предположим, у меня есть некоторые KVC-совместимые объекты, подобные следующим:
class Person : NSObject {
var office: Office
var firstName: String
var lastName: String
var reports: [Report]
init( office: Office,
firstName: String,
lastName: String,
reports: [Report] ) {
...
}
}
class Office : NSObject {
var building: String
var room: Int
init( building: String,
room: Int ) {
...
}
}
class Report : NSObject {
var title: String
var contents: String
init( title: String,
contents: String ) {
...
}
}
И я создаю экземпляр, человек
let person = Person(
office: Office(
building: "Main",
room: 2 ),
firstName: "Bob",
lastName: "Roberts",
reports: [
Report( title: "Today's weather", contents: "..." ),
Report( title: "Traffic", contents: "..." ),
Report( title: "Stocks", contents: "..." )
] )
Я могу использовать person.value(forKeyPath:) для доступа к свойствам и вложенным свойствам человека следующим образом:
person.value(forKeyPath: "firstName") // "Bob"
person.value(forKeyPath: "office.room" ) // 2
Тем не менее, есть ли в KVC путь к заголовку из второго доклада?
Что-то вроде
person.value(forKeyPath: "reports[1].title" ) // "Traffic"
2 ответа
Решение
Это возможно с родным KVC Swift даже со структурами, среда выполнения ObjC не требуется
struct Person {
var office: Office
var firstName: String
var lastName: String
var reports: [Report]
}
struct Office {
var building: String
var room: Int
}
struct Report {
var title: String
var contents: String
}
let person = Person(office: Office(building: "Main", room: 2 ),
firstName: "Bob", lastName: "Roberts",
reports: [
Report( title: "Today's weather", contents: "..." ),
Report( title: "Traffic", contents: "..." ),
Report( title: "Stocks", contents: "..." )
])
let keyPath = \Person.reports[1].title
let title = person[keyPath: keyPath] // "Traffic"
Здесь пояснение: вам не нужно менять класс NSObject на struct. Даже класс может работать. Я считаю это ответом.
class Person : NSObject {
var office: Office!
@objc var firstName: String!
var lastName: String!
@objc var reports: [Report]!
init( office: Office,
firstName: String,
lastName: String,
reports: [Report] ) {
super.init()
self.office = office
self.firstName = firstName
self.lastName = lastName
self.reports = reports
}
}
class Office : NSObject {
var building: String!
var room: Int!
init( building: String,
room: Int ) {
self.building = building
self.room = room
}
}
class Report : NSObject {
var title: String!
var contents: String!
init( title: String,
contents: String ) {
self.title = title
self.contents = contents
}
}
let person = Person(
office: Office(
building: "Main",
room: 2 ),
firstName: "Bob",
lastName: "Roberts",
reports: [
Report( title: "Today's weather", contents: "..." ),
Report( title: "Traffic", contents: "..." ),
Report( title: "Stocks", contents: "..." )
] )
person[keyPath: \Person.reports[1].title]