├── Fit Vein Tests
├── Fit_Vein_Tests.swift
├── FunctionsTests.swift
├── HealthStatTests.swift
├── IntervalWorkoutTests.swift
└── PostTests.swift
├── Fit Vein UI Tests
├── Fit_Vein_UI_Tests.swift
└── Fit_Vein_UI_TestsLaunchTests.swift
├── Fit Vein.xcodeproj
├── project.pbxproj
├── project.xcworkspace
│ ├── contents.xcworkspacedata
│ └── xcshareddata
│ │ └── IDEWorkspaceChecks.plist
└── xcuserdata
│ └── vader.xcuserdatad
│ └── xcschemes
│ └── xcschememanagement.plist
├── Fit Vein
├── Assets.xcassets
│ ├── AccentColor.colorset
│ │ └── Contents.json
│ ├── AppIcon.appiconset
│ │ ├── 100.png
│ │ ├── 1024.png
│ │ ├── 114.png
│ │ ├── 120.png
│ │ ├── 144.png
│ │ ├── 152.png
│ │ ├── 167.png
│ │ ├── 180.png
│ │ ├── 20.png
│ │ ├── 29.png
│ │ ├── 40.png
│ │ ├── 50.png
│ │ ├── 57.png
│ │ ├── 58.png
│ │ ├── 60.png
│ │ ├── 72.png
│ │ ├── 76.png
│ │ ├── 80.png
│ │ ├── 87.png
│ │ └── Contents.json
│ ├── Contents.json
│ ├── SignInBackgroundImage.imageset
│ │ ├── Contents.json
│ │ └── gym.jpg
│ └── SignUpBackgroundImage.imageset
│ │ ├── Contents.json
│ │ └── gym.jpg
├── Constants&Utils
│ ├── Enums.swift
│ ├── Errors.swift
│ ├── Functions.swift
│ ├── HealthKitRepository.swift
│ ├── NetworkManager.swift
│ └── SwiftUIImagePicker.swift
├── ContentView.swift
├── FIrebase Utils
│ ├── FirebaseStorageManager.swift
│ ├── FirestoreManager.swift
│ ├── RCValues.swift
│ └── SessionStore.swift
├── Fit Vein.entitlements
├── Fit_VeinApp.swift
├── Images
│ ├── FitVeinIconDark.png
│ ├── FitVeinIconLight.png
│ ├── blank-profile-hi.png
│ ├── gym.jpg
│ └── sprint2.png
├── LoggedUserView.swift
├── Lottie
│ ├── 20022-character-squat-animation.json
│ ├── 43323-men-doing-rope-exercise.json
│ ├── 70772-smiley-loader.json
│ ├── avocadoWorkout.json
│ ├── baboonExercising.json
│ ├── countdown.json
│ ├── countdown2.json
│ ├── distance.json
│ ├── downArrows.json
│ ├── dumbleLoading.json
│ ├── flame.json
│ ├── heartRate.json
│ ├── hello.json
│ ├── levelUp.json
│ ├── loading.json
│ ├── noInternetConnection.json
│ ├── runnerAddWorkout.json
│ ├── shoesWalking.json
│ ├── skeleton.json
│ ├── success2.json
│ ├── time.json
│ ├── time2.json
│ └── wrongData.json
├── LottieView.swift
├── Medals
│ ├── medal-2.png
│ ├── medal-3.png
│ ├── medal-4.png
│ ├── medal.png
│ ├── medalFirstComment.png
│ ├── medalFirstLike.png
│ ├── medalFirstOwnWorkout.png
│ ├── medalFirstPost.png
│ ├── medalFirstWorkout.png
│ ├── medalSecondLevel.png
│ └── medalThirdLevel.png
├── Models
│ ├── HealthStat.swift
│ ├── IntervalWorkout.swift
│ ├── Post.swift
│ └── Profile.swift
├── Onboarding Photos
│ ├── AddWorkoutView.png
│ ├── FinishedWorkoutView.png
│ ├── HealthTabView.png
│ ├── HomeTabView.png
│ ├── MedalView.png
│ ├── PostCommentsView.png
│ ├── ProfileView.png
│ ├── SearchFriendsView.png
│ ├── SettingsView.png
│ ├── WorkoutTimerView.png
│ ├── WorkoutsTabView.png
│ ├── WorkoutsTabViewList.png
│ └── WorkoutsView.png
├── OnboardingView.swift
├── PreLaunchView.swift
├── Preview Content
│ └── Preview Assets.xcassets
│ │ └── Contents.json
├── ViewModels
│ ├── HealthKitViewModel.swift
│ ├── HomeViewModel.swift
│ ├── MedalsViewModel.swift
│ ├── ProfileViewModel.swift
│ ├── SignInViewModel.swift
│ ├── SignUpViewModel.swift
│ └── WorkoutViewModel.swift
├── Views
│ ├── AuthenticationViews
│ │ ├── SignInView.swift
│ │ └── SignUpView.swift
│ ├── HelpViews
│ │ ├── CustomTextField.swift
│ │ ├── HomeTabFetchingView.swift
│ │ ├── ProfileTabFetchingView.swift
│ │ └── RingProgressViewStyle.swift
│ ├── HomeTabViews
│ │ ├── CommentsViews
│ │ │ ├── HomeTabCommentsView.swift
│ │ │ ├── HomeTabCommentsViewPostView.swift
│ │ │ └── PostCommentsView.swift
│ │ ├── HomeView.swift
│ │ ├── NotificationsView.swift
│ │ ├── PostsViews
│ │ │ ├── AddPostView.swift
│ │ │ ├── EditPostView.swift
│ │ │ ├── HomeTabSubViewPostDetailsView.swift
│ │ │ ├── HomeTabSubViewPostsView.swift
│ │ │ └── HomeTabSubViewShareView.swift
│ │ └── SearchFriendsView.swift
│ ├── ProfileTabViews
│ │ ├── HealthTabView.swift
│ │ ├── ProfileView.swift
│ │ ├── SettingsView.swift
│ │ └── WorkoutTabView.swift
│ └── WorkoutTabViews
│ │ ├── FinishedWorkoutView.swift
│ │ ├── SingleWorkoutWindowView.swift
│ │ ├── WorkoutTimerView.swift
│ │ └── WorkoutView.swift
├── en.lproj
│ └── Localizable.strings
└── pl.lproj
│ └── Localizable.strings
├── LICENSE
└── README.md
/Fit Vein Tests/Fit_Vein_Tests.swift:
--------------------------------------------------------------------------------
1 | //
2 | // Fit_Vein_Tests.swift
3 | // Fit Vein Tests
4 | //
5 | // Created by Łukasz Janiszewski on 04/01/2022.
6 | //
7 |
8 | import XCTest
9 | import SwiftUI
10 | @testable import Fit_Vein
11 |
12 | class Fit_Vein_Tests: XCTestCase {
13 |
14 | // override func setUpWithError() throws {
15 | // // Put setup code here. This method is called before the invocation of each test method in the class.
16 | // }
17 | //
18 | // override func tearDownWithError() throws {
19 | // // Put teardown code here. This method is called after the invocation of each test method in the class.
20 | // }
21 | //
22 | // func testExample() throws {
23 | // // This is an example of a functional test case.
24 | // // Use XCTAssert and related functions to verify your tests produce the correct results.
25 | // // Any test you write for XCTest can be annotated as throws and async.
26 | // // Mark your test throws to produce an unexpected failure when your test encounters an uncaught error.
27 | // // Mark your test async to allow awaiting for asynchronous code to complete. Check the results with assertions afterwards.
28 | // }
29 | //
30 | // func testPerformanceExample() throws {
31 | // // This is an example of a performance test case.
32 | // measure {
33 | // // Put the code you want to measure the time of here.
34 | // }
35 | // }
36 | }
37 |
--------------------------------------------------------------------------------
/Fit Vein Tests/FunctionsTests.swift:
--------------------------------------------------------------------------------
1 | //
2 | // FunctionsTests.swift
3 | // Fit Vein Tests
4 | //
5 | // Created by Łukasz Janiszewski on 04/01/2022.
6 | //
7 |
8 | import XCTest
9 | @testable import Fit_Vein
10 |
11 | class FunctionsTests: XCTestCase {
12 |
13 | let dateFormatter = DateFormatter()
14 |
15 | func testYearsBetweenDate() {
16 | dateFormatter.dateFormat = "dd/MM/yy"
17 | let startDate = dateFormatter.date(from: "01/02/2016")!
18 | let endDate = dateFormatter.date(from: "02/02/2016")!
19 |
20 | let result = yearsBetweenDate(startDate: startDate, endDate: endDate)
21 | XCTAssertEqual(result, 0)
22 | }
23 |
24 | func testGetWorkoutsDivider() {
25 | var workoutsCount = 11
26 | let result = getWorkoutsDivider(workoutsCount: workoutsCount)
27 | XCTAssertEqual(result, 1)
28 |
29 | workoutsCount = 1
30 | XCTAssertEqual(result, 1)
31 | }
32 |
33 | func testGetShortDate() {
34 | let stringDate = "2022-01-04 18:04:41 +0000"
35 | dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss ZZZZ"
36 | let date = dateFormatter.date(from: stringDate)!
37 | let result = getShortDate(longDate: date)
38 | XCTAssertEqual(result, "4 stycznia 2022 o 7:04 PM")
39 | }
40 |
41 | }
42 |
--------------------------------------------------------------------------------
/Fit Vein Tests/HealthStatTests.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HealthStatTests.swift
3 | // Fit Vein Tests
4 | //
5 | // Created by Łukasz Janiszewski on 04/01/2022.
6 | //
7 |
8 | import XCTest
9 | @testable import Fit_Vein
10 |
11 | class HealthStatTests: XCTestCase {
12 |
13 | var healthStat: HealthStat!
14 | let dateFormatter = DateFormatter()
15 | var date: Date = Date()
16 |
17 | override func setUp() {
18 | super.setUp()
19 | dateFormatter.dateFormat = "dd/MM/yy"
20 | date = dateFormatter.date(from: "01/02/2016")!
21 | healthStat = HealthStat(stat: nil, date: date)
22 | }
23 |
24 | override func tearDown() {
25 | super.tearDown()
26 | healthStat = nil
27 | }
28 |
29 | func testHealthStatIDInitialization() {
30 | XCTAssertNil(healthStat.stat)
31 | XCTAssertEqual(healthStat.date, date)
32 | }
33 | }
34 |
--------------------------------------------------------------------------------
/Fit Vein Tests/IntervalWorkoutTests.swift:
--------------------------------------------------------------------------------
1 | //
2 | // IntervalWorkoutTests.swift
3 | // Fit Vein Tests
4 | //
5 | // Created by Łukasz Janiszewski on 04/01/2022.
6 | //
7 |
8 | import XCTest
9 | @testable import Fit_Vein
10 |
11 | class IntervalWorkoutTests: XCTestCase {
12 |
13 | var intervalWorkout: IntervalWorkout!
14 | let dateFormatter = DateFormatter()
15 | var date: Date = Date()
16 |
17 | override func setUp() {
18 | super.setUp()
19 | dateFormatter.dateFormat = "dd/MM/yy"
20 | date = dateFormatter.date(from: "01/02/2016")!
21 | intervalWorkout = IntervalWorkout(forPreviews: true, id: "id", usersID: "usersID", type: "type", date: date, isFinished: true, calories: 32, series: 5, workTime: 30, restTime: 15, completedDuration: 10, completedSeries: 0)
22 | }
23 |
24 | override func tearDown() {
25 | super.tearDown()
26 | intervalWorkout = nil
27 | }
28 |
29 | func testPostSecondInit() {
30 | XCTAssertEqual(intervalWorkout.id, "id")
31 | XCTAssertEqual(intervalWorkout.usersID, "usersID")
32 | XCTAssertEqual(intervalWorkout.type, "type")
33 | XCTAssertEqual(intervalWorkout.date, date)
34 | XCTAssertEqual(intervalWorkout.isFinished, true)
35 | XCTAssertEqual(intervalWorkout.calories, 32)
36 | XCTAssertEqual(intervalWorkout.series, 5)
37 | XCTAssertEqual(intervalWorkout.workTime, 30)
38 | XCTAssertEqual(intervalWorkout.restTime, 15)
39 | XCTAssertEqual(intervalWorkout.completedDuration, 10)
40 | XCTAssertEqual(intervalWorkout.completedSeries, 0)
41 | }
42 |
43 | func testSetDataOnEnd() {
44 | intervalWorkout.setDataOnEnd(calories: 200, completedDuration: 40, completedSeries: 2)
45 | XCTAssertEqual(intervalWorkout.isFinished, true)
46 | XCTAssertEqual(intervalWorkout.calories, 200)
47 | XCTAssertEqual(intervalWorkout.completedDuration, 40)
48 | XCTAssertEqual(intervalWorkout.completedSeries, 2)
49 | }
50 |
51 | func testSetUsersID() {
52 | intervalWorkout.setUsersID(usersID: "usersID2")
53 | XCTAssertEqual(intervalWorkout.usersID, "usersID2")
54 | }
55 |
56 | }
57 |
--------------------------------------------------------------------------------
/Fit Vein Tests/PostTests.swift:
--------------------------------------------------------------------------------
1 | //
2 | // PostTests.swift
3 | // Fit Vein Tests
4 | //
5 | // Created by Łukasz Janiszewski on 04/01/2022.
6 | //
7 |
8 | import XCTest
9 | @testable import Fit_Vein
10 |
11 | class PostTests: XCTestCase {
12 | var post: Post!
13 | var post2: Post!
14 |
15 | var comment: Comment!
16 | var comment2: Comment!
17 |
18 | override func setUp() {
19 | super.setUp()
20 | post = Post(authorID: "authorID", authorFirstName: "authorFirstName", authorUsername: "authorUsername", authorProfilePictureURL: "authorProfilePictureURL", text: "text")
21 | post2 = Post(id: "id", authorID: "authorID", authorFirstName: "authorFirstName", authorUsername: "authorUsername", authorProfilePictureURL: "authorProfilePictureURL", addDate: Date(), text: "text", reactionsUsersIDs: nil, commentedUsersIDs: nil, comments: nil)
22 |
23 | comment = Comment(authorID: "authorID", postID: "postID", authorFirstName: "authorFirstName", authorUsername: "authorUsername", authorProfilePictureURL: "authorProfilePictureURL", text: "text")
24 | comment2 = Comment(id: "id", authorID: "authorID", postID: "postID", authorFirstName: "authorFirstName", authorUsername: "authorUsername", authorProfilePictureURL: "authorProfilePictureURL", addDate: Date(), text: "text", reactionsUsersIDs: nil)
25 | }
26 |
27 | override func tearDown() {
28 | super.tearDown()
29 | post = nil
30 | post2 = nil
31 |
32 | comment = nil
33 | comment2 = nil
34 | }
35 |
36 | func testPostFirstInit() {
37 | XCTAssertEqual(post.authorID, "authorID")
38 | XCTAssertEqual(post.authorFirstName, "authorFirstName")
39 | XCTAssertEqual(post.authorUsername, "authorUsername")
40 | XCTAssertEqual(post.authorProfilePictureURL, "authorProfilePictureURL")
41 | XCTAssertEqual(post.text, "text")
42 | }
43 |
44 | func testPostSecondInit() {
45 | XCTAssertEqual(post2.id, "id")
46 | XCTAssertEqual(post2.authorID, "authorID")
47 | XCTAssertEqual(post2.authorFirstName, "authorFirstName")
48 | XCTAssertEqual(post2.authorUsername, "authorUsername")
49 | XCTAssertEqual(post2.authorProfilePictureURL, "authorProfilePictureURL")
50 | XCTAssertEqual(post2.text, "text")
51 | XCTAssertNil(post2.reactionsUsersIDs)
52 | XCTAssertNil(post2.commentedUsersIDs)
53 | XCTAssertNil(post2.comments)
54 | }
55 |
56 | func testCommentFirstInit() {
57 | XCTAssertEqual(comment.authorID, "authorID")
58 | XCTAssertEqual(comment.postID, "postID")
59 | XCTAssertEqual(comment.authorFirstName, "authorFirstName")
60 | XCTAssertEqual(comment.authorUsername, "authorUsername")
61 | XCTAssertEqual(comment.authorProfilePictureURL, "authorProfilePictureURL")
62 | XCTAssertEqual(comment.text, "text")
63 | }
64 |
65 | func testCommentSecondInit() {
66 | XCTAssertEqual(comment2.id, "id")
67 | XCTAssertEqual(comment2.authorID, "authorID")
68 | XCTAssertEqual(comment.postID, "postID")
69 | XCTAssertEqual(comment2.authorFirstName, "authorFirstName")
70 | XCTAssertEqual(comment2.authorUsername, "authorUsername")
71 | XCTAssertEqual(comment2.authorProfilePictureURL, "authorProfilePictureURL")
72 | XCTAssertEqual(comment2.text, "text")
73 | XCTAssertNil(comment2.reactionsUsersIDs)
74 | }
75 | }
76 |
--------------------------------------------------------------------------------
/Fit Vein UI Tests/Fit_Vein_UI_TestsLaunchTests.swift:
--------------------------------------------------------------------------------
1 | //
2 | // Fit_Vein_UI_TestsLaunchTests.swift
3 | // Fit Vein UI Tests
4 | //
5 | // Created by Łukasz Janiszewski on 04/01/2022.
6 | //
7 |
8 | import XCTest
9 |
10 | class Fit_Vein_UI_TestsLaunchTests: XCTestCase {
11 |
12 | override class var runsForEachTargetApplicationUIConfiguration: Bool {
13 | true
14 | }
15 |
16 | override func setUpWithError() throws {
17 | continueAfterFailure = false
18 | }
19 |
20 | func testLaunch() throws {
21 | let app = XCUIApplication()
22 | app.launch()
23 |
24 | // Insert steps here to perform after app launch but before taking a screenshot,
25 | // such as logging into a test account or navigating somewhere in the app
26 |
27 | let attachment = XCTAttachment(screenshot: app.screenshot())
28 | attachment.name = "Launch Screen"
29 | attachment.lifetime = .keepAlways
30 | add(attachment)
31 | }
32 | }
33 |
--------------------------------------------------------------------------------
/Fit Vein.xcodeproj/project.xcworkspace/contents.xcworkspacedata:
--------------------------------------------------------------------------------
1 |
2 |
4 |
6 |
7 |
8 |
--------------------------------------------------------------------------------
/Fit Vein.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | IDEDidComputeMac32BitWarning
6 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/Fit Vein.xcodeproj/xcuserdata/vader.xcuserdatad/xcschemes/xcschememanagement.plist:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | SchemeUserState
6 |
7 | Fit Vein.xcscheme_^#shared#^_
8 |
9 | orderHint
10 | 25
11 |
12 |
13 | SuppressBuildableAutocreation
14 |
15 | 0AC4DD802715A9E4003D5FED
16 |
17 | primary
18 |
19 |
20 |
21 |
22 |
23 |
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AccentColor.colorset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "colors" : [
3 | {
4 | "color" : {
5 | "platform" : "ios",
6 | "reference" : "systemGreenColor"
7 | },
8 | "idiom" : "universal"
9 | },
10 | {
11 | "appearances" : [
12 | {
13 | "appearance" : "luminosity",
14 | "value" : "light"
15 | }
16 | ],
17 | "color" : {
18 | "platform" : "ios",
19 | "reference" : "systemGreenColor"
20 | },
21 | "idiom" : "universal"
22 | },
23 | {
24 | "appearances" : [
25 | {
26 | "appearance" : "luminosity",
27 | "value" : "dark"
28 | }
29 | ],
30 | "color" : {
31 | "platform" : "ios",
32 | "reference" : "systemGreenColor"
33 | },
34 | "idiom" : "universal"
35 | },
36 | {
37 | "color" : {
38 | "platform" : "ios",
39 | "reference" : "systemGreenColor"
40 | },
41 | "idiom" : "iphone"
42 | },
43 | {
44 | "appearances" : [
45 | {
46 | "appearance" : "luminosity",
47 | "value" : "light"
48 | }
49 | ],
50 | "color" : {
51 | "platform" : "ios",
52 | "reference" : "systemGreenColor"
53 | },
54 | "idiom" : "iphone"
55 | },
56 | {
57 | "appearances" : [
58 | {
59 | "appearance" : "luminosity",
60 | "value" : "dark"
61 | }
62 | ],
63 | "color" : {
64 | "platform" : "ios",
65 | "reference" : "systemGreenColor"
66 | },
67 | "idiom" : "iphone"
68 | }
69 | ],
70 | "info" : {
71 | "author" : "xcode",
72 | "version" : 1
73 | }
74 | }
75 |
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/100.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/100.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/1024.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/1024.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/114.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/114.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/120.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/120.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/144.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/144.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/152.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/152.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/167.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/167.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/180.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/180.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/20.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/20.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/29.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/29.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/40.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/40.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/50.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/50.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/57.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/57.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/58.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/58.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/60.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/60.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/72.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/72.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/76.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/76.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/80.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/80.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/87.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/AppIcon.appiconset/87.png
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/AppIcon.appiconset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "images" : [
3 | {
4 | "filename" : "40.png",
5 | "idiom" : "iphone",
6 | "scale" : "2x",
7 | "size" : "20x20"
8 | },
9 | {
10 | "filename" : "60.png",
11 | "idiom" : "iphone",
12 | "scale" : "3x",
13 | "size" : "20x20"
14 | },
15 | {
16 | "filename" : "29.png",
17 | "idiom" : "iphone",
18 | "scale" : "1x",
19 | "size" : "29x29"
20 | },
21 | {
22 | "filename" : "58.png",
23 | "idiom" : "iphone",
24 | "scale" : "2x",
25 | "size" : "29x29"
26 | },
27 | {
28 | "filename" : "87.png",
29 | "idiom" : "iphone",
30 | "scale" : "3x",
31 | "size" : "29x29"
32 | },
33 | {
34 | "filename" : "80.png",
35 | "idiom" : "iphone",
36 | "scale" : "2x",
37 | "size" : "40x40"
38 | },
39 | {
40 | "filename" : "120.png",
41 | "idiom" : "iphone",
42 | "scale" : "3x",
43 | "size" : "40x40"
44 | },
45 | {
46 | "filename" : "57.png",
47 | "idiom" : "iphone",
48 | "scale" : "1x",
49 | "size" : "57x57"
50 | },
51 | {
52 | "filename" : "114.png",
53 | "idiom" : "iphone",
54 | "scale" : "2x",
55 | "size" : "57x57"
56 | },
57 | {
58 | "filename" : "120.png",
59 | "idiom" : "iphone",
60 | "scale" : "2x",
61 | "size" : "60x60"
62 | },
63 | {
64 | "filename" : "180.png",
65 | "idiom" : "iphone",
66 | "scale" : "3x",
67 | "size" : "60x60"
68 | },
69 | {
70 | "filename" : "20.png",
71 | "idiom" : "ipad",
72 | "scale" : "1x",
73 | "size" : "20x20"
74 | },
75 | {
76 | "filename" : "40.png",
77 | "idiom" : "ipad",
78 | "scale" : "2x",
79 | "size" : "20x20"
80 | },
81 | {
82 | "filename" : "29.png",
83 | "idiom" : "ipad",
84 | "scale" : "1x",
85 | "size" : "29x29"
86 | },
87 | {
88 | "filename" : "58.png",
89 | "idiom" : "ipad",
90 | "scale" : "2x",
91 | "size" : "29x29"
92 | },
93 | {
94 | "filename" : "40.png",
95 | "idiom" : "ipad",
96 | "scale" : "1x",
97 | "size" : "40x40"
98 | },
99 | {
100 | "filename" : "80.png",
101 | "idiom" : "ipad",
102 | "scale" : "2x",
103 | "size" : "40x40"
104 | },
105 | {
106 | "filename" : "50.png",
107 | "idiom" : "ipad",
108 | "scale" : "1x",
109 | "size" : "50x50"
110 | },
111 | {
112 | "filename" : "100.png",
113 | "idiom" : "ipad",
114 | "scale" : "2x",
115 | "size" : "50x50"
116 | },
117 | {
118 | "filename" : "72.png",
119 | "idiom" : "ipad",
120 | "scale" : "1x",
121 | "size" : "72x72"
122 | },
123 | {
124 | "filename" : "144.png",
125 | "idiom" : "ipad",
126 | "scale" : "2x",
127 | "size" : "72x72"
128 | },
129 | {
130 | "filename" : "76.png",
131 | "idiom" : "ipad",
132 | "scale" : "1x",
133 | "size" : "76x76"
134 | },
135 | {
136 | "filename" : "152.png",
137 | "idiom" : "ipad",
138 | "scale" : "2x",
139 | "size" : "76x76"
140 | },
141 | {
142 | "filename" : "167.png",
143 | "idiom" : "ipad",
144 | "scale" : "2x",
145 | "size" : "83.5x83.5"
146 | },
147 | {
148 | "filename" : "1024.png",
149 | "idiom" : "ios-marketing",
150 | "scale" : "1x",
151 | "size" : "1024x1024"
152 | }
153 | ],
154 | "info" : {
155 | "author" : "xcode",
156 | "version" : 1
157 | }
158 | }
159 |
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "info" : {
3 | "author" : "xcode",
4 | "version" : 1
5 | }
6 | }
7 |
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/SignInBackgroundImage.imageset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "images" : [
3 | {
4 | "filename" : "gym.jpg",
5 | "idiom" : "universal",
6 | "scale" : "1x"
7 | },
8 | {
9 | "idiom" : "universal",
10 | "scale" : "2x"
11 | },
12 | {
13 | "idiom" : "universal",
14 | "scale" : "3x"
15 | }
16 | ],
17 | "info" : {
18 | "author" : "xcode",
19 | "version" : 1
20 | }
21 | }
22 |
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/SignInBackgroundImage.imageset/gym.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/SignInBackgroundImage.imageset/gym.jpg
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/SignUpBackgroundImage.imageset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "images" : [
3 | {
4 | "filename" : "gym.jpg",
5 | "idiom" : "universal",
6 | "scale" : "1x"
7 | },
8 | {
9 | "idiom" : "universal",
10 | "scale" : "2x"
11 | },
12 | {
13 | "idiom" : "universal",
14 | "scale" : "3x"
15 | }
16 | ],
17 | "info" : {
18 | "author" : "xcode",
19 | "version" : 1
20 | }
21 | }
22 |
--------------------------------------------------------------------------------
/Fit Vein/Assets.xcassets/SignUpBackgroundImage.imageset/gym.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Assets.xcassets/SignUpBackgroundImage.imageset/gym.jpg
--------------------------------------------------------------------------------
/Fit Vein/Constants&Utils/Enums.swift:
--------------------------------------------------------------------------------
1 | //
2 | // Enums.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 12/10/2021.
6 | //
7 |
8 | import Foundation
9 |
10 | public enum Country: String, CaseIterable, Identifiable {
11 | case poland
12 |
13 | public var id: String { self.rawValue }
14 | }
15 |
16 | public enum Language: String, CaseIterable, Identifiable {
17 | case english
18 | case polish
19 |
20 | public var id: String { self.rawValue }
21 | }
22 |
--------------------------------------------------------------------------------
/Fit Vein/Constants&Utils/Errors.swift:
--------------------------------------------------------------------------------
1 | //
2 | // Errors.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 13/10/2021.
6 | //
7 |
8 | import Foundation
9 |
10 | enum AuthError: String, Error {
11 | case authFailed = "Failed to authenticate the user"
12 | case signOutFailed = "Failed to sign out"
13 | }
14 |
15 | enum DatabaseError: String, Error {
16 | case createUserDataFailed = "Failed to create user data in database"
17 | case fetchUserDataFailed = "Failed to fetch data from database"
18 | case deleteUserDataFailed = "Failed to delete user data from database"
19 | case editUserDataFailed = "Failed to edit user data in database"
20 | }
21 |
--------------------------------------------------------------------------------
/Fit Vein/Constants&Utils/Functions.swift:
--------------------------------------------------------------------------------
1 | //
2 | // Functions.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 13/10/2021.
6 | //
7 |
8 | import Foundation
9 | import SwiftUI
10 |
11 | public func yearsBetweenDate(startDate: Date, endDate: Date) -> Int {
12 | let calendar = Calendar.current
13 | let components = calendar.dateComponents([.year], from: startDate, to: endDate)
14 | return components.year!
15 | }
16 |
17 | public func getWorkoutsDivider(workoutsCount: Int) -> Int {
18 | var workoutsCountNumber = workoutsCount
19 | while workoutsCountNumber > 10 {
20 | workoutsCountNumber = workoutsCountNumber / 10
21 | }
22 | return workoutsCountNumber % 10
23 | }
24 |
25 | public func getShortDate(longDate: Date) -> String {
26 | let dateFormatter = DateFormatter()
27 | dateFormatter.dateStyle = .long
28 | dateFormatter.timeStyle = .short
29 |
30 | return dateFormatter.string(from: longDate)
31 | }
32 |
33 | public func getTextTimeFromDuration(duration: Int) -> Text {
34 | var secondsRemaining = 0
35 | var minutesRemaining = 0
36 |
37 | if duration >= 60 {
38 | minutesRemaining = Int(duration / 60)
39 | secondsRemaining = duration - (60 * minutesRemaining)
40 | } else {
41 | minutesRemaining = 0
42 | secondsRemaining = duration
43 | }
44 |
45 | if minutesRemaining < 10 {
46 | if secondsRemaining < 10 {
47 | return Text("0\(minutesRemaining):0\(secondsRemaining)")
48 | } else {
49 | return Text("0\(minutesRemaining):\(secondsRemaining)")
50 | }
51 | } else {
52 | if secondsRemaining < 10 {
53 | return Text("\(minutesRemaining):0\(secondsRemaining)")
54 | } else {
55 | return Text("\(minutesRemaining):\(secondsRemaining)")
56 | }
57 | }
58 | }
59 |
60 | public func firstDayOfWeek() -> Date {
61 | return Calendar(identifier: .iso8601).date(from: Calendar(identifier: .iso8601).dateComponents([.yearForWeekOfYear, .weekOfYear], from: Date())) ?? Date()
62 | }
63 |
--------------------------------------------------------------------------------
/Fit Vein/Constants&Utils/HealthKitRepository.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HealthKitRepository.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 24/11/2021.
6 | //
7 |
8 | import Foundation
9 | import HealthKit
10 |
11 | final class HealthKitRepository {
12 | var healthStore: HKHealthStore?
13 |
14 | let allTypes = Set([
15 | HKObjectType.quantityType(forIdentifier: .stepCount)!,
16 | HKObjectType.quantityType(forIdentifier: .activeEnergyBurned)!,
17 | HKObjectType.quantityType(forIdentifier: .distanceWalkingRunning)!,
18 | HKObjectType.quantityType(forIdentifier: .appleExerciseTime)!,
19 | HKObjectType.quantityType(forIdentifier: .heartRate)!
20 | ])
21 |
22 | var query: HKStatisticsCollectionQuery?
23 |
24 | init() {
25 | if HKHealthStore.isHealthDataAvailable() {
26 | healthStore = HKHealthStore()
27 | }
28 | }
29 |
30 | func requestAuthorization(completion: @escaping ((Bool) -> Void)) {
31 | guard let store = healthStore else {
32 | return
33 | }
34 |
35 | store.requestAuthorization(toShare: [], read: allTypes) { (success, error) in
36 | if let error = error {
37 | print("HealthKit authorization error: \(error)")
38 | } else {
39 | print("Health authorization success")
40 | completion(success)
41 | }
42 | }
43 | }
44 |
45 | func requestHealthStats(by category: String, completion: @escaping (([HealthStat]) -> Void)) {
46 | guard let store = healthStore, let type = HKObjectType.quantityType(forIdentifier: typeByCategory(category: category)) else {
47 | return
48 | }
49 |
50 | let startDate = Calendar.current.date(byAdding: .day, value: -7, to: Date()) ?? Date()
51 | let endDate = Date()
52 | let anchorDate = firstDayOfWeek()
53 | let dailyComponent = DateComponents(day: 1)
54 |
55 | var healthStats = [HealthStat]()
56 |
57 | let predicate = HKQuery.predicateForSamples(withStart: startDate, end: endDate, options: .strictStartDate)
58 |
59 | query = HKStatisticsCollectionQuery(quantityType: type, quantitySamplePredicate: predicate, options: type == HKObjectType.quantityType(forIdentifier: typeByCategory(category: "heartRate")) ? .discreteAverage : .cumulativeSum, anchorDate: anchorDate, intervalComponents: dailyComponent)
60 |
61 | query?.initialResultsHandler = { (query, statistics, error) in
62 | if let error = error {
63 | print("Error in requestHealthStats functions in initialResultsHandler: \(error)")
64 | } else {
65 | statistics?.enumerateStatistics(from: startDate, to: endDate, with: { (stats, _) in
66 | if stats.quantityType == HKObjectType.quantityType(forIdentifier: .heartRate) {
67 | let stat = HealthStat(stat: stats.averageQuantity(), date: stats.startDate)
68 | healthStats.append(stat)
69 | } else {
70 | let stat = HealthStat(stat: stats.sumQuantity(), date: stats.startDate)
71 | healthStats.append(stat)
72 | }
73 | })
74 |
75 | completion(healthStats)
76 | }
77 | }
78 |
79 | guard let query = query else {
80 | return
81 | }
82 |
83 | store.execute(query)
84 | }
85 |
86 | private func typeByCategory(category: String) -> HKQuantityTypeIdentifier {
87 | switch category {
88 | case "stepCount":
89 | return .stepCount
90 | case "activeEnergyBurned":
91 | return .activeEnergyBurned
92 | case "distanceWalkingRunning":
93 | return .distanceWalkingRunning
94 | case "appleExerciseTime":
95 | return .appleExerciseTime
96 | case "heartRate":
97 | return .heartRate
98 | default:
99 | return .stepCount
100 | }
101 | }
102 | }
103 |
--------------------------------------------------------------------------------
/Fit Vein/Constants&Utils/NetworkManager.swift:
--------------------------------------------------------------------------------
1 | //
2 | // NetworkManager.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 09/01/2022.
6 | //
7 |
8 | import Foundation
9 | import Network
10 |
11 | class NetworkManager: ObservableObject {
12 | let monitor = NWPathMonitor()
13 | let queue = DispatchQueue(label: "NetworkManager")
14 | @Published var isConnected = true
15 |
16 | init() {
17 | monitor.pathUpdateHandler = { path in
18 | DispatchQueue.main.async {
19 | self.isConnected = path.status == .satisfied
20 | }
21 | }
22 |
23 | monitor.start(queue: queue)
24 | }
25 | }
26 |
--------------------------------------------------------------------------------
/Fit Vein/Constants&Utils/SwiftUIImagePicker.swift:
--------------------------------------------------------------------------------
1 | //
2 | // SwiftUIImagePicker.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 20/10/2021.
6 | //
7 |
8 | import UIKit
9 | import SwiftUI
10 |
11 | struct ImagePicker: UIViewControllerRepresentable {
12 |
13 | var sourceType: UIImagePickerController.SourceType = .photoLibrary
14 |
15 | @Binding var selectedImage: UIImage
16 | @Environment(\.presentationMode) private var presentationMode
17 |
18 | func makeUIViewController(context: UIViewControllerRepresentableContext) -> UIImagePickerController {
19 |
20 | let imagePicker = UIImagePickerController()
21 | imagePicker.allowsEditing = true
22 | imagePicker.sourceType = sourceType
23 | imagePicker.delegate = context.coordinator
24 |
25 | return imagePicker
26 | }
27 |
28 | func updateUIViewController(_ uiViewController: UIImagePickerController, context: UIViewControllerRepresentableContext) {
29 |
30 | }
31 |
32 | func makeCoordinator() -> Coordinator {
33 | Coordinator(self)
34 | }
35 |
36 |
37 | final class Coordinator: NSObject, UIImagePickerControllerDelegate, UINavigationControllerDelegate {
38 |
39 | var parent: ImagePicker
40 |
41 | init(_ parent: ImagePicker) {
42 | self.parent = parent
43 | }
44 |
45 | func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey : Any]) {
46 | if let image = info[.editedImage] as? UIImage {
47 | parent.selectedImage = image
48 | } else if let image = info[UIImagePickerController.InfoKey.originalImage] as? UIImage {
49 | parent.selectedImage = image
50 | }
51 |
52 | parent.presentationMode.wrappedValue.dismiss()
53 | }
54 | }
55 | }
56 |
--------------------------------------------------------------------------------
/Fit Vein/ContentView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // ContentView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 12/10/2021.
6 | //
7 |
8 | import SwiftUI
9 | import LocalAuthentication
10 |
11 | struct ContentView: View {
12 | @EnvironmentObject private var sessionStore: SessionStore
13 | @Environment(\.colorScheme) var colorScheme
14 | @AppStorage("locked") var biometricLock: Bool = false
15 | @AppStorage("shouldShowOnboarding") var shouldShowOnboarding: Bool = true
16 | @State private var unlocked = false
17 |
18 | var body: some View {
19 | GeometryReader { geometry in
20 | let screenWidth = geometry.size.width
21 | let screenHeight = geometry.size.height
22 |
23 | NavigationView {
24 | if self.unlocked || !self.biometricLock {
25 | if sessionStore.session != nil {
26 | LoggedUserView()
27 | .ignoresSafeArea(.keyboard)
28 | .navigationBarHidden(true)
29 | } else {
30 | SignInView()
31 | .environmentObject(sessionStore)
32 | .ignoresSafeArea(.keyboard)
33 | }
34 | } else {
35 | VStack {
36 | Text(String(localized: "ContentView_welcome_label"))
37 | .font(.title)
38 | .padding(.bottom, screenHeight * 0.02)
39 |
40 | HStack(spacing: screenWidth * 0.0001) {
41 | Image(uiImage: UIImage(named: colorScheme == .dark ? "FitVeinIconDark" : "FitVeinIconLight")!)
42 | .resizable()
43 | .scaledToFill()
44 | .frame(width: screenWidth * 0.2, height: screenHeight * 0.2)
45 | .padding(.horizontal, screenWidth * 0.05)
46 |
47 | Text("Fit")
48 | .fontWeight(.bold)
49 | .foregroundColor(.accentColor)
50 | Text("Vein")
51 | .fontWeight(.bold)
52 | }
53 | .font(.system(size: screenHeight * 0.1))
54 |
55 | Spacer()
56 |
57 | Button(action: {
58 | authenticate()
59 | }, label: {
60 | Image(systemName: "faceid")
61 | })
62 | .font(.system(size: screenHeight * 0.08))
63 | .padding(.bottom, screenHeight * 0.02)
64 |
65 | Text(String(localized: "ContentView_unlock_the_app_label"))
66 | .foregroundColor(Color(uiColor: UIColor.lightGray))
67 |
68 | Spacer()
69 | }
70 | .padding()
71 |
72 | }
73 | }
74 | .navigationViewStyle(.stack)
75 | .onAppear {
76 | // In case the app remains in logged-in state
77 | // sessionStore.signOut()
78 | if self.biometricLock && !shouldShowOnboarding {
79 | authenticate()
80 | }
81 | sessionStore.listen()
82 | }
83 | }
84 | }
85 |
86 | func authenticate() {
87 | let context = LAContext()
88 | var error: NSError?
89 |
90 | // check whether biometric authentication is possible
91 | if context.canEvaluatePolicy(.deviceOwnerAuthenticationWithBiometrics, error: &error) {
92 | // it's possible, so go ahead and use it
93 | let reason = "Used to take new pictures of the user."
94 |
95 | context.evaluatePolicy(.deviceOwnerAuthenticationWithBiometrics, localizedReason: reason) { success, authenticationError in
96 | // authentication has now completed
97 | DispatchQueue.main.async {
98 | if success {
99 | // authenticated successfully
100 | self.unlocked = true
101 | } else {
102 | // there was a problem
103 | }
104 | }
105 | }
106 | } else {
107 | // no biometrics
108 | }
109 | }
110 | }
111 |
112 | struct ContentView_Previews: PreviewProvider {
113 | static var previews: some View {
114 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
115 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
116 | ContentView()
117 | .preferredColorScheme(colorScheme)
118 | .previewDevice(PreviewDevice(rawValue: deviceName))
119 | .previewDisplayName(deviceName)
120 | .environmentObject(SessionStore(forPreviews: true))
121 | }
122 | }
123 | }
124 | }
125 |
--------------------------------------------------------------------------------
/Fit Vein/FIrebase Utils/FirebaseStorageManager.swift:
--------------------------------------------------------------------------------
1 | //
2 | // FirebaseStorageManager.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 13/10/2021.
6 | //
7 |
8 | import Foundation
9 | import Firebase
10 |
11 | @MainActor
12 | class FirebaseStorageManager: ObservableObject {
13 | private let storageRef = Storage.storage().reference()
14 |
15 | func uploadImageToStorage(image: UIImage, userID: String, completion: @escaping ((String?, Bool) -> ())) {
16 | let imageUUID = UUID().uuidString
17 | let userImagesStorageRef = storageRef.child("images/\(userID)/\(imageUUID)")
18 |
19 | let data = image.jpegData(compressionQuality: 1)
20 |
21 | let metadata = StorageMetadata()
22 | metadata.contentType = "image/jpeg"
23 | metadata.customMetadata = ["username": userID]
24 |
25 | if let data = data {
26 | userImagesStorageRef.putData(data, metadata: metadata) { _, error in
27 | if let error = error {
28 | print("Error uploading photo: \(error.localizedDescription)")
29 | completion(nil, false)
30 | } else {
31 | completion(imageUUID, true)
32 | }
33 | }
34 | } else {
35 | completion(nil, false)
36 | }
37 | }
38 |
39 | func uploadPostImageToStorage(image: UIImage, userID: String, postID: String, completion: @escaping ((String?), Bool) -> ()) {
40 | let imageUUID = UUID().uuidString
41 | let userImagesStorageRef = storageRef.child("images/\(userID)/\(postID)/\(imageUUID)")
42 |
43 | let data = image.jpegData(compressionQuality: 1)
44 |
45 | let metadata = StorageMetadata()
46 | metadata.contentType = "image/jpeg"
47 | metadata.customMetadata = ["username": userID, "post": postID]
48 |
49 | if let data = data {
50 | userImagesStorageRef.putData(data, metadata: metadata) { _, error in
51 | if let error = error {
52 | print("Error uploading photo: \(error.localizedDescription)")
53 | completion(nil, false)
54 | } else {
55 | completion(imageUUID, true)
56 | }
57 | }
58 | } else {
59 | completion(nil, false)
60 | }
61 | }
62 |
63 | func deleteImageFromStorage(userPhotoURL: String, userID: String, completion: @escaping ((Bool) -> ())) {
64 | let userImagesStorageRef = storageRef.child("images/\(userID)/\(userPhotoURL)")
65 |
66 | userImagesStorageRef.delete() { (error) in
67 | if let error = error {
68 | print("Error deleting image from storage: \(error.localizedDescription)")
69 | completion(false)
70 | } else {
71 | print("Successfully deleted image from storage")
72 | completion(true)
73 | }
74 | }
75 | }
76 |
77 | func deletePostImageFromStorage(photoURL: String, userID: String, postID: String, completion: @escaping ((Bool) -> ())) {
78 | let userImagesStorageRef = storageRef.child("images/\(userID)/\(postID)/\(photoURL)")
79 |
80 | userImagesStorageRef.delete() { (error) in
81 | if let error = error {
82 | print("Error deleting image from storage: \(error.localizedDescription)")
83 | completion(false)
84 | } else {
85 | print("Successfully deleted image from storage")
86 | completion(true)
87 | }
88 | }
89 | }
90 |
91 | func getDownloadURLForImage(stringURL: String, userID: String, completion: @escaping ((URL?, Bool) -> ())) {
92 | let path = "images/\(userID)/\(stringURL)"
93 | let userImagesStorageRef = storageRef.child(path)
94 | userImagesStorageRef.downloadURL() { url, error in
95 | if let error = error {
96 | print("Error getting download URL: \(error.localizedDescription)")
97 | completion(nil, false)
98 | } else {
99 | completion(url, true)
100 | }
101 | }
102 | }
103 |
104 | func getDownloadURLForPostImage(stringURL: String, userID: String, postID: String, completion: @escaping ((URL?, Bool) -> ())) {
105 | let path = "images/\(userID)/\(postID)/\(stringURL)"
106 | let userImagesStorageRef = storageRef.child(path)
107 | userImagesStorageRef.downloadURL() { url, error in
108 | if let error = error {
109 | print("Error getting download URL: \(error.localizedDescription)")
110 | completion(nil, false)
111 | } else {
112 | completion(url, true)
113 | }
114 | }
115 | }
116 |
117 | func downloadImageFromStorage(userID: String, userPhotoURL: String, completion: @escaping ((UIImage?, Bool) -> ())) {
118 | let userImagesStorageRef = storageRef.child("images/\(userID)/\(userPhotoURL)")
119 |
120 | userImagesStorageRef.getData(maxSize: 1 * 100 * 1024 * 1024) { (data, error) in
121 | if let error = error {
122 | print("Error downloading file: ", error.localizedDescription)
123 | completion(nil, false)
124 | } else {
125 | if let data = data {
126 | let image = UIImage(data: data)!
127 | completion(image, true)
128 | } else {
129 | completion(nil, false)
130 | }
131 | }
132 | }
133 | }
134 | }
135 |
--------------------------------------------------------------------------------
/Fit Vein/FIrebase Utils/RCValues.swift:
--------------------------------------------------------------------------------
1 | //
2 | // RCValues.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 03/01/2022.
6 | //
7 |
8 | //import Foundation
9 | //import SwiftUI
10 | //import Firebase
11 | //
12 | //var appPrimaryColor: Color {
13 | // RCValues.sharedInstance.color(forKey: .appPrimaryColor)
14 | //}
15 | //
16 | //class RCValues {
17 | // enum ValueKey: String {
18 | // case appPrimaryColor
19 | // }
20 | //
21 | // static let sharedInstance = RCValues()
22 | //
23 | // var loadingDoneCallback: (() -> Void)?
24 | // var fetchComplete = false
25 | //
26 | // private init() {
27 | // loadDefaultValues()
28 | // fetchCloudValues()
29 | // }
30 | //
31 | // func loadDefaultValues() {
32 | // let appDefaults: [String: Any?] = [
33 | // ValueKey.appPrimaryColor.rawValue : "#00D100"
34 | // ]
35 | // RemoteConfig.remoteConfig().setDefaults(appDefaults as? [String: NSObject])
36 | // }
37 | //
38 | // func activateDebugMode() {
39 | // let settings = RemoteConfigSettings()
40 | // // WARNING: Don't actually do this in production!
41 | // settings.minimumFetchInterval = 0
42 | // RemoteConfig.remoteConfig().configSettings = settings
43 | // }
44 | //
45 | // func fetchCloudValues() {
46 | // activateDebugMode()
47 | //
48 | // RemoteConfig.remoteConfig().fetch { [weak self] _, error in
49 | // if let error = error {
50 | // print("Error fetching remote values: \(error)")
51 | // return
52 | //
53 | // }
54 | //
55 | // RemoteConfig.remoteConfig().activate { _, _ in
56 | // print("Retrieved values from the cloud!")
57 | // }
58 | //
59 | // self?.fetchComplete = true
60 | //
61 | // DispatchQueue.main.async {
62 | // self?.loadingDoneCallback?()
63 | // }
64 | // }
65 | // }
66 | //
67 | // func color(forKey key: ValueKey) -> Color {
68 | // let colorAsHexString = RemoteConfig.remoteConfig()[key.rawValue].stringValue ?? "#00D100"
69 | // let convertedColor = Color(hex: colorAsHexString)
70 | // if self.fetchComplete {
71 | // return convertedColor
72 | // } else {
73 | // return Color(.red)
74 | // }
75 | //
76 | // }
77 | //
78 | // func bool(forKey key: ValueKey) -> Bool {
79 | // RemoteConfig.remoteConfig()[key.rawValue].boolValue
80 | // }
81 | //
82 | // func string(forKey key: ValueKey) -> String {
83 | // RemoteConfig.remoteConfig()[key.rawValue].stringValue ?? ""
84 | // }
85 | //
86 | // func double(forKey key: ValueKey) -> Double {
87 | // RemoteConfig.remoteConfig()[key.rawValue].numberValue.doubleValue
88 | // }
89 | //}
90 |
--------------------------------------------------------------------------------
/Fit Vein/FIrebase Utils/SessionStore.swift:
--------------------------------------------------------------------------------
1 | //
2 | // SessionStore.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 12/10/2021.
6 | //
7 |
8 | import Foundation
9 | import Firebase
10 | import Combine
11 |
12 | struct User {
13 | var uid: String
14 | var email: String
15 | }
16 |
17 | @MainActor
18 | class SessionStore: ObservableObject {
19 |
20 | @Published var session: User?
21 | private var firestoreManager = FirestoreManager()
22 | private var firebaseStorageManager = FirebaseStorageManager()
23 |
24 | var handle: AuthStateDidChangeListenerHandle?
25 | private var authRef = Auth.auth()
26 | public var currentUser = Auth.auth().currentUser
27 |
28 | init(forPreviews: Bool) {
29 | if forPreviews {
30 | self.session = User(uid: "uid", email: "email")
31 | }
32 | }
33 |
34 | func listen() {
35 | handle = authRef.addStateDidChangeListener({ (auth, user) in
36 | if let user = user {
37 | self.session = User(uid: user.uid, email: user.email!)
38 | self.authRef = Auth.auth()
39 | self.currentUser = Auth.auth().currentUser
40 | } else {
41 | self.session = nil
42 | }
43 | })
44 | }
45 |
46 | func signIn(email: String, password: String, completion: @escaping ((Bool) -> ())) {
47 | authRef.signIn(withEmail: email, password: password) { (result, error) in
48 | if let error = error {
49 | print("Error signing in: \(error.localizedDescription)")
50 | completion(false)
51 | } else {
52 | print("Successfully signed in")
53 | completion(true)
54 | }
55 | }
56 | }
57 |
58 | func signUp(email: String, password: String, completion: @escaping ((String?, Bool) -> ())) {
59 | authRef.createUser(withEmail: email, password: password) { (result, error) in
60 | if let error = error {
61 | print("Error signing up: \(error.localizedDescription)")
62 | completion(nil, false)
63 | } else {
64 | print("Successfully signed up")
65 | completion(result!.user.uid, true)
66 | }
67 | }
68 | }
69 |
70 | func signOut() -> Bool {
71 | do {
72 | try Auth.auth().signOut()
73 | self.session = nil
74 | unbind()
75 | return true
76 | } catch {
77 | }
78 | return false
79 | }
80 |
81 | func unbind () {
82 | if let handle = handle {
83 | Auth.auth().removeStateDidChangeListener(handle)
84 | }
85 | }
86 |
87 | // deinit {
88 | // Task {
89 | // await unbind()
90 | // }
91 | // }
92 |
93 | func sendRecoveryEmail(email: String, completion: @escaping ((Bool) -> ())) {
94 | authRef.sendPasswordReset(withEmail: email) { (error) in
95 | if let error = error {
96 | print("Error sending recovery email: \(error.localizedDescription)")
97 | completion(false)
98 | } else {
99 | print("Recovery email has been sent")
100 | completion(true)
101 | }
102 | }
103 | }
104 |
105 | func changeEmailAddress(userID: String, oldEmailAddress: String, password: String, newEmailAddress: String, completion: @escaping ((Bool) -> ())) {
106 | let credential = EmailAuthProvider.credential(withEmail: oldEmailAddress, password: password)
107 |
108 | currentUser?.reauthenticate(with: credential) { [self] (result, error) in
109 | if let error = error {
110 | print("Error re-authenticating user \(error)")
111 | completion(false)
112 | } else {
113 | currentUser?.updateEmail(to: newEmailAddress) { (error) in
114 | if let error = error {
115 | print("Error changing email address: \(error.localizedDescription)")
116 | completion(false)
117 | } else {
118 | completion(true)
119 | }
120 | }
121 | }
122 | }
123 | }
124 |
125 | func changePassword(emailAddress: String, oldPassword: String, newPassword: String, completion: @escaping ((Bool) -> ())) {
126 | let credential = EmailAuthProvider.credential(withEmail: emailAddress, password: oldPassword)
127 |
128 | currentUser?.reauthenticate(with: credential) { [self] (result, error) in
129 | if let error = error {
130 | print("Error re-authenticating user \(error)")
131 | completion(false)
132 | } else {
133 | currentUser?.updatePassword(to: newPassword) { (error) in
134 | if let error = error {
135 | print("Error changing password: \(error.localizedDescription)")
136 | completion(false)
137 | } else {
138 | print("Successfully changed password")
139 | completion(true)
140 | }
141 | }
142 | }
143 | }
144 | }
145 |
146 | func deleteUser(email: String, password: String, completion: @escaping ((Bool) -> ())) {
147 | let credential = EmailAuthProvider.credential(withEmail: email, password: password)
148 |
149 | currentUser?.reauthenticate(with: credential) { [self] (result, error) in
150 | if let error = error {
151 | print("Error re-authenticating user \(error)")
152 | completion(false)
153 | } else {
154 | currentUser?.delete { (error) in
155 | if let error = error {
156 | print("Could not delete user: \(error)")
157 | completion(false)
158 | } else {
159 | let result = self.signOut()
160 | completion(result)
161 | }
162 | }
163 | }
164 | }
165 | }
166 | }
167 |
--------------------------------------------------------------------------------
/Fit Vein/Fit Vein.entitlements:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | com.apple.developer.healthkit
6 |
7 | com.apple.developer.healthkit.access
8 |
9 |
10 |
11 |
--------------------------------------------------------------------------------
/Fit Vein/Fit_VeinApp.swift:
--------------------------------------------------------------------------------
1 | //
2 | // Fit_VeinApp.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 12/10/2021.
6 | //
7 |
8 | import SwiftUI
9 | import Firebase
10 |
11 | @main
12 | struct Fit_VeinApp: App {
13 |
14 | @UIApplicationDelegateAdaptor(AppDelegate.self) var appDelegate
15 |
16 | var body: some Scene {
17 | WindowGroup {
18 | let sessionStore = SessionStore(forPreviews: false)
19 | PreLaunchView()
20 | .environmentObject(sessionStore)
21 | }
22 | }
23 | }
24 |
25 | class AppDelegate: NSObject, UIApplicationDelegate {
26 | func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey : Any]? = nil) -> Bool {
27 | FirebaseApp.configure()
28 |
29 | // Only to use along with Firebase A/B Testing (device ID Token / Installation Token)
30 | // Installations.installations().authTokenForcingRefresh(true, completion: { (result, error) in
31 | // if let error = error {
32 | // print("Error fetching token: \(error)")
33 | // return
34 | // }
35 | // guard let result = result else { return }
36 | // print("Installation auth token: \(result.authToken)")
37 | // })
38 |
39 | // Only to use along with Remote Config values in RCValues class
40 | // _ = RCValues.sharedInstance
41 |
42 | return true
43 | }
44 | }
45 |
46 | extension UIApplication {
47 | struct Constants {
48 | static let CFBundleShortVersionString = "CFBundleShortVersionString"
49 | }
50 | class func appVersion() -> String {
51 | return Bundle.main.object(forInfoDictionaryKey: Constants.CFBundleShortVersionString) as! String
52 | }
53 |
54 | class func appBuild() -> String {
55 | return Bundle.main.object(forInfoDictionaryKey: kCFBundleVersionKey as String) as! String
56 | }
57 |
58 | class func versionBuild() -> String {
59 | let version = appVersion(), build = appBuild()
60 |
61 | return version == build ? "\(version)" : "\(version) (\(build))"
62 | }
63 | }
64 |
65 | extension View {
66 | /// Hide or show the view based on a boolean value.
67 | ///
68 | /// Example for visibility:
69 | ///
70 | /// Text("Label")
71 | /// .isHidden(true)
72 | ///
73 | /// Example for complete removal:
74 | ///
75 | /// Text("Label")
76 | /// .isHidden(true, remove: true)
77 | ///
78 | /// - Parameters:
79 | /// - hidden: Set to `false` to show the view. Set to `true` to hide the view.
80 | /// - remove: Boolean value indicating whether or not to remove the view.
81 | @ViewBuilder func isHidden(_ hidden: Bool, remove: Bool = false) -> some View {
82 | if hidden {
83 | if !remove {
84 | self.hidden()
85 | }
86 | } else {
87 | self
88 | }
89 | }
90 | }
91 |
92 | extension Array: RawRepresentable where Element: Codable {
93 | public init?(rawValue: String) {
94 | guard let data = rawValue.data(using: .utf8),
95 | let result = try? JSONDecoder().decode([Element].self, from: data)
96 | else {
97 | return nil
98 | }
99 | self = result
100 | }
101 |
102 | public var rawValue: String {
103 | guard let data = try? JSONEncoder().encode(self),
104 | let result = String(data: data, encoding: .utf8)
105 | else {
106 | return "[]"
107 | }
108 | return result
109 | }
110 | }
111 |
112 | extension StringProtocol {
113 | var letters: String {
114 | return String(compactMap {
115 | guard let unicodeScalar = $0.unicodeScalars.first else { return nil }
116 | return CharacterSet.letters.contains(unicodeScalar) ? $0 : nil
117 | })
118 | }
119 | }
120 |
121 | extension StringProtocol {
122 | public func removeCharactersFromString(string: String, character: String, before: Bool, upToCharacter: String?) -> String {
123 | if let startingCharacterindex = string.range(of: character)?.lowerBound {
124 | var substring = ""
125 |
126 | if before {
127 | substring = String(string.prefix(upTo: startingCharacterindex))
128 | } else {
129 | if upToCharacter != nil {
130 | if let endingCharacterIndex = string.range(of: upToCharacter!)?.lowerBound {
131 | let lastCharacterIndex = string.endIndex
132 | substring = String(string[.. some View {
147 | for v in vars { print(v) }
148 | return EmptyView()
149 | }
150 | }
151 |
152 | extension View {
153 | @ViewBuilder
154 | func `if`(_ condition: Bool, transform: (Self) -> Transform) -> some View {
155 | if condition { transform(self) }
156 | else { self }
157 | }
158 | }
159 |
160 | struct NavigationBarModifier: ViewModifier {
161 |
162 | var backgroundColor: UIColor?
163 | var titleColor: UIColor?
164 |
165 | init(backgroundColor: UIColor?, titleColor: UIColor?) {
166 | self.backgroundColor = backgroundColor
167 | let coloredAppearance = UINavigationBarAppearance()
168 | coloredAppearance.configureWithTransparentBackground()
169 | coloredAppearance.backgroundColor = backgroundColor
170 | coloredAppearance.titleTextAttributes = [.foregroundColor: titleColor ?? .white]
171 | coloredAppearance.largeTitleTextAttributes = [.foregroundColor: titleColor ?? .white]
172 |
173 | UINavigationBar.appearance().standardAppearance = coloredAppearance
174 | UINavigationBar.appearance().compactAppearance = coloredAppearance
175 | UINavigationBar.appearance().scrollEdgeAppearance = coloredAppearance
176 | }
177 |
178 | func body(content: Content) -> some View {
179 | ZStack{
180 | content
181 | VStack {
182 | GeometryReader { geometry in
183 | Color(self.backgroundColor ?? .clear)
184 | .frame(height: geometry.safeAreaInsets.top)
185 | .edgesIgnoringSafeArea(.top)
186 | Spacer()
187 | }
188 | }
189 | }
190 | }
191 | }
192 |
193 | extension View {
194 | func navigationBarColor(backgroundColor: UIColor?, titleColor: UIColor?) -> some View {
195 | self.modifier(NavigationBarModifier(backgroundColor: backgroundColor, titleColor: titleColor))
196 | }
197 | }
198 |
199 | extension UIColor {
200 | convenience init(red: Int, green: Int, blue: Int, a: CGFloat = 1.0) {
201 | self.init(
202 | red: CGFloat(red) / 255.0,
203 | green: CGFloat(green) / 255.0,
204 | blue: CGFloat(blue) / 255.0,
205 | alpha: a
206 | )
207 | }
208 |
209 | convenience init(rgb: Int, a: CGFloat = 1.0) {
210 | self.init(
211 | red: (rgb >> 16) & 0xFF,
212 | green: (rgb >> 8) & 0xFF,
213 | blue: rgb & 0xFF,
214 | a: a
215 | )
216 | }
217 | }
218 |
219 | extension UIScreen{
220 | static let screenWidth = UIScreen.main.bounds.size.width
221 | static let screenHeight = UIScreen.main.bounds.size.height
222 | static let screenSize = UIScreen.main.bounds.size
223 | }
224 |
225 | extension Date: RawRepresentable {
226 | private static let formatter = ISO8601DateFormatter()
227 |
228 | public var rawValue: String {
229 | Date.formatter.string(from: self)
230 | }
231 |
232 | public init?(rawValue: String) {
233 | self = Date.formatter.date(from: rawValue) ?? Date()
234 | }
235 | }
236 |
237 | extension UIApplication {
238 | func endEditing() {
239 | sendAction(#selector(UIResponder.resignFirstResponder), to: nil, from: nil, for: nil)
240 | }
241 | }
242 |
--------------------------------------------------------------------------------
/Fit Vein/Images/FitVeinIconDark.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Images/FitVeinIconDark.png
--------------------------------------------------------------------------------
/Fit Vein/Images/FitVeinIconLight.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Images/FitVeinIconLight.png
--------------------------------------------------------------------------------
/Fit Vein/Images/blank-profile-hi.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Images/blank-profile-hi.png
--------------------------------------------------------------------------------
/Fit Vein/Images/gym.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Images/gym.jpg
--------------------------------------------------------------------------------
/Fit Vein/Images/sprint2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Images/sprint2.png
--------------------------------------------------------------------------------
/Fit Vein/LoggedUserView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // LoggedUserView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 20/10/2021.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct LoggedUserView: View {
11 | @StateObject private var homeViewModel = HomeViewModel()
12 | @StateObject private var workoutViewModel = WorkoutViewModel(forPreviews: false)
13 | @StateObject private var profileViewModel = ProfileViewModel()
14 | @StateObject private var medalsViewModel = MedalsViewModel()
15 | @StateObject private var networkManager = NetworkManager()
16 |
17 | @AppStorage("isTabBarHidden") var isTabBarHidden: Bool = false
18 |
19 | @AppStorage("shouldShowOnboarding") var shouldShowOnboarding: Bool = true
20 | @State private var shouldShowOnboardingState: Bool = false
21 |
22 | @State var selectedTab: Tab = .home
23 |
24 | enum Tab: String {
25 | case home
26 | case workout
27 | case profile
28 | }
29 |
30 | private var tabItems = [
31 | TabItem(text: String(localized: "LoggedUserView_home_tab_label"), icon: "house", tab: .home),
32 | TabItem(text: String(localized: "LoggedUserView_workout_tab_label"), icon: "figure.walk", tab: .workout),
33 | TabItem(text: String(localized: "LoggedUserView_profile_tab_label"), icon: "person", tab: .profile)
34 | ]
35 |
36 | var body: some View {
37 | GeometryReader { geometry in
38 | let screenWidth = geometry.size.width
39 | let screenHeight = geometry.size.height
40 |
41 | Group {
42 | switch selectedTab {
43 | case .home:
44 | withAnimation(.linear) {
45 | HomeView()
46 | .environmentObject(homeViewModel)
47 | .environmentObject(profileViewModel)
48 | .environmentObject(medalsViewModel)
49 | .environmentObject(networkManager)
50 | .onAppear {
51 | UserDefaults.standard.set(false, forKey: "isTabBarHidden")
52 | }
53 | }
54 | case .workout:
55 | withAnimation(.linear) {
56 | WorkoutView()
57 | .environmentObject(workoutViewModel)
58 | .environmentObject(medalsViewModel)
59 | .environmentObject(networkManager)
60 | .onAppear {
61 | UserDefaults.standard.set(false, forKey: "isTabBarHidden")
62 | }
63 | }
64 | case .profile:
65 | withAnimation(.linear) {
66 | ProfileView()
67 | .environmentObject(profileViewModel)
68 | .environmentObject(medalsViewModel)
69 | .environmentObject(networkManager)
70 | .onAppear {
71 | UserDefaults.standard.set(false, forKey: "isTabBarHidden")
72 | }
73 | }
74 | }
75 | }
76 | .frame(maxWidth: .infinity, maxHeight: .infinity)
77 |
78 | HStack {
79 | Spacer()
80 |
81 | ForEach(tabItems) { tabItem in
82 | Spacer()
83 |
84 | Button {
85 | withAnimation(.spring(response: 0.3, dampingFraction: 0.7)) {
86 | selectedTab = tabItem.tab
87 | }
88 | } label: {
89 | VStack(spacing: 0) {
90 | Image(systemName: tabItem.icon)
91 | .symbolVariant(.fill)
92 | .font(.body.bold())
93 | .frame(width: 44, height: 29)
94 | Text(tabItem.text)
95 | .font(.caption2)
96 | .lineLimit(1)
97 | }
98 | .frame(maxWidth: .infinity)
99 | .foregroundColor(selectedTab == tabItem.tab ? .accentColor : Color(uiColor: .systemGray))
100 | }
101 | .foregroundStyle(selectedTab == tabItem.tab ? .primary : .secondary)
102 |
103 | Spacer()
104 | }
105 |
106 | Spacer()
107 | }
108 | .padding(.horizontal, 7)
109 | .padding(.top, 10)
110 | .frame(height: screenHeight * 0.11, alignment: .top)
111 | .background(.ultraThinMaterial, in: RoundedRectangle(cornerRadius: 40, style: .continuous))
112 | .overlay(
113 | HStack {
114 | if selectedTab == .workout {
115 | Spacer()
116 | }
117 |
118 | if selectedTab == .profile {
119 | Spacer()
120 | }
121 |
122 | Rectangle()
123 | .fill(Color.accentColor)
124 | .frame(width: screenWidth * 0.15, height: screenHeight * 0.007)
125 | .cornerRadius(3)
126 | .frame(width: screenWidth * 0.235)
127 | .frame(maxHeight: .infinity, alignment: .top)
128 |
129 | if selectedTab == .home {
130 | Spacer()
131 | }
132 |
133 | if selectedTab == .workout {
134 | Spacer()
135 | }
136 | }
137 | .padding(selectedTab == .home ? .leading : .trailing, selectedTab == .workout ? 0 : screenWidth * 0.074)
138 | )
139 | .frame(maxHeight: .infinity, alignment: .bottom)
140 | .ignoresSafeArea()
141 | .isHidden(isTabBarHidden)
142 | }
143 | .onAppear {
144 | self.shouldShowOnboardingState = shouldShowOnboarding
145 | }
146 | .fullScreenCover(isPresented: $shouldShowOnboarding, onDismiss: {
147 | shouldShowOnboarding = false
148 | shouldShowOnboardingState = false
149 | }, content: {
150 | OnboardingView()
151 | })
152 | }
153 |
154 | struct TabItem: Identifiable {
155 | var id = UUID()
156 | var text: String
157 | var icon: String
158 | var tab: Tab
159 | }
160 | }
161 |
162 | struct LoggedUserView_Previews: PreviewProvider {
163 | static var previews: some View {
164 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
165 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
166 | LoggedUserView()
167 | .preferredColorScheme(colorScheme)
168 | .previewDevice(PreviewDevice(rawValue: deviceName))
169 | .previewDisplayName(deviceName)
170 | }
171 | }
172 | }
173 | }
174 |
--------------------------------------------------------------------------------
/Fit Vein/Lottie/70772-smiley-loader.json:
--------------------------------------------------------------------------------
1 | {"v":"5.5.7","meta":{"g":"LottieFiles AE 0.1.21","a":"","k":"","d":"","tc":""},"fr":60,"ip":0,"op":361,"w":400,"h":400,"nm":"Loader","ddd":0,"assets":[{"id":"comp_0","layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Left Eye","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.555,"y":1},"o":{"x":1,"y":0},"t":60,"s":[800,627.5,0],"to":[-4.5,2.083,0],"ti":[4.5,-2.083,0]},{"i":{"x":0.335,"y":0.335},"o":{"x":0.167,"y":0.167},"t":120,"s":[773,640,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.763,"y":1},"o":{"x":1,"y":0},"t":257,"s":[773,640,0],"to":[4.5,-2.083,0],"ti":[-4.5,2.083,0]},{"t":310,"s":[800,627.5,0]}],"ix":2},"a":{"a":0,"k":[20,-36,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0,0,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":10,"s":[0,0,100]},{"i":{"x":[0.555,0.555,0.667],"y":[1,1,1]},"o":{"x":[1,1,0.333],"y":[0,0,0]},"t":60,"s":[140,140,100]},{"i":{"x":[0.497,0.497,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":120,"s":[100,100,100]},{"i":{"x":[0,0,0.667],"y":[1,1,1]},"o":{"x":[1,1,0.167],"y":[0,0,0]},"t":257,"s":[100,100,100]},{"i":{"x":[0.117,0.117,0.833],"y":[1,1,1]},"o":{"x":[0.14,0.14,0.333],"y":[0,0,0]},"t":310,"s":[140,140,100]},{"t":359,"s":[0,0,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[35,35],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":124,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[0.094117647059,0.250980392157,0.160784313725,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[20,-36],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":600,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Right Eye","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.555,"y":1},"o":{"x":1,"y":0},"t":60,"s":[800,627.5,0],"to":[4.667,2.083,0],"ti":[-4.667,-2.083,0]},{"i":{"x":0.76,"y":0.76},"o":{"x":0.167,"y":0.167},"t":120,"s":[828,640,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.769,"y":1},"o":{"x":1,"y":0},"t":257,"s":[828,640,0],"to":[-4.667,-2.083,0],"ti":[4.667,2.083,0]},{"t":310,"s":[800,627.5,0]}],"ix":2},"a":{"a":0,"k":[20,-36,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0,0,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":10,"s":[0,0,100]},{"i":{"x":[0.555,0.555,0.667],"y":[1,1,1]},"o":{"x":[1,1,0.333],"y":[0,0,0]},"t":60,"s":[140,140,100]},{"i":{"x":[0.497,0.497,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":120,"s":[100,100,100]},{"i":{"x":[0,0,0.667],"y":[1,1,1]},"o":{"x":[1,1,0.167],"y":[0,0,0]},"t":257,"s":[100,100,100]},{"i":{"x":[0.112,0.112,0.833],"y":[1,1,1]},"o":{"x":[0.141,0.141,0.333],"y":[0,0,0]},"t":310,"s":[140,140,100]},{"t":359,"s":[0,0,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":1,"k":[{"i":{"x":[0,0.457],"y":[1,1]},"o":{"x":[0.936,0.455],"y":[0,0]},"t":150,"s":[35,35]},{"i":{"x":[0.748,0.667],"y":[1,1]},"o":{"x":[0.862,0.333],"y":[0,0]},"t":170,"s":[35,12]},{"i":{"x":[0,0.457],"y":[1,1]},"o":{"x":[0.392,0.855],"y":[0,0]},"t":182,"s":[38,10]},{"i":{"x":[0.107,0.667],"y":[1,1]},"o":{"x":[0.862,0.333],"y":[0,0]},"t":193,"s":[35,12]},{"t":213,"s":[35,35]}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":124,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[0.094117647059,0.250980392157,0.160784313725,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[20,-36],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":600,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Smile Lip","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.239],"y":[1]},"o":{"x":[0.373],"y":[0]},"t":60,"s":[-180]},{"i":{"x":[0.452],"y":[1]},"o":{"x":[0.877],"y":[0]},"t":77,"s":[-210]},{"i":{"x":[0.381],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":120,"s":[0]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.015],"y":[0]},"t":230,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.836],"y":[0]},"t":247,"s":[-20]},{"t":310,"s":[180]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.761,"y":0.761},"o":{"x":0.167,"y":0.167},"t":120,"s":[800,640,0],"to":[0,0,0],"ti":[0,0,0]},{"t":230,"s":[800,640,0]}],"ix":2},"a":{"a":0,"k":[-136.248,-44.248,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.761,0.761,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":120,"s":[100.678,100,100]},{"t":230,"s":[100.678,100,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[147.503,147.503],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":124,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"st","c":{"a":0,"k":[0.094117647059,0.250980392157,0.160784313725,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":16,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-136.248,-44.248],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":10,"s":[11]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":60,"s":[0]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.134],"y":[0]},"t":230,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":310,"s":[0]},{"t":359,"s":[11]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":10,"s":[11]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":60,"s":[23]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.134],"y":[0]},"t":230,"s":[23]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":310,"s":[23]},{"t":359,"s":[11]}],"ix":2},"o":{"a":0,"k":48,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":600,"st":0,"bm":0}]}],"layers":[{"ddd":0,"ind":1,"ty":0,"nm":"Smile_lottie","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[200,144,0],"ix":2},"a":{"a":0,"k":[800,600,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"w":1600,"h":1200,"ip":0,"op":600,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"BG","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[200,200,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[202.25,204,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[800,600],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[0.373502604167,1,0.643871292413,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":600,"st":0,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/Lottie/downArrows.json:
--------------------------------------------------------------------------------
1 | {"v":"5.5.7","meta":{"g":"LottieFiles AE 0.1.20","a":"","k":"","d":"","tc":""},"fr":30,"ip":0,"op":33,"w":500,"h":500,"nm":"arrow animation","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Layer 5 Outlines","sr":1,"ks":{"o":{"a":0,"k":50,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":2,"s":[250,178,0],"to":[0,6.667,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":16,"s":[250,218.00000000000003,0],"to":[0,0,0],"ti":[0,6.667,0]},{"t":30,"s":[250,178,0]}],"ix":2},"a":{"a":0,"k":[72.596,41.955,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[57.596,-26.956],[-0.001,26.956],[-57.597,-26.956]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.25098039215686274,0.8392156862745098,0.20392156862745098,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":30,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[72.596,41.955],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":35,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Layer 4 Outlines","sr":1,"ks":{"o":{"a":0,"k":80,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":3,"s":[250,250.99999999999997,0],"to":[0,6.667,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":17,"s":[250,291,0],"to":[0,0,0],"ti":[0,6.667,0]},{"t":32,"s":[250,250.99999999999997,0]}],"ix":2},"a":{"a":0,"k":[72.596,41.955,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[57.596,-26.955],[-0.001,26.955],[-57.597,-26.955]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.25098039215686274,0.8392156862745098,0.20392156862745098,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":30,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[72.596,41.955],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":35,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Layer 3 Outlines","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":4,"s":[250,324,0],"to":[0,6.667,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":18,"s":[250,364,0],"to":[0,0,0],"ti":[0,6.667,0]},{"t":34,"s":[250,324,0]}],"ix":2},"a":{"a":0,"k":[132.596,101.955,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[57.596,-26.955],[-0.001,26.955],[-57.597,-26.955]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.25098039215686274,0.8392156862745098,0.20392156862745098,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":30,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[132.596,101.955],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":35,"st":0,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/Lottie/heartRate.json:
--------------------------------------------------------------------------------
1 | {"v":"5.5.8","fr":29.9700012207031,"ip":0,"op":219.000008920053,"w":500,"h":500,"nm":"Comp 1","ddd":0,"assets":[{"id":"comp_0","layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[1040,250,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[-5.667,-29.333],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-1032,24],[-952,24],[-929,-63],[-912,25],[-881,25],[-856,-130],[-822,119],[-811,18],[-762,18],[-746,-78],[-722,16],[-637,16],[-614,114],[-598,18],[-498,18]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.8156862745098039,0.00784313725490196,0.10588235294117647,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":10,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":52,"s":[0]},{"t":174.000007087165,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":2,"s":[0]},{"t":127.000005172816,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":180.00000733155,"st":0,"bm":0}]}],"layers":[{"ddd":0,"ind":1,"ty":0,"nm":"Beat1","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[992,250,0],"ix":2},"a":{"a":0,"k":[1000,250,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"w":2000,"h":500,"ip":213.000008675668,"op":386.000015722102,"st":206.000008390552,"bm":0},{"ddd":0,"ind":2,"ty":0,"nm":"Beat1","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[992,250,0],"ix":2},"a":{"a":0,"k":[1000,250,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"w":2000,"h":500,"ip":98.0000039916218,"op":278.000011323172,"st":98.0000039916218,"bm":0},{"ddd":0,"ind":3,"ty":0,"nm":"Beat1","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[992,250,0],"ix":2},"a":{"a":0,"k":[1000,250,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"w":2000,"h":500,"ip":-2.00000008146167,"op":166.000006761319,"st":-14.0000005702317,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[1033,250,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[-5.667,-29.333],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-1035,24],[-952,24],[-929,-63],[-912,25],[-881,25],[-856,-130],[-822,119],[-811,18],[-762,18],[-746,-78],[-722,16],[-637,16],[-614,114],[-598,18],[-498,18]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.8156862745098039,0.00784313725490196,0.10588235294117647,1],"ix":3},"o":{"a":0,"k":30,"ix":4},"w":{"a":0,"k":10,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":385.000015681371,"st":0,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/Lottie/loading.json:
--------------------------------------------------------------------------------
1 | {"v":"5.8.1","fr":29.9700012207031,"ip":0,"op":50.0000020365418,"w":568,"h":228,"nm":"Loading","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":19,"s":[100]},{"t":40.0000016292334,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[193,126.453,0],"ix":2,"l":2},"a":{"a":0,"k":[-144.773,14,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,-0.174]},"t":0,"s":[67.692,100,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":21,"s":[94.692,139.886,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,6.95]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":25,"s":[94,138.864,100]},{"t":47.0000019143492,"s":[67,98.977,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[36,115],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.274563389198,0.637646963082,0.308735985849,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-148,12.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":50.0000020365418,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":4,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":20,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":23,"s":[100]},{"t":44.0000017921567,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[275.273,126.458,0],"ix":2,"l":2},"a":{"a":0,"k":[-144.773,14,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":4,"s":[67.692,100,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,-0.25]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":25,"s":[94.692,139.886,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":29,"s":[94,138.864,100]},{"t":51.0000020772726,"s":[67,98.977,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[36,115],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.274563389198,0.637646963082,0.308735985849,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-148,12.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":4.00000016292334,"op":54.0000021994651,"st":4.00000016292334,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Shape Layer 3","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":26,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":29,"s":[100]},{"t":50.0000020365418,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[357,126.088,0],"ix":2,"l":2},"a":{"a":0,"k":[-144.773,14,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,-0.174]},"t":10,"s":[67.692,100,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":31,"s":[94.692,139.886,100]},{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,6.95]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":35,"s":[94,138.864,100]},{"t":57.0000023216576,"s":[67,98.977,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[36,115],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.274563389198,0.637646963082,0.308735985849,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-148,12.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":10.0000004073083,"op":60.0000024438501,"st":10.0000004073083,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/Lottie/skeleton.json:
--------------------------------------------------------------------------------
1 | {"v":"4.8.0","meta":{"g":"LottieFiles AE ","a":"","k":"","d":"","tc":""},"fr":24,"ip":0,"op":48,"w":650,"h":611,"nm":"Artboard 1","ddd":0,"assets":[{"id":"image_0","w":650,"h":611,"u":"","p":"","e":1}],"layers":[{"ddd":0,"ind":1,"ty":2,"nm":"Artboard 1.png","cl":"png","refId":"image_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":24,"s":[14]},{"t":48,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[325,305.5,0],"ix":2},"a":{"a":0,"k":[325,305.5,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":48,"st":0,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/Lottie/success2.json:
--------------------------------------------------------------------------------
1 | {"v":"5.5.7","meta":{"g":"LottieFiles AE 0.1.20","a":"","k":"","d":"","tc":""},"fr":30,"ip":0,"op":60,"w":608,"h":608,"nm":"Comp 1","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"thik chino","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[301.5,281,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-119,30],[-44,96],[112,-50]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":17,"s":[0]},{"t":37,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":25,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"fill","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[304,304,0],"ix":2},"a":{"a":0,"k":[-38,0,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0,0,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":16,"s":[0,0,100]},{"t":21,"s":[100,100,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[100.516,0],[0,-100.516],[-100.516,0],[0,100.516]],"o":[[-100.516,0],[0,100.516],[100.516,0],[0,-100.516]],"v":[[0,-182],[-182,0],[0,182],[182,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.223528992896,0.349019996793,0.517646998985,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.170349854114,0.819607843137,0.445329823213,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-38,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"stock","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[304,304,0],"ix":2},"a":{"a":0,"k":[-38,0,0],"ix":1},"s":{"a":0,"k":[107,107,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[100.516,0],[0,-100.516],[-100.516,0],[0,100.516]],"o":[[-100.516,0],[0,100.516],[100.516,0],[0,-100.516]],"v":[[0,-182],[-182,0],[0,182],[182,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[100]},{"t":20,"s":[0]}],"ix":1},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.170934250776,0.764705882353,0.424896808699,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":25,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-38,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60,"st":0,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/Lottie/time.json:
--------------------------------------------------------------------------------
1 | {"v":"5.5.4","fr":60,"ip":0,"op":480,"w":48,"h":48,"nm":"Timer","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"seconds","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[235]},{"t":479,"s":[595]}],"ix":10},"p":{"a":0,"k":[24.375,23.781,0],"ix":2},"a":{"a":0,"k":[-21.625,0.32,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[4,16],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":100,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"st","c":{"a":0,"k":[0.49411764705882355,0.8274509803921568,0.12941176470588237,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.49411764705882355,0.8274509803921568,0.12941176470588237,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-21.625,0.5],"ix":2},"a":{"a":0,"k":[0,6.25],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":480,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"minutes","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[24.375,23.781,0],"ix":2},"a":{"a":0,"k":[-21.625,2.422,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[4,20],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":112,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"st","c":{"a":0,"k":[0.49411764705882355,0.8274509803921568,0.12941176470588237,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.49411764705882355,0.8274509803921568,0.12941176470588237,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-21.625,-5.75],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":480,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"background","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[24.379,23.758,0],"ix":2},"a":{"a":0,"k":[0.504,2.008,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[24.5,24.5],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[0.49411764705882355,0.8274509803921568,0.12941176470588237,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.26666666666666666,0.2549019607843137,0.2549019607843137,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0.5,2],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[186.718,186.718],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":480,"st":0,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/Lottie/time2.json:
--------------------------------------------------------------------------------
1 | {"v":"5.4.3","fr":29.9700012207031,"ip":0,"op":60.0000024438501,"w":640,"h":640,"nm":"Comp 2","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[313.25,237.77,0],"ix":2},"a":{"a":0,"k":[2.596,-12.179,0],"ix":1},"s":{"a":0,"k":[1054.029,1054.029,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.829,0],[0,0],[0,-0.828],[-0.829,0],[0,0],[0,0.828]],"o":[[0,0],[-0.829,0],[0,0.828],[0,0],[0.829,0],[0,-0.828]],"v":[[17.887,6.197],[7.212,6.197],[5.712,7.697],[7.212,9.197],[17.887,9.197],[19.387,7.697]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.8156862745098039,0.00784313725490196,0.10588235294117647,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0.29,"y":1},"o":{"x":0.71,"y":0},"n":"0p29_1_0p71_0","t":0,"s":[-0.231,-25.763],"e":[-0.231,-24.909],"to":[0,0],"ti":[0,0]},{"i":{"x":0.29,"y":1},"o":{"x":0.71,"y":0},"n":"0p29_1_0p71_0","t":9,"s":[-0.231,-24.909],"e":[-0.231,-25.763],"to":[0,0],"ti":[0,0]},{"t":13.0000005295009}],"ix":2},"a":{"a":0,"k":[9.722,7.87],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 4","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.829,0],[0,0],[0,-0.828],[-0.829,0],[0,0],[0,0.828]],"o":[[0,0],[-0.829,0],[0,0.828],[0,0],[0.829,0],[0,-0.828]],"v":[[3.535,-2.136],[-8.804,-2.136],[-10.304,-0.636],[-8.804,0.864],[3.535,0.864],[5.035,-0.636]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.8156862745098039,0.00784313725490196,0.10588235294117647,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[2.773,-1.262],"ix":2},"a":{"a":0,"k":[3.542,-0.631],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"n":["0p833_0p833_0p167_0p167"],"t":12,"s":[-125],"e":[595]},{"t":55.0000022401959}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 5","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.829,0],[0,0],[0,-0.828],[-0.829,0],[0,0],[0,0.828]],"o":[[0,0],[-0.829,0],[0,0.828],[0,0],[0.829,0],[0,-0.828]],"v":[[3.535,-2.136],[-8.804,-2.136],[-10.304,-0.636],[-8.804,0.864],[3.535,0.864],[5.035,-0.636]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0,0,0,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[2.662,0.347],"ix":2},"a":{"a":0,"k":[5.15,-0.521],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":90,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 3","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60.0000024438501,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-89,"ix":10},"p":{"a":0,"k":[312.714,348,0],"ix":2},"a":{"a":0,"k":[0.579,0.579,0],"ix":1},"s":{"a":0,"k":[1054.029,1054.029,100],"ix":6}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"s","pt":{"a":0,"k":{"i":[[10.204,0],[0,10.204],[-10.204,0],[0,-10.204]],"o":[[-10.204,0],[0,-10.204],[10.204,0],[0,10.204]],"v":[[0.5,19.005],[-18.005,0.5],[0.5,-18.005],[19.005,0.5]],"c":true},"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[11.858,0],[0,-11.858],[-11.858,0],[0,11.857]],"o":[[-11.858,0],[0,11.857],[11.858,0],[0,-11.858]],"v":[[0.579,-20.927],[-20.927,0.579],[0.579,22.084],[22.084,0.579]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0,0,0,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[23.727,23.032],"ix":2},"a":{"a":0,"k":[23.727,23.032],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 5","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":60.0000024438501,"st":0,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/Lottie/wrongData.json:
--------------------------------------------------------------------------------
1 | {"v":"5.5.7","meta":{"g":"LottieFiles AE 0.1.20","a":"","k":"","d":"","tc":""},"fr":60,"ip":0,"op":240,"w":1920,"h":1080,"nm":"Comp 1","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 6","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":90,"ix":10},"p":{"a":0,"k":[960,540,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[69.737,69.737,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-150,-151],[152,151]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":62,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.429],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":91,"s":[0]},{"t":121,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":240,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 5","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[960,540,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[69.737,69.737,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-150,-151],[152,151]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":62,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.429],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":91,"s":[0]},{"t":121,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":240,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Shape Layer 3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[960,540,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0,0,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":60,"s":[0,0,100]},{"t":91,"s":[100,100,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[720,720],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":58,"s":[100]},{"t":121,"s":[0]}],"ix":1},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.487058877945,0.067615278065,0.067615278065,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":20,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[86.756,86.756],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":240,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[960,540,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0,0,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":60,"s":[0,0,100]},{"t":91,"s":[100,100,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[720,720],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"fl","c":{"a":0,"k":[0.869803873698,0.243545158237,0.243545158237,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[86.756,86.756],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":240,"st":0,"bm":0}],"markers":[]}
--------------------------------------------------------------------------------
/Fit Vein/LottieView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // LottieView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 09/01/2022.
6 | //
7 |
8 | import SwiftUI
9 | import Lottie
10 |
11 | struct LottieView: UIViewRepresentable {
12 | var name: String
13 | var loopMode: LottieLoopMode = .loop
14 | var contentMode: UIView.ContentMode = .scaleAspectFit
15 | var paused: Bool = false
16 |
17 | var animationView = AnimationView()
18 |
19 | func makeUIView(context: UIViewRepresentableContext) -> UIView {
20 | let view = UIView(frame: .zero)
21 |
22 | animationView.animation = Animation.named(name)
23 | animationView.contentMode = contentMode
24 | animationView.loopMode = loopMode
25 | animationView.backgroundBehavior = .pauseAndRestore
26 | animationView.play()
27 |
28 | animationView.translatesAutoresizingMaskIntoConstraints = false
29 | view.addSubview(animationView)
30 |
31 | NSLayoutConstraint.activate([
32 | animationView.heightAnchor.constraint(equalTo: view.heightAnchor),
33 | animationView.widthAnchor.constraint(equalTo: view.widthAnchor)
34 | ])
35 |
36 | return view
37 | }
38 |
39 | func updateUIView(_ uiView: UIView, context: UIViewRepresentableContext) {
40 | }
41 |
42 | func changeAnimationState(pause: Bool) {
43 | if pause {
44 | self.animationView.pause()
45 | } else {
46 | self.animationView.play()
47 | }
48 | }
49 | }
50 |
--------------------------------------------------------------------------------
/Fit Vein/Medals/medal-2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medal-2.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medal-3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medal-3.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medal-4.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medal-4.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medal.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medal.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medalFirstComment.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medalFirstComment.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medalFirstLike.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medalFirstLike.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medalFirstOwnWorkout.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medalFirstOwnWorkout.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medalFirstPost.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medalFirstPost.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medalFirstWorkout.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medalFirstWorkout.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medalSecondLevel.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medalSecondLevel.png
--------------------------------------------------------------------------------
/Fit Vein/Medals/medalThirdLevel.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Medals/medalThirdLevel.png
--------------------------------------------------------------------------------
/Fit Vein/Models/HealthStat.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HealthStat.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 24/11/2021.
6 | //
7 |
8 | import Foundation
9 | import HealthKit
10 |
11 | struct HealthStat: Identifiable {
12 | let id: UUID
13 | let stat: HKQuantity?
14 | let date: Date
15 |
16 | init(id: UUID = UUID(), stat: HKQuantity?, date: Date) {
17 | self.id = id
18 | self.stat = stat
19 | self.date = date
20 | }
21 | }
22 |
--------------------------------------------------------------------------------
/Fit Vein/Models/IntervalWorkout.swift:
--------------------------------------------------------------------------------
1 | //
2 | // IntervalWorkout.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 25/10/2021.
6 | //
7 |
8 | import Foundation
9 |
10 | struct IntervalWorkout: Codable, Identifiable {
11 | var id: String
12 | var usersID: String
13 | var type: String
14 | var date: Date
15 | var isFinished: Bool
16 | var duration: Int?
17 | var calories: Int?
18 | var series: Int?
19 | var workTime: Int?
20 | var restTime: Int?
21 | var completedDuration: Int?
22 | var completedSeries: Int?
23 |
24 | init(id: String, usersID: String, type: String, date: Date, isFinished: Bool = false, calories: Int?, series: Int?, workTime: Int?, restTime: Int?) {
25 | self.id = id
26 | self.usersID = usersID
27 | self.type = type
28 | self.date = date
29 | self.duration = (series! * workTime!) + (series! * restTime!)
30 | self.isFinished = isFinished
31 | self.calories = calories
32 | self.series = series
33 | self.workTime = workTime
34 | self.restTime = restTime
35 | }
36 |
37 | init(forPreviews: Bool, id: String, usersID: String, type: String, date: Date, isFinished: Bool = false, calories: Int?, series: Int?, workTime: Int?, restTime: Int?, completedDuration: Int?, completedSeries: Int?) {
38 | self.id = id
39 | self.usersID = usersID
40 | self.type = type
41 | self.date = date
42 | self.duration = (series! * workTime!) + (series! * restTime!)
43 | self.isFinished = isFinished
44 | self.calories = calories
45 | self.series = series
46 | self.workTime = workTime
47 | self.restTime = restTime
48 | self.completedDuration = completedDuration
49 | self.completedSeries = completedSeries
50 | }
51 |
52 | mutating func setDataOnEnd(calories: Int?, completedDuration: Int?, completedSeries: Int?) {
53 | self.isFinished = true
54 | self.calories = calories
55 | self.completedDuration = completedDuration
56 | self.completedSeries = completedSeries
57 | }
58 |
59 | mutating func setUsersID(usersID: String) {
60 | self.usersID = usersID
61 | }
62 | }
63 |
--------------------------------------------------------------------------------
/Fit Vein/Models/Post.swift:
--------------------------------------------------------------------------------
1 | //
2 | // Post.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 28/10/2021.
6 | //
7 |
8 | import Foundation
9 |
10 | struct Post: Codable, Identifiable {
11 | var id: String
12 | var authorID: String
13 | var authorFirstName: String
14 | var authorUsername: String
15 | var authorProfilePictureURL: String
16 | var addDate: Date
17 | var text: String
18 | var reactionsUsersIDs: [String]?
19 | var commentedUsersIDs: [String]?
20 | var comments: [Comment]?
21 | var photoURL: String?
22 |
23 | init(authorID: String, authorFirstName: String, authorUsername: String, authorProfilePictureURL: String, text: String) {
24 | self.id = UUID().uuidString
25 | self.authorID = authorID
26 | self.authorFirstName = authorFirstName
27 | self.authorUsername = authorUsername
28 | self.authorProfilePictureURL = authorProfilePictureURL
29 | self.addDate = Date()
30 | self.text = text
31 | }
32 |
33 | init(id: String, authorID: String, authorFirstName: String, authorUsername: String, authorProfilePictureURL: String, addDate: Date, text: String, reactionsUsersIDs: [String]?, commentedUsersIDs: [String]?, comments: [Comment]?, photoURL: String? = nil) {
34 | self.id = id
35 | self.authorID = authorID
36 | self.authorFirstName = authorFirstName
37 | self.authorUsername = authorUsername
38 | self.authorProfilePictureURL = authorProfilePictureURL
39 | self.addDate = addDate
40 | self.text = text
41 | self.reactionsUsersIDs = reactionsUsersIDs
42 | self.commentedUsersIDs = commentedUsersIDs
43 | self.comments = comments
44 | self.photoURL = photoURL
45 | }
46 | }
47 |
48 | struct Comment: Codable, Identifiable {
49 | var id: String
50 | var authorID: String
51 | var postID: String
52 | var authorFirstName: String
53 | var authorUsername: String
54 | var authorProfilePictureURL: String
55 | var addDate: Date
56 | var reactionsUsersIDs: [String]?
57 | var text: String
58 |
59 | init(authorID: String, postID: String, authorFirstName: String, authorUsername: String, authorProfilePictureURL: String, text: String) {
60 | self.id = UUID().uuidString
61 | self.authorID = authorID
62 | self.postID = postID
63 | self.authorFirstName = authorFirstName
64 | self.authorUsername = authorUsername
65 | self.authorProfilePictureURL = authorProfilePictureURL
66 | self.addDate = Date()
67 | self.text = text
68 | }
69 |
70 | init(id: String, authorID: String, postID: String, authorFirstName: String, authorUsername: String, authorProfilePictureURL: String, addDate: Date, text: String, reactionsUsersIDs: [String]?) {
71 | self.id = id
72 | self.authorID = authorID
73 | self.postID = postID
74 | self.authorFirstName = authorFirstName
75 | self.authorUsername = authorUsername
76 | self.authorProfilePictureURL = authorProfilePictureURL
77 | self.addDate = addDate
78 | self.text = text
79 | self.reactionsUsersIDs = reactionsUsersIDs
80 | }
81 | }
82 |
--------------------------------------------------------------------------------
/Fit Vein/Models/Profile.swift:
--------------------------------------------------------------------------------
1 | //
2 | // Profile.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 20/10/2021.
6 | //
7 |
8 | import Foundation
9 |
10 | struct Profile: Codable, Identifiable {
11 | var id: String
12 | var firstName: String
13 | var username: String
14 | var birthDate: Date
15 | var age: Int
16 | var country: String
17 | var language: String
18 | var gender: String
19 | var email: String
20 | var profilePictureURL: String?
21 | var followedIDs: [String]?
22 | var reactedPostsIDs: [String]?
23 | var commentedPostsIDs: [String]?
24 | var reactedCommentsIDs: [String]?
25 | var completedWorkouts: Int = 0
26 | var level: Int = 1
27 | }
28 |
29 |
30 |
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/AddWorkoutView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/AddWorkoutView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/FinishedWorkoutView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/FinishedWorkoutView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/HealthTabView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/HealthTabView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/HomeTabView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/HomeTabView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/MedalView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/MedalView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/PostCommentsView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/PostCommentsView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/ProfileView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/ProfileView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/SearchFriendsView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/SearchFriendsView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/SettingsView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/SettingsView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/WorkoutTimerView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/WorkoutTimerView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/WorkoutsTabView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/WorkoutsTabView.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/WorkoutsTabViewList.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/WorkoutsTabViewList.png
--------------------------------------------------------------------------------
/Fit Vein/Onboarding Photos/WorkoutsView.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ljaniszewski00/Fit-Vein/d33c72da202ad617d8f4751ba9d638cea8da25fb/Fit Vein/Onboarding Photos/WorkoutsView.png
--------------------------------------------------------------------------------
/Fit Vein/OnboardingView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // OnboardingView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 24/10/2021.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct OnboardingView: View {
11 | @Environment(\.dismiss) var dismiss
12 | @Environment(\.colorScheme) var colorScheme
13 |
14 | private var imagesNames: [String] = ["", "HomeTabView", "PostCommentsView", "SearchFriendsView", "WorkoutsView", "AddWorkoutView", "WorkoutTimerView", "FinishedWorkoutView", "ProfileView", "HealthTabView", "WorkoutsTabView", "WorkoutsTabViewList", "SettingsView", "MedalView"]
15 |
16 | private var titles: [String] = [String(localized: "Onboarding_welcome_title"), String(localized: "Onboarding_home_tab_view_title"), String(localized: "Onboarding_post_comments_title"), String(localized: "Onboarding_search_friends_view_title"), String(localized: "Onboarding_workouts_view_title"), String(localized: "Onboarding_add_workout_view_title"), String(localized: "Onboarding_workout_timer_view_title"), String(localized: "Onboarding_finished_workout_view_title"), String(localized: "Onboarding_profile_view_title"), String(localized: "Onboarding_health_tab_view_title"), String(localized: "Onboarding_workouts_tab_view_title"), String(localized: "Onboarding_workouts_tab_view_list_title"), String(localized: "Onboarding_settings_view_title"), String(localized: "Onboarding_medal_view_title"), String(localized: "Onboarding_thanks_title")]
17 |
18 | private var descriptions: [String] = [String(localized: "Onboarding_welcome_description"), String(localized: "Onboarding_home_tab_view_description"), String(localized: "Onboarding_post_comments_description"), String(localized: "Onboarding_search_friends_view_description"), String(localized: "Onboarding_workouts_view_description"), String(localized: "Onboarding_add_workout_view_description"), String(localized: "Onboarding_workout_timer_view_description"), String(localized: "Onboarding_finished_workout_view_description"), String(localized: "Onboarding_profile_view_description"), String(localized: "Onboarding_health_tab_view_description"), String(localized: "Onboarding_workouts_tab_view_description"), String(localized: "Onboarding_workouts_tab_view_list_description"), String(localized: "Onboarding_settings_view_description"), String(localized: "Onboarding_medal_view_description"), String(localized: "Onboarding_thanks_description")]
19 |
20 | var body: some View {
21 | GeometryReader { geometry in
22 | let screenWidth = geometry.size.width
23 | let screenHeight = geometry.size.height
24 |
25 | VStack {
26 | ZStack() {
27 | TabView {
28 | ForEach(0...imagesNames.count, id: \.self) { number in
29 | VStack {
30 | Spacer()
31 |
32 | switch number {
33 | case 0:
34 | LottieView(name: "hello", loopMode: .loop, contentMode: .scaleAspectFill)
35 | .frame(width: screenWidth * 0.7, height: screenHeight * 0.35)
36 | case 14:
37 | LottieView(name: "baboonExercising", loopMode: .loop, contentMode: .scaleAspectFill)
38 | .frame(width: screenWidth)
39 | default:
40 | Image(uiImage: UIImage(named: imagesNames[number])!)
41 | .resizable()
42 | .scaledToFill()
43 | .frame(width: screenWidth * 0.5, height: screenHeight * 0.5)
44 | }
45 |
46 | Spacer()
47 |
48 | VStack(alignment: .center) {
49 | Text(titles[number])
50 | .font(.largeTitle)
51 | .bold()
52 | .padding(.bottom, screenHeight * 0.025)
53 |
54 | Text(descriptions[number])
55 | .font(.system(size: number == 0 ? screenHeight * 0.03 : screenHeight * 0.02))
56 | .padding(.bottom, screenHeight * 0.025)
57 |
58 | Spacer()
59 | }
60 | .padding()
61 | .frame(width: screenWidth, height: screenHeight * 0.3)
62 | .background(Color.accentColor)
63 | }
64 | }
65 | }
66 | .tabViewStyle(PageTabViewStyle())
67 | }
68 | HStack {
69 | Button(action: {
70 | withAnimation {
71 | dismiss()
72 | }
73 | }, label: {
74 | ZStack {
75 | RoundedRectangle(cornerRadius: 10, style: .continuous)
76 | .foregroundColor(Color(uiColor: UIColor(red: 180, green: 255, blue: 180)))
77 |
78 | HStack {
79 | Text(String(localized: "Onboarding_get_started_button"))
80 | .fontWeight(.bold)
81 | .foregroundColor(Color(uiColor: UIColor(red: 100, green: 215, blue: 100)))
82 | }
83 | .padding(.horizontal)
84 | .frame(height: screenHeight * 0.05)
85 | }
86 | .frame(width: screenWidth * 0.4, height: screenHeight * 0.06)
87 | })
88 | }
89 | .padding()
90 | .frame(width: screenWidth)
91 | .background(Color(uiColor: .systemGray6))
92 | }
93 | }
94 | }
95 | }
96 |
97 | struct OnboardingView_Previews: PreviewProvider {
98 | static var previews: some View {
99 | OnboardingView()
100 | }
101 | }
102 |
--------------------------------------------------------------------------------
/Fit Vein/PreLaunchView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // PreLaunchView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 03/01/2022.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct PreLaunchView: View {
11 | @EnvironmentObject var sessionStore: SessionStore
12 | @Environment(\.colorScheme) var colorScheme
13 | @State private var showContentView: Bool = false
14 |
15 | var body: some View {
16 | GeometryReader { geometry in
17 | let screenWidth = geometry.size.width
18 | let screenHeight = geometry.size.height
19 |
20 | Group {
21 | if showContentView {
22 | ContentView()
23 | .environmentObject(sessionStore)
24 | } else {
25 | VStack(alignment: .center) {
26 | Image(uiImage: UIImage(named: colorScheme == .dark ? "FitVeinIconDark" : "FitVeinIconLight")!)
27 | .resizable()
28 | .scaledToFill()
29 | .frame(width: screenWidth * 0.2, height: screenHeight * 0.2)
30 | .padding(.top, screenHeight * 0.15)
31 |
32 | HStack {
33 | Text("Fit")
34 | .foregroundColor(.accentColor)
35 | .fontWeight(.bold)
36 | Text("Vein")
37 | .fontWeight(.bold)
38 | }
39 | .font(.system(size: screenHeight * 0.1))
40 |
41 | LottieView(name: "dumbleLoading", loopMode: .playOnce, contentMode: .scaleAspectFill)
42 | .frame(width: screenWidth * 0.6, height: screenHeight * 0.3)
43 | .padding(.top, screenHeight * 0.05)
44 |
45 | Spacer()
46 | }
47 | .frame(width: screenWidth)
48 | }
49 | }
50 | .onAppear {
51 | withAnimation(.linear.delay(2.5)) {
52 | showContentView = true
53 | }
54 | }
55 |
56 | }
57 | }
58 | }
59 |
60 | struct PreLaunchView_Previews: PreviewProvider {
61 | static var previews: some View {
62 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
63 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
64 | PreLaunchView()
65 | .preferredColorScheme(colorScheme)
66 | .previewDevice(PreviewDevice(rawValue: deviceName))
67 | .previewDisplayName(deviceName)
68 | .environmentObject(SessionStore(forPreviews: true))
69 | }
70 | }
71 | }
72 | }
73 |
--------------------------------------------------------------------------------
/Fit Vein/Preview Content/Preview Assets.xcassets/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "info" : {
3 | "author" : "xcode",
4 | "version" : 1
5 | }
6 | }
7 |
--------------------------------------------------------------------------------
/Fit Vein/ViewModels/HealthKitViewModel.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HealthKitViewModel.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 24/11/2021.
6 | //
7 |
8 | import Foundation
9 | import SwiftUI
10 | import HealthKit
11 |
12 | @MainActor
13 | class HealthKitViewModel: ObservableObject {
14 | private var healthKitRepository = HealthKitRepository()
15 | @Published var stepCount = [HealthStat]()
16 | @Published var activeEnergyBurned = [HealthStat]()
17 | @Published var distanceWalkingRunning = [HealthStat]()
18 | @Published var appleExerciseTime = [HealthStat]()
19 | @Published var heartRate = [HealthStat]()
20 |
21 | init() {
22 | self.healthKitRepository.requestAuthorization() { success in
23 | self.healthKitRepository.requestHealthStats(by: "stepCount") { hStats in
24 | DispatchQueue.main.async {
25 | self.stepCount = hStats
26 | }
27 | }
28 |
29 | self.healthKitRepository.requestHealthStats(by: "activeEnergyBurned") { hStats in
30 | DispatchQueue.main.async {
31 | self.activeEnergyBurned = hStats
32 | }
33 | }
34 |
35 | self.healthKitRepository.requestHealthStats(by: "distanceWalkingRunning") { hStats in
36 | DispatchQueue.main.async {
37 | self.distanceWalkingRunning = hStats
38 | }
39 | }
40 |
41 | self.healthKitRepository.requestHealthStats(by: "appleExerciseTime") { hStats in
42 | DispatchQueue.main.async {
43 | self.appleExerciseTime = hStats
44 | }
45 | }
46 |
47 | self.healthKitRepository.requestHealthStats(by: "heartRate") { hStats in
48 | DispatchQueue.main.async {
49 | self.heartRate = hStats
50 | }
51 | }
52 | }
53 | }
54 |
55 | let measurementFormatter = MeasurementFormatter()
56 |
57 | func value(from stat: HKQuantity?) -> (value: Float, units: String) {
58 | guard let stat = stat else {
59 | return (0, "-")
60 | }
61 |
62 | measurementFormatter.unitStyle = .short
63 |
64 | if stat.is(compatibleWith: .kilocalorie()) {
65 | let value = stat.doubleValue(for: .kilocalorie())
66 | return(Float(value), stat.description.letters)
67 | } else if stat.is(compatibleWith: .meter()) {
68 | let value = stat.doubleValue(for: .mile())
69 | let unit = Measurement(value: value, unit: UnitLength.miles)
70 | return (Float(round(value * 100) / 100.0), measurementFormatter.string(from: unit).letters)
71 | } else if stat.is(compatibleWith: .count()) {
72 | let value = stat.doubleValue(for: .count())
73 | return (Float(value), stat.description.letters == "count" ? "" : stat.description.letters)
74 | } else if stat.is(compatibleWith: .minute()) {
75 | let value = stat.doubleValue(for: .minute())
76 | return (Float(value), stat.description.letters)
77 | } else {
78 | let value = stat.doubleValue(for: HKUnit(from: "count/min"))
79 | return (Float(value), "BPM")
80 | }
81 | }
82 | }
83 |
--------------------------------------------------------------------------------
/Fit Vein/ViewModels/MedalsViewModel.swift:
--------------------------------------------------------------------------------
1 | //
2 | // MedalsViewModel.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 29/01/2022.
6 | //
7 |
8 | import Foundation
9 |
10 | @MainActor
11 | class MedalsViewModel: ObservableObject {
12 | var sessionStore = SessionStore(forPreviews: false)
13 | private let firestoreManager = FirestoreManager()
14 |
15 | private(set) var allMedalsDescriptions: [String: String] = [
16 | "medalFirstLevel": String(localized: "ProfileView_medal_first_level"),
17 | "medalSecondLevel": String(localized: "ProfileView_medal_second_level"),
18 | "medalThirdLevel": String(localized: "ProfileView_medal_third_level"),
19 | "medalFirstPost": String(localized: "ProfileView_medal_first_post"),
20 | "medalFirstComment": String(localized: "ProfileView_medal_first_comment"),
21 | "medalFirstLike": String(localized: "ProfileView_medal_first_like"),
22 | "medalFirstWorkout": String(localized: "ProfileView_medal_first_workout"),
23 | "medalFirstOwnWorkout": String(localized: "ProfileView_medal_first_own_workout")]
24 |
25 | @Published var allUsersMedals: [String] = []
26 |
27 | init() {
28 | fetchUsersMedals()
29 | }
30 |
31 | func fetchUsersMedals() {
32 | if let currentUser = sessionStore.currentUser {
33 | firestoreManager.fetchDataForMedalsViewModel(userID: currentUser.uid) { medals in
34 | self.allUsersMedals = medals.sorted(by: <)
35 | }
36 | }
37 | }
38 |
39 | func giveUserMedal(medalName: String) {
40 | if let currentUser = sessionStore.currentUser {
41 | if !allUsersMedals.contains(medalName) {
42 | firestoreManager.giveUserMedal(userID: currentUser.uid, medalName: medalName) { success in }
43 | }
44 | }
45 | }
46 | }
47 |
--------------------------------------------------------------------------------
/Fit Vein/ViewModels/SignInViewModel.swift:
--------------------------------------------------------------------------------
1 | //
2 | // SignInViewModel.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 13/10/2021.
6 | //
7 |
8 | import Foundation
9 |
10 | @MainActor
11 | class SignInViewModel: ObservableObject {
12 | var sessionStore = SessionStore(forPreviews: false)
13 | private let firestoreManager = FirestoreManager()
14 | private let firebaseStorageManager = FirebaseStorageManager()
15 |
16 | func signIn(email: String, password: String, completion: @escaping ((Bool) -> ())) {
17 | self.sessionStore.signIn(email: email, password: password) { success in
18 | completion(success)
19 | }
20 | }
21 |
22 | func sendRecoveryEmail(email: String, completion: @escaping ((Bool) -> ())) {
23 | self.sessionStore.sendRecoveryEmail(email: email) { success in
24 | completion(success)
25 | }
26 | }
27 | }
28 |
--------------------------------------------------------------------------------
/Fit Vein/ViewModels/SignUpViewModel.swift:
--------------------------------------------------------------------------------
1 | //
2 | // SignUpViewModel.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 13/10/2021.
6 | //
7 |
8 | import Foundation
9 | import UIKit
10 | import SwiftUI
11 |
12 | @MainActor
13 | class SignUpViewModel: ObservableObject {
14 | var sessionStore = SessionStore(forPreviews: false)
15 | private let firestoreManager = FirestoreManager()
16 | private let firebaseStorageManager = FirebaseStorageManager()
17 |
18 | func signUp(firstName: String, userName: String, birthDate: Date, country: String, language: String, email: String, password: String, gender: String, completion: @escaping ((Bool) -> ())) {
19 | self.sessionStore.signUp(email: email, password: password) { (userID, success) in
20 | if success {
21 | if let userID = userID {
22 | self.firestoreManager.signUpDataCreation(id: userID, firstName: firstName, username: userName, birthDate: birthDate, country: country, language: language, email: email, gender: gender) { profile, success in
23 | completion(success)
24 | }
25 | } else {
26 | completion(false)
27 | }
28 | } else {
29 | completion(false)
30 | }
31 | }
32 | }
33 |
34 | func checkUsernameDuplicate(username: String) async throws -> Bool {
35 | try await firestoreManager.checkUsernameDuplicate(username: username)
36 | }
37 |
38 | func checkEmailDuplicate(email: String) async throws -> Bool {
39 | try await firestoreManager.checkEmailDuplicate(email: email)
40 | }
41 | }
42 |
--------------------------------------------------------------------------------
/Fit Vein/ViewModels/WorkoutViewModel.swift:
--------------------------------------------------------------------------------
1 | //
2 | // WorkoutViewModel.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 20/10/2021.
6 | //
7 |
8 | import Foundation
9 | import SwiftUI
10 |
11 | @MainActor
12 | class WorkoutViewModel: ObservableObject {
13 | var sessionStore = SessionStore(forPreviews: false)
14 | @Published var workout: IntervalWorkout?
15 | @Published var workoutsList: [IntervalWorkout] = [IntervalWorkout(id: UUID().uuidString, usersID: "9999", type: "Interval", date: Date(), isFinished: false, calories: 200, series: 8, workTime: 45, restTime: 15),
16 | IntervalWorkout(id: UUID().uuidString, usersID: "9999", type: "Interval", date: Date(), isFinished: false, calories: 260, series: 10, workTime: 30, restTime: 15),
17 | IntervalWorkout(id: UUID().uuidString, usersID: "9999", type: "Interval", date: Date(), isFinished: false, calories: 140, series: 5, workTime: 60, restTime: 30),
18 | IntervalWorkout(id: UUID().uuidString, usersID: "9999", type: "Interval", date: Date(), isFinished: false, calories: 110, series: 7, workTime: 45, restTime: 20),
19 | IntervalWorkout(id: UUID().uuidString, usersID: "9999", type: "Interval", date: Date(), isFinished: false, calories: 260, series: 8, workTime: 45, restTime: 20)]
20 | @AppStorage("usersWorkoutsList") var usersWorkoutsList: [IntervalWorkout] = []
21 | private let firestoreManager = FirestoreManager()
22 | private let firebaseStorageManager = FirebaseStorageManager()
23 |
24 | init(forPreviews: Bool) {
25 | if forPreviews {
26 | self.workout = IntervalWorkout(id: UUID().uuidString, usersID: "9999", type: "Interval", date: Date(), isFinished: false, calories: 0, series: 8, workTime: 45, restTime: 15)
27 | }
28 | }
29 |
30 | func setup(sessionStore: SessionStore) {
31 | self.sessionStore = sessionStore
32 | }
33 |
34 | func startWorkout(workout: IntervalWorkout) {
35 | self.workout = workout
36 | }
37 |
38 | func stopWorkout(calories: Int, completedDuration: Int, completedSeries: Int) {
39 | self.workout?.setDataOnEnd(calories: calories, completedDuration: completedDuration, completedSeries: completedSeries)
40 | }
41 |
42 | func saveWorkoutToDatabase(completion: @escaping ((Bool) -> ())) {
43 | if self.workout != nil {
44 | if self.sessionStore.currentUser != nil {
45 | self.workout!.setUsersID(usersID: self.sessionStore.currentUser!.uid)
46 | self.firestoreManager.workoutDataCreation(id: workout!.id, usersID: workout!.usersID, type: workout!.type, date: workout!.date, isFinished: workout!.isFinished, calories: workout!.calories, series: workout!.series, workTime: workout!.workTime, restTime: workout!.restTime, completedDuration: workout!.completedDuration, completedSeries: workout!.completedSeries) { success in
47 | if success {
48 | print("Successfully saved workout to the database.")
49 | completion(true)
50 | } else {
51 | print("Error saving workout to the database.")
52 | completion(false)
53 | }
54 | }
55 | } else {
56 | completion(false)
57 | print("Error saving workout to the database: current user is nil")
58 | }
59 | }
60 | }
61 |
62 | func addUserWorkout(series: Int, workTime: Int, restTime: Int) {
63 | self.usersWorkoutsList.append(IntervalWorkout(id: UUID().uuidString, usersID: "9999", type: "Interval", date: Date(), isFinished: false, calories: 200, series: series, workTime: workTime, restTime: restTime))
64 | }
65 |
66 | func deleteUserWorkout(indexSet: IndexSet) {
67 | self.usersWorkoutsList.remove(atOffsets: indexSet)
68 | }
69 | }
70 |
71 |
72 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HelpViews/CustomTextField.swift:
--------------------------------------------------------------------------------
1 | //
2 | // CustomTextField.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 11/01/2022.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct CustomTextField: View {
11 | var isSecureField: Bool = false
12 | var textFieldProperty: String
13 | var textFieldImageName: String
14 | var textFieldSignsLimit: Int = 0
15 | @Binding var text: String
16 | @Binding var isFocusedParentView: Bool
17 | @FocusState private var isFocused: Bool
18 |
19 | var body: some View {
20 | VStack {
21 | VStack(alignment: .leading, spacing: 4) {
22 | HStack(spacing: 15) {
23 | Group {
24 | if !isSecureField {
25 | TextField("", text: $text)
26 | .onTapGesture {
27 | withAnimation(.easeIn) {
28 | isFocusedParentView = true
29 | isFocused = true
30 | }
31 | }
32 | .onSubmit {
33 | withAnimation(.easeOut) {
34 | isFocused = false
35 | isFocusedParentView = false
36 | }
37 | }
38 | } else {
39 | SecureField("", text: $text)
40 | .onTapGesture {
41 | withAnimation(.easeIn) {
42 | isFocusedParentView = true
43 | isFocused = true
44 | }
45 | }
46 | .onSubmit {
47 | withAnimation(.easeOut) {
48 | isFocused = false
49 | isFocusedParentView = false
50 | }
51 | }
52 | }
53 | }
54 | .focused($isFocused)
55 | .disableAutocorrection(true)
56 | .autocapitalization(.none)
57 |
58 | Image(systemName: textFieldImageName)
59 | .foregroundColor(isFocused ? .accentColor : .gray)
60 | }
61 | .padding(.top, isFocused ? 15 : 0)
62 | .background(
63 | Text(textFieldProperty)
64 | .scaleEffect(text.isEmpty ? (isFocused ? 0.8 : 1) : (isFocused ? 0.8 : 0.8))
65 | .offset(x: text.isEmpty ? (isFocused ? -7 : 0) : -7, y: text.isEmpty ? (isFocused ? -20 : 0) : -20)
66 | .foregroundColor(isFocused ? .accentColor : .gray)
67 |
68 | ,alignment: .leading
69 | )
70 | .padding(.horizontal)
71 |
72 | Rectangle()
73 | .fill(isFocused ? Color.accentColor : Color.gray)
74 | .opacity(isFocused ? 1 : 0.5)
75 | .frame(height: 1)
76 | }
77 | .padding(.vertical, isFocused ? 18 : (text.isEmpty ? 12 : 20))
78 | .background(Color.gray.opacity(0.09))
79 | .cornerRadius(5)
80 |
81 | HStack {
82 | Spacer()
83 |
84 | if textFieldSignsLimit != 0 {
85 | Text("\(text.count)/\(textFieldSignsLimit)")
86 | .font(.caption)
87 | .foregroundColor(text.count > textFieldSignsLimit ? .red : .gray)
88 | .padding(.trailing)
89 | .padding(.top, 4)
90 | }
91 | }
92 | }
93 | .padding()
94 | }
95 | }
96 |
97 | struct CustomTextField_Previews: PreviewProvider {
98 | static var previews: some View {
99 | PreviewWrapper()
100 | }
101 |
102 | struct PreviewWrapper: View {
103 | @State(initialValue: "") var text: String
104 | @State(initialValue: false) var isFocused: Bool
105 |
106 | var body: some View {
107 | CustomTextField(textFieldProperty: "Username", textFieldImageName: "", textFieldSignsLimit: 15, text: $text, isFocusedParentView: $isFocused)
108 | }
109 | }
110 | }
111 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HelpViews/ProfileTabFetchingView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // ProfileTabFetchingView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 23/11/2021.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct ProfileTabFetchingView: View {
11 | var body: some View {
12 | GeometryReader { geometry in
13 | let screenWidth = geometry.size.width
14 | let screenHeight = geometry.size.height
15 |
16 | VStack {
17 | HStack {
18 | RoundedRectangle(cornerRadius: 50)
19 | .foregroundColor(Color(uiColor: .systemGray6))
20 | .frame(width: screenWidth * 0.4, height: screenHeight * 0.2)
21 |
22 | VStack {
23 | HStack {
24 | RoundedRectangle(cornerRadius: 25)
25 | .foregroundColor(Color(uiColor: .systemGray6))
26 | .frame(width: screenWidth * 0.4, height: screenHeight * 0.03)
27 |
28 | Spacer()
29 | }
30 | .padding(.top, screenHeight * 0.02)
31 |
32 | HStack {
33 | RoundedRectangle(cornerRadius: 25)
34 | .foregroundColor(Color(uiColor: .systemGray6))
35 | .frame(width: screenWidth * 0.4, height: screenHeight * 0.03)
36 |
37 | Spacer()
38 | }
39 |
40 | Spacer()
41 | }
42 | .frame(width: screenWidth * 0.5, height: screenHeight * 0.2)
43 | }
44 | .padding()
45 |
46 | VStack(spacing: screenHeight * 0.04) {
47 | RoundedRectangle(cornerRadius: 25)
48 | .foregroundColor(Color(uiColor: .systemGray6))
49 | .frame(width: screenWidth * 0.9, height: screenHeight * 0.03)
50 |
51 | RoundedRectangle(cornerRadius: 25)
52 | .foregroundColor(Color(uiColor: .systemGray6))
53 | .frame(width: screenWidth * 0.35, height: screenHeight * 0.02)
54 |
55 | HStack {
56 | Spacer()
57 |
58 | RoundedRectangle(cornerRadius: 25)
59 | .foregroundColor(Color(uiColor: .systemGray6))
60 | .frame(width: screenWidth * 0.3, height: screenHeight * 0.03)
61 |
62 | Spacer()
63 |
64 | RoundedRectangle(cornerRadius: 25)
65 | .foregroundColor(Color(uiColor: .systemGray6))
66 | .frame(width: screenWidth * 0.3, height: screenHeight * 0.03)
67 |
68 | Spacer()
69 | }
70 | .padding()
71 |
72 | HStack {
73 | RoundedRectangle(cornerRadius: 25)
74 | .foregroundColor(Color(uiColor: .systemGray6))
75 | .frame(width: screenWidth * 0.5, height: screenHeight * 0.05)
76 |
77 | Spacer()
78 | }
79 |
80 | LazyVGrid(columns: [GridItem(.flexible()),
81 | GridItem(.flexible())], spacing: 0) {
82 | ForEach(0..<5) { _ in
83 | RoundedRectangle(cornerRadius: 25)
84 | .frame(height: screenHeight * 0.2)
85 | .foregroundColor(Color(uiColor: .systemGray6))
86 | .padding()
87 | }
88 | }
89 | }
90 | .padding()
91 | }
92 |
93 | }
94 | }
95 | }
96 |
97 | struct ProfileTabFetchingView_Previews: PreviewProvider {
98 | static var previews: some View {
99 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
100 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
101 |
102 | ProfileTabFetchingView()
103 | .preferredColorScheme(colorScheme)
104 | .previewDevice(PreviewDevice(rawValue: deviceName))
105 | .previewDisplayName(deviceName)
106 | }
107 | }
108 | }
109 | }
110 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HelpViews/RingProgressViewStyle.swift:
--------------------------------------------------------------------------------
1 | //
2 | // RingProgressViewStyle.swift
3 | // Fit Vein
4 | //
5 | // Created by SwiftUI Recipes on 12/08/2021.
6 | // https://swiftuirecipes.com
7 | // https://github.com/globulus/swiftui-circle-progress-bar
8 | //
9 |
10 | import SwiftUI
11 |
12 | public struct RingProgressViewStyle: ProgressViewStyle {
13 | private let defaultSize: CGFloat = 36
14 | private let lineWidth: CGFloat = 6
15 | private let defaultProgress = 0.2 // CHANGE
16 |
17 | // tracks the rotation angle for the indefinite progress bar
18 | @State private var fillRotationAngle = Angle.degrees(-90) // ADD
19 |
20 | public func makeBody(configuration: ProgressViewStyleConfiguration) -> some View {
21 | VStack {
22 | configuration.label
23 | progressCircleView(fractionCompleted: configuration.fractionCompleted ?? defaultProgress,
24 | isIndefinite: configuration.fractionCompleted == nil) // UPDATE
25 | configuration.currentValueLabel
26 | }
27 | }
28 |
29 | private func progressCircleView(fractionCompleted: Double,
30 | isIndefinite: Bool) -> some View { // UPDATE
31 | // this is the circular "track", which is a full circle at all times
32 | Circle()
33 | .strokeBorder(Color.gray.opacity(0.5), lineWidth: lineWidth, antialiased: true)
34 | .overlay(fillView(fractionCompleted: fractionCompleted, isIndefinite: isIndefinite)) // UPDATE
35 | .frame(width: defaultSize, height: defaultSize)
36 | }
37 |
38 | private func fillView(fractionCompleted: Double,
39 | isIndefinite: Bool) -> some View { // UPDATE
40 | Circle() // the fill view is also a circle
41 | .trim(from: 0, to: CGFloat(fractionCompleted))
42 | .stroke(Color.secondary, lineWidth: lineWidth)
43 | .frame(width: defaultSize - lineWidth, height: defaultSize - lineWidth)
44 | .rotationEffect(fillRotationAngle) // UPDATE
45 | // triggers the infinite rotation animation for indefinite progress views
46 | .onAppear {
47 | if isIndefinite {
48 | withAnimation(.easeInOut(duration: 1).repeatForever()) {
49 | fillRotationAngle = .degrees(270)
50 | }
51 | }
52 | }
53 | }
54 | }
55 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HomeTabViews/CommentsViews/HomeTabCommentsViewPostView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HomeTabCommentsViewPostView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 13/01/2022.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct HomeTabCommentsViewPostView: View {
11 | @EnvironmentObject private var homeViewModel: HomeViewModel
12 | @EnvironmentObject private var profileViewModel: ProfileViewModel
13 | @EnvironmentObject private var medalsViewModel: MedalsViewModel
14 | @Environment(\.colorScheme) var colorScheme
15 |
16 | private var post: Post
17 |
18 | init(post: Post) {
19 | self.post = post
20 | }
21 |
22 | var body: some View {
23 | let screenWidth = UIScreen.screenWidth
24 | let screenHeight = UIScreen.screenHeight
25 |
26 | VStack {
27 | Text(post.text)
28 | .fixedSize(horizontal: false, vertical: true)
29 | .padding()
30 | .padding(.top)
31 |
32 | if let post = self.homeViewModel.getCurrentPostDetails(postID: post.id) {
33 | if let postPhotoURL = post.photoURL {
34 | Group {
35 | if let postPictureURL = self.homeViewModel.postsPicturesURLs[post.id] {
36 | AsyncImage(url: postPictureURL) { phase in
37 | if let image = phase.image {
38 | image
39 | .resizable()
40 | .scaledToFill()
41 | } else {
42 | ProgressView()
43 | }
44 | }
45 | }
46 | }
47 | }
48 | }
49 |
50 | HStack {
51 | if post.reactionsUsersIDs != nil {
52 | if post.reactionsUsersIDs!.count != 0 {
53 | Image(systemName: post.reactionsUsersIDs!.contains(self.profileViewModel.profile!.id) ? "hand.thumbsup.fill" : "hand.thumbsup")
54 | .foregroundColor(.accentColor)
55 | .padding(.leading, screenWidth * 0.05)
56 |
57 | Text("\(post.reactionsUsersIDs!.count)")
58 | .foregroundColor(Color(uiColor: .systemGray2))
59 | }
60 |
61 | }
62 |
63 | Spacer()
64 |
65 | if let postComments = homeViewModel.postsComments[post.id] {
66 | Text("\(postComments.count) \(String(localized: "CommentView_comment_number_label"))")
67 | .foregroundColor(Color(uiColor: .systemGray2))
68 | .padding(.trailing, screenWidth * 0.05)
69 | }
70 | }
71 | .padding(.top, screenHeight * 0.02)
72 |
73 | HStack {
74 | Spacer()
75 |
76 | if let reactionsUsersIDs = profileViewModel.profile!.reactedPostsIDs {
77 | Button(action: {
78 | withAnimation {
79 | if reactionsUsersIDs.contains(post.id) {
80 | self.homeViewModel.removeReactionFromPost(postID: post.id) { success in }
81 | } else {
82 | self.homeViewModel.reactToPost(postID: post.id) { success in
83 | self.medalsViewModel.giveUserMedal(medalName: "medalFirstLike")
84 | }
85 | }
86 | }
87 | }, label: {
88 | HStack {
89 | Image(systemName: "hand.thumbsup")
90 | .symbolVariant(reactionsUsersIDs.contains(post.id) ? .fill : .none)
91 | Text(String(localized: "CommentView_post_like_button"))
92 | }
93 | .foregroundColor(reactionsUsersIDs.contains(post.id) ? .accentColor : (colorScheme == .dark ? .white : .black))
94 | })
95 | } else {
96 | Button(action: {
97 | withAnimation {
98 | self.homeViewModel.reactToPost(postID: post.id) { success in
99 | self.medalsViewModel.giveUserMedal(medalName: "medalFirstLike")
100 | }
101 | }
102 | }, label: {
103 | HStack {
104 | Image(systemName: "hand.thumbsup")
105 | Text(String(localized: "CommentView_post_like_button"))
106 | }
107 | .foregroundColor(colorScheme == .dark ? .white : .black)
108 | })
109 | }
110 |
111 | Spacer()
112 | }
113 | .padding(screenHeight * 0.014)
114 | .background(.ultraThinMaterial, in: RoundedRectangle(cornerRadius: 10))
115 |
116 | Divider()
117 | }
118 | }
119 | }
120 |
121 | struct HomeTabCommentsViewPostView_Previews: PreviewProvider {
122 | static var previews: some View {
123 | let homeViewModel = HomeViewModel(forPreviews: true)
124 | let profileViewModel = ProfileViewModel(forPreviews: true)
125 | let comments = [Comment(id: "id1", authorID: "1", postID: "1", authorFirstName: "Maciej", authorUsername: "maciej.j223", authorProfilePictureURL: "nil", addDate: Date(), text: "Good job!", reactionsUsersIDs: ["2", "3"]), Comment(id: "id2", authorID: "3", postID: "1", authorFirstName: "Kamil", authorUsername: "kamil.j223", authorProfilePictureURL: "nil", addDate: Date(), text: "Let's Go!", reactionsUsersIDs: ["1", "3"])]
126 | let post = Post(id: "1", authorID: "1", authorFirstName: "Jan", authorUsername: "jan23.d", authorProfilePictureURL: "", addDate: Date(), text: "Did this today!", reactionsUsersIDs: nil, commentedUsersIDs: nil, comments: comments)
127 |
128 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
129 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
130 | NavigationView {
131 | HomeTabCommentsViewPostView(post: post)
132 | .environmentObject(homeViewModel)
133 | .environmentObject(profileViewModel)
134 | .preferredColorScheme(colorScheme)
135 | .previewDevice(PreviewDevice(rawValue: deviceName))
136 | .previewDisplayName(deviceName)
137 | }
138 | }
139 | }
140 | }
141 | }
142 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HomeTabViews/HomeView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HomeView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 20/10/2021.
6 | //
7 |
8 |
9 | import SwiftUI
10 |
11 | struct HomeView: View {
12 | @EnvironmentObject private var homeViewModel: HomeViewModel
13 | @EnvironmentObject private var profileViewModel: ProfileViewModel
14 | @EnvironmentObject private var medalsViewModel: MedalsViewModel
15 | @EnvironmentObject private var networkManager: NetworkManager
16 | @Environment(\.colorScheme) var colorScheme
17 |
18 | var body: some View {
19 | let screenWidth = UIScreen.screenWidth
20 | let screenHeight = UIScreen.screenHeight
21 |
22 | NavigationView {
23 | Group {
24 | if !networkManager.isConnected {
25 | VStack {
26 | Spacer()
27 | HStack {
28 | Spacer()
29 | LottieView(name: "noInternetConnection", loopMode: .loop)
30 | .frame(width: screenWidth * 0.7, height: screenHeight * 0.7)
31 | Spacer()
32 | }
33 | Spacer()
34 | }
35 | } else {
36 | if profileViewModel.profile != nil {
37 | withAnimation {
38 | ScrollView(.vertical, showsIndicators: false) {
39 | HomeTabSubViewShareView().environmentObject(homeViewModel).environmentObject(profileViewModel).environmentObject(medalsViewModel)
40 |
41 | Group {
42 | if let posts = homeViewModel.posts {
43 | if posts.count != 0 {
44 | LazyVStack {
45 | ForEach(posts) { post in
46 | HomeTabSubViewPostsView(post: post).environmentObject(homeViewModel).environmentObject(profileViewModel)
47 | .environmentObject(medalsViewModel)
48 | .background(Color(uiColor: .systemGray6), in: RoundedRectangle(cornerRadius: 10))
49 | }
50 | }
51 | .padding(.bottom, screenHeight * 0.07)
52 | } else {
53 | VStack {
54 | Text(String(localized: "HomeView_nothing_to_present"))
55 | .fontWeight(.bold)
56 | .foregroundColor(.accentColor)
57 | .padding(.top, screenHeight * 0.07)
58 |
59 | Spacer()
60 | }
61 | }
62 | } else {
63 | if let followedIDs = self.profileViewModel.profile!.followedIDs {
64 | if followedIDs.count != 0 {
65 | // HomeTabPostsFetchingView()
66 | // .frame(width: screenWidth, height: screenHeight)
67 | VStack {
68 | Text(String(localized: "HomeView_nothing_to_present"))
69 | .fontWeight(.bold)
70 | .foregroundColor(.accentColor)
71 | .padding(.top, screenHeight * 0.07)
72 |
73 | Spacer()
74 | }
75 | } else {
76 | VStack {
77 | Text(String(localized: "HomeView_add_friends_label"))
78 | .fontWeight(.bold)
79 | .foregroundColor(.accentColor)
80 | .padding(.top, screenHeight * 0.07)
81 |
82 | Spacer()
83 | }
84 | }
85 | } else {
86 | VStack {
87 | Text(String(localized: "HomeView_add_friends_label"))
88 | .fontWeight(.bold)
89 | .foregroundColor(.accentColor)
90 | .padding(.top, screenHeight * 0.07)
91 |
92 | Spacer()
93 | }
94 | }
95 | }
96 | }
97 | }
98 | .padding(.top, screenHeight * 0.001)
99 | }
100 | }
101 | }
102 | }
103 | .navigationTitle("")
104 | .navigationBarTitleDisplayMode(.inline)
105 | .toolbar {
106 | ToolbarItem(placement: .navigationBarLeading) {
107 | Image(uiImage: UIImage(named: colorScheme == .dark ? "FitVeinIconDark" : "FitVeinIconLight")!)
108 | .resizable()
109 | .scaledToFit()
110 | .frame(width: screenWidth * 0.1, height: screenHeight * 0.1)
111 | }
112 |
113 | ToolbarItem(placement: .navigationBarTrailing) {
114 | NavigationLink(destination: SearchFriendsView().environmentObject(homeViewModel).environmentObject(profileViewModel).ignoresSafeArea(.keyboard)) {
115 | Image(systemName: "magnifyingglass")
116 | .foregroundColor(.accentColor)
117 | }
118 | .disabled(!networkManager.isConnected)
119 | }
120 |
121 | ToolbarItem(placement: .navigationBarTrailing) {
122 | NavigationLink(destination: NotificationsView()) {
123 | Image(systemName: "bell")
124 | .foregroundColor(.accentColor)
125 | }
126 | .disabled(!networkManager.isConnected)
127 | }
128 | }
129 | }
130 | .navigationViewStyle(.stack)
131 | }
132 | }
133 |
134 | struct HomeView_Previews: PreviewProvider {
135 | static var previews: some View {
136 | let homeViewModel = HomeViewModel(forPreviews: true)
137 | let profileViewModel = ProfileViewModel(forPreviews: true)
138 |
139 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
140 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
141 | HomeView()
142 | .environmentObject(homeViewModel)
143 | .environmentObject(profileViewModel)
144 | .preferredColorScheme(colorScheme)
145 | .previewDevice(PreviewDevice(rawValue: deviceName))
146 | .previewDisplayName(deviceName)
147 | }
148 | }
149 | }
150 | }
151 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HomeTabViews/NotificationsView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // NotificationsView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 28/10/2021.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct NotificationsView: View {
11 | var body: some View {
12 | GeometryReader { geometry in
13 | let screenWidth = geometry.size.width
14 | let screenHeight = geometry.size.height
15 |
16 | NavigationView {
17 | VStack {
18 | HStack {
19 | Spacer()
20 |
21 | LottieView(name: "notifications", loopMode: .loop)
22 | .frame(width: screenWidth * 0.25, height: screenHeight * 0.15)
23 | }
24 | .padding()
25 | .padding(.trailing, screenWidth * 0.05)
26 |
27 | HStack {
28 | Spacer()
29 | LottieView(name: "skeleton", loopMode: .loop)
30 | .frame(width: screenWidth, height: screenHeight * 0.5)
31 | Spacer()
32 | }
33 | Spacer()
34 | }
35 | .navigationTitle(String(localized: "NotificationsView_navigation_title"))
36 | }
37 | .navigationViewStyle(.stack)
38 | }
39 | }
40 | }
41 |
42 | struct NotificationsView_Previews: PreviewProvider {
43 | static var previews: some View {
44 | NotificationsView()
45 | }
46 | }
47 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HomeTabViews/PostsViews/HomeTabSubViewPostDetailsView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HomeTabSubViewPostDetailsView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 08/01/2022.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct HomeTabSubViewPostDetailsView: View {
11 | @EnvironmentObject private var homeViewModel: HomeViewModel
12 | @EnvironmentObject private var profileViewModel: ProfileViewModel
13 |
14 | @State private var showPostOptions = false
15 | @State private var showEditPostSheet = false
16 |
17 | private var currentUserID: String
18 | private var post: Post
19 |
20 | init(currentUserID: String, post: Post) {
21 | self.currentUserID = currentUserID
22 | self.post = post
23 | }
24 |
25 | var body: some View {
26 | let screenWidth = UIScreen.screenWidth
27 | let screenHeight = UIScreen.screenHeight
28 |
29 | VStack {
30 | HStack {
31 | Group {
32 | if let profilePictureURL = self.homeViewModel.postsAuthorsProfilePicturesURLs[post.id] {
33 | AsyncImage(url: profilePictureURL) { phase in
34 | if let image = phase.image {
35 | image
36 | .resizable()
37 | } else {
38 | Image(uiImage: UIImage(named: "blank-profile-hi")!)
39 | .resizable()
40 | }
41 | }
42 | } else {
43 | Image(uiImage: UIImage(named: "blank-profile-hi")!)
44 | .resizable()
45 | }
46 | }
47 | .aspectRatio(contentMode: .fill)
48 | .clipShape(RoundedRectangle(cornerRadius: 50))
49 | .frame(width: screenWidth * 0.06, height: screenHeight * 0.06)
50 | .padding(.horizontal)
51 |
52 | VStack(spacing: screenHeight * 0.005) {
53 | HStack {
54 | Text(post.authorFirstName)
55 | .fontWeight(.bold)
56 | Text("•")
57 | Text(post.authorUsername)
58 | Spacer()
59 |
60 | if currentUserID == post.authorID {
61 | Button(action: {
62 | withAnimation {
63 | showPostOptions.toggle()
64 | }
65 | }, label: {
66 | Image(systemName: "ellipsis")
67 | .foregroundColor(.accentColor)
68 | })
69 |
70 | }
71 | }
72 |
73 | HStack {
74 | Text(getShortDate(longDate: post.addDate))
75 | .font(.system(size: screenHeight * 0.016))
76 | .foregroundColor(Color(uiColor: .systemGray2))
77 | Spacer()
78 | }
79 | }
80 | .padding(.horizontal)
81 | }
82 | .padding()
83 |
84 | Text(post.text)
85 | .fixedSize(horizontal: false, vertical: true)
86 | .padding(.horizontal)
87 | .padding(.bottom)
88 |
89 | if let post = self.homeViewModel.getCurrentPostDetails(postID: post.id) {
90 | if let postPhotoURL = post.photoURL {
91 | Group {
92 | if let postPictureURL = self.homeViewModel.postsPicturesURLs[post.id] {
93 | AsyncImage(url: postPictureURL) { phase in
94 | if let image = phase.image {
95 | image
96 | .resizable()
97 | .scaledToFill()
98 | } else {
99 | ProgressView()
100 | }
101 | }
102 | }
103 | }
104 | .padding(.bottom)
105 | }
106 | }
107 | }
108 | .confirmationDialog(String(localized: "HomeView_confirmation_dialog_text"), isPresented: $showPostOptions, titleVisibility: .visible) {
109 | Button(String(localized: "HomeView_confirmation_dialog_edit")) {
110 | showEditPostSheet.toggle()
111 | }
112 |
113 | Button(String(localized: "HomeView_confirmation_dialog_delete"), role: .destructive) {
114 | self.homeViewModel.deletePost(postID: post.id, postPictureURL: post.photoURL) { success in }
115 | }
116 |
117 | Button(String(localized: "HomeView_confirmation_dialog_cancel"), role: .cancel) {}
118 | }
119 | .sheet(isPresented: $showEditPostSheet) {
120 | EditPostView(post: post).environmentObject(homeViewModel).environmentObject(profileViewModel).ignoresSafeArea(.keyboard)
121 | }
122 | }
123 | }
124 |
125 | struct HomeTabSubViewPostDetailsView_Previews: PreviewProvider {
126 | static var previews: some View {
127 | let homeViewModel = HomeViewModel(forPreviews: true)
128 | let profileViewModel = ProfileViewModel(forPreviews: true)
129 | let comments = [Comment(id: "id1", authorID: "1", postID: "1", authorFirstName: "Maciej", authorUsername: "maciej.j223", authorProfilePictureURL: "nil", addDate: Date(), text: "Good job!", reactionsUsersIDs: ["2", "3"]), Comment(id: "id2", authorID: "3", postID: "1", authorFirstName: "Kamil", authorUsername: "kamil.j223", authorProfilePictureURL: "nil", addDate: Date(), text: "Let's Go!", reactionsUsersIDs: ["1", "3"])]
130 | let post = Post(id: "1", authorID: "1", authorFirstName: "Jan", authorUsername: "jan23.d", authorProfilePictureURL: "", addDate: Date(), text: "Did this today!", reactionsUsersIDs: nil, commentedUsersIDs: nil, comments: comments)
131 |
132 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
133 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
134 | HomeTabSubViewPostDetailsView(currentUserID: "id1", post: post)
135 | .environmentObject(homeViewModel)
136 | .environmentObject(profileViewModel)
137 | .preferredColorScheme(colorScheme)
138 | .previewDevice(PreviewDevice(rawValue: deviceName))
139 | .previewDisplayName(deviceName)
140 | }
141 | }
142 | }
143 | }
144 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HomeTabViews/PostsViews/HomeTabSubViewPostsView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HomeTabSubViewPostsView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 08/01/2022.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct HomeTabSubViewPostsView: View {
11 | @EnvironmentObject private var homeViewModel: HomeViewModel
12 | @EnvironmentObject private var profileViewModel: ProfileViewModel
13 | @EnvironmentObject private var medalsViewModel: MedalsViewModel
14 |
15 | @Environment(\.colorScheme) var colorScheme
16 |
17 | private var post: Post
18 |
19 | init(post: Post) {
20 | self.post = post
21 | }
22 |
23 | var body: some View {
24 | let screenWidth = UIScreen.screenWidth
25 | let screenHeight = UIScreen.screenHeight
26 |
27 | VStack {
28 | VStack {
29 | HomeTabSubViewPostDetailsView(currentUserID: self.profileViewModel.profile!.id, post: post)
30 | .environmentObject(homeViewModel).environmentObject(profileViewModel).environmentObject(medalsViewModel)
31 | }
32 |
33 | VStack {
34 | HStack {
35 | if post.reactionsUsersIDs != nil {
36 | if post.reactionsUsersIDs!.count != 0 {
37 | Image(systemName: post.reactionsUsersIDs!.contains(self.profileViewModel.profile!.id) ? "hand.thumbsup.fill" : "hand.thumbsup")
38 | .foregroundColor(.accentColor)
39 | .padding(.leading, screenWidth * 0.05)
40 |
41 | Text("\(post.reactionsUsersIDs!.count)")
42 | .foregroundColor(Color(uiColor: .systemGray2))
43 | }
44 |
45 | }
46 |
47 | Spacer()
48 |
49 | if let postComments = homeViewModel.postsComments[post.id] {
50 | Text("\(postComments.count) \(String(localized: "HomeView_post_comments_number"))")
51 | .foregroundColor(Color(uiColor: .systemGray2))
52 | .padding(.trailing, screenWidth * 0.05)
53 | }
54 | }
55 | }
56 |
57 | VStack {
58 | HStack(spacing: 0) {
59 | Spacer()
60 |
61 | if let reactionsUsersIDs = profileViewModel.profile!.reactedPostsIDs {
62 | Button(action: {
63 | withAnimation {
64 | if reactionsUsersIDs.contains(post.id) {
65 | self.homeViewModel.removeReactionFromPost(postID: post.id) { success in }
66 | } else {
67 | self.homeViewModel.reactToPost(postID: post.id) { success in
68 | self.medalsViewModel.giveUserMedal(medalName: "medalFirstLike")
69 | }
70 | }
71 | }
72 | }, label: {
73 | HStack {
74 | Image(systemName: "hand.thumbsup")
75 | .symbolVariant(reactionsUsersIDs.contains(post.id) ? .fill : .none)
76 | Text(String(localized: "HomeView_like_button"))
77 | }
78 | .foregroundColor(reactionsUsersIDs.contains(post.id) ? .green : (colorScheme == .dark ? .white : .black))
79 | })
80 | } else {
81 | Button(action: {
82 | withAnimation {
83 | self.homeViewModel.reactToPost(postID: post.id) { success in
84 | self.medalsViewModel.giveUserMedal(medalName: "medalFirstLike")
85 | }
86 | }
87 | }, label: {
88 | HStack {
89 | Image(systemName: "hand.thumbsup")
90 | Text(String(localized: "HomeView_like_button"))
91 | }
92 | .foregroundColor(colorScheme == .dark ? .white : .black)
93 | })
94 | }
95 |
96 | Spacer()
97 |
98 | Divider()
99 |
100 | Spacer()
101 |
102 | NavigationLink(destination: PostCommentsView(post: post).environmentObject(homeViewModel).environmentObject(profileViewModel).environmentObject(medalsViewModel).ignoresSafeArea(.keyboard)) {
103 | HStack {
104 | Image(systemName: "bubble.left")
105 | Text(String(localized: "HomeView_comment_button"))
106 | }
107 | .foregroundColor(colorScheme == .dark ? .white : .black)
108 | }
109 | NavigationLink(destination: EmptyView()) {
110 | EmptyView()
111 | }
112 |
113 | Spacer()
114 | }
115 | }
116 | .padding(.vertical, screenHeight * 0.015)
117 | .background(.ultraThinMaterial, in: RoundedRectangle(cornerRadius: 10))
118 | }
119 | }
120 | }
121 |
122 | struct HomeTabSubViewPostsView_Previews: PreviewProvider {
123 | static var previews: some View {
124 | let homeViewModel = HomeViewModel(forPreviews: true)
125 | let profileViewModel = ProfileViewModel(forPreviews: true)
126 |
127 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
128 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
129 | HomeTabSubViewPostsView(post: homeViewModel.posts![0])
130 | .environmentObject(homeViewModel)
131 | .environmentObject(profileViewModel)
132 | .preferredColorScheme(colorScheme)
133 | .previewDevice(PreviewDevice(rawValue: deviceName))
134 | .previewDisplayName(deviceName)
135 | }
136 | }
137 | }
138 | }
139 |
--------------------------------------------------------------------------------
/Fit Vein/Views/HomeTabViews/PostsViews/HomeTabSubViewShareView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // HomeTabSubViewShareView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 08/01/2022.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct HomeTabSubViewShareView: View {
11 | @EnvironmentObject private var homeViewModel: HomeViewModel
12 | @EnvironmentObject private var profileViewModel: ProfileViewModel
13 | @EnvironmentObject private var medalsViewModel: MedalsViewModel
14 |
15 | @State private var showAddPostSheet = false
16 |
17 | var body: some View {
18 | let screenWidth = UIScreen.screenWidth
19 | let screenHeight = UIScreen.screenHeight
20 |
21 | VStack {
22 | HStack {
23 | Group {
24 | if let profilePicturePhoto = profileViewModel.profilePicturePhoto {
25 | Image(uiImage: profilePicturePhoto)
26 | .resizable()
27 | } else {
28 | Image(uiImage: UIImage(named: "blank-profile-hi")!)
29 | .resizable()
30 | }
31 | }
32 | .aspectRatio(contentMode: .fill)
33 | .clipShape(RoundedRectangle(cornerRadius: 50))
34 | .frame(width: screenWidth * 0.05, height: screenHeight * 0.05)
35 | .padding(.trailing, screenWidth * 0.04)
36 |
37 | Text(String(localized: "HomeView_share"))
38 | .foregroundColor(Color(uiColor: .systemGray))
39 | .font(.system(size: screenHeight * 0.02))
40 |
41 | Spacer()
42 | }
43 | .padding()
44 | .padding(.leading)
45 | .onTapGesture {
46 | withAnimation {
47 | showAddPostSheet.toggle()
48 | }
49 | }
50 |
51 | Divider()
52 |
53 | Text(String(localized: "HomeView_friends_activity"))
54 | .foregroundColor(.accentColor)
55 | .font(.system(size: screenHeight * 0.03, weight: .bold))
56 | .padding()
57 | .background(Rectangle().foregroundColor(Color(uiColor: .systemGray6)).frame(width: screenWidth))
58 |
59 | }
60 | .sheet(isPresented: $showAddPostSheet) {
61 | AddPostView().environmentObject(homeViewModel).environmentObject(profileViewModel).environmentObject(medalsViewModel).ignoresSafeArea(.keyboard)
62 | }
63 | }
64 | }
65 |
66 | struct HomeTabSubViewShareView_Previews: PreviewProvider {
67 | static var previews: some View {
68 | let homeViewModel = HomeViewModel(forPreviews: true)
69 | let profileViewModel = ProfileViewModel(forPreviews: true)
70 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
71 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
72 | HomeTabSubViewShareView()
73 | .environmentObject(homeViewModel)
74 | .environmentObject(profileViewModel)
75 | .previewDevice(PreviewDevice(rawValue: deviceName))
76 | .previewDisplayName(deviceName)
77 | }
78 | }
79 | }
80 | }
81 |
--------------------------------------------------------------------------------
/Fit Vein/Views/ProfileTabViews/WorkoutTabView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // WorkoutTabView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 25/10/2021.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct WorkoutTabView: View {
11 | @EnvironmentObject private var profileViewModel: ProfileViewModel
12 | @State private var howToDisplay = 0
13 |
14 | var body: some View {
15 | GeometryReader { geometry in
16 |
17 | NavigationView {
18 | Group {
19 | if howToDisplay == 0 {
20 | WorkoutTabViewWindows()
21 | } else {
22 | WorkoutTabViewList()
23 | }
24 | }
25 | .toolbar {
26 | ToolbarItem(placement: .navigationBarTrailing) {
27 | Picker("", selection: $howToDisplay) {
28 | Image(systemName: "squareshape.split.2x2").tag(0)
29 | Image(systemName: "list.bullet").tag(1)
30 | }
31 | .pickerStyle(SegmentedPickerStyle())
32 | }
33 | }
34 | }
35 | .navigationViewStyle(.stack)
36 | }
37 | }
38 | }
39 |
40 | struct WorkoutTabViewWindows: View {
41 | @EnvironmentObject private var profileViewModel: ProfileViewModel
42 |
43 | var body: some View {
44 | GeometryReader { geometry in
45 | let screenWidth = geometry.size.width
46 | let screenHeight = geometry.size.height
47 |
48 | TabView {
49 | ForEach(profileViewModel.workouts!) { workout in
50 | SingleWorkoutWindowView(workout: workout)
51 | .frame(width: screenWidth, height: screenHeight)
52 | }
53 | }
54 | .tabViewStyle(.page)
55 | .navigationTitle(String(localized: "WorkoutTabView_windows_navigation_title"))
56 | .navigationBarHidden(false)
57 | }
58 | }
59 | }
60 |
61 | struct WorkoutTabViewList: View {
62 | @EnvironmentObject var profileViewModel: ProfileViewModel
63 |
64 | var body: some View {
65 | GeometryReader { geometry in
66 | let screenWidth = geometry.size.width
67 | let screenHeight = geometry.size.height
68 |
69 | List(profileViewModel.workouts!) { workout in
70 | NavigationLink(destination: SingleWorkoutWindowView(workout: workout)
71 | .navigationTitle(String(localized: "WorkoutTabView_list_single_workout_navigation_title"))
72 | .navigationBarHidden(false)
73 | .frame(width: screenWidth, height: screenHeight))
74 | {
75 | HStack {
76 | Image(uiImage: UIImage(named: "sprint2")!)
77 | .resizable()
78 | .scaledToFit()
79 | .frame(width: screenWidth * 0.1, height: screenHeight * 0.1)
80 | .padding(.trailing)
81 |
82 | VStack {
83 | Text(String(localized: "WorkoutTabView_list_workout_type_name"))
84 | .font(.title3)
85 | .fontWeight(.bold)
86 | Text(getShortDate(longDate: workout.date))
87 | }
88 | }
89 | }
90 | }
91 | .navigationTitle(String(localized: "WorkoutTabView_list_navigation_title"))
92 | .navigationBarHidden(false)
93 | }
94 | }
95 | }
96 |
97 | struct WorkoutTabView_Previews: PreviewProvider {
98 | static var previews: some View {
99 | let profileViewModel = ProfileViewModel(forPreviews: true)
100 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
101 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
102 |
103 | WorkoutTabView()
104 | .environmentObject(profileViewModel)
105 | .preferredColorScheme(colorScheme)
106 | .previewDevice(PreviewDevice(rawValue: deviceName))
107 | .previewDisplayName(deviceName)
108 |
109 | WorkoutTabViewWindows()
110 | .environmentObject(profileViewModel)
111 | .preferredColorScheme(colorScheme)
112 | .previewDevice(PreviewDevice(rawValue: deviceName))
113 | .previewDisplayName(deviceName)
114 |
115 | WorkoutTabViewList()
116 | .environmentObject(profileViewModel)
117 | .preferredColorScheme(colorScheme)
118 | .previewDevice(PreviewDevice(rawValue: deviceName))
119 | .previewDisplayName(deviceName)
120 | }
121 | }
122 | }
123 | }
124 |
--------------------------------------------------------------------------------
/Fit Vein/Views/WorkoutTabViews/FinishedWorkoutView.swift:
--------------------------------------------------------------------------------
1 | //
2 | // FinishedWorkoutView.swift
3 | // Fit Vein
4 | //
5 | // Created by Łukasz Janiszewski on 26/10/2021.
6 | //
7 |
8 | import SwiftUI
9 |
10 | struct FinishedWorkoutView: View {
11 | @EnvironmentObject private var workoutViewModel: WorkoutViewModel
12 | @EnvironmentObject private var medalsViewModel: MedalsViewModel
13 | @EnvironmentObject private var networkManager: NetworkManager
14 | @Environment(\.dismiss) var dismiss
15 | @State private var backToBeginning = false
16 |
17 | @State private var error = false
18 |
19 | var body: some View {
20 | GeometryReader { geometry in
21 | let screenWidth = geometry.size.width
22 | let screenHeight = geometry.size.height
23 |
24 | if backToBeginning || workoutViewModel.workout == nil {
25 | withAnimation(.linear) {
26 | WorkoutView()
27 | .environmentObject(workoutViewModel)
28 | .environmentObject(medalsViewModel)
29 | .environmentObject(networkManager)
30 | }
31 | } else {
32 | VStack(spacing: screenHeight * 0.05) {
33 | SingleWorkoutWindowView(workout: workoutViewModel.workout!)
34 | .frame(height: screenHeight * 0.75)
35 |
36 | if error {
37 | HStack {
38 | LottieView(name: "wrongData", loopMode: .loop, contentMode: .scaleAspectFill)
39 | .frame(width: screenWidth * 0.15, height: screenHeight * 0.05)
40 | .padding(.leading)
41 | .offset(y: -screenHeight * 0.013)
42 | Text(String(localized: "FinisehdWorkoutView_saving_error"))
43 | .foregroundColor(.red)
44 | .font(.system(size: screenWidth * 0.035, weight: .bold))
45 | Spacer()
46 | }
47 | .padding(.horizontal)
48 | .offset(y: -screenHeight * 0.05)
49 | }
50 |
51 | HStack {
52 | Spacer()
53 |
54 | Button(action: {
55 | error = false
56 | self.workoutViewModel.saveWorkoutToDatabase() { success in
57 | withAnimation {
58 | if success {
59 | medalsViewModel.giveUserMedal(medalName: "medalFirstWorkout")
60 | backToBeginning = true
61 | } else {
62 | error = true
63 | }
64 | }
65 | }
66 | }, label: {
67 | Text(String(localized: "FinishedWorkoutView_save_button"))
68 | .fontWeight(.bold)
69 | .foregroundColor(Color(uiColor: .systemGray5))
70 | })
71 | .background(RoundedRectangle(cornerRadius: 25).frame(width: screenWidth * 0.35, height: screenHeight * 0.07).foregroundColor(.accentColor))
72 | .padding()
73 | .disabled(!networkManager.isConnected)
74 |
75 | Spacer()
76 |
77 | Button(action: {
78 | withAnimation(.linear) {
79 | backToBeginning = true
80 | }
81 | }, label: {
82 | Text(String(localized: "FinishedWorkoutView_discard_button"))
83 | .foregroundColor(Color(uiColor: .systemGray5))
84 | })
85 | .background(RoundedRectangle(cornerRadius: 25).frame(width: screenWidth * 0.35, height: screenHeight * 0.07).foregroundColor(.red))
86 | .padding()
87 |
88 | Spacer()
89 | }
90 | .padding(.bottom, screenHeight * 0.05)
91 | }
92 | .onDisappear {
93 | dismiss()
94 | }
95 | }
96 | }
97 | }
98 | }
99 |
100 | struct FinishedWorkoutView_Previews: PreviewProvider {
101 | static var previews: some View {
102 | let sessionStore = SessionStore(forPreviews: true)
103 | let workoutViewModel = WorkoutViewModel(forPreviews: true)
104 | let networkManager = NetworkManager()
105 | ForEach(ColorScheme.allCases, id: \.self) { colorScheme in
106 | ForEach(["iPhone XS MAX", "iPhone 8"], id: \.self) { deviceName in
107 | FinishedWorkoutView()
108 | .preferredColorScheme(colorScheme)
109 | .previewDevice(PreviewDevice(rawValue: deviceName))
110 | .previewDisplayName(deviceName)
111 | .environmentObject(sessionStore)
112 | .environmentObject(workoutViewModel)
113 | .environmentObject(networkManager)
114 | }
115 | }
116 | }
117 | }
118 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | Creative Commons Legal Code
2 |
3 | CC0 1.0 Universal
4 |
5 | CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE
6 | LEGAL SERVICES. DISTRIBUTION OF THIS DOCUMENT DOES NOT CREATE AN
7 | ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS
8 | INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES
9 | REGARDING THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS
10 | PROVIDED HEREUNDER, AND DISCLAIMS LIABILITY FOR DAMAGES RESULTING FROM
11 | THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS PROVIDED
12 | HEREUNDER.
13 |
14 | Statement of Purpose
15 |
16 | The laws of most jurisdictions throughout the world automatically confer
17 | exclusive Copyright and Related Rights (defined below) upon the creator
18 | and subsequent owner(s) (each and all, an "owner") of an original work of
19 | authorship and/or a database (each, a "Work").
20 |
21 | Certain owners wish to permanently relinquish those rights to a Work for
22 | the purpose of contributing to a commons of creative, cultural and
23 | scientific works ("Commons") that the public can reliably and without fear
24 | of later claims of infringement build upon, modify, incorporate in other
25 | works, reuse and redistribute as freely as possible in any form whatsoever
26 | and for any purposes, including without limitation commercial purposes.
27 | These owners may contribute to the Commons to promote the ideal of a free
28 | culture and the further production of creative, cultural and scientific
29 | works, or to gain reputation or greater distribution for their Work in
30 | part through the use and efforts of others.
31 |
32 | For these and/or other purposes and motivations, and without any
33 | expectation of additional consideration or compensation, the person
34 | associating CC0 with a Work (the "Affirmer"), to the extent that he or she
35 | is an owner of Copyright and Related Rights in the Work, voluntarily
36 | elects to apply CC0 to the Work and publicly distribute the Work under its
37 | terms, with knowledge of his or her Copyright and Related Rights in the
38 | Work and the meaning and intended legal effect of CC0 on those rights.
39 |
40 | 1. Copyright and Related Rights. A Work made available under CC0 may be
41 | protected by copyright and related or neighboring rights ("Copyright and
42 | Related Rights"). Copyright and Related Rights include, but are not
43 | limited to, the following:
44 |
45 | i. the right to reproduce, adapt, distribute, perform, display,
46 | communicate, and translate a Work;
47 | ii. moral rights retained by the original author(s) and/or performer(s);
48 | iii. publicity and privacy rights pertaining to a person's image or
49 | likeness depicted in a Work;
50 | iv. rights protecting against unfair competition in regards to a Work,
51 | subject to the limitations in paragraph 4(a), below;
52 | v. rights protecting the extraction, dissemination, use and reuse of data
53 | in a Work;
54 | vi. database rights (such as those arising under Directive 96/9/EC of the
55 | European Parliament and of the Council of 11 March 1996 on the legal
56 | protection of databases, and under any national implementation
57 | thereof, including any amended or successor version of such
58 | directive); and
59 | vii. other similar, equivalent or corresponding rights throughout the
60 | world based on applicable law or treaty, and any national
61 | implementations thereof.
62 |
63 | 2. Waiver. To the greatest extent permitted by, but not in contravention
64 | of, applicable law, Affirmer hereby overtly, fully, permanently,
65 | irrevocably and unconditionally waives, abandons, and surrenders all of
66 | Affirmer's Copyright and Related Rights and associated claims and causes
67 | of action, whether now known or unknown (including existing as well as
68 | future claims and causes of action), in the Work (i) in all territories
69 | worldwide, (ii) for the maximum duration provided by applicable law or
70 | treaty (including future time extensions), (iii) in any current or future
71 | medium and for any number of copies, and (iv) for any purpose whatsoever,
72 | including without limitation commercial, advertising or promotional
73 | purposes (the "Waiver"). Affirmer makes the Waiver for the benefit of each
74 | member of the public at large and to the detriment of Affirmer's heirs and
75 | successors, fully intending that such Waiver shall not be subject to
76 | revocation, rescission, cancellation, termination, or any other legal or
77 | equitable action to disrupt the quiet enjoyment of the Work by the public
78 | as contemplated by Affirmer's express Statement of Purpose.
79 |
80 | 3. Public License Fallback. Should any part of the Waiver for any reason
81 | be judged legally invalid or ineffective under applicable law, then the
82 | Waiver shall be preserved to the maximum extent permitted taking into
83 | account Affirmer's express Statement of Purpose. In addition, to the
84 | extent the Waiver is so judged Affirmer hereby grants to each affected
85 | person a royalty-free, non transferable, non sublicensable, non exclusive,
86 | irrevocable and unconditional license to exercise Affirmer's Copyright and
87 | Related Rights in the Work (i) in all territories worldwide, (ii) for the
88 | maximum duration provided by applicable law or treaty (including future
89 | time extensions), (iii) in any current or future medium and for any number
90 | of copies, and (iv) for any purpose whatsoever, including without
91 | limitation commercial, advertising or promotional purposes (the
92 | "License"). The License shall be deemed effective as of the date CC0 was
93 | applied by Affirmer to the Work. Should any part of the License for any
94 | reason be judged legally invalid or ineffective under applicable law, such
95 | partial invalidity or ineffectiveness shall not invalidate the remainder
96 | of the License, and in such case Affirmer hereby affirms that he or she
97 | will not (i) exercise any of his or her remaining Copyright and Related
98 | Rights in the Work or (ii) assert any associated claims and causes of
99 | action with respect to the Work, in either case contrary to Affirmer's
100 | express Statement of Purpose.
101 |
102 | 4. Limitations and Disclaimers.
103 |
104 | a. No trademark or patent rights held by Affirmer are waived, abandoned,
105 | surrendered, licensed or otherwise affected by this document.
106 | b. Affirmer offers the Work as-is and makes no representations or
107 | warranties of any kind concerning the Work, express, implied,
108 | statutory or otherwise, including without limitation warranties of
109 | title, merchantability, fitness for a particular purpose, non
110 | infringement, or the absence of latent or other defects, accuracy, or
111 | the present or absence of errors, whether or not discoverable, all to
112 | the greatest extent permissible under applicable law.
113 | c. Affirmer disclaims responsibility for clearing rights of other persons
114 | that may apply to the Work or any use thereof, including without
115 | limitation any person's Copyright and Related Rights in the Work.
116 | Further, Affirmer disclaims responsibility for obtaining any necessary
117 | consents, permissions or other rights required for any use of the
118 | Work.
119 | d. Affirmer understands and acknowledges that Creative Commons is not a
120 | party to this document and has no duty or obligation with respect to
121 | this CC0 or use of the Work.
122 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Fit-Vein
2 |
3 | ## Table of Content
4 | * [General Info](#general-info)
5 | * [Technologies](#technologies)
6 | * [Status](#status)
7 | * [Requirements](#requirements)
8 | * [Functionalities](#functionalities)
9 | * [Screenshots](#screenshots)
10 | * [Preview Video](#preview)
11 |
12 |
13 | ## General info
14 | iOS fitness app in which user can perform various interval workouts, share their activity with people followed by creating posts, adding reactions and commenting others activity. What's more, it is possible to monitor health data delivered from the HealthKit and see already completed workouts.
15 |
16 |
17 | ## Technologies
18 | Swift 5.5 [SwiftUI + UIKit]
19 | Xcode 13
20 | Firebase, Firebase Firestore, Firebase Storage
21 | HealthKit
22 | Lottie Library
23 |
24 |
25 | ## Status
26 | Finished
27 |
28 |
29 | ## Requirements
30 | Apple iPhone with iOS 15+ installed
31 |
32 |
33 | ## Functionalities
34 | * Home Tab in which users can share recent workouts by creating posts, add photos to them, write whatever they want and react by liking and commenting posts of other users
35 | * The possibility of performing interval training during which they have time, series, calories etc. counter
36 | * Users can make use of use pre-created workouts or create new, custom ones
37 | * Up-to-date HealthKit data in the Profile tab where users can see their daily burned calories, distance traveled and average heartRate measured by iPhone or Apple Watch
38 | * Achieving new levels along with increasing number of completed workouts, new levels bring new medals
39 | * The possibility of changing user's data including e-mail address, password, signing out and deleting user's account
40 | * Biometric security (FaceID and TouchID)
41 | * Special offline mode where all network functions are turned off. Still it is possible to do workouts and see HealthKit data
42 | * During workout we are allowed to minimize the app or lock the screen and workouts progress will still be calculated in background and displayed properly updated after coming back to app
43 | * Full English and Polish localization
44 | * Light and dark color themes
45 | * Full onboarding, executing at first app's launch, taking the user on a tour across all of the app's most vital functionalities.
46 |
47 |
48 | ## Screenshots
49 |
50 | 
51 |
52 | 
53 | 
54 | 
55 | 
56 | 
57 | 
58 | 
59 | 
60 | 
61 |
62 |
63 | ## Preview Video
64 | ### Polish UI Localization
65 | https://user-images.githubusercontent.com/78795431/173231961-d1ca756b-2498-467a-8709-29c1f9e4cfd6.mov
66 |
--------------------------------------------------------------------------------