Files
SideStore/AltWidget/Widgets/AppDetailWidget.swift

211 lines
8.3 KiB
Swift
Raw Normal View History

2020-09-15 13:51:29 -07:00
//
// AppDetailWidget.swift
// AltWidgetExtension
2020-09-15 13:51:29 -07:00
//
// Created by Riley Testut on 9/14/20.
// Copyright © 2020 Riley Testut. All rights reserved.
//
import WidgetKit
import SwiftUI
import AltStoreCore
struct AppDetailWidget: Widget
{
private let kind: String = "AppDetail"
public var body: some WidgetConfiguration {
let configuration = IntentConfiguration(kind: kind,
intent: ViewAppIntent.self,
provider: AppsTimelineProvider()) { (entry) in
AppDetailWidgetView(entry: entry)
}
.supportedFamilies([.systemSmall])
.configurationDisplayName("App Status")
.description("View remaining days until your sideloaded apps expire. Tap the countdown timer to refresh them in the background.")
if #available(iOS 17, *)
{
return configuration
.contentMarginsDisabled()
}
else
{
return configuration
}
}
}
private struct AppDetailWidgetView: View
2020-09-15 13:51:29 -07:00
{
var entry: AppsEntry
2020-09-15 13:51:29 -07:00
var body: some View {
Group {
if let app = self.entry.apps.first
2020-09-15 13:51:29 -07:00
{
let daysRemaining = app.expirationDate.numberOfCalendarDays(since: self.entry.date)
2020-09-15 13:51:29 -07:00
GeometryReader { (geometry) in
Group {
VStack(alignment: .leading) {
2020-09-15 13:51:29 -07:00
VStack(alignment: .leading, spacing: 5) {
let imageHeight = geometry.size.height * 0.4
2020-09-15 13:51:29 -07:00
Image(uiImage: app.icon ?? UIImage())
.resizable()
.aspectRatio(CGSize(width: 1, height: 1), contentMode: .fit)
.frame(height: imageHeight)
.mask(RoundedRectangle(cornerRadius: imageHeight / 5.0, style: .continuous))
2020-09-15 13:51:29 -07:00
Text(app.name.uppercased())
.font(.system(size: 12, weight: .semibold, design: .rounded))
.foregroundColor(.white)
.lineLimit(1)
.minimumScaleFactor(0.5)
}
.fixedSize(horizontal: false, vertical: true)
2020-09-15 13:51:29 -07:00
Spacer(minLength: 0)
HStack(alignment: .center) {
let expirationText: Text = {
switch daysRemaining
{
case ..<0: return Text("Expired")
case 1: return Text("1 day")
default: return Text("\(daysRemaining) days")
}
}()
2020-09-15 13:51:29 -07:00
(
Text("Expires in\n")
.font(.system(size: 13, weight: .semibold, design: .rounded))
.foregroundColor(Color.white.opacity(0.45)) +
expirationText
2020-09-15 13:51:29 -07:00
.font(.system(size: 15, weight: .semibold, design: .rounded))
.foregroundColor(.white)
)
.lineLimit(2)
.lineSpacing(1.0)
.minimumScaleFactor(0.5)
Spacer()
if daysRemaining >= 0
{
Countdown(startDate: app.refreshedDate,
endDate: app.expirationDate,
currentDate: self.entry.date)
.font(.system(size: 20, weight: .semibold, design: .rounded))
.foregroundColor(Color.white)
.opacity(0.8)
.fixedSize(horizontal: true, vertical: false)
.offset(x: 5)
.invalidatableContentIfAvailable()
}
2020-09-15 13:51:29 -07:00
}
.fixedSize(horizontal: false, vertical: true)
.activatesRefreshAllAppsIntent()
2020-09-15 13:51:29 -07:00
}
.padding()
2020-09-15 13:51:29 -07:00
}
}
}
else
{
VStack {
// Put conditional inside VStack, or else an empty view will be returned
// if isPlaceholder == false, which messes up layout.
if !entry.isPlaceholder
{
Text("App Not Found")
.font(.system(.body, design: .rounded))
.fontWeight(.semibold)
.foregroundColor(Color.white.opacity(0.4))
}
}
.frame(maxWidth: .infinity, maxHeight: .infinity)
}
}
.widgetBackground(
backgroundView(
icon: entry.apps.first?.icon,
tintColor: entry.apps.first?.tintColor
)
)
2020-09-15 13:51:29 -07:00
}
}
private extension AppDetailWidgetView
2020-09-15 13:51:29 -07:00
{
func backgroundView(icon: UIImage? = nil, tintColor: UIColor? = nil) -> some View
{
let icon = icon ?? UIImage(named: "SideStore")!
let tintColor = tintColor ?? .gray
2020-09-15 13:51:29 -07:00
let imageHeight = 60 as CGFloat
let saturation = 1.8
let blurRadius = 5 as CGFloat
let tintOpacity = 0.45
// 1024x1024 images are not supported by previews but supported by device
// so we scale the image to 97% so as to reduce its actual size but not too much
// to somewhere below value, acceptable by previews ie < 1042x948
let scalingFactor = 0.97
let resizedSize = CGSize(
width: icon.size.width * scalingFactor,
height: icon.size.height * scalingFactor
)
let resizedIcon = icon.resizing(to: resizedSize)!
2020-09-15 13:51:29 -07:00
return ZStack(alignment: .topTrailing) {
// Blurred Image
GeometryReader { geometry in
ZStack {
Image(uiImage: resizedIcon)
2020-09-15 13:51:29 -07:00
.resizable()
.aspectRatio(contentMode: .fill)
.frame(width: imageHeight, height: imageHeight, alignment: .center)
.saturation(saturation)
.blur(radius: blurRadius, opaque: true)
.scaleEffect(geometry.size.width / imageHeight, anchor: .center)
// .onAppear {
// print("Geometry size: \(geometry.size)")
// print("Image height: \(imageHeight), Geometry width: \(geometry.size.width)")
// print("Icon size: \(icon.size)")
// }
2020-09-15 13:51:29 -07:00
Color(tintColor)
.opacity(tintOpacity)
}
}
Image("Badge")
.resizable()
.frame(width: 26, height: 26)
.padding()
}
}
}
@available(iOS 17, *)
#Preview(as: .systemSmall) {
AppDetailWidget()
} timeline: {
let expiredDate = Date().addingTimeInterval(1 * 60 * 60 * 24 * 7)
let (altstore, _, _, longAltStore, _, _) = AppSnapshot.makePreviewSnapshots()
AppsEntry(date: Date(), apps: [altstore])
AppsEntry(date: Date(), apps: [longAltStore])
AppsEntry(date: expiredDate, apps: [altstore])
AppsEntry(date: Date(), apps: [])
AppsEntry(date: Date(), apps: [], isPlaceholder: true)
2020-09-15 13:51:29 -07:00
}