Skip to main content

Merge pull request #7 from alexwlchan/swiftui

ID
8d90f16
date
2023-06-08 09:38:01+00:00
author
Alex Chan <alex@alexwlchan.net>
parents
2eb8250, bff1a60
message
Merge pull request #7 from alexwlchan/swiftui

Start building a proper Mac app in SwiftUI
changed files
20 files, 1128 additions

Changed files

.gitignore (0) → .gitignore (32)

diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..f495a82
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,2 @@
+*.xcuserstate
+*.xcworkspacedata

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

diff --git a/BlinkReviewer/BlinkReviewer.xcodeproj/project.pbxproj b/BlinkReviewer/BlinkReviewer.xcodeproj/project.pbxproj
new file mode 100644
index 0000000..80f28a6
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer.xcodeproj/project.pbxproj
@@ -0,0 +1,611 @@
+// !$*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 /* ThumbnailImage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D7511B2A31A7B1005859E7 /* ThumbnailImage.swift */; };
+		94D7511E2A31B243005859E7 /* PreviewImage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D7511D2A31B243005859E7 /* PreviewImage.swift */; };
+		94D751202A31B53E005859E7 /* AlbumInfo.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D7511F2A31B53E005859E7 /* AlbumInfo.swift */; };
+		94D751222A31BD8E005859E7 /* PhotoReviewer.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D751212A31BD8E005859E7 /* PhotoReviewer.swift */; };
+		94D7512B2A31D6AC005859E7 /* AssetHelpers.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94D7512A2A31D6AC005859E7 /* AssetHelpers.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 /* ThumbnailImage.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ThumbnailImage.swift; sourceTree = "<group>"; };
+		94D7511D2A31B243005859E7 /* PreviewImage.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PreviewImage.swift; sourceTree = "<group>"; };
+		94D7511F2A31B53E005859E7 /* AlbumInfo.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AlbumInfo.swift; sourceTree = "<group>"; };
+		94D751212A31BD8E005859E7 /* PhotoReviewer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PhotoReviewer.swift; sourceTree = "<group>"; };
+		94D7512A2A31D6AC005859E7 /* AssetHelpers.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AssetHelpers.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 = (
+				94D751272A31D61D005859E7 /* Photos */,
+				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 /* ThumbnailImage.swift */,
+				94D7511D2A31B243005859E7 /* PreviewImage.swift */,
+				94D7511F2A31B53E005859E7 /* AlbumInfo.swift */,
+				94D751212A31BD8E005859E7 /* PhotoReviewer.swift */,
+			);
+			path = Views;
+			sourceTree = "<group>";
+		};
+		94D751272A31D61D005859E7 /* Photos */ = {
+			isa = PBXGroup;
+			children = (
+				94D7512A2A31D6AC005859E7 /* AssetHelpers.swift */,
+			);
+			path = Photos;
+			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 /* ThumbnailImage.swift in Sources */,
+				94D750F22A31A796005859E7 /* ContentView.swift in Sources */,
+				94D7512B2A31D6AC005859E7 /* AssetHelpers.swift in Sources */,
+				94D7511E2A31B243005859E7 /* PreviewImage.swift in Sources */,
+				94D750F02A31A796005859E7 /* BlinkReviewerApp.swift in Sources */,
+				94D751202A31B53E005859E7 /* AlbumInfo.swift in Sources */,
+				94D751222A31BD8E005859E7 /* PhotoReviewer.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 = "";
+				INFOPLIST_KEY_NSPhotoLibraryUsageDescription = "This app needs full Photo Library permissions.";
+				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 = "";
+				INFOPLIST_KEY_NSPhotoLibraryUsageDescription = "This app needs full Photo Library permissions.";
+				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/Assets.xcassets/IMG_5934.imageset/Contents.json (0) → BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/Contents.json (207)

diff --git a/BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/Contents.json b/BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/Contents.json
new file mode 100644
index 0000000..3f75115
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/Contents.json
@@ -0,0 +1,14 @@
+{
+  "images" : [
+    {
+      "filename" : "IMG_5934.jpg",
+      "idiom" : "universal",
+      "scale" : "1x",
+      "unassigned" : true
+    }
+  ],
+  "info" : {
+    "author" : "xcode",
+    "version" : 1
+  }
+}

BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/IMG_5934.jpg (0) → BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/IMG_5934.jpg (3790742)

diff --git a/BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/IMG_5934.jpg b/BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/IMG_5934.jpg
new file mode 100644
index 0000000..37b60bf
Binary files /dev/null and b/BlinkReviewer/BlinkReviewer/Assets.xcassets/IMG_5934.imageset/IMG_5934.jpg differ

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 (213)

diff --git a/BlinkReviewer/BlinkReviewer/ContentView.swift b/BlinkReviewer/BlinkReviewer/ContentView.swift
new file mode 100644
index 0000000..905ad9c
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/ContentView.swift
@@ -0,0 +1,14 @@
+//
+//  ContentView.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import SwiftUI
+
+struct ContentView: View {
+    var body: some View {
+        PhotoReviewer(assets: getAllPhotos())
+    }
+}

BlinkReviewer/BlinkReviewer/Photos/AssetHelpers.swift (0) → BlinkReviewer/BlinkReviewer/Photos/AssetHelpers.swift (2426)

diff --git a/BlinkReviewer/BlinkReviewer/Photos/AssetHelpers.swift b/BlinkReviewer/BlinkReviewer/Photos/AssetHelpers.swift
new file mode 100644
index 0000000..80ba71a
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Photos/AssetHelpers.swift
@@ -0,0 +1,82 @@
+//
+//  Helpers.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import Foundation
+import Photos
+import SwiftUI
+
+/// Returns a list of all the images in the Photos Library.
+func getAllPhotos() -> [PHAsset] {
+    var photos: [PHAsset] = []
+    
+    PHAsset.fetchAssets(with: PHAssetMediaType.image, options: nil)
+        .enumerateObjects({ (asset, _, _) in
+            photos.append(asset)
+        })
+    
+    return photos
+}
+
+extension PHAsset {
+    /// Returns a list of all the albums that contain this asset.
+    func albums() -> [PHAssetCollection] {
+        var result: [PHAssetCollection] = []
+        
+        PHAssetCollection
+            .fetchAssetCollectionsContaining(self, with: .album, options: nil)
+            .enumerateObjects({ (collection, index, stop) in
+                result.append(collection)
+            })
+        
+        return result
+    }
+    
+    private func getImageForSize(size: CGSize) -> NSImage {
+        // This implementation is based on code in a Stack Overflow answer
+        // by Francois Nadeau: https://stackoverflow.com/a/48755517/1558022
+
+        let options = PHImageRequestOptions()
+        
+        // do I still need this?
+        options.isSynchronous = true
+
+        // If i don't set this value, then sometimes I get an error like
+        // this in the `info` variable:
+        //
+        //      Error Domain=PHPhotosErrorDomain Code=3164 "(null)"
+        //
+        // This means that the asset is in the cloud, and by default Photos
+        // isn't allowed to download assets here.  Apple's documentation
+        // suggests adding this option as the fix.
+        //
+        // See https://developer.apple.com/documentation/photokit/phphotoserror/phphotoserrornetworkaccessrequired
+        options.isNetworkAccessAllowed = true
+
+        var image = NSImage()
+        
+        PHCachingImageManager()
+            .requestImage(
+                for: self,
+                targetSize: size,
+                contentMode: .aspectFill,
+                options: options,
+                resultHandler: { (result, info) -> Void in
+                    image = result!
+                }
+            )
+
+        return image
+    }
+    
+    func getThumbnail() -> NSImage {
+        return getImageForSize(size: CGSize(width: 70, height: 70))
+    }
+    
+    func getImage() -> NSImage {
+        return getImageForSize(size: PHImageManagerMaximumSize)
+    }
+}

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/AlbumInfo.swift (0) → BlinkReviewer/BlinkReviewer/Views/AlbumInfo.swift (1023)

diff --git a/BlinkReviewer/BlinkReviewer/Views/AlbumInfo.swift b/BlinkReviewer/BlinkReviewer/Views/AlbumInfo.swift
new file mode 100644
index 0000000..31b06d3
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Views/AlbumInfo.swift
@@ -0,0 +1,36 @@
+//
+//  AlbumInfo.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import SwiftUI
+import Photos
+
+/// This view shows the names of the albums that a given asset is in.
+///
+/// Each album is shown as a separate "pill" in the list, for example:
+///
+///     [Cats] [Cross-stitch] [Stuff I did in 2023]
+///
+struct AlbumInfo: View {
+    var asset: PHAsset
+    
+    var body: some View {
+        HStack {
+            ForEach(asset.albums(), id: \.localIdentifier) { album in
+                if let title = album.localizedTitle {
+                    // The icon was chosen to match the one used for albums
+                    // in the sidebar in Photos.
+                    Text("\(Image(systemName: "rectangle.stack")) \(title)")
+                        .fontWeight(.bold)
+                        .font(.title2)
+                        .padding(5)
+                        .background(.white.opacity(0.9))
+                        .cornerRadius(7.0)
+                }
+            }
+        }.padding()
+    }
+}

BlinkReviewer/BlinkReviewer/Views/PhotoReviewer.swift (0) → BlinkReviewer/BlinkReviewer/Views/PhotoReviewer.swift (2024)

diff --git a/BlinkReviewer/BlinkReviewer/Views/PhotoReviewer.swift b/BlinkReviewer/BlinkReviewer/Views/PhotoReviewer.swift
new file mode 100644
index 0000000..6b97356
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Views/PhotoReviewer.swift
@@ -0,0 +1,65 @@
+//
+//  PhotoReviewer.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import SwiftUI
+import Photos
+
+struct PhotoReviewer: View {
+    var assets: [PHAsset]
+    @State private var selectedAssetIndex: Int = 0
+    
+    var body: some View {
+        VStack {
+            Divider()
+            ScrollViewReader { proxy in
+                ScrollView(.horizontal) {
+                    LazyHStack(spacing: 5) {
+                        // TODO: placeholder images for start/end
+                        // TODO: Allow tapping thumbnails to jump to that
+                        ForEach(assets, id: \.localIdentifier) { asset in
+                            ThumbnailImage(thumbnail: asset.getThumbnail(), isSelected: assets[selectedAssetIndex].localIdentifier == asset.localIdentifier)
+                        }
+                    }.padding()
+                }.frame(height: 70)
+                    .onChange(of: selectedAssetIndex, perform: { newIndex in
+                        withAnimation {
+                            proxy.scrollTo(assets[newIndex].localIdentifier, anchor: .center)
+                        }
+                        
+                    })
+            }
+            Divider()
+            
+            PreviewImage(asset: assets[selectedAssetIndex])
+            
+            Spacer()
+        }.onAppear {
+            NSEvent.addLocalMonitorForEvents(matching: .keyDown) { event in
+                handleKeyEvent(event)
+                return event
+            }
+        }
+    }
+    
+    private func handleKeyEvent(_ event: NSEvent) {
+        switch event.keyCode {
+            case 123: // Left arrow key
+                if selectedAssetIndex > 0 {
+                    selectedAssetIndex -= 1
+                }
+            
+            case 124: // Right arrow key
+                if selectedAssetIndex < assets.count - 1 {
+                    selectedAssetIndex += 1
+                }
+            
+            default:
+                print(event)
+                break
+        }
+    }
+}

BlinkReviewer/BlinkReviewer/Views/PreviewImage.swift (0) → BlinkReviewer/BlinkReviewer/Views/PreviewImage.swift (752)

diff --git a/BlinkReviewer/BlinkReviewer/Views/PreviewImage.swift b/BlinkReviewer/BlinkReviewer/Views/PreviewImage.swift
new file mode 100644
index 0000000..c4444c4
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Views/PreviewImage.swift
@@ -0,0 +1,32 @@
+//
+//  PreviewImage.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import Photos
+import SwiftUI
+
+struct PreviewImage: View {
+    var asset: PHAsset
+    
+    var body: some View {
+        ZStack {
+            VStack {
+                HStack {
+                    Spacer()
+                        
+                    Image(nsImage: asset.getImage())
+                        .resizable()
+                        .aspectRatio(contentMode: .fit)
+                        .overlay(alignment: Alignment(horizontal: .center, vertical: .top)) {
+                            AlbumInfo(asset: asset)
+                        }
+                        
+                    Spacer()
+                }
+            }.padding()
+        }
+    }
+}

BlinkReviewer/BlinkReviewer/Views/ThumbnailImage.swift (0) → BlinkReviewer/BlinkReviewer/Views/ThumbnailImage.swift (1271)

diff --git a/BlinkReviewer/BlinkReviewer/Views/ThumbnailImage.swift b/BlinkReviewer/BlinkReviewer/Views/ThumbnailImage.swift
new file mode 100644
index 0000000..2eed657
--- /dev/null
+++ b/BlinkReviewer/BlinkReviewer/Views/ThumbnailImage.swift
@@ -0,0 +1,47 @@
+//
+//  ThumbnailItem.swift
+//  BlinkReviewer
+//
+//  Created by Alex Chan on 08/06/2023.
+//
+
+import SwiftUI
+
+/// Renders a square thumbnail for an image.
+///
+/// The image will be expanded to fill the square, and may be clipped
+/// if the original aspect ratio isn't square.
+struct ThumbnailImage: View {
+    var thumbnail: NSImage
+    var isSelected: Bool
+    
+    var size: CGFloat {
+        isSelected ? 70.0 : 50.0
+    }
+    
+    var body: some View {
+        Image(nsImage: thumbnail)
+            .resizable()
+            // Note: it's taken several attempts to get this working correctly;
+            // it behaves differently in the running app to the SwiftUI preview.
+            //
+            // Expected properties:
+            //
+            //    - Thumbnails are square
+            //    - Thumbnails are expanded to fill the square, but they prefer
+            //      to crop rather than stretch the image
+            //
+            .scaledToFill()
+            .frame(width: size, height: size, alignment: .center)
+            .clipped()
+    }
+}
+
+struct ThumbnailItem_Previews: PreviewProvider {
+    static var previews: some View {
+        ThumbnailImage(
+            thumbnail: NSImage(named: "IMG_5934")!,
+            isSelected: true
+        )
+    }
+}

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)
+    }
+}