Skip to main content

Add the basic SwiftUI app

ID
3e6e144
date
2023-06-08 06:34:53+00:00
author
Alex Chan <alex@alexwlchan.net>
parent
2eb8250
message
Add the basic SwiftUI app
changed files
13 files, 877 additions

Changed files

BlinkReviewer/BlinkReviewer.xcodeproj/project.pbxproj (0) → BlinkReviewer/BlinkReviewer.xcodeproj/project.pbxproj (22546)

diff --git a/BlinkReviewer/BlinkReviewer.xcodeproj/project.pbxproj b/BlinkReviewer/BlinkReviewer.xcodeproj/project.pbxproj
new file mode 100644
index 0000000..9403922
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer.xcodeproj/project.pbxproj
@@ -0,0 +1,585 @@
+// !$*UTF8*$!
+{
+	archiveVersion = 1;
+	classes = {
+	};
+	objectVersion = 56;
+	objects = {
+
+/* Begin PBXBuildFile section */
+		94D750F02A31A796005859E7 /* BlinkReviewerApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D750EF2A31A796005859E7 /* BlinkReviewerApp.swift */; };
+		94D750F22A31A796005859E7 /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D750F12A31A796005859E7 /* ContentView.swift */; };
+		94D750F42A31A797005859E7 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 94D750F32A31A797005859E7 /* Assets.xcassets */; };
+		94D750F72A31A797005859E7 /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 94D750F62A31A797005859E7 /* Preview Assets.xcassets */; };
+		94D751022A31A798005859E7 /* BlinkReviewerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D751012A31A798005859E7 /* BlinkReviewerTests.swift */; };
+		94D7510C2A31A798005859E7 /* BlinkReviewerUITests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D7510B2A31A798005859E7 /* BlinkReviewerUITests.swift */; };
+		94D7510E2A31A798005859E7 /* BlinkReviewerUITestsLaunchTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D7510D2A31A798005859E7 /* BlinkReviewerUITestsLaunchTests.swift */; };
+		94D7511C2A31A7B1005859E7 /* ThumbnailItem.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D7511B2A31A7B1005859E7 /* ThumbnailItem.swift */; };
+/* End PBXBuildFile section */
+
+/* Begin PBXContainerItemProxy section */
+		94D750FE2A31A798005859E7 /* PBXContainerItemProxy */ = {
+			isa = PBXContainerItemProxy;
+			containerPortal = 94D750E42A31A796005859E7 /* Project object */;
+			proxyType = 1;
+			remoteGlobalIDString = 94D750EB2A31A796005859E7;
+			remoteInfo = BlinkReviewer;
+		};
+		94D751082A31A798005859E7 /* PBXContainerItemProxy */ = {
+			isa = PBXContainerItemProxy;
+			containerPortal = 94D750E42A31A796005859E7 /* Project object */;
+			proxyType = 1;
+			remoteGlobalIDString = 94D750EB2A31A796005859E7;
+			remoteInfo = BlinkReviewer;
+		};
+/* End PBXContainerItemProxy section */
+
+/* Begin PBXFileReference section */
+		94D750EC2A31A796005859E7 /* BlinkReviewer.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = BlinkReviewer.app; sourceTree = BUILT_PRODUCTS_DIR; };
+		94D750EF2A31A796005859E7 /* BlinkReviewerApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlinkReviewerApp.swift; sourceTree = "<group>"; };
+		94D750F12A31A796005859E7 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = "<group>"; };
+		94D750F32A31A797005859E7 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; };
+		94D750F62A31A797005859E7 /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = "<group>"; };
+		94D750F82A31A797005859E7 /* BlinkReviewer.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = BlinkReviewer.entitlements; sourceTree = "<group>"; };
+		94D750FD2A31A798005859E7 /* BlinkReviewerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = BlinkReviewerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
+		94D751012A31A798005859E7 /* BlinkReviewerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlinkReviewerTests.swift; sourceTree = "<group>"; };
+		94D751072A31A798005859E7 /* BlinkReviewerUITests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = BlinkReviewerUITests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
+		94D7510B2A31A798005859E7 /* BlinkReviewerUITests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlinkReviewerUITests.swift; sourceTree = "<group>"; };
+		94D7510D2A31A798005859E7 /* BlinkReviewerUITestsLaunchTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlinkReviewerUITestsLaunchTests.swift; sourceTree = "<group>"; };
+		94D7511B2A31A7B1005859E7 /* ThumbnailItem.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ThumbnailItem.swift; sourceTree = "<group>"; };
+/* End PBXFileReference section */
+
+/* Begin PBXFrameworksBuildPhase section */
+		94D750E92A31A796005859E7 /* Frameworks */ = {
+			isa = PBXFrameworksBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+		94D750FA2A31A798005859E7 /* Frameworks */ = {
+			isa = PBXFrameworksBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+		94D751042A31A798005859E7 /* Frameworks */ = {
+			isa = PBXFrameworksBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+/* End PBXFrameworksBuildPhase section */
+
+/* Begin PBXGroup section */
+		94D750E32A31A796005859E7 = {
+			isa = PBXGroup;
+			children = (
+				94D750EE2A31A796005859E7 /* BlinkReviewer */,
+				94D751002A31A798005859E7 /* BlinkReviewerTests */,
+				94D7510A2A31A798005859E7 /* BlinkReviewerUITests */,
+				94D750ED2A31A796005859E7 /* Products */,
+			);
+			sourceTree = "<group>";
+		};
+		94D750ED2A31A796005859E7 /* Products */ = {
+			isa = PBXGroup;
+			children = (
+				94D750EC2A31A796005859E7 /* BlinkReviewer.app */,
+				94D750FD2A31A798005859E7 /* BlinkReviewerTests.xctest */,
+				94D751072A31A798005859E7 /* BlinkReviewerUITests.xctest */,
+			);
+			name = Products;
+			sourceTree = "<group>";
+		};
+		94D750EE2A31A796005859E7 /* BlinkReviewer */ = {
+			isa = PBXGroup;
+			children = (
+				94D7511A2A31A7A6005859E7 /* Views */,
+				94D750EF2A31A796005859E7 /* BlinkReviewerApp.swift */,
+				94D750F12A31A796005859E7 /* ContentView.swift */,
+				94D750F32A31A797005859E7 /* Assets.xcassets */,
+				94D750F82A31A797005859E7 /* BlinkReviewer.entitlements */,
+				94D750F52A31A797005859E7 /* Preview Content */,
+			);
+			path = BlinkReviewer;
+			sourceTree = "<group>";
+		};
+		94D750F52A31A797005859E7 /* Preview Content */ = {
+			isa = PBXGroup;
+			children = (
+				94D750F62A31A797005859E7 /* Preview Assets.xcassets */,
+			);
+			path = "Preview Content";
+			sourceTree = "<group>";
+		};
+		94D751002A31A798005859E7 /* BlinkReviewerTests */ = {
+			isa = PBXGroup;
+			children = (
+				94D751012A31A798005859E7 /* BlinkReviewerTests.swift */,
+			);
+			path = BlinkReviewerTests;
+			sourceTree = "<group>";
+		};
+		94D7510A2A31A798005859E7 /* BlinkReviewerUITests */ = {
+			isa = PBXGroup;
+			children = (
+				94D7510B2A31A798005859E7 /* BlinkReviewerUITests.swift */,
+				94D7510D2A31A798005859E7 /* BlinkReviewerUITestsLaunchTests.swift */,
+			);
+			path = BlinkReviewerUITests;
+			sourceTree = "<group>";
+		};
+		94D7511A2A31A7A6005859E7 /* Views */ = {
+			isa = PBXGroup;
+			children = (
+				94D7511B2A31A7B1005859E7 /* ThumbnailItem.swift */,
+			);
+			path = Views;
+			sourceTree = "<group>";
+		};
+/* End PBXGroup section */
+
+/* Begin PBXNativeTarget section */
+		94D750EB2A31A796005859E7 /* BlinkReviewer */ = {
+			isa = PBXNativeTarget;
+			buildConfigurationList = 94D751112A31A798005859E7 /* Build configuration list for PBXNativeTarget "BlinkReviewer" */;
+			buildPhases = (
+				94D750E82A31A796005859E7 /* Sources */,
+				94D750E92A31A796005859E7 /* Frameworks */,
+				94D750EA2A31A796005859E7 /* Resources */,
+			);
+			buildRules = (
+			);
+			dependencies = (
+			);
+			name = BlinkReviewer;
+			productName = BlinkReviewer;
+			productReference = 94D750EC2A31A796005859E7 /* BlinkReviewer.app */;
+			productType = "com.apple.product-type.application";
+		};
+		94D750FC2A31A798005859E7 /* BlinkReviewerTests */ = {
+			isa = PBXNativeTarget;
+			buildConfigurationList = 94D751142A31A798005859E7 /* Build configuration list for PBXNativeTarget "BlinkReviewerTests" */;
+			buildPhases = (
+				94D750F92A31A798005859E7 /* Sources */,
+				94D750FA2A31A798005859E7 /* Frameworks */,
+				94D750FB2A31A798005859E7 /* Resources */,
+			);
+			buildRules = (
+			);
+			dependencies = (
+				94D750FF2A31A798005859E7 /* PBXTargetDependency */,
+			);
+			name = BlinkReviewerTests;
+			productName = BlinkReviewerTests;
+			productReference = 94D750FD2A31A798005859E7 /* BlinkReviewerTests.xctest */;
+			productType = "com.apple.product-type.bundle.unit-test";
+		};
+		94D751062A31A798005859E7 /* BlinkReviewerUITests */ = {
+			isa = PBXNativeTarget;
+			buildConfigurationList = 94D751172A31A798005859E7 /* Build configuration list for PBXNativeTarget "BlinkReviewerUITests" */;
+			buildPhases = (
+				94D751032A31A798005859E7 /* Sources */,
+				94D751042A31A798005859E7 /* Frameworks */,
+				94D751052A31A798005859E7 /* Resources */,
+			);
+			buildRules = (
+			);
+			dependencies = (
+				94D751092A31A798005859E7 /* PBXTargetDependency */,
+			);
+			name = BlinkReviewerUITests;
+			productName = BlinkReviewerUITests;
+			productReference = 94D751072A31A798005859E7 /* BlinkReviewerUITests.xctest */;
+			productType = "com.apple.product-type.bundle.ui-testing";
+		};
+/* End PBXNativeTarget section */
+
+/* Begin PBXProject section */
+		94D750E42A31A796005859E7 /* Project object */ = {
+			isa = PBXProject;
+			attributes = {
+				BuildIndependentTargetsInParallel = 1;
+				LastSwiftUpdateCheck = 1430;
+				LastUpgradeCheck = 1430;
+				TargetAttributes = {
+					94D750EB2A31A796005859E7 = {
+						CreatedOnToolsVersion = 14.3.1;
+					};
+					94D750FC2A31A798005859E7 = {
+						CreatedOnToolsVersion = 14.3.1;
+						TestTargetID = 94D750EB2A31A796005859E7;
+					};
+					94D751062A31A798005859E7 = {
+						CreatedOnToolsVersion = 14.3.1;
+						TestTargetID = 94D750EB2A31A796005859E7;
+					};
+				};
+			};
+			buildConfigurationList = 94D750E72A31A796005859E7 /* Build configuration list for PBXProject "BlinkReviewer" */;
+			compatibilityVersion = "Xcode 14.0";
+			developmentRegion = en;
+			hasScannedForEncodings = 0;
+			knownRegions = (
+				en,
+				Base,
+			);
+			mainGroup = 94D750E32A31A796005859E7;
+			productRefGroup = 94D750ED2A31A796005859E7 /* Products */;
+			projectDirPath = "";
+			projectRoot = "";
+			targets = (
+				94D750EB2A31A796005859E7 /* BlinkReviewer */,
+				94D750FC2A31A798005859E7 /* BlinkReviewerTests */,
+				94D751062A31A798005859E7 /* BlinkReviewerUITests */,
+			);
+		};
+/* End PBXProject section */
+
+/* Begin PBXResourcesBuildPhase section */
+		94D750EA2A31A796005859E7 /* Resources */ = {
+			isa = PBXResourcesBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+				94D750F72A31A797005859E7 /* Preview Assets.xcassets in Resources */,
+				94D750F42A31A797005859E7 /* Assets.xcassets in Resources */,
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+		94D750FB2A31A798005859E7 /* Resources */ = {
+			isa = PBXResourcesBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+		94D751052A31A798005859E7 /* Resources */ = {
+			isa = PBXResourcesBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+/* End PBXResourcesBuildPhase section */
+
+/* Begin PBXSourcesBuildPhase section */
+		94D750E82A31A796005859E7 /* Sources */ = {
+			isa = PBXSourcesBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+				94D7511C2A31A7B1005859E7 /* ThumbnailItem.swift in Sources */,
+				94D750F22A31A796005859E7 /* ContentView.swift in Sources */,
+				94D750F02A31A796005859E7 /* BlinkReviewerApp.swift in Sources */,
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+		94D750F92A31A798005859E7 /* Sources */ = {
+			isa = PBXSourcesBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+				94D751022A31A798005859E7 /* BlinkReviewerTests.swift in Sources */,
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+		94D751032A31A798005859E7 /* Sources */ = {
+			isa = PBXSourcesBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+				94D7510E2A31A798005859E7 /* BlinkReviewerUITestsLaunchTests.swift in Sources */,
+				94D7510C2A31A798005859E7 /* BlinkReviewerUITests.swift in Sources */,
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+/* End PBXSourcesBuildPhase section */
+
+/* Begin PBXTargetDependency section */
+		94D750FF2A31A798005859E7 /* PBXTargetDependency */ = {
+			isa = PBXTargetDependency;
+			target = 94D750EB2A31A796005859E7 /* BlinkReviewer */;
+			targetProxy = 94D750FE2A31A798005859E7 /* PBXContainerItemProxy */;
+		};
+		94D751092A31A798005859E7 /* PBXTargetDependency */ = {
+			isa = PBXTargetDependency;
+			target = 94D750EB2A31A796005859E7 /* BlinkReviewer */;
+			targetProxy = 94D751082A31A798005859E7 /* PBXContainerItemProxy */;
+		};
+/* End PBXTargetDependency section */
+
+/* Begin XCBuildConfiguration section */
+		94D7510F2A31A798005859E7 /* Debug */ = {
+			isa = XCBuildConfiguration;
+			buildSettings = {
+				ALWAYS_SEARCH_USER_PATHS = NO;
+				CLANG_ANALYZER_NONNULL = YES;
+				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;
+				CLANG_CXX_LANGUAGE_STANDARD = "gnu++20";
+				CLANG_ENABLE_MODULES = YES;
+				CLANG_ENABLE_OBJC_ARC = YES;
+				CLANG_ENABLE_OBJC_WEAK = YES;
+				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
+				CLANG_WARN_BOOL_CONVERSION = YES;
+				CLANG_WARN_COMMA = YES;
+				CLANG_WARN_CONSTANT_CONVERSION = YES;
+				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
+				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
+				CLANG_WARN_DOCUMENTATION_COMMENTS = YES;
+				CLANG_WARN_EMPTY_BODY = YES;
+				CLANG_WARN_ENUM_CONVERSION = YES;
+				CLANG_WARN_INFINITE_RECURSION = YES;
+				CLANG_WARN_INT_CONVERSION = YES;
+				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
+				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
+				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
+				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
+				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
+				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
+				CLANG_WARN_STRICT_PROTOTYPES = YES;
+				CLANG_WARN_SUSPICIOUS_MOVE = YES;
+				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;
+				CLANG_WARN_UNREACHABLE_CODE = YES;
+				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
+				COPY_PHASE_STRIP = NO;
+				DEBUG_INFORMATION_FORMAT = dwarf;
+				ENABLE_STRICT_OBJC_MSGSEND = YES;
+				ENABLE_TESTABILITY = YES;
+				GCC_C_LANGUAGE_STANDARD = gnu11;
+				GCC_DYNAMIC_NO_PIC = NO;
+				GCC_NO_COMMON_BLOCKS = YES;
+				GCC_OPTIMIZATION_LEVEL = 0;
+				GCC_PREPROCESSOR_DEFINITIONS = (
+					"DEBUG=1",
+					"$(inherited)",
+				);
+				GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
+				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
+				GCC_WARN_UNDECLARED_SELECTOR = YES;
+				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
+				GCC_WARN_UNUSED_FUNCTION = YES;
+				GCC_WARN_UNUSED_VARIABLE = YES;
+				MACOSX_DEPLOYMENT_TARGET = 13.3;
+				MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE;
+				MTL_FAST_MATH = YES;
+				ONLY_ACTIVE_ARCH = YES;
+				SDKROOT = macosx;
+				SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG;
+				SWIFT_OPTIMIZATION_LEVEL = "-Onone";
+			};
+			name = Debug;
+		};
+		94D751102A31A798005859E7 /* Release */ = {
+			isa = XCBuildConfiguration;
+			buildSettings = {
+				ALWAYS_SEARCH_USER_PATHS = NO;
+				CLANG_ANALYZER_NONNULL = YES;
+				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;
+				CLANG_CXX_LANGUAGE_STANDARD = "gnu++20";
+				CLANG_ENABLE_MODULES = YES;
+				CLANG_ENABLE_OBJC_ARC = YES;
+				CLANG_ENABLE_OBJC_WEAK = YES;
+				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
+				CLANG_WARN_BOOL_CONVERSION = YES;
+				CLANG_WARN_COMMA = YES;
+				CLANG_WARN_CONSTANT_CONVERSION = YES;
+				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
+				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
+				CLANG_WARN_DOCUMENTATION_COMMENTS = YES;
+				CLANG_WARN_EMPTY_BODY = YES;
+				CLANG_WARN_ENUM_CONVERSION = YES;
+				CLANG_WARN_INFINITE_RECURSION = YES;
+				CLANG_WARN_INT_CONVERSION = YES;
+				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
+				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
+				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
+				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
+				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
+				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
+				CLANG_WARN_STRICT_PROTOTYPES = YES;
+				CLANG_WARN_SUSPICIOUS_MOVE = YES;
+				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;
+				CLANG_WARN_UNREACHABLE_CODE = YES;
+				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
+				COPY_PHASE_STRIP = NO;
+				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
+				ENABLE_NS_ASSERTIONS = NO;
+				ENABLE_STRICT_OBJC_MSGSEND = YES;
+				GCC_C_LANGUAGE_STANDARD = gnu11;
+				GCC_NO_COMMON_BLOCKS = YES;
+				GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
+				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
+				GCC_WARN_UNDECLARED_SELECTOR = YES;
+				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
+				GCC_WARN_UNUSED_FUNCTION = YES;
+				GCC_WARN_UNUSED_VARIABLE = YES;
+				MACOSX_DEPLOYMENT_TARGET = 13.3;
+				MTL_ENABLE_DEBUG_INFO = NO;
+				MTL_FAST_MATH = YES;
+				SDKROOT = macosx;
+				SWIFT_COMPILATION_MODE = wholemodule;
+				SWIFT_OPTIMIZATION_LEVEL = "-O";
+			};
+			name = Release;
+		};
+		94D751122A31A798005859E7 /* Debug */ = {
+			isa = XCBuildConfiguration;
+			buildSettings = {
+				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
+				ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor;
+				CODE_SIGN_ENTITLEMENTS = BlinkReviewer/BlinkReviewer.entitlements;
+				CODE_SIGN_STYLE = Automatic;
+				COMBINE_HIDPI_IMAGES = YES;
+				CURRENT_PROJECT_VERSION = 1;
+				DEVELOPMENT_ASSET_PATHS = "\"BlinkReviewer/Preview Content\"";
+				ENABLE_PREVIEWS = YES;
+				GENERATE_INFOPLIST_FILE = YES;
+				INFOPLIST_KEY_NSHumanReadableCopyright = "";
+				LD_RUNPATH_SEARCH_PATHS = (
+					"$(inherited)",
+					"@executable_path/../Frameworks",
+				);
+				MARKETING_VERSION = 1.0;
+				PRODUCT_BUNDLE_IDENTIFIER = net.alexwlchan.BlinkReviewer;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				SWIFT_EMIT_LOC_STRINGS = YES;
+				SWIFT_VERSION = 5.0;
+			};
+			name = Debug;
+		};
+		94D751132A31A798005859E7 /* Release */ = {
+			isa = XCBuildConfiguration;
+			buildSettings = {
+				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
+				ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor;
+				CODE_SIGN_ENTITLEMENTS = BlinkReviewer/BlinkReviewer.entitlements;
+				CODE_SIGN_STYLE = Automatic;
+				COMBINE_HIDPI_IMAGES = YES;
+				CURRENT_PROJECT_VERSION = 1;
+				DEVELOPMENT_ASSET_PATHS = "\"BlinkReviewer/Preview Content\"";
+				ENABLE_PREVIEWS = YES;
+				GENERATE_INFOPLIST_FILE = YES;
+				INFOPLIST_KEY_NSHumanReadableCopyright = "";
+				LD_RUNPATH_SEARCH_PATHS = (
+					"$(inherited)",
+					"@executable_path/../Frameworks",
+				);
+				MARKETING_VERSION = 1.0;
+				PRODUCT_BUNDLE_IDENTIFIER = net.alexwlchan.BlinkReviewer;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				SWIFT_EMIT_LOC_STRINGS = YES;
+				SWIFT_VERSION = 5.0;
+			};
+			name = Release;
+		};
+		94D751152A31A798005859E7 /* Debug */ = {
+			isa = XCBuildConfiguration;
+			buildSettings = {
+				ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES;
+				BUNDLE_LOADER = "$(TEST_HOST)";
+				CODE_SIGN_STYLE = Automatic;
+				CURRENT_PROJECT_VERSION = 1;
+				GENERATE_INFOPLIST_FILE = YES;
+				MACOSX_DEPLOYMENT_TARGET = 13.3;
+				MARKETING_VERSION = 1.0;
+				PRODUCT_BUNDLE_IDENTIFIER = net.alexwlchan.BlinkReviewerTests;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				SWIFT_EMIT_LOC_STRINGS = NO;
+				SWIFT_VERSION = 5.0;
+				TEST_HOST = "$(BUILT_PRODUCTS_DIR)/BlinkReviewer.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/BlinkReviewer";
+			};
+			name = Debug;
+		};
+		94D751162A31A798005859E7 /* Release */ = {
+			isa = XCBuildConfiguration;
+			buildSettings = {
+				ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES;
+				BUNDLE_LOADER = "$(TEST_HOST)";
+				CODE_SIGN_STYLE = Automatic;
+				CURRENT_PROJECT_VERSION = 1;
+				GENERATE_INFOPLIST_FILE = YES;
+				MACOSX_DEPLOYMENT_TARGET = 13.3;
+				MARKETING_VERSION = 1.0;
+				PRODUCT_BUNDLE_IDENTIFIER = net.alexwlchan.BlinkReviewerTests;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				SWIFT_EMIT_LOC_STRINGS = NO;
+				SWIFT_VERSION = 5.0;
+				TEST_HOST = "$(BUILT_PRODUCTS_DIR)/BlinkReviewer.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/BlinkReviewer";
+			};
+			name = Release;
+		};
+		94D751182A31A798005859E7 /* Debug */ = {
+			isa = XCBuildConfiguration;
+			buildSettings = {
+				ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES;
+				CODE_SIGN_STYLE = Automatic;
+				CURRENT_PROJECT_VERSION = 1;
+				GENERATE_INFOPLIST_FILE = YES;
+				MARKETING_VERSION = 1.0;
+				PRODUCT_BUNDLE_IDENTIFIER = net.alexwlchan.BlinkReviewerUITests;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				SWIFT_EMIT_LOC_STRINGS = NO;
+				SWIFT_VERSION = 5.0;
+				TEST_TARGET_NAME = BlinkReviewer;
+			};
+			name = Debug;
+		};
+		94D751192A31A798005859E7 /* Release */ = {
+			isa = XCBuildConfiguration;
+			buildSettings = {
+				ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES;
+				CODE_SIGN_STYLE = Automatic;
+				CURRENT_PROJECT_VERSION = 1;
+				GENERATE_INFOPLIST_FILE = YES;
+				MARKETING_VERSION = 1.0;
+				PRODUCT_BUNDLE_IDENTIFIER = net.alexwlchan.BlinkReviewerUITests;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				SWIFT_EMIT_LOC_STRINGS = NO;
+				SWIFT_VERSION = 5.0;
+				TEST_TARGET_NAME = BlinkReviewer;
+			};
+			name = Release;
+		};
+/* End XCBuildConfiguration section */
+
+/* Begin XCConfigurationList section */
+		94D750E72A31A796005859E7 /* Build configuration list for PBXProject "BlinkReviewer" */ = {
+			isa = XCConfigurationList;
+			buildConfigurations = (
+				94D7510F2A31A798005859E7 /* Debug */,
+				94D751102A31A798005859E7 /* Release */,
+			);
+			defaultConfigurationIsVisible = 0;
+			defaultConfigurationName = Release;
+		};
+		94D751112A31A798005859E7 /* Build configuration list for PBXNativeTarget "BlinkReviewer" */ = {
+			isa = XCConfigurationList;
+			buildConfigurations = (
+				94D751122A31A798005859E7 /* Debug */,
+				94D751132A31A798005859E7 /* Release */,
+			);
+			defaultConfigurationIsVisible = 0;
+			defaultConfigurationName = Release;
+		};
+		94D751142A31A798005859E7 /* Build configuration list for PBXNativeTarget "BlinkReviewerTests" */ = {
+			isa = XCConfigurationList;
+			buildConfigurations = (
+				94D751152A31A798005859E7 /* Debug */,
+				94D751162A31A798005859E7 /* Release */,
+			);
+			defaultConfigurationIsVisible = 0;
+			defaultConfigurationName = Release;
+		};
+		94D751172A31A798005859E7 /* Build configuration list for PBXNativeTarget "BlinkReviewerUITests" */ = {
+			isa = XCConfigurationList;
+			buildConfigurations = (
+				94D751182A31A798005859E7 /* Debug */,
+				94D751192A31A798005859E7 /* Release */,
+			);
+			defaultConfigurationIsVisible = 0;
+			defaultConfigurationName = Release;
+		};
+/* End XCConfigurationList section */
+	};
+	rootObject = 94D750E42A31A796005859E7 /* Project object */;
+}

BlinkReviewer/BlinkReviewer.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist (0) → BlinkReviewer/BlinkReviewer.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist (238)

diff --git a/BlinkReviewer/BlinkReviewer.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/BlinkReviewer/BlinkReviewer.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist
new file mode 100644
index 0000000..18d9810
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist
@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
+<plist version="1.0">
+<dict>
+	<key>IDEDidComputeMac32BitWarning</key>
+	<true/>
+</dict>
+</plist>

BlinkReviewer/BlinkReviewer/Assets.xcassets/AccentColor.colorset/Contents.json (0) → BlinkReviewer/BlinkReviewer/Assets.xcassets/AccentColor.colorset/Contents.json (123)

diff --git a/BlinkReviewer/BlinkReviewer/Assets.xcassets/AccentColor.colorset/Contents.json b/BlinkReviewer/BlinkReviewer/Assets.xcassets/AccentColor.colorset/Contents.json
new file mode 100644
index 0000000..eb87897
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Assets.xcassets/AccentColor.colorset/Contents.json
@@ -0,0 +1,11 @@
+{
+  "colors" : [
+    {
+      "idiom" : "universal"
+    }
+  ],
+  "info" : {
+    "author" : "xcode",
+    "version" : 1
+  }
+}

BlinkReviewer/BlinkReviewer/Assets.xcassets/AppIcon.appiconset/Contents.json (0) → BlinkReviewer/BlinkReviewer/Assets.xcassets/AppIcon.appiconset/Contents.json (904)

diff --git a/BlinkReviewer/BlinkReviewer/Assets.xcassets/AppIcon.appiconset/Contents.json b/BlinkReviewer/BlinkReviewer/Assets.xcassets/AppIcon.appiconset/Contents.json
new file mode 100644
index 0000000..3f00db4
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Assets.xcassets/AppIcon.appiconset/Contents.json
@@ -0,0 +1,58 @@
+{
+  "images" : [
+    {
+      "idiom" : "mac",
+      "scale" : "1x",
+      "size" : "16x16"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "2x",
+      "size" : "16x16"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "1x",
+      "size" : "32x32"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "2x",
+      "size" : "32x32"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "1x",
+      "size" : "128x128"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "2x",
+      "size" : "128x128"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "1x",
+      "size" : "256x256"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "2x",
+      "size" : "256x256"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "1x",
+      "size" : "512x512"
+    },
+    {
+      "idiom" : "mac",
+      "scale" : "2x",
+      "size" : "512x512"
+    }
+  ],
+  "info" : {
+    "author" : "xcode",
+    "version" : 1
+  }
+}

BlinkReviewer/BlinkReviewer/Assets.xcassets/Contents.json (0) → BlinkReviewer/BlinkReviewer/Assets.xcassets/Contents.json (63)

diff --git a/BlinkReviewer/BlinkReviewer/Assets.xcassets/Contents.json b/BlinkReviewer/BlinkReviewer/Assets.xcassets/Contents.json
new file mode 100644
index 0000000..73c0059
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Assets.xcassets/Contents.json
@@ -0,0 +1,6 @@
+{
+  "info" : {
+    "author" : "xcode",
+    "version" : 1
+  }
+}

BlinkReviewer/BlinkReviewer/BlinkReviewer.entitlements (0) → BlinkReviewer/BlinkReviewer/BlinkReviewer.entitlements (322)

diff --git a/BlinkReviewer/BlinkReviewer/BlinkReviewer.entitlements b/BlinkReviewer/BlinkReviewer/BlinkReviewer.entitlements
new file mode 100644
index 0000000..f2ef3ae
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/BlinkReviewer.entitlements
@@ -0,0 +1,10 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
+<plist version="1.0">
+<dict>
+    <key>com.apple.security.app-sandbox</key>
+    <true/>
+    <key>com.apple.security.files.user-selected.read-only</key>
+    <true/>
+</dict>
+</plist>

BlinkReviewer/BlinkReviewer/BlinkReviewerApp.swift (0) → BlinkReviewer/BlinkReviewer/BlinkReviewerApp.swift (241)

diff --git a/BlinkReviewer/BlinkReviewer/BlinkReviewerApp.swift b/BlinkReviewer/BlinkReviewer/BlinkReviewerApp.swift
new file mode 100644
index 0000000..db7a252
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/BlinkReviewerApp.swift
@@ -0,0 +1,17 @@
+//
+//  BlinkReviewerApp.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import SwiftUI
+
+@main
+struct BlinkReviewerApp: App {
+    var body: some Scene {
+        WindowGroup {
+            ContentView()
+        }
+    }
+}

BlinkReviewer/BlinkReviewer/ContentView.swift (0) → BlinkReviewer/BlinkReviewer/ContentView.swift (917)

diff --git a/BlinkReviewer/BlinkReviewer/ContentView.swift b/BlinkReviewer/BlinkReviewer/ContentView.swift
new file mode 100644
index 0000000..09b690c
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/ContentView.swift
@@ -0,0 +1,39 @@
+//
+//  ContentView.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import SwiftUI
+
+struct ContentView: View {
+    var body: some View {
+        VStack {
+            Divider()
+            ScrollViewReader { value in
+                ScrollView(.horizontal) {
+                    Button("Jump to #8") {
+                        value.scrollTo(8, anchor: .center)
+                    }
+                    .padding()
+                    
+                    LazyHStack(spacing: 10) {
+                        ForEach(0..<100, id: \.self) { index in
+                            ThumbnailItem(label: "\(index)")
+                        }
+                    }.padding()
+                }.frame(height: 100)
+            }
+            
+        }
+        Divider()
+        Spacer()
+    }
+}
+
+struct ContentView_Previews: PreviewProvider {
+    static var previews: some View {
+        ContentView()
+    }
+}

BlinkReviewer/BlinkReviewer/Preview Content/Preview Assets.xcassets/Contents.json (0) → BlinkReviewer/BlinkReviewer/Preview Content/Preview Assets.xcassets/Contents.json (63)

diff --git a/BlinkReviewer/BlinkReviewer/Preview Content/Preview Assets.xcassets/Contents.json b/BlinkReviewer/BlinkReviewer/Preview Content/Preview Assets.xcassets/Contents.json
new file mode 100644
index 0000000..73c0059
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Preview Content/Preview Assets.xcassets/Contents.json
@@ -0,0 +1,6 @@
+{
+  "info" : {
+    "author" : "xcode",
+    "version" : 1
+  }
+}

BlinkReviewer/BlinkReviewer/Views/ThumbnailItem.swift (0) → BlinkReviewer/BlinkReviewer/Views/ThumbnailItem.swift (513)

diff --git a/BlinkReviewer/BlinkReviewer/Views/ThumbnailItem.swift b/BlinkReviewer/BlinkReviewer/Views/ThumbnailItem.swift
new file mode 100644
index 0000000..f00eb8c
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Views/ThumbnailItem.swift
@@ -0,0 +1,28 @@
+//
+//  ThumbnailItem.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import SwiftUI
+
+struct ThumbnailItem: View {
+    @State var label: String
+    
+    var body: some View {
+        ZStack {
+            Circle()
+                .fill(.yellow)
+                .frame(width: 70, height: 70)
+            Text(label)
+                .font(.title)
+        }
+    }
+}
+
+struct ThumbnailItem_Previews: PreviewProvider {
+    static var previews: some View {
+        ThumbnailItem(label: "1")
+    }
+}

BlinkReviewer/BlinkReviewerTests/BlinkReviewerTests.swift (0) → BlinkReviewer/BlinkReviewerTests/BlinkReviewerTests.swift (1244)

diff --git a/BlinkReviewer/BlinkReviewerTests/BlinkReviewerTests.swift b/BlinkReviewer/BlinkReviewerTests/BlinkReviewerTests.swift
new file mode 100644
index 0000000..52ae162
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewerTests/BlinkReviewerTests.swift
@@ -0,0 +1,36 @@
+//
+//  BlinkReviewerTests.swift
+//  BlinkReviewerTests
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import XCTest
+@testable import BlinkReviewer
+
+final class BlinkReviewerTests: XCTestCase {
+
+    override func setUpWithError() throws {
+        // Put setup code here. This method is called before the invocation of each test method in the class.
+    }
+
+    override func tearDownWithError() throws {
+        // Put teardown code here. This method is called after the invocation of each test method in the class.
+    }
+
+    func testExample() throws {
+        // This is an example of a functional test case.
+        // Use XCTAssert and related functions to verify your tests produce the correct results.
+        // Any test you write for XCTest can be annotated as throws and async.
+        // Mark your test throws to produce an unexpected failure when your test encounters an uncaught error.
+        // Mark your test async to allow awaiting for asynchronous code to complete. Check the results with assertions afterwards.
+    }
+
+    func testPerformanceExample() throws {
+        // This is an example of a performance test case.
+        self.measure {
+            // Put the code you want to measure the time of here.
+        }
+    }
+
+}

BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITests.swift (0) → BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITests.swift (1392)

diff --git a/BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITests.swift b/BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITests.swift
new file mode 100644
index 0000000..993c3a4
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITests.swift
@@ -0,0 +1,41 @@
+//
+//  BlinkReviewerUITests.swift
+//  BlinkReviewerUITests
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import XCTest
+
+final class BlinkReviewerUITests: XCTestCase {
+
+    override func setUpWithError() throws {
+        // Put setup code here. This method is called before the invocation of each test method in the class.
+
+        // In UI tests it is usually best to stop immediately when a failure occurs.
+        continueAfterFailure = false
+
+        // In UI tests it’s important to set the initial state - such as interface orientation - required for your tests before they run. The setUp method is a good place to do this.
+    }
+
+    override func tearDownWithError() throws {
+        // Put teardown code here. This method is called after the invocation of each test method in the class.
+    }
+
+    func testExample() throws {
+        // UI tests must launch the application that they test.
+        let app = XCUIApplication()
+        app.launch()
+
+        // Use XCTAssert and related functions to verify your tests produce the correct results.
+    }
+
+    func testLaunchPerformance() throws {
+        if #available(macOS 10.15, iOS 13.0, tvOS 13.0, watchOS 7.0, *) {
+            // This measures how long it takes to launch your application.
+            measure(metrics: [XCTApplicationLaunchMetric()]) {
+                XCUIApplication().launch()
+            }
+        }
+    }
+}

BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITestsLaunchTests.swift (0) → BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITestsLaunchTests.swift (820)

diff --git a/BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITestsLaunchTests.swift b/BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITestsLaunchTests.swift
new file mode 100644
index 0000000..920bf87
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewerUITests/BlinkReviewerUITestsLaunchTests.swift
@@ -0,0 +1,32 @@
+//
+//  BlinkReviewerUITestsLaunchTests.swift
+//  BlinkReviewerUITests
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import XCTest
+
+final class BlinkReviewerUITestsLaunchTests: XCTestCase {
+
+    override class var runsForEachTargetApplicationUIConfiguration: Bool {
+        true
+    }
+
+    override func setUpWithError() throws {
+        continueAfterFailure = false
+    }
+
+    func testLaunch() throws {
+        let app = XCUIApplication()
+        app.launch()
+
+        // Insert steps here to perform after app launch but before taking a screenshot,
+        // such as logging into a test account or navigating somewhere in the app
+
+        let attachment = XCTAttachment(screenshot: app.screenshot())
+        attachment.name = "Launch Screen"
+        attachment.lifetime = .keepAlways
+        add(attachment)
+    }
+}