@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.620",
|
||||
"green" : "0.870",
|
||||
"red" : "0.910"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.620",
|
||||
"green" : "0.870",
|
||||
"red" : "0.910"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"filename" : "beefdairysquare_36.png",
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
After Width: | Height: | Size: 12 KiB |
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.560",
|
||||
"green" : "0.230",
|
||||
"red" : "0.080"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.560",
|
||||
"green" : "0.230",
|
||||
"red" : "0.080"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"filename" : "bullseye-cover2-1024x1024.jpg",
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
After Width: | Height: | Size: 26 KiB |
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.000",
|
||||
"green" : "0.810",
|
||||
"red" : "1.000"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.000",
|
||||
"green" : "0.810",
|
||||
"red" : "1.000"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"filename" : "DGS-podcastArt.png",
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
After Width: | Height: | Size: 31 KiB |
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.700",
|
||||
"green" : "0.670",
|
||||
"red" : "0.010"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.700",
|
||||
"green" : "0.670",
|
||||
"red" : "0.010"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"filename" : "jjgo_logo.png",
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
After Width: | Height: | Size: 179 KiB |
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.350",
|
||||
"green" : "0.040",
|
||||
"red" : "0.080"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.350",
|
||||
"green" : "0.040",
|
||||
"red" : "0.080"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"filename" : "Screenshot 2023-05-21 at 18.58.17.png",
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
After Width: | Height: | Size: 271 KiB |
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.340",
|
||||
"green" : "0.130",
|
||||
"red" : "0.050"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.340",
|
||||
"green" : "0.130",
|
||||
"red" : "0.050"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"filename" : "oh-no-ross-and-carrie-cover-1024x1024.jpg",
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
After Width: | Height: | Size: 46 KiB |
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.750",
|
||||
"green" : "0.080",
|
||||
"red" : "0.030"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.750",
|
||||
"green" : "0.080",
|
||||
"red" : "0.030"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"filename" : "spy_logo.jpg",
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
After Width: | Height: | Size: 40 KiB |
@ -0,0 +1,6 @@
|
||||
{
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.871",
|
||||
"green" : "0.314",
|
||||
"red" : "0.490"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.871",
|
||||
"green" : "0.314",
|
||||
"red" : "0.490"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.980",
|
||||
"green" : "0.980",
|
||||
"red" : "0.980"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.020",
|
||||
"green" : "0.020",
|
||||
"red" : "0.020"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.900",
|
||||
"green" : "0.900",
|
||||
"red" : "0.900"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.250",
|
||||
"green" : "0.250",
|
||||
"red" : "0.250"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.090",
|
||||
"green" : "0.090",
|
||||
"red" : "0.090"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.910",
|
||||
"green" : "0.910",
|
||||
"red" : "0.910"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.666",
|
||||
"green" : "0.666",
|
||||
"red" : "0.666"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.833",
|
||||
"green" : "0.833",
|
||||
"red" : "0.833"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.833",
|
||||
"green" : "0.833",
|
||||
"red" : "0.833"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.333",
|
||||
"green" : "0.333",
|
||||
"red" : "0.333"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,6 @@
|
||||
{
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.100",
|
||||
"green" : "0.100",
|
||||
"red" : "0.100"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.100",
|
||||
"green" : "0.100",
|
||||
"red" : "0.100"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.900",
|
||||
"green" : "0.900",
|
||||
"red" : "0.900"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.900",
|
||||
"green" : "0.900",
|
||||
"red" : "0.900"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.910",
|
||||
"green" : "0.910",
|
||||
"red" : "0.910"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.910",
|
||||
"green" : "0.910",
|
||||
"red" : "0.910"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.090",
|
||||
"green" : "0.090",
|
||||
"red" : "0.090"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "1.000",
|
||||
"blue" : "0.090",
|
||||
"green" : "0.090",
|
||||
"red" : "0.090"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "0.800",
|
||||
"blue" : "0.800",
|
||||
"green" : "0.800",
|
||||
"red" : "0.800"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "0.800",
|
||||
"blue" : "0.800",
|
||||
"green" : "0.800",
|
||||
"red" : "0.800"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
{
|
||||
"colors" : [
|
||||
{
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "0.800",
|
||||
"blue" : "0.250",
|
||||
"green" : "0.250",
|
||||
"red" : "0.250"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
},
|
||||
{
|
||||
"appearances" : [
|
||||
{
|
||||
"appearance" : "luminosity",
|
||||
"value" : "dark"
|
||||
}
|
||||
],
|
||||
"color" : {
|
||||
"color-space" : "srgb",
|
||||
"components" : {
|
||||
"alpha" : "0.800",
|
||||
"blue" : "0.250",
|
||||
"green" : "0.250",
|
||||
"red" : "0.250"
|
||||
}
|
||||
},
|
||||
"idiom" : "universal"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
@ -1,26 +0,0 @@
|
||||
//
|
||||
// ContentView.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-12.
|
||||
//
|
||||
|
||||
import SwiftUI
|
||||
|
||||
struct ContentView: View {
|
||||
var body: some View {
|
||||
VStack {
|
||||
Image(systemName: "globe")
|
||||
.imageScale(.large)
|
||||
.foregroundColor(.accentColor)
|
||||
Text("Hello, World!")
|
||||
}
|
||||
.padding()
|
||||
}
|
||||
}
|
||||
|
||||
struct ContentView_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
ContentView()
|
||||
}
|
||||
}
|
@ -0,0 +1,15 @@
|
||||
//
|
||||
// PodcastEpisode.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-21.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
struct Episode {
|
||||
var id: UUID
|
||||
var publicationDate: Date
|
||||
var title: String
|
||||
var description: String
|
||||
var duration: TimeInterval // duration in seconds
|
||||
}
|
@ -0,0 +1,59 @@
|
||||
//
|
||||
// Podcast.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-16.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import SwiftUI
|
||||
|
||||
struct Podcast {
|
||||
var id: UUID
|
||||
var image: UIImage
|
||||
var title: String
|
||||
var by: String
|
||||
private var _episodes: [Episode]
|
||||
var rating: Double
|
||||
var reviews: Int
|
||||
var genre: String
|
||||
var frequency: String
|
||||
var backgroundColor: Color
|
||||
var backgroundIsDark: Bool
|
||||
|
||||
init(
|
||||
id: UUID,
|
||||
image: UIImage,
|
||||
title: String,
|
||||
by: String,
|
||||
episodes: [Episode],
|
||||
rating: Double,
|
||||
reviews: Int,
|
||||
genre: String,
|
||||
frequency: String = "Unknown",
|
||||
backgroundColor: Color = Color.theme.background,
|
||||
backgroundIsDark: Bool = false
|
||||
) {
|
||||
self.id = id
|
||||
self.image = image
|
||||
self.title = title
|
||||
self.by = by
|
||||
self._episodes = episodes
|
||||
self.rating = rating
|
||||
self.reviews = reviews
|
||||
self.genre = genre
|
||||
self.frequency = frequency
|
||||
self.backgroundColor = backgroundColor
|
||||
self.backgroundIsDark = backgroundIsDark
|
||||
}
|
||||
|
||||
|
||||
var episodes: [Episode] {
|
||||
_episodes.sorted(by: { $0.publicationDate > $1.publicationDate })
|
||||
}
|
||||
|
||||
var latestEpisodeDescription: String {
|
||||
guard !episodes.isEmpty else { return "No episodes available" }
|
||||
return episodes.first?.description ?? "No description available"
|
||||
}
|
||||
}
|
@ -0,0 +1,77 @@
|
||||
//
|
||||
// NowPlayingBar.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-21.
|
||||
//
|
||||
|
||||
// Thanks to Luca Jonscher for this tutorial: https://itnext.io/add-a-now-playing-bar-with-swiftui-to-your-app-d515b03f05e3
|
||||
// This bar is almost all made by them. The mistakes are all mine.
|
||||
import SwiftUI
|
||||
|
||||
struct NowPlayingBar<Content: View>: View {
|
||||
var content: Content
|
||||
|
||||
@ViewBuilder var body: some View {
|
||||
ZStack {
|
||||
Blur(style: .systemMaterial) // Use Blur view as background
|
||||
.frame(width: UIScreen.main.bounds.size.width, height: 64)
|
||||
|
||||
VStack {
|
||||
HStack {
|
||||
Button(action: {}) {
|
||||
HStack {
|
||||
Image("jjho_logo")
|
||||
.resizable()
|
||||
.frame(width: 48, height: 48)
|
||||
.shadow(radius: 4, x: 0, y: 2)
|
||||
.padding(.leading)
|
||||
VStack(alignment: .leading) {
|
||||
Text("Acting in Bat Faith")
|
||||
.foregroundColor(Color.theme.primary)
|
||||
Text("1 February 2023")
|
||||
.font(.caption)
|
||||
.foregroundColor(Color.theme.secondary)
|
||||
}.padding()
|
||||
Spacer()
|
||||
}
|
||||
}
|
||||
.buttonStyle(PlainButtonStyle())
|
||||
|
||||
Button(action: {}) {
|
||||
Image(systemName: "play.fill")
|
||||
.font(.title3)
|
||||
}
|
||||
.buttonStyle(PlainButtonStyle())
|
||||
.padding(.horizontal)
|
||||
|
||||
Button(action: {}) {
|
||||
Image(systemName: "gobackward.30")
|
||||
.font(.title3)
|
||||
}
|
||||
.buttonStyle(PlainButtonStyle())
|
||||
.padding(.trailing, 30)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
struct Blur: UIViewRepresentable {
|
||||
var style: UIBlurEffect.Style = .systemChromeMaterial
|
||||
|
||||
func makeUIView(context: Context) -> UIVisualEffectView {
|
||||
return UIVisualEffectView(effect: UIBlurEffect(style: style))
|
||||
}
|
||||
|
||||
func updateUIView(_ uiView: UIVisualEffectView, context: Context) {
|
||||
uiView.effect = UIBlurEffect(style: style)
|
||||
}
|
||||
}
|
||||
|
||||
struct NowPlayingBar_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
NowPlayingBar(content: Text("Hello Bar"))
|
||||
}
|
||||
}
|
@ -0,0 +1,83 @@
|
||||
//
|
||||
// PodcastEpisodeViewCell.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-12.
|
||||
//
|
||||
|
||||
import SwiftUI
|
||||
|
||||
|
||||
struct EpisodeViewCell: View {
|
||||
|
||||
let episode: Episode
|
||||
|
||||
var body: some View {
|
||||
VStack {
|
||||
Divider()
|
||||
.foregroundColor(Color.theme.backgroundSecondary)
|
||||
.edgesIgnoringSafeArea(.trailing)
|
||||
.padding(.leading)
|
||||
|
||||
VStack(alignment: .leading) {
|
||||
|
||||
Text(formatter.localizedString(for: episode.publicationDate, relativeTo: Date()))
|
||||
.font(.subheadline)
|
||||
.foregroundColor(Color.theme.secondary)
|
||||
|
||||
(Text(episode.title)
|
||||
.font(.headline)
|
||||
.foregroundColor(Color.theme.primary)
|
||||
+ Text("\n\(episode.description)")
|
||||
.font(.body)
|
||||
.foregroundColor(Color.theme.secondary))
|
||||
.lineLimit(4)
|
||||
.truncationMode(.tail)
|
||||
|
||||
HStack {
|
||||
Image(systemName: "play.fill")
|
||||
.foregroundColor(Color.theme.accent)
|
||||
.padding()
|
||||
.background(Color.theme.backgroundSecondary)
|
||||
.clipShape(Circle())
|
||||
Text(timeString(time: episode.duration))
|
||||
.foregroundColor(Color.theme.accent)
|
||||
Spacer()
|
||||
Text(Strings.threeDots)
|
||||
.foregroundColor(Color.theme.secondary)
|
||||
.padding(.horizontal)
|
||||
}
|
||||
}
|
||||
.padding()
|
||||
}
|
||||
}
|
||||
|
||||
private func timeString(time: TimeInterval) -> String {
|
||||
let hours = Int(time) / 3600
|
||||
let minutes = Int(time) / 60 % 60
|
||||
|
||||
var timeComponents = [String]()
|
||||
if hours > 0 {
|
||||
timeComponents.append("\(hours) hr")
|
||||
}
|
||||
if minutes > 0 {
|
||||
timeComponents.append("\(minutes) min")
|
||||
}
|
||||
|
||||
return timeComponents.joined(separator: " ")
|
||||
}
|
||||
|
||||
private let formatter: RelativeDateTimeFormatter = {
|
||||
let formatter = RelativeDateTimeFormatter()
|
||||
formatter.unitsStyle = .full
|
||||
return formatter
|
||||
}()
|
||||
|
||||
}
|
||||
|
||||
struct EpisodeViewCell_Previews: PreviewProvider {
|
||||
|
||||
static var previews: some View {
|
||||
EpisodeViewCell(episode: Stub.episodes[1])
|
||||
}
|
||||
}
|
@ -0,0 +1,28 @@
|
||||
//
|
||||
// Color.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-21.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import SwiftUI
|
||||
|
||||
extension Color {
|
||||
static let theme = ColorTheme()
|
||||
}
|
||||
|
||||
struct ColorTheme {
|
||||
let primary = Color("primary")
|
||||
let secondary = Color("secondary")
|
||||
let background = Color("background")
|
||||
let backgroundSecondary = Color("backgroundSecondary")
|
||||
let accent = Color("accent")
|
||||
let shadow = Color("shadow")
|
||||
let unchangingPrimaryDark = Color("primaryDark")
|
||||
let unchangingPrimaryLight = Color("primaryLight")
|
||||
let unchangingSecondaryDark = Color("secondaryDark")
|
||||
let unchangingSecondaryLight = Color("secondaryLight")
|
||||
let unchangingBackgroundDark = Color("backgroundDark")
|
||||
let unchangingBackgroundLight = Color("backgroundLight")
|
||||
}
|
@ -0,0 +1,49 @@
|
||||
//
|
||||
// LibraryView.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-16.
|
||||
//
|
||||
|
||||
import SwiftUI
|
||||
|
||||
struct LibraryView: View {
|
||||
|
||||
var podcasts: [Podcast]
|
||||
|
||||
var body: some View {
|
||||
NavigationView {
|
||||
ScrollView {
|
||||
VStack(alignment: .leading) {
|
||||
Text("Podcasts")
|
||||
.font(.largeTitle)
|
||||
.fontWeight(.bold)
|
||||
.foregroundColor(Color.theme.primary)
|
||||
.padding()
|
||||
|
||||
let columns = [
|
||||
GridItem(.flexible(), spacing: 16),
|
||||
GridItem(.flexible(), spacing: 16)
|
||||
]
|
||||
|
||||
LazyVGrid(columns: columns, spacing: 16) {
|
||||
ForEach(podcasts, id: \.id) { podcast in
|
||||
NavigationLink(
|
||||
destination: PodcastDetailView(podcast: podcast)) {
|
||||
PodcastViewCell(podcast: podcast)
|
||||
}
|
||||
}
|
||||
}
|
||||
.padding(.horizontal)
|
||||
}
|
||||
}
|
||||
}
|
||||
.navigationViewStyle(StackNavigationViewStyle())
|
||||
}
|
||||
}
|
||||
|
||||
struct LibraryView_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
LibraryView(podcasts: Stub.podcasts)
|
||||
}
|
||||
}
|
@ -0,0 +1,50 @@
|
||||
//
|
||||
// MainView.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-12.
|
||||
//
|
||||
|
||||
import SwiftUI
|
||||
|
||||
struct MainView: View {
|
||||
var body: some View {
|
||||
ZStack {
|
||||
|
||||
TabView {
|
||||
Text("Check out the library instead")
|
||||
.tabItem {
|
||||
Label("Listen Now", systemImage: "play")
|
||||
}
|
||||
Text("I swear, the library is where it's at")
|
||||
.tabItem {
|
||||
Label("Browse", systemImage: "square.grid.2x2")
|
||||
}
|
||||
LibraryView(podcasts: Stub.podcasts)
|
||||
.tabItem {
|
||||
Label("Library", systemImage: "book")
|
||||
}
|
||||
Text("Nothing to see here. The library, on the other hand...")
|
||||
.tabItem {
|
||||
Label("Search", systemImage: "magnifyingglass")
|
||||
}
|
||||
}
|
||||
.accentColor(Color.theme.accent)
|
||||
.tabViewStyle(.automatic)
|
||||
|
||||
VStack {
|
||||
Spacer()
|
||||
VStack{
|
||||
NowPlayingBar(content: Text("Gahhh!"))
|
||||
}
|
||||
.offset(y: -42)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
struct MainView_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
MainView()
|
||||
}
|
||||
}
|
@ -0,0 +1,109 @@
|
||||
//
|
||||
// PodcastDetailView.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-12.
|
||||
//
|
||||
|
||||
import SwiftUI
|
||||
|
||||
struct PodcastDetailView: View {
|
||||
|
||||
let podcast: Podcast
|
||||
|
||||
var body: some View {
|
||||
ScrollView {
|
||||
ZStack() {
|
||||
podcast.backgroundColor.ignoresSafeArea(.all, edges: .all)
|
||||
VStack(alignment: .center) {
|
||||
VStack(alignment: .center) {
|
||||
|
||||
Image(uiImage: podcast.image)
|
||||
.resizable()
|
||||
.scaledToFit()
|
||||
.cornerRadius(12)
|
||||
.shadow(color: Color.theme.unchangingPrimaryLight, radius: 10, x: 0, y: 5)
|
||||
.padding(.horizontal, 48)
|
||||
.padding(.vertical, 16)
|
||||
|
||||
Text(podcast.title)
|
||||
.font(.title)
|
||||
.foregroundColor(podcast.backgroundIsDark ? Color.theme.unchangingPrimaryDark : Color.theme.unchangingPrimaryLight)
|
||||
.multilineTextAlignment(.center)
|
||||
|
||||
Text(podcast.by)
|
||||
.font(.headline)
|
||||
.foregroundColor(podcast.backgroundIsDark ? Color.theme.unchangingSecondaryDark : Color.theme.unchangingSecondaryLight)
|
||||
.multilineTextAlignment(.center)
|
||||
|
||||
Button(action: {}) {
|
||||
HStack {
|
||||
Image(systemName: "play.fill")
|
||||
.padding(.horizontal, 4)
|
||||
Text(Strings.latestEpisode)
|
||||
}}
|
||||
.padding(.vertical)
|
||||
.padding(.horizontal, 64)
|
||||
.background(podcast.backgroundIsDark ? Color.theme.unchangingBackgroundLight : Color.theme.unchangingBackgroundDark)
|
||||
.foregroundColor(podcast.backgroundIsDark ? Color.theme.unchangingPrimaryLight : Color.theme.unchangingPrimaryDark)
|
||||
.clipShape(RoundedRectangle(cornerSize: CGSize(width: 12.0, height: 12.0)))
|
||||
|
||||
// TODO replace '...' with Strings.readFurtherPrompt
|
||||
Text(podcast.latestEpisodeDescription)
|
||||
.lineLimit(3)
|
||||
.truncationMode(.tail)
|
||||
.padding()
|
||||
.foregroundColor(podcast.backgroundIsDark ? Color.theme.unchangingPrimaryDark : Color.theme.unchangingPrimaryLight)
|
||||
|
||||
HStack() {
|
||||
Text("\(Image(systemName: "star.fill")) \(podcast.rating, specifier: "%.1f") (\(podcast.reviews)) \(Strings.classySeparator) \(podcast.genre) \(Strings.classySeparator) \(podcast.frequency)")
|
||||
.padding(.horizontal)
|
||||
|
||||
Spacer()
|
||||
}
|
||||
.foregroundColor(podcast.backgroundIsDark ? Color.theme.unchangingSecondaryDark : Color.theme.unchangingSecondaryLight)
|
||||
}
|
||||
|
||||
Divider()
|
||||
.foregroundColor(Color.theme.backgroundSecondary)
|
||||
|
||||
ZStack() {
|
||||
Color.theme.background.ignoresSafeArea(.all, edges: .all)
|
||||
|
||||
VStack(alignment: .leading) {
|
||||
HStack {
|
||||
Text(Strings.episodes)
|
||||
.font(.title2)
|
||||
.fontWeight(.bold)
|
||||
.foregroundColor(Color.theme.primary)
|
||||
.padding()
|
||||
|
||||
Image(systemName: "chevron.down")
|
||||
.foregroundColor(Color.theme.accent)
|
||||
|
||||
Spacer()
|
||||
|
||||
Text(Strings.seeAll)
|
||||
.foregroundColor(Color.theme.accent)
|
||||
.padding(.trailing)
|
||||
}
|
||||
|
||||
|
||||
ForEach(podcast.episodes, id: \.id) { episode in
|
||||
EpisodeViewCell(episode: episode)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
struct PodcastDetailView_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
PodcastDetailView(podcast: Stub.podcasts[2])
|
||||
PodcastDetailView(podcast: Stub.podcasts[2])
|
||||
}
|
||||
}
|
@ -0,0 +1,54 @@
|
||||
//
|
||||
// PodcastViewCell.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-21.
|
||||
//
|
||||
|
||||
import SwiftUI
|
||||
|
||||
struct PodcastViewCell: View {
|
||||
let podcast: Podcast
|
||||
|
||||
var body: some View {
|
||||
VStack(alignment: .leading) {
|
||||
Image(uiImage: podcast.image)
|
||||
.resizable()
|
||||
.scaledToFit()
|
||||
.cornerRadius(12)
|
||||
|
||||
Text(podcast.title)
|
||||
.foregroundColor(Color.theme.primary)
|
||||
|
||||
Text("Updated \(smartDate(podcast.episodes.first?.publicationDate ?? Date()))") .foregroundColor(Color.theme.secondary)
|
||||
.font(.footnote)
|
||||
}
|
||||
}
|
||||
|
||||
func smartDate(_ date: Date) -> String {
|
||||
let calendar = Calendar.current
|
||||
let now = Date()
|
||||
let components = calendar.dateComponents([.day, .year], from: date, to: now)
|
||||
|
||||
if let day = components.day {
|
||||
switch day {
|
||||
case 0:
|
||||
return "Today"
|
||||
case 1..<7:
|
||||
return "\(day)d ago"
|
||||
default:
|
||||
let formatter = DateFormatter()
|
||||
formatter.dateFormat = "dd MMMM yyyy"
|
||||
return formatter.string(from: date)
|
||||
}
|
||||
} else {
|
||||
return "Unknown"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
struct PodcastViewCell_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
PodcastViewCell(podcast: Stub.podcasts[1])
|
||||
}
|
||||
}
|
@ -0,0 +1,17 @@
|
||||
//
|
||||
// Strings.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-21.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
struct Strings {
|
||||
static let threeDots = "···"
|
||||
static let classySeparator = "·"
|
||||
static let latestEpisode = "Latest Episode"
|
||||
static let readFurtherPrompt = "MORE"
|
||||
static let episodes = "Episodes"
|
||||
static let seeAll = "See All"
|
||||
}
|
@ -0,0 +1,149 @@
|
||||
//
|
||||
// Stub.swift
|
||||
// PodcastsClone
|
||||
//
|
||||
// Created by etudiant on 2023-05-21.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import SwiftUI
|
||||
|
||||
struct Stub {
|
||||
static let episodes: [Episode] = [
|
||||
Episode(
|
||||
id: UUID(),
|
||||
publicationDate: Date.now.addingTimeInterval(-1000000000),
|
||||
title: "A New Ipsum",
|
||||
description: "Stand in doorway, unwilling to chose whether to stay in or go out more napping, more napping all the napping is exhausting sleep i'm bored inside, let me out i'm lonely outside, let me in i can't make up my mind whether to go in or out, guess i'll just stand partway in and partway out, contemplating the universe for half an hour how dare you nudge me with your foot?!?! leap into the air in greatest offense!",
|
||||
duration: 3463
|
||||
),
|
||||
Episode(
|
||||
id: UUID(),
|
||||
publicationDate: Date.now.addingTimeInterval(-1000000),
|
||||
title: "Return of the Hooman",
|
||||
description: "Catch mouse and gave it as a present mewl for food at 4am drink water out of the faucet and have secret plans. Stretch chase dog then run away. Kitty. Mouse if it fits, i sits. Bite off human's toes. If human is on laptop sit on the keyboard.",
|
||||
duration: 4480
|
||||
),
|
||||
Episode(
|
||||
id: UUID(),
|
||||
publicationDate: Date.now.addingTimeInterval(-100000000),
|
||||
title: "Cat Ipsum Strikes Back",
|
||||
description: "Chase after silly colored fish toys around the house i want to go outside let me go outside nevermind inside is better or get video posted to internet for chasing red dot eat owner's food wack the mini furry mouse so cat meoooow i iz master of hoomaan, not hoomaan master of i, oooh damn dat dog but stuff and things. Cats making all the muffins.",
|
||||
duration: 4028
|
||||
),
|
||||
]
|
||||
|
||||
static let episodesOld = Stub.episodes.map { episode in
|
||||
Episode(
|
||||
id: episode.id,
|
||||
publicationDate: Date.now.addingTimeInterval((episode.publicationDate.timeIntervalSinceNow - Date.now.timeIntervalSinceNow) * 10),
|
||||
title: episode.title,
|
||||
description: episode.description,
|
||||
duration: episode.duration
|
||||
)
|
||||
}
|
||||
|
||||
static let episodesRecent = Stub.episodes.map { episode in
|
||||
Episode(
|
||||
id: episode.id,
|
||||
publicationDate: Date.now.addingTimeInterval((episode.publicationDate.timeIntervalSinceNow - Date.now.timeIntervalSinceNow) / 10.0),
|
||||
title: episode.title,
|
||||
description: episode.description,
|
||||
duration: episode.duration
|
||||
)
|
||||
}
|
||||
|
||||
static let podcasts: [Podcast] = [
|
||||
Podcast(
|
||||
id: UUID(),
|
||||
image: UIImage(named: "jjho_logo")!,
|
||||
title: "Podcast Title 1",
|
||||
by: "Author 1",
|
||||
episodes: episodesOld,
|
||||
rating: 4.2,
|
||||
reviews: 2139,
|
||||
genre: "Genre 1",
|
||||
frequency: "Weekly",
|
||||
backgroundColor: Color("jjhoColor"),
|
||||
backgroundIsDark: true
|
||||
),
|
||||
Podcast(
|
||||
id: UUID(),
|
||||
image: UIImage(named: "jjgo_logo")!,
|
||||
title: "Podcast Title 2",
|
||||
by: "Author 2",
|
||||
episodes: episodes,
|
||||
rating: 4.2,
|
||||
reviews: 211,
|
||||
genre: "Genre 2",
|
||||
frequency: "Twice weekly",
|
||||
backgroundColor: Color("jjgoColor"),
|
||||
backgroundIsDark: true
|
||||
),
|
||||
Podcast(
|
||||
id: UUID(),
|
||||
image: UIImage(named: "spy_logo")!,
|
||||
title: "Podcast Title 3",
|
||||
by: "Author 3",
|
||||
episodes: episodesRecent,
|
||||
rating: 4.812039,
|
||||
reviews: 3981,
|
||||
genre: "Genre 3",
|
||||
frequency: "Complete",
|
||||
backgroundColor: Color("spyColor"),
|
||||
backgroundIsDark: true
|
||||
),
|
||||
Podcast(
|
||||
id: UUID(),
|
||||
image: UIImage(named: "bdnp_logo")!,
|
||||
title: "Podcast Title 4",
|
||||
by: "Author 4",
|
||||
episodes: episodesOld,
|
||||
rating: 4.2,
|
||||
reviews: 211,
|
||||
genre: "Genre 4",
|
||||
frequency: "Monthly",
|
||||
backgroundColor: Color("bdnpColor"),
|
||||
backgroundIsDark: false
|
||||
),
|
||||
Podcast(
|
||||
id: UUID(),
|
||||
image: UIImage(named: "bewjt_logo")!,
|
||||
title: "Podcast Title 5",
|
||||
by: "Author 5",
|
||||
episodes: episodes,
|
||||
rating: 4.2,
|
||||
reviews: 211,
|
||||
genre: "Genre 5",
|
||||
frequency: "Daily",
|
||||
backgroundColor: Color("bewjtColor"),
|
||||
backgroundIsDark: true
|
||||
),
|
||||
Podcast(
|
||||
id: UUID(),
|
||||
image: UIImage(named: "onrac_logo")!,
|
||||
title: "Podcast Title 6",
|
||||
by: "Author 6",
|
||||
episodes: episodesRecent,
|
||||
rating: 4.2,
|
||||
reviews: 211,
|
||||
genre: "Genre 6",
|
||||
frequency: "Weekly",
|
||||
backgroundColor: Color("onracColor"),
|
||||
backgroundIsDark: true
|
||||
),
|
||||
Podcast(
|
||||
id: UUID(),
|
||||
image: UIImage(named: "dgs_logo")!,
|
||||
title: "Podcast Title 7",
|
||||
by: "Author 7",
|
||||
episodes: episodes,
|
||||
rating: 4.2,
|
||||
reviews: 211,
|
||||
genre: "Genre 7",
|
||||
frequency: "Complete",
|
||||
backgroundColor: Color("dgsColor"),
|
||||
backgroundIsDark: false
|
||||
),
|
||||
]
|
||||
}
|
@ -1,3 +1,83 @@
|
||||
# AD_iOS
|
||||
# Apple Podcasts Clone
|
||||
|
||||
iOS Apple Podcasts mini-clone
|
||||
## Project Overview
|
||||
|
||||
Apple Podcasts allows users to listen to podcasts and interact with a
|
||||
sophisticated UI to control their playback.
|
||||
|
||||
This clone does not.
|
||||
|
||||
Apple Podcasts Clone is an application built on the SwiftUI
|
||||
framework. It focuses on the `Library` "master-detail".
|
||||
|
||||
* ### :lipstick: The View
|
||||
As of the 21st of May 2023, it is a <em>façade</em>: only the
|
||||
View part of the project is serviceable.
|
||||
|
||||
<img src="./docs/light_lib_up.png" height="700" style="margin:20px">
|
||||
<img src="./docs/light_lib_down.png" height="700" style="margin:20px">
|
||||
<br>
|
||||
<img src="./docs/light_pod_up.png" height="700" style="margin:20px">
|
||||
<img src="./docs/light_pod_down.png" height="700" style="margin:20px">
|
||||
<br>
|
||||
<img src="./docs/dark_lib_up.png" height="700" style="margin:20px">
|
||||
<img src="./docs/dark_lib_down.png" height="700" style="margin:20px">
|
||||
<br>
|
||||
<img src="./docs/dark_pod_up.png" height="700" style="margin:20px">
|
||||
<img src="./docs/dark_pod_down.png" height="700" style="margin:20px">
|
||||
|
||||
<details><summary> more details... </summary>
|
||||
|
||||
* #### Library ("master") 💐
|
||||
|
||||
Users can browse a `Library` of `Podcasts` and select any podcast to inspect.
|
||||
|
||||
* #### Podcast ("detail") 🌻
|
||||
|
||||
Users can browse a `Podcast` of `Episodes`.
|
||||
|
||||
* #### Dark/Light themes 🌙 ☀️
|
||||
|
||||
This clone replicates the original dark/light themes by Apple Podcasts.
|
||||
|
||||
To test this, you can change your device's (or your emulator's) display
|
||||
setting to dark/light theme.
|
||||
|
||||
* #### Top bar ☝️
|
||||
|
||||
In Apple Podcasts, a stylish, slightly transparent top bar contains certain menu
|
||||
options, and displays the name of the current
|
||||
section once the user has scrolled past the corresponding header.
|
||||
|
||||
This clone has no such thing.
|
||||
|
||||
* #### Bottom bar 👇
|
||||
|
||||
In Apple Podcasts, a consistent, stylish, slightly transparent bottom bar allows
|
||||
navigation between different views, and playing songs.
|
||||
|
||||
This clone tried to emulate that, but only the Library is there to visit, and the
|
||||
"Now Playing" bar is static.
|
||||
|
||||
</detail>
|
||||
|
||||
* ### :necktie: The Model
|
||||
|
||||
A very basic Model is in place, to provide some sample data to views more easily.
|
||||
|
||||
## Installation
|
||||
|
||||
To run the Apple Podcasts Clone, you must use XCode -- your best bet is using a macOS machine.
|
||||
|
||||
Once in XCode, you can look through the `Views` -- they come with their own `Previews`. Find the `MainView`,
|
||||
and have a look around inside the preview. Sometimes, XCode is pretty neat. Sometimes...
|
||||
|
||||
## Some known limitations and shortcomings
|
||||
|
||||
Concerning the View part of this project:
|
||||
* in the "detail" Podcast view, instead of saying "MORE", latest episode descriptions spanning more than 3 lines are truncated with a "..."
|
||||
* throughout the app, the smart date formattings are close to the original app, but not quite as smart.
|
||||
* the top bar was left unimplemented
|
||||
* the "now playing bar" and the tabview don't have the same transparency / colors
|
||||
* instead of i18, this clone has a very basic struct that provides static strings at need. Some string literals are still being used in the views.
|
||||
* and many others will join this list, no doubt.
|
||||
|