SwiftData Cramcktime с использованием предиката макроса с общей моделью на основе протоколовIOS

Программируем под IOS
Ответить
Anonymous
 SwiftData Cramcktime с использованием предиката макроса с общей моделью на основе протоколов

Сообщение Anonymous »

Я работаю со SwiftData и пытаюсь поделиться логикой по нескольким моделям с использованием протоколов и расширений протоколов.
Я создал некоторые общие протоколы, такие как запрос , StatusResentable и отслеживаемые , которые мои модели SwiftData (E.G., PET ) Convorm p> p> p> p> p> p> p> p> p> prember prember prem. это: < /p>
@Model
final class Pet {
var id: UUID
var name: String
var statusRaw: String
// ... other properties
}
< /code>
и я определяю эти протоколы: < /p>
protocol StatusRepresentable: AnyObject, PersistentModel {
var statusRaw: String { get set }
}

extension StatusRepresentable {
var status: Status {
get { Status(rawValue: statusRaw) ?? .active }
set { statusRaw = newValue.rawValue }
}

func changeStatus(to newStatus: Status) {
if newStatus != status {
self.updateTimestamp(onChange: newStatus)
self.statusRaw = newStatus.rawValue
}
}
}
< /code>
и: < /p>
protocol Queryable: AnyObject, Identifiable, StatusRepresentable, PersistentModel {}

extension Queryable {
static var activePredicate: Predicate {
.withStatus(.active)
}

static func predicate(for id: UUID) -> Predicate where Self.ID == UUID {
.withId(id)
}
}
< /code>
Вот проблематичная часть: < /h4>
Я использую общее расширение предиката, как это: < /p>
extension Predicate {
static func withStatus(_ status: Status...) -> Predicate {
let rawValues = status.map { $0.rawValue }
return #Predicate {
rawValues.contains($0.statusRaw)
}
}
}
< /code>
Затем в моем представлении Swiftui я использую его так: < /p>
struct ComponentActiveList: View {
@Query private var activePets: [Pet]

init() {
self._activePets = Query(
filter: .activePredicate, // or .withStatus(.active)
sort: \.name,
order: .forward
)
}

var body: some View {
// ...
}
}
< /code>
Проблема: < /h4>
Он компилируется хорошо, но сбои во время выполнения с этой ошибкой (упрощенная): < /p>
keyPath: \.statusRaw
Thread 1: EXC_BREAKPOINT (code=1, subcode=0x...)
< /code>
В расширенном макросе я вижу это: < /p>
Foundation.Predicate({
PredicateExpressions.build_contains(
PredicateExpressions.build_Arg(rawValues),
PredicateExpressions.build_KeyPath(
root: PredicateExpressions.build_Arg($0),
keyPath: \.statusRaw
)
)
})
< /code>
Кажется, что у макроса возникают проблемы с разрешением \ .statusraw < /code> через расширение протокола /динамический поиск. Я предполагаю, что это имеет какое -то отношение к SwiftData + `#predicate не в состоянии разрешить свойства, ограниченные протоколом во время выполнения? /> extension Predicate {
static func pets(with status: Status...) -> Predicate {
let rawValues = status.map { $0.rawValue }
return #Predicate {
rawValues.contains($0.statusRaw)
}
}

static func pet(with id: UUID) -> Predicate {
#Predicate { $0.id == id }
}
}
< /code>
В качестве обходного пути я в настоящее время вернул весь код протокола и дублировал предикатую логику для каждой модели напрямую. Но в идеале я хотел бы определить их в одном месте с помощью протоколов или дженериков.


Подробнее здесь: https://stackoverflow.com/questions/797 ... eric-model
Ответить

Быстрый ответ

Изменение регистра текста: 
Смайлики
:) :( :oops: :roll: :wink: :muza: :clever: :sorry: :angel: :read: *x)
Ещё смайлики…
   
К этому ответу прикреплено по крайней мере одно вложение.

Если вы не хотите добавлять вложения, оставьте поля пустыми.

Максимально разрешённый размер вложения: 15 МБ.

Вернуться в «IOS»