#swift #swiftui #widget #widgetkit
#swift #swiftui #виджет #widgetkit
Вопрос:
Что я хотел бы сделать, так это предоставить пользователю возможность выбрать, является ли это widget background
изображение, взятое из http
или gradient background
.
В настоящее время у меня есть следующая структура заметок, но я не могу заставить ее работать.
Поэтому typeBg
должно быть значение по умолчанию, если оно не передано, оно должно принимать значение по умолчанию.
Значения image и bgColors
должны быть необязательными параметрами.
struct Note: Identifiable, Codable {
let title: String
let message: String
let image: String?
let bgColors: [Color?]//[String?]
let typeBg: String? = "color"
var id = UUID()
}
Но я получаю только ошибки в примечании к структуре:
Тип «Примечание» не соответствует протоколу «Декодируемый»
Тип «Примечание» не соответствует протоколу «Кодируемый»
Что я хотел бы сделать, так это:
если typeBg
из Struct == 'url'
, то я принимаю в качестве значения image
URL-адрес.
если typeBg
из Struct == 'gradient'
, то я принимаю в качестве значения bgColors
, которое представляет собой массив цветов.
contentView:
SmallWidget(entry: Note(title: "Title", message: "Mex", bgColors: bgColors, typeBg: "gradient"))
SmallWidget:
struct SmallWidget: View {
var entry: Note
@Environment(.colorScheme) var colorScheme
func bg() -> AnyView { //<- No work
switch entry.typeBg {
case "url":
return AnyView(NetworkImage(url: URL(string: entry.image))
case "gradient":
return AnyView(
LinearGradient(
gradient: Gradient(colors: entry.bgColors),
startPoint: .top,
endPoint: .bottom)
)
default:
return AnyView(Color.blue)
}
var body: some View {
GeometryReader { geo in
VStack(alignment: .center){
Text(entry.title)
.font(.title)
.bold()
.minimumScaleFactor(0.5)
.foregroundColor(.white)
.shadow(
color: Color.black,
radius: 1.0,
x: CGFloat(4),
y: CGFloat(4))
Text(entry.message)
.foregroundColor(Color.gray)
.shadow(
color: Color.black,
radius: 1.0,
x: CGFloat(4),
y: CGFloat(4))
}
.frame(maxWidth: .infinity, maxHeight: .infinity)
.edgesIgnoringSafeArea(.all)
}
.background(bg)
//.background(gradient)
//.background(NetworkImage(url: URL(string: entry.image)))
}
}
struct NetworkImage: View {
public let url: URL?
var body: some View {
Group {
if let url = url, let imageData = try? Data(contentsOf: url),
let uiImage = UIImage(data: imageData) {
Image(uiImage: uiImage)
.resizable()
.aspectRatio(contentMode: .fill)
}
else {
ProgressView()
}
}
}
}
Ответ №1:
Это заняло довольно много времени, потому Color
что это не Codable
так, поэтому пришлось создать пользовательскую версию. Вот что я получил:
struct Note: Identifiable, Codable {
enum CodingKeys: CodingKey {
case title, message, background
}
let id = UUID()
let title: String
let message: String
let background: NoteBackground
}
extension Note {
enum NoteBackground: Codable {
enum NoteBackgroundError: Error {
case failedToDecode
}
case url(String)
case gradient([Color])
init(from decoder: Decoder) throws {
let container = try decoder.singleValueContainer()
if let url = try? container.decode(String.self) {
self = .url(url)
return
}
if let gradient = try? container.decode([ColorWrapper].self) {
self = .gradient(gradient.map(.color))
return
}
throw NoteBackgroundError.failedToDecode
}
func encode(to encoder: Encoder) throws {
var container = encoder.singleValueContainer()
switch self {
case let .url(url):
try container.encode(url)
case let .gradient(gradient):
let colors = gradient.map(ColorWrapper.init(color:))
try container.encode(colors)
}
}
}
}
Чтобы Color
быть Codable
, он обернут в ColorWrapper
:
enum ColorConvert {
struct Components: Codable {
let red: Double
let green: Double
let blue: Double
let opacity: Double
}
static func toColor(from components: Components) -> Color {
Color(
red: components.red,
green: components.green,
blue: components.blue,
opacity: components.opacity
)
}
static func toComponents(from color: Color) -> Components? {
guard let components = color.cgColor?.components else { return nil }
guard components.count == 4 else { return nil }
let converted = components.map(Double.init)
return Components(
red: converted[0],
green: converted[1],
blue: converted[2],
opacity: converted[3]
)
}
}
struct ColorWrapper: Codable {
let color: Color
init(color: Color) {
self.color = color
}
init(from decoder: Decoder) throws {
let container = try decoder.singleValueContainer()
let components = try container.decode(ColorConvert.Components.self)
color = ColorConvert.toColor(from: components)
}
func encode(to encoder: Encoder) throws {
var container = encoder.singleValueContainer()
let components = ColorConvert.toComponents(from: color)
try container.encode(components)
}
}
Затем его можно использовать следующим образом:
struct ContentView: View {
let data = Note(title: "Title", message: "Message", background: .url("https://google.com"))
//let data = Note(title: "Title", message: "Message", background: .gradient([Color(red: 1, green: 0.5, blue: 0.2), Color(red: 0.3, green: 0.7, blue: 0.8)]))
var body: some View {
Text(String(describing: data))
.onAppear(perform: test)
}
private func test() {
do {
let encodedData = try JSONEncoder().encode(data)
print("encoded", encodedData.base64EncodedString())
let decodedData = try JSONDecoder().decode(Note.self, from: encodedData)
print("decoded", String(describing: decodedData))
} catch let error {
fatalError("Error: (error.localizedDescription)")
}
}
}
Примечание: Color
кодируемый вами код не может быть чем-то вроде Color.red
— он должен быть сделан из компонентов RGB, например, с использованием Color(red:green:blue:)
инициализатора.
Для вас вы могли бы сделать что-то подобное, чтобы изменить фон в зависимости от entry
background
:
@ViewBuilder func bg() -> some View {
switch entry.background {
case let .url(url):
NetworkImage(url: URL(string: url))
case let .gradient(colors):
LinearGradient(
gradient: Gradient(colors: colors),
startPoint: .top,
endPoint: .bottom
)
/// CAN ADD ANOTHER CASE TO `NoteBackground` ENUM FOR SOLID COLOR HERE
}
}
Комментарии:
1. Спасибо за вашу помощь, я тестирую его, но у меня не получается заставить его работать должным образом. Проблемы заключаются в следующем: 1) Фон заметки, как вы его назвали фоном, должен быть фоном виджета SmallWidget, см. Код выше. Я пытаюсь сделать что-то вроде
GeometryReader {geo in ...}.background (entry.background)
, но это не работает, выдает следующую ошибку:Instance method 'background (_: alignment :)' requires that 'Note.NoteBackground 'conform to' View '
. Итак, как я могу использовать разработанную вами структуру?2. 2) В случае, если фоном является URL-адрес изображения, от меня ускользает, как изображение извлекается внутри
NoteBackground
, например, для извлечения изображения из http, который я используюNetworkImage
, вы можете увидеть выше кода.3. @Paul Надеюсь, пример, который я добавил, был понятен. Вы используете не
entry.background
какView
, а вместо этого данные для этих представлений. Сделайте то, что вы делали раньшеGeometryReader { geo in ... }.background(bg)
.4. Спасибо, кажется, работает, ваше мнение по одному вопросу. Как вы можете видеть на изображении, у меня есть средство выбора, чтобы выбрать, откуда должно быть взято изображение. Чтобы избежать необходимости делать это:
if select == 0 {SmallWidget (entry: Note(title: arrayBg[select], message: "(select)", background: .url(url)))}
так и для всех остальных. По вашему мнению, есть способ объявить такую вещь:array = [.url(url), .url(url), .gradient(bgColors)]
а затем просто сделайтеSmallWidget (entry: Note(title: arrayBg [select], message: " (select)", background: array[select]))