Фон виджета SwiftUI на основе переданного значения URL-адреса изображения или градиентного фона

#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]))