зеркало из https://github.com/nextcloud/ios.git
205 строки
9.7 KiB
Swift
205 строки
9.7 KiB
Swift
//
|
|
// DashboardWidgetView.swift
|
|
// Widget
|
|
//
|
|
// Created by Marino Faggiana on 20/08/22.
|
|
// Copyright © 2022 Marino Faggiana. All rights reserved.
|
|
//
|
|
// Author Marino Faggiana <marino.faggiana@nextcloud.com>
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
//
|
|
|
|
import SwiftUI
|
|
import WidgetKit
|
|
|
|
struct DashboardWidgetView: View {
|
|
var entry: DashboardDataEntry
|
|
var body: some View {
|
|
GeometryReader { geo in
|
|
if entry.isEmpty {
|
|
VStack(alignment: .center) {
|
|
Image(systemName: "checkmark")
|
|
.resizable()
|
|
.scaledToFit()
|
|
.font(Font.system(.body).weight(.light))
|
|
.frame(width: 50, height: 50)
|
|
Text(NSLocalizedString("_no_items_", comment: ""))
|
|
.font(.system(size: 25))
|
|
.padding()
|
|
Text(NSLocalizedString("_check_back_later_", comment: ""))
|
|
.font(.system(size: 15))
|
|
}
|
|
.frame(width: geo.size.width, height: geo.size.height)
|
|
}
|
|
|
|
ZStack(alignment: .topLeading) {
|
|
HStack {
|
|
Image(uiImage: entry.titleImage)
|
|
.renderingMode(.template)
|
|
.resizable()
|
|
.scaledToFill()
|
|
.frame(width: 20, height: 20)
|
|
|
|
Text(entry.title)
|
|
.font(.system(size: 15))
|
|
.fontWeight(.bold)
|
|
.multilineTextAlignment(.center)
|
|
.textCase(.uppercase)
|
|
.lineLimit(1)
|
|
}
|
|
.frame(width: geo.size.width - 20)
|
|
.padding([.top, .leading, .trailing], 10)
|
|
|
|
if !entry.isEmpty {
|
|
|
|
VStack(alignment: .leading) {
|
|
|
|
VStack(spacing: 0) {
|
|
|
|
ForEach(entry.datas, id: \.id) { element in
|
|
|
|
Link(destination: element.link) {
|
|
|
|
HStack {
|
|
|
|
let subTitleColor = Color(white: 0.5)
|
|
|
|
if entry.isPlaceholder {
|
|
Circle()
|
|
.fill(Color(.systemGray4))
|
|
.frame(width: 35, height: 35)
|
|
} else if let color = element.imageColor {
|
|
Image(uiImage: element.icon)
|
|
.renderingMode(.template)
|
|
.resizable()
|
|
.frame(width: 20, height: 20)
|
|
.foregroundColor(Color(color))
|
|
} else if element.template {
|
|
if entry.dashboard?.itemIconsRound ?? false {
|
|
Image(uiImage: element.icon)
|
|
.renderingMode(.template)
|
|
.resizable()
|
|
.scaledToFill()
|
|
.frame(width: 20, height: 20)
|
|
.foregroundColor(.white)
|
|
.padding(8)
|
|
.background(Color(.systemGray4))
|
|
.clipShape(Circle())
|
|
} else {
|
|
Image(uiImage: element.icon)
|
|
.renderingMode(.template)
|
|
.resizable()
|
|
.scaledToFill()
|
|
.frame(width: 25, height: 25)
|
|
.clipped()
|
|
.cornerRadius(5)
|
|
}
|
|
} else {
|
|
if entry.dashboard?.itemIconsRound ?? false || element.avatar {
|
|
Image(uiImage: element.icon)
|
|
.resizable()
|
|
.scaledToFill()
|
|
.frame(width: 35, height: 35)
|
|
.clipShape(Circle())
|
|
} else {
|
|
Image(uiImage: element.icon)
|
|
.resizable()
|
|
.scaledToFill()
|
|
.frame(width: 35, height: 35)
|
|
.clipped()
|
|
.cornerRadius(5)
|
|
}
|
|
}
|
|
|
|
VStack(alignment: .leading, spacing: 2) {
|
|
|
|
Text(element.title)
|
|
.font(.system(size: 12))
|
|
.fontWeight(.regular)
|
|
|
|
Text(element.subTitle)
|
|
.font(.system(size: CGFloat(10)))
|
|
.foregroundColor(subTitleColor)
|
|
}
|
|
Spacer()
|
|
}
|
|
.padding(.leading, 10)
|
|
.frame(height: 50)
|
|
}
|
|
if element != entry.datas.last {
|
|
Divider()
|
|
.padding(.leading, 54)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.padding(.top, 35)
|
|
.redacted(reason: entry.isPlaceholder ? .placeholder : [])
|
|
}
|
|
|
|
if let buttons = entry.buttons, !buttons.isEmpty, !entry.isPlaceholder {
|
|
|
|
HStack(spacing: 10) {
|
|
let brandColor = Color(NCBrandColor.shared.getElement(account: entry.account))
|
|
let brandTextColor = Color(NCBrandColor.shared.getText(account: entry.account))
|
|
|
|
ForEach(buttons, id: \.index) { element in
|
|
Link(destination: URL(string: element.link)!, label: {
|
|
|
|
Text(element.text)
|
|
.font(.system(size: 15))
|
|
.padding(7)
|
|
.background(brandColor)
|
|
.foregroundColor(brandTextColor)
|
|
.border(brandColor, width: 1)
|
|
.cornerRadius(.infinity)
|
|
})
|
|
}
|
|
}
|
|
.frame(width: geo.size.width - 10, height: geo.size.height - 25, alignment: .bottomTrailing)
|
|
}
|
|
|
|
HStack {
|
|
|
|
Image(systemName: entry.footerImage)
|
|
.resizable()
|
|
.scaledToFit()
|
|
.frame(width: 15, height: 15)
|
|
.font(Font.system(.body).weight(.light))
|
|
.foregroundColor(entry.isPlaceholder ? Color(.systemGray4) : Color(NCBrandColor.shared.getElement(account: entry.account)))
|
|
|
|
Text(entry.footerText)
|
|
.font(.caption2)
|
|
.lineLimit(1)
|
|
.foregroundColor(entry.isPlaceholder ? Color(.systemGray4) : Color(NCBrandColor.shared.getElement(account: entry.account)))
|
|
}
|
|
.padding(.horizontal, 15.0)
|
|
.frame(maxWidth: geo.size.width, maxHeight: geo.size.height - 2, alignment: .bottomTrailing)
|
|
}
|
|
}
|
|
.widgetBackground(Color(UIColor.systemBackground))
|
|
}
|
|
}
|
|
|
|
struct DashboardWidget_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
let datas = Array(dashboardDatasTest[0...4])
|
|
let title = "Dashboard"
|
|
let titleImage = UIImage(named: "widget")!
|
|
let entry = DashboardDataEntry(date: Date(), datas: datas, dashboard: nil, buttons: nil, isPlaceholder: false, isEmpty: true, titleImage: titleImage, title: title, footerImage: "checkmark.icloud", footerText: "Nextcloud widget", account: "")
|
|
DashboardWidgetView(entry: entry).previewContext(WidgetPreviewContext(family: .systemLarge))
|
|
}
|
|
}
|