Guia sobre SwiftData - Models

Photo by fabio on Unsplash

Guia sobre SwiftData - Models

Umas das novidades do Swift e da Apple, sabe que ela lançou o SwiftData para ajudar a simplificar o uso do CoreData junto com o SwiftUI

O que é o SwiftData

Primeiro de tudo é que não podemos confundir o SwiftData com um Banco de Dados, o SwiftData é basicamente um ORM (Mapeamento objeto-relacional (Object–relational mapping)). Para curiosidade dos demais, o banco de dados padrão do iOS é o SQLite.

SwiftData é uma framework lançada para substituir a antiga framework chamada CoreData que vem da era do Objective-C (Saudades colchetes). Mesmo seu projeto sendo em Swift/SwftUI você conseguia integrar o CoreData, mas não é uma solução nativa para essas ferramentas.

SwiftData foi introduzido no iOS 17 e não funciona nas versões abaixo, uma pena para projeto que ainda não podem migrar a versão minima.

Models

Vamos falar sobre as Models, que basicamente seriam nossas tabelas em um banco de dados relacional.

@Model class Book {
  var title: String
  var pages: Double
  var theme: String
}

Primeiro de tudo que precisamos adicionar a Macro @Model e não podemos trabalhar com Struct, temos que obrigatóriamente com as class. Basicamente esse é um modelo da classe Livro que será salvo no nosso banco de dados.

Atributos

Você deve estar se perguntando, mas se eu quiser adicionar um atributo unico, pode ser o title ou talvez um id. Nesse caso vamos adicionar outra macro na frente do nosso atríbuto.

@Model class Book {
  @Attribute(.unique) var title: String
  var pages: Double
  var theme: String
}

Dessa forma não conseguiremos salvar dois livros com o mesmo título.

Vou deixar uma lista aqui de atríbutos que podemos ser usados:

 /// Ensures the property's value is unique across all models of the same type.
 public static var unique: Schema.Attribute.Option { get }

/// Transforms the property's value between an in-memory form and a persisted form.
public static func transformable(by transformerType: ValueTransformer.Type) -> Schema.Attribute.Option
public static func transformable(by transformerName: String) -> Schema.Attribute.Option

/// Stores the property's value as binary data adjacent to the model storage.
public static var externalStorage: Schema.Attribute.Option { get }

/// Stores the property's value in an encrypted form.
public static var allowsCloudEncryption: Schema.Attribute.Option { get }

/// Preserves the property's value in the persistent history when the context deletes the owning model.
public static var preserveValueOnDeletion: Schema.Attribute.Option { get }

/// Track changes to this property but do not persist
public static var ephemeral: Schema.Attribute.Option { get }

/// Indexes the property's value so it can appear in Spotlight search results.
public static var spotlight: Schema.Attribute.Option { get }

Atributos Transitórios (transient attributes)

São atributos que fazem parte da nossa model, mas que não queremos salvar no banco de dados por algum motivo específico.

@Model class Book {
  @Attribute(.unique) var title: String
  var pages: Double
  var theme: String

  // Will not save this two properties 
  @Transient var readingSpeedPerPage = 0
  var fullTitle: String {
        "\(title) - \(theme)"
  }
}

SwiftData vai automaticamente salvar somente as Stored Properties do seu modelo, qualquer Computed Properties será tratada como transient, caso você queria que alguma Stored Property seja temporário basta colocar a macro @Transient na frente.

Relationship

@Model class Book {
  @Attribute(.unique) var title: String
  var pages: Double
  var theme: String

  // Will not save this two properties 
  @Transient var readingSpeedPerPage = 0
  var fullTitle: String {
        "\(title) - \(theme)"
  }
}

@Model class Author {
  @Attribute(.unique) var name: String
  @Relationship(deleteRule: .cascade) var books: [Books]
}

Para criar uma relação entre duas classes precisamos criar um atributo, nesse caso eu estou dizendo que o Author conhece seus livros mas o livro não sabe quem é seu Author, e adicionar a Macro @Relationship e adicionar o tipo de regra de exclusão. Segue uma lista das regras disponíveis:

case cascade
A rule that deletes any related models.

case deny
A rule that prevents the deletion of a model because it contains one or more references to other models.

case noAction
A rule that doesn’t make changes to any related models.

case nullify
A rule that nullifies the related model’s reference to the deleted model.

Basicamente:

  • Cascade - Se apagarmos o Author todos os livros que ele tem como referencia vão ser apagados.

  • Deny - Só vai deixar apagar o Author depois que apagarmos todos os livros

  • noAction - Apaga o Author mas deixa os livros salvos

  • nullify - Se nossa classe Book conhecesse a classe Author essa propriedade ficaria com valor nil se apagarmos o Author desses livros.

Por padrão o SwiftData a regra de deleção é nullify

Para terminar esse artigo vamos fazer nossa classe livro conhecer o autor, porque existem dois jeito de fazer isso.

Uma maneira é deixar explicito na macro de relação

@Model class Book {
  @Attribute(.unique) var title: String
  var pages: Double
  var theme: String
  var author: Author

  // Will not save this two properties 
  @Transient var readingSpeedPerPage = 0
  var fullTitle: String {
        "\(title) - \(theme)"
  }
}

@Model class Author {
  @Attribute(.unique) var name: String
  @Relationship(deleteRule: .cascade,  inverse: \Book.author) var books: [Books]
}

A outra é deixar o atributo Author como Optional

@Model class Book {
  @Attribute(.unique) var title: String
  var pages: Double
  var theme: String
  var author: Author?

  // Will not save this two properties 
  @Transient var readingSpeedPerPage = 0
  var fullTitle: String {
        "\(title) - \(theme)"
  }
}

@Model class Author {
  @Attribute(.unique) var name: String
  @Relationship(deleteRule: .cascade) var books: [Books]
}