From 49d5e89a854adc673a56b2c1d18af83ab3bbbc12 Mon Sep 17 00:00:00 2001 From: Elliot Boschwitz Date: Fri, 12 Feb 2021 17:00:45 -0800 Subject: [PATCH] Major code refactor + bug fixes (#23) * Refactors to adhere to MVVM architecture (#14, #21) * Adds CI pipeline (#19) * Fixes bugs (#20) * README updates --- .github/workflows/ios.yml | 42 +++ LobeTests/Info.plist | 22 ++ LobeTests/LobeTests.swift | 33 +++ Lobe_iOS.xcodeproj/project.pbxproj | 233 +++++++++++++++-- Lobe_iOS/AppDelegate.swift | 8 + Lobe_iOS/CaptureSessionViewController.swift | 109 ++++++++ Lobe_iOS/ContentView.swift | 152 ----------- Lobe_iOS/ImagePicker.swift | 54 ---- Lobe_iOS/Info.plist | 21 +- Lobe_iOS/Lobe.entitlements | 16 ++ Lobe_iOS/Models/CaptureSessionManager.swift | 231 ++++++++++++++++ Lobe_iOS/Models/PredictionLayer.swift | 79 ++++++ Lobe_iOS/Models/Project.swift | 22 ++ Lobe_iOS/MyViewController.swift | 246 ------------------ Lobe_iOS/PlayViewModel.swift | 75 ++++++ Lobe_iOS/README.md | 38 +++ Lobe_iOS/SceneDelegate.swift | 17 +- Lobe_iOS/Views/CameraView.swift | 56 ++++ Lobe_iOS/Views/ImagePicker.swift | 61 +++++ Lobe_iOS/Views/ImagePreview.swift | 46 ++++ Lobe_iOS/Views/PlayView.swift | 190 ++++++++++++++ .../PredictionLabelView.swift} | 31 ++- Lobe_iOS/Views/README.md | 8 + README.md | 41 +-- assets/codeDiagram.png | Bin 0 -> 36174 bytes 25 files changed, 1309 insertions(+), 522 deletions(-) create mode 100644 .github/workflows/ios.yml create mode 100644 LobeTests/Info.plist create mode 100644 LobeTests/LobeTests.swift create mode 100644 Lobe_iOS/CaptureSessionViewController.swift delete mode 100644 Lobe_iOS/ContentView.swift delete mode 100644 Lobe_iOS/ImagePicker.swift create mode 100644 Lobe_iOS/Lobe.entitlements create mode 100644 Lobe_iOS/Models/CaptureSessionManager.swift create mode 100644 Lobe_iOS/Models/PredictionLayer.swift create mode 100644 Lobe_iOS/Models/Project.swift delete mode 100644 Lobe_iOS/MyViewController.swift create mode 100644 Lobe_iOS/PlayViewModel.swift create mode 100644 Lobe_iOS/README.md create mode 100644 Lobe_iOS/Views/CameraView.swift create mode 100644 Lobe_iOS/Views/ImagePicker.swift create mode 100644 Lobe_iOS/Views/ImagePreview.swift create mode 100644 Lobe_iOS/Views/PlayView.swift rename Lobe_iOS/{UpdateTextViewExternal.swift => Views/PredictionLabelView.swift} (72%) create mode 100644 Lobe_iOS/Views/README.md create mode 100644 assets/codeDiagram.png diff --git a/.github/workflows/ios.yml b/.github/workflows/ios.yml new file mode 100644 index 0000000..982c807 --- /dev/null +++ b/.github/workflows/ios.yml @@ -0,0 +1,42 @@ +name: App-CI + +on: + push: + branches: [ u/elbosc/main ] + pull_request: + branches: [ u/elbosc/main ] + +jobs: + build: + name: Build and Test Default Scheme + runs-on: macos-11.0 + + steps: + - name: Checkout + uses: actions/checkout@v2 + - name: Set Default Scheme + run: | + scheme_list=$(xcodebuild -list -json | tr -d "\n") + default=$(echo $scheme_list | ruby -e "require 'json'; puts JSON.parse(STDIN.gets)['project']['targets'][0]") + echo $default | cat >default + echo Using default scheme: $default + - name: Build + env: + scheme: ${{ 'default' }} + platform: ${{ 'iOS Simulator' }} + run: | + device=`instruments -s -devices | grep -oE 'iPhone.*?[^\(]+' | head -1 | awk '{$1=$1;print}'` + if [ $scheme = default ]; then scheme=$(cat default); fi + if [ "`ls -A | grep -i \\.xcworkspace\$`" ]; then filetype_parameter="workspace" && file_to_build="`ls -A | grep -i \\.xcworkspace\$`"; else filetype_parameter="project" && file_to_build="`ls -A | grep -i \\.xcodeproj\$`"; fi + file_to_build=`echo $file_to_build | awk '{$1=$1;print}'` + xcodebuild build-for-testing -scheme "$scheme" -"$filetype_parameter" "$file_to_build" -destination "platform=$platform,name=$device" + - name: Test + env: + scheme: ${{ 'default' }} + platform: ${{ 'iOS Simulator' }} + run: | + device=`instruments -s -devices | grep -oE 'iPhone.*?[^\(]+' | head -1 | awk '{$1=$1;print}'` + if [ $scheme = default ]; then scheme=$(cat default); fi + if [ "`ls -A | grep -i \\.xcworkspace\$`" ]; then filetype_parameter="workspace" && file_to_build="`ls -A | grep -i \\.xcworkspace\$`"; else filetype_parameter="project" && file_to_build="`ls -A | grep -i \\.xcodeproj\$`"; fi + file_to_build=`echo $file_to_build | awk '{$1=$1;print}'` + xcodebuild test-without-building -scheme "$scheme" -"$filetype_parameter" "$file_to_build" -destination "platform=$platform,name=$device" diff --git a/LobeTests/Info.plist b/LobeTests/Info.plist new file mode 100644 index 0000000..64d65ca --- /dev/null +++ b/LobeTests/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + + diff --git a/LobeTests/LobeTests.swift b/LobeTests/LobeTests.swift new file mode 100644 index 0000000..c459aa7 --- /dev/null +++ b/LobeTests/LobeTests.swift @@ -0,0 +1,33 @@ +// +// LobeTests.swift +// LobeTests +// +// Created by Elliot Boschwitz on 12/8/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import XCTest + +class LobeTests: 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. + } + + func testPerformanceExample() throws { + // This is an example of a performance test case. + measure { + // Put the code you want to measure the time of here. + } + } + +} diff --git a/Lobe_iOS.xcodeproj/project.pbxproj b/Lobe_iOS.xcodeproj/project.pbxproj index 3df972b..9ab55d0 100644 --- a/Lobe_iOS.xcodeproj/project.pbxproj +++ b/Lobe_iOS.xcodeproj/project.pbxproj @@ -9,33 +9,70 @@ /* Begin PBXBuildFile section */ 76A9AF54247F138B002086F7 /* LobeModel.mlmodel in Sources */ = {isa = PBXBuildFile; fileRef = 76A9AF53247F138B002086F7 /* LobeModel.mlmodel */; }; 76CDE1A4247F2E6D0096E882 /* ImagePicker.swift in Sources */ = {isa = PBXBuildFile; fileRef = 76CDE1A3247F2E6D0096E882 /* ImagePicker.swift */; }; - 76ED73F3248987E3003B4F6B /* MyViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 76ED73F2248987E3003B4F6B /* MyViewController.swift */; }; - 76ED73F5248988AE003B4F6B /* UpdateTextViewExternal.swift in Sources */ = {isa = PBXBuildFile; fileRef = 76ED73F4248988AE003B4F6B /* UpdateTextViewExternal.swift */; }; + 76ED73F3248987E3003B4F6B /* CaptureSessionViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 76ED73F2248987E3003B4F6B /* CaptureSessionViewController.swift */; }; + 76ED73F5248988AE003B4F6B /* PredictionLabelView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 76ED73F4248988AE003B4F6B /* PredictionLabelView.swift */; }; 9B7B03EB2475DBF300D34020 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9B7B03EA2475DBF300D34020 /* AppDelegate.swift */; }; 9B7B03ED2475DBF300D34020 /* SceneDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9B7B03EC2475DBF300D34020 /* SceneDelegate.swift */; }; - 9B7B03EF2475DBF300D34020 /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9B7B03EE2475DBF300D34020 /* ContentView.swift */; }; + 9B7B03EF2475DBF300D34020 /* PlayView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9B7B03EE2475DBF300D34020 /* PlayView.swift */; }; 9B7B03F12475DBF600D34020 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 9B7B03F02475DBF600D34020 /* Assets.xcassets */; }; 9B7B03F42475DBF600D34020 /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 9B7B03F32475DBF600D34020 /* Preview Assets.xcassets */; }; 9B7B03F72475DBF600D34020 /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 9B7B03F52475DBF600D34020 /* LaunchScreen.storyboard */; }; + B513D84525809D62009D91E2 /* LobeTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = B513D84425809D62009D91E2 /* LobeTests.swift */; }; + B5173D5725774E990065A01F /* PlayViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = B5173D5625774E990065A01F /* PlayViewModel.swift */; }; + B5173D5B25774F400065A01F /* ImagePreview.swift in Sources */ = {isa = PBXBuildFile; fileRef = B5173D5A25774F400065A01F /* ImagePreview.swift */; }; + B53C73762533C150007A0231 /* Project.swift in Sources */ = {isa = PBXBuildFile; fileRef = B53C73752533C150007A0231 /* Project.swift */; }; + B55A1ED32574F368003EE8AD /* CameraView.swift in Sources */ = {isa = PBXBuildFile; fileRef = B55A1ED22574F368003EE8AD /* CameraView.swift */; }; + B55A1ED825760BAC003EE8AD /* PredictionLayer.swift in Sources */ = {isa = PBXBuildFile; fileRef = B55A1ED725760BAC003EE8AD /* PredictionLayer.swift */; }; + B55D41462599BB48007C9DBF /* CaptureSessionManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = B55D41452599BB48007C9DBF /* CaptureSessionManager.swift */; }; + B5A9B5272557373E00FD595B /* CloudKit.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = B5A9B5262557373E00FD595B /* CloudKit.framework */; }; /* End PBXBuildFile section */ +/* Begin PBXContainerItemProxy section */ + B513D84725809D62009D91E2 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 9B7B03DF2475DBF300D34020 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 9B7B03E62475DBF300D34020; + remoteInfo = Lobe_iOS; + }; +/* End PBXContainerItemProxy section */ + /* Begin PBXFileReference section */ 76A9AF53247F138B002086F7 /* LobeModel.mlmodel */ = {isa = PBXFileReference; lastKnownFileType = file.mlmodel; name = LobeModel.mlmodel; path = ../LobeModel.mlmodel; sourceTree = ""; }; 76CDE1A3247F2E6D0096E882 /* ImagePicker.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImagePicker.swift; sourceTree = ""; }; - 76ED73F2248987E3003B4F6B /* MyViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MyViewController.swift; sourceTree = ""; }; - 76ED73F4248988AE003B4F6B /* UpdateTextViewExternal.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UpdateTextViewExternal.swift; sourceTree = ""; }; + 76ED73F2248987E3003B4F6B /* CaptureSessionViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CaptureSessionViewController.swift; sourceTree = ""; }; + 76ED73F4248988AE003B4F6B /* PredictionLabelView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PredictionLabelView.swift; sourceTree = ""; }; 9B7B03E72475DBF300D34020 /* Lobe.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Lobe.app; sourceTree = BUILT_PRODUCTS_DIR; }; 9B7B03EA2475DBF300D34020 /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; 9B7B03EC2475DBF300D34020 /* SceneDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SceneDelegate.swift; sourceTree = ""; }; - 9B7B03EE2475DBF300D34020 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = ""; }; + 9B7B03EE2475DBF300D34020 /* PlayView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayView.swift; sourceTree = ""; }; 9B7B03F02475DBF600D34020 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; 9B7B03F32475DBF600D34020 /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = ""; }; 9B7B03F62475DBF600D34020 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; }; 9B7B03F82475DBF600D34020 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; + B513D84225809D62009D91E2 /* LobeTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = LobeTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + B513D84425809D62009D91E2 /* LobeTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LobeTests.swift; sourceTree = ""; }; + B513D84625809D62009D91E2 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; + B5173D5625774E990065A01F /* PlayViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayViewModel.swift; sourceTree = ""; }; + B5173D5A25774F400065A01F /* ImagePreview.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImagePreview.swift; sourceTree = ""; }; + B53C73752533C150007A0231 /* Project.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Project.swift; sourceTree = ""; }; + B55A1ED22574F368003EE8AD /* CameraView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CameraView.swift; sourceTree = ""; }; + B55A1ED725760BAC003EE8AD /* PredictionLayer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PredictionLayer.swift; sourceTree = ""; }; + B55D41452599BB48007C9DBF /* CaptureSessionManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CaptureSessionManager.swift; sourceTree = ""; }; + B5A9B5232557372800FD595B /* Lobe.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = Lobe.entitlements; sourceTree = ""; }; + B5A9B5262557373E00FD595B /* CloudKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = CloudKit.framework; path = System/Library/Frameworks/CloudKit.framework; sourceTree = SDKROOT; }; /* End PBXFileReference section */ /* Begin PBXFrameworksBuildPhase section */ 9B7B03E42475DBF300D34020 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + B5A9B5272557373E00FD595B /* CloudKit.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + B513D83F25809D62009D91E2 /* Frameworks */ = { isa = PBXFrameworksBuildPhase; buildActionMask = 2147483647; files = ( @@ -49,7 +86,9 @@ isa = PBXGroup; children = ( 9B7B03E92475DBF300D34020 /* Lobe_iOS */, + B513D84325809D62009D91E2 /* LobeTests */, 9B7B03E82475DBF300D34020 /* Products */, + B5A9B5252557373E00FD595B /* Frameworks */, ); sourceTree = ""; }; @@ -57,6 +96,7 @@ isa = PBXGroup; children = ( 9B7B03E72475DBF300D34020 /* Lobe.app */, + B513D84225809D62009D91E2 /* LobeTests.xctest */, ); name = Products; sourceTree = ""; @@ -64,14 +104,15 @@ 9B7B03E92475DBF300D34020 /* Lobe_iOS */ = { isa = PBXGroup; children = ( + B56240FA257ED11B00709520 /* Views */, + B55C4E1325D4F7820075F68B /* Models */, + B5173D5625774E990065A01F /* PlayViewModel.swift */, + 76ED73F2248987E3003B4F6B /* CaptureSessionViewController.swift */, + B5A9B5232557372800FD595B /* Lobe.entitlements */, 9B7B03EA2475DBF300D34020 /* AppDelegate.swift */, + 9B7B03F52475DBF600D34020 /* LaunchScreen.storyboard */, 9B7B03EC2475DBF300D34020 /* SceneDelegate.swift */, - 9B7B03EE2475DBF300D34020 /* ContentView.swift */, - 76ED73F4248988AE003B4F6B /* UpdateTextViewExternal.swift */, - 76ED73F2248987E3003B4F6B /* MyViewController.swift */, - 76CDE1A3247F2E6D0096E882 /* ImagePicker.swift */, 9B7B03F02475DBF600D34020 /* Assets.xcassets */, - 9B7B03F52475DBF600D34020 /* LaunchScreen.storyboard */, 76A9AF53247F138B002086F7 /* LobeModel.mlmodel */, 9B7B03F82475DBF600D34020 /* Info.plist */, 9B7B03F22475DBF600D34020 /* Preview Content */, @@ -87,6 +128,45 @@ path = "Preview Content"; sourceTree = ""; }; + B513D84325809D62009D91E2 /* LobeTests */ = { + isa = PBXGroup; + children = ( + B513D84425809D62009D91E2 /* LobeTests.swift */, + B513D84625809D62009D91E2 /* Info.plist */, + ); + path = LobeTests; + sourceTree = ""; + }; + B55C4E1325D4F7820075F68B /* Models */ = { + isa = PBXGroup; + children = ( + B55D41452599BB48007C9DBF /* CaptureSessionManager.swift */, + B55A1ED725760BAC003EE8AD /* PredictionLayer.swift */, + B53C73752533C150007A0231 /* Project.swift */, + ); + path = Models; + sourceTree = ""; + }; + B56240FA257ED11B00709520 /* Views */ = { + isa = PBXGroup; + children = ( + 9B7B03EE2475DBF300D34020 /* PlayView.swift */, + 76ED73F4248988AE003B4F6B /* PredictionLabelView.swift */, + 76CDE1A3247F2E6D0096E882 /* ImagePicker.swift */, + B55A1ED22574F368003EE8AD /* CameraView.swift */, + B5173D5A25774F400065A01F /* ImagePreview.swift */, + ); + path = Views; + sourceTree = ""; + }; + B5A9B5252557373E00FD595B /* Frameworks */ = { + isa = PBXGroup; + children = ( + B5A9B5262557373E00FD595B /* CloudKit.framework */, + ); + name = Frameworks; + sourceTree = ""; + }; /* End PBXGroup section */ /* Begin PBXNativeTarget section */ @@ -107,19 +187,41 @@ productReference = 9B7B03E72475DBF300D34020 /* Lobe.app */; productType = "com.apple.product-type.application"; }; + B513D84125809D62009D91E2 /* LobeTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = B513D84B25809D62009D91E2 /* Build configuration list for PBXNativeTarget "LobeTests" */; + buildPhases = ( + B513D83E25809D62009D91E2 /* Sources */, + B513D83F25809D62009D91E2 /* Frameworks */, + B513D84025809D62009D91E2 /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + B513D84825809D62009D91E2 /* PBXTargetDependency */, + ); + name = LobeTests; + productName = LobeTests; + productReference = B513D84225809D62009D91E2 /* LobeTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; /* End PBXNativeTarget section */ /* Begin PBXProject section */ 9B7B03DF2475DBF300D34020 /* Project object */ = { isa = PBXProject; attributes = { - LastSwiftUpdateCheck = 1140; - LastUpgradeCheck = 1140; - ORGANIZATIONNAME = "Adam Menges"; + LastSwiftUpdateCheck = 1220; + LastUpgradeCheck = 1200; + ORGANIZATIONNAME = Microsoft; TargetAttributes = { 9B7B03E62475DBF300D34020 = { CreatedOnToolsVersion = 11.4.1; }; + B513D84125809D62009D91E2 = { + CreatedOnToolsVersion = 12.2; + TestTargetID = 9B7B03E62475DBF300D34020; + }; }; }; buildConfigurationList = 9B7B03E22475DBF300D34020 /* Build configuration list for PBXProject "Lobe_iOS" */; @@ -136,6 +238,7 @@ projectRoot = ""; targets = ( 9B7B03E62475DBF300D34020 /* Lobe_iOS */, + B513D84125809D62009D91E2 /* LobeTests */, ); }; /* End PBXProject section */ @@ -151,6 +254,13 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + B513D84025809D62009D91E2 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXResourcesBuildPhase section */ /* Begin PBXSourcesBuildPhase section */ @@ -158,18 +268,40 @@ isa = PBXSourcesBuildPhase; buildActionMask = 2147483647; files = ( + B55D41462599BB48007C9DBF /* CaptureSessionManager.swift in Sources */, 76A9AF54247F138B002086F7 /* LobeModel.mlmodel in Sources */, 9B7B03EB2475DBF300D34020 /* AppDelegate.swift in Sources */, + B5173D5B25774F400065A01F /* ImagePreview.swift in Sources */, + B55A1ED32574F368003EE8AD /* CameraView.swift in Sources */, 76CDE1A4247F2E6D0096E882 /* ImagePicker.swift in Sources */, 9B7B03ED2475DBF300D34020 /* SceneDelegate.swift in Sources */, - 76ED73F5248988AE003B4F6B /* UpdateTextViewExternal.swift in Sources */, - 76ED73F3248987E3003B4F6B /* MyViewController.swift in Sources */, - 9B7B03EF2475DBF300D34020 /* ContentView.swift in Sources */, + 76ED73F5248988AE003B4F6B /* PredictionLabelView.swift in Sources */, + 76ED73F3248987E3003B4F6B /* CaptureSessionViewController.swift in Sources */, + B55A1ED825760BAC003EE8AD /* PredictionLayer.swift in Sources */, + B5173D5725774E990065A01F /* PlayViewModel.swift in Sources */, + 9B7B03EF2475DBF300D34020 /* PlayView.swift in Sources */, + B53C73762533C150007A0231 /* Project.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + B513D83E25809D62009D91E2 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + B513D84525809D62009D91E2 /* LobeTests.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; /* End PBXSourcesBuildPhase section */ +/* Begin PBXTargetDependency section */ + B513D84825809D62009D91E2 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 9B7B03E62475DBF300D34020 /* Lobe_iOS */; + targetProxy = B513D84725809D62009D91E2 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + /* Begin PBXVariantGroup section */ 9B7B03F52475DBF600D34020 /* LaunchScreen.storyboard */ = { isa = PBXVariantGroup; @@ -208,6 +340,7 @@ 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; @@ -268,6 +401,7 @@ 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; @@ -300,17 +434,20 @@ isa = XCBuildConfiguration; buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CODE_SIGN_ENTITLEMENTS = Lobe_iOS/Lobe.entitlements; + CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; DEVELOPMENT_ASSET_PATHS = "\"Lobe_iOS/Preview Content\""; - DEVELOPMENT_TEAM = NGV85M44ZU; + DEVELOPMENT_TEAM = UBF8T346G9; ENABLE_PREVIEWS = YES; INFOPLIST_FILE = Lobe_iOS/Info.plist; LD_RUNPATH_SEARCH_PATHS = ( "$(inherited)", "@executable_path/Frameworks", ); - PRODUCT_BUNDLE_IDENTIFIER = "adammenges.Lobe-iOS"; + PRODUCT_BUNDLE_IDENTIFIER = com.microsoft.lobe; PRODUCT_NAME = Lobe; + PROVISIONING_PROFILE_SPECIFIER = ""; SWIFT_VERSION = 5.0; TARGETED_DEVICE_FAMILY = "1,2"; }; @@ -320,19 +457,64 @@ isa = XCBuildConfiguration; buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CODE_SIGN_ENTITLEMENTS = Lobe_iOS/Lobe.entitlements; + CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; DEVELOPMENT_ASSET_PATHS = "\"Lobe_iOS/Preview Content\""; - DEVELOPMENT_TEAM = NGV85M44ZU; + DEVELOPMENT_TEAM = UBF8T346G9; ENABLE_PREVIEWS = YES; INFOPLIST_FILE = Lobe_iOS/Info.plist; LD_RUNPATH_SEARCH_PATHS = ( "$(inherited)", "@executable_path/Frameworks", ); - PRODUCT_BUNDLE_IDENTIFIER = "adammenges.Lobe-iOS"; + PRODUCT_BUNDLE_IDENTIFIER = com.microsoft.lobe; PRODUCT_NAME = Lobe; + PROVISIONING_PROFILE_SPECIFIER = ""; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Release; + }; + B513D84925809D62009D91E2 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CODE_SIGN_STYLE = Automatic; + DEVELOPMENT_TEAM = UBF8T346G9; + INFOPLIST_FILE = LobeTests/Info.plist; + IPHONEOS_DEPLOYMENT_TARGET = 14.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@loader_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = test.LobeTests; + PRODUCT_NAME = "$(TARGET_NAME)"; SWIFT_VERSION = 5.0; TARGETED_DEVICE_FAMILY = "1,2"; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Lobe.app/Lobe"; + }; + name = Debug; + }; + B513D84A25809D62009D91E2 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CODE_SIGN_STYLE = Automatic; + DEVELOPMENT_TEAM = UBF8T346G9; + INFOPLIST_FILE = LobeTests/Info.plist; + IPHONEOS_DEPLOYMENT_TARGET = 14.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@loader_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = test.LobeTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Lobe.app/Lobe"; }; name = Release; }; @@ -357,6 +539,15 @@ defaultConfigurationIsVisible = 0; defaultConfigurationName = Release; }; + B513D84B25809D62009D91E2 /* Build configuration list for PBXNativeTarget "LobeTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + B513D84925809D62009D91E2 /* Debug */, + B513D84A25809D62009D91E2 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; /* End XCConfigurationList section */ }; rootObject = 9B7B03DF2475DBF300D34020 /* Project object */; diff --git a/Lobe_iOS/AppDelegate.swift b/Lobe_iOS/AppDelegate.swift index 5c0ffa2..833f72f 100644 --- a/Lobe_iOS/AppDelegate.swift +++ b/Lobe_iOS/AppDelegate.swift @@ -1,3 +1,11 @@ +// +// AppDelegate.swift +// Lobe_iOS +// +// Created by Adam Menges on 5/20/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + import UIKit @UIApplicationMain diff --git a/Lobe_iOS/CaptureSessionViewController.swift b/Lobe_iOS/CaptureSessionViewController.swift new file mode 100644 index 0000000..761bd4f --- /dev/null +++ b/Lobe_iOS/CaptureSessionViewController.swift @@ -0,0 +1,109 @@ +// +// CaptureSessionViewController.swift +// Lobe_iOS +// +// Created by Kathy Zhou on 6/4/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import AVKit +import Foundation + +/// Defines tap gesture delegate protocol. +protocol CaptureSessionGestureDelegate { + func viewRecognizedDoubleTap() + func viewRecognizedTripleTap(_ view: UIView) +} + +/// View controller for video capture session. It's responsibilities include: +/// 1. Setting camera output to UI view. +/// 2. Handling orientation changes. +/// 3. Managing tap gestures. +class CaptureSessionViewController: UIViewController { + var previewLayer: AVCaptureVideoPreviewLayer? + var tripleTapGesture: UITapGestureRecognizer? + var doubleTapGesture: UITapGestureRecognizer? + var gestureDelegate: CaptureSessionGestureDelegate? + + override func viewDidLoad() { + super.viewDidLoad() + + /// Define gesture event listeners. We don't use SwiftUI since there isn't support for + /// recognizing a double tap gesture when a triple tap gesture is also present. + let doubleTapGesture = UITapGestureRecognizer(target: self, action:#selector(self.handleDoubleTap(_:))) + doubleTapGesture.numberOfTapsRequired = 2 + view.addGestureRecognizer(doubleTapGesture) + + let tripleTapGesture = UITapGestureRecognizer(target: self, action:#selector(self.handleTripleTap(_:))) + tripleTapGesture.numberOfTapsRequired = 3 + view.addGestureRecognizer(tripleTapGesture) + doubleTapGesture.require(toFail: tripleTapGesture) + } + + /// Set video configuration for subview layout + override func viewDidLayoutSubviews() { + super.viewDidLayoutSubviews() + self.configureVideoOrientation(for: self.previewLayer) + } + + /// Update video configuration when device orientation changes + override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) { + super.viewWillTransition(to: size, with: coordinator) + self.configureVideoOrientation(for: self.previewLayer) + } + + /// Configures orientation of preview layer for AVCapture session. + func configureVideoOrientation(for previewLayer: AVCaptureVideoPreviewLayer?) { + if let preview = previewLayer, + let connection = preview.connection { + let orientation = UIDevice.current.orientation + + if connection.isVideoOrientationSupported { + var videoOrientation: AVCaptureVideoOrientation + + switch orientation { + case .portrait: + videoOrientation = .portrait + case .portraitUpsideDown: + videoOrientation = .portraitUpsideDown + case .landscapeLeft: + videoOrientation = .landscapeRight + case .landscapeRight: + videoOrientation = .landscapeLeft + default: + videoOrientation = UIApplication.shared.windows.first?.windowScene?.interfaceOrientation.asAVCaptureVideoOrientation() ?? .portrait + } + connection.videoOrientation = videoOrientation + } + preview.frame = self.view.bounds + } + } + + /// Double tap flips camera. + @objc func handleDoubleTap(_ sender: UITapGestureRecognizer? = nil) { + self.gestureDelegate?.viewRecognizedDoubleTap() + } + + /// Triple tap creates screen shot. + @objc func handleTripleTap(_ sender: UITapGestureRecognizer? = nil) { + self.gestureDelegate?.viewRecognizedTripleTap(self.view) + } +} + +/// Conversion helper for AVCaptureSession orientation changes. +extension UIInterfaceOrientation { + func asAVCaptureVideoOrientation() -> AVCaptureVideoOrientation { + switch self { + case .portrait: + return .portrait + case .landscapeLeft: + return .landscapeLeft + case .landscapeRight: + return .landscapeRight + case .portraitUpsideDown: + return .portraitUpsideDown + default: + return .portrait + } + } +} diff --git a/Lobe_iOS/ContentView.swift b/Lobe_iOS/ContentView.swift deleted file mode 100644 index 80ed66d..0000000 --- a/Lobe_iOS/ContentView.swift +++ /dev/null @@ -1,152 +0,0 @@ -import SwiftUI -import AVKit -import Vision - -var useCamera: Bool = true - -struct ContentView: View { - - var controller: MyViewController = MyViewController() - @State var showImagePicker: Bool = false - @State private var image: UIImage? - @State var scaling: CGSize = .init(width: 1, height: 1) - @State private var offset = CGSize.zero - - var body: some View { - GeometryReader { geometry in - - VStack { - if (self.image != nil) { - /* Placeholder for displaying an image from the photo library. */ - Image(uiImage: self.image!) - .resizable() - .aspectRatio(self.image!.size, contentMode: .fill) - .scaleEffect(1 / self.scaling.height) - .offset(self.offset) - /* Gesture for swiping down to dismiss the image. */ - .gesture(DragGesture() - .onChanged ({value in - self.scaling = value.translation - self.scaling.height = max(self.scaling.height / 50, 1) - self.offset = value.translation - }) - .onEnded {_ in - self.offset = .zero - if self.scaling.height > 1.5 { - self.image = nil - useCamera = true - self.controller.changeStatus(useCam: useCamera, img: self.controller.camImage!) - } - self.scaling = .init(width: 1, height: 1) - } - ) - .opacity(1 / self.scaling.height < 1 ? 0.5: 1) - } else { - /* Background camera. */ - MyRepresentable(controller: self.controller) - /* Gesture for swiping up the photo library. */ - .gesture( - DragGesture() - .onEnded {value in - if value.translation.height < 0 { - withAnimation{ - self.showImagePicker = true - } - self.controller.changeStatus(useCam: false, img: self.controller.camImage!) - } - } - ) - } - } - .frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: .infinity) - .background(Color.black) - .edgesIgnoringSafeArea(.all) - - HStack { - Spacer() - /* Icon for closing the image*/ - Image("x") - .resizable() - .opacity(self.image != nil ? 1: 0) - .frame(width: geometry.size.width / 15, height: geometry.size.width / 15) - .onTapGesture { - self.image = nil - useCamera = true - self.controller.changeStatus(useCam: useCamera, img: self.controller.camImage!) - } - }.padding() - - VStack { - Spacer() - UpdateTextViewExternal(viewModel: self.controller) - HStack { - - /* Button for openning the photo library. */ - Button(action: { - withAnimation { - self.showImagePicker = true - } - self.controller.changeStatus(useCam: false, img: self.controller.camImage!) - }) { - Image("PhotoLib") - .renderingMode(.original) - .frame(width: geometry.size.width / 3, height: geometry.size.height / 16) - }.opacity(0) // not displaying the button - - /* button for taking screenshot. */ - Button(action: { - self.controller.screenShotMethod() - }) { - Image("Button") - .renderingMode(.original) - .frame(width: geometry.size.width / 3, height: geometry.size.width / 9) - }.opacity(0) // not displaying the button - - /* button for flipping the camera. */ - Button(action: { - self.controller.flipCamera() - }) { - Image("Swap") - .renderingMode(.original) - .frame(width: geometry.size.width / 3, height: geometry.size.height / 16) - }.opacity(0) // not displaying the button - } - .frame(width: geometry.size.width, - height: geometry.size.height / 30, alignment: .bottom) - .opacity(self.image == nil ? 1: 0) // hide the buttons when displaying an image from the photo library - } - - ImagePicker(image: self.$image, isShown: self.$showImagePicker, controller: self.controller, sourceType: .photoLibrary) - .edgesIgnoringSafeArea(.all) - .offset(x: 0, y: self.showImagePicker ? 0: UIApplication.shared.keyWindow?.frame.height ?? 0) - }.statusBar(hidden: true) - } - -} - - -/* Gadget to build colors from Hashtag Color Code Hex. */ -extension UIColor { - convenience init(red: Int, green: Int, blue: Int) { - assert(red >= 0 && red <= 255, "Invalid red component") - assert(green >= 0 && green <= 255, "Invalid green component") - assert(blue >= 0 && blue <= 255, "Invalid blue component") - - self.init(red: CGFloat(red) / 255.0, green: CGFloat(green) / 255.0, blue: CGFloat(blue) / 255.0, alpha: 1.0) - } - - convenience init(rgb: Int) { - self.init( - red: (rgb >> 16) & 0xFF, - green: (rgb >> 8) & 0xFF, - blue: rgb & 0xFF - ) - } - -} - -struct ContentView_Previews: PreviewProvider { - static var previews: some View { - ContentView() - } -} diff --git a/Lobe_iOS/ImagePicker.swift b/Lobe_iOS/ImagePicker.swift deleted file mode 100644 index 4c362a4..0000000 --- a/Lobe_iOS/ImagePicker.swift +++ /dev/null @@ -1,54 +0,0 @@ -import Foundation -import SwiftUI - -class ImagePickerCoordinator: NSObject, UINavigationControllerDelegate, UIImagePickerControllerDelegate { - - @Binding var image: UIImage? - @Binding var isShown: Bool - var controller: MyViewController - - init(image: Binding, isShown: Binding, controller: MyViewController) { - _image = image - _isShown = isShown - self.controller = controller - } - func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey : Any]) { - if let uiImage = info[UIImagePickerController.InfoKey.originalImage] as? UIImage { - image = uiImage - isShown = false - useCamera = false - controller.changeStatus(useCam: false, img: uiImage) - } - } - func imagePickerControllerDidCancel(_ picker: UIImagePickerController) { - isShown = false - self.controller.changeStatus(useCam: true, img: self.controller.camImage!) - } - -} - -/* Image picker. */ -struct ImagePicker: UIViewControllerRepresentable { - - typealias UIViewControllerType = UIImagePickerController - typealias Coordinator = ImagePickerCoordinator - - @Binding var image: UIImage? - @Binding var isShown: Bool - var controller: MyViewController - var sourceType: UIImagePickerController.SourceType = .camera - - func updateUIViewController(_ uiViewController: UIImagePickerController, context: UIViewControllerRepresentableContext) { - } - func makeCoordinator() -> ImagePicker.Coordinator { - return ImagePickerCoordinator(image: $image, isShown: $isShown, controller: controller) - } - func makeUIViewController(context: UIViewControllerRepresentableContext) -> UIImagePickerController { - let picker = UIImagePickerController() - picker.sourceType = sourceType - picker.delegate = context.coordinator - picker.modalPresentationStyle = .fullScreen - return picker - } - -} diff --git a/Lobe_iOS/Info.plist b/Lobe_iOS/Info.plist index 8d55f5b..1fbb014 100644 --- a/Lobe_iOS/Info.plist +++ b/Lobe_iOS/Info.plist @@ -2,14 +2,6 @@ - UIAppFonts - - labgrotesque-bold.ttf - - NSPhotoLibraryAddUsageDescription - To save screenshot - NSCameraUsageDescription - This app needs to access the camera CFBundleDevelopmentRegion $(DEVELOPMENT_LANGUAGE) CFBundleExecutable @@ -28,6 +20,19 @@ 1 LSRequiresIPhoneOS + NSAppTransportSecurity + + NSAllowsArbitraryLoads + + + NSCameraUsageDescription + This app needs to access the camera + NSPhotoLibraryAddUsageDescription + To save screenshot + UIAppFonts + + labgrotesque-bold.ttf + UIApplicationSceneManifest UIApplicationSupportsMultipleScenes diff --git a/Lobe_iOS/Lobe.entitlements b/Lobe_iOS/Lobe.entitlements new file mode 100644 index 0000000..1d1012c --- /dev/null +++ b/Lobe_iOS/Lobe.entitlements @@ -0,0 +1,16 @@ + + + + + aps-environment + development + com.apple.developer.icloud-container-identifiers + + com.apple.developer.icloud-services + + CloudDocuments + + com.apple.developer.ubiquity-container-identifiers + + + diff --git a/Lobe_iOS/Models/CaptureSessionManager.swift b/Lobe_iOS/Models/CaptureSessionManager.swift new file mode 100644 index 0000000..b8d843f --- /dev/null +++ b/Lobe_iOS/Models/CaptureSessionManager.swift @@ -0,0 +1,231 @@ +// +// CaptureSessionViewModel.swift +// Lobe_iOS +// +// Created by Elliot Boschwitz on 12/27/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import AVKit +import Combine +import SwiftUI +import VideoToolbox + +/// View model for camera view. +class CaptureSessionManager: NSObject { + @Published var previewLayer: AVCaptureVideoPreviewLayer? + @Published var capturedImageOutput: UIImage? + var captureSession: AVCaptureSession? + private var backCam = AVCaptureDevice.DiscoverySession(deviceTypes: [.builtInWideAngleCamera], mediaType: AVMediaType.video, position: .back).devices.first + private var frontCam = AVCaptureDevice.DiscoverySession(deviceTypes: [.builtInWideAngleCamera], mediaType: AVMediaType.video, position: .front).devices.first + private var dataOutput: AVCaptureVideoDataOutput? + private var captureDevice: AVCaptureDevice? + private var disposables = Set() + private var totalFrameCount = 0 + + override init() { + self.captureDevice = self.backCam + } + + /// Resets camera feed, which: + /// 1. Creates capture session for specified device. + /// 2. Creates preview layer. + /// 3. Creates new video data output. + /// 4. Starts capture session. + func resetCameraFeed() { + guard let captureDevice = self.captureDevice else { + print("No capture device found on reset camera feed.") + return + } + /// Tear down existing capture session to remove output for buffer delegate. + self.captureSession = nil + self.dataOutput = nil + + /// Create new capture session and preview layer. + let captureSession = self.createCaptureSession(for: captureDevice) + let previewLayer = self.createPreviewLayer(for: captureSession) + let dataOutput = AVCaptureVideoDataOutput() + + /// Set delegate of video output buffer to self. + dataOutput.setSampleBufferDelegate(self, queue: DispatchQueue(label: "videoQueue")) + captureSession.startRunning() + captureSession.addOutput(dataOutput) + + self.captureSession = captureSession + self.previewLayer = previewLayer + self.dataOutput = dataOutput + } + + /// On disable, stop running capture session and then tear down. + /// Both steps are required to prroperly shut down camera session. + func tearDown() { + self.captureSession?.stopRunning() + self.captureSession = nil + } + + /// Creates a capture session given input device as param. + private func createCaptureSession(for captureDevice: AVCaptureDevice) -> AVCaptureSession { + let captureSession = AVCaptureSession() + + do { + let input = try AVCaptureDeviceInput(device: captureDevice) + captureSession.addInput(input) + } catch { + print("Could not create AVCaptureDeviceInput in viewDidLoad.") + } + + return captureSession + } + + /// Sets up preview layer which gets displayed in view controller. + func createPreviewLayer(for captureSession: AVCaptureSession) -> AVCaptureVideoPreviewLayer { + let previewLayer = AVCaptureVideoPreviewLayer(session: captureSession) + previewLayer.videoGravity = AVLayerVideoGravity.resizeAspectFill + return previewLayer + } + + /// Toggles between front and back cam. + func rotateCamera() { + self.captureDevice = (captureDevice == backCam) ? frontCam : backCam + self.resetCameraFeed() + } + + /// Wrapper for screen shot. + func takeScreenShot(in view: UIView) { + guard let camImage = self.capturedImageOutput else { + fatalError("Could not call takeScreenShot") + } + + /// Create a `UIImageView` for overlaying the shutter animation over the camera view. + /// Remove it from the super view after image is saved to storage. + let imageView = UIImageView(image: camImage) + screenShotAnimate(in: view, imageView: imageView) + UIImageWriteToSavedPhotosAlbum(camImage, nil, nil, nil) + imageView.removeFromSuperview() + } + + /// Provides flash animation when screenshot is triggered. + private func screenShotAnimate(in view: UIView, imageView: UIImageView) { + imageView.contentMode = .scaleAspectFit + imageView.frame = view.frame + + let black = UIImage(named: "Black") + let blackView = UIImageView(image: black) + imageView.contentMode = .scaleAspectFill + blackView.frame = view.frame + view.addSubview(blackView) + blackView.alpha = 1 + + /// Shutter animation. + UIView.animate(withDuration: 0.3, delay: 0, options: UIView.AnimationOptions.curveLinear, animations: { + blackView.alpha = 0 + }, completion: nil) + } +} + +extension CaptureSessionManager: AVCaptureVideoDataOutputSampleBufferDelegate { + /// Delegate method for `AVCaptureVideoDataOutputSampleBufferDelegate`: formats image for inference. + /// The delegate is set in the capture session view model. + func captureOutput(_ output: AVCaptureOutput, didOutput sampleBuffer: CMSampleBuffer, from connection: AVCaptureConnection) { + /// Skip frames to optimize. + totalFrameCount += 1 + if totalFrameCount % 20 != 0{ return } + + guard let pixelBuffer: CVPixelBuffer = CMSampleBufferGetImageBuffer(sampleBuffer), + let image = UIImage(pixelBuffer: pixelBuffer), + let previewLayer = self.previewLayer, + let videoOrientation = previewLayer.connection?.videoOrientation else { + print("Failed creating image at captureOutput.") + return + } + + /// Determine rotation by radians given device orientation and camera device + var radiansToRotate = CGFloat(0) + switch videoOrientation { + case .portrait: + radiansToRotate = .pi / 2 + break + case .portraitUpsideDown: + radiansToRotate = (3 * .pi) / 2 + break + case .landscapeLeft: + if (self.captureDevice == self.backCam) { + radiansToRotate = .pi + } + break + case .landscapeRight: + if (self.captureDevice == self.frontCam) { + radiansToRotate = .pi + } + break + default: + break + } + + /// Rotate image and flip over x-axis if using front-facing cam. + let isUsingFrontCam = self.captureDevice == self.frontCam + guard let rotatedImage = image.rotate(radians: radiansToRotate, flipX: isUsingFrontCam) else { + fatalError("Could not rotate or crop image.") + } + + self.capturedImageOutput = rotatedImage + } +} + +/// Helpers for editing images. +extension UIImage { + var isPortrait: Bool { size.height > size.width } + var isLandscape: Bool { size.width > size.height } + var breadth: CGFloat { min(size.width, size.height) } + var breadthSize: CGSize { .init(width: breadth, height: breadth) } + + func squared(isOpaque: Bool = false) -> UIImage? { + guard let cgImage = cgImage? + .cropping(to: .init(origin: .init(x: isLandscape ? ((size.width-size.height)/2).rounded(.down) : 0, + y: isPortrait ? ((size.height-size.width)/2).rounded(.down) : 0), + size: breadthSize)) else { return nil } + let format = imageRendererFormat + format.opaque = isOpaque + return UIGraphicsImageRenderer(size: breadthSize, format: format).image { _ in + UIImage(cgImage: cgImage, scale: 1, orientation: imageOrientation) + .draw(in: .init(origin: .zero, size: breadthSize)) + } + } + public convenience init?(pixelBuffer: CVPixelBuffer) { + var cgImage: CGImage? + VTCreateCGImageFromCVPixelBuffer(pixelBuffer, options: nil, imageOut: &cgImage) + + guard let myImage = cgImage else { + return nil + } + + self.init(cgImage: myImage) + } + + func rotate(radians: CGFloat, flipX: Bool = false) -> UIImage? { + var newSize = CGRect(origin: CGPoint.zero, size: self.size).applying(CGAffineTransform(rotationAngle: CGFloat(radians))).size + // Trim off the extremely small float value to prevent core graphics from rounding it up + newSize.width = floor(newSize.width) + newSize.height = floor(newSize.height) + + UIGraphicsBeginImageContextWithOptions(newSize, false, self.scale) + let context = UIGraphicsGetCurrentContext()! + + // Move origin to middle + context.translateBy(x: newSize.width/2, y: newSize.height/2) + + // Flip x-axis if specified (used to correct front-facing cam + if flipX { context.scaleBy(x: -1, y: 1) } + + // Rotate around middle + context.rotate(by: CGFloat(radians)) + + // Draw the image at its center + self.draw(in: CGRect(x: -self.size.width/2, y: -self.size.height/2, width: self.size.width, height: self.size.height)) + + let newImage = UIGraphicsGetImageFromCurrentImageContext() + UIGraphicsEndImageContext() + + return newImage + } +} diff --git a/Lobe_iOS/Models/PredictionLayer.swift b/Lobe_iOS/Models/PredictionLayer.swift new file mode 100644 index 0000000..a8afe1d --- /dev/null +++ b/Lobe_iOS/Models/PredictionLayer.swift @@ -0,0 +1,79 @@ +// +// PredictionLayer.swift +// Lobe_iOS +// +// Created by Elliot Boschwitz on 11/30/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import Combine +import SwiftUI +import Vision + +/// Backend logic for predicting classifiers for a given image. +class PredictionLayer: NSObject { + @Published var classificationResult: VNClassificationObservation? + var model: VNCoreMLModel? + + /// Used for debugging image output + @Published var imageForPrediction: UIImage? + + init(model: VNCoreMLModel?) { + self.model = model + } + + /// Prediction handler which updates `classificationResult` publisher. + func getPrediction(forImage image: UIImage) { + let requestHandler = createPredictionRequestHandler(forImage: image) + + /// Add image to publisher if enviornment variable is enabled. + /// Used for debugging purposes. + if Bool(ProcessInfo.processInfo.environment["SHOW_FORMATTED_IMAGE"] ?? "false") ?? false { + self.imageForPrediction = image + } + + /// Create request handler. + let request = createModelRequest( + /// Set classification result to publisher + onComplete: { [weak self] request in + guard let classifications = request.results as? [VNClassificationObservation], + !classifications.isEmpty else { + self?.classificationResult = nil + return + } + let topClassifications = classifications.prefix(1) + self?.classificationResult = topClassifications[0] + }, onError: { [weak self] error in + print("Error getting predictions: \(error)") + self?.classificationResult = nil + }) + + try? requestHandler.perform([request]) + } + + /// Creates request handler and formats image for prediciton processing. + private func createPredictionRequestHandler(forImage image: UIImage) -> VNImageRequestHandler { + /* Crop to square images and send to the model. */ + guard let cgImage = image.cgImage else { + fatalError("Could not create cgImage in captureOutput") + } + + let ciImage = CIImage(cgImage: cgImage) + let requestHandler = VNImageRequestHandler(ciImage: ciImage) + return requestHandler + } + + private func createModelRequest(onComplete: @escaping (VNRequest) -> (), onError: @escaping (Error) -> ()) -> VNCoreMLRequest { + guard let model = model else { + fatalError("Model not found in prediction layer") + } + + let request = VNCoreMLRequest(model: model, completionHandler: { (request, error) in + if let error = error { + onError(error) + } + onComplete(request) + }) + return request + } +} diff --git a/Lobe_iOS/Models/Project.swift b/Lobe_iOS/Models/Project.swift new file mode 100644 index 0000000..30dcfe6 --- /dev/null +++ b/Lobe_iOS/Models/Project.swift @@ -0,0 +1,22 @@ +// +// Project.swift +// Lobe_iOS +// +// Created by Elliot Boschwitz on 10/11/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import Foundation +import Vision + +/// Project class. +struct Project { + var model: VNCoreMLModel? + + /// Initialize Project instance with MLModel. + init(mlModel: MLModel?) { + if let mlModel = mlModel { + self.model = try? VNCoreMLModel(for: mlModel) + } + } +} diff --git a/Lobe_iOS/MyViewController.swift b/Lobe_iOS/MyViewController.swift deleted file mode 100644 index e4de7ec..0000000 --- a/Lobe_iOS/MyViewController.swift +++ /dev/null @@ -1,246 +0,0 @@ -import Foundation -import SwiftUI -import AVKit -import Vision - -struct MyRepresentable: UIViewControllerRepresentable{ - - @State var controller: MyViewController - func makeUIViewController(context: Context) -> MyViewController { - return self.controller - } - func updateUIViewController(_ uiViewController: MyViewController, context: Context) { - - } -} - -/* Camera session; ML request handling. */ -class MyViewController: UIViewController, AVCaptureVideoDataOutputSampleBufferDelegate, ObservableObject { - - @Published var classificationLabel: String? - var backCam: AVCaptureDevice! - var frontCam: AVCaptureDevice! - var captureDevice: AVCaptureDevice! - var captureSession = AVCaptureSession() - var previewLayer: AVCaptureVideoPreviewLayer? - var useCam: Bool = true - var img: UIImage? - var confidence: Float? - var camImage: UIImage? - var totalFrameCount = 0 - - var tripleTapGesture = UITapGestureRecognizer() - var doubleTapGesture = UITapGestureRecognizer() - - @objc func handleDoubleTap(_ sender: UITapGestureRecognizer? = nil) { - flipCamera() - } - @objc func handleTripleTap(_ sender: UITapGestureRecognizer? = nil) { - screenShotMethod() - } - @objc func screenShotMethod() { - let imageView = UIImageView(image: self.camImage!) - imageView.contentMode = .scaleAspectFit - imageView.frame = view.frame - - let black = UIImage(named: "Black") - let blackView = UIImageView(image: black) - imageView.contentMode = .scaleAspectFill - blackView.frame = view.frame - view.addSubview(blackView) - blackView.alpha = 1 - - /* Shutter animation. */ - UIView.animate(withDuration: 0.3, delay: 0, options: UIView.AnimationOptions.curveLinear, animations: { - blackView.alpha = 0 - }, completion: nil) - - if useCamera{ - UIView.transition(with: view, duration: 1, options: .curveEaseIn, animations: nil) - view.addSubview(imageView) - self.changeStatus(useCam: false, img: camImage!) - } - let layer = UIApplication.shared.keyWindow!.layer - let scale = UIScreen.main.scale - UIGraphicsBeginImageContextWithOptions(layer.frame.size, false, scale); - layer.render(in: UIGraphicsGetCurrentContext()!) - let screenshot = UIGraphicsGetImageFromCurrentImageContext() - UIGraphicsEndImageContext() - UIImageWriteToSavedPhotosAlbum(screenshot!, nil, nil, nil) - if useCamera { - imageView.removeFromSuperview() - self.changeStatus(useCam: true, img: self.camImage!) - } - } - @objc func flipCamera() { - UIView.transition(with: view, duration: 0.5, options: .transitionFlipFromLeft, animations: nil) - if captureDevice == backCam{ - captureDevice = frontCam} - else { - captureDevice = backCam} - captureSession = AVCaptureSession() - guard let input = try? AVCaptureDeviceInput(device: self.captureDevice) else {return} - captureSession.addInput(input) - captureSession.startRunning() - setPreviewLayer() - setOutput() - } - override func viewDidLoad() { - super.viewDidLoad() - - doubleTapGesture = UITapGestureRecognizer(target: self, action:#selector(self.handleDoubleTap(_:))) - doubleTapGesture.numberOfTapsRequired = 2 - view.addGestureRecognizer(doubleTapGesture) - - tripleTapGesture = UITapGestureRecognizer(target: self, action:#selector(self.handleTripleTap(_:))) - tripleTapGesture.numberOfTapsRequired = 3 - view.addGestureRecognizer(tripleTapGesture) - doubleTapGesture.require(toFail: tripleTapGesture) - - backCam = AVCaptureDevice.DiscoverySession(deviceTypes: [.builtInWideAngleCamera], mediaType: AVMediaType.video, position: .back).devices.first - frontCam = AVCaptureDevice.DiscoverySession(deviceTypes: [.builtInWideAngleCamera], mediaType: AVMediaType.video, position: .front).devices.first - captureDevice = backCam - let input: AVCaptureInput! - if self.captureDevice != nil { - input = try! AVCaptureDeviceInput(device: self.captureDevice) - } else { - return - } - captureSession.addInput(input) - captureSession.startRunning() - setPreviewLayer() - setOutput() - } - func setPreviewLayer() { - previewLayer = AVCaptureVideoPreviewLayer(session: captureSession) - previewLayer!.videoGravity = AVLayerVideoGravity.resizeAspectFill - view.layer.addSublayer(previewLayer!) - previewLayer!.frame = view.frame - } - func setOutput() { - let dataOutput = AVCaptureVideoDataOutput() - dataOutput.setSampleBufferDelegate(self, queue: DispatchQueue(label: "videoQueue")) - captureSession.addOutput(dataOutput) - } - func getDevice(position: AVCaptureDevice.Position) -> AVCaptureDevice? { - let cam = AVCaptureDevice.DiscoverySession(deviceTypes: [.builtInWideAngleCamera], mediaType: AVMediaType.video, position: position).devices - return cam.first - } - func changeStatus(useCam: Bool, img: UIImage){ - if useCam { - self.useCam = true - self.img = nil - } else { - self.useCam = false - self.img = img - } - } - func captureOutput(_ output: AVCaptureOutput, didOutput sampleBuffer: CMSampleBuffer, from connection: AVCaptureConnection) { - - /* Skip frames to optimize. */ - totalFrameCount += 1 - if totalFrameCount % 20 != 0{ return } - - guard let pixelBuffer: CVPixelBuffer = CMSampleBufferGetImageBuffer(sampleBuffer) else { return } - - let curImg = UIImage(pixelBuffer: pixelBuffer) - let rotatedImage = curImg!.rotate(radians: .pi / 2) - /* Crop the captured image to be the size of the screen. */ - self.camImage = rotatedImage.crop(height: (previewLayer?.frame.height)!, width: (previewLayer?.frame.width)!) - - guard let model = try? VNCoreMLModel(for: LobeModel().model) else { return } - let request = VNCoreMLRequest(model: model) { (finishReq, err) in - self.processClassifications(for: finishReq, error: err) - } - - /* Crop to square images and send to the model. */ - if self.useCam { - try? VNImageRequestHandler(ciImage: CIImage(cgImage: (self.camImage?.squared()?.cgImage!)!)).perform([request]) - } else { - try? VNImageRequestHandler(ciImage: CIImage(cgImage: (self.img?.squared()?.cgImage!)!)).perform([request]) - } - } - func processClassifications(for request: VNRequest, error: Error?) { - DispatchQueue.main.async { - guard let results = request.results else { - self.classificationLabel = "Unable to classify image.\n\(error!.localizedDescription)" - return - } - let classifications = results as! [VNClassificationObservation] - - if classifications.isEmpty { - self.classificationLabel = "Nothing recognized." - } else { - /* Display top classifications ranked by confidence in the UI. */ - let topClassifications = classifications.prefix(1) - self.classificationLabel = topClassifications[0].identifier - self.confidence = topClassifications[0].confidence - } - } - } -} - -/* Helpers for editing images. */ -import VideoToolbox -extension UIImage { - var isPortrait: Bool { size.height > size.width } - var isLandscape: Bool { size.width > size.height } - var breadth: CGFloat { min(size.width, size.height) } - var breadthSize: CGSize { .init(width: breadth, height: breadth) } - - func squared(isOpaque: Bool = false) -> UIImage? { - guard let cgImage = cgImage? - .cropping(to: .init(origin: .init(x: isLandscape ? ((size.width-size.height)/2).rounded(.down) : 0, - y: isPortrait ? ((size.height-size.width)/2).rounded(.down) : 0), - size: breadthSize)) else { return nil } - let format = imageRendererFormat - format.opaque = isOpaque - return UIGraphicsImageRenderer(size: breadthSize, format: format).image { _ in - UIImage(cgImage: cgImage, scale: 1, orientation: imageOrientation) - .draw(in: .init(origin: .zero, size: breadthSize)) - } - } - func crop(isOpaque: Bool = false, height: CGFloat, width: CGFloat) -> UIImage? { - let newWidth = size.width - let newHeight = height / width * size.width - var screenSize: CGSize { .init(width: newWidth, height: newHeight)} - guard let cgImage = cgImage? - .cropping(to: .init(origin: .init(x: 0, - y: ((size.height - newHeight) / 2)), - size: screenSize)) else { return nil } - let format = imageRendererFormat - format.opaque = isOpaque - return UIGraphicsImageRenderer(size: screenSize, format: format).image { _ in - UIImage(cgImage: cgImage, scale: 1, orientation: imageOrientation) - .draw(in: .init(origin: .zero, size: screenSize)) - } - } - public convenience init?(pixelBuffer: CVPixelBuffer) { - var cgImage: CGImage? - VTCreateCGImageFromCVPixelBuffer(pixelBuffer, options: nil, imageOut: &cgImage) - - guard let myImage = cgImage else { - return nil - } - - self.init(cgImage: myImage) - } - func rotate(radians: CGFloat) -> UIImage { - let rotatedSize = CGRect(origin: .zero, size: size) - .applying(CGAffineTransform(rotationAngle: CGFloat(radians))) - .integral.size - UIGraphicsBeginImageContext(rotatedSize) - if let context = UIGraphicsGetCurrentContext() { - let origin = CGPoint(x: rotatedSize.width / 2.0, - y: rotatedSize.height / 2.0) - context.translateBy(x: origin.x, y: origin.y) - context.rotate(by: radians) - draw(in: CGRect(x: -origin.y, y: -origin.x, - width: size.width, height: size.height)) - let rotatedImage = UIGraphicsGetImageFromCurrentImageContext() - UIGraphicsEndImageContext() - return rotatedImage ?? self - } - return self - } -} diff --git a/Lobe_iOS/PlayViewModel.swift b/Lobe_iOS/PlayViewModel.swift new file mode 100644 index 0000000..d9cd7d4 --- /dev/null +++ b/Lobe_iOS/PlayViewModel.swift @@ -0,0 +1,75 @@ +// +// PlayViewModel.swift +// Lobe_iOS +// +// Created by Elliot Boschwitz on 12/1/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import Combine +import SwiftUI + +enum PlayViewMode { + case Camera + case ImagePreview + case NotLoaded +} + +/// View model for the Play View +class PlayViewModel: ObservableObject { + @Published var classificationLabel: String? + @Published var confidence: Float? + @Published var viewMode: PlayViewMode = PlayViewMode.NotLoaded + @Published var showImagePicker: Bool = false + @Published var imageFromPhotoPicker: UIImage? + var captureSessionManager: CaptureSessionManager + let project: Project + var imagePredicter: PredictionLayer + private var disposables = Set() + + init(project: Project) { + self.project = project + self.imagePredicter = PredictionLayer(model: project.model) + self.captureSessionManager = CaptureSessionManager() + + /// Subscribes to two publishers: + /// 1. `capturedImageOutput` published from `Camera` mode. + /// 2. `imageFromPhotoPicker` published from `ImagePreview` mode. + /// If either of the above publishers emit, we send it's output to the prediction layer for classification results. + self.self.$imageFromPhotoPicker + .merge(with: captureSessionManager.$capturedImageOutput) + .compactMap { $0 } // remove non-nill values + .receive(on: DispatchQueue.global(qos: .userInitiated)) + .sink(receiveValue: { [weak self] image in + guard let squaredImage = image.squared() else { + print("Could not create squared image in PlayViewModel.") + return + } + self?.imagePredicter.getPrediction(forImage: squaredImage) + }) + .store(in: &disposables) + + /// Subscribe to classifier results from prediction layer + self.imagePredicter.$classificationResult + .receive(on: DispatchQueue.main) + .sink(receiveValue: {[weak self] classificationResult in + guard let _classificationResult = classificationResult else { + self?.classificationLabel = "Loading Results..." + return + } + self?.classificationLabel = _classificationResult.identifier + self?.confidence = _classificationResult.confidence + + }) + .store(in: &disposables) + + /// Update camera session if toggled between view mode. + self.$viewMode + .receive(on: DispatchQueue.main) + .sink(receiveValue: { [weak self] _viewMode in + if _viewMode == .Camera { self?.captureSessionManager.resetCameraFeed() } + else { self?.captureSessionManager.tearDown() } + }) + .store(in: &disposables) + } +} diff --git a/Lobe_iOS/README.md b/Lobe_iOS/README.md new file mode 100644 index 0000000..c7acc33 --- /dev/null +++ b/Lobe_iOS/README.md @@ -0,0 +1,38 @@ +# Files in iOS Bootstrap + +This project adheres to [MVVM architecture](https://www.raywenderlich.com/34-design-patterns-by-tutorials-mvvm). MVVM is a design pattern which organizes objects by model, view model, and view. Our app leverages MVVM as follows: + +![Code Diagram](https://github.com/lobe/iOS-bootstrap/raw/master/assets/codeDiagram.png) + +*Arrows designate subscriptions in their pointed direction, i.e `PlayViewModel` subscribes to `PredictionLayer`.* + +## View +View files define the look-and-feel of the app. This starter project defines a `PlayView` superview which imports all other view objects. Please consult the README in the [`/Views`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Views) folder for more information. + +## View Model +[`PlayViewModel`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/PlayViewModel.swift) is the view model which publishes data to the `PlayView` view. View models act as important intermediaries between the view and model by de-coupling business logic from views and by subscribing to changes from the model. + +`PlayViewModel` publishes changes to the view by subscribing to the following events: +1. New images (either from video capture or image preview) will trigger a prediction request. +2. Responses to prediction requests update the UI to display the prediction result. +3. Switching to `ImagePreview` mode (which uses prediction on a chosen image from the device's library) will tear-down the video capture session manager. + +## Model +The model layer handles all data processing in the app, publishing results to any subscribers. In this case, `PlayViewModel` is the only subscriber to model objects, which are: +- [`CaptureSessionManager`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Models/CaptureSessionManager.swift): publishes select frames from the video capture feed. +- [`PredictionLayer`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Models/PredictionLayer.swift): uses the imported Core ML model to publish the results from prediction requests. +- [`Project`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Models/Project.swift): a struct for managing Core ML models. + +## Other Files +[`CaptureSessionViewController`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/CaptureSessionViewController.swift) is an exception to the MVVM rule. Although we leverage the SwiftUI library whenever possible, we still need the older UIKit library for select purposes relating to video capture handling. + +Thankfully, [we can integrate UIKit easily into SwiftUI](https://developer.apple.com/tutorials/swiftui/interfacing-with-uikit) with `UIViewControllerRepresentable`, a struct that manages view controllers directly in a SwiftUI view. In our app, [`CameraView`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Views/CameraView.swift) is a `UIViewControllerRepresentable` which creates `CaptureSessionViewController`. This view controller is responsible for: +1. Setting the view frame to the video feed. +2. Handling device orientation changes, ensuring the video feed is correctly oriented. +3. Managing tap gestures. UIKit handles conflicts between multiple tap gesture handlers better than SwiftUI, at the time of this writing. [Click here](https://github.com/lobe/iOS-bootstrap#in-app-gestures) to read more about tap-gestures in iOS-bootstrap. + +## Useful Links + +For further reading and guidance for Swift best practices: +- [Ray Wenderlich](https://www.raywenderlich.com/4161005-mvvm-with-combine-tutorial-for-ios) has a great tutorial showcasing MVVM with the [Combine](https://developer.apple.com/documentation/combine) library, which is used to define publishers and subscribers between MVVM layers. We use Combine a lot in the view model and model layers. +- [Design+Code](https://designcode.io/swi\ftui2-course) has in-depth material for creating a Swift app in iOS 14. diff --git a/Lobe_iOS/SceneDelegate.swift b/Lobe_iOS/SceneDelegate.swift index dcee5d0..a33af08 100644 --- a/Lobe_iOS/SceneDelegate.swift +++ b/Lobe_iOS/SceneDelegate.swift @@ -1,3 +1,11 @@ +// +// SceneDelegate.swift +// Lobe_iOS +// +// Created by Adam Menges on 5/20/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + import UIKit import SwiftUI @@ -12,13 +20,18 @@ class SceneDelegate: UIResponder, UIWindowSceneDelegate { // This delegate does not imply the connecting scene or session are new (see `application:configurationForConnectingSceneSession` instead). // Create the SwiftUI view that provides the window contents. - let contentView = ContentView() + let modelUrl = LobeModel.urlOfModelInThisBundle + let model = try? LobeModel(contentsOf: modelUrl).model + let project = Project(mlModel: model) + let viewModel = PlayViewModel(project: project) + let view = PlayView(viewModel: viewModel) // Use a UIHostingController as window root view controller. if let windowScene = scene as? UIWindowScene { let window = UIWindow(windowScene: windowScene) - window.rootViewController = UIHostingController(rootView: contentView) + window.rootViewController = UIHostingController(rootView: view) self.window = window + self.window?.tintColor = UIColor(rgb: 0x00DDAD) window.makeKeyAndVisible() } } diff --git a/Lobe_iOS/Views/CameraView.swift b/Lobe_iOS/Views/CameraView.swift new file mode 100644 index 0000000..f933218 --- /dev/null +++ b/Lobe_iOS/Views/CameraView.swift @@ -0,0 +1,56 @@ +// +// CameraView.swift +// Lobe_iOS +// +// Created by Elliot Boschwitz on 11/29/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import AVKit +import SwiftUI +import UIKit +import Vision + +struct CameraView: UIViewControllerRepresentable { + var captureSessionManager: CaptureSessionManager + + init(captureSessionManager: CaptureSessionManager) { + self.captureSessionManager = captureSessionManager + } + + func makeUIViewController(context: Context) -> CaptureSessionViewController { + let vc = CaptureSessionViewController() + vc.gestureDelegate = context.coordinator + return vc + } + + /// Update preview layer when state changes for camera device + func updateUIViewController(_ uiViewController: CaptureSessionViewController, context: Context) { + /// Set view with previewlayer + let previewLayer = self.captureSessionManager.previewLayer + uiViewController.previewLayer = previewLayer + uiViewController.configureVideoOrientation(for: previewLayer) + if previewLayer != nil { uiViewController.view.layer.addSublayer(previewLayer!) } + else { print("Preview layer null in updateUIViewController.") } + } + + func makeCoordinator() -> Coordinator { + Coordinator(self) + } + + class Coordinator: NSObject, CaptureSessionGestureDelegate { + var parent: CameraView + + init(_ parent: CameraView) { + self.parent = parent + } + + func viewRecognizedDoubleTap() { + parent.captureSessionManager.rotateCamera() + } + + func viewRecognizedTripleTap(_ view: UIView) { + parent.captureSessionManager.takeScreenShot(in: view) + } + } +} diff --git a/Lobe_iOS/Views/ImagePicker.swift b/Lobe_iOS/Views/ImagePicker.swift new file mode 100644 index 0000000..2029459 --- /dev/null +++ b/Lobe_iOS/Views/ImagePicker.swift @@ -0,0 +1,61 @@ +// +// ImagePicker.swift +// Lobe_iOS +// +// Created by Kathy Zhou on 5/27/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + + +import Foundation +import SwiftUI + + +/* Image picker. */ +struct ImagePicker: UIViewControllerRepresentable { + // dismisses view when document is selected + @Environment(\.presentationMode) var presentationMode + + typealias UIViewControllerType = UIImagePickerController + typealias Coordinator = ImagePickerCoordinator + + @Binding var image: UIImage? + @Binding var viewMode: PlayViewMode + let predictionLayer: PredictionLayer + + var sourceType: UIImagePickerController.SourceType = .camera + + func updateUIViewController(_ uiViewController: UIImagePickerController, context: UIViewControllerRepresentableContext) { + } + + func makeCoordinator() -> ImagePicker.Coordinator { + ImagePickerCoordinator(self) + } + + func makeUIViewController(context: UIViewControllerRepresentableContext) -> UIImagePickerController { + let picker = UIImagePickerController() + picker.sourceType = sourceType + picker.delegate = context.coordinator + picker.modalPresentationStyle = .fullScreen + return picker + } + + class ImagePickerCoordinator: NSObject, UINavigationControllerDelegate, UIImagePickerControllerDelegate { + var parent: ImagePicker + + init(_ parent: ImagePicker) { + self.parent = parent + } + + func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey : Any]) { + defer { parent.presentationMode.wrappedValue.dismiss() } + + if let uiImage = info[UIImagePickerController.InfoKey.originalImage] as? UIImage { + self.parent.image = uiImage + self.parent.predictionLayer.getPrediction(forImage: uiImage) + self.parent.viewMode = .ImagePreview + } + } + } + +} diff --git a/Lobe_iOS/Views/ImagePreview.swift b/Lobe_iOS/Views/ImagePreview.swift new file mode 100644 index 0000000..90ebc5a --- /dev/null +++ b/Lobe_iOS/Views/ImagePreview.swift @@ -0,0 +1,46 @@ +// +// ImagePreview.swift +// Lobe_iOS +// +// Created by Elliot Boschwitz on 12/1/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import SwiftUI + +/// Shows image preview from photo library on top of PlayView. +struct ImagePreview: View { + @Binding var image: UIImage? + @Binding var viewMode: PlayViewMode + @State private var offset = CGSize.zero + @State private var scaling: CGSize = .init(width: 1, height: 1) + + var body: some View { + if let image = self.image { + Image(uiImage: image) + .resizable() + .aspectRatio(image.size, contentMode: .fill) + .scaleEffect(1 / self.scaling.height) + .offset(self.offset) + + /* Gesture for swiping down to dismiss the image. */ + .gesture(DragGesture() + .onChanged ({value in + self.scaling = value.translation + self.scaling.height = max(self.scaling.height / 50, 1) + self.offset = value.translation + }) + .onEnded {_ in + self.offset = .zero + if self.scaling.height > 1.5 { + self.image = nil + self.viewMode = .Camera + } + self.scaling = .init(width: 1, height: 1) + } + ) + .opacity(1 / self.scaling.height < 1 ? 0.5: 1) + } + } +} + diff --git a/Lobe_iOS/Views/PlayView.swift b/Lobe_iOS/Views/PlayView.swift new file mode 100644 index 0000000..9bea368 --- /dev/null +++ b/Lobe_iOS/Views/PlayView.swift @@ -0,0 +1,190 @@ +// +// PlayView.swift +// Lobe_iOS +// +// Created by Adam Menges on 5/20/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + +import AVKit +import SwiftUI + +struct PlayView: View { + @Environment(\.presentationMode) var presentationMode: Binding + @ObservedObject var viewModel: PlayViewModel + + init(viewModel: PlayViewModel) { + self.viewModel = viewModel + } + + var body: some View { + NavigationView { + GeometryReader { geometry in + VStack { + switch(self.viewModel.viewMode) { + // Background camera view. + case .Camera: + ZStack { + CameraView(captureSessionManager: self.viewModel.captureSessionManager) + // Gesture for swiping up the photo library. + .gesture( + DragGesture() + .onEnded {value in + if value.translation.height < 0 { + withAnimation{ + self.viewModel.showImagePicker.toggle() + } + } + } + ) + } + + // Placeholder for displaying an image from the photo library. + case .ImagePreview: + ImagePreview(image: self.$viewModel.imageFromPhotoPicker, viewMode: self.$viewModel.viewMode) + + // TO-DO: loading screen here + case .NotLoaded: + Text("View Loading...") + } + } + .frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: .infinity) + .background(Color.black) + .edgesIgnoringSafeArea(.all) + + VStack { + /// Show processed image that gets used for prediction. + /// Used for debugging purposes + if Bool(ProcessInfo.processInfo.environment["SHOW_FORMATTED_IMAGE"] ?? "false") ?? false { + if let imageForProcessing = self.viewModel.imagePredicter.imageForPrediction { + Image(uiImage: imageForProcessing) + .resizable() + .scaledToFit() + .frame(width: 300, height: 300) + .border(Color.blue, width: 8) + } + } + Spacer() + PredictionLabelView(classificationLabel: self.$viewModel.classificationLabel, confidence: self.$viewModel.confidence) + } + } + .statusBar(hidden: true) + .navigationBarBackButtonHidden(true) + .navigationBarItems(trailing: + HStack { + /// Render `rotateCameraButton` for all modes--this is a workaround for bug where right padding is off for `ImagePreview` mode. + rotateCameraButton + .disabled(self.viewModel.viewMode != .Camera) + .opacity(self.viewModel.viewMode == .Camera ? 1 : 0) + /// Photo picker button if in camera mode, else we show button to toggle to camera mode + if (self.viewModel.viewMode == .Camera) { + openPhotoPickerButton + } else { + showCameraModeButton + } + } + .buttonStyle(PlayViewButtonStyle()) + ) + .sheet(isPresented: self.$viewModel.showImagePicker) { + ImagePicker(image: self.$viewModel.imageFromPhotoPicker, viewMode: self.$viewModel.viewMode, predictionLayer: self.viewModel.imagePredicter, sourceType: .photoLibrary) + .edgesIgnoringSafeArea(.all) + } + .onAppear { + self.viewModel.viewMode = .Camera + } + .onDisappear { + /// Disable capture session + self.viewModel.viewMode = .NotLoaded + } + } + .navigationViewStyle(StackNavigationViewStyle()) + } +} + +extension PlayView { + /// Button style for navigation row + struct PlayViewButtonStyle: ButtonStyle { + func makeBody(configuration: Configuration) -> some View { + configuration.label + .padding(10) + .foregroundColor(.white) + .background(Color.black.opacity(0.35).blur(radius: 20)) + .cornerRadius(8) + } + } + + /// Button for opening photo picker + var openPhotoPickerButton: some View { + Button(action: { + self.viewModel.showImagePicker.toggle() + }) { + Image(systemName: "photo.fill") + } + } + + /// Button for enabling camera mode + var showCameraModeButton: some View { + Button(action: { + self.viewModel.viewMode = .Camera + }) { + Image(systemName: "camera.viewfinder") + } + } + + /// Button for rotating camera + var rotateCameraButton: some View { + Button(action: { self.viewModel.captureSessionManager.rotateCamera() }) { + Image(systemName: "camera.rotate.fill") + } + } +} + +/// Gadget to build colors from Hashtag Color Code Hex. +extension UIColor { + convenience init(red: Int, green: Int, blue: Int) { + assert(red >= 0 && red <= 255, "Invalid red component") + assert(green >= 0 && green <= 255, "Invalid green component") + assert(blue >= 0 && blue <= 255, "Invalid blue component") + + self.init(red: CGFloat(red) / 255.0, green: CGFloat(green) / 255.0, blue: CGFloat(blue) / 255.0, alpha: 1.0) + } + + convenience init(rgb: Int) { + self.init( + red: (rgb >> 16) & 0xFF, + green: (rgb >> 8) & 0xFF, + blue: rgb & 0xFF + ) + } + +} + +struct PlayView_Previews: PreviewProvider { + struct TestImage: View { + var body: some View { + Image("testing_image") + .frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: .infinity) + .edgesIgnoringSafeArea(.all) + } + } + + static var previews: some View { + let viewModel = PlayViewModel(project: Project(mlModel: nil)) + viewModel.viewMode = .Camera + + return Group { + NavigationView { + ZStack { + TestImage() + PlayView(viewModel: viewModel) + } + } + .previewDevice("iPhone 12") + ZStack { + TestImage() + PlayView(viewModel: viewModel) + } + .previewDevice("iPad Pro (11-inch) (2nd generation)") + } + } +} diff --git a/Lobe_iOS/UpdateTextViewExternal.swift b/Lobe_iOS/Views/PredictionLabelView.swift similarity index 72% rename from Lobe_iOS/UpdateTextViewExternal.swift rename to Lobe_iOS/Views/PredictionLabelView.swift index aefeb6f..96cadec 100644 --- a/Lobe_iOS/UpdateTextViewExternal.swift +++ b/Lobe_iOS/Views/PredictionLabelView.swift @@ -1,3 +1,11 @@ +// +// PredictionLabelView.swift +// Lobe_iOS +// +// Created by Kathy Zhou on 6/4/20. +// Copyright © 2020 Microsoft. All rights reserved. +// + import Foundation import SwiftUI @@ -8,10 +16,10 @@ struct VisualEffectView: UIViewRepresentable { } /* View for displaying the green bar containing the prediction label. */ -struct UpdateTextViewExternal: View { - @ObservedObject var viewModel: MyViewController - @State var showImagePicker: Bool = false - @State private var image: UIImage? +struct PredictionLabelView: View { + @State private var showImagePicker: Bool = false + @Binding var classificationLabel: String? + @Binding var confidence: Float? var body: some View { GeometryReader { geometry in @@ -25,17 +33,17 @@ struct UpdateTextViewExternal: View { Rectangle() .foregroundColor(Color(UIColor(rgb: 0x00DDAD))) - .frame(width: min(CGFloat(self.viewModel.confidence ?? 0) * geometry.size.width / 1.2, geometry.size.width / 1.2)) + .frame(width: min(CGFloat(self.confidence ?? 0) * geometry.size.width / 1.2, geometry.size.width / 1.2)) .animation(.linear) - - Text(self.viewModel.classificationLabel ?? "Loading...") - .padding() - .foregroundColor(.white) + + Text(self.classificationLabel ?? "Loading...") .font(.system(size: 28)) + .foregroundColor(.white) + .padding() } } .frame(width: geometry.size.width / 1.2, - height: 65, + height: 75, alignment: .center ) .cornerRadius(17.0) @@ -46,6 +54,7 @@ struct UpdateTextViewExternal: View { } } + struct UpdateTextViewExternal_Previews: PreviewProvider { static var previews: some View { GeometryReader { geometry in @@ -56,7 +65,7 @@ struct UpdateTextViewExternal_Previews: PreviewProvider { .edgesIgnoringSafeArea(.all) .frame(width: geometry.size.width, height: geometry.size.height) - UpdateTextViewExternal(viewModel: MyViewController()).zIndex(0) + PredictionLabelView(classificationLabel: .constant(nil), confidence: .constant(nil)) }.frame(width: geometry.size.width, height: geometry.size.height) } diff --git a/Lobe_iOS/Views/README.md b/Lobe_iOS/Views/README.md new file mode 100644 index 0000000..ffc593f --- /dev/null +++ b/Lobe_iOS/Views/README.md @@ -0,0 +1,8 @@ +# View Objects +The Lobe iOS-bootstrap app organizes view logic into the following objects: + +- [`PlayView`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Views/PlayView.swift) is the superview object, which handles rendering logic for both `Camera` and `ImagePreview` modes. `PlayView` also formats the location of overlayed buttons and labels in the frame. +- [`CameraView`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Views/CameraView.swift) is a `UIViewControllerRepresentable`, rather than a `View`. It manages a view controller which sets the video feed to the view frame, described in more detail [here](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS#other-files). +- [`ImagePreview`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Views/ImagePreview.swift) displays a `UIImage` as selected from the `ImagePicker` photo picker. +- [`ImagePicker`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Views/ImagePicker.swift) is another `UIViewControllerRepresentable` which integrates `UIImagePickerControllerDelegate`, a UIKit delegate for handling selected images from an image picker. The selected image is used for `ImagePreview` mode. +- [`PredictionLabelView`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS/Views/PredictionLabelView.swift) defines the view for the UI label displaying prediction text and confidence percentage. \ No newline at end of file diff --git a/README.md b/README.md index 9f24786..846e8d3 100644 --- a/README.md +++ b/README.md @@ -4,9 +4,7 @@ iOS Bootstrap takes the machine learning model created in Lobe, and adds it to a project on iOS that uses CoreML and [SwiftUI](https://developer.apple.com/xcode/swiftui/). We help you along the way with everything you need to do to integrate it in your project. -
- -## Installing Your Development Environment +## Installing Development Environment You need to get you setup so you can build, launch, and play with your app. These instructions are written for macOS, the only system you can develop iOS apps on. @@ -42,9 +40,7 @@ Once it's done, double click on the `Lobe_iOS.xcodeproj` file in your project di Now we need to export your custom model from Lobe. If you'd like, you can skip to the [deploying your app](#deploying-your-app) section if you just want to see this app working with the default sample model. -
- -### Step 3 - Exporting your model +### Step 3 - Exporting Model After your machine learning is done training, and you are getting good results, you can export your model by going into the file menu and clicking export. Lobe supports a bunch of industry standard platforms. For this project, we'll select CoreML, the standard for Apple's platforms. @@ -52,9 +48,7 @@ Once you have the CoreML model, rename it to `LobeModel.mlmodel` and drag it int ![Illustration of Finder](https://github.com/lobe/iOS-bootstrap/raw/master/assets/modeldrag.png) -
- -## Step 4 - Deploying your app +### Step 4 - Deploying App Next, we'll want to get this app onto your phone so you can see it working live with your device's camera. To do this, plug in your device via a USB-Lightning cable and, in the open Xcode window, press the play button in the top left corner of the window: @@ -68,31 +62,22 @@ And there you have it! You're app should be running on your device. If Xcode pop And finally, if you'd like to post your app (running your custom image classification model) to the App Store, you're more than welcome to do so. [Follow the instructions here](https://developer.apple.com/app-store/submitting/) to get the process rolling. You'll need to have an Apple Developer account. -
- -## Tips and Tricks - -This app is meant as a starting place for your own project. Below is a high level overview of the project to get you started. Like any good bootstrap app, this project has been kept intentionally simple. There are only two main components in two files, `ContentView.swift` and `MyViewController.swift`. - -#### `ContentView.swift` - -This file contains all the main UI, built using SwiftUI. If you'd like to adjust the placement of any UI elements or add you own, start here. If you'd like a primer on SwiftUI, start with this: [Build a SwiftUI app for iOS 14](https://designcode.io/swiftui2-course) - -#### `MyViewController.swift` +## Miscellaneous Information -This file contains all parts that needed to be done using the old style UIKit. Mainly this is making the camera view. Luckily, this is all ported back to SwiftUI using Apple's `UIViewControllerRepresentable` API. This allows us to make the camera view, and then use it like any other SwiftUI view above. You'll also see the CoreML prediction call here. +### In-App Gestures -#### `UpdateTextViewExternal.swift` +The Lobe iOS bootstrap app supports the following gestures: +- **Swipe Up**: opens an image picker for the device's photo library. The selected image is previewed and used for prediction. +- **Double Tap**: toggles between front and back-facing cameras for the video feed. +- **Triple Tap**: saves a screenshot of the video feed, omitting overlayed UI components in the capture. -Includes the small amount of SwiftUI for the prediction bar at the bottom of the screen. +### Device Support -#### Miscellaneous Pointers +This app works for iPhones and iPads running iOS/iPadOS 13.4 or greater. -- This project contains a sample icon and other assets, feel free to use these or create your own. -- When you're using the app, swiping up on the screen pulls open the image picker. -- Double tapping flips the camera around to the front facing camera. Double tapping again flips the camera back to the front. +## Understanding the Code -
+Follow the README in the [`/Lobe_iOS`](https://github.com/lobe/iOS-bootstrap/tree/master/Lobe_iOS) folder. ## Contributing diff --git a/assets/codeDiagram.png b/assets/codeDiagram.png new file mode 100644 index 0000000000000000000000000000000000000000..f0c2efdf5e42720d165296b9831b56d668744402 GIT binary patch literal 36174 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 F{{k