Add SwiftUI + Redux

This commit is contained in:
Ivan Vorobei
2019-06-07 08:58:01 +03:00
parent 5b758db8cd
commit f385cd4929
29 changed files with 1311 additions and 0 deletions

View File

@@ -0,0 +1,495 @@
// !$*UTF8*$!
{
archiveVersion = 1;
classes = {
};
objectVersion = 50;
objects = {
/* Begin PBXBuildFile section */
6905595122A82C0400639314 /* AppState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6905595022A82C0400639314 /* AppState.swift */; };
69250A8B22A97BE300332EF0 /* Badge.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69250A8A22A97BE300332EF0 /* Badge.swift */; };
69516A5D22A8743500E62A5E /* Reducer.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69516A5C22A8743500E62A5E /* Reducer.swift */; };
69516A5F22A8745800E62A5E /* FluxState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69516A5E22A8745800E62A5E /* FluxState.swift */; };
69516A6122A8747600E62A5E /* Action.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69516A6022A8747600E62A5E /* Action.swift */; };
69516A6322A874CE00E62A5E /* UsersStateReducer.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69516A6222A874CE00E62A5E /* UsersStateReducer.swift */; };
69516A6522A874F900E62A5E /* UsersAction.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69516A6422A874F900E62A5E /* UsersAction.swift */; };
6994C88022A81130006C5F62 /* UserEditForm.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6994C87F22A81130006C5F62 /* UserEditForm.swift */; };
69C1521A22A8CB3600149392 /* TabbarView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69C1521922A8CB3600149392 /* TabbarView.swift */; };
69C1522222A8CF4600149392 /* MapView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69C1522122A8CF4600149392 /* MapView.swift */; };
69C1522522A8D01300149392 /* MapKit.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 69C1522422A8D01300149392 /* MapKit.framework */; };
69E06C2D22A6FD6B0081D614 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69E06C2C22A6FD6B0081D614 /* AppDelegate.swift */; };
69E06C2F22A6FD6B0081D614 /* SceneDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69E06C2E22A6FD6B0081D614 /* SceneDelegate.swift */; };
69E06C3122A6FD6B0081D614 /* UsersListView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69E06C3022A6FD6B0081D614 /* UsersListView.swift */; };
69E06C3322A6FD6D0081D614 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 69E06C3222A6FD6D0081D614 /* Assets.xcassets */; };
69E06C3622A6FD6D0081D614 /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 69E06C3522A6FD6D0081D614 /* Preview Assets.xcassets */; };
69E06C3922A6FD6D0081D614 /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 69E06C3722A6FD6D0081D614 /* LaunchScreen.storyboard */; };
69E06C4422A6FE940081D614 /* User.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69E06C4322A6FE940081D614 /* User.swift */; };
69E06C4622A6FF3C0081D614 /* UsersState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69E06C4522A6FF3C0081D614 /* UsersState.swift */; };
69E06C4922A7016B0081D614 /* UserRow.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69E06C4822A7016B0081D614 /* UserRow.swift */; };
69E06C4C22A706450081D614 /* UserDetailView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69E06C4B22A706450081D614 /* UserDetailView.swift */; };
/* End PBXBuildFile section */
/* Begin PBXFileReference section */
6905595022A82C0400639314 /* AppState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppState.swift; sourceTree = "<group>"; };
69250A8A22A97BE300332EF0 /* Badge.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Badge.swift; sourceTree = "<group>"; };
69516A5C22A8743500E62A5E /* Reducer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Reducer.swift; sourceTree = "<group>"; };
69516A5E22A8745800E62A5E /* FluxState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FluxState.swift; sourceTree = "<group>"; };
69516A6022A8747600E62A5E /* Action.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Action.swift; sourceTree = "<group>"; };
69516A6222A874CE00E62A5E /* UsersStateReducer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UsersStateReducer.swift; sourceTree = "<group>"; };
69516A6422A874F900E62A5E /* UsersAction.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UsersAction.swift; sourceTree = "<group>"; };
6994C87F22A81130006C5F62 /* UserEditForm.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserEditForm.swift; sourceTree = "<group>"; };
69C1521922A8CB3600149392 /* TabbarView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TabbarView.swift; sourceTree = "<group>"; };
69C1522122A8CF4600149392 /* MapView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MapView.swift; sourceTree = "<group>"; };
69C1522422A8D01300149392 /* MapKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = MapKit.framework; path = System/Library/Frameworks/MapKit.framework; sourceTree = SDKROOT; };
69E06C2922A6FD6B0081D614 /* SwiftUIDemo.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = SwiftUIDemo.app; sourceTree = BUILT_PRODUCTS_DIR; };
69E06C2C22A6FD6B0081D614 /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = "<group>"; };
69E06C2E22A6FD6B0081D614 /* SceneDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SceneDelegate.swift; sourceTree = "<group>"; };
69E06C3022A6FD6B0081D614 /* UsersListView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UsersListView.swift; sourceTree = "<group>"; };
69E06C3222A6FD6D0081D614 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; };
69E06C3522A6FD6D0081D614 /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = "<group>"; };
69E06C3822A6FD6D0081D614 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = "<group>"; };
69E06C3A22A6FD6D0081D614 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
69E06C4322A6FE940081D614 /* User.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = User.swift; sourceTree = "<group>"; };
69E06C4522A6FF3C0081D614 /* UsersState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UsersState.swift; sourceTree = "<group>"; };
69E06C4822A7016B0081D614 /* UserRow.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserRow.swift; sourceTree = "<group>"; };
69E06C4B22A706450081D614 /* UserDetailView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserDetailView.swift; sourceTree = "<group>"; };
/* End PBXFileReference section */
/* Begin PBXFrameworksBuildPhase section */
69E06C2622A6FD6B0081D614 /* Frameworks */ = {
isa = PBXFrameworksBuildPhase;
buildActionMask = 2147483647;
files = (
69C1522522A8D01300149392 /* MapKit.framework in Frameworks */,
);
runOnlyForDeploymentPostprocessing = 0;
};
/* End PBXFrameworksBuildPhase section */
/* Begin PBXGroup section */
69250A8922A97BD500332EF0 /* component */ = {
isa = PBXGroup;
children = (
69250A8A22A97BE300332EF0 /* Badge.swift */,
);
path = component;
sourceTree = "<group>";
};
69516A5822A873C900E62A5E /* states */ = {
isa = PBXGroup;
children = (
69516A5E22A8745800E62A5E /* FluxState.swift */,
6905595022A82C0400639314 /* AppState.swift */,
69E06C4522A6FF3C0081D614 /* UsersState.swift */,
);
path = states;
sourceTree = "<group>";
};
69516A5922A873D100E62A5E /* actions */ = {
isa = PBXGroup;
children = (
69516A6022A8747600E62A5E /* Action.swift */,
69516A6422A874F900E62A5E /* UsersAction.swift */,
);
path = actions;
sourceTree = "<group>";
};
69516A5A22A873D700E62A5E /* reducers */ = {
isa = PBXGroup;
children = (
69516A5C22A8743500E62A5E /* Reducer.swift */,
69516A6222A874CE00E62A5E /* UsersStateReducer.swift */,
);
path = reducers;
sourceTree = "<group>";
};
69C1521F22A8CF2A00149392 /* users */ = {
isa = PBXGroup;
children = (
69250A8922A97BD500332EF0 /* component */,
69E06C4722A7015F0081D614 /* rows */,
69E06C3022A6FD6B0081D614 /* UsersListView.swift */,
69E06C4B22A706450081D614 /* UserDetailView.swift */,
6994C87F22A81130006C5F62 /* UserEditForm.swift */,
);
path = users;
sourceTree = "<group>";
};
69C1522022A8CF3800149392 /* map */ = {
isa = PBXGroup;
children = (
69C1522122A8CF4600149392 /* MapView.swift */,
);
path = map;
sourceTree = "<group>";
};
69C1522322A8D01300149392 /* Frameworks */ = {
isa = PBXGroup;
children = (
69C1522422A8D01300149392 /* MapKit.framework */,
);
name = Frameworks;
sourceTree = "<group>";
};
69E06C2022A6FD6B0081D614 = {
isa = PBXGroup;
children = (
69E06C2B22A6FD6B0081D614 /* SwiftUIDemo */,
69E06C2A22A6FD6B0081D614 /* Products */,
69C1522322A8D01300149392 /* Frameworks */,
);
sourceTree = "<group>";
};
69E06C2A22A6FD6B0081D614 /* Products */ = {
isa = PBXGroup;
children = (
69E06C2922A6FD6B0081D614 /* SwiftUIDemo.app */,
);
name = Products;
sourceTree = "<group>";
};
69E06C2B22A6FD6B0081D614 /* SwiftUIDemo */ = {
isa = PBXGroup;
children = (
69F83D4822A85D8E00268B3F /* flux */,
69E06C4A22A7061E0081D614 /* views */,
69E06C2C22A6FD6B0081D614 /* AppDelegate.swift */,
69E06C2E22A6FD6B0081D614 /* SceneDelegate.swift */,
69E06C3222A6FD6D0081D614 /* Assets.xcassets */,
69E06C3722A6FD6D0081D614 /* LaunchScreen.storyboard */,
69E06C3A22A6FD6D0081D614 /* Info.plist */,
69E06C3422A6FD6D0081D614 /* Preview Content */,
);
path = SwiftUIDemo;
sourceTree = "<group>";
};
69E06C3422A6FD6D0081D614 /* Preview Content */ = {
isa = PBXGroup;
children = (
69E06C3522A6FD6D0081D614 /* Preview Assets.xcassets */,
);
path = "Preview Content";
sourceTree = "<group>";
};
69E06C4022A6FD730081D614 /* models */ = {
isa = PBXGroup;
children = (
69E06C4322A6FE940081D614 /* User.swift */,
);
path = models;
sourceTree = "<group>";
};
69E06C4722A7015F0081D614 /* rows */ = {
isa = PBXGroup;
children = (
69E06C4822A7016B0081D614 /* UserRow.swift */,
);
path = rows;
sourceTree = "<group>";
};
69E06C4A22A7061E0081D614 /* views */ = {
isa = PBXGroup;
children = (
69C1522022A8CF3800149392 /* map */,
69C1521F22A8CF2A00149392 /* users */,
69C1521922A8CB3600149392 /* TabbarView.swift */,
);
path = views;
sourceTree = "<group>";
};
69F83D4822A85D8E00268B3F /* flux */ = {
isa = PBXGroup;
children = (
69516A5A22A873D700E62A5E /* reducers */,
69516A5922A873D100E62A5E /* actions */,
69516A5822A873C900E62A5E /* states */,
69E06C4022A6FD730081D614 /* models */,
);
path = flux;
sourceTree = "<group>";
};
/* End PBXGroup section */
/* Begin PBXNativeTarget section */
69E06C2822A6FD6B0081D614 /* SwiftUIDemo */ = {
isa = PBXNativeTarget;
buildConfigurationList = 69E06C3D22A6FD6D0081D614 /* Build configuration list for PBXNativeTarget "SwiftUIDemo" */;
buildPhases = (
69E06C2522A6FD6B0081D614 /* Sources */,
69E06C2622A6FD6B0081D614 /* Frameworks */,
69E06C2722A6FD6B0081D614 /* Resources */,
);
buildRules = (
);
dependencies = (
);
name = SwiftUIDemo;
productName = SwiftUIDemo;
productReference = 69E06C2922A6FD6B0081D614 /* SwiftUIDemo.app */;
productType = "com.apple.product-type.application";
};
/* End PBXNativeTarget section */
/* Begin PBXProject section */
69E06C2122A6FD6B0081D614 /* Project object */ = {
isa = PBXProject;
attributes = {
LastSwiftUpdateCheck = 1100;
LastUpgradeCheck = 1100;
ORGANIZATIONNAME = "Thomas Ricouarf";
TargetAttributes = {
69E06C2822A6FD6B0081D614 = {
CreatedOnToolsVersion = 11.0;
};
};
};
buildConfigurationList = 69E06C2422A6FD6B0081D614 /* Build configuration list for PBXProject "SwiftUIDemo" */;
compatibilityVersion = "Xcode 9.3";
developmentRegion = en;
hasScannedForEncodings = 0;
knownRegions = (
en,
Base,
);
mainGroup = 69E06C2022A6FD6B0081D614;
productRefGroup = 69E06C2A22A6FD6B0081D614 /* Products */;
projectDirPath = "";
projectRoot = "";
targets = (
69E06C2822A6FD6B0081D614 /* SwiftUIDemo */,
);
};
/* End PBXProject section */
/* Begin PBXResourcesBuildPhase section */
69E06C2722A6FD6B0081D614 /* Resources */ = {
isa = PBXResourcesBuildPhase;
buildActionMask = 2147483647;
files = (
69E06C3922A6FD6D0081D614 /* LaunchScreen.storyboard in Resources */,
69E06C3622A6FD6D0081D614 /* Preview Assets.xcassets in Resources */,
69E06C3322A6FD6D0081D614 /* Assets.xcassets in Resources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
/* End PBXResourcesBuildPhase section */
/* Begin PBXSourcesBuildPhase section */
69E06C2522A6FD6B0081D614 /* Sources */ = {
isa = PBXSourcesBuildPhase;
buildActionMask = 2147483647;
files = (
69E06C4622A6FF3C0081D614 /* UsersState.swift in Sources */,
69516A6522A874F900E62A5E /* UsersAction.swift in Sources */,
69516A6122A8747600E62A5E /* Action.swift in Sources */,
69E06C4422A6FE940081D614 /* User.swift in Sources */,
69516A6322A874CE00E62A5E /* UsersStateReducer.swift in Sources */,
69E06C2D22A6FD6B0081D614 /* AppDelegate.swift in Sources */,
69516A5D22A8743500E62A5E /* Reducer.swift in Sources */,
69E06C2F22A6FD6B0081D614 /* SceneDelegate.swift in Sources */,
69E06C4C22A706450081D614 /* UserDetailView.swift in Sources */,
69250A8B22A97BE300332EF0 /* Badge.swift in Sources */,
69C1522222A8CF4600149392 /* MapView.swift in Sources */,
69E06C3122A6FD6B0081D614 /* UsersListView.swift in Sources */,
6905595122A82C0400639314 /* AppState.swift in Sources */,
6994C88022A81130006C5F62 /* UserEditForm.swift in Sources */,
69516A5F22A8745800E62A5E /* FluxState.swift in Sources */,
69C1521A22A8CB3600149392 /* TabbarView.swift in Sources */,
69E06C4922A7016B0081D614 /* UserRow.swift in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
/* End PBXSourcesBuildPhase section */
/* Begin PBXVariantGroup section */
69E06C3722A6FD6D0081D614 /* LaunchScreen.storyboard */ = {
isa = PBXVariantGroup;
children = (
69E06C3822A6FD6D0081D614 /* Base */,
);
name = LaunchScreen.storyboard;
sourceTree = "<group>";
};
/* End PBXVariantGroup section */
/* Begin XCBuildConfiguration section */
69E06C3B22A6FD6D0081D614 /* 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++14";
CLANG_CXX_LIBRARY = "libc++";
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_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;
IPHONEOS_DEPLOYMENT_TARGET = 13.0;
MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE;
MTL_FAST_MATH = YES;
ONLY_ACTIVE_ARCH = YES;
SDKROOT = iphoneos;
SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG;
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
};
name = Debug;
};
69E06C3C22A6FD6D0081D614 /* 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++14";
CLANG_CXX_LIBRARY = "libc++";
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_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;
IPHONEOS_DEPLOYMENT_TARGET = 13.0;
MTL_ENABLE_DEBUG_INFO = NO;
MTL_FAST_MATH = YES;
SDKROOT = iphoneos;
SWIFT_COMPILATION_MODE = wholemodule;
SWIFT_OPTIMIZATION_LEVEL = "-O";
VALIDATE_PRODUCT = YES;
};
name = Release;
};
69E06C3E22A6FD6D0081D614 /* Debug */ = {
isa = XCBuildConfiguration;
buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CODE_SIGN_STYLE = Automatic;
DEVELOPMENT_ASSET_PATHS = "SwiftUIDemo/Preview\\ Content";
DEVELOPMENT_TEAM = Z6P74P6T99;
ENABLE_PREVIEWS = YES;
INFOPLIST_FILE = SwiftUIDemo/Info.plist;
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",
);
PRODUCT_BUNDLE_IDENTIFIER = com.thomasricouard.SwiftUIDemo;
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_VERSION = 5.0;
TARGETED_DEVICE_FAMILY = "1,2";
};
name = Debug;
};
69E06C3F22A6FD6D0081D614 /* Release */ = {
isa = XCBuildConfiguration;
buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CODE_SIGN_STYLE = Automatic;
DEVELOPMENT_ASSET_PATHS = "SwiftUIDemo/Preview\\ Content";
DEVELOPMENT_TEAM = Z6P74P6T99;
ENABLE_PREVIEWS = YES;
INFOPLIST_FILE = SwiftUIDemo/Info.plist;
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",
);
PRODUCT_BUNDLE_IDENTIFIER = com.thomasricouard.SwiftUIDemo;
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_VERSION = 5.0;
TARGETED_DEVICE_FAMILY = "1,2";
};
name = Release;
};
/* End XCBuildConfiguration section */
/* Begin XCConfigurationList section */
69E06C2422A6FD6B0081D614 /* Build configuration list for PBXProject "SwiftUIDemo" */ = {
isa = XCConfigurationList;
buildConfigurations = (
69E06C3B22A6FD6D0081D614 /* Debug */,
69E06C3C22A6FD6D0081D614 /* Release */,
);
defaultConfigurationIsVisible = 0;
defaultConfigurationName = Release;
};
69E06C3D22A6FD6D0081D614 /* Build configuration list for PBXNativeTarget "SwiftUIDemo" */ = {
isa = XCConfigurationList;
buildConfigurations = (
69E06C3E22A6FD6D0081D614 /* Debug */,
69E06C3F22A6FD6D0081D614 /* Release */,
);
defaultConfigurationIsVisible = 0;
defaultConfigurationName = Release;
};
/* End XCConfigurationList section */
};
rootObject = 69E06C2122A6FD6B0081D614 /* Project object */;
}

View File

@@ -0,0 +1,7 @@
<?xml version="1.0" encoding="UTF-8"?>
<Workspace
version = "1.0">
<FileRef
location = "self:SwiftUIDemo.xcodeproj">
</FileRef>
</Workspace>

View File

@@ -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>

View File

@@ -0,0 +1,41 @@
//
// AppDelegate.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 04/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import UIKit
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
// Override point for customization after application launch.
return true
}
func applicationWillTerminate(_ application: UIApplication) {
// Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:.
}
// MARK: UISceneSession Lifecycle
func application(_ application: UIApplication, configurationForConnecting connectingSceneSession: UISceneSession, options: UIScene.ConnectionOptions) -> UISceneConfiguration {
// Called when a new scene session is being created.
// Use this method to select a configuration to create the new scene with.
return UISceneConfiguration(name: "Default Configuration", sessionRole: connectingSceneSession.role)
}
func application(_ application: UIApplication, didDiscardSceneSessions sceneSessions: Set<UISceneSession>) {
// Called when the user discards a scene session.
// If any sessions were discarded while the application was not running, this will be called shortly after application:didFinishLaunchingWithOptions.
// Use this method to release any resources that were specific to the discarded scenes, as they will not return.
}
}

View File

@@ -0,0 +1,98 @@
{
"images" : [
{
"idiom" : "iphone",
"size" : "20x20",
"scale" : "2x"
},
{
"idiom" : "iphone",
"size" : "20x20",
"scale" : "3x"
},
{
"idiom" : "iphone",
"size" : "29x29",
"scale" : "2x"
},
{
"idiom" : "iphone",
"size" : "29x29",
"scale" : "3x"
},
{
"idiom" : "iphone",
"size" : "40x40",
"scale" : "2x"
},
{
"idiom" : "iphone",
"size" : "40x40",
"scale" : "3x"
},
{
"idiom" : "iphone",
"size" : "60x60",
"scale" : "2x"
},
{
"idiom" : "iphone",
"size" : "60x60",
"scale" : "3x"
},
{
"idiom" : "ipad",
"size" : "20x20",
"scale" : "1x"
},
{
"idiom" : "ipad",
"size" : "20x20",
"scale" : "2x"
},
{
"idiom" : "ipad",
"size" : "29x29",
"scale" : "1x"
},
{
"idiom" : "ipad",
"size" : "29x29",
"scale" : "2x"
},
{
"idiom" : "ipad",
"size" : "40x40",
"scale" : "1x"
},
{
"idiom" : "ipad",
"size" : "40x40",
"scale" : "2x"
},
{
"idiom" : "ipad",
"size" : "76x76",
"scale" : "1x"
},
{
"idiom" : "ipad",
"size" : "76x76",
"scale" : "2x"
},
{
"idiom" : "ipad",
"size" : "83.5x83.5",
"scale" : "2x"
},
{
"idiom" : "ios-marketing",
"size" : "1024x1024",
"scale" : "1x"
}
],
"info" : {
"version" : 1,
"author" : "xcode"
}
}

View File

@@ -0,0 +1,6 @@
{
"info" : {
"version" : 1,
"author" : "xcode"
}
}

View File

@@ -0,0 +1,22 @@
{
"images" : [
{
"idiom" : "universal",
"filename" : "user-image.png",
"scale" : "1x"
},
{
"idiom" : "universal",
"filename" : "user-image@2x.png",
"scale" : "2x"
},
{
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"version" : 1,
"author" : "xcode"
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

View File

@@ -0,0 +1,25 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="13122.16" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES" initialViewController="01J-lp-oVM">
<dependencies>
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="13104.12"/>
<capability name="Safe area layout guides" minToolsVersion="9.0"/>
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
</dependencies>
<scenes>
<!--View Controller-->
<scene sceneID="EHf-IW-A2E">
<objects>
<viewController id="01J-lp-oVM" sceneMemberID="viewController">
<view key="view" contentMode="scaleToFill" id="Ze5-6b-2t3">
<rect key="frame" x="0.0" y="0.0" width="375" height="667"/>
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
<color key="backgroundColor" xcode11CocoaTouchSystemColor="systemBackgroundColor" cocoaTouchSystemColor="whiteColor"/>
<viewLayoutGuide key="safeArea" id="6Tk-OE-BBY"/>
</view>
</viewController>
<placeholder placeholderIdentifier="IBFirstResponder" id="iYj-Kq-Ea1" userLabel="First Responder" sceneMemberID="firstResponder"/>
</objects>
<point key="canvasLocation" x="53" y="375"/>
</scene>
</scenes>
</document>

View File

@@ -0,0 +1,62 @@
<?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>CFBundleDevelopmentRegion</key>
<string>$(DEVELOPMENT_LANGUAGE)</string>
<key>CFBundleExecutable</key>
<string>$(EXECUTABLE_NAME)</string>
<key>CFBundleIdentifier</key>
<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
<key>CFBundleInfoDictionaryVersion</key>
<string>6.0</string>
<key>CFBundleName</key>
<string>$(PRODUCT_NAME)</string>
<key>CFBundlePackageType</key>
<string>$(PRODUCT_BUNDLE_PACKAGE_TYPE)</string>
<key>CFBundleShortVersionString</key>
<string>1.0</string>
<key>CFBundleVersion</key>
<string>1</string>
<key>LSRequiresIPhoneOS</key>
<true/>
<key>UIApplicationSceneManifest</key>
<dict>
<key>UIApplicationSupportsMultipleScenes</key>
<false/>
<key>UISceneConfigurations</key>
<dict>
<key>UIWindowSceneSessionRoleApplication</key>
<array>
<dict>
<key>UILaunchStoryboardName</key>
<string>LaunchScreen</string>
<key>UISceneConfigurationName</key>
<string>Default Configuration</string>
<key>UISceneDelegateClassName</key>
<string>$(PRODUCT_MODULE_NAME).SceneDelegate</string>
</dict>
</array>
</dict>
</dict>
<key>UILaunchStoryboardName</key>
<string>LaunchScreen</string>
<key>UIRequiredDeviceCapabilities</key>
<array>
<string>armv7</string>
</array>
<key>UISupportedInterfaceOrientations</key>
<array>
<string>UIInterfaceOrientationPortrait</string>
<string>UIInterfaceOrientationLandscapeLeft</string>
<string>UIInterfaceOrientationLandscapeRight</string>
</array>
<key>UISupportedInterfaceOrientations~ipad</key>
<array>
<string>UIInterfaceOrientationPortrait</string>
<string>UIInterfaceOrientationPortraitUpsideDown</string>
<string>UIInterfaceOrientationLandscapeLeft</string>
<string>UIInterfaceOrientationLandscapeRight</string>
</array>
</dict>
</plist>

View File

@@ -0,0 +1,6 @@
{
"info" : {
"version" : 1,
"author" : "xcode"
}
}

View File

@@ -0,0 +1,24 @@
//
// SceneDelegate.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 04/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import UIKit
import SwiftUI
class SceneDelegate: UIResponder, UIWindowSceneDelegate {
var window: UIWindow?
func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) {
let window = UIWindow(frame: UIScreen.main.bounds)
window.rootViewController = UIHostingController(rootView: TabbarView().environmentObject(store))
self.window = window
window.makeKeyAndVisible()
}
}

View File

@@ -0,0 +1,13 @@
//
// Action.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import Foundation
protocol Action {
}

View File

@@ -0,0 +1,19 @@
//
// UsersAction.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import Foundation
enum UserActions: Action {
case addUser
case deleteUser(index: Int)
case move(from: Int, to: Int)
case editUser(id: Int, name: String, username: String)
case testEditFirstUser
case startEditUser
case stopEditUser
}

View File

@@ -0,0 +1,20 @@
//
// User.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 04/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import Foundation
import SwiftUI
struct User: Identifiable {
let id: Int
var name: String
var username: String
let imageName = "person"
}
let sampleData = [User(id: 0, name: "user 1", username: "@user1"),
User(id: 1, name: "user 2", username: "@user2")]

View File

@@ -0,0 +1,14 @@
//
// Reducer.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import Foundation
protocol Reducer {
associatedtype StateType: FluxState
func reduce(state: StateType, action: Action) -> StateType
}

View File

@@ -0,0 +1,42 @@
//
// UsersStateReducer.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import Foundation
struct UserStateReducer: Reducer {
func reduce(state: UsersState, action: Action) -> UsersState {
var state = state
switch action {
case UserActions.addUser:
state.users.append(User(id: state.users.count,
name: "New user \(state.users.count + 1)",
username: "@newuser\(state.users.count + 1)"))
case let UserActions.deleteUser(index):
state.users.remove(at: index)
case let UserActions.move(from, to):
let user = state.users.remove(at: from)
state.users.insert(user, at: to)
case let UserActions.editUser(id, name, username):
var user = state.users[id]
user.name = name
user.username = username
state.users[id] = user
case UserActions.testEditFirstUser:
if !state.users.isEmpty {
state.users[0] = User(id: 0, name: "user1", username: "u\ns\ne\nr\nn\na\nm\ne")
}
case UserActions.startEditUser:
state.isEditingUser = true
case UserActions.stopEditUser:
state.isEditingUser = false
default:
break
}
return state
}
}

View File

@@ -0,0 +1,34 @@
//
// AppStore.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import Foundation
import SwiftUI
import Combine
final class AppState: BindableObject {
var didChange = PassthroughSubject<AppState, Never>()
var usersState: UsersState
init(usersState: UsersState = UsersState()) {
self.usersState = usersState
}
func dispatch(action: Action) {
usersState = UserStateReducer().reduce(state: usersState, action: action)
didChange.send(self)
}
}
let store = AppState()
#if DEBUG
let sampleStore = AppState(usersState: UsersState(users: sampleData))
#endif

View File

@@ -0,0 +1,11 @@
//
// FluxState.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import Foundation
protocol FluxState { }

View File

@@ -0,0 +1,20 @@
//
// UsersStore.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 04/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import Foundation
import SwiftUI
import Combine
struct UsersState: FluxState {
var users: [User]
var isEditingUser = false
init(users: [User] = []) {
self.users = users
}
}

View File

@@ -0,0 +1,32 @@
//
// MainView.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import SwiftUI
struct TabbarView : View {
@EnvironmentObject var state: AppState
@State var selectedIndex: Int = 0
var body: some View {
TabbedView(selection: $selectedIndex) {
UsersListView()
.tabItemLabel(Text("Users"))
MapView()
.tabItemLabel(Text("Map"))
}
}
}
#if DEBUG
struct MainView_Previews : PreviewProvider {
static var previews: some View {
TabbarView(selectedIndex: 0).environmentObject(sampleStore)
}
}
#endif

View File

@@ -0,0 +1,36 @@
//
// MapVie.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import SwiftUI
import UIKit
import MapKit
struct MapViewRepresentable: UIViewRepresentable {
func makeUIView(context: Context) -> MKMapView {
return MKMapView()
}
func updateUIView(_ uiView: MKMapView, context: Context) {
}
}
struct MapView : View {
var body: some View {
MapViewRepresentable()
}
}
#if DEBUG
struct MapVie_Previews : PreviewProvider {
static var previews: some View {
MapView()
}
}
#endif

View File

@@ -0,0 +1,53 @@
//
// UserDetailView.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 04/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import SwiftUI
import Combine
struct UserDetailView : View {
@EnvironmentObject var state: AppState
let userId: Int
var editModal: Modal {
let user = state.usersState.users[userId]
return Modal(UserEditForm(userId: user.id, saveHandler: { saved in
self.state.dispatch(action: UserActions.stopEditUser)
}).environmentObject(state)) {
self.state.dispatch(action: UserActions.stopEditUser)
}
}
var body: some View {
let user = state.usersState.users[userId]
return VStack {
Image(systemName: user.imageName)
Text(user.name)
Text(user.username).lineLimit(0)
}
.navigationBarTitle(Text(user.name), displayMode: .inline)
.navigationBarItems(trailing:
Button(action: {
self.state.dispatch(action: UserActions.startEditUser)
}) {
Text("Edit user")
}
.presentation(self.state.usersState.isEditingUser ? self.editModal : nil))
}
}
#if DEBUG
struct UserDetailView_Previews : PreviewProvider {
static var previews: some View {
NavigationView {
UserDetailView(userId: 0).environmentObject(sampleStore)
}
}
}
#endif

View File

@@ -0,0 +1,88 @@
//
// UserEditForm.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 05/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import SwiftUI
struct UserEditForm : View {
@EnvironmentObject var state: AppState
let userId: Int
let saveHandler: ((Bool) -> Swift.Void)?
@State var newUserName = ""
@State var newUserUsername = ""
@State var showSaved = false
@State var showError = false
var body: some View {
let user = state.usersState.users[userId]
return NavigationView {
VStack(alignment: .leading, spacing: 10) {
Text("User name")
TextField($newUserName, placeholder: Text("New name"))
.textFieldStyle(.roundedBorder)
Divider()
Text("Username")
TextField($newUserUsername, placeholder: Text("New username"))
.textFieldStyle(.roundedBorder)
}.padding(16)
Button(action: save) {
Text("Save")
.padding(8)
.foregroundColor(.white)
.background(Color.green)
.cornerRadius(8)
}
.navigationBarItems(trailing: Button(action: close) {
Text("Close")
})
.navigationBarTitle(Text("Edit \(user.name)"), displayMode: .inline)
Badge(text: "Saved successfully", color: .green, show: $showSaved)
Badge(text: "Missing username or name", color: .red, show: $showError)
}
}
func save() {
guard !newUserName.isEmpty && !newUserUsername.isEmpty else {
withAnimation{
showError = true
}
DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) {
self.showError = false
}
return
}
withAnimation {
showSaved = true
}
DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) {
self.showSaved = false
}
state.dispatch(action: UserActions.editUser(id: userId, name: newUserName, username: newUserUsername))
saveHandler?(true)
}
func close() {
saveHandler?(false)
}
}
#if DEBUG
struct UserEditForm_Previews : PreviewProvider {
static var previews: some View {
UserEditForm(userId: 0, saveHandler: nil).environmentObject(sampleStore)
}
}
#endif

View File

@@ -0,0 +1,66 @@
//
// ContentView.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 04/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import SwiftUI
struct UsersListView : View {
@EnvironmentObject var state: AppState
var body: some View {
NavigationView {
List {
Section {
Button(action: addUser) {
Text("Add user")
}
Button(action: targetUpdate) {
Text("Update first")
}
}
Section {
ForEach(state.usersState.users) {user in
NavigationButton(destination: UserDetailView(userId: user.id)) {
UserRow(user: user)
}
}
.onDelete(perform: delete)
.onMove(perform: move)
}
}
.listStyle(.grouped)
.navigationBarTitle(Text("Users (\(state.usersState.users.count))"))
.navigationBarItems(trailing: EditButton())
}
}
func addUser() {
state.dispatch(action: UserActions.addUser)
}
func targetUpdate() {
state.dispatch(action: UserActions.testEditFirstUser)
}
func delete(at offset: IndexSet) {
state.dispatch(action: UserActions.deleteUser(index: offset.first!))
}
func move(from: IndexSet, to: Int) {
state.dispatch(action: UserActions.move(from: from.first!, to: to))
}
}
#if DEBUG
struct ContentView_Previews : PreviewProvider {
static var previews: some View {
UsersListView().environmentObject(sampleStore)
}
}
#endif

View File

@@ -0,0 +1,32 @@
//
// GreenBadge.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 06/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import SwiftUI
struct Badge : View {
let text: String
let color: Color
@Binding var show: Bool
var animation: Animation {
Animation
.spring(initialVelocity: 5)
.speed(2)
}
var body: some View {
Text(text)
.color(.white)
.padding()
.background(color)
.cornerRadius(8)
.scaleEffect(show ? 1: 0.5)
.opacity(show ? 1 : 0)
.animation(animation)
}
}

View File

@@ -0,0 +1,36 @@
//
// UserRw.swift
// SwiftUIDemo
//
// Created by Thomas Ricouard on 04/06/2019.
// Copyright © 2019 Thomas Ricouarf. All rights reserved.
//
import SwiftUI
struct UserRow : View {
let user: User
var body: some View {
VStack {
HStack {
Image(systemName: user.imageName)
VStack {
Text(user.name)
Text(user.username)
.color(.secondary)
.lineLimit(0)
}
}
}
}
}
#if DEBUG
struct UserRw_Previews : PreviewProvider {
static var previews: some View {
UserRow(user: sampleData[0])
}
}
#endif

View File

@@ -33,6 +33,7 @@ Also include:
- Movie
- InstaFake
- TempusRomanumII
- SwiftUI + Redux
## Projects