diff --git a/.github/ISSUE_TEMPLATE/hmm-issue-template.md b/.github/ISSUE_TEMPLATE/hmm-issue-template.md new file mode 100644 index 00000000..79252a3e --- /dev/null +++ b/.github/ISSUE_TEMPLATE/hmm-issue-template.md @@ -0,0 +1,11 @@ +--- +name: Hmm issue template +about: 'Hmm의 ' +title: '' +labels: '' +assignees: '' + +--- + +# 인수 조건 +- [ ] diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md new file mode 100644 index 00000000..49f77bbd --- /dev/null +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -0,0 +1,9 @@ +## 연관된 이슈 + +- closed #이슈번호 + +## 작업 내용 및 고민 내용 + +## 스크린샷 + +## 리뷰 요구사항 diff --git a/.github/workflows/ios-ci.yml b/.github/workflows/ios-ci.yml new file mode 100644 index 00000000..88ada6ac --- /dev/null +++ b/.github/workflows/ios-ci.yml @@ -0,0 +1,126 @@ +name: iOS CI + +on: + push: + branches: [ main, dev ] + pull_request: + branches: [ main, dev ] + +jobs: + build-and-test: + name: Build and Test + runs-on: macos-latest + + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Set up Xcode + uses: maxim-lobanov/setup-xcode@v1 + with: + xcode-version: latest-stable + + - name: Show Xcode version + run: xcodebuild -version + + - name: Find latest iOS Simulator + id: find-simulator + run: | + echo "🔍 사용 가능한 시뮬레이터 검색 중..." + + # simctl을 사용하여 사용 가능한 iPhone 시뮬레이터 가져오기 + ALL_DEVICES=$(xcrun simctl list devices available | grep "iPhone") + + echo "📋 찾은 iPhone 시뮬레이터:" + echo "$ALL_DEVICES" + + # 숫자가 있는 iPhone만 추출하고 번호로 정렬 (iPhone 16, 17, 18... 등) + # sed로 "iPhone <숫자>" 패턴만 추출하고, 숫자로 정렬하여 가장 큰 것 선택 + NUMBERED_IPHONES=$(echo "$ALL_DEVICES" | grep -E "iPhone [0-9]+" | sed -E 's/.*iPhone ([0-9]+).*/\1/' | sort -n -u) + LATEST_NUMBER=$(echo "$NUMBERED_IPHONES" | tail -1) + + echo "📊 찾은 iPhone 버전: $(echo $NUMBERED_IPHONES | tr '\n' ' ')" + echo "🎯 최신 버전: iPhone $LATEST_NUMBER" + + # 최신 버전의 iPhone을 우선순위로 찾기 (Pro Max > Pro > Plus > 기본) + SIMULATOR_ID="" + for MODEL in "iPhone $LATEST_NUMBER Pro Max" "iPhone $LATEST_NUMBER Pro" "iPhone $LATEST_NUMBER Plus" "iPhone $LATEST_NUMBER"; do + FOUND=$(echo "$ALL_DEVICES" | grep "$MODEL" | tail -1) + if [ ! -z "$FOUND" ]; then + echo "✅ '$MODEL' 발견!" + # UUID 추출 + SIMULATOR_ID=$(echo "$FOUND" | grep -oE '[A-F0-9]{8}-[A-F0-9]{4}-[A-F0-9]{4}-[A-F0-9]{4}-[A-F0-9]{12}') + SIMULATOR_NAME=$(echo "$FOUND" | sed -E 's/^[[:space:]]*([^(]+).*/\1/' | xargs) + + echo "simulator_id=$SIMULATOR_ID" >> $GITHUB_OUTPUT + echo "simulator_name=$SIMULATOR_NAME" >> $GITHUB_OUTPUT + echo "✅ 선택된 시뮬레이터: $SIMULATOR_NAME" + echo "📱 시뮬레이터 ID: $SIMULATOR_ID" + break + fi + done + + # 시뮬레이터를 찾지 못한 경우 + if [ -z "$SIMULATOR_ID" ]; then + echo "❌ 사용 가능한 iPhone 시뮬레이터를 찾을 수 없습니다!" + echo "📋 전체 시뮬레이터 목록:" + xcrun simctl list devices available + exit 1 + fi + + - name: Show selected simulator + run: | + echo "🎯 사용할 시뮬레이터: ${{ steps.find-simulator.outputs.simulator_name }}" + echo "🆔 시뮬레이터 ID: ${{ steps.find-simulator.outputs.simulator_id }}" + + - name: Cache SPM packages + uses: actions/cache@v4 + with: + path: | + ~/Library/Developer/Xcode/DerivedData/**/SourcePackages + **/Package.resolved + key: ${{ runner.os }}-spm-${{ hashFiles('**/Package.resolved') }} + restore-keys: | + ${{ runner.os }}-spm- + + - name: Build SoloDeveloperTraining + run: | + cd SoloDeveloperTraining + xcodebuild clean build \ + -project SoloDeveloperTraining.xcodeproj \ + -scheme SoloDeveloperTraining \ + -destination "platform=iOS Simulator,id=${{ steps.find-simulator.outputs.simulator_id }}" \ + CODE_SIGNING_ALLOWED=NO \ + CODE_SIGNING_REQUIRED=NO + + - name: Test SoloDeveloperTraining + run: | + cd SoloDeveloperTraining + xcodebuild test \ + -project SoloDeveloperTraining.xcodeproj \ + -scheme SoloDeveloperTraining \ + -destination "platform=iOS Simulator,id=${{ steps.find-simulator.outputs.simulator_id }}" \ + CODE_SIGNING_ALLOWED=NO \ + CODE_SIGNING_REQUIRED=NO \ + || echo "No tests found or tests failed - continuing..." + + - name: Build SoloDeveloperTraining-Dev + run: | + cd SoloDeveloperTraining + xcodebuild clean build \ + -project SoloDeveloperTraining.xcodeproj \ + -scheme SoloDeveloperTraining-Dev \ + -destination "platform=iOS Simulator,id=${{ steps.find-simulator.outputs.simulator_id }}" \ + CODE_SIGNING_ALLOWED=NO \ + CODE_SIGNING_REQUIRED=NO + + - name: Test SoloDeveloperTraining-Dev + run: | + cd SoloDeveloperTraining + xcodebuild test \ + -project SoloDeveloperTraining.xcodeproj \ + -scheme SoloDeveloperTraining-Dev \ + -destination "platform=iOS Simulator,id=${{ steps.find-simulator.outputs.simulator_id }}" \ + CODE_SIGNING_ALLOWED=NO \ + CODE_SIGNING_REQUIRED=NO \ + || echo "No tests found or tests failed - continuing..." \ No newline at end of file diff --git a/.github/workflows/release-note.yml b/.github/workflows/release-note.yml new file mode 100644 index 00000000..87aa2f29 --- /dev/null +++ b/.github/workflows/release-note.yml @@ -0,0 +1,199 @@ +name: Release Note Generator + +on: + pull_request: + types: [closed] + branches: + - main + +permissions: + contents: write + pull-requests: read + +jobs: + create_release_note: + # PR이 실제로 merge되었을 때만 실행 + if: github.event.pull_request.merged == true + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + with: + fetch-depth: 0 + + # Xcode 프로젝트에서 버전 정보 추출 + - name: Extract Xcode version info + id: xcode_version + run: | + # 프로젝트 이름 + PROJECT_NAME="SoloDeveloperTraining" + + # MARKETING_VERSION 추출 + MARKETING_VERSION=$(grep -m 1 "MARKETING_VERSION = " "${PROJECT_NAME}/${PROJECT_NAME}.xcodeproj/project.pbxproj" | sed 's/.*MARKETING_VERSION = \(.*\);/\1/' | tr -d ' ') + + # BUILD_NUMBER 추출 + BUILD_NUMBER=$(grep -m 1 "CURRENT_PROJECT_VERSION = " "${PROJECT_NAME}/${PROJECT_NAME}.xcodeproj/project.pbxproj" | awk -F'[ ;]' '{print $3}') + + echo "marketing_version=$MARKETING_VERSION" >> $GITHUB_OUTPUT + echo "build_number=$BUILD_NUMBER" >> $GITHUB_OUTPUT + + echo "📱 Extracted Marketing Version: $MARKETING_VERSION" + echo "🔢 Extracted Build Number: $BUILD_NUMBER" + + # 버전이 변경되었는지 확인 + - name: Check if tag exists + id: check_tag + run: | + if git rev-parse "v${{ steps.xcode_version.outputs.marketing_version }}" >/dev/null 2>&1; then + echo "exists=true" >> $GITHUB_OUTPUT + echo "⚠️ Tag v${{ steps.xcode_version.outputs.marketing_version }} already exists" + else + echo "exists=false" >> $GITHUB_OUTPUT + echo "✅ Tag v${{ steps.xcode_version.outputs.marketing_version }} does not exist" + fi + + # main 브랜치에 포함된 커밋을 기준으로 PR 조회 + - name: Get merged PRs in main branch + # 동일한 태그가 존재하지 않을 경우만 실행 + if: steps.check_tag.outputs.exists == 'false' + id: get_prs + uses: actions/github-script@v7 + with: + script: | + try { + const owner = context.repo.owner; + const repo = context.repo.repo; + const pull_number = context.payload.pull_request.number; + + const prNumbers = new Set(); + + console.log(`🔎 Analyzing commits from merged PR #${pull_number}...`); + + // 1. 현재 main으로 머지된 PR에 포함된 모든 커밋 목록을 가져옵니다. + const commits = await github.paginate( + github.rest.pulls.listCommits, + { + owner, + repo, + pull_number: pull_number + } + ); + + // 2. 각 커밋과 연결된 PR들을 역추적합니다. + for (const commit of commits) { + const linkedPRs = await github.rest.repos.listPullRequestsAssociatedWithCommit({ + owner, + repo, + commit_sha: commit.sha + }); + + linkedPRs.data.forEach(pr => { + // 머지된 PR이고, 현재 main으로 머지된 PR 자체가 아닌 경우(하위 PR인 경우) 추가 + if (pr.merged_at && pr.number !== pull_number) { + prNumbers.add(pr.number); + } + }); + } + + // 3. 현재 PR을 포함시킵니다. + prNumbers.add(pull_number); + + console.log(`✅ Found ${prNumbers.size} unique sub-PRs.`); + + const prDetails = []; + for (const number of Array.from(prNumbers)) { + const { data: pr } = await github.rest.pulls.get({ + owner, + repo, + pull_number: number + }); + prDetails.push(pr); + } + + prDetails.sort((a, b) => new Date(b.merged_at) - new Date(a.merged_at)); + + // 4. PR 라벨에 따라 카테고라이징 합니다. + const features = prDetails.filter(pr => pr.labels.some(l => ['Feature','UI','Design'].includes(l.name))); + const bugfixes = prDetails.filter(pr => pr.labels.some(l => ['Fix','Bug'].includes(l.name))); + const maintenance = prDetails.filter(pr => pr.labels.some(l => ['Chore','Refactor','Remove','Docs','Test'].includes(l.name))); + const others = prDetails.filter(pr => !pr.labels.some(l => ['Feature','UI','Design','Fix','Bug','Chore','Refactor','Remove','Docs','Test','Someday','Release'].includes(l.name))); + + console.log(`📊 PR Breakdown - Features: ${features.length}, Fixes: ${bugfixes.length}, Maint: ${maintenance.length}, Others: ${others.length}`); + + // 5. 문서 내용을 추가합니다. + let releaseNotes = '## What\'s Changed\n\n'; + const formatPR = (pr) => `- ${pr.title} @${pr.user.login} ([#${pr.number}](${pr.html_url}))\n`; + + if (features.length) releaseNotes += `### 🚀 New Features\n${features.map(formatPR).join('')}\n`; + if (bugfixes.length) releaseNotes += `### 🐛 Bug Fixes\n${bugfixes.map(formatPR).join('')}\n`; + if (maintenance.length) releaseNotes += `### 🚩 Maintenance\n${maintenance.map(formatPR).join('')}\n`; + if (others.length) releaseNotes += `### 📝 Others\n${others.map(formatPR).join('')}\n`; + + return releaseNotes; + + } catch (error) { + console.error('❌ Error:', error.message); + core.setFailed(error.message); + throw error; + } + + # 릴리즈 생성 및 업데이트 + - name: Create or Update Release + if: steps.check_tag.outputs.exists == 'false' + uses: actions/github-script@v7 + with: + script: | + try { + const marketingVersion = '${{ steps.xcode_version.outputs.marketing_version }}'; + const buildNumber = '${{ steps.xcode_version.outputs.build_number }}'; + const tagName = `v${marketingVersion}`; + const releaseNotes = ${{ steps.get_prs.outputs.result }}; + + if (!marketingVersion || marketingVersion === '') { + throw new Error('Marketing version is empty or invalid'); + } + if (!buildNumber || buildNumber === '') { + throw new Error('Build number is empty or invalid'); + } + + console.log(`📦 Creating/updating release: ${tagName}`); + console.log(`📱 Version: ${marketingVersion}`); + console.log(`🔢 Build: ${buildNumber}`); + + const fullReleaseNotes = `**Version**: ${marketingVersion}\n**Build**: ${buildNumber}\n\n${releaseNotes}`; + + const { data: releases } = await github.rest.repos.listReleases({ + owner: context.repo.owner, + repo: context.repo.repo + }); + + const existingRelease = releases.find(r => r.tag_name === tagName); + + if (existingRelease) { + console.log(`🔄 Updating existing release: ${tagName}`); + await github.rest.repos.updateRelease({ + owner: context.repo.owner, + repo: context.repo.repo, + release_id: existingRelease.id, + body: fullReleaseNotes + }); + console.log(`✅ Updated release: ${tagName} (Build: ${buildNumber})`); + } else { + console.log(`🆕 Creating new release: ${tagName}`); + const release = await github.rest.repos.createRelease({ + owner: context.repo.owner, + repo: context.repo.repo, + tag_name: tagName, + name: `v${marketingVersion}`, + body: fullReleaseNotes, + draft: false, + prerelease: false + }); + console.log(`✅ Created release: ${tagName} (Build: ${buildNumber})`); + console.log(`🔗 Release URL: ${release.data.html_url}`); + } + + } catch (error) { + console.error('❌ Error creating/updating release:', error.message); + core.setFailed(`Failed to create/update release: ${error.message}`); + throw error; + } \ No newline at end of file diff --git a/.gitignore b/.gitignore index 52fe2f71..002557e8 100644 --- a/.gitignore +++ b/.gitignore @@ -1,62 +1,110 @@ +# Created by https://www.toptal.com/developers/gitignore/api/swift,xcode,macos +# Edit at https://www.toptal.com/developers/gitignore?templates=swift,xcode,macos +### macOS ### +# General +.DS_Store +.AppleDouble +.LSOverride +# Icon must end with two \r +Icon +# Thumbnails +._* +# Files that might appear in the root of a volume +.DocumentRevisions-V100 +.fseventsd +.Spotlight-V100 +.TemporaryItems +.Trashes +.VolumeIcon.icns +.com.apple.timemachine.donotpresent +# Directories potentially created on remote AFP share +.AppleDB +.AppleDesktop +Network Trash Folder +Temporary Items +.apdisk +### macOS Patch ### +# iCloud generated files +*.icloud +### Swift ### # Xcode # # gitignore contributors: remember to update Global/Xcode.gitignore, Objective-C.gitignore & Swift.gitignore - ## User settings +*.xcuserstate +*.xcuserdatad xcuserdata/ - +## compatibility with Xcode 8 and earlier (ignoring not required starting Xcode 9) +*.xcscmblueprint +*.xccheckout +## compatibility with Xcode 3 and earlier (ignoring not required starting Xcode 4) +build/ +DerivedData/ +*.moved-aside +*.pbxuser +!default.pbxuser +*.mode1v3 +!default.mode1v3 +*.mode2v3 +!default.mode2v3 +*.perspectivev3 +!default.perspectivev3 ## Obj-C/Swift specific *.hmap - ## App packaging *.ipa *.dSYM.zip *.dSYM - ## Playgrounds timeline.xctimeline playground.xcworkspace - # Swift Package Manager -# # Add this line if you want to avoid checking in source code from Swift Package Manager dependencies. # Packages/ # Package.pins # Package.resolved # *.xcodeproj -# # Xcode automatically generates this directory with a .xcworkspacedata file and xcuserdata # hence it is not needed unless you have added a package configuration file to your project # .swiftpm - .build/ - # CocoaPods -# # We recommend against adding the Pods directory to your .gitignore. However # you should judge for yourself, the pros and cons are mentioned at: # https://guides.cocoapods.org/using/using-cocoapods.html#should-i-check-the-pods-directory-into-source-control -# # Pods/ -# # Add this line if you want to avoid checking in source code from the Xcode workspace # *.xcworkspace - # Carthage -# # Add this line if you want to avoid checking in source code from Carthage dependencies. # Carthage/Checkouts - Carthage/Build/ - +# Accio dependency management +Dependencies/ +.accio/ # fastlane -# # It is recommended to not store the screenshots in the git repo. # Instead, use fastlane to re-generate the screenshots whenever they are needed. # For more information about the recommended setup visit: # https://docs.fastlane.tools/best-practices/source-control/#source-control - fastlane/report.xml fastlane/Preview.html fastlane/screenshots/**/*.png fastlane/test_output +# Code Injection +# After new code Injection tools there’s a generated folder /iOSInjectionProject +# https://github.com/johnno1962/injectionforxcode +iOSInjectionProject/ +### Xcode ### +## Xcode 8 and earlier +### Xcode Patch ### +*.xcodeproj/* +!*.xcodeproj/project.pbxproj +!*.xcodeproj/xcshareddata/ +!*.xcodeproj/project.xcworkspace/ +!*.xcworkspace/contents.xcworkspacedata +/*.gcno +**/xcshareddata/WorkspaceSettings.xcsettings +Secret.swift +# End of https://www.toptal.com/developers/gitignore/api/swift,xcode,macos + diff --git a/Prototype/Prototype/Prototype.xcodeproj/project.pbxproj b/Prototype/Prototype/Prototype.xcodeproj/project.pbxproj new file mode 100644 index 00000000..6389af48 --- /dev/null +++ b/Prototype/Prototype/Prototype.xcodeproj/project.pbxproj @@ -0,0 +1,365 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 77; + objects = { + +/* Begin PBXBuildFile section */ + 28CCE4832EF321FA00385818 /* Lottie in Frameworks */ = {isa = PBXBuildFile; productRef = 28CCE4822EF321FA00385818 /* Lottie */; }; +/* End PBXBuildFile section */ + +/* Begin PBXFileReference section */ + 08267DF92EF1A312005A0066 /* Prototype.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Prototype.app; sourceTree = BUILT_PRODUCTS_DIR; }; +/* End PBXFileReference section */ + +/* Begin PBXFileSystemSynchronizedRootGroup section */ + 08267DFB2EF1A312005A0066 /* Prototype */ = { + isa = PBXFileSystemSynchronizedRootGroup; + path = Prototype; + sourceTree = ""; + }; +/* End PBXFileSystemSynchronizedRootGroup section */ + +/* Begin PBXFrameworksBuildPhase section */ + 08267DF62EF1A312005A0066 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 28CCE4832EF321FA00385818 /* Lottie in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 08267DF02EF1A312005A0066 = { + isa = PBXGroup; + children = ( + 08267DFB2EF1A312005A0066 /* Prototype */, + 08267DFA2EF1A312005A0066 /* Products */, + ); + sourceTree = ""; + }; + 08267DFA2EF1A312005A0066 /* Products */ = { + isa = PBXGroup; + children = ( + 08267DF92EF1A312005A0066 /* Prototype.app */, + ); + name = Products; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXNativeTarget section */ + 08267DF82EF1A312005A0066 /* Prototype */ = { + isa = PBXNativeTarget; + buildConfigurationList = 08267E042EF1A314005A0066 /* Build configuration list for PBXNativeTarget "Prototype" */; + buildPhases = ( + 08267DF52EF1A312005A0066 /* Sources */, + 08267DF62EF1A312005A0066 /* Frameworks */, + 08267DF72EF1A312005A0066 /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + ); + fileSystemSynchronizedGroups = ( + 08267DFB2EF1A312005A0066 /* Prototype */, + ); + name = Prototype; + packageProductDependencies = ( + 28CCE4822EF321FA00385818 /* Lottie */, + ); + productName = Prototype; + productReference = 08267DF92EF1A312005A0066 /* Prototype.app */; + productType = "com.apple.product-type.application"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 08267DF12EF1A312005A0066 /* Project object */ = { + isa = PBXProject; + attributes = { + BuildIndependentTargetsInParallel = 1; + LastSwiftUpdateCheck = 2610; + LastUpgradeCheck = 2610; + TargetAttributes = { + 08267DF82EF1A312005A0066 = { + CreatedOnToolsVersion = 26.1.1; + }; + }; + }; + buildConfigurationList = 08267DF42EF1A312005A0066 /* Build configuration list for PBXProject "Prototype" */; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + en, + Base, + ); + mainGroup = 08267DF02EF1A312005A0066; + minimizedProjectReferenceProxies = 1; + packageReferences = ( + 28CCE4812EF321FA00385818 /* XCRemoteSwiftPackageReference "lottie-ios" */, + ); + preferredProjectObjectVersion = 77; + productRefGroup = 08267DFA2EF1A312005A0066 /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 08267DF82EF1A312005A0066 /* Prototype */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 08267DF72EF1A312005A0066 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 08267DF52EF1A312005A0066 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin XCBuildConfiguration section */ + 08267E022EF1A314005A0066 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = dwarf; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_TESTABILITY = YES; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + 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 = 26.1; + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; + MTL_FAST_MATH = YES; + ONLY_ACTIVE_ARCH = YES; + SDKROOT = iphoneos; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + }; + name = Debug; + }; + 08267E032EF1A314005A0066 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + 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 = 26.1; + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MTL_ENABLE_DEBUG_INFO = NO; + MTL_FAST_MATH = YES; + SDKROOT = iphoneos; + SWIFT_COMPILATION_MODE = wholemodule; + VALIDATE_PRODUCT = YES; + }; + name = Release; + }; + 08267E052EF1A314005A0066 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_PREVIEWS = YES; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES; + INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES; + INFOPLIST_KEY_UILaunchScreen_Generation = YES; + INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight"; + INFOPLIST_KEY_UISupportedInterfaceOrientations_iPhone = "UIInterfaceOrientationPortrait UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight"; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.hmm.Prototype; + PRODUCT_NAME = "$(TARGET_NAME)"; + STRING_CATALOG_GENERATE_SYMBOLS = YES; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_DEFAULT_ACTOR_ISOLATION = MainActor; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Debug; + }; + 08267E062EF1A314005A0066 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_PREVIEWS = YES; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES; + INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES; + INFOPLIST_KEY_UILaunchScreen_Generation = YES; + INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight"; + INFOPLIST_KEY_UISupportedInterfaceOrientations_iPhone = "UIInterfaceOrientationPortrait UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight"; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.hmm.Prototype; + PRODUCT_NAME = "$(TARGET_NAME)"; + STRING_CATALOG_GENERATE_SYMBOLS = YES; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_DEFAULT_ACTOR_ISOLATION = MainActor; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Release; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 08267DF42EF1A312005A0066 /* Build configuration list for PBXProject "Prototype" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 08267E022EF1A314005A0066 /* Debug */, + 08267E032EF1A314005A0066 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 08267E042EF1A314005A0066 /* Build configuration list for PBXNativeTarget "Prototype" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 08267E052EF1A314005A0066 /* Debug */, + 08267E062EF1A314005A0066 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + +/* Begin XCRemoteSwiftPackageReference section */ + 28CCE4812EF321FA00385818 /* XCRemoteSwiftPackageReference "lottie-ios" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/airbnb/lottie-ios"; + requirement = { + kind = upToNextMajorVersion; + minimumVersion = 4.5.2; + }; + }; +/* End XCRemoteSwiftPackageReference section */ + +/* Begin XCSwiftPackageProductDependency section */ + 28CCE4822EF321FA00385818 /* Lottie */ = { + isa = XCSwiftPackageProductDependency; + package = 28CCE4812EF321FA00385818 /* XCRemoteSwiftPackageReference "lottie-ios" */; + productName = Lottie; + }; +/* End XCSwiftPackageProductDependency section */ + }; + rootObject = 08267DF12EF1A312005A0066 /* Project object */; +} diff --git a/Prototype/Prototype/Prototype.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/Prototype/Prototype/Prototype.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..919434a6 --- /dev/null +++ b/Prototype/Prototype/Prototype.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/Prototype/Prototype/Prototype.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/Prototype/Prototype/Prototype.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved new file mode 100644 index 00000000..9805a57e --- /dev/null +++ b/Prototype/Prototype/Prototype.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -0,0 +1,15 @@ +{ + "originHash" : "0a79f2009ea5ba6a9e7c2c6471d05458a7ac9a981d2fd7ab914f1eb741af1ea1", + "pins" : [ + { + "identity" : "lottie-ios", + "kind" : "remoteSourceControl", + "location" : "https://github.com/airbnb/lottie-ios", + "state" : { + "revision" : "a004050748dc197c56256a14dca49a035d74726c", + "version" : "4.5.2" + } + } + ], + "version" : 3 +} diff --git a/Prototype/Prototype/Prototype/ApplyDesign/ApplyDesignResourceView.swift b/Prototype/Prototype/Prototype/ApplyDesign/ApplyDesignResourceView.swift new file mode 100644 index 00000000..9c61f060 --- /dev/null +++ b/Prototype/Prototype/Prototype/ApplyDesign/ApplyDesignResourceView.swift @@ -0,0 +1,359 @@ +// +// ApplyDesignResourceView.swift +// Prototype +// +// Created by sunjae on 12/17/25. +// + +import SwiftUI +import WebKit +import Lottie +import SpriteKit +import RealityKit +import Combine + +struct ApplyDesignResourceView: View { + var body: some View { + ScrollView { + VStack(alignment: .leading, spacing: 10) { + Text("Description").font(.title2) + Text( + "로고나 캐릭터에 필요한 gif, lottie 등 다양한 형태의 디자인 리소스를 가능한 방법으로 적용하고 확인합니다." + ) + Divider() + .padding(.bottom, 16) + VStack(spacing: 16) { + Logo() + Character2D() + Character3D() + } + }.padding() + } + } +} + +struct GIFView: UIViewRepresentable { + + enum RenderType { + case webView // 웹뷰 방식 + case imageSource // 이미지 소스 변환 및 재생 방식 + } + + let gifName: String + let renderType: RenderType + + func makeUIView(context: Context) -> UIView { + switch renderType { + case .webView: + return makeWebView() + case .imageSource: + return makeImageView() + } + } + + func updateUIView(_ uiView: UIView, context: Context) {} +} + +private extension GIFView { + func makeWebView() -> WKWebView { + let webView = WKWebView() + webView.scrollView.isScrollEnabled = false + webView.isOpaque = false + webView.backgroundColor = .clear + + if let path = Bundle.main.path(forResource: gifName, ofType: "gif") { + let url = URL(fileURLWithPath: path) + let data = try? Data(contentsOf: url) + + webView.load( + data ?? Data(), + mimeType: "image/gif", + characterEncodingName: "UTF-8", + baseURL: url + ) + } + return webView + } + + func makeImageView() -> UIView { + let container = UIView() + container.backgroundColor = .clear + + let imageView = UIImageView() + imageView.backgroundColor = .clear + imageView.contentMode = .scaleAspectFit + imageView.clipsToBounds = true + imageView.translatesAutoresizingMaskIntoConstraints = false + + container.addSubview(imageView) + + NSLayoutConstraint.activate([ + imageView.leadingAnchor.constraint(equalTo: container.leadingAnchor), + imageView.trailingAnchor.constraint(equalTo: container.trailingAnchor), + imageView.topAnchor.constraint(equalTo: container.topAnchor), + imageView.bottomAnchor.constraint(equalTo: container.bottomAnchor) + ]) + + // GIF 세팅 + guard + let url = Bundle.main.url(forResource: gifName, withExtension: "gif"), + let data = try? Data(contentsOf: url), + let source = CGImageSourceCreateWithData(data as CFData, nil) + else { return container } + + let frameCount = CGImageSourceGetCount(source) + var images: [UIImage] = [] + + for index in 0.., with event: UIEvent?) { + if self.isJumping { return } + self.isJumping = true + smileAnimation() + } +} + +struct SpriteCharacterView: View { + var body: some View { + VStack { + Text("* 터치하면 점프합니다.") + SpriteView( + scene: CharacterScene(), + options: [.allowsTransparency] + ) + .frame(width: 200, height: 200) + } + } +} + +struct RealityControlView: View { + @StateObject private var viewModel = Character3DViewModel() + + var body: some View { + RealityCharacterView(viewModel: viewModel) + .frame(width: 200, height: 200) + .gesture( + DragGesture() + .onChanged { value in + viewModel.dragOffset = value.translation + } + ) + } +} + +final class Character3DViewModel: ObservableObject { + @Published var dragOffset: CGSize = .zero +} + +struct RealityCharacterView: UIViewRepresentable { + @ObservedObject var viewModel: Character3DViewModel + + func makeUIView(context: Context) -> ARView { + let view = ARView(frame: .zero) + view.environment.background = .color(.clear) + + let character = try! Entity.load(named: "sample_character_3D") + character.generateCollisionShapes(recursive: true) + + // 사이즈 자동 맞춤 + let bounds = character.visualBounds(relativeTo: nil) + let size = bounds.extents + let maxDimension = max(size.x, size.y, size.z) + let targetSize: Float = 0.4 + let scale = targetSize / maxDimension + character.scale = SIMD3(repeating: scale) + + let anchor = AnchorEntity(world: .zero) + anchor.addChild(character) + view.scene.addAnchor(anchor) + + context.coordinator.character = character + + return view + } + + func updateUIView(_ uiView: ARView, context: Context) { + guard let character = context.coordinator.character else { return } + let offset = viewModel.dragOffset + character.position.x = Float(offset.width) * 0.001 + character.position.y = Float(-offset.height) * 0.001 + } + + func makeCoordinator() -> Coordinator { + Coordinator() + } + + final class Coordinator { + var character: Entity? + var viewModel: Character3DViewModel? + } +} + +// MARK: - 로고 +struct Logo: View { + + enum Option: String, CaseIterable { + case webViewGIF = "GIF - WebView" + case imageSourceGIF = "GIF - ImageSource" + case lottie = "Lottie" + } + + @State private var selectedOption: Option = .webViewGIF + + var body: some View { + VStack(alignment: .leading, spacing: 10) { + Text("로고") + .font(.title) + .bold() + + Picker("방법 선택", selection: $selectedOption) { + ForEach(Option.allCases, id: \.self) { type in + Text(type.rawValue) + } + } + .pickerStyle(.segmented) + Group { + switch selectedOption { + case .webViewGIF: + GIFView(gifName: "logo_gif", renderType: .webView) + .frame(width: 200, height: 200) + case .imageSourceGIF: + GIFView(gifName: "logo_gif", renderType: .imageSource) + .frame(width: 200, height: 200) + case .lottie: + LottieView(animation: .named("logo_lottie")) + .playing(loopMode: .playOnce).frame(width: 200, height: 200) // 재생 모드 설정 가능 + } + }.frame(maxWidth: .infinity) + .background(Color.white) + } + } +} + +// MARK: - 2D캐릭터 표시 및 제어 +struct Character2D: View { + + enum Option: String, CaseIterable { + case gif = "GIF" + case lottie = "Lottie" + case spriteKit = "SpriteKit" + } + + @State private var selectedOption: Option = .gif + + var body: some View { + VStack(alignment: .leading, spacing: 10) { + Text("2D캐릭터") + .font(.title) + .bold() + + Picker("방법 선택", selection: $selectedOption) { + ForEach(Option.allCases, id: \.self) { type in + Text(type.rawValue) + } + } + .pickerStyle(.segmented) + Group { + switch selectedOption { + case .gif: + GIFView(gifName: "character_blink_gif", renderType: .webView) + .frame(width: 200, height: 200) + case .lottie: + LottieView(animation: .named("character_smile_lottie")) + .playing(loopMode: .loop).frame(width: 200, height: 200) // 재생 모드 설정 가능 + case .spriteKit: + SpriteCharacterView() + } + }.frame(maxWidth: .infinity) + } + } +} + +// MARK: - 3D캐릭터 표시 및 제어 +struct Character3D: View { + + var body: some View { + VStack(alignment: .leading, spacing: 10) { + Text("3D캐릭터") + .font(.title) + .bold() + Text("* 드래그 시 이동합니다.") + RealityControlView() + .frame(maxWidth: .infinity) + } + } +} + +#Preview { + ApplyDesignResourceView() +} diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/character_blink_gif.gif b/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/character_blink_gif.gif new file mode 100644 index 00000000..86dba77c Binary files /dev/null and b/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/character_blink_gif.gif differ diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/character_smile_gif.gif b/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/character_smile_gif.gif new file mode 100644 index 00000000..384c5a2d Binary files /dev/null and b/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/character_smile_gif.gif differ diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/logo_gif.gif b/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/logo_gif.gif new file mode 100644 index 00000000..935ca797 Binary files /dev/null and b/Prototype/Prototype/Prototype/ApplyDesign/Resources/GIF/logo_gif.gif differ diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/Lottie/character_smile_lottie.json b/Prototype/Prototype/Prototype/ApplyDesign/Resources/Lottie/character_smile_lottie.json new file mode 100644 index 00000000..47b0e2d5 --- /dev/null +++ b/Prototype/Prototype/Prototype/ApplyDesign/Resources/Lottie/character_smile_lottie.json @@ -0,0 +1 @@ +{"v":"5.7.5","fr":100,"ip":0,"op":243,"w":201,"h":200,"nm":"Comp 1","ddd":0,"metadata":{},"assets":[{"id":"0","layers":[{"ddd":0,"ind":1,"ty":3,"nm":"","sr":1,"ks":{"p":{"a":0,"k":[201,200],"ix":2},"a":{"a":0,"k":[200,200],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0},{"ddd":0,"refId":"1","w":202,"h":202,"ind":2,"ty":0,"nm":"Gemini_Generated_Image_9e5vou9e5vou9e5v-Photoroom","sr":1,"ks":{"p":{"a":0,"k":[-1,-1],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0,"parent":1}]},{"id":"2","layers":[{"ddd":0,"refId":"3","ind":3,"ty":2,"nm":"Image","sr":1,"ks":{"p":{"a":0,"k":[101,101],"ix":2},"a":{"a":0,"k":[512,512],"ix":2},"s":{"a":0,"k":[19.53125,19.53125],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Fill mask - box","sr":1,"ks":{"p":{"a":0,"k":[1,1],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"shapes":[{"ty":"rc","d":1,"s":{"a":0,"k":[202,202],"ix":2},"p":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2}},{"ty":"fl","c":{"a":0,"k":[0,0,0],"ix":2},"o":{"a":0,"k":0,"ix":2},"r":1,"bm":0}],"ip":0,"op":244,"st":0,"bm":0}]},{"id":"1","layers":[{"ddd":0,"ind":5,"ty":4,"nm":"Fill mask","sr":1,"ks":{"p":{"a":0,"k":[1,1],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"td":1,"ao":0,"shapes":[{"ty":"gr","nm":"Fill mask","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[200,200],"ix":2},"p":{"a":0,"k":[0,0],"ix":2},"r":{"a":0,"k":0,"ix":2}},{"ty":"fl","c":{"a":0,"k":[0,0,0],"ix":2},"o":{"a":0,"k":100,"ix":2},"r":1,"bm":0},{"ty":"tr","p":{"a":0,"k":[100,100],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]}],"ip":0,"op":244,"st":0,"bm":0},{"ddd":0,"refId":"2","w":202,"h":202,"ind":2,"ty":0,"nm":"Gemini_Generated_Image_9e5vou9e5vou9e5v-Photoroom","sr":1,"ks":{"p":{"a":0,"k":[100,100],"ix":2},"a":{"a":0,"k":[100,100],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0,"tt":1}]},{"id":"3","u":"","p":"","w":1024,"h":1024,"e":1},{"id":"4","layers":[{"ddd":0,"refId":"5","ind":6,"ty":2,"nm":"Image","sr":1,"ks":{"p":{"a":0,"k":[101,100],"ix":2},"a":{"a":0,"k":[512,512],"ix":2},"s":{"a":0,"k":[19.53125,19.53125],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0},{"ddd":0,"ind":7,"ty":4,"nm":"Fill mask - box","sr":1,"ks":{"p":{"a":0,"k":[1,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"shapes":[{"ty":"rc","d":1,"s":{"a":0,"k":[202,201],"ix":2},"p":{"a":0,"k":[100,100.5],"ix":2},"r":{"a":0,"k":0,"ix":2}},{"ty":"fl","c":{"a":0,"k":[0,0,0],"ix":2},"o":{"a":0,"k":0,"ix":2},"r":1,"bm":0}],"ip":0,"op":244,"st":0,"bm":0}]},{"id":"6","layers":[{"ddd":0,"ind":8,"ty":4,"nm":"Fill mask","sr":1,"ks":{"p":{"a":0,"k":[1,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"td":1,"ao":0,"shapes":[{"ty":"gr","nm":"Fill mask","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[200,200],"ix":2},"p":{"a":0,"k":[0,0],"ix":2},"r":{"a":0,"k":0,"ix":2}},{"ty":"fl","c":{"a":0,"k":[0,0,0],"ix":2},"o":{"a":0,"k":100,"ix":2},"r":1,"bm":0},{"ty":"tr","p":{"a":0,"k":[100,100],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]}],"ip":0,"op":244,"st":0,"bm":0},{"ddd":0,"refId":"4","w":202,"h":201,"ind":9,"ty":0,"nm":"Gemini_Generated_Image_a49ogpa49ogpa49o-Photoroom","sr":1,"ks":{"p":{"a":0,"k":[100,100.5],"ix":2},"a":{"a":0,"k":[100,100.5],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0,"tt":1}]},{"id":"5","u":"","p":"","w":1024,"h":1024,"e":1}],"layers":[{"ddd":0,"ind":12345679,"ty":4,"nm":"Group Layer 8","sr":1,"ks":{"p":{"a":0,"k":[146.75,185.702868852459,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[20.59426229508197,20.59426229508197,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[220.741,37.184],[225.501,35.896],[228.749,32.36800000000001],[229.981,27.216000000000008],[228.749,22.12],[225.501,18.592],[220.741,17.304],[215.981,18.592],[212.677,22.12],[211.501,27.216000000000008],[212.677,32.36800000000001],[215.981,35.896],[220.741,37.184],[220.741,37.184],[220.741,37.184]],"i":[[0,0],[-1.380999999999972,0.8586999999999989],[-0.7839999999999918,1.493299999999991],[0,1.903999999999996],[0.8220000000000027,1.493299999999991],[1.382000000000062,0.8586999999999989],[1.79200000000003,0],[1.418999999999983,-0.8586999999999989],[0.8220000000000027,-1.493300000000005],[0,-1.904000000000011],[-0.7839999999999918,-1.5307000000000102],[-1.380999999999972,-0.8586999999999989],[-1.754000000000019,0],[0,0],[0,0]],"o":[[1.79200000000003,0],[1.382000000000062,-0.8586999999999989],[0.8220000000000027,-1.5307000000000102],[0,-1.904000000000011],[-0.7839999999999918,-1.493300000000005],[-1.380999999999972,-0.8586999999999989],[-1.754000000000019,0],[-1.380999999999972,0.8586999999999989],[-0.7839999999999918,1.493299999999991],[0,1.903999999999996],[0.8220000000000027,1.493299999999991],[1.418999999999983,0.8586999999999989],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[221.357,43.06400000000001],[214.917,41.608],[210.49300000000005,37.408],[211.221,36.232],[211.221,42.392],[205.173,42.392],[205.173,0],[211.501,0],[211.501,18.36800000000001],[210.49300000000005,16.912000000000006],[214.973,12.88],[221.357,11.424000000000007],[229.085,13.49600000000001],[234.51700000000005,19.152],[236.533,27.216000000000008],[234.51700000000005,35.28],[229.141,40.992],[221.357,43.06400000000001],[221.357,43.06400000000001],[221.357,43.06400000000001]],"i":[[0,0],[1.942000000000007,0.9706999999999937],[1.045999999999935,1.829300000000003],[-0.2426666666666506,0.3919999999999959],[0,-2.053333333333327],[2.015999999999963,0],[0,14.13066666666667],[-2.109333333333325,0],[0,-6.122666666666674],[0.3360000000000127,0.4853333333333296],[-1.865999999999985,0.9707000000000079],[-2.38900000000001,0],[-2.277000000000044,-1.3813000000000102],[-1.30600000000004,-2.389300000000006],[0,-2.986699999999999],[1.343999999999937,-2.389300000000006],[2.27800000000002,-1.4187000000000012],[2.912000000000035,0],[0,0],[0,0]],"o":[[-2.351999999999975,0],[-1.903999999999996,-0.9707000000000079],[0.2426666666666506,-0.3919999999999959],[0,2.053333333333327],[-2.015999999999963,0],[0,-14.13066666666667],[2.109333333333325,0],[0,6.122666666666667],[-0.3360000000000127,-0.4853333333333296],[1.120000000000005,-1.717300000000009],[1.867000000000075,-0.9706999999999937],[2.875,0],[2.314999999999941,1.381299999999996],[1.343999999999937,2.389300000000006],[0,2.986699999999999],[-1.30600000000004,2.389300000000006],[-2.27699999999993,1.381299999999996],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[181.87,43.06400000000001],[176.438,42],[172.854,38.976],[171.566,34.384],[172.63,29.960000000000008],[176.046,26.656000000000006],[181.814,24.752],[192.342,23.016000000000005],[192.342,28],[183.046,29.624],[179.35,31.248],[178.174,34.16],[179.462,37.016000000000005],[182.878,38.08],[187.35799999999995,36.96000000000001],[190.38199999999995,33.992],[191.446,29.792],[191.446,22.008],[189.766,18.36800000000001],[185.398,16.912000000000006],[180.974,18.256],[178.23,21.616],[172.966,18.98400000000001],[175.71,15.064000000000007],[180.134,12.376],[185.566,11.424000000000007],[191.894,12.768],[196.206,16.52],[197.774,22.008],[197.774,42.392],[191.726,42.392],[191.726,36.904],[193.014,37.072],[190.27,40.264],[186.518,42.336],[181.87,43.06400000000001],[181.87,43.06400000000001],[181.87,43.06400000000001]],"i":[[0,0],[1.567999999999984,0.7092999999999989],[0.8589999999999236,1.2693000000000012],[0,1.7547],[-0.7089999999999463,1.306699999999992],[-1.530000000000086,0.8960000000000008],[-2.313999999999965,0.3733000000000004],[-3.509333333333302,0.5786666666666633],[0,-1.661333333333332],[3.098666666666645,-0.541333333333327],[0.7839999999999918,-0.784000000000006],[0,-1.194699999999997],[-0.8579999999999472,-0.7467000000000041],[-1.381000000000085,0],[-1.268999999999892,0.7466999999999899],[-0.7089999999999463,1.2319999999999993],[0,1.530699999999996],[0,2.594666666666669],[1.120000000000005,0.9332999999999885],[1.829999999999927,0],[1.269999999999982,-0.8960000000000008],[0.5979999999999563,-1.381299999999996],[1.754666666666708,0.8773333333333255],[-1.269000000000005,1.11999999999999],[-1.680000000000064,0.6346999999999952],[-1.903999999999996,0],[-1.828999999999951,-0.8960000000000008],[-1.008000000000038,-1.6053],[0,-2.090699999999998],[0,-6.794666666666672],[2.015999999999963,0],[0,1.829333333333338],[-0.4293333333333749,-0.05599999999999739],[1.120000000000005,-0.8959999999999866],[1.418999999999983,-0.4852999999999952],[1.717999999999961,0],[0,0],[0,0]],"o":[[-2.052999999999997,0],[-1.529999999999973,-0.7467000000000041],[-0.8580000000000609,-1.306699999999992],[0,-1.642700000000005],[0.7469999999999573,-1.306700000000006],[1.530999999999949,-0.8960000000000008],[3.509333333333302,-0.5786666666666633],[0,1.661333333333332],[-3.098666666666645,0.541333333333327],[-1.680000000000064,0.2987000000000108],[-0.7839999999999918,0.7467000000000041],[0,1.157300000000006],[0.8959999999999582,0.7092999999999989],[1.717999999999961,0],[1.307000000000016,-0.7467000000000041],[0.7100000000000364,-1.2693000000000012],[0,-2.594666666666669],[0,-1.493299999999991],[-1.081999999999994,-0.9707000000000079],[-1.680000000000064,0],[-1.232000000000085,0.8586999999999989],[-1.754666666666708,-0.8773333333333255],[0.5599999999999454,-1.493300000000005],[1.269999999999982,-1.157300000000006],[1.717999999999961,-0.6347000000000094],[2.389999999999986,0],[1.866999999999962,0.8960000000000008],[1.045999999999935,1.568000000000012],[0,6.794666666666672],[-2.015999999999963,0],[0,-1.829333333333338],[0.4293333333333749,0.05599999999999739],[-0.70900000000006,1.2319999999999993],[-1.081999999999994,0.8960000000000008],[-1.380999999999972,0.4853000000000094],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[159.072,42.392],[159.072,0],[165.4,0],[165.4,42.392],[159.072,42.392],[159.072,42.392],[159.072,42.392]],"i":[[0,0],[0,14.13066666666667],[-2.109333333333325,0],[0,-14.13066666666667],[2.109333333333325,0],[0,0],[0,0]],"o":[[0,-14.13066666666667],[2.109333333333325,0],[0,14.13066666666667],[-2.109333333333325,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[138.952,43.06400000000001],[130.888,40.992],[125.456,35.28],[123.496,27.16],[125.456,19.040000000000006],[130.832,13.49600000000001],[138.448,11.424000000000007],[144.552,12.600000000000009],[149.088,15.848],[151.888,20.49600000000001],[152.896,26.096],[152.84,27.608],[152.616,29.064000000000007],[128.48,29.064000000000007],[128.48,24.024],[149.032,24.024],[146.008,26.320000000000007],[145.616,21.448000000000008],[142.816,18.032],[138.448,16.744],[133.968,18.032],[130.944,21.616],[130.104,27.216000000000008],[130.944,32.592],[134.192,36.176],[139.008,37.464],[143.65599999999995,36.232],[146.736,33.040000000000006],[151.888,35.56],[149.088,39.42400000000001],[144.60799999999995,42.11200000000001],[138.952,43.06400000000001],[138.952,43.06400000000001],[138.952,43.06400000000001]],"i":[[0,0],[2.351999999999975,1.381299999999996],[1.307000000000016,2.389300000000006],[0,2.986699999999999],[-1.307000000000016,2.35199999999999],[-2.240000000000009,1.343999999999994],[-2.836999999999989,0],[-1.79200000000003,-0.784000000000006],[-1.231999999999971,-1.381299999999996],[-0.6349999999999909,-1.754700000000014],[0,-1.978700000000003],[0.03699999999992087,-0.5227000000000004],[0.1119999999999663,-0.4480000000000075],[8.04533333333336,0],[0,1.680000000000007],[-6.850666666666712,0],[1.008000000000038,-0.7653333333333308],[0.6349999999999909,1.4187000000000012],[1.269000000000005,0.8213000000000079],[1.680000000000064,0],[1.307000000000016,-0.8586999999999989],[0.70900000000006,-1.567999999999998],[-0.1490000000000009,-2.202700000000007],[-0.7469999999999573,-1.530699999999996],[-1.380999999999972,-0.8586999999999989],[-1.79200000000003,0],[-1.268999999999892,0.8213000000000079],[-0.7469999999999573,1.306699999999992],[-1.717333333333386,-0.8400000000000034],[1.269000000000005,-1.157300000000006],[1.755000000000109,-0.6720000000000113],[2.052999999999997,0],[0,0],[0,0]],"o":[[-3.024000000000001,0],[-2.315000000000055,-1.4187000000000012],[-1.307000000000016,-2.426699999999997],[0,-3.061299999999989],[1.343999999999937,-2.352000000000004],[2.240000000000009,-1.3813000000000102],[2.277000000000044,0],[1.79200000000003,0.7839999999999918],[1.232000000000085,1.344000000000008],[0.6720000000000255,1.7547],[0,0.4852999999999952],[-0.03700000000003456,0.5227000000000004],[-8.04533333333336,0],[0,-1.680000000000007],[6.850666666666712,0],[-1.008000000000038,0.7653333333333308],[0.3729999999999336,-1.829300000000003],[-0.59699999999998,-1.456000000000003],[-1.232000000000085,-0.8586999999999989],[-1.67999999999995,0],[-1.306999999999903,0.8213000000000079],[-0.7089999999999463,1.530699999999996],[-0.1870000000000118,2.053299999999993],[0.7839999999999918,1.53070000000001],[1.418999999999983,0.8586999999999989],[1.828999999999951,0],[1.307000000000016,-0.8212999999999937],[1.717333333333386,0.8400000000000034],[-0.59699999999998,1.4187000000000012],[-1.231999999999971,1.11999999999999],[-1.716999999999985,0.6346999999999952],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[111.001,7.951999999999998],[111.001,0.6720000000000041],[117.329,0.6720000000000041],[117.329,7.951999999999998],[111.001,7.951999999999998],[111.001,7.951999999999998],[111.001,7.951999999999998]],"i":[[0,0],[0,2.426666666666662],[-2.109333333333325,0],[0,-2.426666666666662],[2.109333333333325,0],[0,0],[0,0]],"o":[[0,-2.426666666666662],[2.109333333333325,0],[0,2.426666666666662],[-2.109333333333325,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[111.001,42.392],[111.001,12.096],[117.329,12.096],[117.329,42.392],[111.001,42.392],[111.001,42.392],[111.001,42.392]],"i":[[0,0],[0,10.09866666666667],[-2.109333333333325,0],[0,-10.09866666666667],[2.109333333333325,0],[0,0],[0,0]],"o":[[0,-10.09866666666666],[2.109333333333325,0],[0,10.09866666666666],[-2.109333333333325,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[101.41,42.72800000000001],[94.01800000000003,40.040000000000006],[91.38599999999997,32.48],[91.38599999999997,17.808000000000007],[86.06600000000003,17.808000000000007],[86.06600000000003,12.096],[86.90599999999995,12.096],[90.21000000000004,10.864],[91.38599999999997,7.504000000000005],[91.38599999999997,5.152000000000001],[97.71400000000006,5.152000000000001],[97.71400000000006,12.096],[104.602,12.096],[104.602,17.808000000000007],[97.71400000000006,17.808000000000007],[97.71400000000006,32.2],[98.21799999999996,34.888000000000005],[99.84199999999998,36.568],[102.754,37.128],[103.76200000000006,37.072],[104.826,36.96000000000001],[104.826,42.392],[103.09,42.616],[101.41,42.72800000000001],[101.41,42.72800000000001],[101.41,42.72800000000001]],"i":[[0,0],[1.754999999999995,1.792000000000002],[0,3.248000000000005],[0,4.890666666666661],[1.773333333333312,0],[0,1.903999999999996],[-0.2799999999999727,0],[-0.7839999999999918,0.8212999999999937],[0,1.4187000000000012],[0,0.7839999999999989],[-2.109333333333325,0],[0,-2.314666666666668],[-2.295999999999935,0],[0,-1.903999999999996],[2.295999999999935,0],[0,-4.797333333333327],[-0.3360000000000127,-0.7467000000000041],[-0.7469999999999573,-0.4106999999999914],[-1.19500000000005,0],[-0.3730000000000473,0.03730000000000189],[-0.3360000000000127,0.03729999999998768],[0,-1.810666666666663],[0.6349999999999909,-0.07469999999999288],[0.4850000000000136,0],[0,0],[0,0]],"o":[[-3.173000000000002,0],[-1.754999999999995,-1.792000000000002],[0,-4.890666666666661],[-1.773333333333312,0],[0,-1.903999999999996],[0.2799999999999727,0],[1.419000000000096,0],[0.7839999999999918,-0.8213000000000079],[0,-0.7839999999999989],[2.109333333333325,0],[0,2.314666666666668],[2.295999999999935,0],[0,1.903999999999996],[-2.295999999999935,0],[0,4.797333333333327],[0,1.045299999999997],[0.3360000000000127,0.7092999999999989],[0.7470000000000709,0.3733000000000004],[0.2989999999999782,0],[0.3729999999999336,-0.03730000000000189],[0,1.810666666666663],[-0.5230000000000246,0.0747000000000071],[-0.6349999999999909,0.0747000000000071],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[78.71499999999997,42.72800000000001],[71.32299999999998,40.040000000000006],[68.69100000000003,32.48],[68.69100000000003,17.808000000000007],[63.37099999999998,17.808000000000007],[63.37099999999998,12.096],[64.21100000000001,12.096],[67.51499999999999,10.864],[68.69100000000003,7.504000000000005],[68.69100000000003,5.152000000000001],[75.019,5.152000000000001],[75.019,12.096],[81.90700000000004,12.096],[81.90700000000004,17.808000000000007],[75.019,17.808000000000007],[75.019,32.2],[75.52300000000002,34.888000000000005],[77.14699999999999,36.568],[80.05900000000003,37.128],[81.06700000000001,37.072],[82.13099999999997,36.96000000000001],[82.13099999999997,42.392],[80.39499999999998,42.616],[78.71499999999997,42.72800000000001],[78.71499999999997,42.72800000000001],[78.71499999999997,42.72800000000001]],"i":[[0,0],[1.754000000000019,1.792000000000002],[0,3.248000000000005],[0,4.890666666666661],[1.773333333333369,0],[0,1.903999999999996],[-0.2800000000000296,0],[-0.7839999999999918,0.8212999999999937],[0,1.4187000000000012],[0,0.7839999999999989],[-2.109333333333325,0],[0,-2.314666666666668],[-2.295999999999992,0],[0,-1.903999999999996],[2.295999999999992,0],[0,-4.797333333333327],[-0.3360000000000127,-0.7467000000000041],[-0.7470000000000141,-0.4106999999999914],[-1.19500000000005,0],[-0.3740000000000236,0.03730000000000189],[-0.3360000000000127,0.03729999999998768],[0,-1.810666666666663],[0.6340000000000146,-0.07469999999999288],[0.4850000000000136,0],[0,0],[0,0]],"o":[[-3.173999999999978,0],[-1.754999999999995,-1.792000000000002],[0,-4.890666666666661],[-1.773333333333369,0],[0,-1.903999999999996],[0.2800000000000296,0],[1.418000000000006,0],[0.7839999999999918,-0.8213000000000079],[0,-0.7839999999999989],[2.109333333333325,0],[0,2.314666666666668],[2.295999999999992,0],[0,1.903999999999996],[-2.295999999999992,0],[0,4.797333333333327],[0,1.045299999999997],[0.3359999999999559,0.7092999999999989],[0.7460000000000377,0.3733000000000004],[0.297999999999945,0],[0.3730000000000473,-0.03730000000000189],[0,1.810666666666663],[-0.5230000000000246,0.0747000000000071],[-0.6349999999999909,0.0747000000000071],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[44.18799999999999,37.184],[48.94799999999998,35.896],[52.19600000000003,32.36800000000001],[53.428,27.216000000000008],[52.19600000000003,22.12],[48.94799999999998,18.592],[44.18799999999999,17.304],[39.428,18.592],[36.12400000000002,22.12],[34.94799999999998,27.216000000000008],[36.12400000000002,32.36800000000001],[39.428,35.896],[44.18799999999999,37.184],[44.18799999999999,37.184],[44.18799999999999,37.184]],"i":[[0,0],[-1.381999999999948,0.8586999999999989],[-0.7840000000000487,1.493299999999991],[0,1.903999999999996],[0.8209999999999695,1.493299999999991],[1.381000000000029,0.8586999999999989],[1.79200000000003,0],[1.418000000000006,-0.8586999999999989],[0.8209999999999695,-1.493300000000005],[0,-1.904000000000011],[-0.7840000000000487,-1.5307000000000102],[-1.382000000000005,-0.8586999999999989],[-1.754999999999995,0],[0,0],[0,0]],"o":[[1.79200000000003,0],[1.381000000000029,-0.8586999999999989],[0.8209999999999695,-1.5307000000000102],[0,-1.904000000000011],[-0.7840000000000487,-1.493300000000005],[-1.381999999999948,-0.8586999999999989],[-1.754999999999995,0],[-1.382000000000005,0.8586999999999989],[-0.7840000000000487,1.493299999999991],[0,1.903999999999996],[0.8209999999999695,1.493299999999991],[1.418000000000006,0.8586999999999989],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[44.18799999999999,43.06400000000001],[36.18000000000001,40.992],[30.46800000000002,35.336],[28.33999999999997,27.216000000000008],[30.46800000000002,19.096],[36.18000000000001,13.49600000000001],[44.18799999999999,11.424000000000007],[52.19600000000003,13.49600000000001],[57.85199999999998,19.096],[59.98000000000002,27.216000000000008],[57.85199999999998,35.392],[52.139999999999986,41.048],[44.18799999999999,43.06400000000001],[44.18799999999999,43.06400000000001],[44.18799999999999,43.06400000000001]],"i":[[0,0],[2.425999999999988,1.381299999999996],[1.418000000000006,2.389300000000006],[0,3.024000000000001],[-1.41900000000004,2.352000000000004],[-2.389999999999986,1.343999999999994],[-2.949999999999989,0],[-2.352000000000032,-1.3813000000000102],[-1.381999999999948,-2.389300000000006],[0,-3.061300000000003],[1.418000000000006,-2.389299999999992],[2.38900000000001,-1.381299999999996],[2.912000000000035,0],[0,0],[0,0]],"o":[[-2.911999999999978,0],[-2.389999999999986,-1.381299999999996],[-1.41900000000004,-2.389299999999992],[0,-3.061300000000003],[1.418000000000006,-2.389300000000006],[2.38900000000001,-1.3813000000000102],[2.98599999999999,0],[2.388999999999953,1.343999999999994],[1.418000000000006,2.352000000000004],[0,3.061299999999989],[-1.418999999999983,2.389300000000006],[-2.389999999999986,1.343999999999994],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[0,42.392],[0,0.6720000000000041],[6.608000000000004,0.6720000000000041],[6.608000000000004,36.512],[24.639999999999986,36.512],[24.639999999999986,42.392],[0,42.392],[0,42.392],[0,42.392]],"i":[[0,0],[0,13.90666666666666],[-2.202666666666687,0],[0,-11.94666666666666],[-6.01066666666668,0],[0,-1.959999999999994],[8.21333333333331,0],[0,0],[0,0]],"o":[[0,-13.90666666666667],[2.202666666666687,0],[0,11.94666666666667],[6.01066666666668,0],[0,1.959999999999994],[-8.21333333333331,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"fl","c":{"a":0,"k":[1,1,1],"ix":2},"o":{"a":0,"k":100,"ix":2},"r":1,"bm":0},{"ty":"tr","p":{"a":0,"k":[98.08047485351562,-21.67217254638672],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[99.99999403953552,99.99999403953552],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[246.681,42.392],[246.681,0],[253.009,0],[253.009,18.032],[252.001,17.248],[255.585,12.936000000000007],[261.297,11.424000000000007],[267.23299999999995,12.88],[271.265,16.912000000000006],[272.721,22.792],[272.721,42.392],[266.449,42.392],[266.449,24.528000000000006],[265.553,20.664],[263.201,18.2],[259.729,17.304],[256.25699999999995,18.2],[253.849,20.664],[253.009,24.528000000000006],[253.009,42.392],[246.681,42.392],[246.681,42.392],[246.681,42.392]],"i":[[0,0],[0,14.13066666666667],[-2.109333333333325,0],[0,-6.010666666666665],[0.3360000000000127,0.2613333333333259],[-1.643000000000029,0.9706999999999937],[-2.166000000000054,0],[-1.717999999999961,-0.9706999999999937],[-0.9710000000000036,-1.717300000000009],[0,-2.202699999999993],[0,-6.533333333333331],[2.090666666666721,0],[0,5.954666666666668],[0.59699999999998,1.045299999999997],[1.007999999999925,0.5600000000000023],[1.305999999999926,0],[1.045000000000073,-0.5973000000000042],[0.59699999999998,-1.082700000000003],[0,-1.493300000000005],[0,-5.954666666666668],[2.109333333333325,0],[0,0],[0,0]],"o":[[0,-14.13066666666667],[2.109333333333325,0],[0,6.010666666666665],[-0.3360000000000127,-0.2613333333333259],[0.7459999999999809,-1.903999999999996],[1.641999999999967,-1.0080000000000098],[2.240000000000009,0],[1.717000000000098,0.9707000000000079],[0.9700000000000273,1.717299999999994],[0,6.533333333333331],[-2.090666666666721,0],[0,-5.954666666666668],[0,-1.5307000000000102],[-0.5599999999999454,-1.082700000000003],[-1.008000000000038,-0.5973000000000042],[-1.270000000000095,0],[-1.007999999999953,0.5600000000000023],[-0.5600000000000023,1.082700000000003],[0,5.954666666666668],[-2.109333333333325,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-354.5325317382812,-77.50520324707031],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[237.089,42.72800000000001],[229.697,40.040000000000006],[227.065,32.48],[227.065,17.808000000000007],[221.745,17.808000000000007],[221.745,12.096],[222.585,12.096],[225.889,10.864],[227.065,7.504000000000005],[227.065,5.152000000000001],[233.393,5.152000000000001],[233.393,12.096],[240.281,12.096],[240.281,17.808000000000007],[233.393,17.808000000000007],[233.393,32.2],[233.897,34.888000000000005],[235.521,36.568],[238.433,37.128],[239.441,37.072],[240.505,36.96000000000001],[240.505,42.392],[238.769,42.616],[237.089,42.72800000000001],[237.089,42.72800000000001],[237.089,42.72800000000001]],"i":[[0,0],[1.755000000000052,1.792000000000002],[0,3.248000000000005],[0,4.890666666666661],[1.773333333333369,0],[0,1.903999999999996],[-0.2800000000000296,0],[-0.7839999999999918,0.8212999999999937],[0,1.4187000000000012],[0,0.7839999999999989],[-2.109333333333325,0],[0,-2.314666666666668],[-2.295999999999992,0],[0,-1.903999999999996],[2.295999999999992,0],[0,-4.797333333333327],[-0.3360000000000127,-0.7467000000000041],[-0.7459999999999809,-0.4106999999999914],[-1.194000000000017,0],[-0.3729999999999905,0.03730000000000189],[-0.3360000000000127,0.03729999999998768],[0,-1.810666666666663],[0.6350000000000477,-0.07469999999999288],[0.48599999999999,0],[0,0],[0,0]],"o":[[-3.173000000000002,0],[-1.753999999999962,-1.792000000000002],[0,-4.890666666666661],[-1.773333333333369,0],[0,-1.903999999999996],[0.2800000000000296,0],[1.418999999999983,0],[0.7839999999999918,-0.8213000000000079],[0,-0.7839999999999989],[2.109333333333325,0],[0,2.314666666666668],[2.295999999999992,0],[0,1.903999999999996],[-2.295999999999992,0],[0,4.797333333333327],[0,1.045299999999997],[0.3359999999999559,0.7092999999999989],[0.7470000000000141,0.3733000000000004],[0.2989999999999782,0],[0.3740000000000236,-0.03730000000000189],[0,1.810666666666663],[-0.5220000000000482,0.0747000000000071],[-0.6339999999999577,0.0747000000000071],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-354.5325317382812,-77.50520324707031],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[210.259,7.951999999999998],[210.259,0.6720000000000041],[216.587,0.6720000000000041],[216.587,7.951999999999998],[210.259,7.951999999999998],[210.259,7.951999999999998],[210.259,7.951999999999998]],"i":[[0,0],[0,2.426666666666662],[-2.109333333333325,0],[0,-2.426666666666662],[2.109333333333325,0],[0,0],[0,0]],"o":[[0,-2.426666666666662],[2.109333333333325,0],[0,2.426666666666662],[-2.109333333333325,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-354.5325317382812,-77.50520324707031],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[210.259,42.392],[210.259,12.096],[216.587,12.096],[216.587,42.392],[210.259,42.392],[210.259,42.392],[210.259,42.392]],"i":[[0,0],[0,10.09866666666667],[-2.109333333333325,0],[0,-10.09866666666667],[2.109333333333325,0],[0,0],[0,0]],"o":[[0,-10.09866666666666],[2.109333333333325,0],[0,10.09866666666666],[-2.109333333333325,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-354.5325317382812,-77.50520324707031],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[169.688,42.392],[159.272,12.096],[165.992,12.096],[173.944,36.232],[171.592,36.232],[179.712,12.096],[185.48,12.096],[193.544,36.232],[191.192,36.232],[199.2,12.096],[205.92,12.096],[195.448,42.392],[189.736,42.392],[181.56,17.696],[183.632,17.696],[175.456,42.392],[169.688,42.392],[169.688,42.392],[169.688,42.392]],"i":[[0,0],[3.47199999999998,10.09866666666667],[-2.240000000000009,0],[-2.650666666666666,-8.045333333333332],[0.7839999999999918,0],[-2.706666666666649,8.045333333333332],[-1.922666666666657,0],[-2.687999999999988,-8.045333333333332],[0.7839999999999918,0],[-2.669333333333327,8.045333333333332],[-2.240000000000009,0],[3.490666666666641,-10.09866666666667],[1.903999999999996,0],[2.725333333333367,8.232],[-0.6906666666666865,0],[2.72533333333331,-8.232],[1.922666666666657,0],[0,0],[0,0]],"o":[[-3.47199999999998,-10.09866666666666],[2.240000000000009,0],[2.650666666666666,8.045333333333332],[-0.7839999999999918,0],[2.706666666666649,-8.045333333333332],[1.922666666666657,0],[2.687999999999988,8.045333333333332],[-0.7839999999999918,0],[2.669333333333327,-8.045333333333332],[2.240000000000009,0],[-3.490666666666641,10.09866666666666],[-1.903999999999996,0],[-2.725333333333367,-8.232],[0.6906666666666865,0],[-2.72533333333331,8.232],[-1.922666666666657,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-354.5325317382812,-77.50520324707031],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"fl","c":{"a":0,"k":[1,1,1],"ix":2},"o":{"a":0,"k":100,"ix":2},"r":1,"bm":0},{"ty":"tr","p":{"a":0,"k":[155.86146545410156,56.001014709472656],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[99.99999403953552,99.99999403953552],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[132.444,43.06400000000001],[124.38,40.992],[118.948,35.28],[116.988,27.16],[118.948,19.040000000000006],[124.324,13.49600000000001],[131.94,11.424000000000007],[138.044,12.600000000000009],[142.58,15.848],[145.38,20.49600000000001],[146.388,26.096],[146.332,27.608],[146.108,29.064000000000007],[121.972,29.064000000000007],[121.972,24.024],[142.524,24.024],[139.5,26.320000000000007],[139.108,21.448000000000008],[136.308,18.032],[131.94,16.744],[127.46,18.032],[124.436,21.616],[123.596,27.216000000000008],[124.436,32.592],[127.684,36.176],[132.5,37.464],[137.148,36.232],[140.228,33.040000000000006],[145.38,35.56],[142.58,39.42400000000001],[138.1,42.11200000000001],[132.444,43.06400000000001],[132.444,43.06400000000001],[132.444,43.06400000000001]],"i":[[0,0],[2.351999999999975,1.381299999999996],[1.305999999999983,2.389300000000006],[0,2.986699999999999],[-1.307000000000016,2.35199999999999],[-2.240000000000009,1.343999999999994],[-2.838000000000022,0],[-1.79200000000003,-0.784000000000006],[-1.232000000000028,-1.381299999999996],[-0.6350000000000477,-1.754700000000014],[0,-1.978700000000003],[0.03699999999997772,-0.5227000000000004],[0.1120000000000232,-0.4480000000000075],[8.045333333333303,0],[0,1.680000000000007],[-6.850666666666655,0],[1.007999999999981,-0.7653333333333308],[0.6340000000000146,1.4187000000000012],[1.269000000000005,0.8213000000000079],[1.67999999999995,0],[1.305999999999983,-0.8586999999999989],[0.7090000000000032,-1.567999999999998],[-0.1499999999999773,-2.202700000000007],[-0.7470000000000141,-1.530699999999996],[-1.382000000000005,-0.8586999999999989],[-1.791999999999973,0],[-1.269999999999982,0.8213000000000079],[-0.7469999999999573,1.306699999999992],[-1.717333333333329,-0.8400000000000034],[1.269000000000005,-1.157300000000006],[1.754000000000019,-0.6720000000000113],[2.052999999999997,0],[0,0],[0,0]],"o":[[-3.024000000000001,0],[-2.314999999999998,-1.4187000000000012],[-1.307000000000016,-2.426699999999997],[0,-3.061299999999989],[1.343999999999994,-2.352000000000004],[2.240000000000009,-1.3813000000000102],[2.276999999999987,0],[1.791999999999973,0.7839999999999918],[1.231999999999971,1.344000000000008],[0.6719999999999686,1.7547],[0,0.4852999999999952],[-0.03800000000001091,0.5227000000000004],[-8.045333333333303,0],[0,-1.680000000000007],[6.850666666666655,0],[-1.007999999999981,0.7653333333333308],[0.3730000000000473,-1.829300000000003],[-0.5979999999999563,-1.456000000000003],[-1.232000000000028,-0.8586999999999989],[-1.680000000000007,0],[-1.307000000000016,0.8213000000000079],[-0.7100000000000364,1.530699999999996],[-0.186999999999955,2.053299999999993],[0.7839999999999918,1.53070000000001],[1.418000000000006,0.8586999999999989],[1.829000000000008,0],[1.305999999999983,-0.8212999999999937],[1.717333333333329,0.8400000000000034],[-0.5980000000000132,1.4187000000000012],[-1.232000000000028,1.11999999999999],[-1.718000000000018,0.6346999999999952],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-211.7300415039062,-77.67320251464844],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[95.32,37.184],[100.024,35.896],[103.328,32.36800000000001],[104.56,27.216000000000008],[103.328,22.12],[100.024,18.592],[95.32,17.304],[90.56,18.592],[87.256,22.12],[86.08000000000001,27.216000000000008],[87.256,32.36800000000001],[90.50399999999999,35.896],[95.32,37.184],[95.32,37.184],[95.32,37.184]],"i":[[0,0],[-1.381,0.8586999999999989],[-0.7839999999999918,1.493299999999991],[0,1.903999999999996],[0.820999999999998,1.493299999999991],[1.419000000000011,0.8586999999999989],[1.754999999999995,0],[1.418999999999983,-0.8586999999999989],[0.7839999999999918,-1.493300000000005],[0,-1.904000000000011],[-0.7839999999999918,-1.5307000000000102],[-1.381,-0.8586999999999989],[-1.792000000000002,0],[0,0],[0,0]],"o":[[1.754999999999995,0],[1.419000000000011,-0.8586999999999989],[0.820999999999998,-1.5307000000000102],[0,-1.904000000000011],[-0.7839999999999918,-1.493300000000005],[-1.381,-0.8586999999999989],[-1.754999999999995,0],[-1.419000000000011,0.8586999999999989],[-0.7839999999999918,1.493299999999991],[0,1.903999999999996],[0.7839999999999918,1.493299999999991],[1.419000000000011,0.8586999999999989],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-211.7300415039062,-77.67320251464844],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[94.70400000000001,43.06400000000001],[86.864,40.992],[81.43199999999999,35.28],[79.47200000000001,27.216000000000008],[81.488,19.152],[86.91999999999999,13.49600000000001],[94.648,11.424000000000007],[101.088,12.88],[105.512,16.912000000000006],[104.56,18.36800000000001],[104.56,0],[110.832,0],[110.832,42.392],[104.84,42.392],[104.84,36.232],[105.568,37.408],[101.088,41.608],[94.70400000000001,43.06400000000001],[94.70400000000001,43.06400000000001],[94.70400000000001,43.06400000000001]],"i":[[0,0],[2.314999999999998,1.381299999999996],[1.344000000000023,2.389300000000006],[0,2.986699999999999],[-1.343999999999994,2.389300000000006],[-2.276999999999987,1.381299999999996],[-2.875,0],[-1.8669999999999902,-0.9706999999999937],[-1.082999999999998,-1.717300000000009],[0.3173333333333233,-0.4853333333333296],[0,6.122666666666674],[-2.090666666666664,0],[0,-14.13066666666667],[1.99733333333333,0],[0,2.053333333333327],[-0.242666666666679,-0.3919999999999959],[1.941000000000003,-0.9707000000000079],[2.314999999999998,0],[0,0],[0,0]],"o":[[-2.912000000000006,0],[-2.277000000000015,-1.4187000000000012],[-1.306999999999988,-2.389300000000006],[0,-2.986699999999999],[1.343999999999994,-2.389300000000006],[2.277000000000015,-1.3813000000000102],[2.426999999999992,0],[1.867000000000019,0.9707000000000079],[-0.3173333333333233,0.4853333333333296],[0,-6.122666666666674],[2.090666666666664,0],[0,14.13066666666667],[-1.99733333333333,0],[0,-2.053333333333327],[0.242666666666679,0.3919999999999959],[-1.045000000000016,1.829300000000003],[-1.941000000000003,0.9706999999999937],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-211.7300415039062,-77.67320251464844],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[57.40100000000001,43.06400000000001],[51.968999999999994,42],[48.38499999999999,38.976],[47.09700000000001,34.384],[48.161,29.960000000000008],[51.577,26.656000000000006],[57.345,24.752],[67.87299999999999,23.016000000000005],[67.87299999999999,28],[58.577,29.624],[54.881,31.248],[53.70500000000001,34.16],[54.992999999999995,37.016000000000005],[58.40899999999999,38.08],[62.88900000000001,36.96000000000001],[65.91300000000001,33.992],[66.977,29.792],[66.977,22.008],[65.297,18.36800000000001],[60.929,16.912000000000006],[56.505,18.256],[53.761,21.616],[48.496999999999986,18.98400000000001],[51.240999999999985,15.064000000000007],[55.66499999999999,12.376],[61.09700000000001,11.424000000000007],[67.42500000000001,12.768],[71.737,16.52],[73.305,22.008],[73.305,42.392],[67.257,42.392],[67.257,36.904],[68.54499999999999,37.072],[65.80099999999999,40.264],[62.04900000000001,42.336],[57.40100000000001,43.06400000000001],[57.40100000000001,43.06400000000001],[57.40100000000001,43.06400000000001]],"i":[[0,0],[1.568000000000012,0.7092999999999989],[0.8590000000000089,1.2693000000000012],[0,1.7547],[-0.7090000000000032,1.306699999999992],[-1.531000000000006,0.8960000000000008],[-2.314999999999998,0.3733000000000004],[-3.509333333333331,0.5786666666666633],[0,-1.661333333333332],[3.098666666666674,-0.541333333333327],[0.7839999999999918,-0.784000000000006],[0,-1.194699999999997],[-0.8590000000000089,-0.7467000000000041],[-1.381,0],[-1.269000000000005,0.7466999999999899],[-0.7090000000000032,1.2319999999999993],[0,1.530699999999996],[0,2.594666666666669],[1.120000000000005,0.9332999999999885],[1.829000000000008,0],[1.269000000000005,-0.8960000000000008],[0.5970000000000084,-1.381299999999996],[1.754666666666679,0.8773333333333255],[-1.268999999999977,1.11999999999999],[-1.680000000000007,0.6346999999999952],[-1.903999999999996,0],[-1.829000000000008,-0.8960000000000008],[-1.0080000000000098,-1.6053],[0,-2.090699999999998],[0,-6.794666666666672],[2.015999999999991,0],[0,1.829333333333338],[-0.429333333333318,-0.05599999999999739],[1.120000000000005,-0.8959999999999866],[1.418999999999983,-0.4852999999999952],[1.716999999999985,0],[0,0],[0,0]],"o":[[-2.053000000000026,0],[-1.531000000000006,-0.7467000000000041],[-0.8589999999999804,-1.306699999999992],[0,-1.642700000000005],[0.7469999999999857,-1.306700000000006],[1.531000000000006,-0.8960000000000008],[3.509333333333331,-0.5786666666666633],[0,1.661333333333332],[-3.098666666666674,0.541333333333327],[-1.680000000000007,0.2987000000000108],[-0.7839999999999918,0.7467000000000041],[0,1.157300000000006],[0.896000000000015,0.7092999999999989],[1.717000000000013,0],[1.306999999999988,-0.7467000000000041],[0.7089999999999748,-1.2693000000000012],[0,-2.594666666666669],[0,-1.493299999999991],[-1.082999999999998,-0.9707000000000079],[-1.680000000000007,0],[-1.2319999999999993,0.8586999999999989],[-1.754666666666679,-0.8773333333333255],[0.5600000000000023,-1.493300000000005],[1.269000000000005,-1.157300000000006],[1.717000000000013,-0.6347000000000094],[2.388999999999982,0],[1.86699999999999,0.8960000000000008],[1.045000000000016,1.568000000000012],[0,6.794666666666672],[-2.015999999999991,0],[0,-1.829333333333338],[0.429333333333318,0.05599999999999739],[-0.7089999999999748,1.2319999999999993],[-1.082999999999998,0.8960000000000008],[-1.381,0.4853000000000094],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-211.7300415039062,-77.67320251464844],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"sh","d":1,"ks":{"a":0,"k":{"c":true,"v":[[0,42.392],[0,0.6720000000000041],[6.159999999999997,0.6720000000000041],[21.84,22.400000000000006],[18.75999999999999,22.400000000000006],[34.16,0.6720000000000041],[40.31999999999999,0.6720000000000041],[40.31999999999999,42.392],[33.768,42.392],[33.768,8.456000000000003],[36.232,9.128],[20.49600000000001,30.632000000000005],[19.824000000000012,30.632000000000005],[4.424000000000007,9.128],[6.608000000000004,8.456000000000003],[6.608000000000004,42.392],[0,42.392],[0,42.392],[0,42.392]],"i":[[0,0],[0,13.90666666666666],[-2.053333333333342,0],[-5.226666666666659,-7.242666666666665],[1.026666666666671,0],[-5.133333333333326,7.242666666666672],[-2.053333333333342,0],[0,-13.90666666666667],[2.183999999999997,0],[0,11.312],[-0.8213333333333424,-0.2240000000000038],[5.245333333333321,-7.168000000000006],[0.2239999999999895,0],[5.133333333333326,7.168000000000006],[-0.7280000000000086,0.2240000000000038],[0,-11.312],[2.202666666666659,0],[0,0],[0,0]],"o":[[0,-13.90666666666667],[2.053333333333342,0],[5.226666666666659,7.242666666666672],[-1.026666666666671,0],[5.133333333333326,-7.242666666666665],[2.053333333333342,0],[0,13.90666666666666],[-2.183999999999997,0],[0,-11.312],[0.8213333333333424,0.2240000000000038],[-5.245333333333321,7.168000000000006],[-0.2239999999999895,0],[-5.133333333333326,-7.168000000000006],[0.7280000000000086,-0.2240000000000038],[0,11.312],[-2.202666666666659,0],[0,0],[0,0],[0,0]]}}},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[-211.7300415039062,-77.67320251464844],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"fl","c":{"a":0,"k":[1,1,1],"ix":2},"o":{"a":0,"k":100,"ix":2},"r":1,"bm":0},{"ty":"tr","p":{"a":0,"k":[2.91259765625,56.001014709472656],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[99.99999403953552,99.99999403953552],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[702.6863719370097,144],"ix":2},"p":{"a":0,"k":[0,0],"ix":2},"r":{"a":0,"k":72,"ix":2}},{"ty":"fl","c":{"a":0,"k":[0,0,0],"ix":2},"o":{"a":0,"k":100,"ix":2},"r":1,"bm":0},{"ty":"tm","s":{"a":0,"k":0,"ix":2},"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":0,"ix":2},"m":1},{"ty":"tr","p":{"a":0,"k":[56.54167175292969,-0.000022762338630855083],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[99.99999403953552,99.99999403953552],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":80,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]},{"ty":"tr","p":{"a":0,"k":[122.0000003294881,25.00000012138912],"ix":2},"a":{"a":0,"k":[56.54167175292969,-0.00002288818359375],"ix":2},"s":{"a":0,"k":[34.403572049765366,34.403572049765366],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}}]}],"ip":0,"op":244,"st":0,"bm":0},{"ddd":0,"ind":10,"ty":3,"nm":"","sr":1,"ks":{"p":{"a":0,"k":[200,200],"ix":2},"a":{"a":0,"k":[200,200],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0},{"ddd":0,"refId":"0","w":202,"h":201,"ind":11,"ty":0,"nm":"Frame 104","sr":1,"ks":{"p":{"a":0,"k":[-1,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":1,"k":[{"t":43,"s":[100],"h":1},{"t":44,"s":[100],"h":1},{"t":45,"s":[100],"h":1},{"t":46,"s":[100],"h":1},{"t":47,"s":[100],"h":1},{"t":48,"s":[100],"h":1},{"t":49,"s":[100],"h":1},{"t":50,"s":[100],"h":1},{"t":51,"s":[100],"h":1},{"t":52,"s":[100],"h":1},{"t":53,"s":[100],"h":1},{"t":54,"s":[100],"h":1},{"t":55,"s":[100],"h":1},{"t":56,"s":[100],"h":1},{"t":57,"s":[100],"h":1},{"t":58,"s":[100],"h":1},{"t":59,"s":[100],"h":1},{"t":60,"s":[100],"h":1},{"t":61,"s":[100],"h":1},{"t":62,"s":[100],"h":1},{"t":63,"s":[100],"h":1},{"t":64,"s":[100],"h":1},{"t":65,"s":[100],"h":1},{"t":66,"s":[100],"h":1},{"t":67,"s":[100],"h":1},{"t":68,"s":[100],"h":1},{"t":69,"s":[100],"h":1},{"t":70,"s":[100],"h":1},{"t":71,"s":[100],"h":1},{"t":72,"s":[100],"h":1},{"t":73,"s":[100],"h":1},{"t":74,"s":[100],"h":1},{"t":75,"s":[100],"h":1},{"t":76,"s":[100],"h":1},{"t":77,"s":[100],"h":1},{"t":78,"s":[100],"h":1},{"t":79,"s":[100],"h":1},{"t":80,"s":[100],"h":1},{"t":81,"s":[100],"h":1},{"t":82,"s":[100],"h":1},{"t":83,"s":[100],"h":1},{"t":84,"s":[100],"h":1},{"t":85,"s":[100],"h":1},{"t":86,"s":[100],"h":1},{"t":87,"s":[100],"h":1},{"t":88,"s":[100],"h":1},{"t":89,"s":[100],"h":1},{"t":90,"s":[100],"h":1},{"t":91,"s":[100],"h":1},{"t":92,"s":[100],"h":1},{"t":93,"s":[0],"h":1},{"t":94,"s":[0],"h":1},{"t":95,"s":[0],"h":1},{"t":96,"s":[0],"h":1},{"t":97,"s":[0],"h":1},{"t":98,"s":[0],"h":1},{"t":99,"s":[0],"h":1},{"t":100,"s":[0],"h":1},{"t":101,"s":[0],"h":1},{"t":102,"s":[0],"h":1},{"t":103,"s":[0],"h":1},{"t":104,"s":[0],"h":1},{"t":105,"s":[0],"h":1},{"t":106,"s":[0],"h":1},{"t":107,"s":[0],"h":1},{"t":108,"s":[0],"h":1},{"t":109,"s":[0],"h":1},{"t":110,"s":[0],"h":1},{"t":111,"s":[0],"h":1},{"t":112,"s":[0],"h":1},{"t":113,"s":[0],"h":1},{"t":114,"s":[0],"h":1},{"t":115,"s":[0],"h":1},{"t":116,"s":[0],"h":1},{"t":117,"s":[0],"h":1},{"t":118,"s":[0],"h":1},{"t":119,"s":[0],"h":1},{"t":120,"s":[0],"h":1},{"t":121,"s":[0],"h":1},{"t":122,"s":[0],"h":1},{"t":123,"s":[0],"h":1},{"t":124,"s":[0],"h":1},{"t":125,"s":[0],"h":1},{"t":126,"s":[0],"h":1},{"t":127,"s":[0],"h":1},{"t":128,"s":[0],"h":1},{"t":129,"s":[0],"h":1},{"t":130,"s":[0],"h":1},{"t":131,"s":[0],"h":1},{"t":132,"s":[0],"h":1},{"t":133,"s":[0],"h":1},{"t":134,"s":[0],"h":1},{"t":135,"s":[0],"h":1},{"t":136,"s":[0],"h":1},{"t":137,"s":[0],"h":1},{"t":138,"s":[0],"h":1},{"t":139,"s":[0],"h":1},{"t":140,"s":[0],"h":1},{"t":141,"s":[0],"h":1},{"t":142,"s":[0],"h":1},{"t":143,"s":[100],"h":1},{"t":144,"s":[100],"h":1},{"t":145,"s":[100],"h":1},{"t":146,"s":[100],"h":1},{"t":147,"s":[100],"h":1},{"t":148,"s":[100],"h":1},{"t":149,"s":[100],"h":1},{"t":150,"s":[100],"h":1},{"t":151,"s":[100],"h":1},{"t":152,"s":[100],"h":1},{"t":153,"s":[100],"h":1},{"t":154,"s":[100],"h":1},{"t":155,"s":[100],"h":1},{"t":156,"s":[100],"h":1},{"t":157,"s":[100],"h":1},{"t":158,"s":[100],"h":1},{"t":159,"s":[100],"h":1},{"t":160,"s":[100],"h":1},{"t":161,"s":[100],"h":1},{"t":162,"s":[100],"h":1},{"t":163,"s":[100],"h":1},{"t":164,"s":[100],"h":1},{"t":165,"s":[100],"h":1},{"t":166,"s":[100],"h":1},{"t":167,"s":[100],"h":1},{"t":168,"s":[100],"h":1},{"t":169,"s":[100],"h":1},{"t":170,"s":[100],"h":1},{"t":171,"s":[100],"h":1},{"t":172,"s":[100],"h":1},{"t":173,"s":[100],"h":1},{"t":174,"s":[100],"h":1},{"t":175,"s":[100],"h":1},{"t":176,"s":[100],"h":1},{"t":177,"s":[100],"h":1},{"t":178,"s":[100],"h":1},{"t":179,"s":[100],"h":1},{"t":180,"s":[100],"h":1},{"t":181,"s":[100],"h":1},{"t":182,"s":[100],"h":1},{"t":183,"s":[100],"h":1},{"t":184,"s":[100],"h":1},{"t":185,"s":[100],"h":1},{"t":186,"s":[100],"h":1},{"t":187,"s":[100],"h":1},{"t":188,"s":[100],"h":1},{"t":189,"s":[100],"h":1},{"t":190,"s":[100],"h":1},{"t":191,"s":[100],"h":1},{"t":192,"s":[100],"h":1},{"t":193,"s":[0],"h":1},{"t":194,"s":[0],"h":1},{"t":195,"s":[0],"h":1},{"t":196,"s":[0],"h":1},{"t":197,"s":[0],"h":1},{"t":198,"s":[0],"h":1},{"t":199,"s":[0],"h":1},{"t":200,"s":[0],"h":1},{"t":201,"s":[0],"h":1},{"t":202,"s":[0],"h":1},{"t":203,"s":[0],"h":1},{"t":204,"s":[0],"h":1},{"t":205,"s":[0],"h":1},{"t":206,"s":[0],"h":1},{"t":207,"s":[0],"h":1},{"t":208,"s":[0],"h":1},{"t":209,"s":[0],"h":1},{"t":210,"s":[0],"h":1},{"t":211,"s":[0],"h":1},{"t":212,"s":[0],"h":1},{"t":213,"s":[0],"h":1},{"t":214,"s":[0],"h":1},{"t":215,"s":[0],"h":1},{"t":216,"s":[0],"h":1},{"t":217,"s":[0],"h":1},{"t":218,"s":[0],"h":1},{"t":219,"s":[0],"h":1},{"t":220,"s":[0],"h":1},{"t":221,"s":[0],"h":1},{"t":222,"s":[0],"h":1},{"t":223,"s":[0],"h":1},{"t":224,"s":[0],"h":1},{"t":225,"s":[0],"h":1},{"t":226,"s":[0],"h":1},{"t":227,"s":[0],"h":1},{"t":228,"s":[0],"h":1},{"t":229,"s":[0],"h":1},{"t":230,"s":[0],"h":1},{"t":231,"s":[0],"h":1},{"t":232,"s":[0],"h":1},{"t":233,"s":[0],"h":1},{"t":234,"s":[0],"h":1},{"t":235,"s":[0],"h":1},{"t":236,"s":[0],"h":1},{"t":237,"s":[0],"h":1},{"t":238,"s":[0],"h":1},{"t":239,"s":[0],"h":1},{"t":240,"s":[0],"h":1},{"t":241,"s":[0],"h":1},{"t":242,"s":[0],"h":1},{"t":243,"s":[100],"h":1}],"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0,"parent":10},{"ddd":0,"ind":12,"ty":3,"nm":"Frame 105","sr":1,"ks":{"p":{"a":0,"k":[478,200],"ix":2},"a":{"a":0,"k":[478,200],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0},{"ddd":0,"ind":13,"ty":3,"nm":"","sr":1,"ks":{"p":{"a":0,"k":[200,200],"ix":2},"a":{"a":0,"k":[200,200],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0,"parent":12},{"ddd":0,"refId":"6","w":202,"h":201,"ind":9,"ty":0,"nm":"Gemini_Generated_Image_a49ogpa49ogpa49o-Photoroom","sr":1,"ks":{"p":{"a":0,"k":[-1,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":2},"s":{"a":0,"k":[100,100],"ix":2},"r":{"a":0,"k":0,"ix":2},"o":{"a":0,"k":100,"ix":2},"sk":{"a":0,"k":0,"ix":2},"sa":{"a":0,"k":0,"ix":2}},"ao":0,"ip":0,"op":244,"st":0,"bm":0,"parent":13}],"markers":[]} \ No newline at end of file diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/Lottie/logo_lottie.json b/Prototype/Prototype/Prototype/ApplyDesign/Resources/Lottie/logo_lottie.json new file mode 100644 index 00000000..98e350cf --- /dev/null +++ b/Prototype/Prototype/Prototype/ApplyDesign/Resources/Lottie/logo_lottie.json @@ -0,0 +1 @@ +{"nm":"로고","ddd":0,"h":300,"w":300,"meta":{"g":"LottieFiles Figma v106"},"layers":[{"ty":4,"nm":"Gemini_Generated_Image_dacc7qdacc7qdacc-Photoroom 1 (Border)","sr":1,"st":0,"op":107,"ip":0,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[25,25]},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"p":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[86.40087257324568,125.40026882804793],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[86.40097822738178,110.40026952837809],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[86.40097822738178,110.40026952837809],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[93.36547983950662,131.2937685528882],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[90.71947922701725,123.35576892350292],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[91.60297943152757,126.00626879975457],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[91.3564793744684,125.26676883428087],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[91.4029793852321,125.4062688277678],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[91.40647938604226,125.41676882727756],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[91.39697938384322,125.3882688286082],"t":106.4929928779602},{"s":[91.40097938476914,125.40026882804793],"t":107}]},"r":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[21.407910113145988],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[-2.6024811970472683],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[-2.6024811970472683],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[30.841592858920876],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[18.13529377756663],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[22.377929922077783],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[21.19421763048524],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[21.41751426967005],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[21.434321543587203],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[21.388701800097806],"t":106.4929928779602},{"s":[21.407910113145988],"t":107}]},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"shapes":[{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":0,"k":{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}}},{"ty":"st","bm":0,"hd":false,"nm":"","lc":2,"lj":2,"ml":1,"o":{"a":0,"k":0},"w":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[1],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[1],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[1],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[1],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[1],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[1],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[1],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[1],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[1],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[1],"t":106.4929928779602},{"s":[1],"t":107}]},"c":{"a":0,"k":[0,0,0]}}],"ind":1},{"ty":4,"nm":"Gemini_Generated_Image_dacc7qdacc7qdacc-Photoroom 1 (Mask)","sr":1,"st":0,"op":107,"ip":0,"hd":false,"ddd":0,"bm":0,"hasMask":false,"td":1,"ao":0,"ks":{"a":{"a":0,"k":[0,0]},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"p":{"a":0,"k":[0,0]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"shapes":[{"ty":"gr","bm":0,"hd":false,"nm":"","it":[{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[50,0],[50,50],[0,50],[0,0]]}],"t":107}]}},{"ty":"fl","bm":0,"hd":false,"nm":"","c":{"a":0,"k":[1,0,0]},"r":2,"o":{"a":0,"k":100}},{"ty":"tr","a":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[25,25],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[25,25],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[25,25],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[25,25],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[25,25],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[25,25],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[25,25],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[25,25],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[25,25],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[25,25],"t":106.4929928779602},{"s":[25,25],"t":107}]},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"p":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[86.40087257324568,125.40026882804793],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[86.40097822738178,110.40026952837809],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[86.40097822738178,110.40026952837809],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[93.36547983950662,131.2937685528882],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[90.71947922701725,123.35576892350292],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[91.60297943152757,126.00626879975457],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[91.3564793744684,125.26676883428087],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[91.4029793852321,125.4062688277678],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[91.40647938604226,125.41676882727756],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[91.39697938384322,125.3882688286082],"t":106.4929928779602},{"s":[91.40097938476914,125.40026882804793],"t":107}]},"r":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[21.407910113145988],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[-2.6024811970472683],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[-2.6024811970472683],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[30.841592858920876],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[18.13529377756663],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[22.377929922077783],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[21.19421763048524],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[21.41751426967005],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[21.434321543587203],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[21.388701800097806],"t":106.4929928779602},{"s":[21.407910113145988],"t":107}]},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}}]}],"ind":2},{"ty":2,"nm":"Gemini_Generated_Image_dacc7qdacc7qdacc-Photoroom 1","sr":1,"st":0,"op":107,"ip":0,"hd":false,"ddd":0,"bm":0,"tt":1,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[512,512]},"s":{"a":0,"k":[4.8828125,4.8828125]},"sk":{"a":0,"k":0},"p":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[86.40087257324568,125.40026882804793],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[86.40097822738178,110.40026952837809],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[86.40097822738178,110.40026952837809],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[93.36547983950662,131.2937685528882],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[90.71947922701725,123.35576892350292],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[91.60297943152757,126.00626879975457],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[91.3564793744684,125.26676883428087],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[91.4029793852321,125.4062688277678],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[91.40647938604226,125.41676882727756],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[91.39697938384322,125.3882688286082],"t":106.4929928779602},{"s":[91.40097938476914,125.40026882804793],"t":107}]},"r":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[21.407910113145988],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[-2.6024811970472683],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[-2.6024811970472683],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[30.841592858920876],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[18.13529377756663],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[22.377929922077783],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[21.19421763048524],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[21.41751426967005],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[21.434321543587203],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[21.388701800097806],"t":106.4929928779602},{"s":[21.407910113145988],"t":107}]},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[0],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[100],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[100],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[100],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[100],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[100],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[100],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[100],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[100],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[100],"t":106.4929928779602},{"s":[100],"t":107}]}},"refId":"0a224e8e34eadd0e46bc2059027a2254cd08e303","ind":3},{"ty":4,"nm":"1인","sr":1,"st":0,"op":107.55,"ip":0,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[28.5,8],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[28.5,8],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[28.5,8],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[25.02,8],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[26.34,8],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[25.9,8],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[26.02,8],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[26,8],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[26,8],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[26,8],"t":106.4929928779602},{"s":[26,8],"t":107}]},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"p":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[82.5,94],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[82.5,70],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[82.5,70],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[77.61,105.88],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[79.48,92.5],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[78.86,97.03],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[79.03,95.77],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[79,96.01],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[79,96.03],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[79,95.98],"t":106.4929928779602},{"s":[79,96],"t":107}]},"r":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[0],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[-10],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[-10],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[3.93],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[-1.36],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[0.41],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[-0.08],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[0.01],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[0.02],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[0],"t":106.4929928779602},{"s":[0.01],"t":107}]},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[0],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[100],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[100],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[100],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[100],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[100],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[100],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[100],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[100],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[100],"t":106.4929928779602},{"s":[100],"t":107}]}},"shapes":[{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[15.01,-6.09],[15.01,23],[8.86,23],[8.86,-0.25],[8.69,-0.25],[2.03,3.92],[2.03,-1.53],[9.23,-6.09],[15.01,-6.09]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[15.0142,-6.09091],[15.0142,23],[8.86364,23],[8.86364,-0.252842],[8.69318,-0.252842],[2.03125,3.9233],[2.03125,-1.53125],[9.23295,-6.09091],[15.0142,-6.09091]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[15.0142,-6.09091],[15.0142,23],[8.86364,23],[8.86364,-0.252842],[8.69318,-0.252842],[2.03125,3.9233],[2.03125,-1.53125],[9.23295,-6.09091],[15.0142,-6.09091]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[12.92289994,-4.824638539],[12.92289994,20.214199999999998],[7.629015227,20.214199999999998],[7.629015227,0.20024580502],[7.482304172,0.20024580502],[1.7483102229999998,3.794693543],[1.7483102229999998,-0.9000852229999999],[7.946899358999999,-4.824638539],[12.92289994,-4.824638539]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[13.71744082,-5.305728967],[13.71744082,21.2726],[8.098082231,21.2726],[8.098082231,0.028105618060000004],[7.9423483159999995,0.028105618060000004],[1.855806619,3.843554579],[1.855806619,-1.139881619],[8.435504427,-5.305728967],[13.71744082,-5.305728967]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[13.45214344,-5.145092764],[13.45214344,20.9192],[7.941460652,20.9192],[7.941460652,0.08558326552000001],[7.788739472,0.08558326552000001],[1.819913548,3.8272398680000004],[1.819913548,-1.059813548],[8.272359084,-5.145092764],[13.45214344,-5.145092764]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[13.52616246,-5.189910901],[13.52616246,21.0178],[7.985158693,21.0178],[7.985158693,0.06954677417999999],[7.831596948,0.06954677417999999],[1.829927857,3.831791737],[1.829927857,-1.082152857],[8.317877281,-5.189910901],[13.52616246,-5.189910901]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[13.51219944,-5.181456364],[13.51219944,20.999200000000002],[7.976915452,20.999200000000002],[7.976915452,0.07257191351999998],[7.823512272,0.07257191351999998],[1.828038748,3.830933068],[1.828038748,-1.077938748],[8.309290683999999,-5.181456364],[13.51219944,-5.181456364]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[13.511148460000001,-5.180820001],[13.511148460000001,20.997799999999998],[7.976294993,20.997799999999998],[7.976294993,0.07279961218000003],[7.822903748,0.07279961218000003],[1.8278965569999999,3.8308684370000003],[1.8278965569999999,-1.0776215569999998],[8.308644380999999,-5.180820001],[13.511148460000001,-5.180820001]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[13.51400112,-5.182547272],[13.51400112,21.0016],[7.977979096,21.0016],[7.977979096,0.07218157296],[7.824555456,0.07218157296],[1.828282504,3.831043864],[1.828282504,-1.078482504],[8.310398631999998,-5.182547272],[13.51400112,-5.182547272]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[13.51,-5.18],[13.51,21],[7.98,21],[7.98,0.07],[7.82,0.07],[1.83,3.83],[1.83,-1.08],[8.31,-5.18],[13.51,-5.18]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[47.05,-10.48],[52.37,-10.48],[52.37,16.12],[47.05,16.12],[47.05,-10.48]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[47.0503,-10.48],[52.3703,-10.48],[52.3703,16.12],[47.0503,16.12],[47.0503,-10.48]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[47.0503,-10.48],[52.3703,-10.48],[52.3703,16.12],[47.0503,16.12],[47.0503,-10.48]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[40.5524215,-8.6023708],[45.1313987,-8.6023708],[45.1313987,14.292515199999999],[40.5524215,14.292515199999999],[40.5524215,-8.6023708]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.021139500000004,-9.3157324],[47.8816511,-9.3157324],[47.8816511,14.9868256],[43.021139500000004,14.9868256],[43.021139500000004,-9.3157324]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.196834,-9.0775408],[46.963341199999995,-9.0775408],[46.963341199999995,14.7549952],[42.196834,14.7549952],[42.196834,-9.0775408]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.4268185,-9.1439972],[47.2195533,-9.1439972],[47.2195533,14.8196768],[42.4268185,14.8196768],[42.4268185,-9.1439972]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.383434,-9.1314608],[47.1712212,-9.1314608],[47.1712212,14.807475199999999],[42.383434,14.807475199999999],[42.383434,-9.1314608]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.3801685,-9.1305172],[47.1675833,-9.1305172],[47.1675833,14.8065568],[42.3801685,14.8065568],[42.3801685,-9.1305172]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.389032,-9.133078399999999],[47.1774576,-9.133078399999999],[47.1774576,14.8090496],[42.389032,14.8090496],[42.389032,-9.133078399999999]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.39,-9.13],[47.17,-9.13],[47.17,14.81],[42.39,14.81],[42.39,-9.13]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[27.69,21.64],[53.33,21.64],[53.33,25.92],[27.69,25.92],[27.69,21.64]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[27.6903,21.64],[53.3303,21.64],[53.3303,25.92],[27.6903,25.92],[27.6903,21.64]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[27.6903,21.64],[53.3303,21.64],[53.3303,25.92],[27.6903,25.92],[27.6903,21.64]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[23.8890759,19.0436344],[45.9576803,19.0436344],[45.9576803,22.7274732],[23.8890759,22.7274732],[23.8890759,19.0436344]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[25.333262700000002,20.0300632],[48.758735900000005,20.0300632],[48.758735900000005,23.9403996],[25.333262700000002,23.9403996],[25.333262700000002,20.0300632]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.851048400000003,19.7006944],[47.8234628,19.7006944],[47.8234628,23.5354032],[24.851048400000003,23.5354032],[24.851048400000003,19.7006944]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.9855881,19.7925896],[48.0844077,19.7925896],[48.0844077,23.6483988],[24.9855881,23.6483988],[24.9855881,19.7925896]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.960208400000003,19.7752544],[48.0351828,19.7752544],[48.0351828,23.6270832],[24.960208400000003,23.6270832],[24.960208400000003,19.7752544]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.9582981,19.773949599999998],[48.0314777,19.773949599999998],[48.0314777,23.6254788],[24.9582981,23.6254788],[24.9582981,19.773949599999998]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.963483200000002,19.7774912],[48.0415344,19.7774912],[48.0415344,23.6298336],[24.963483200000002,23.6298336],[24.963483200000002,19.7774912]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.96,19.78],[48.04,19.78],[48.04,23.63],[24.96,23.63],[24.96,19.78]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[27.69,13.48],[33.01,13.48],[33.01,23.4],[27.69,23.4],[27.69,13.48]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[27.6903,13.48],[33.0103,13.48],[33.0103,23.4],[27.6903,23.4],[27.6903,13.48]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[27.6903,13.48],[33.0103,13.48],[33.0103,23.4],[27.6903,23.4],[27.6903,13.48]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[23.8890759,12.0202408],[28.468053100000002,12.0202408],[28.468053100000002,20.558484],[23.8890759,20.558484],[23.8890759,12.0202408]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[25.333262700000002,12.5748424],[30.1937743,12.5748424],[30.1937743,21.638052],[25.333262700000002,21.638052],[25.333262700000002,12.5748424]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.851048400000003,12.3896608],[29.617555600000003,12.3896608],[29.617555600000003,21.277584],[24.851048400000003,21.277584],[24.851048400000003,12.3896608]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.9855881,12.4413272],[29.778322900000003,12.4413272],[29.778322900000003,21.378156],[24.9855881,21.378156],[24.9855881,12.4413272]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.960208400000003,12.4315808],[29.747995600000003,12.4315808],[29.747995600000003,21.359184],[24.960208400000003,21.359184],[24.960208400000003,12.4315808]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.9582981,12.4308472],[29.7457129,12.4308472],[29.7457129,21.357756],[24.9582981,21.357756],[24.9582981,12.4308472]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.963483200000002,12.432838400000001],[29.751908800000002,12.432838400000001],[29.751908800000002,21.361632],[24.963483200000002,21.361632],[24.963483200000002,12.432838400000001]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[24.96,12.43],[29.75,12.43],[29.75,21.36],[24.96,21.36],[24.96,12.43]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[-1.52,-0.8],[-0.88,-1.41],[0,-1.84],[0.91,-1.44],[1.52,-0.83],[1.89,0],[1.52,0.8],[0.91,1.41],[0,1.79],[-0.88,1.41],[-1.52,0.8],[-1.89,0]],"o":[[1.89,0],[1.52,0.8],[0.91,1.41],[0,1.79],[-0.88,1.41],[-1.52,0.8],[-1.89,0],[-1.52,-0.83],[-0.88,-1.44],[0,-1.84],[0.91,-1.41],[1.52,-0.8],[0,0]],"v":[[32.21,-8.12],[37.33,-6.92],[40.93,-3.6],[42.29,1.28],[40.93,6.12],[37.33,9.48],[32.21,10.68],[27.09,9.48],[23.45,6.12],[22.13,1.28],[23.45,-3.6],[27.09,-6.92],[32.21,-8.12]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[-1.5200000000000031,-0.7999999999999998],[-0.8800000000000026,-1.4133400000000003],[0,-1.8400020000000001],[0.9067000000000007,-1.4400000000000004],[1.519999999999996,-0.82667],[1.8933000000000035,0],[1.5199999999999996,0.7999999999999989],[0.9067000000000007,1.4133300000000002],[0,1.7866699999999998],[-0.879999999999999,1.4133300000000002],[-1.5199999999999996,0.7999999999999998],[-1.8932999999999964,0]],"o":[[1.8933000000000035,0],[1.519999999999996,0.7999999999999998],[0.9067000000000007,1.4133300000000002],[0,1.7866699999999998],[-0.8800000000000026,1.4133300000000002],[-1.5200000000000031,0.7999999999999989],[-1.8932999999999964,0],[-1.5199999999999996,-0.82667],[-0.879999999999999,-1.4400000000000004],[0,-1.8400020000000001],[0.9067000000000007,-1.4133400000000003],[1.5199999999999996,-0.7999999999999998],[0,0]],"v":[[32.2103,-8.12],[37.3303,-6.92],[40.9303,-3.6],[42.2903,1.28],[40.9303,6.12],[37.3303,9.48],[32.2103,10.68],[27.0903,9.48],[23.4503,6.12],[22.1303,1.28],[23.4503,-3.6],[27.0903,-6.92],[32.2103,-8.12]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[-1.5200000000000031,-0.7999999999999998],[-0.8800000000000026,-1.4133400000000003],[0,-1.8400020000000001],[0.9067000000000007,-1.4400000000000004],[1.519999999999996,-0.82667],[1.8933000000000035,0],[1.5199999999999996,0.7999999999999989],[0.9067000000000007,1.4133300000000002],[0,1.7866699999999998],[-0.879999999999999,1.4133300000000002],[-1.5199999999999996,0.7999999999999998],[-1.8932999999999964,0]],"o":[[1.8933000000000035,0],[1.519999999999996,0.7999999999999998],[0.9067000000000007,1.4133300000000002],[0,1.7866699999999998],[-0.8800000000000026,1.4133300000000002],[-1.5200000000000031,0.7999999999999989],[-1.8932999999999964,0],[-1.5199999999999996,-0.82667],[-0.879999999999999,-1.4400000000000004],[0,-1.8400020000000001],[0.9067000000000007,-1.4133400000000003],[1.5199999999999996,-0.7999999999999998],[0,0]],"v":[[32.2103,-8.12],[37.3303,-6.92],[40.9303,-3.6],[42.2903,1.28],[40.9303,6.12],[37.3303,9.48],[32.2103,10.68],[27.0903,9.48],[23.4503,6.12],[22.1303,1.28],[23.4503,-3.6],[27.0903,-6.92],[32.2103,-8.12]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[-1.3082792000000016,-0.6885679999999997],[-0.7574248000000012,-1.2164675139999996],[0,-1.5837084],[0.7803639700000031,-1.2394224000000003],[1.3082791999999945,-0.7115189570000021],[1.6296240299999996,0],[1.308279200000003,0.6885679999999988],[0.7803639700000031,1.2164714430000003],[0,1.5378005570000002],[-0.7574247999999977,1.2164714429999999],[-1.308279199999998,0.6885679999999997],[-1.6296240300000022,0]],"o":[[1.6296240299999996,0],[1.3082791999999945,0.6885679999999997],[0.7803639700000031,1.2164714429999999],[0,1.5378005570000002],[-0.7574248000000012,1.2164714430000003],[-1.3082792000000016,0.6885679999999988],[-1.6296240300000022,0],[-1.308279199999998,-0.7115189570000021],[-0.7574247999999977,-1.2394224000000003],[0,-1.5837084],[0.7803639700000031,-1.2164675139999996],[1.308279200000003,-0.6885679999999997],[0,0]],"v":[[27.779485100000002,-6.5710952],[32.186320300000006,-5.5382432],[35.28487629999999,-2.6806859999999997],[36.455441900000004,1.5195788],[35.28487629999999,5.6854152],[32.186320300000006,8.577400800000001],[27.779485100000002,9.610252800000001],[23.3726499,8.577400800000001],[20.239665499999997,5.6854152],[19.1035283,1.5195788],[20.239665499999997,-2.6806859999999997],[23.3726499,-5.5382432],[27.779485100000002,-6.5710952]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[-1.3887176000000023,-0.7309039999999998],[-0.8039944000000017,-1.2912646419999998],[0,-1.6810812],[0.8283624100000022,-1.3156272000000002],[1.3887175999999952,-0.7552679210000013],[1.729801590000001,0],[1.3887176000000017,0.7309039999999989],[0.8283624100000022,1.2912632790000003],[0,1.632352721],[-0.8039943999999981,1.291263279],[-1.3887175999999988,0.7309039999999998],[-1.7298015900000001,0]],"o":[[1.729801590000001,0],[1.3887175999999952,0.7309039999999998],[0.8283624100000022,1.291263279],[0,1.632352721],[-0.8039944000000017,1.2912632790000003],[-1.3887176000000023,0.7309039999999989],[-1.7298015900000001,0],[-1.3887175999999988,-0.7552679210000013],[-0.8039943999999981,-1.3156272000000002],[0,-1.6810812],[0.8283624100000022,-1.2912646419999998],[1.3887176000000017,-0.7309039999999998],[0,0]],"v":[[29.4628703,-7.1595656],[34.140655900000006,-6.0632095999999995],[37.4297239,-3.0299579999999997],[38.6722607,1.4285564],[37.4297239,5.8505256],[34.140655900000006,8.9203224],[29.4628703,10.0166784],[24.7850847,8.9203224],[21.4594715,5.8505256],[20.2534799,1.4285564],[21.4594715,-3.0299579999999997],[24.7850847,-6.0632095999999995],[29.4628703,-7.1595656]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[-1.361859200000002,-0.7167679999999997],[-0.7884448000000015,-1.2662898639999998],[0,-1.6485684],[0.8123357200000025,-1.2901824000000002],[1.361859199999995,-0.7406601320000016],[1.6963522800000006,0],[1.3618592000000023,0.7167679999999989],[0.8123357200000025,1.2662902680000003],[0,1.6007817320000002],[-0.788444799999998,1.266290268],[-1.3618591999999985,0.7167679999999997],[-1.6963522800000008,0]],"o":[[1.6963522800000006,0],[1.361859199999995,0.7167679999999997],[0.8123357200000025,1.266290268],[0,1.6007817320000002],[-0.7884448000000015,1.2662902680000003],[-1.361859200000002,0.7167679999999989],[-1.6963522800000008,0],[-1.3618591999999985,-0.7406601320000016],[-0.788444799999998,-1.2901824000000002],[0,-1.6485684],[0.8123357200000025,-1.2662898639999998],[1.3618592000000023,-0.7167679999999997],[0,0]],"v":[[28.9007876,-6.9630752000000005],[33.48810280000001,-5.8879231999999995],[36.7135588,-2.913336],[37.9320644,1.4589488],[36.7135588,5.7953952],[33.48810280000001,8.805820800000001],[28.9007876,9.8809728],[24.3134724,8.805820800000001],[21.052177999999998,5.7953952],[19.8695108,1.4589488],[21.052177999999998,-2.913336],[24.3134724,-5.8879231999999995],[28.9007876,-6.9630752000000005]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[-1.3693528000000021,-0.7207119999999998],[-0.7927832000000016,-1.2732579259999999],[0,-1.6576396],[0.8168072300000024,-1.2972816000000003],[1.369352799999995,-0.7447357630000015],[1.7056847700000006,0],[1.3693528000000021,0.7207119999999989],[0.8168072300000024,1.2732578370000003],[0,1.609590163],[-0.792783199999998,1.273257837],[-1.3693527999999986,0.7207119999999998],[-1.7056847700000006,0]],"o":[[1.7056847700000006,0],[1.369352799999995,0.7207119999999998],[0.8168072300000024,1.273257837],[0,1.609590163],[-0.7927832000000016,1.2732578370000003],[-1.3693528000000021,0.7207119999999989],[-1.7056847700000006,0],[-1.3693527999999986,-0.7447357630000015],[-0.792783199999998,-1.2972816000000003],[0,-1.6576396],[0.8168072300000024,-1.2732579259999999],[1.3693528000000021,-0.7207119999999998],[0,0]],"v":[[29.0576109,-7.0178968],[33.6701677,-5.9368288],[36.9133717,-2.945874],[38.1385821,1.4504692],[36.9133717,5.8107768],[33.6701677,8.8377672],[29.0576109,9.9188352],[24.4450541,8.8377672],[21.1658145,5.8107768],[19.9766397,1.4504692],[21.1658145,-2.945874],[24.4450541,-5.9368288],[29.0576109,-7.0178968]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[-1.3679392000000021,-0.7199679999999997],[-0.7919648000000016,-1.2719434639999998],[0,-1.6559284],[0.8159637200000025,-1.2959424000000004],[1.367939199999995,-0.7439669320000015],[1.7039242800000007,0],[1.3679392000000021,0.7199679999999988],[0.8159637200000025,1.2719434680000004],[0,1.607928532],[-0.791964799999998,1.271943468],[-1.3679391999999986,0.7199679999999997],[-1.7039242800000007,0]],"o":[[1.7039242800000007,0],[1.367939199999995,0.7199679999999997],[0.8159637200000025,1.271943468],[0,1.607928532],[-0.7919648000000016,1.2719434680000004],[-1.3679392000000021,0.7199679999999988],[-1.7039242800000007,0],[-1.3679391999999986,-0.7439669320000015],[-0.791964799999998,-1.2959424000000004],[0,-1.6559284],[0.8159637200000025,-1.2719434639999998],[1.3679392000000021,-0.7199679999999997],[0,0]],"v":[[29.028027599999998,-7.0075552000000005],[33.63582280000001,-5.9276032],[36.875678799999996,-2.939736],[38.0996244,1.4520688],[36.875678799999996,5.8078752],[33.63582280000001,8.8317408],[29.028027599999998,9.9116928],[24.4202324,8.8317408],[21.144378,5.8078752],[19.9564308,1.4520688],[21.144378,-2.939736],[24.4202324,-5.9276032],[29.028027599999998,-7.0075552000000005]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[-1.3678328000000022,-0.7199119999999998],[-0.7919032000000016,-1.2718445259999998],[0,-1.6557996],[0.8159002300000024,-1.2958416000000001],[1.367832799999995,-0.7439090630000016],[1.7037917700000005,0],[1.3678328000000022,0.7199119999999989],[0.8159002300000024,1.2718445370000002],[0,1.607803463],[-0.791903199999998,1.271844537],[-1.3678327999999986,0.7199119999999998],[-1.7037917700000007,0]],"o":[[1.7037917700000005,0],[1.367832799999995,0.7199119999999998],[0.8159002300000024,1.271844537],[0,1.607803463],[-0.7919032000000016,1.2718445370000002],[-1.3678328000000022,0.7199119999999989],[-1.7037917700000007,0],[-1.3678327999999986,-0.7439090630000016],[-0.791903199999998,-1.2958416000000001],[0,-1.6557996],[0.8159002300000024,-1.2718445259999998],[1.3678328000000022,-0.7199119999999998],[0,0]],"v":[[29.0258009,-7.0067768],[33.6332377,-5.9269088],[36.872841699999995,-2.9392739999999997],[38.0966921,1.4521892],[36.872841699999995,5.8076568],[33.6332377,8.8312872],[29.0258009,9.911155200000001],[24.418364099999998,8.8312872],[21.1427645,5.8076568],[19.9549097,1.4521892],[21.1427645,-2.9392739999999997],[24.418364099999998,-5.9269088],[29.0258009,-7.0067768]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[-1.368121600000002,-0.7200639999999998],[-0.7920704000000016,-1.2721130719999998],[0,-1.6561492],[0.8160725600000025,-1.2961152000000002],[1.368121599999995,-0.7440661360000016],[1.7041514400000006,0],[1.368121600000002,0.7200639999999989],[0.8160725600000025,1.2721130640000002],[0,1.6081429360000001],[-0.7920703999999981,1.272113064],[-1.3681215999999985,0.7200639999999998],[-1.7041514400000006,0]],"o":[[1.7041514400000006,0],[1.368121599999995,0.7200639999999998],[0.8160725600000025,1.272113064],[0,1.6081429360000001],[-0.7920704000000016,1.2721130640000002],[-1.368121600000002,0.7200639999999989],[-1.7041514400000006,0],[-1.3681215999999985,-0.7440661360000016],[-0.7920703999999981,-1.2961152000000002],[0,-1.6561492],[0.8160725600000025,-1.2721130719999998],[1.368121600000002,-0.7200639999999998],[0,0]],"v":[[29.0318448,-7.0088896],[33.6402544,-5.9287936],[36.880542399999996,-2.940528],[38.1046512,1.4518624],[36.880542399999996,5.8082496],[33.6402544,8.832518400000001],[29.0318448,9.9126144],[24.4234352,8.832518400000001],[21.147143999999997,5.8082496],[19.9590384,1.4518624],[21.147143999999997,-2.940528],[24.4234352,-5.9287936],[29.0318448,-7.0088896]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[-1.37,-0.72],[-0.79,-1.27],[0,-1.66],[0.82,-1.3],[1.37,-0.74],[1.7,0],[1.37,0.72],[0.82,1.27],[0,1.61],[-0.79,1.27],[-1.37,0.72],[-1.7,0]],"o":[[1.7,0],[1.37,0.72],[0.82,1.27],[0,1.61],[-0.79,1.27],[-1.37,0.72],[-1.7,0],[-1.37,-0.74],[-0.79,-1.3],[0,-1.66],[0.82,-1.27],[1.37,-0.72],[0,0]],"v":[[29.03,-7.01],[33.64,-5.93],[36.88,-2.94],[38.1,1.45],[36.88,5.81],[33.64,8.83],[29.03,9.91],[24.42,8.83],[21.15,5.81],[19.96,1.45],[21.15,-2.94],[24.42,-5.93],[29.03,-7.01]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0.75,-0.4],[0.43,-0.72],[0,-1.04],[-0.43,-0.72],[-0.75,-0.4],[-0.91,0],[-0.75,0.37],[-0.43,0.69],[0,0.99],[0.43,0.69],[0.75,0.37],[0.91,0]],"o":[[-0.91,0],[-0.75,0.37],[-0.43,0.69],[0,0.99],[0.43,0.69],[0.75,0.37],[0.91,0],[0.75,-0.4],[0.43,-0.72],[0,-1.04],[-0.43,-0.72],[-0.75,-0.4],[0,0]],"v":[[32.21,-3.56],[29.73,-2.96],[27.97,-1.32],[27.33,1.28],[27.97,3.84],[29.73,5.48],[32.21,6.04],[34.69,5.48],[36.45,3.84],[37.09,1.28],[36.45,-1.32],[34.69,-2.96],[32.21,-3.56]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0.7467000000000006,-0.3999999999999999],[0.42669999999999675,-0.72],[0,-1.040001],[-0.4267000000000003,-0.7199999999999998],[-0.7467000000000006,-0.40000000000000036],[-0.9066999999999972,0],[-0.746699999999997,0.3733299999999993],[-0.42669999999999675,0.6933300000000004],[0,0.9866599999999999],[0.42670000000000385,0.6933320000000001],[0.746699999999997,0.37333000000000016],[0.9067000000000007,0]],"o":[[-0.9066999999999972,0],[-0.7467000000000006,0.37333000000000016],[-0.4267000000000003,0.6933320000000001],[0,0.9866599999999999],[0.42669999999999675,0.6933300000000004],[0.7467000000000006,0.3733299999999993],[0.9067000000000007,0],[0.746699999999997,-0.40000000000000036],[0.42670000000000385,-0.7199999999999998],[0,-1.040001],[-0.42669999999999675,-0.72],[-0.746699999999997,-0.3999999999999999],[0,0]],"v":[[32.2103,-3.56],[29.7303,-2.96],[27.9703,-1.32],[27.3303,1.28],[27.9703,3.84],[29.7303,5.48],[32.2103,6.04],[34.6903,5.48],[36.4503,3.84],[37.0903,1.28],[36.4503,-1.32],[34.6903,-2.96],[32.2103,-3.56]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0.7467000000000006,-0.3999999999999999],[0.42669999999999675,-0.72],[0,-1.040001],[-0.4267000000000003,-0.7199999999999998],[-0.7467000000000006,-0.40000000000000036],[-0.9066999999999972,0],[-0.746699999999997,0.3733299999999993],[-0.42669999999999675,0.6933300000000004],[0,0.9866599999999999],[0.42670000000000385,0.6933320000000001],[0.746699999999997,0.37333000000000016],[0.9067000000000007,0]],"o":[[-0.9066999999999972,0],[-0.7467000000000006,0.37333000000000016],[-0.4267000000000003,0.6933320000000001],[0,0.9866599999999999],[0.42669999999999675,0.6933300000000004],[0.7467000000000006,0.3733299999999993],[0.9067000000000007,0],[0.746699999999997,-0.40000000000000036],[0.42670000000000385,-0.7199999999999998],[0,-1.040001],[-0.42669999999999675,-0.72],[-0.746699999999997,-0.3999999999999999],[0,0]],"v":[[32.2103,-3.56],[29.7303,-2.96],[27.9703,-1.32],[27.3303,1.28],[27.9703,3.84],[29.7303,5.48],[32.2103,6.04],[34.6903,5.48],[36.4503,3.84],[37.0903,1.28],[36.4503,-1.32],[34.6903,-2.96],[32.2103,-3.56]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0.6426503700000006,-0.3442840000000005],[0.36722317000000176,-0.6197098071],[0,-0.8951393999999999],[-0.3672231700000004,-0.6197111999999997],[-0.6426503700000006,-0.3442840000000003],[-0.7803639699999996,0],[-0.642650369999997,0.3213330429999995],[-0.36722317000000176,0.596760243],[0,0.8492364859999999],[0.36722316999999893,0.5967594572000001],[0.642650370000002,0.32133304299999976],[0.7803639700000031,0]],"o":[[-0.7803639699999996,0],[-0.6426503700000006,0.32133304299999976],[-0.3672231700000004,0.5967594572000001],[0,0.8492364859999999],[0.36722317000000176,0.596760243],[0.6426503700000006,0.3213330429999995],[0.7803639700000031,0],[0.642650370000002,-0.3442840000000003],[0.36722316999999893,-0.6197111999999997],[0,-0.8951393999999999],[-0.36722317000000176,-0.6197098071],[-0.642650369999997,-0.3442840000000005],[0,0]],"v":[[27.779485100000002,-2.6462575999999998],[25.6449243,-2.1298315999999997],[24.1300747,-0.7182685929],[23.5792203,1.5195788],[24.1300747,3.7229963999999995],[25.6449243,5.1345608],[27.779485100000002,5.6165584],[29.914045899999998,5.1345608],[31.428895500000003,3.7229963999999995],[31.9797499,1.5195788],[31.428895500000003,-0.7182685929],[29.914045899999998,-2.1298315999999997],[27.779485100000002,-2.6462575999999998]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0.6821816100000005,-0.3654520000000003],[0.38982000999999983,-0.6578127363],[0,-0.9501762],[-0.3898200100000003,-0.6578135999999997],[-0.6821816100000005,-0.36545200000000033],[-0.8283624099999987,0],[-0.682181609999997,0.3410880789999994],[-0.38982000999999983,0.6334496790000002],[0,0.9014473579999999],[0.3898200100000008,0.6334499516000001],[0.6821816100000001,0.3410880789999999],[0.8283624100000022,0]],"o":[[-0.8283624099999987,0],[-0.6821816100000005,0.3410880789999999],[-0.3898200100000003,0.6334499516000001],[0,0.9014473579999999],[0.38982000999999983,0.6334496790000002],[0.6821816100000005,0.3410880789999994],[0.8283624100000022,0],[0.6821816100000001,-0.36545200000000033],[0.3898200100000008,-0.6578135999999997],[0,-0.9501762],[-0.38982000999999983,-0.6578127363],[-0.682181609999997,-0.3654520000000003],[0,0]],"v":[[29.4628703,-2.9934127999999998],[27.1970679,-2.4452347999999997],[25.5890791,-0.9468824637],[25.0043559,1.4285564],[25.5890791,3.7674491999999997],[27.1970679,5.2658024],[29.4628703,5.777435199999999],[31.728672699999997,5.2658024],[33.3366615,3.7674491999999997],[33.921384700000004,1.4285564],[33.3366615,-0.9468824637],[31.728672699999997,-2.4452347999999997],[29.4628703,-2.9934127999999998]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0.6689821200000006,-0.3583840000000003],[0.3822749200000005,-0.6450901596],[0,-0.9317993999999999],[-0.38227492000000035,-0.6450911999999996],[-0.6689821200000006,-0.3583840000000003],[-0.812335719999999,0],[-0.668982119999997,0.3344918679999994],[-0.3822749200000005,0.6211990680000001],[0,0.8840141359999999],[0.3822749200000002,0.6211989872000001],[0.6689821200000008,0.33449186799999986],[0.8123357200000025,0]],"o":[[-0.812335719999999,0],[-0.6689821200000006,0.33449186799999986],[-0.38227492000000035,0.6211989872000001],[0,0.8840141359999999],[0.3822749200000005,0.6211990680000001],[0.6689821200000006,0.3344918679999994],[0.8123357200000025,0],[0.6689821200000008,-0.3583840000000003],[0.3822749200000002,-0.6450911999999996],[0,-0.9317993999999999],[-0.3822749200000005,-0.6450901596],[-0.668982119999997,-0.3583840000000003],[0,0]],"v":[[28.9007876,-2.8774976],[26.6788068,-2.3399216],[25.1019172,-0.8705482404],[24.5285028,1.4589488],[25.1019172,3.7526064],[26.6788068,5.2219808],[28.9007876,5.7237184],[31.122768399999998,5.2219808],[32.699658,3.7526064],[33.273072400000004,1.4589488],[32.699658,-0.8705482404],[31.122768399999998,-2.3399216],[28.9007876,-2.8774976]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0.6726648300000007,-0.36035600000000034],[0.3843800300000003,-0.6486398089],[0,-0.9369265999999999],[-0.3843800300000003,-0.6486407999999997],[-0.6726648300000007,-0.36035600000000034],[-0.8168072299999989,0],[-0.6726648299999971,0.3363322369999994],[-0.3843800300000003,0.6246170370000002],[0,0.888878074],[0.38438003000000037,0.6246170548000001],[0.6726648300000005,0.33633223699999987],[0.8168072300000024,0]],"o":[[-0.8168072299999989,0],[-0.6726648300000007,0.33633223699999987],[-0.3843800300000003,0.6246170548000001],[0,0.888878074],[0.3843800300000003,0.6246170370000002],[0.6726648300000007,0.3363322369999994],[0.8168072300000024,0],[0.6726648300000005,-0.36035600000000034],[0.38438003000000037,-0.6486407999999997],[0,-0.9369265999999999],[-0.3843800300000003,-0.6486398089],[-0.6726648299999971,-0.36035600000000034],[0,0]],"v":[[29.0576109,-2.9098384],[26.8234037,-2.3693044],[25.2378373,-0.8918457911000001],[24.6612677,1.4504692],[25.2378373,3.7567475999999997],[26.8234037,5.2342072],[29.0576109,5.738705599999999],[31.2918181,5.2342072],[32.877384500000005,3.7567475999999997],[33.453954100000004,1.4504692],[32.877384500000005,-0.8918457911000001],[31.2918181,-2.3693044],[29.0576109,-2.9098384]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0.6719701200000006,-0.3599840000000003],[0.38398292000000034,-0.6479701996],[0,-0.9359594],[-0.38398292000000034,-0.6479711999999997],[-0.6719701200000006,-0.3599840000000003],[-0.815963719999999,0],[-0.671970119999997,0.3359850679999994],[-0.38398292000000034,0.6239722680000002],[0,0.8879605359999999],[0.38398292000000034,0.6239722672000001],[0.6719701200000006,0.33598506799999983],[0.8159637200000025,0]],"o":[[-0.815963719999999,0],[-0.6719701200000006,0.33598506799999983],[-0.38398292000000034,0.6239722672000001],[0,0.8879605359999999],[0.38398292000000034,0.6239722680000002],[0.6719701200000006,0.3359850679999994],[0.8159637200000025,0],[0.6719701200000006,-0.3599840000000003],[0.38398292000000034,-0.6479711999999997],[0,-0.9359594],[-0.38398292000000034,-0.6479701996],[-0.671970119999997,-0.3599840000000003],[0,0]],"v":[[29.028027599999998,-2.9037376],[26.7961268,-2.3637616],[25.2121972,-0.8878282004],[24.6362228,1.4520688],[25.2121972,3.7559663999999997],[26.7961268,5.2319008],[29.028027599999998,5.7358784],[31.2599284,5.2319008],[32.843858000000004,3.7559663999999997],[33.419832400000004,1.4520688],[32.843858000000004,-0.8878282004],[31.2599284,-2.3637616],[29.028027599999998,-2.9037376]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0.6719178300000006,-0.35995600000000033],[0.38395303000000036,-0.6479197989],[0,-0.9358865999999999],[-0.38395303000000036,-0.6479207999999996],[-0.6719178300000006,-0.35995600000000033],[-0.8159002299999989,0],[-0.671917829999997,0.3359589369999994],[-0.38395303000000036,0.6239237370000001],[0,0.8878914739999999],[0.38395303000000036,0.6239237348000001],[0.6719178300000006,0.33595893699999985],[0.8159002300000024,0]],"o":[[-0.8159002299999989,0],[-0.6719178300000006,0.33595893699999985],[-0.38395303000000036,0.6239237348000001],[0,0.8878914739999999],[0.38395303000000036,0.6239237370000001],[0.6719178300000006,0.3359589369999994],[0.8159002300000024,0],[0.6719178300000006,-0.35995600000000033],[0.38395303000000036,-0.6479207999999996],[0,-0.9358865999999999],[-0.38395303000000036,-0.6479197989],[-0.671917829999997,-0.35995600000000033],[0,0]],"v":[[29.0258009,-2.9032783999999996],[26.7940737,-2.3633444],[25.210267299999998,-0.8875258011],[24.6343377,1.4521892],[25.210267299999998,3.7559075999999996],[26.7940737,5.2317272],[29.0258009,5.7356656],[31.2575281,5.2317272],[32.8413345,3.7559075999999996],[33.417264100000004,1.4521892],[32.8413345,-0.8875258011],[31.2575281,-2.3633444],[29.0258009,-2.9032783999999996]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0.6720597600000006,-0.3600320000000003],[0.38403416000000035,-0.6480566007999999],[0,-0.9360842],[-0.38403416000000035,-0.6480575999999997],[-0.6720597600000006,-0.36003200000000035],[-0.816072559999999,0],[-0.672059759999997,0.3360298639999994],[-0.38403416000000035,0.6240554640000001],[0,0.8880789279999999],[0.38403416000000035,0.6240554656000001],[0.6720597600000006,0.33602986399999984],[0.8160725600000025,0]],"o":[[-0.816072559999999,0],[-0.6720597600000006,0.33602986399999984],[-0.38403416000000035,0.6240554656000001],[0,0.8880789279999999],[0.38403416000000035,0.6240554640000001],[0.6720597600000006,0.3360298639999994],[0.8160725600000025,0],[0.6720597600000006,-0.36003200000000035],[0.38403416000000035,-0.6480575999999997],[0,-0.9360842],[-0.38403416000000035,-0.6480566007999999],[-0.672059759999997,-0.3600320000000003],[0,0]],"v":[[29.0318448,-2.9045248],[26.7996464,-2.3644768],[25.2155056,-0.8883465992],[24.639454399999998,1.4518624],[25.2155056,3.7560672],[26.7996464,5.232198400000001],[29.0318448,5.7362432],[31.2640432,5.232198400000001],[32.848184,3.7560672],[33.424235200000005,1.4518624],[32.848184,-0.8883465992],[31.2640432,-2.3644768],[29.0318448,-2.9045248]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0.67,-0.36],[0.38,-0.65],[0,-0.94],[-0.38,-0.65],[-0.67,-0.36],[-0.82,0],[-0.67,0.34],[-0.38,0.62],[0,0.89],[0.38,0.62],[0.67,0.34],[0.82,0]],"o":[[-0.82,0],[-0.67,0.34],[-0.38,0.62],[0,0.89],[0.38,0.62],[0.67,0.34],[0.82,0],[0.67,-0.36],[0.38,-0.65],[0,-0.94],[-0.38,-0.65],[-0.67,-0.36],[0,0]],"v":[[29.03,-2.9],[26.8,-2.36],[25.21,-0.89],[24.64,1.45],[25.21,3.76],[26.8,5.23],[29.03,5.74],[31.26,5.23],[32.85,3.76],[33.42,1.45],[32.85,-0.89],[31.26,-2.36],[29.03,-2.9]]}],"t":107}]}},{"ty":"fl","bm":0,"hd":false,"nm":"","c":{"a":0,"k":[0,0,0]},"r":1,"o":{"a":0,"k":100}}],"ind":4},{"ty":4,"nm":"개발자 키우기","sr":1,"st":0,"op":107.55,"ip":0,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[76.5,32]},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"p":{"a":0,"k":[169.5,179]},"r":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[0],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[18.95],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[18.95],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[-7.44],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[2.59],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[-0.76],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[0.17],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[-0.01],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[-0.02],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[0.02],"t":106.4929928779602},{"s":[0],"t":107}]},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"shapes":[{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":0,"k":{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[28.28,-10.52],[33.32,-10.52],[33.32,26.52],[28.28,26.52],[28.28,-10.52]]}}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":0,"k":{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[23.56,3.44],[29.72,3.44],[29.72,7.72],[23.56,7.72],[23.56,3.44]]}}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0.45,-2.35],[1.01,-2.11],[1.73,-1.81],[2.59,-1.52],[0,0],[-1.65,1.97],[-0.72,2.4],[0,2.96],[0,0]],"o":[[0,0],[0,2.61],[-0.43,2.35],[-1.01,2.08],[-1.71,1.79],[0,0],[2.8,-1.65],[1.68,-2],[0.72,-2.43],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0.4532999999999987,-2.346669],[1.0132999999999992,-2.1066700000000003],[1.7332999999999998,-1.8133],[2.58667,-1.5199999999999996],[0,0],[-1.6533299999999995,1.9733],[-0.7199999999999989,2.3999999999999995],[0,2.96],[0,0]],"o":[[0,0],[0,2.61333],[-0.4267000000000003,2.34667],[-1.013300000000001,2.08],[-1.706669999999999,1.7866999999999997],[0,0],[2.8,-1.6532999999999998],[1.6799999999999997,-2.000000000000001],[0.7200000000000006,-2.4266680000000003],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0.45,-2.35],[1.01,-2.11],[1.73,-1.81],[2.59,-1.52],[0,0],[-1.65,1.97],[-0.72,2.4],[0,2.96],[0,0]],"o":[[0,0],[0,2.61],[-0.43,2.35],[-1.01,2.08],[-1.71,1.79],[0,0],[2.8,-1.65],[1.68,-2],[0.72,-2.43],[0,0],[0,0]],"v":[[12.52,-5.96],[17.64,-5.96],[16.96,1.48],[14.8,8.16],[10.68,14],[4.24,18.96],[1.16,15.24],[7.84,9.8],[11.44,3.2],[12.52,-4.88],[12.52,-5.96]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":0,"k":{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[3,-5.96],[14.08,-5.96],[14.08,-1.72],[3,-1.72],[3,-5.96]]}}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":0,"k":{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[20.04,-9.56],[25.04,-9.56],[25.04,24.92],[20.04,24.92],[20.04,-9.56]]}}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.88,-8.88],[45.16,-8.88],[45.16,-4.48],[52.44,-4.48],[52.44,-8.88],[57.68,-8.88],[57.68,7.32],[39.88,7.32],[39.88,-8.88]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.8769,-8.88],[45.1569,-8.88],[45.1569,-4.48],[52.4369,-4.48],[52.4369,-8.88],[57.6769,-8.88],[57.6769,7.32],[39.8769,7.32],[39.8769,-8.88]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[39.88,-8.88],[45.16,-8.88],[45.16,-4.48],[52.44,-4.48],[52.44,-8.88],[57.68,-8.88],[57.68,7.32],[39.88,7.32],[39.88,-8.88]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.16,-0.52],[45.16,3.2],[52.44,3.2],[52.44,-0.52],[45.16,-0.52]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.1569,-0.52],[45.1569,3.2],[52.4369,3.2],[52.4369,-0.52],[45.1569,-0.52]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[45.16,-0.52],[45.16,3.2],[52.44,3.2],[52.44,-0.52],[45.16,-0.52]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.64,-10.48],[67.96,-10.48],[67.96,8.32],[62.64,8.32],[62.64,-10.48]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.6369,-10.48],[67.9569,-10.48],[67.9569,8.32],[62.6369,8.32],[62.6369,-10.48]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[62.64,-10.48],[67.96,-10.48],[67.96,8.32],[62.64,8.32],[62.64,-10.48]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.84,-3.4],[72.84,-3.4],[72.84,0.96],[65.84,0.96],[65.84,-3.4]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.8369,-3.4],[72.8369,-3.4],[72.8369,0.96],[65.8369,0.96],[65.8369,-3.4]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.84,-3.4],[72.84,-3.4],[72.84,0.96],[65.84,0.96],[65.84,-3.4]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.52,9.72],[67.96,9.72],[67.96,19.84],[48.84,19.84],[48.84,24.2],[43.56,24.2],[43.56,16.04],[62.72,16.04],[62.72,13.8],[43.52,13.8],[43.52,9.72]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5169,9.72],[67.9569,9.72],[67.9569,19.84],[48.8369,19.84],[48.8369,24.2],[43.5569,24.2],[43.5569,16.04],[62.7169,16.04],[62.7169,13.8],[43.5169,13.8],[43.5169,9.72]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.52,9.72],[67.96,9.72],[67.96,19.84],[48.84,19.84],[48.84,24.2],[43.56,24.2],[43.56,16.04],[62.72,16.04],[62.72,13.8],[43.52,13.8],[43.52,9.72]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.56,22.16],[69.04,22.16],[69.04,26.32],[43.56,26.32],[43.56,22.16]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.5569,22.16],[69.0369,22.16],[69.0369,26.32],[43.5569,26.32],[43.5569,22.16]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[43.56,22.16],[69.04,22.16],[69.04,26.32],[43.56,26.32],[43.56,22.16]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.45,-2.05],[0.85,-1.81],[1.31,-1.41],[1.71,-0.8],[0,0],[-1.12,1.17],[-0.75,1.44],[-0.37,1.55],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05],[-0.43,2.05],[-0.85,1.79],[-1.28,1.41],[0,0],[1.55,-0.72],[1.15,-1.17],[0.77,-1.44],[0.37,-1.57],[0,0],[0,0]],"v":[[84.31,-4.64],[88.51,-4.64],[88.51,-0.48],[87.83,5.68],[85.91,11.48],[82.67,16.28],[78.19,19.6],[75.19,15.4],[79.19,12.56],[82.03,8.64],[83.75,4.16],[84.31,-0.48],[84.31,-4.64]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.4532999999999987,-2.0533399999999995],[0.8533000000000044,-1.8133300000000006],[1.306600000000003,-1.4133000000000013],[1.7066000000000088,-0.8000000000000007],[0,0],[-1.1199999999999903,1.1732999999999993],[-0.7467000000000041,1.4399999999999995],[-0.37340000000000373,1.5466699999999998],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05333],[-0.42669999999999675,2.05333],[-0.8533999999999935,1.7866999999999997],[-1.2800000000000011,1.4132999999999996],[0,0],[1.546599999999998,-0.7200000000000006],[1.1466000000000065,-1.1733000000000011],[0.7733000000000061,-1.4400000000000004],[0.3733000000000004,-1.5733300000000003],[0,0],[0,0]],"v":[[84.3138,-4.64],[88.5138,-4.64],[88.5138,-0.48],[87.8338,5.68],[85.9138,11.48],[82.6738,16.28],[78.1938,19.6],[75.1938,15.4],[79.1938,12.56],[82.0338,8.64],[83.7538,4.16],[84.3138,-0.48],[84.3138,-4.64]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0.45,-2.05],[0.85,-1.81],[1.31,-1.41],[1.71,-0.8],[0,0],[-1.12,1.17],[-0.75,1.44],[-0.37,1.55],[0,1.52],[0,0]],"o":[[0,0],[0,0],[0,2.05],[-0.43,2.05],[-0.85,1.79],[-1.28,1.41],[0,0],[1.55,-0.72],[1.15,-1.17],[0.77,-1.44],[0.37,-1.57],[0,0],[0,0]],"v":[[84.31,-4.64],[88.51,-4.64],[88.51,-0.48],[87.83,5.68],[85.91,11.48],[82.67,16.28],[78.19,19.6],[75.19,15.4],[79.19,12.56],[82.03,8.64],[83.75,4.16],[84.31,-0.48],[84.31,-4.64]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.37,-1.47],[-0.72,-1.36],[-1.12,-1.12],[-1.52,-0.69],[0,0],[1.28,1.33],[0.88,1.71],[0.43,1.92],[0,1.97],[0,0]],"o":[[0,0],[0,0],[0,1.39],[0.37,1.47],[0.75,1.36],[1.15,1.09],[0,0],[-1.71,-0.8],[-1.25,-1.36],[-0.85,-1.73],[-0.43,-1.95],[0,0],[0,0]],"v":[[85.43,-4.64],[89.63,-4.64],[89.63,-0.48],[90.19,3.8],[91.83,8.04],[94.63,11.76],[98.63,14.44],[95.67,18.68],[91.19,15.48],[87.99,10.88],[86.07,5.4],[85.43,-0.48],[85.43,-4.64]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.3733999999999895,-1.4666699999999997],[-0.7199999999999989,-1.3599999999999994],[-1.1199999999999903,-1.1199999999999992],[-1.519999999999996,-0.6932999999999989],[0,0],[1.2800000000000011,1.3333000000000013],[0.8800000000000097,1.7066999999999997],[0.42659999999999343,1.92],[0,1.97333],[0,0]],"o":[[0,0],[0,0],[0,1.386666],[0.3733000000000004,1.4666700000000006],[0.7466000000000008,1.3600000000000012],[1.1466000000000065,1.093300000000001],[0,0],[-1.7066999999999979,-0.8000000000000007],[-1.2533999999999992,-1.3600000000000012],[-0.8533999999999935,-1.7333300000000005],[-0.42670000000001096,-1.9466700000000006],[0,0],[0,0]],"v":[[85.4338,-4.64],[89.6338,-4.64],[89.6338,-0.48],[90.1938,3.8],[91.8338,8.04],[94.6338,11.76],[98.6338,14.44],[95.6738,18.68],[91.1938,15.48],[87.9938,10.88],[86.0738,5.4],[85.4338,-0.48],[85.4338,-4.64]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[-0.37,-1.47],[-0.72,-1.36],[-1.12,-1.12],[-1.52,-0.69],[0,0],[1.28,1.33],[0.88,1.71],[0.43,1.92],[0,1.97],[0,0]],"o":[[0,0],[0,0],[0,1.39],[0.37,1.47],[0.75,1.36],[1.15,1.09],[0,0],[-1.71,-0.8],[-1.25,-1.36],[-0.85,-1.73],[-0.43,-1.95],[0,0],[0,0]],"v":[[85.43,-4.64],[89.63,-4.64],[89.63,-0.48],[90.19,3.8],[91.83,8.04],[94.63,11.76],[98.63,14.44],[95.67,18.68],[91.19,15.48],[87.99,10.88],[86.07,5.4],[85.43,-0.48],[85.43,-4.64]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.63,-6.96],[96.99,-6.96],[96.99,-2.56],[76.63,-2.56],[76.63,-6.96]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.6338,-6.96],[96.9938,-6.96],[96.9938,-2.56],[76.6338,-2.56],[76.6338,-6.96]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[76.63,-6.96],[96.99,-6.96],[96.99,-2.56],[76.63,-2.56],[76.63,-6.96]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.67,-10.48],[105.03,-10.48],[105.03,26.56],[99.67,26.56],[99.67,-10.48]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.6738,-10.48],[105.034,-10.48],[105.034,26.56],[99.6738,26.56],[99.6738,-10.48]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[99.67,-10.48],[105.03,-10.48],[105.03,26.56],[99.67,26.56],[99.67,-10.48]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.87,3.64],[110.39,3.64],[110.39,8],[103.87,8],[103.87,3.64]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.874,3.64],[110.394,3.64],[110.394,8],[103.874,8],[103.874,3.64]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[103.87,3.64],[110.39,3.64],[110.39,8],[103.87,8],[103.87,3.64]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0.48,-2.64],[1.23,-2.37],[2.21,-2.05],[3.49,-1.73],[0,0],[-1.89,1.63],[-1.09,1.89],[-0.45,2.21],[0,2.61],[0,0]],"o":[[0,0],[0,2.85],[-0.48,2.61],[-1.23,2.37],[-2.21,2.05],[0,0],[2.85,-1.41],[1.89,-1.63],[1.09,-1.89],[0.48,-2.21],[0,0],[0,0]],"v":[[55.15,41.04],[60.31,41.04],[59.59,49.28],[57.03,56.76],[51.87,63.4],[43.31,69.08],[40.51,64.96],[47.63,60.4],[52.11,55.12],[54.43,48.96],[55.15,41.72],[55.15,41.04]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0.480000000000004,-2.6400000000000006],[1.226700000000001,-2.3733000000000004],[2.213300000000004,-2.0533],[3.493400000000001,-1.7332999999999998],[0,0],[-1.8932999999999964,1.6266999999999996],[-1.0932999999999993,1.8933000000000035],[-0.4532999999999987,2.2132999999999967],[0,2.6133000000000024],[0,0]],"o":[[0,0],[0,2.8533000000000044],[-0.4799999999999969,2.6133000000000024],[-1.2265999999999977,2.3733000000000004],[-2.2134,2.0533],[0,0],[2.8534000000000006,-1.4132999999999925],[1.8933999999999997,-1.6266999999999996],[1.0934000000000026,-1.8932999999999964],[0.480000000000004,-2.213300000000004],[0,0],[0,0]],"v":[[55.1512,41.04],[60.3112,41.04],[59.5913,49.28],[57.0312,56.76],[51.8713,63.4],[43.3112,69.08],[40.5112,64.96],[47.6312,60.4],[52.1112,55.12],[54.4312,48.96],[55.1512,41.72],[55.1512,41.04]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0.48,-2.64],[1.23,-2.37],[2.21,-2.05],[3.49,-1.73],[0,0],[-1.89,1.63],[-1.09,1.89],[-0.45,2.21],[0,2.61],[0,0]],"o":[[0,0],[0,2.85],[-0.48,2.61],[-1.23,2.37],[-2.21,2.05],[0,0],[2.85,-1.41],[1.89,-1.63],[1.09,-1.89],[0.48,-2.21],[0,0],[0,0]],"v":[[55.15,41.04],[60.31,41.04],[59.59,49.28],[57.03,56.76],[51.87,63.4],[43.31,69.08],[40.51,64.96],[47.63,60.4],[52.11,55.12],[54.43,48.96],[55.15,41.72],[55.15,41.04]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.75,41.04],[57.79,41.04],[57.79,45.28],[42.75,45.28],[42.75,41.04]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.7512,41.04],[57.7912,41.04],[57.7912,45.28],[42.7512,45.28],[42.7512,41.04]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[42.75,41.04],[57.79,41.04],[57.79,45.28],[42.75,45.28],[42.75,41.04]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.11,50.12],[55.11,54.2],[41.51,55.44],[40.83,50.96],[55.11,50.12]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.1112,50.12],[55.1112,54.2],[41.5112,55.44],[40.8312,50.96],[55.1112,50.12]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.11,50.12],[55.11,54.2],[41.51,55.44],[40.83,50.96],[55.11,50.12]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.79,37.48],[71.11,37.48],[71.11,74.52],[65.79,74.52],[65.79,37.48]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.7913,37.48],[71.1113,37.48],[71.1113,74.52],[65.7913,74.52],[65.7913,37.48]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[65.79,37.48],[71.11,37.48],[71.11,74.52],[65.79,74.52],[65.79,37.48]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.47,58.2],[110.99,58.2],[110.99,62.48],[77.47,62.48],[77.47,58.2]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.4681,58.2],[110.988,58.2],[110.988,62.48],[77.4681,62.48],[77.4681,58.2]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[77.47,58.2],[110.99,58.2],[110.99,62.48],[77.47,62.48],[77.47,58.2]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.43,61],[96.75,61],[96.75,74.56],[91.43,74.56],[91.43,61]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.4281,61],[96.7481,61],[96.7481,74.56],[91.4281,74.56],[91.4281,61]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[91.43,61],[96.75,61],[96.75,74.56],[91.43,74.56],[91.43,61]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[-2,-0.69],[-1.12,-1.25],[0,-1.68],[1.12,-1.25],[2.03,-0.69],[2.64,0],[2.03,0.67],[1.15,1.23],[0,1.71],[-1.12,1.25],[-2,0.67],[-2.64,0]],"o":[[2.64,0],[2.03,0.67],[1.12,1.25],[0,1.71],[-1.12,1.23],[-2,0.67],[-2.64,0],[-2,-0.69],[-1.12,-1.25],[0,-1.68],[1.15,-1.25],[2.03,-0.69],[0,0]],"v":[[94.15,38.76],[101.11,39.8],[105.83,42.68],[107.51,47.08],[105.83,51.52],[101.11,54.4],[94.15,55.4],[87.15,54.4],[82.43,51.52],[80.75,47.08],[82.43,42.68],[87.15,39.8],[94.15,38.76]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[-1.999900000000011,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1199999999999903,-1.253300000000003],[2.027000000000001,-0.6933000000000007],[2.6400000000000006,0],[2.0267000000000053,0.6666999999999987],[1.1466999999999956,1.226699999999994],[0,1.706700000000005],[-1.1200000000000045,1.253300000000003],[-2,0.6667000000000058],[-2.6400000000000006,0]],"o":[[2.6400000000000006,0],[2.027000000000001,0.6667000000000058],[1.1199999999999903,1.253300000000003],[0,1.706700000000005],[-1.1200000000000045,1.226699999999994],[-1.999900000000011,0.6666999999999987],[-2.6400000000000006,0],[-2,-0.6933000000000007],[-1.1200000000000045,-1.253300000000003],[0,-1.6799999999999997],[1.1466999999999956,-1.253300000000003],[2.0267000000000053,-0.6933000000000007],[0,0]],"v":[[94.1481,38.76],[101.108,39.8],[105.828,42.68],[107.508,47.08],[105.828,51.52],[101.108,54.4],[94.1481,55.4],[87.1481,54.4],[82.4281,51.52],[80.7481,47.08],[82.4281,42.68],[87.1481,39.8],[94.1481,38.76]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[-2,-0.69],[-1.12,-1.25],[0,-1.68],[1.12,-1.25],[2.03,-0.69],[2.64,0],[2.03,0.67],[1.15,1.23],[0,1.71],[-1.12,1.25],[-2,0.67],[-2.64,0]],"o":[[2.64,0],[2.03,0.67],[1.12,1.25],[0,1.71],[-1.12,1.23],[-2,0.67],[-2.64,0],[-2,-0.69],[-1.12,-1.25],[0,-1.68],[1.15,-1.25],[2.03,-0.69],[0,0]],"v":[[94.15,38.76],[101.11,39.8],[105.83,42.68],[107.51,47.08],[105.83,51.52],[101.11,54.4],[94.15,55.4],[87.15,54.4],[82.43,51.52],[80.75,47.08],[82.43,42.68],[87.15,39.8],[94.15,38.76]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[1.2,-0.32],[0.64,-0.61],[0,-0.91],[-0.64,-0.61],[-1.17,-0.32],[-1.6,0],[-1.17,0.29],[-0.64,0.61],[0,0.88],[0.67,0.59],[1.17,0.32],[1.65,0]],"o":[[-1.6,0],[-1.17,0.32],[-0.64,0.59],[0,0.88],[0.64,0.61],[1.2,0.29],[1.65,0],[1.17,-0.32],[0.67,-0.61],[0,-0.91],[-0.64,-0.61],[-1.17,-0.32],[0,0]],"v":[[94.11,42.96],[89.91,43.44],[87.19,44.84],[86.23,47.08],[87.19,49.32],[89.91,50.72],[94.11,51.16],[98.35,50.72],[101.07,49.32],[102.07,47.08],[101.07,44.84],[98.35,43.44],[94.11,42.96]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[1.1999999999999886,-0.3200000000000003],[0.6400000000000006,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[-1.5999999999999943,0],[-1.1732999999999976,0.2933000000000021],[-0.6400000000000006,0.6133000000000024],[0,0.8800000000000026],[0.6670000000000016,0.5866999999999933],[1.1734000000000009,0.3200000000000003],[1.6534000000000049,0]],"o":[[-1.5999999999999943,0],[-1.1732999999999976,0.3200000000000003],[-0.6400000000000006,0.5866999999999933],[0,0.8800000000000026],[0.6400000000000006,0.6133000000000024],[1.1999999999999886,0.2933000000000021],[1.6534000000000049,0],[1.1734000000000009,-0.3200000000000003],[0.6670000000000016,-0.6133000000000024],[0,-0.9067000000000007],[-0.6400000000000006,-0.6133000000000024],[-1.1732999999999976,-0.3200000000000003],[0,0]],"v":[[94.1081,42.96],[89.9081,43.44],[87.1881,44.84],[86.2281,47.08],[87.1881,49.32],[89.9081,50.72],[94.1081,51.16],[98.3481,50.72],[101.068,49.32],[102.068,47.08],[101.068,44.84],[98.3481,43.44],[94.1081,42.96]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[1.2,-0.32],[0.64,-0.61],[0,-0.91],[-0.64,-0.61],[-1.17,-0.32],[-1.6,0],[-1.17,0.29],[-0.64,0.61],[0,0.88],[0.67,0.59],[1.17,0.32],[1.65,0]],"o":[[-1.6,0],[-1.17,0.32],[-0.64,0.59],[0,0.88],[0.64,0.61],[1.2,0.29],[1.65,0],[1.17,-0.32],[0.67,-0.61],[0,-0.91],[-0.64,-0.61],[-1.17,-0.32],[0,0]],"v":[[94.11,42.96],[89.91,43.44],[87.19,44.84],[86.23,47.08],[87.19,49.32],[89.91,50.72],[94.11,51.16],[98.35,50.72],[101.07,49.32],[102.07,47.08],[101.07,44.84],[98.35,43.44],[94.11,42.96]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.19,37.48],[145.51,37.48],[145.51,74.52],[140.19,74.52],[140.19,37.48]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.185,37.48],[145.505,37.48],[145.505,74.52],[140.185,74.52],[140.185,37.48]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[140.19,37.48],[145.51,37.48],[145.51,74.52],[140.19,74.52],[140.19,37.48]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0.56,-2.59],[1.28,-2.27],[2.21,-1.95],[3.28,-1.6],[0,0],[-2.11,2.08],[-0.93,2.64],[0,3.31],[0,0]],"o":[[0,0],[0,2.83],[-0.53,2.56],[-1.25,2.27],[-2.19,1.92],[0,0],[3.57,-1.73],[2.13,-2.11],[0.93,-2.67],[0,0],[0,0]],"v":[[129.26,41.32],[134.51,41.32],[133.67,49.44],[130.95,56.68],[125.75,63],[117.55,68.28],[114.75,64.08],[123.27,58.36],[127.87,51.24],[129.26,42.28],[129.26,41.32]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0.5600000000000023,-2.5867000000000004],[1.2800000000000011,-2.2667],[2.212999999999994,-1.9466999999999999],[3.280000000000001,-1.5999999999999943],[0,0],[-2.1069999999999993,2.0799999999999983],[-0.9329999999999927,2.6400000000000006],[0,3.3066999999999993],[0,0]],"o":[[0,0],[0,2.8267000000000024],[-0.532999999999987,2.5600000000000023],[-1.252999999999986,2.2667],[-2.1869999999999976,1.9200000000000017],[0,0],[3.5729999999999933,-1.7332999999999998],[2.1329999999999956,-2.1066999999999965],[0.9330000000000069,-2.6666999999999987],[0,0],[0,0]],"v":[[129.265,41.32],[134.505,41.32],[133.665,49.44],[130.945,56.68],[125.745,63],[117.545,68.28],[114.745,64.08],[123.265,58.36],[127.865,51.24],[129.265,42.28],[129.265,41.32]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0.56,-2.59],[1.28,-2.27],[2.21,-1.95],[3.28,-1.6],[0,0],[-2.11,2.08],[-0.93,2.64],[0,3.31],[0,0]],"o":[[0,0],[0,2.83],[-0.53,2.56],[-1.25,2.27],[-2.19,1.92],[0,0],[3.57,-1.73],[2.13,-2.11],[0.93,-2.67],[0,0],[0,0]],"v":[[129.26,41.32],[134.51,41.32],[133.67,49.44],[130.95,56.68],[125.75,63],[117.55,68.28],[114.75,64.08],[123.27,58.36],[127.87,51.24],[129.26,42.28],[129.26,41.32]]}],"t":107}]}},{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":1,"k":[{"h":1,"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.75,41.32],[131.75,41.32],[131.75,45.56],[116.75,45.56],[116.75,41.32]]}],"t":0},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":1},{"o":{"x":0,"y":0},"i":{"x":1.42,"y":1.42},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":49},{"o":{"x":1.42,"y":1.42},"i":{"x":0.82,"y":0.82},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":57.21328469685146},{"o":{"x":0.82,"y":0.82},"i":{"x":1.07,"y":1.07},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":65.42656939370292},{"o":{"x":1.07,"y":1.07},"i":{"x":0.97,"y":0.97},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":73.63985409055437},{"o":{"x":0.97,"y":0.97},"i":{"x":1.01,"y":1.01},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":81.85313878740584},{"o":{"x":1.01,"y":1.01},"i":{"x":0.99,"y":0.99},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":90.06642348425729},{"o":{"x":0.99,"y":0.99},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":98.27970818110875},{"o":{"x":1,"y":1},"i":{"x":1,"y":1},"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.745,41.32],[131.745,41.32],[131.745,45.56],[116.745,45.56],[116.745,41.32]]}],"t":106.4929928779602},{"s":[{"c":true,"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[116.75,41.32],[131.75,41.32],[131.75,45.56],[116.75,45.56],[116.75,41.32]]}],"t":107}]}},{"ty":"fl","bm":0,"hd":false,"nm":"","c":{"a":0,"k":[0,0,0]},"r":1,"o":{"a":0,"k":100}}],"ind":5},{"ty":4,"nm":"Frame 95 Bg","sr":1,"st":0,"op":107.55,"ip":0,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[150,150]},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"p":{"a":0,"k":[150,150]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"shapes":[{"ty":"sh","bm":0,"hd":false,"nm":"","d":1,"ks":{"a":0,"k":{"c":true,"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[300,0],[300,300],[0,300],[0,0]]}}},{"ty":"fl","bm":0,"hd":false,"nm":"","c":{"a":0,"k":[1,1,1]},"r":1,"o":{"a":0,"k":100}}],"ind":6}],"v":"5.7.0","fr":60,"op":106.55,"ip":0,"assets":[{"id":"0a224e8e34eadd0e46bc2059027a2254cd08e303","e":1,"w":1024,"h":1024,"p":"","u":""}]} \ No newline at end of file diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_close.png b/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_close.png new file mode 100644 index 00000000..c6bf8e0b Binary files /dev/null and b/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_close.png differ diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_idle.png b/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_idle.png new file mode 100644 index 00000000..d4839e06 Binary files /dev/null and b/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_idle.png differ diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_smile.png b/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_smile.png new file mode 100644 index 00000000..72b37fe5 Binary files /dev/null and b/Prototype/Prototype/Prototype/ApplyDesign/Resources/PNG/character_smile.png differ diff --git a/Prototype/Prototype/Prototype/ApplyDesign/Resources/sample_character_3D.usdz b/Prototype/Prototype/Prototype/ApplyDesign/Resources/sample_character_3D.usdz new file mode 100644 index 00000000..34223b78 Binary files /dev/null and b/Prototype/Prototype/Prototype/ApplyDesign/Resources/sample_character_3D.usdz differ diff --git a/Prototype/Prototype/Prototype/Assets.xcassets/AccentColor.colorset/Contents.json b/Prototype/Prototype/Prototype/Assets.xcassets/AccentColor.colorset/Contents.json new file mode 100644 index 00000000..eb878970 --- /dev/null +++ b/Prototype/Prototype/Prototype/Assets.xcassets/AccentColor.colorset/Contents.json @@ -0,0 +1,11 @@ +{ + "colors" : [ + { + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Prototype/Prototype/Prototype/Assets.xcassets/AppIcon.appiconset/Contents.json b/Prototype/Prototype/Prototype/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 00000000..23058801 --- /dev/null +++ b/Prototype/Prototype/Prototype/Assets.xcassets/AppIcon.appiconset/Contents.json @@ -0,0 +1,35 @@ +{ + "images" : [ + { + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "tinted" + } + ], + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Prototype/Prototype/Prototype/Assets.xcassets/Contents.json b/Prototype/Prototype/Prototype/Assets.xcassets/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/Prototype/Prototype/Prototype/Assets.xcassets/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/Contents.json b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/bokeh.imageset/Contents.json b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/bokeh.imageset/Contents.json new file mode 100644 index 00000000..8e9b4dc0 --- /dev/null +++ b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/bokeh.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "filename" : "bokeh.png", + "idiom" : "universal", + "scale" : "1x" + }, + { + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/bokeh.imageset/bokeh.png b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/bokeh.imageset/bokeh.png new file mode 100644 index 00000000..c6805516 Binary files /dev/null and b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/bokeh.imageset/bokeh.png differ diff --git a/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/spark.imageset/Contents.json b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/spark.imageset/Contents.json new file mode 100644 index 00000000..5fdc9eae --- /dev/null +++ b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/spark.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "filename" : "spark.png", + "idiom" : "universal", + "scale" : "1x" + }, + { + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/spark.imageset/spark.png b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/spark.imageset/spark.png new file mode 100644 index 00000000..3383a218 Binary files /dev/null and b/Prototype/Prototype/Prototype/Assets.xcassets/Particle Sprite Atlas.spriteatlas/spark.imageset/spark.png differ diff --git a/Prototype/Prototype/Prototype/ContentView.swift b/Prototype/Prototype/Prototype/ContentView.swift new file mode 100644 index 00000000..1a37bde4 --- /dev/null +++ b/Prototype/Prototype/Prototype/ContentView.swift @@ -0,0 +1,41 @@ +// +// ContentView.swift +// Prototype +// +// Created by SeoJunYoung on 12/16/25. +// + +import SwiftUI +import SpriteKit + +struct ContentView: View { + let gameSceme: SKScene = { + let scene = StackGameScene() + scene.scaleMode = .resizeFill + return scene + }() + + var body: some View { + NavigationStack { + List { + NavigationLink("물리 엔진 검증") { + SpriteView(scene: gameSceme) + .ignoresSafeArea() + } + NavigationLink("자이로 센서 검증") { + MotionView() + } + NavigationLink("탭 이벤트 검증") { + VarifyTapEventCalculationView() + } + NavigationLink("디자인 리소스 적용 검증") { + ApplyDesignResourceView() + } + } + } + } +} + +#Preview { + ContentView() +} diff --git a/Prototype/Prototype/Prototype/Motion/Motion.swift b/Prototype/Prototype/Prototype/Motion/Motion.swift new file mode 100644 index 00000000..7194b258 --- /dev/null +++ b/Prototype/Prototype/Prototype/Motion/Motion.swift @@ -0,0 +1,181 @@ +// +// Motion.swift +// Prototype +// +// Created by 김성훈 on 12/17/25. +// + +import SwiftUI + +struct MotionView: View { + private enum Texts { + static let descriptionTitle = "Description" + static let descriptionContent = "기기를 기울여 Core Motion 변화를 확인합니다. 실기기에서만 테스트할 수 있습니다." + + static let selectedSensorTitle = "사용할 센서 값" + static let calibratedGravityX = "보정된 gravityX 입력값" + static let unselectedSensorTitle = "측정 센서 값들" + + static let gravityX = "gravityX" + static let gravityY = "gravityY" + static let gravityZ = "gravityZ" + + static let roll = "좌우 기울기(Roll)" + static let pitch = "앞뒤 기울기(Pitch)" + static let yaw = "수평 회전(Yaw)" + + static let rotationRateRoll = "Roll 속도" + static let rotationRatePitch = "Pitch 속도" + static let rotationRateYaw = "Yaw 속도" + + static let accelerationX = "가속도(x)" + static let accelerationY = "가속도(y)" + static let accelerationZ = "가속도(z)" + + static let recalibrateButton = "센서 초기값 다시 설정하기" + static let character = "캐릭터" + } + + @State private var motionManager = MotionManager() + + var body: some View { + VStack(alignment: .leading, spacing: 10) { + + // 1. Description + VStack(alignment: .leading, spacing: 20) { + Text(Texts.descriptionTitle) + .font(.title2) + Text(Texts.descriptionContent) + } + + Divider() + + // 2. 선택에 적합한 센서 값 + VStack(alignment: .leading, spacing: 5) { + Text(Texts.selectedSensorTitle) + .font(.title2) + .bold() + + HStack { + Text(Texts.calibratedGravityX) + Spacer() + Text(String(format: "%.3f", motionManager.calibratedGravityX)) + .foregroundStyle(.blue) + .bold() + } + } + + Divider() + + // 3. 측정한 센서 값 + VStack(alignment: .leading, spacing: 5) { + Text(Texts.unselectedSensorTitle) + .font(.title2) + .bold() + + // Gravity + HStack { + Text(Texts.gravityX) + Spacer() + Text(String(format: "%.3f", motionManager.gravityX)) + .foregroundStyle(abs(motionManager.gravityX) > 0.7 ? .red : .primary) + } + HStack { + Text(Texts.gravityY) + Spacer() + Text(String(format: "%.3f", motionManager.gravityY)) + } + HStack { + Text(Texts.gravityZ) + Spacer() + Text(String(format: "%.3f", motionManager.gravityZ)) + } + + // 기울기 + HStack { + Text(Texts.roll) + Spacer() + Text(String(format: "%.3f", motionManager.roll)) + } + HStack { + Text(Texts.pitch) + Spacer() + Text(String(format: "%.3f", motionManager.pitch)) + } + HStack { + Text(Texts.yaw) + Spacer() + Text(String(format: "%.3f", motionManager.yaw)) + } + + // 회전 속도 + HStack { + Text(Texts.rotationRateRoll) + Spacer() + Text(String(format: "%.3f", motionManager.rotationY)) + } + HStack { + Text(Texts.rotationRatePitch) + Spacer() + Text(String(format: "%.3f", motionManager.rotationX)) + } + HStack { + Text(Texts.rotationRateYaw) + Spacer() + Text(String(format: "%.3f", motionManager.rotationZ)) + } + + // 가속도 + HStack { + Text(Texts.accelerationX) + Spacer() + Text(String(format: "%.3f", motionManager.userAccelX)) + } + HStack { + Text(Texts.accelerationY) + Spacer() + Text(String(format: "%.3f", motionManager.userAccelY)) + } + HStack { + Text(Texts.accelerationZ) + Spacer() + Text(String(format: "%.3f", motionManager.userAccelZ)) + } + } + + Divider() + + // 4. 초기화 버튼 + Button(Texts.recalibrateButton) { + motionManager.recalibrate() + } + .buttonStyle(.bordered) + .frame(maxWidth: .infinity) + .disabled(!motionManager.isCalibratable) + + if !motionManager.isCalibratable { + Text("기기가 너무 기울어져서 재설정할 수 없습니다.") + .font(.caption) + .foregroundStyle(.red) + } + + Spacer() + + // 5. 캐릭터 + VStack { + Rectangle() + .fill(.gray) + .frame(width: 40, height: 40) + Text(Texts.character) + .font(.caption) + } + .offset(x: motionManager.characterX, y: 0) + .frame(maxWidth: .infinity) + } + .padding() + } +} + +#Preview { + MotionView() +} diff --git a/Prototype/Prototype/Prototype/Motion/MotionManager.swift b/Prototype/Prototype/Prototype/Motion/MotionManager.swift new file mode 100644 index 00000000..147e5335 --- /dev/null +++ b/Prototype/Prototype/Prototype/Motion/MotionManager.swift @@ -0,0 +1,89 @@ +// +// MotionManager.swift +// Prototype +// +// Created by 최범수 on 2025-12-18. +// + +import CoreMotion + +@Observable +class MotionManager { + private let motionManager = CMMotionManager() + + // 설정값 + private let updateInterval = 1.0 / 60.0 // 주사율 16.67 ms + private let threshold: Double = 0.05 // 데드존 (무반응 구간) + private let movementSpeed: CGFloat = 1000.0 // 곱할 속도 (민감도) + private let calibrationLimit: Double = 0.7 // 보정 허용 범위 + private var baselineGravityX: Double = 0 // 사용자 기준점 (0점) + + // View에서 사용할 데이터들 + var gravityX: Double = 0 + var gravityY: Double = 0 + var gravityZ: Double = 0 + var roll: Double = 0 + var pitch: Double = 0 + var yaw: Double = 0 + var userAccelX: Double = 0 + var userAccelY: Double = 0 + var userAccelZ: Double = 0 + var rotationX: Double = 0 + var rotationY: Double = 0 + var rotationZ: Double = 0 + var characterX: CGFloat = 0 + var calibratedGravityX: Double = 0 + var isCalibratable: Bool = true + + init() { startMotionUpdates() } + deinit { motionManager.stopDeviceMotionUpdates() } + + func startMotionUpdates() { + guard motionManager.isDeviceMotionAvailable else { return } + motionManager.deviceMotionUpdateInterval = updateInterval + + motionManager.startDeviceMotionUpdates(to: .main) { [weak self] motion, error in + guard let self = self, let motion = motion else { return } + + self.gravityX = motion.gravity.x + self.gravityY = motion.gravity.y + self.gravityZ = motion.gravity.z + + // 라디안 -> 각도 변환 + self.roll = motion.attitude.roll * 180 / .pi + self.pitch = motion.attitude.pitch * 180 / .pi + self.yaw = motion.attitude.yaw * 180 / .pi + + self.userAccelX = motion.userAcceleration.x + self.userAccelY = motion.userAcceleration.y + self.userAccelZ = motion.userAcceleration.z + + self.rotationX = motion.rotationRate.x + self.rotationY = motion.rotationRate.y + self.rotationZ = motion.rotationRate.z + + self.isCalibratable = abs(motion.gravity.x) <= self.calibrationLimit + + // 보정 및 클램핑 (-1.0 ~ 1.0 으로 보정) + let rawInput = motion.gravity.x - self.baselineGravityX + let clampedInput = max(-1.0, min(1.0, rawInput)) + self.calibratedGravityX = clampedInput + + // 캐릭터 이동 + if abs(clampedInput) > self.threshold { + self.characterX += CGFloat(clampedInput) * self.movementSpeed * CGFloat(self.updateInterval) + + // 화면 밖 방지 + let screenLimit: CGFloat = 150 + self.characterX = max(-screenLimit, min(screenLimit, self.characterX)) + } + } + } + + func recalibrate() { + if abs(self.gravityX) <= self.calibrationLimit { + self.baselineGravityX = self.gravityX + self.characterX = 0 + } + } +} diff --git a/Prototype/Prototype/Prototype/PrototypeApp.swift b/Prototype/Prototype/Prototype/PrototypeApp.swift new file mode 100644 index 00000000..d6e69dfa --- /dev/null +++ b/Prototype/Prototype/Prototype/PrototypeApp.swift @@ -0,0 +1,17 @@ +// +// PrototypeApp.swift +// Prototype +// +// Created by SeoJunYoung on 12/16/25. +// + +import SwiftUI + +@main +struct PrototypeApp: App { + var body: some Scene { + WindowGroup { + ContentView() + } + } +} diff --git a/Prototype/Prototype/Prototype/StackGame/BlockNode.swift b/Prototype/Prototype/Prototype/StackGame/BlockNode.swift new file mode 100644 index 00000000..2f07217c --- /dev/null +++ b/Prototype/Prototype/Prototype/StackGame/BlockNode.swift @@ -0,0 +1,42 @@ +// +// BlockNode.swift +// Prototype +// +// Created by 최범수 on 2025-12-18. +// + +import SpriteKit + +final class BlockNode: SKSpriteNode { + private var action: SKAction? + + init(size: CGSize, color: UIColor) { + super.init(texture: nil, color: color, size: size) + self.name = "block" + } + + required init?(coder aDecoder: NSCoder) { + return nil + } + + func startMoving(distance: CGFloat) { + let duration = Double.random(in: 0.5...1.5) + let moveRight = SKAction.moveBy(x: distance, y: 0, duration: duration) + let moveLeft = SKAction.moveBy(x: -distance, y: 0, duration: duration) + let sequence = SKAction.sequence([moveRight, moveLeft]) + let repeatAction = SKAction.repeatForever(sequence) + self.action = repeatAction + self.run(repeatAction) + } + + func stopMoving() { + self.removeAllActions() + } + + func enableGravity() { + self.physicsBody = SKPhysicsBody(rectangleOf: self.size) + self.physicsBody?.isDynamic = true + self.physicsBody?.allowsRotation = true + self.physicsBody?.mass = 1 + } +} diff --git a/Prototype/Prototype/Prototype/StackGame/StackGameScene.swift b/Prototype/Prototype/Prototype/StackGame/StackGameScene.swift new file mode 100644 index 00000000..7d50b7c9 --- /dev/null +++ b/Prototype/Prototype/Prototype/StackGame/StackGameScene.swift @@ -0,0 +1,259 @@ +// +// StackGameScene.swift +// Prototype +// +// Created by 최범수 on 2025-12-17. +// + +import SpriteKit + +final class StackGameScene: SKScene { + private var isBlockProcessing: Bool = false + private var currentBlock: BlockNode? + private var previousBlock: BlockNode? + private var blocks: [BlockNode] = [] + + private let blockSize = CGSize(width: 80, height: 40) + private let blockColors: [UIColor] = [ + .systemRed, .systemBlue, .systemGreen, .systemYellow, + .systemPurple, .systemOrange, .systemPink, .systemTeal + ] + + private var score: Int = 0 + private var currentHeight: CGFloat = 40 + + private let scoreLabel: SKLabelNode = { + let label = SKLabelNode(fontNamed: "Chalkduster") + label.fontSize = 40 + label.fontColor = .black + label.text = 0.description + return label + }() + + override func didMove(to view: SKView) { + self.backgroundColor = .white + setupGravity() + setupGround() + setupCamera() + startGame() + } + + override func touchesBegan(_ touches: Set, with event: UIEvent?) { + guard currentBlock != nil, + !isBlockProcessing + else { return } + + dropBlock() + } + + private func setupGravity() { + self.physicsWorld.gravity = CGVector(dx: 0, dy: -9.8) + } + + private func setupGround() { + let size = CGSize(width: frame.size.width, height: 10) + let ground = SKSpriteNode(color: .gray, size: size) + ground.position = CGPoint(x: frame.midX, y: 0) + ground.physicsBody = SKPhysicsBody(rectangleOf: size) + ground.physicsBody?.isDynamic = false + self.addChild(ground) + } + + private func setupCamera() { + let cameraNode = SKCameraNode() + cameraNode.position = CGPoint(x: frame.midX, y: frame.midY) + cameraNode.addChild(scoreLabel) + addChild(cameraNode) + self.camera = cameraNode + } + + private func startGame() { + isBlockProcessing = false + blocks.removeAll() + score = 0 + currentHeight = 40 + + blocks.forEach { $0.removeFromParent() } + blocks.removeAll() + + camera?.position = CGPoint(x: frame.midX, y: frame.midY) + + putInitialBlock() + + spawnBlock() + } + + private func putInitialBlock() { + let firstBlock = BlockNode(size: blockSize, color: .gray) + firstBlock.physicsBody = SKPhysicsBody(rectangleOf: blockSize) + firstBlock.physicsBody?.isDynamic = false + firstBlock.position = CGPoint(x: frame.midX, y: currentHeight) + + previousBlock = firstBlock + currentHeight += firstBlock.size.height + addChild(firstBlock) + blocks.append(firstBlock) + } + + private func spawnBlock() { + isBlockProcessing = false + + let color = blockColors[score % blockColors.count] + let block = BlockNode(size: blockSize, color: color) + + let spawnY = (camera?.position.y ?? frame.midY) + frame.height / 2 - 100 + let leftEdge = blockSize.width / 2 + let rightEdge = frame.width - blockSize.width / 2 + + block.position = CGPoint(x: leftEdge, y: spawnY) + block.startMoving(distance: rightEdge - leftEdge) + + currentBlock = block + addChild(block) + } + + private func dropBlock() { + guard let block = currentBlock, + !isBlockProcessing + else { return } + + isBlockProcessing = true + + block.stopMoving() + block.enableGravity() + + evaluateBlock() + } + + private func evaluateBlock() { + guard let block = currentBlock, + let previous = previousBlock + else { return } + + let targetY = previous.position.y + previous.size.height + + if block.position.y <= targetY + blockSize.height { + checkAlignment() + } else { + DispatchQueue.main.asyncAfter(deadline: .now() + 0.01) { [weak self] in + self?.evaluateBlock() + } + } + } + + private func checkAlignment() { + guard let current = currentBlock, + let previous = previousBlock + else { return } + + let previousLeft = previous.position.x - previous.size.width / 2 + let previousRight = previous.position.x + previous.size.width / 2 + let previousRange = previousLeft...previousRight + + if previousRange.contains(current.position.x) { + placeBlockSuccess() + } else { + placeBlockFail() + } + } + private func placeBlockSuccess() { + guard let block = currentBlock else { return } + + block.physicsBody?.isDynamic = false + block.position = CGPoint(x: block.position.x, y: currentHeight) + + score += 1 + scoreLabel.text = score.description + + showSuccessParticle(at: block.position) + + playSuccessSound() + + blocks.append(block) + previousBlock = block + currentHeight += blockSize.height + + let moveCamera = SKAction.moveBy(x: 0, y: blockSize.height, duration: 0.3) + moveCamera.timingMode = .easeInEaseOut + camera?.run(moveCamera) + + DispatchQueue.main.asyncAfter(deadline: .now() + 0.3) { [weak self] in + self?.spawnBlock() + } + } + + private func placeBlockFail() { + guard let block = currentBlock, + let previous = previousBlock + else { return } + + score = max(0, score - 5) + scoreLabel.text = score.description + + block.physicsBody?.isDynamic = true + block.physicsBody?.allowsRotation = true + block.physicsBody?.restitution = 0.3 + block.physicsBody?.friction = 0.7 + + let targetY = previous.position.y + previous.size.height + block.position.y = targetY + 5 + + showFailureParticle(at: block.position) + + playFailureSound() + + DispatchQueue.main.asyncAfter(deadline: .now() + 1.5) { [weak self] in + block.removeFromParent() + self?.spawnBlock() + } + } + + private func showSuccessParticle(at position: CGPoint) { + let emitter = SKEmitterNode() + emitter.particleTexture = SKTexture(imageNamed: "spark") + emitter.particleBirthRate = 200 + emitter.numParticlesToEmit = 50 + emitter.particleLifetime = 1.0 + emitter.emissionAngle = 0 + emitter.emissionAngleRange = .pi * 2 + emitter.particleSpeed = 100 + emitter.particleSpeedRange = 50 + emitter.particleScale = 0.3 + emitter.particleScaleRange = 0.2 + emitter.particleAlpha = 1.0 + emitter.particleAlphaSpeed = -1.0 + emitter.particleColorBlendFactor = 1.0 + emitter.particleColor = .systemYellow + emitter.position = position + + addChild(emitter) + } + + private func showFailureParticle(at position: CGPoint) { + let emitter = SKEmitterNode() + emitter.particleTexture = SKTexture(imageNamed: "bokeh") + emitter.particleBirthRate = 100 + emitter.numParticlesToEmit = 30 + emitter.particleLifetime = 1.5 + emitter.emissionAngle = 0 + emitter.emissionAngleRange = .pi * 2 + emitter.particleSpeed = 50 + emitter.particleSpeedRange = 30 + emitter.particleScale = 0.5 + emitter.particleScaleRange = 0.3 + emitter.particleAlpha = 0.8 + emitter.particleAlphaSpeed = -0.5 + emitter.particleColor = .red + emitter.position = position + + addChild(emitter) + } + + private func playSuccessSound() { + run(SKAction.playSoundFileNamed("success.wav", waitForCompletion: false)) + } + + private func playFailureSound() { + run(SKAction.playSoundFileNamed("failure.wav", waitForCompletion: false)) + } +} diff --git a/Prototype/Prototype/Prototype/StackGame/failure.wav b/Prototype/Prototype/Prototype/StackGame/failure.wav new file mode 100644 index 00000000..22275ad3 Binary files /dev/null and b/Prototype/Prototype/Prototype/StackGame/failure.wav differ diff --git a/Prototype/Prototype/Prototype/StackGame/success.wav b/Prototype/Prototype/Prototype/StackGame/success.wav new file mode 100644 index 00000000..20f8bff2 Binary files /dev/null and b/Prototype/Prototype/Prototype/StackGame/success.wav differ diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Currency.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Currency.swift new file mode 100644 index 00000000..56ff9d3c --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Currency.swift @@ -0,0 +1,37 @@ +// +// Currency.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +/// 게임 내 화폐 시스템을 위한 공통 프로토콜 +protocol Currency: AnyObject { + /// 화폐 수량의 타입 + associatedtype Value: SignedNumeric & Comparable + + /// 현재 보유 중인 화폐 수량 + var amount: Value { get set } + + /// 지정된 수량만큼 화폐를 획득합니다. + /// - Parameter value: 획득할 화폐 수량 + func earn(_ value: Value) + + /// 지정된 수량만큼 화폐를 소비합니다. + /// - Parameter value: 소비할 화폐 수량 + /// - Returns: 소비 성공 시 `true`, 잔액 부족 시 `false` + @discardableResult + func spend(_ value: Value) -> Bool +} + +extension Currency { + func earn(_ value: Value) { + amount += value + } + + func spend(_ value: Value) -> Bool { + guard amount >= value else { return false } + amount -= value + return true + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Diamond.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Diamond.swift new file mode 100644 index 00000000..2d4e39e2 --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Diamond.swift @@ -0,0 +1,14 @@ +// +// Diamond.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +final class Diamond: Currency { + var amount: Int32 + + init(amount: Int32) { + self.amount = amount + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Money.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Money.swift new file mode 100644 index 00000000..ef326027 --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Money.swift @@ -0,0 +1,14 @@ +// +// Money.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +final class Money: Currency { + var amount: Double + + init(amount: Double) { + self.amount = amount + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Wallet.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Wallet.swift new file mode 100644 index 00000000..6091da5b --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Currency/Wallet.swift @@ -0,0 +1,16 @@ +// +// Wallet.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +final class Wallet { + private(set) var money: Money + private(set) var diamond: Diamond + + init(money: Money, diamond: Diamond) { + self.money = money + self.diamond = diamond + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Item/ConsumableItem.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Item/ConsumableItem.swift new file mode 100644 index 00000000..7aa0684b --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Item/ConsumableItem.swift @@ -0,0 +1,26 @@ +// +// ConsumableItem.swift +// Prototype +// +// Created by SeoJunYoung on 12/18/25. +// + +import Foundation + +/// 소비 가능한 아이템 +final class ConsumableItem { + /// 아이템 이름 + let name: String + + /// 효과 지속 시간 (초) + let duration: TimeInterval + + /// 보상 배율 + let multiplier: Double + + init(name: String, duration: TimeInterval, multiplier: Double) { + self.name = name + self.duration = duration + self.multiplier = multiplier + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Item/Inventory.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Item/Inventory.swift new file mode 100644 index 00000000..3c25063c --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Item/Inventory.swift @@ -0,0 +1,41 @@ +// +// Inventory.swift +// Prototype +// +// Created by SeoJunYoung on 12/18/25. +// + +/// 아이템 인벤토리 +final class Inventory { + /// 보유한 아이템과 개수 [아이템 이름: 개수] + private(set) var items: [String: Int] = [:] + + /// 특정 아이템을 추가합니다. + /// - Parameters: + /// - item: 추가할 아이템 + /// - count: 추가할 개수 (기본값: 1) + func add(item: ConsumableItem, count: Int = 1) { + let currentCount = items[item.name] ?? 0 + items[item.name] = currentCount + count + } + + /// 특정 아이템을 사용합니다. + /// - Parameter item: 사용할 아이템 + /// - Returns: 사용 성공 시 `true`, 아이템이 없는 경우 `false` + @discardableResult + func use(item: ConsumableItem) -> Bool { + guard let currentCount = items[item.name], currentCount > 0 else { + return false + } + + items[item.name] = currentCount - 1 + return true + } + + /// 특정 아이템의 보유 개수를 반환합니다. + /// - Parameter item: 확인할 아이템 + /// - Returns: 보유 개수 + func count(of item: ConsumableItem) -> Int { + items[item.name] ?? 0 + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Policy.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Policy.swift new file mode 100644 index 00000000..d12f9f92 --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Policy.swift @@ -0,0 +1,57 @@ +// +// Policy.swift +// Prototype +// +// Created by SeoJunYoung on 12/18/25. +// + +struct Policy { + + /// 스킬 강화별 가중치 [스킬이름 : 레벨 : 가중치] + static let skillUpgradeWeights: [String: [Int: Int]] = [ + "웹 개발 초급": [ + 1: 1, + 2: 2, + 3: 3, + 4: 4, + 5: 5, + 6: 6, + 7: 7, + 8: 8, + 9: 9, + 10: 10 + ], + "웹 개발 중급": [ + 1: 5, + 2: 10, + 3: 15, + 4: 20, + 5: 25, + 6: 30, + 7: 35, + 8: 40, + 9: 45, + 10: 50 + ], + "웹 개발 고급": [ + 1: 10, + 2: 20, + 3: 30, + 4: 40, + 5: 50, + 6: 60, + 7: 70, + 8: 80, + 9: 90, + 10: 100 + ] + ] + + /// 피버 단계별 배율 [단계 : 배율] + static let feverMultipliers: [Int: Double] = [ + 0: 1.0, // 기본 (가중치 없음) + 1: 1.2, // 20% 증가 + 2: 1.5, // 50% 증가 + 3: 2.0 // 100% 증가 (2배) + ] +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/RewardCalculator.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/RewardCalculator.swift new file mode 100644 index 00000000..a8017fd8 --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/RewardCalculator.swift @@ -0,0 +1,45 @@ +// +// RewardCalculator.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +struct RewardCalculator { + private let user: User + private let feverSystem: FeverSystem + private let buffSystem: BuffSystem + + init(user: User, feverSystem: FeverSystem, buffSystem: BuffSystem) { + self.user = user + self.feverSystem = feverSystem + self.buffSystem = buffSystem + } + + /// 탭 당 획득 가능한 재산을 계산합니다. + /// - Returns: 탭 당 재산 + func calculateMoneyPerTap() async -> Double { + let skillLevels = await user.skillSet.currentSkillLevels + let weights = Policy.skillUpgradeWeights + + var totalWeight = 0 + + for (skillName, level) in skillLevels { + guard + let skillWeights = weights[skillName], + let weight = skillWeights[level] + else { + continue + } + + totalWeight += weight + } + + // 기본 값에 스킬 가중치를 더하고 피버 배율, 버프 배율 적용 + let baseAmount = Double(totalWeight) + let feverMultiplier = feverSystem.getCurrentMultiplier() + let buffMultiplier = buffSystem.currentMultiplier + + return baseAmount * feverMultiplier * buffMultiplier + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Skill/Skill.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Skill/Skill.swift new file mode 100644 index 00000000..4511fd39 --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Skill/Skill.swift @@ -0,0 +1,20 @@ +// +// Skill.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +/// 게임 내 스킬 시스템을 위한 공통 구조체 +final class Skill { + /// 스킬 이름 + var title: String + + /// 최대 업그레이드 레벨 + var maxUpgradeLevel: Int + + init(title: String, maxUpgradeLevel: Int) { + self.title = title + self.maxUpgradeLevel = maxUpgradeLevel + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Skill/SkillSet.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Skill/SkillSet.swift new file mode 100644 index 00000000..7c634d1f --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/Skill/SkillSet.swift @@ -0,0 +1,65 @@ +// +// SkillSet.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +/// 사용자가 보유한 스킬들을 관리하는 구조체 +final class SkillSet { + /// 스킬 이름을 키로, 현재 레벨을 값으로 가지는 딕셔너리 + private(set) var currentSkillLevels: [String: Int] = [:] + + /// 특정 스킬의 레벨을 1 증가시킵니다. + /// - Parameter skill: 업그레이드할 스킬 + /// - Returns: 업그레이드 성공 시 `true`, 최대 레벨 도달 시 `false` + @discardableResult + func upgrade(skill: Skill) -> Bool { + let currentLevel = currentSkillLevels[skill.title] ?? 0 + + guard currentLevel < skill.maxUpgradeLevel else { + return false + } + + currentSkillLevels[skill.title] = currentLevel + 1 + return true + } + + /// 특정 스킬의 레벨을 1 감소시킵니다. + /// - Parameter skill: 다운그레이드할 스킬 + /// - Returns: 다운그레이드 성공 시 `true`, 레벨이 0이거나 스킬을 보유하지 않은 경우 `false` + @discardableResult + func downgrade(skill: Skill) -> Bool { + let currentLevel = currentSkillLevels[skill.title] ?? 0 + + guard currentLevel > 0 else { + return false + } + + currentSkillLevels[skill.title] = currentLevel - 1 + return true + } +} + +private extension SkillSet { + /// 특정 스킬의 현재 레벨을 반환합니다. + /// - Parameter skill: 레벨을 확인할 스킬 + /// - Returns: 현재 레벨 (보유하지 않은 스킬은 0) + func level(of skill: Skill) -> Int { + currentSkillLevels[skill.title] ?? 0 + } + + /// 특정 스킬을 보유하고 있는지 확인합니다. + /// - Parameter skill: 확인할 스킬 + /// - Returns: 보유 여부 + func hasSkill(_ skill: Skill) -> Bool { + currentSkillLevels[skill.title] != nil + } + + /// 특정 스킬이 최대 레벨에 도달했는지 확인합니다. + /// - Parameter skill: 확인할 스킬 + /// - Returns: 최대 레벨 도달 여부 + func isMaxLevel(skill: Skill) -> Bool { + level(of: skill) >= skill.maxUpgradeLevel + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/BuffSystem.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/BuffSystem.swift new file mode 100644 index 00000000..d9a11cee --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/BuffSystem.swift @@ -0,0 +1,64 @@ +// +// BuffSystem.swift +// Prototype +// +// Created by SeoJunYoung on 12/18/25. +// + +import Foundation + +/// 버프 효과를 관리하는 시스템 +final class BuffSystem { + /// 현재 활성화된 버프의 배율 + private(set) var currentMultiplier: Double = 1.0 + + /// 버프 타이머 + private var buffTimer: Timer? + + /// 버프가 활성화되어 있는지 확인 + var isActive: Bool { + buffTimer != nil + } + + /// 남은 버프 시간 (초) + private(set) var remainingTime: TimeInterval = 0 + + /// 버프를 활성화합니다. + /// - Parameter item: 사용할 소비 아이템 + /// - Returns: 활성화 성공 시 `true`, 이미 버프가 활성화된 경우 `false` + @discardableResult + func activate(item: ConsumableItem) -> Bool { + // 이미 버프가 활성화되어 있으면 실패 + guard !isActive else { + return false + } + + currentMultiplier = item.multiplier + remainingTime = item.duration + + // 1초마다 남은 시간 감소 + buffTimer = Timer.scheduledTimer(withTimeInterval: 1.0, repeats: true) { [weak self] _ in + guard let self = self else { return } + + self.remainingTime -= 1 + + if self.remainingTime <= 0 { + self.deactivate() + } + } + + return true + } + + /// 버프를 비활성화합니다. + func deactivate() { + buffTimer?.invalidate() + buffTimer = nil + currentMultiplier = 1.0 + remainingTime = 0 + } + + deinit { + deactivate() + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/FeverSystem.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/FeverSystem.swift new file mode 100644 index 00000000..c7d1724e --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/FeverSystem.swift @@ -0,0 +1,47 @@ +// +// FeverSystem.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +final class FeverSystem { + /// 현재 피버 단계 (0~3) + private(set) var currentLevel: Int = 0 + + /// 최대 피버 단계 + let maxLevel: Int = 3 + + /// 현재 피버 단계의 배율을 반환합니다. + /// - Returns: 피버 배율 + func getCurrentMultiplier() -> Double { + Policy.feverMultipliers[currentLevel] ?? 1.0 + } + + /// 피버 단계를 1 증가시킵니다. + /// - Returns: 증가 성공 시 `true`, 최대 단계 도달 시 `false` + @discardableResult + func levelUp() -> Bool { + guard currentLevel < maxLevel else { + return false + } + currentLevel += 1 + return true + } + + /// 피버 단계를 1 감소시킵니다. + /// - Returns: 감소 성공 시 `true`, 이미 0단계인 경우 `false` + @discardableResult + func levelDown() -> Bool { + guard currentLevel > 0 else { + return false + } + currentLevel -= 1 + return true + } + + /// 피버를 초기화합니다. + func reset() { + currentLevel = 0 + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/TapGameSystem.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/TapGameSystem.swift new file mode 100644 index 00000000..b059ac34 --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/System/TapGameSystem.swift @@ -0,0 +1,26 @@ +// +// TapGameSystem.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +final class TapGameSystem { + let user: User + let rewardCalculator: RewardCalculator + let feverSystem: FeverSystem + + init(user: User, rewardCalculator: RewardCalculator, feverSystem: FeverSystem) { + self.user = user + self.rewardCalculator = rewardCalculator + self.feverSystem = feverSystem + } + + /// 탭 이벤트를 수행하고 획득한 재산을 반환합니다. + /// - Returns: 탭 당 획득한 돈 + func tap() async -> Double { + let earnedMoney = await rewardCalculator.calculateMoneyPerTap() + await user.wallet.money.earn(earnedMoney) + return earnedMoney + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/User.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/User.swift new file mode 100644 index 00000000..ff4c482c --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Models/User.swift @@ -0,0 +1,27 @@ +// +// User.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +actor User { + /// 닉네임 + let nickname: String + + /// 재화 지갑 + private(set) var wallet: Wallet + + /// 유저의 스킬 목록 + private(set) var skillSet: SkillSet + + /// 아이템 인벤토리 + private(set) var inventory: Inventory + + init(nickname: String, wallet: Wallet, skillSet: SkillSet, inventory: Inventory) { + self.nickname = nickname + self.wallet = wallet + self.skillSet = skillSet + self.inventory = inventory + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Presentation/VarifyTapEventCalculationView.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Presentation/VarifyTapEventCalculationView.swift new file mode 100644 index 00000000..620c510b --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Presentation/VarifyTapEventCalculationView.swift @@ -0,0 +1,220 @@ +// +// VarifyTapEventCalculationView.swift +// Prototype +// +// Created by SeoJunYoung on 12/17/25. +// + +import SwiftUI + +fileprivate enum Constant { + enum Text { + static let description: String = """ + 클릭커 게임이 동작할 수 있는 구조를 설계 및 확인 합니다. + """ + static let descriptionSectionHeader: String = "Description" + static let currentMoneyHeader: String = "현재 재산" + static let currencyPerTapHeader: String = "탭 당 획득 재산" + static let navigationTitle: String = "탭 이벤트 연산 확인" + static let tapArea: String = "Tap Button" + } + + enum Fonts { + static let large: Font = .system(size: 30, weight: .bold) + static let title: Font = .system(size: 25, weight: .bold) + static let subTitle: Font = .system(size: 20, weight: .semibold) + static let body: Font = .system(size: 18, weight: .regular) + } + + enum Spacing { + static let section: CGFloat = 16 + } + + enum Colors { + static let primary: Color = .blue + } + + enum Size { + static let tapButtonHeight: CGFloat = 50 + } +} + +struct VarifyTapEventCalculationView: View { + @State var viewModel = VarifyTapEventCalculationViewModel() + + var body: some View { + ScrollView { + VStack(alignment: .leading, spacing: Constant.Spacing.section) { + descriptionSection + Divider() + dashBoardSection + Divider() + feverSection + Divider() + buffSection + Divider() + skillSection + Divider() + tapButton + } + .navigationTitle(Constant.Text.navigationTitle) + .navigationBarTitleDisplayMode(.inline) + .padding() + .task { + await viewModel.loadInitialData() + } + } + } + + @ViewBuilder + var descriptionSection: some View { + Text(Constant.Text.descriptionSectionHeader) + .font(Constant.Fonts.body) + Text(Constant.Text.description) + .font(Constant.Fonts.body) + } + + @ViewBuilder + var feverSection: some View { + VStack(alignment: .leading, spacing: 8) { + Text("피버 시스템") + .font(Constant.Fonts.subTitle) + + HStack { + Text("현재 단계: \(viewModel.feverLevel)") + .font(Constant.Fonts.body) + Spacer() + Text("배율: x\(String(format: "%.1f", viewModel.feverMultiplier))") + .font(Constant.Fonts.body) + .foregroundStyle(Constant.Colors.primary) + } + + HStack(spacing: 8) { + Button("단계 올리기") { + viewModel.increaseFeverLevel() + } + .buttonStyle(.bordered) + + Button("단계 내리기") { + viewModel.decreaseFeverLevel() + } + .buttonStyle(.bordered) + + Button("초기화") { + viewModel.resetFever() + } + .buttonStyle(.bordered) + } + } + } + + @ViewBuilder + var buffSection: some View { + VStack(alignment: .leading, spacing: 8) { + Text("버프 시스템 (소비 아이템)") + .font(Constant.Fonts.subTitle) + + HStack { + Text("버프 배율: x\(String(format: "%.1f", viewModel.buffMultiplier))") + .font(Constant.Fonts.body) + Spacer() + if viewModel.buffRemainingTime > 0 { + Text("남은 시간: \(Int(viewModel.buffRemainingTime))초") + .font(Constant.Fonts.body) + .foregroundStyle(.orange) + } + } + + HStack { + Text(viewModel.doubleRewardItem.name) + .font(Constant.Fonts.body) + Spacer() + Text("보유: \(viewModel.itemCounts[viewModel.doubleRewardItem.name] ?? 0)개") + .font(Constant.Fonts.body) + } + + HStack(spacing: 8) { + Button("아이템 추가") { + viewModel.addItem() + } + .buttonStyle(.bordered) + + Button("아이템 사용") { + viewModel.useItem() + } + .buttonStyle(.borderedProminent) + .disabled(viewModel.buffRemainingTime > 0) + } + } + } + + @ViewBuilder + var skillSection: some View { + VStack(alignment: .leading, spacing: 8) { + Text("스킬 업그레이드") + .font(Constant.Fonts.subTitle) + + ForEach(viewModel.availableSkills, id: \.title) { skill in + HStack { + Text(skill.title) + Spacer() + Text("Lv. \(viewModel.skillLevels[skill.title] ?? 0)") + Button("+") { + viewModel.upgradeSkill(skill) + } + .buttonStyle(.bordered) + Button("-") { + viewModel.downgradeSkill(skill) + } + .buttonStyle(.bordered) + } + } + } + } + + @ViewBuilder + var dashBoardSection: some View { + VStack(alignment: .leading, spacing: 0) { + Text(Constant.Text.currentMoneyHeader) + .font(Constant.Fonts.subTitle) + + Text(String(format: "%.0f", viewModel.currentMoney)) + .font(Constant.Fonts.large) + .foregroundStyle(Constant.Colors.primary) + } + + VStack(alignment: .leading, spacing: 0) { + Text(Constant.Text.currencyPerTapHeader) + .font(Constant.Fonts.subTitle) + + Text(String(format: "%.0f", viewModel.moneyPerTap)) + .font(Constant.Fonts.large) + .foregroundStyle(Constant.Colors.primary) + } + } + + @ViewBuilder + var tapButton: some View { + Button { + viewModel.tap() + } label: { + Text(Constant.Text.tapArea) + .font(Constant.Fonts.subTitle) + .foregroundStyle(Color.white) + .frame( + maxWidth: .infinity, + maxHeight: Constant.Size.tapButtonHeight, + alignment: .center + ) + .background(Constant.Colors.primary) + .cornerRadius(10) + } + .frame(height: Constant.Size.tapButtonHeight) + } +} + +#Preview { + NavigationStack { + VarifyTapEventCalculationView() + } +} diff --git a/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Presentation/VarifyTapEventCalculationViewModel.swift b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Presentation/VarifyTapEventCalculationViewModel.swift new file mode 100644 index 00000000..ce406610 --- /dev/null +++ b/Prototype/Prototype/Prototype/VarifyTapEventCaluation/Presentation/VarifyTapEventCalculationViewModel.swift @@ -0,0 +1,171 @@ +// +// VarifyTapEventCalculationViewModel.swift +// Prototype +// +// Created by SeoJunYoung on 12/18/25. +// + +import Observation +import Foundation + +@Observable +final class VarifyTapEventCalculationViewModel { + private let user: User + private let tapSystem: TapGameSystem + private let feverSystem: FeverSystem + private let buffSystem: BuffSystem + + // UI 상태 + var currentMoney: Double = 0 + var moneyPerTap: Double = 0 + var skillLevels: [String: Int] = [:] + var feverLevel: Int = 0 + var feverMultiplier: Double = 1.0 + var buffMultiplier: Double = 1.0 + var buffRemainingTime: TimeInterval = 0 + var itemCounts: [String: Int] = [:] + + // 사용 가능한 스킬 목록 + let availableSkills: [Skill] = [ + Skill(title: "웹 개발 초급", maxUpgradeLevel: 10), + Skill(title: "웹 개발 중급", maxUpgradeLevel: 10), + Skill(title: "웹 개발 고급", maxUpgradeLevel: 10) + ] + + // 사용 가능한 아이템 + let doubleRewardItem = ConsumableItem(name: "2배 보상 물약", duration: 30, multiplier: 2.0) + + init() { + let user = User( + nickname: "ProtoType", + wallet: .init(money: .init(amount: 0), diamond: .init(amount: 0)), + skillSet: .init(), + inventory: .init() + ) + let feverSystem = FeverSystem() + let buffSystem = BuffSystem() + + self.user = user + self.feverSystem = feverSystem + self.buffSystem = buffSystem + self.tapSystem = TapGameSystem( + user: user, + rewardCalculator: .init(user: user, feverSystem: feverSystem, buffSystem: buffSystem), + feverSystem: feverSystem + ) + } + + /// 초기 데이터 로드 + func loadInitialData() async { + await updateUIState() + } + + /// 탭 이벤트 + func tap() { + Task { + let earned = await tapSystem.tap() + await updateUIState() + print("획득한 재산: \(earned)") + } + } + + /// 스킬 업그레이드 + func upgradeSkill(_ skill: Skill) { + Task { + let success = await user.skillSet.upgrade(skill: skill) + if success { + await updateUIState() + } + } + } + + /// 스킬 다운그레이드 + func downgradeSkill(_ skill: Skill) { + Task { + let success = await user.skillSet.downgrade(skill: skill) + if success { + await updateUIState() + } + } + } + + /// 피버 레벨 증가 + func increaseFeverLevel() { + feverSystem.levelUp() + Task { + await updateUIState() + } + } + + /// 피버 레벨 감소 + func decreaseFeverLevel() { + feverSystem.levelDown() + Task { + await updateUIState() + } + } + + /// 피버 초기화 + func resetFever() { + feverSystem.reset() + Task { + await updateUIState() + } + } + + /// 아이템 추가 (검증용) + func addItem() { + Task { + await user.inventory.add(item: doubleRewardItem, count: 1) + await updateUIState() + } + } + + /// 아이템 사용 + func useItem() { + Task { + // 이미 버프가 활성화되어 있으면 사용 불가 + guard !buffSystem.isActive else { + print("이미 버프가 활성화되어 있습니다.") + return + } + + // 인벤토리에서 아이템 사용 + let hasItem = await user.inventory.use(item: doubleRewardItem) + guard hasItem else { + print("아이템이 부족합니다.") + return + } + + // 버프 활성화 + buffSystem.activate(item: doubleRewardItem) + await updateUIState() + + // 1초마다 UI 업데이트 (남은 시간 표시) + startBuffTimer() + } + } + + /// 버프 타이머 시작 + private func startBuffTimer() { + Task { + while buffSystem.isActive { + try? await Task.sleep(nanoseconds: 1_000_000_000) // 1초 + await updateUIState() + } + } + } + + /// UI 상태 업데이트 + @MainActor + private func updateUIState() async { + currentMoney = await user.wallet.money.amount + moneyPerTap = await tapSystem.rewardCalculator.calculateMoneyPerTap() + skillLevels = await user.skillSet.currentSkillLevels + feverLevel = feverSystem.currentLevel + feverMultiplier = feverSystem.getCurrentMultiplier() + buffMultiplier = buffSystem.currentMultiplier + buffRemainingTime = buffSystem.remainingTime + itemCounts[doubleRewardItem.name] = await user.inventory.count(of: doubleRewardItem) + } +} diff --git a/README.md b/README.md index c331ae98..461c3850 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,126 @@ -# iOS01-Hmm +# iOS01-Hmm (흠) 흠이 없어....그래서 흠이야 + +## 팀원 소개 +| 이름 | Sophia 김선재 | Oliver 김성훈 | Raven 서준영 | Edward 최범수 | +|:----:|:------:|:------:|:------:|:------:| +| 캠퍼 ID | S004 | S005 | S016 | S037 | +| 사진 | image |image| image | image| +| 역할 | 팀원 | 🤴🏻 팀장 | 팀원 | 팀원 | + +# 🎮 개발자 키우기 +image + +# 📝 개요 + +> `개발자 키우기`는 백수에서 시작해 월드클래스 iOS 개발자로 성장하는 시뮬레이션 게임입니다. +> + +본 프로젝트는 단순한 탭 반복 위주의 방치형 게임에서 벗어나, +iOS 기기 고유의 입력 방식과 물리 시스템을 적극 활용한 게임 경험을 목표로 기획되었습니다. +사용자는 화면 터치뿐만 아니라 기기의 기울기를 직접 조작하며 다양한 미니게임을 플레이하게 됩니다. + +게임의 핵심 루프는 **`미니게임 → 재산 획득 → 스킬/아이템/부동산 강화 → 커리어 성장 및 미션 보상`** 으로 구성되어 있으며, +플레이가 누적될수록 더 효율적인 성장 전략을 설계할 수 있도록 설계되었습니다. +이를 통해 단순한 반복이 아닌, 선택과 전략이 개입되는 성장 구조를 제공합니다. + +기술적으로는 `CoreMotion`을 활용한 기기 기울기 입력 처리와 +`SpriteKit` 물리 엔진을 활용한 오브젝트 상호작용을 통해 +iOS 기기 특유의 ‘손맛’과 직관적인 조작감을 구현했습니다. + +또한 `SwiftUI` 기반 UI 구성과 `Actor`를 활용한 동시성 제어를 통해 +게임 로직과 상태 관리를 명확히 분리하고, 안정적이고 부드러운 플레이 경험을 제공하는 것을 목표로 했습니다. + +# 🔨 실행 방법 / 최소 지원 버전 +``` +타깃을 SoloDeveloperTraining으로 설정 후 빌드합니다. +``` +``` +Minimum Deployment Target: iOS 17.0 +``` + +# ⚔️ 사용 기술 스택 + +| 구분 | 스택 | +|---|---| +| **Language** | Swift 6.0 | +| **UI** | SwiftUI, Observable | +| **Framework** | CoreMotion, SpriteKit | +| **Async** | Swift Concurrency | +| **Tools** | SwiftLint | +| **CI/CD** | Github Actions | + +# 🦿 프로젝트 구조 +``` +SoloDeveloperTraining/ + ├── Prototype/ # 프로토타입 프로젝트 + └── SoloDeveloperTraining/ # 메인 프로젝트 + ├── SoloDeveloperTraining.xcodeproj + └── SoloDeveloperTraining/ + │ + ├── App/ # 앱 진입점 (AppDelegate, SceneDelegate 등) + │ + ├── DesignSystem/ # 디자인 시스템 + │ + ├── Extensions/ # Swift 확장 기능 + │ + ├── GameCore/ # 게임 핵심 로직 + │ └── Models/ + │ ├── Games/ # 미니게임 + │ ├── Items/ # 아이템 모델 + │ ├── Storages/ # 저장소 모델 + │ ├── Systems/ # 게임 시스템 + │ └── User/ # 유저 관련 + │ + ├── Production/ # 프로덕션 코드 + │ ├── Data/ # 데이터 레이어 + │ ├── Error/ # 에러 처리 + │ ├── FeedbackSystem/ # 피드백 시스템 (햅틱, 사운드 등) + │ ├── Presentation/ # UI 레이어 + │ └── Utility/ # 유틸리티 함수들 + │ + ├── Development/ # 개발용 코드 + │ └── Presentation/ # 개발용 UI + │ + └── Resources/ # 리소스 파일 + ├── Assets.xcassets/ # 이미지, 컬러 에셋 + ├── Audio/ # 오디오 파일 + └── Fonts/ # 폰트 파일 +``` +# 🚀 주요 기능 + +## 1. 커리어 성장 시스템 +- **9단계 커리어 등급** + - **`백수 → 노트북 보유자 → 개발자 지망생 → ... → 월드클래스 개발자`** +- **누적 재산 기반 자동 승급**: 플레이하며 자연스럽게 성장하는 시스템입니다. +- **등급별 콘텐츠 해금**: 새로운 미니게임 모드와 강화 콘텐츠를 해금할 수 있습니다. + +## 2. 4가지 미니게임 +| 코드짜기 | 언어 맞추기 | 버그 피하기 | 데이터 쌓기 | +| --- | --- | --- | --- | +|image|image|image|image| +| 반복적인 화면 터치(탭)를 통해 재산을 획득할 수 있습니다. | 올바른 언어 아이콘을 매칭 터치하여 재산을 획득할 수 있습니다. | CoreMotion 자이로 센서를 활용하여 기기 기울여 재산을 획득할 수 있습니다. | SpriteKit 물리 엔진을 기반으로 타이밍에 맞춰 터치하여 재산을 획득할 수 있습니다. | + +## 3. 피버 시스템 +- **3단계 피버 게이지**: 0~300%까지 노란색 → 주황색 → 빨간색으로 시각화 했습니다. +- **단계별 배율**: 100% 도달 시 x배, 200% 도달 시 y배, 300% 도달 시 z배 획득 할 수 있습니다. +- **게이지 변화**: 액션 성공 시 증가하고, n초마다 자동 감소합니다. + +| 0단계 | 1단계 | 2단계 | 3단계 | +| --- | --- | --- | --- | +|image|image|image|image| + + +## 4. 경제 시스템 + +| 스킬| 아이템 | 부동산 | +| --- | --- | --- | +|image|image|image| +| - 업무 4개 모드마다 초급/중급/고급의 스킬이 존재합니다.
- 레벨이 올라갈수록 각 업무의 액션 재산이 증가합니다. | - 커피, 박하스로 일시적 버프 효과를 획득합니다.
- 키보드, 마우스, 모니터, 의자 각각의 8등급의 강화 시스템이 존재합니다.
- 등급이 높아질수록 강화 성공 확률이 감소합니다. | - 길바닥 → 반지하 → … → 펜트하우스의 등급이 존재합니다.
- 배경을 변경할 수 있고, 부동산은 하나만 소유 가능합니다.| + + +## 5. 부가 콘텐츠 +| 퀴즈 | 미션 | 튜토리얼 | 설정 | +| --- | --- | --- | --- | +| image| image| image| image| +| 개발 밈과 관련된 퀴즈로 다이아 보상을 획득할 수 있습니다. | 다양한 목표 달성으로 지속적인 플레이를 보장하며 보상을 획득할 수 있습니다. | 게임 시스템의 기본적인 학습을 할 수 있습니다. | 사운드, 효과음, 햅틱에 대한 설정을 조절할 수 있습니다. | diff --git a/SoloDeveloperTraining/.swiftlint.yml b/SoloDeveloperTraining/.swiftlint.yml new file mode 100644 index 00000000..84209c78 --- /dev/null +++ b/SoloDeveloperTraining/.swiftlint.yml @@ -0,0 +1,8 @@ +disabled_rules: + - cyclomatic_complexity + +line_length: + warning: 150 + error: 200 + ignores_comments: true + ignores_urls: true diff --git a/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/project.pbxproj b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/project.pbxproj new file mode 100644 index 00000000..01aa4309 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/project.pbxproj @@ -0,0 +1,724 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 77; + objects = { + +/* Begin PBXContainerItemProxy section */ + BCFC853D2F31FED800447A9A /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 08267F232F0D06BC005A0066 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 08267F2A2F0D06BC005A0066; + remoteInfo = SoloDeveloperTraining; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXFileReference section */ + 08267F2B2F0D06BC005A0066 /* SoloDeveloperTraining.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = SoloDeveloperTraining.app; sourceTree = BUILT_PRODUCTS_DIR; }; + 2896AE422F100CD600D38732 /* SoloDeveloperTraining-Dev.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = "SoloDeveloperTraining-Dev.app"; sourceTree = BUILT_PRODUCTS_DIR; }; + BCFC85392F31FED800447A9A /* SoloDeveloperTrainingTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = SoloDeveloperTrainingTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; +/* End PBXFileReference section */ + +/* Begin PBXFileSystemSynchronizedBuildFileExceptionSet section */ + 2896AE432F100CD600D38732 /* Exceptions for "SoloDeveloperTraining" folder in "SoloDeveloperTraining-Dev" target */ = { + isa = PBXFileSystemSynchronizedBuildFileExceptionSet; + membershipExceptions = ( + "App/Info-Dev.plist", + App/Info.plist, + "Production/Error/PurchasingError+UserReadableError.swift", + "Production/Error/SkillError+UserReadableError.swift", + Production/Error/UserReadableError.swift, + Production/Presentation/CharacterScene.swift, + Production/Presentation/DodgeGameView.swift, + Production/Presentation/FeedbackSettingView.swift, + Production/Presentation/Intro/IntroView.swift, + Production/Presentation/Intro/NicknameSetupView.swift, + Production/Presentation/Intro/TutorialPageView.swift, + Production/Presentation/Intro/TutorialView.swift, + Production/Presentation/LanguageGameView.swift, + Production/Presentation/MainView.swift, + Production/Presentation/MissionView.swift, + Production/Presentation/MultiTouchView.swift, + Production/Presentation/QuizGameView.swift, + Production/Presentation/ShopView.swift, + Production/Presentation/SkillView.swift, + Production/Presentation/StackGameScene.swift, + Production/Presentation/StackGameView.swift, + Production/Presentation/TapGameView.swift, + Production/Presentation/WorkSelectedView.swift, + ); + target = 2896AE3A2F100CD600D38732 /* SoloDeveloperTraining-Dev */; + }; + BC6322A72F0F41BB00EAD3F7 /* Exceptions for "SoloDeveloperTraining" folder in "SoloDeveloperTraining" target */ = { + isa = PBXFileSystemSynchronizedBuildFileExceptionSet; + membershipExceptions = ( + App/Info.plist, + Devlopment/CheatSystem/CheatManager.swift, + Devlopment/Presentation/ContentView.swift, + Devlopment/Presentation/DodgeGameTestView.swift, + Devlopment/Presentation/LanguageGameTestView.swift, + Devlopment/Presentation/MissionTestView.swift, + Devlopment/Presentation/QuizGameTestContentView.swift, + Devlopment/Presentation/QuizGameTestView.swift, + Devlopment/Presentation/ShopTestView.swift, + Devlopment/Presentation/SkillTestView.swift, + Devlopment/Presentation/StackGameScene.swift, + Devlopment/Presentation/StackGameTestView.swift, + Devlopment/Presentation/TabGameView.swift, + ); + target = 08267F2A2F0D06BC005A0066 /* SoloDeveloperTraining */; + }; +/* End PBXFileSystemSynchronizedBuildFileExceptionSet section */ + +/* Begin PBXFileSystemSynchronizedRootGroup section */ + 08267F2D2F0D06BC005A0066 /* SoloDeveloperTraining */ = { + isa = PBXFileSystemSynchronizedRootGroup; + exceptions = ( + BC6322A72F0F41BB00EAD3F7 /* Exceptions for "SoloDeveloperTraining" folder in "SoloDeveloperTraining" target */, + 2896AE432F100CD600D38732 /* Exceptions for "SoloDeveloperTraining" folder in "SoloDeveloperTraining-Dev" target */, + ); + path = SoloDeveloperTraining; + sourceTree = ""; + }; + BCFC853A2F31FED800447A9A /* SoloDeveloperTrainingTests */ = { + isa = PBXFileSystemSynchronizedRootGroup; + path = SoloDeveloperTrainingTests; + sourceTree = ""; + }; +/* End PBXFileSystemSynchronizedRootGroup section */ + +/* Begin PBXFrameworksBuildPhase section */ + 08267F282F0D06BC005A0066 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 2896AE3C2F100CD600D38732 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + BCFC85362F31FED800447A9A /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 08267F222F0D06BC005A0066 = { + isa = PBXGroup; + children = ( + 08267F2D2F0D06BC005A0066 /* SoloDeveloperTraining */, + BCFC853A2F31FED800447A9A /* SoloDeveloperTrainingTests */, + 08267F2C2F0D06BC005A0066 /* Products */, + ); + sourceTree = ""; + }; + 08267F2C2F0D06BC005A0066 /* Products */ = { + isa = PBXGroup; + children = ( + 08267F2B2F0D06BC005A0066 /* SoloDeveloperTraining.app */, + 2896AE422F100CD600D38732 /* SoloDeveloperTraining-Dev.app */, + BCFC85392F31FED800447A9A /* SoloDeveloperTrainingTests.xctest */, + ); + name = Products; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXNativeTarget section */ + 08267F2A2F0D06BC005A0066 /* SoloDeveloperTraining */ = { + isa = PBXNativeTarget; + buildConfigurationList = 08267F362F0D06BE005A0066 /* Build configuration list for PBXNativeTarget "SoloDeveloperTraining" */; + buildPhases = ( + 08267F272F0D06BC005A0066 /* Sources */, + 08267F282F0D06BC005A0066 /* Frameworks */, + 08267F292F0D06BC005A0066 /* Resources */, + B3D95AE82F0FA9850066BDF2 /* ShellScript */, + ); + buildRules = ( + ); + dependencies = ( + ); + fileSystemSynchronizedGroups = ( + 08267F2D2F0D06BC005A0066 /* SoloDeveloperTraining */, + ); + name = SoloDeveloperTraining; + packageProductDependencies = ( + ); + productName = SoloDeveloperTraining; + productReference = 08267F2B2F0D06BC005A0066 /* SoloDeveloperTraining.app */; + productType = "com.apple.product-type.application"; + }; + 2896AE3A2F100CD600D38732 /* SoloDeveloperTraining-Dev */ = { + isa = PBXNativeTarget; + buildConfigurationList = 2896AE3F2F100CD600D38732 /* Build configuration list for PBXNativeTarget "SoloDeveloperTraining-Dev" */; + buildPhases = ( + 2896AE3B2F100CD600D38732 /* Sources */, + 2896AE3C2F100CD600D38732 /* Frameworks */, + 2896AE3D2F100CD600D38732 /* Resources */, + 2896AE3E2F100CD600D38732 /* ShellScript */, + ); + buildRules = ( + ); + dependencies = ( + ); + fileSystemSynchronizedGroups = ( + 08267F2D2F0D06BC005A0066 /* SoloDeveloperTraining */, + ); + name = "SoloDeveloperTraining-Dev"; + packageProductDependencies = ( + ); + productName = SoloDeveloperTraining; + productReference = 2896AE422F100CD600D38732 /* SoloDeveloperTraining-Dev.app */; + productType = "com.apple.product-type.application"; + }; + BCFC85382F31FED800447A9A /* SoloDeveloperTrainingTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = BCFC853F2F31FED800447A9A /* Build configuration list for PBXNativeTarget "SoloDeveloperTrainingTests" */; + buildPhases = ( + BCFC85352F31FED800447A9A /* Sources */, + BCFC85362F31FED800447A9A /* Frameworks */, + BCFC85372F31FED800447A9A /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + BCFC853E2F31FED800447A9A /* PBXTargetDependency */, + ); + fileSystemSynchronizedGroups = ( + BCFC853A2F31FED800447A9A /* SoloDeveloperTrainingTests */, + ); + name = SoloDeveloperTrainingTests; + packageProductDependencies = ( + ); + productName = SoloDeveloperTrainingTests; + productReference = BCFC85392F31FED800447A9A /* SoloDeveloperTrainingTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 08267F232F0D06BC005A0066 /* Project object */ = { + isa = PBXProject; + attributes = { + BuildIndependentTargetsInParallel = 1; + LastSwiftUpdateCheck = 2620; + LastUpgradeCheck = 2610; + TargetAttributes = { + 08267F2A2F0D06BC005A0066 = { + CreatedOnToolsVersion = 26.1.1; + }; + BCFC85382F31FED800447A9A = { + CreatedOnToolsVersion = 26.2; + TestTargetID = 08267F2A2F0D06BC005A0066; + }; + }; + }; + buildConfigurationList = 08267F262F0D06BC005A0066 /* Build configuration list for PBXProject "SoloDeveloperTraining" */; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + en, + Base, + ); + mainGroup = 08267F222F0D06BC005A0066; + minimizedProjectReferenceProxies = 1; + preferredProjectObjectVersion = 77; + productRefGroup = 08267F2C2F0D06BC005A0066 /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 08267F2A2F0D06BC005A0066 /* SoloDeveloperTraining */, + 2896AE3A2F100CD600D38732 /* SoloDeveloperTraining-Dev */, + BCFC85382F31FED800447A9A /* SoloDeveloperTrainingTests */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 08267F292F0D06BC005A0066 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 2896AE3D2F100CD600D38732 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + BCFC85372F31FED800447A9A /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXShellScriptBuildPhase section */ + 2896AE3E2F100CD600D38732 /* ShellScript */ = { + isa = PBXShellScriptBuildPhase; + alwaysOutOfDate = 1; + buildActionMask = 2147483647; + files = ( + ); + inputFileListPaths = ( + ); + inputPaths = ( + ); + outputFileListPaths = ( + ); + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "if [[ \"$(uname -m)\" == arm64 ]]\nthen\n export PATH=\"/opt/homebrew/bin:$PATH\"\nfi\n\nif command -v swiftlint >/dev/null 2>&1\nthen\n swiftlint\nelse\n echo \"warning: `swiftlint` command not found - See https://github.com/realm/SwiftLint#installation for installation instructions.\"\nfi\n"; + }; + B3D95AE82F0FA9850066BDF2 /* ShellScript */ = { + isa = PBXShellScriptBuildPhase; + alwaysOutOfDate = 1; + buildActionMask = 2147483647; + files = ( + ); + inputFileListPaths = ( + ); + inputPaths = ( + ); + outputFileListPaths = ( + ); + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "if [[ \"$(uname -m)\" == arm64 ]]\nthen\n export PATH=\"/opt/homebrew/bin:$PATH\"\nfi\n\nif command -v swiftlint >/dev/null 2>&1\nthen\n swiftlint\nelse\n echo \"warning: `swiftlint` command not found - See https://github.com/realm/SwiftLint#installation for installation instructions.\"\nfi\n"; + }; +/* End PBXShellScriptBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 08267F272F0D06BC005A0066 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 2896AE3B2F100CD600D38732 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + BCFC85352F31FED800447A9A /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + BCFC853E2F31FED800447A9A /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 08267F2A2F0D06BC005A0066 /* SoloDeveloperTraining */; + targetProxy = BCFC853D2F31FED800447A9A /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin XCBuildConfiguration section */ + 08267F342F0D06BE005A0066 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = dwarf; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_TESTABILITY = YES; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + 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 = 26.1; + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; + MTL_FAST_MATH = YES; + ONLY_ACTIVE_ARCH = YES; + OTHER_SWIFT_FLAGS = ""; + SDKROOT = iphoneos; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + }; + name = Debug; + }; + 08267F352F0D06BE005A0066 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + 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 = 26.1; + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MTL_ENABLE_DEBUG_INFO = NO; + MTL_FAST_MATH = YES; + OTHER_SWIFT_FLAGS = ""; + SDKROOT = iphoneos; + SWIFT_COMPILATION_MODE = wholemodule; + VALIDATE_PRODUCT = YES; + }; + name = Release; + }; + 08267F372F0D06BE005A0066 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = app_icon_release; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_PREVIEWS = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = SoloDeveloperTraining/App/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = "개발자 키우기"; + INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.games"; + INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES; + INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES; + INFOPLIST_KEY_UILaunchStoryboardName = "Launch Screen"; + INFOPLIST_KEY_UISupportedInterfaceOrientations = UIInterfaceOrientationPortrait; + INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown"; + INFOPLIST_KEY_UIUserInterfaceStyle = Light; + IPHONEOS_DEPLOYMENT_TARGET = 17; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + MARKETING_VERSION = 1.1.2; + PRODUCT_BUNDLE_IDENTIFIER = kr.codesquad.boostcamp10.SoloDeveloperTraining; + PRODUCT_NAME = "$(TARGET_NAME)"; + STRING_CATALOG_GENERATE_SYMBOLS = YES; + SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; + SUPPORTS_MACCATALYST = NO; + SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; + SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_DEFAULT_ACTOR_ISOLATION = MainActor; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + }; + name = Debug; + }; + 08267F382F0D06BE005A0066 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = app_icon_release; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_PREVIEWS = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = SoloDeveloperTraining/App/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = "개발자 키우기"; + INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.games"; + INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES; + INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES; + INFOPLIST_KEY_UILaunchStoryboardName = "Launch Screen"; + INFOPLIST_KEY_UISupportedInterfaceOrientations = UIInterfaceOrientationPortrait; + INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown"; + INFOPLIST_KEY_UIUserInterfaceStyle = Light; + IPHONEOS_DEPLOYMENT_TARGET = 17; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + MARKETING_VERSION = 1.1.2; + PRODUCT_BUNDLE_IDENTIFIER = kr.codesquad.boostcamp10.SoloDeveloperTraining; + PRODUCT_NAME = "$(TARGET_NAME)"; + STRING_CATALOG_GENERATE_SYMBOLS = YES; + SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; + SUPPORTS_MACCATALYST = NO; + SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; + SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_DEFAULT_ACTOR_ISOLATION = MainActor; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + }; + name = Release; + }; + 2896AE402F100CD600D38732 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = app_icon_dev; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_PREVIEWS = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = "SoloDeveloperTraining/App/Info-Dev.plist"; + INFOPLIST_KEY_CFBundleDisplayName = "개발자 키우기 (Dev)"; + INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES; + INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES; + INFOPLIST_KEY_UILaunchStoryboardName = "Launch Screen"; + INFOPLIST_KEY_UIStatusBarStyle = ""; + INFOPLIST_KEY_UISupportedInterfaceOrientations = UIInterfaceOrientationPortrait; + INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown"; + INFOPLIST_KEY_UIUserInterfaceStyle = Light; + IPHONEOS_DEPLOYMENT_TARGET = 17; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + MARKETING_VERSION = 1.0; + OTHER_SWIFT_FLAGS = "-DDEV_BUILD"; + PRODUCT_BUNDLE_IDENTIFIER = kr.codesquad.boostcamp10.SoloDeveloperTraining.dev; + PRODUCT_NAME = "$(TARGET_NAME)"; + STRING_CATALOG_GENERATE_SYMBOLS = YES; + SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; + SUPPORTS_MACCATALYST = NO; + SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; + SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_DEFAULT_ACTOR_ISOLATION = MainActor; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + }; + name = Debug; + }; + 2896AE412F100CD600D38732 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = app_icon_dev; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = B3PWYBKFUK; + ENABLE_PREVIEWS = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = "SoloDeveloperTraining/App/Info-Dev.plist"; + INFOPLIST_KEY_CFBundleDisplayName = "개발자 키우기 (Dev)"; + INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES; + INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES; + INFOPLIST_KEY_UILaunchStoryboardName = "Launch Screen"; + INFOPLIST_KEY_UIStatusBarStyle = ""; + INFOPLIST_KEY_UISupportedInterfaceOrientations = UIInterfaceOrientationPortrait; + INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown"; + INFOPLIST_KEY_UIUserInterfaceStyle = Light; + IPHONEOS_DEPLOYMENT_TARGET = 17; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + MARKETING_VERSION = 1.0; + OTHER_SWIFT_FLAGS = "-DDEV_BUILD"; + PRODUCT_BUNDLE_IDENTIFIER = kr.codesquad.boostcamp10.SoloDeveloperTraining.dev; + PRODUCT_NAME = "$(TARGET_NAME)"; + STRING_CATALOG_GENERATE_SYMBOLS = YES; + SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; + SUPPORTS_MACCATALYST = NO; + SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; + SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_DEFAULT_ACTOR_ISOLATION = MainActor; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + }; + name = Release; + }; + BCFC85402F31FED800447A9A /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = B3PWYBKFUK; + GENERATE_INFOPLIST_FILE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 17; + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = sunghun.SoloDeveloperTrainingTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + STRING_CATALOG_GENERATE_SYMBOLS = NO; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_EMIT_LOC_STRINGS = NO; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/SoloDeveloperTraining.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/SoloDeveloperTraining"; + }; + name = Debug; + }; + BCFC85412F31FED800447A9A /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = B3PWYBKFUK; + GENERATE_INFOPLIST_FILE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 17; + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = sunghun.SoloDeveloperTrainingTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + STRING_CATALOG_GENERATE_SYMBOLS = NO; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_EMIT_LOC_STRINGS = NO; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/SoloDeveloperTraining.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/SoloDeveloperTraining"; + }; + name = Release; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 08267F262F0D06BC005A0066 /* Build configuration list for PBXProject "SoloDeveloperTraining" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 08267F342F0D06BE005A0066 /* Debug */, + 08267F352F0D06BE005A0066 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 08267F362F0D06BE005A0066 /* Build configuration list for PBXNativeTarget "SoloDeveloperTraining" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 08267F372F0D06BE005A0066 /* Debug */, + 08267F382F0D06BE005A0066 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 2896AE3F2F100CD600D38732 /* Build configuration list for PBXNativeTarget "SoloDeveloperTraining-Dev" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 2896AE402F100CD600D38732 /* Debug */, + 2896AE412F100CD600D38732 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + BCFC853F2F31FED800447A9A /* Build configuration list for PBXNativeTarget "SoloDeveloperTrainingTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + BCFC85402F31FED800447A9A /* Debug */, + BCFC85412F31FED800447A9A /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + }; + rootObject = 08267F232F0D06BC005A0066 /* Project object */; +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..919434a6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTraining.xcscheme b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTraining.xcscheme new file mode 100644 index 00000000..e8192f92 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTraining.xcscheme @@ -0,0 +1,91 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTrainingTests.xcscheme b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTrainingTests.xcscheme new file mode 100644 index 00000000..89673d20 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTrainingTests.xcscheme @@ -0,0 +1,66 @@ + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTrainingUITests.xcscheme b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTrainingUITests.xcscheme new file mode 100644 index 00000000..1e3b2199 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining.xcodeproj/xcshareddata/xcschemes/SoloDeveloperTrainingUITests.xcscheme @@ -0,0 +1,55 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/App/Info-Dev.plist b/SoloDeveloperTraining/SoloDeveloperTraining/App/Info-Dev.plist new file mode 100644 index 00000000..8948e570 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/App/Info-Dev.plist @@ -0,0 +1,12 @@ + + + + + UIAppFonts + + PFStardust-Regular.ttf + PFStardust-Bold.ttf + PFStardust-ExtraBold.ttf + + + diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/App/Info.plist b/SoloDeveloperTraining/SoloDeveloperTraining/App/Info.plist new file mode 100644 index 00000000..8948e570 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/App/Info.plist @@ -0,0 +1,12 @@ + + + + + UIAppFonts + + PFStardust-Regular.ttf + PFStardust-Bold.ttf + PFStardust-ExtraBold.ttf + + + diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/App/Launch Screen.storyboard b/SoloDeveloperTraining/SoloDeveloperTraining/App/Launch Screen.storyboard new file mode 100644 index 00000000..cbb70b75 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/App/Launch Screen.storyboard @@ -0,0 +1,26 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/App/SoloDeveloperTrainingApp.swift b/SoloDeveloperTraining/SoloDeveloperTraining/App/SoloDeveloperTrainingApp.swift new file mode 100644 index 00000000..adbba718 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/App/SoloDeveloperTrainingApp.swift @@ -0,0 +1,181 @@ +// +// SoloDeveloperTrainingApp.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import SwiftUI + +private enum Constant { + enum Animation { + static let transitionDuration: Double = 0.5 // 화면 전환 + static let blinkingDuration: Double = 1.0 // 깜빡임 + } + + enum Padding { + static let nicknamePopupHorizontal: CGFloat = 25 + static let errorPopupVertical: CGFloat = 20 + } + + enum Opacity { + static let overlay: Double = 0.5 + } +} + +@main +struct SoloDeveloperTrainingApp: App { + @State private var hasSeenIntro = false + @State private var showNicknameSetup = false + @State private var showTutorial = false + @State private var user: User? + @State private var showErrorPopup = false + @State private var errorMessage: String = "" + @Environment(\.scenePhase) private var scenePhase + + private let userRepository: UserRepository = FileManagerUserRepository() + + var body: some Scene { + WindowGroup { +#if DEV_BUILD + // Dev 타깃용 루트뷰 + ContentView() +#else + // 운영 타깃용 뷰 + Group { + if hasSeenIntro, let user { + MainView(user: user) + .transition(.opacity) + } else { + IntroView( + hasSeenIntro: $hasSeenIntro, + showNicknameSetup: $showNicknameSetup, + user: user + ) + } + } + .animation(.easeOut(duration: Constant.Animation.transitionDuration), value: hasSeenIntro) + .overlay { + nicknameSetupOverlay + } + .overlay { + errorPopupOverlay + } + .fullScreenCover(isPresented: $showTutorial) { + TutorialView(isPresented: $showTutorial) { + user?.record.tutorialCompleted = true + hasSeenIntro = true + showTutorial = false + } + .onAppear { + Task { + try? await Task.sleep(nanoseconds: UInt64(Constant.Animation.transitionDuration * 1_000_000_000)) + hasSeenIntro = true + } + } + } + .onAppear { + loadUser() + } + .onChange(of: scenePhase) { _, newPhase in + if newPhase == .background || newPhase == .inactive { + saveUser() + } + } +#endif + } + } +} + +#if !DEV_BUILD +private extension SoloDeveloperTrainingApp { + /// 저장된 User를 로드합니다. + func loadUser() { + Task { + do { + if let loadedUser = try await userRepository.load() { + await MainActor.run { + self.user = loadedUser + } + } + } catch { + await MainActor.run { + self.errorMessage = "사용자 데이터를 불러오는데 실패했습니다.\n\(error.localizedDescription)" + self.showErrorPopup = true + } + } + } + } + + /// 현재 User를 저장합니다. + func saveUser() { + guard let user = user else { return } + Task { + do { + try await userRepository.save(user) + } catch { + await MainActor.run { + self.errorMessage = "사용자 데이터를 저장하는데 실패했습니다.\n\(error.localizedDescription)" + self.showErrorPopup = true + } + } + } + } + + @ViewBuilder + var nicknameSetupOverlay: some View { + if showNicknameSetup { + ZStack { + Color.black.opacity(Constant.Opacity.overlay) + .ignoresSafeArea() + + NicknameSetupView( + onStart: { nickname in + user = User(nickname: nickname) + showNicknameSetup = false + withAnimation(.easeOut(duration: Constant.Animation.transitionDuration)) { + hasSeenIntro = true + } + }, + onTutorial: { nickname in + user = User(nickname: nickname) + showNicknameSetup = false + showTutorial = true + } + ) + .padding(.horizontal, Constant.Padding.nicknamePopupHorizontal) + } + } + } + + @ViewBuilder + var errorPopupOverlay: some View { + if showErrorPopup { + ZStack { + Color.black.opacity(Constant.Opacity.overlay) + .ignoresSafeArea() + + Popup(title: "오류") { + VStack(spacing: 0) { + Text(errorMessage) + .textStyle(.body) + .foregroundColor(.black) + .multilineTextAlignment(.center) + .frame(maxWidth: .infinity, alignment: .center) + .padding(.vertical, Constant.Padding.errorPopupVertical) + + HStack(spacing: 0) { + Spacer() + MediumButton(title: "확인", isFilled: true) { + showErrorPopup = false + } + Spacer() + } + } + } + .padding(.horizontal, Constant.Padding.nicknamePopupHorizontal) + } + } + } +} +#endif diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/AppColors.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/AppColors.swift new file mode 100644 index 00000000..63f010ac --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/AppColors.swift @@ -0,0 +1,48 @@ +// +// AppColors.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/12/26. +// + +import SwiftUI + +enum AppColors { + + // MARK: - Gray 100 ~ 700 + static let gray100 = Color("Gray100") + static let gray200 = Color("Gray200") + static let gray300 = Color("Gray300") + static let gray400 = Color("Gray400") + static let gray500 = Color("Gray500") + static let gray600 = Color("Gray600") + static let gray700 = Color("Gray700") + + // MARK: - Orange 100 ~ 700 + static let orange100 = Color("Orange100") + static let orange200 = Color("Orange200") + static let orange300 = Color("Orange300") + static let orange400 = Color("Orange400") + static let orange500 = Color("Orange500") + static let orange600 = Color("Orange600") + static let orange700 = Color("Orange700") + + // MARK: - Beige 100 ~ 400 + static let beige100 = Color("Beige100") + static let beige200 = Color("Beige200") + static let beige300 = Color("Beige300") + static let beige400 = Color("Beige400") + + // MARK: - Accent Colors + static let lightGreen = Color("LightGreen") + static let accentGreen = Color("AccentGreen") + static let accentYellow = Color("AccentYellow") + static let lightOrange = Color("LightOrange") + static let accentRed = Color("AccentRed") + + // MARK: - Pastel Colors + static let pastelYellow = Color("PastelYellow") + static let pastelPink = Color("PastelPink") + static let pastelBlue = Color("PastelBlue") + static let pastelGreen = Color("PastelGreen") +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/BlockItem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/BlockItem.swift new file mode 100644 index 00000000..daecea7f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/BlockItem.swift @@ -0,0 +1,103 @@ +// +// BlockItem.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-13. +// + +import SpriteKit +import SwiftUI + +private enum Constant { + // 움직이는 속도(작을수록 빠름) + static let moveDurationRange = 0.5...1.0 + // 질량 + static let physicsMass: CGFloat = 1.0 + // 탄성 (0으로 설정하여 튀지 않도록) + static let physicsRestitution: CGFloat = 0.0 + // 마찰 (1.0으로 설정하여 미끄러지지 않도록) + static let physicsFriction: CGFloat = 1.0 +} + +final class BlockItem: SKSpriteNode { + private var moveAction: SKAction? + let blockType: BlockType + + init(type: BlockType = .blue) { + self.blockType = type + let texture = SKTexture(imageNamed: type.imageName) + super.init(texture: texture, color: .clear, size: type.size) + self.anchorPoint = CGPoint(x: 0.5, y: 0) + } + + required init?(coder aDecoder: NSCoder) { + return nil + } + + func startMoving(distance: CGFloat) { + let duration = Double.random(in: Constant.moveDurationRange) + let moveRight = SKAction.moveBy(x: distance, y: 0, duration: duration) + let moveLeft = SKAction.moveBy(x: -distance, y: 0, duration: duration) + let sequence = SKAction.sequence([moveRight, moveLeft]) + let repeatAction = SKAction.repeatForever(sequence) + self.moveAction = repeatAction + self.run(repeatAction, withKey: "moving") + } + + func stopMoving() { + self.removeAction(forKey: "moving") + self.moveAction = nil + } + + func enableGravity() { + self.physicsBody = SKPhysicsBody(rectangleOf: self.size) + self.physicsBody?.isDynamic = true + self.physicsBody?.allowsRotation = true + self.physicsBody?.mass = Constant.physicsMass + self.physicsBody?.restitution = Constant.physicsRestitution + self.physicsBody?.friction = Constant.physicsFriction + } + + func fixPosition() { + self.physicsBody?.isDynamic = false + self.physicsBody?.allowsRotation = false + } + + func setupPhysicsBody(isDynamic: Bool = false) { + self.physicsBody = SKPhysicsBody(rectangleOf: self.size) + self.physicsBody?.isDynamic = isDynamic + self.physicsBody?.allowsRotation = false + } +} + +#Preview { + VStack(spacing: 20) { + Text("BlockItem Preview") + .font(.headline) + + ForEach(BlockType.allCases, id: \.self) { type in + VStack(spacing: 8) { + Text(String(describing: type)) + .font(.caption) + .foregroundColor(.secondary) + + SpriteView( + scene: { + let scene = SKScene(size: type.size) + scene.backgroundColor = .clear + scene.scaleMode = .aspectFit + + let block = BlockItem(type: type) + block.position = CGPoint(x: scene.size.width / 2, y: scene.size.height / 2) + scene.addChild(block) + + return scene + }(), + options: [.allowsTransparency] + ) + .frame(width: type.size.width, height: type.size.height) + } + } + } + .padding() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CareerProgressBar.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CareerProgressBar.swift new file mode 100644 index 00000000..f9d4dc7c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CareerProgressBar.swift @@ -0,0 +1,79 @@ +// +// CareerProgressBar.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/14/26. +// + +import SwiftUI + +private enum Constants { + enum Size { + static let barHeight: CGFloat = 22 + } + + enum Spacing { + static let vertical: CGFloat = 4 + static let leadingLabelSpacing: CGFloat = 4 + static let trailingLabelSpacing: CGFloat = 4 + } +} + +struct CareerProgressBar: View { + let career: Career + let totalEarnedMoney: Int + let progress: Double + + var body: some View { + VStack(spacing: Constants.Spacing.vertical) { + ZStack { + // 배경 + Rectangle() + .fill(.gray200) + + // 진행 바 + Rectangle() + .fill(.orange300) + .scaleEffect(x: progress, y: 1, anchor: .leading) + } + .frame(height: Constants.Size.barHeight) + HStack(alignment: .top) { + HStack(spacing: Constants.Spacing.leadingLabelSpacing) { + Text("누적").textStyle(.caption) + CurrencyLabel( + axis: .horizontal, + icon: .gold, + textStyle: .caption, + value: totalEarnedMoney + ) + } + Spacer() + VStack( + alignment: .trailing, + spacing: Constants.Spacing.trailingLabelSpacing + ) { + Text(career.nextCareer?.rawValue ?? "").textStyle(.caption) + CurrencyLabel( + axis: .horizontal, + icon: .gold, + textStyle: .caption, + value: career.nextCareer?.requiredWealth ?? 0 + ) + } + } + } + } +} + +#Preview { + CareerProgressBar( + career: .unemployed, + totalEarnedMoney: 0, + progress: 0.0 + ) + CareerProgressBar( + career: .laptopOwner, + totalEarnedMoney: 1200, + progress: 0.4 + ) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CareerRow.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CareerRow.swift new file mode 100644 index 00000000..1765055c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CareerRow.swift @@ -0,0 +1,119 @@ +// +// CareerRow.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/15/26. +// + +import SwiftUI + +private enum Constant { + static let imageSize: CGSize = .init(width: 49, height: 49) + static let cornerRadius: CGFloat = 4 + static let borderWidth: CGFloat = 1 + static let profileOpacity: CGFloat = 0.6 + + enum Spacing { + static let horizontal: CGFloat = 8 + static let vertical: CGFloat = 6 + } +} + +struct CareerRow: View { + let career: Career + let userCareer: Career + + var body: some View { + HStack(alignment: .center, spacing: Constant.Spacing.horizontal) { + // 프로필 이미지 + Image(displayImageName) + .resizable() + .clipShape(RoundedRectangle(cornerRadius: Constant.cornerRadius)) + .frame(width: Constant.imageSize.width, height: Constant.imageSize.height) + .overlay( + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .stroke(borderColor, lineWidth: Constant.borderWidth) + ) + .opacity(state == .locked ? Constant.profileOpacity : 1.0) + + // 우측 컨텐츠 + VStack(alignment: .leading, spacing: Constant.Spacing.vertical) { + HStack(alignment: .bottom) { + Text(career.rawValue) + .foregroundStyle(textColor) + .textStyle(.subheadline) + Spacer() + if state == .completed { + Text("완료") + .foregroundStyle(.gray400) + .textStyle(.label) + } + } + Text(career.description) + .foregroundStyle(textColor) + .textStyle(.label) + } + } + } +} + +private extension CareerRow { + /// 커리어 Row 상태 + enum CareerState { + case completed + case inProgress + case locked + } + + /// 현재 상태 계산 + var state: CareerState { + let careerIndex = Career.allCases.firstIndex(of: career) ?? 0 + let userIndex = Career.allCases.firstIndex(of: userCareer) ?? 0 + + if career == userCareer { + return .inProgress + } else if careerIndex < userIndex { + return .completed + } else { + return .locked + } + } + + /// 테두리 색상 + var borderColor: Color { + state == .inProgress ? AppColors.accentYellow : AppColors.gray200 + } + + /// 텍스트 색상 + var textColor: Color { + state == .completed ? AppColors.gray400 : .black + } + + /// 표시할 이미지 + var displayImageName: String { + state == .locked ? "profile_locked" : career.imageName + } +} + +#Preview { + VStack(spacing: 12) { + // 완료된 직업 + CareerRow( + career: .unemployed, + userCareer: .aspiringDeveloper + ) + + // 현재 진행중인 직업 + CareerRow( + career: .aspiringDeveloper, + userCareer: .aspiringDeveloper + ) + + // 잠긴 직업 + CareerRow( + career: .juniorDeveloper, + userCareer: .aspiringDeveloper + ) + } + .padding(.horizontal) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CurrencyLabel.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CurrencyLabel.swift new file mode 100644 index 00000000..f0a46b45 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/CurrencyLabel.swift @@ -0,0 +1,71 @@ +// +// CurrencyLabel.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-12. +// + +import SwiftUI + +private enum Constants { + static let iconTextSpacing: CGFloat = 3 +} + +struct CurrencyLabel: View { + private let axis: Axis + private let icon: Icon + private let textStyle: TypographyStyle + private let value: Int + + init(axis: Axis, icon: Icon, textStyle: TypographyStyle = .caption, value: Int) { + self.axis = axis + self.icon = icon + self.textStyle = textStyle + self.value = value + } + + var body: some View { + switch axis { + case .vertical: + VStack(spacing: Constants.iconTextSpacing) { + switch icon { + case .diamond: Image(.iconDiamondGreen) + case .gold: Image(.iconCoinBag) + } + + Text(value.formatted) + .textStyle(textStyle) + } + + case .horizontal: + HStack(spacing: Constants.iconTextSpacing) { + switch icon { + case .diamond: Image(.iconDiamondGreen) + case .gold: Image(.iconCoinBag) + } + + Text(value.formatted) + .textStyle(textStyle) + } + } + } +} + +extension CurrencyLabel { + enum Axis { + case vertical + case horizontal + } + + enum Icon { + case diamond + case gold + } +} + +#Preview { + CurrencyLabel(axis: .vertical, icon: .diamond, textStyle: .body, value: 30540) + CurrencyLabel(axis: .vertical, icon: .gold, textStyle: .callout, value: 30540) + CurrencyLabel(axis: .horizontal, icon: .diamond, textStyle: .caption, value: 30540) + CurrencyLabel(axis: .horizontal, icon: .gold, textStyle: .headline, value: 30540) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/DefaultSegmentControl.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/DefaultSegmentControl.swift new file mode 100644 index 00000000..89a354cc --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/DefaultSegmentControl.swift @@ -0,0 +1,123 @@ +// +// DefaultSegmentControl.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/13/26. +// + +import SwiftUI + +private enum Constant { + static let segmentSpacing: CGFloat = 0 + static let segmentHeight: CGFloat = 40 + static let segmentPadding: CGFloat = 4 + static let lineWidth: CGFloat = 2 + + enum Color { + static let selectedBackground = AppColors.orange500 + static let unselectedBackground = AppColors.gray100 + static let selectedText = SwiftUI.Color.white + static let unselectedText = AppColors.gray500 + static let border = SwiftUI.Color.black + } +} + +struct DefaultSegmentControl: View { + /// 선택된 세그먼트 인덱스 + @Binding var selection: Int + + let segments: [String] + + var body: some View { + HStack(spacing: Constant.segmentSpacing) { + ForEach(0.. Void + + var body: some View { + Button(action: action) { + Text(title) + .textStyle(.subheadline) + .foregroundStyle(textColor) + .frame(maxWidth: .infinity) + .frame(maxHeight: Constant.segmentHeight) + .background(backgroundColor) + .animation(.none, value: isSelected) + } + .buttonStyle(.soundTap) + } + + var backgroundColor: SwiftUI.Color { + isSelected ? Constant.Color.selectedBackground : Constant.Color.unselectedBackground + } + + var textColor: SwiftUI.Color { + isSelected ? Constant.Color.selectedText : Constant.Color.unselectedText + } +} + +#Preview { + @Previewable @State var selection2 = 0 + @Previewable @State var selection3 = 0 + @Previewable @State var selection4 = 0 + + VStack(spacing: 40) { + VStack(spacing: 10) { + Text("2개 세그먼트") + .textStyle(.title) + DefaultSegmentControl( + selection: $selection2, + segments: ["아이템", "부동산"] + ) + Text("선택된 인덱스: \(selection2)") + .textStyle(.body) + } + + VStack(spacing: 10) { + Text("3개 세그먼트") + .textStyle(.title) + DefaultSegmentControl( + selection: $selection3, + segments: ["옵션 1", "옵션 2", "옵션 3"] + ) + Text("선택된 인덱스: \(selection3)") + .textStyle(.body) + } + + VStack(spacing: 10) { + Text("4개 세그먼트") + .textStyle(.title) + DefaultSegmentControl( + selection: $selection4, + segments: ["첫번째", "두번째", "세번째", "네번째"] + ) + Text("선택된 인덱스: \(selection4)") + .textStyle(.body) + } + } + .padding() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/DropItem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/DropItem.swift new file mode 100644 index 00000000..c43bceb8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/DropItem.swift @@ -0,0 +1,50 @@ +// +// DropItem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/13/26. +// + +import SwiftUI + +private enum Constant { + static let itemSize = CGSize(width: 24, height: 24) +} + +struct DropItem: View { + + let type: DropItemType + + var body: some View { + Image(type.imageResource) + .resizable() + .frame(width: Constant.itemSize.width, height: Constant.itemSize.height) + } +} + +extension DropItem { + enum DropItemType { + case smallGold + case largeGold + case bug + + var imageResource: ImageResource { + switch self { + case .smallGold: + return .dodgeDropSmallGold + case .largeGold: + return .dodgeDropLargeGold + case .bug: + return .dodgeDropBug + } + } + } +} + +#Preview { + HStack { + DropItem(type: .smallGold) + DropItem(type: .largeGold) + DropItem(type: .bug) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/EffectLabel.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/EffectLabel.swift new file mode 100644 index 00000000..a3a9053a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/EffectLabel.swift @@ -0,0 +1,125 @@ +// +// EffectLabel.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/13/26. +// + +import SwiftUI + +private enum Constant { + enum Size { + static let icon = CGSize(width: 18, height: 18) + } + + enum Spacing { + static let horizontal: CGFloat = 3 + } + + enum Animation { + // 단계 별 애니메이션 지속 시간 + static let durationSec: Double = 1.5 + // 각 단계의 간격 시간 + static let sleepNanosec: UInt64 = 150_000_000 + // 각 단계에 따른 투명도, offsetY + static let steps: [(Double, CGFloat)] = [ + (1.0, 0), + (0.5, -3), + (0.2, -6), + (0.1, -9), + (0.0, -12) + ] + } +} + +/// EffectLabel 데이터 모델 +struct EffectLabelData: Identifiable { + let id: UUID + let position: CGPoint + let value: Int +} + +struct EffectLabel: View { + let value: Int + let onComplete: () -> Void + + @State private var opacity: Double = 1.0 + @State private var offsetY: CGFloat = 0 + @State private var shouldShow = true + + init( + value: Int, + onComplete: @escaping () -> Void = {} + ) { + self.value = value + self.onComplete = onComplete + } + + private var isZero: Bool { + value == 0 + } + + private var isIncrease: Bool { + value > 0 + } + + private var color: Color { + isZero || isIncrease ? .lightGreen : .accentRed + } + + var body: some View { + if shouldShow { + HStack(spacing: Constant.Spacing.horizontal) { + if !isZero { + Image(isIncrease ? .iconPlus : .iconMinus) + .renderingMode(.template) + .resizable() + .frame( + width: Constant.Size.icon.width, + height: Constant.Size.icon.height + ) + .foregroundStyle(color) + } + + Image(.iconCoinStack) + .resizable() + .frame( + width: Constant.Size.icon.width, + height: Constant.Size.icon.height + ) + + Text(abs(value).formatted()) + .textStyle(.subheadline) + .foregroundStyle(color) + } + .opacity(opacity) + .offset(y: offsetY) + .onAppear { + runAnimation() + } + } + } + + private func runAnimation() { + Task { + for (opacityValue, offsetValue) in Constant.Animation.steps { + withAnimation( + .easeOut(duration: Constant.Animation.durationSec) + ) { + opacity = opacityValue + offsetY = offsetValue + } + try? await Task + .sleep(nanoseconds: Constant.Animation.sleepNanosec) + } + shouldShow = false + onComplete() + } + } +} + +#Preview { + EffectLabel(value: 100000) + EffectLabel(value: -200000) + EffectLabel(value: 0) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/GamePauseWrapper.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/GamePauseWrapper.swift new file mode 100644 index 00000000..bb2e50d9 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/GamePauseWrapper.swift @@ -0,0 +1,129 @@ +// +// GamePauseWrapper.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/27/26. +// + +import SwiftUI + +private enum Constant { + static let opacity: Double = 0.5 + static let animation: Animation = .easeOut(duration: 0.25) + static let blurRadius: CGFloat = 2 + static let iconSize: CGFloat = 28 + + enum Spacing { + static let buttonHorizontal: CGFloat = 50 + static let iconTextHorizontal: CGFloat = 8 + } + + enum ButtonText { + static let leave: String = "나가기" + static let resume: String = "계속하기" + } +} + +struct GamePauseWrapper: ViewModifier { + @Environment(\.scenePhase) private var scenePhase + @State private var isPaused: Bool = false + + @Binding var isGameViewDisappeared: Bool + + let height: CGFloat + let onLeave: () -> Void + let onPause: () -> Void + let onResume: () -> Void + + func body(content: Content) -> some View { + ZStack { + content + .blur(radius: isPaused ? Constant.blurRadius : 0) + if isPaused { + pauseOverlay + .transition(.opacity.combined(with: .scale(scale: 0.95))) + } + } + .onChange(of: scenePhase) { oldPhase, newPhase in + handleScenePhase(oldPhase: oldPhase, newPhase: newPhase) + } + .onChange(of: isGameViewDisappeared) { _, newValue in + if newValue { + handleGameViewDisappeard() + } + } + } +} + +// MARK: - Pause Overlay +private extension GamePauseWrapper { + var pauseOverlay: some View { + ZStack { + Rectangle() + .fill(.white.opacity(Constant.opacity)) + .frame(height: height) + .contentShape(Rectangle()) + .onTapGesture { } // 배경 터치 무효화 + + HStack(spacing: Constant.Spacing.buttonHorizontal) { + pauseButton( + title: Constant.ButtonText.leave, + iconImage: .iconCancel, + action: handleLeave + ) + + pauseButton( + title: Constant.ButtonText.resume, + iconImage: .iconPlay, + action: handleResume + ) + } + } + .frame(maxWidth: .infinity) + } + + func pauseButton( + title: String, + iconImage: ImageResource, + action: @escaping () -> Void + ) -> some View { + Button(action: action) { + HStack(spacing: Constant.Spacing.iconTextHorizontal) { + Image(iconImage) + .resizable() + .frame(width: Constant.iconSize, height: Constant.iconSize) + Text(title) + .textStyle(.title2) + .foregroundStyle(.black) + } + } + .buttonStyle(.soundTap) + } +} + +// MARK: - Actions +private extension GamePauseWrapper { + func handleGameViewDisappeard() { + isPaused = true + onPause() + } + + func handleScenePhase(oldPhase: ScenePhase, newPhase: ScenePhase) { + // 앱이 비활성 상태로 갈 때 즉시 일시정지 + if newPhase != .active { + handleGameViewDisappeard() + } + } + + func handleLeave() { + guard isPaused else { return } + isPaused = false + onLeave() + } + + func handleResume() { + guard isPaused else { return } + isPaused = false + onResume() + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/GameToolBar.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/GameToolBar.swift new file mode 100644 index 00000000..3150d79f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/GameToolBar.swift @@ -0,0 +1,297 @@ +// +// GameToolBar.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/12/26. +// + +import SwiftUI + +private enum Constant { + enum Size { + static let closeButton = CGSize(width: 31, height: 31) + static let coffeeIcon = CGSize(width: 22, height: 22) + static let energyDrinkIcon = CGSize(width: 20, height: 22) + } + + enum Spacing { + static let toolBar: CGFloat = 20 + static let consumableItem: CGFloat = 6 + static let itemIconText: CGFloat = 2 + } + + static let itemCountLabelWidth: CGFloat = 16 + static let feverBarHeight: CGFloat = 15 + static let disabledAlpha: CGFloat = 0.3 +} + +struct GameToolBar: View { + + // MARK: - Properties + /// 닫기 버튼 탭 핸들러 + let closeButtonDidTapHandler: () -> Void + /// 커피 버튼 탭 핸들러 + let coffeeButtonDidTapHandler: () -> Void + /// 에너지 드링크 버튼 탭 핸들러 + let energyDrinkButtonDidTapHandler: () -> Void + + /// 피버 상태 관리 객체 + let feverState: FeverState + + /// 버프 시스템 + let buffSystem: BuffSystem + + /// 커피 보유 개수 + let coffeeCount: Binding + /// 에너지 드링크 보유 개수 + let energyDrinkCount: Binding + + var body: some View { + HStack(spacing: Constant.Spacing.toolBar) { + closeButton + feverBar + HStack(spacing: Constant.Spacing.consumableItem) { + coffeeButton + energyDrinkButton + } + } + } +} + +// MARK: - SubViews +private extension GameToolBar { + /// 닫기 버튼 + var closeButton: some View { + Button { + closeButtonDidTapHandler() + SoundService.shared.stopAllSFX() + SoundService.shared.trigger(.buttonTap) + } label: { + Image(.iconClose) + .resizable() + .frame( + width: Constant.Size.closeButton.width, + height: Constant.Size.closeButton.height + ) + } + .buttonStyle(.soundTap) + } + + /// 피버 게이지 바 + var feverBar: some View { + GeometryReader { geometry in + ZStack(alignment: .leading) { + // 배경 바 (이전 단계 색상) + Rectangle() + .frame(height: Constant.feverBarHeight) + .foregroundStyle(feverBarBackgroundColor) + + // 전경 바 (현재 단계 색상, 진행도에 따라 너비 변경) + Rectangle() + .frame( + width: feverBarFillWidth(totalWidth: geometry.size.width), + height: Constant.feverBarHeight + ) + .foregroundStyle(feverBarFillColor) + + // 피버 배수 텍스트 + if feverState.feverStage != 0 { + Text(String(format: "Fever %.1fx !!", feverState.feverMultiplier)) + .textStyle(.caption2) + .foregroundStyle(.white) + .frame(maxWidth: .infinity) + } + } + } + .frame(height: Constant.feverBarHeight) + } + + /// 커피 아이템 버튼 + var coffeeButton: some View { + Button { + coffeeButtonDidTapHandler() + } label: { + HStack(spacing: Constant.Spacing.itemIconText) { + Image(.iconCoffee) + .resizable() + .frame( + width: Constant.Size.coffeeIcon.width, + height: Constant.Size.coffeeIcon.height + ) + .opacity(coffeeCount.wrappedValue == 0 ? Constant.disabledAlpha : 1.0) + .mask( + GeometryReader { geometry in + VStack(spacing: 0) { + Rectangle() + .opacity(Constant.disabledAlpha) + .frame(height: geometry.size.height * cooldownProgress(for: .coffee)) + Rectangle() + .opacity(1) + .frame(height: geometry.size.height * (1 - cooldownProgress(for: .coffee))) + } + } + ) + Text("\(coffeeCount.wrappedValue)") + .textStyle(.caption2) + .foregroundStyle(isCoffeeBuffActive ? .gray300 : .black) + .frame(width: Constant.itemCountLabelWidth) + } + } + .disabled(isCoffeeBuffActive || coffeeCount.wrappedValue == 0) + } + + /// 에너지 드링크 아이템 버튼 + var energyDrinkButton: some View { + Button { + energyDrinkButtonDidTapHandler() + } label: { + HStack(spacing: Constant.Spacing.itemIconText) { + Image(.iconEnergyDrink) + .resizable() + .frame( + width: Constant.Size.energyDrinkIcon.width, + height: Constant.Size.energyDrinkIcon.height + ) + .opacity(energyDrinkCount.wrappedValue == 0 ? Constant.disabledAlpha : 1.0) + .mask( + GeometryReader { geometry in + VStack(spacing: 0) { + Rectangle() + .opacity(Constant.disabledAlpha) + .frame(height: geometry.size.height * cooldownProgress(for: .energyDrink)) + Rectangle() + .opacity(1) + .frame(height: geometry.size.height * (1 - cooldownProgress(for: .energyDrink))) + } + } + ) + Text("\(energyDrinkCount.wrappedValue)") + .textStyle(.caption2) + .foregroundStyle(isEnergyDrinkBuffActive ? .gray300 : .black) + .frame(width: Constant.itemCountLabelWidth) + } + } + .disabled(isEnergyDrinkBuffActive || energyDrinkCount.wrappedValue == 0) + } +} + +// MARK: - Helper +private extension GameToolBar { + /// 커피 버프 활성화 여부 + var isCoffeeBuffActive: Bool { + buffSystem.coffeeDuration > 0 + } + + /// 에너지 드링크 버프 활성화 여부 + var isEnergyDrinkBuffActive: Bool { + buffSystem.energyDrinkDuration > 0 + } + + /// 피버 바 전경 색상 (현재 단계) + var feverBarFillColor: Color { + switch feverState.feverStage { + case 0: return .gray400 + case 1: return .accentYellow + case 2: return .lightOrange + case 3: return .accentRed + default: return .gray400 + } + } + + /// 피버 바 배경 색상 (이전 단계) + var feverBarBackgroundColor: Color { + switch feverState.feverStage { + case 0: return .gray200 + case 1: return .gray400 + case 2: return .accentYellow + case 3: return .lightOrange + default: return .accentRed + } + } + + /// 현재 피버 단계 내에서의 진행도에 따른 바 너비 계산 + /// - Parameter totalWidth: 피버 바 전체 너비 + /// - Returns: 현재 진행도에 따른 바 너비 + func feverBarFillWidth(totalWidth: CGFloat) -> CGFloat { + let currentPercent = feverState.feverPercent + // 현재 단계 내에서의 진행도 (0.0 ~ 1.0) + let progressRatio: Double + + switch currentPercent { + case 0..<100: + progressRatio = currentPercent / 100.0 + case 100..<200: + progressRatio = (currentPercent - 100) / 100.0 + case 200..<300: + progressRatio = (currentPercent - 200) / 100.0 + case 300...: + progressRatio = min((currentPercent - 300) / 100.0, 1.0) + default: + progressRatio = 0 + } + return totalWidth * progressRatio + } + + /// 소비 아이템 쿨타임 진행도 (0.0 ~ 1.0, 0이 쿨타임 완료, 1이 쿨타임 시작) + func cooldownProgress(for type: ConsumableType) -> Double { + let duration: Int + switch type { + case .coffee: + duration = buffSystem.coffeeDuration + case .energyDrink: + duration = buffSystem.energyDrinkDuration + } + + guard duration > 0 else { return 0.0 } + let totalDuration = Double(type.duration) + let remainingDuration = Double(duration) + return remainingDuration / totalDuration + } +} + +#Preview { + @Previewable @State var coffeeCount: Int = 10 + @Previewable @State var drinkCount: Int = 10 + let feverSystem = FeverSystem(decreaseInterval: 0.1, decreasePercentPerTick: 3) + let buffSystem = BuffSystem() + + VStack { + Button { + if !feverSystem.isRunning { + feverSystem.start() + } + feverSystem.gainFever(20) + } label: { + Text("GainFever") + } + + Button { + buffSystem.useConsumableItem(type: .coffee) + } label: { + Text("Use Coffee") + } + + Button { + buffSystem.useConsumableItem(type: .energyDrink) + } label: { + Text("Use Energy Drink") + } + + GameToolBar( + closeButtonDidTapHandler: { + print("Close") + }, + coffeeButtonDidTapHandler: { + coffeeCount -= 1 + }, + energyDrinkButtonDidTapHandler: { + drinkCount -= 1 + }, + feverState: feverSystem, + buffSystem: buffSystem, + coffeeCount: $coffeeCount, + energyDrinkCount: $drinkCount + ) + .padding() + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/HousingCard.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/HousingCard.swift new file mode 100644 index 00000000..cbe3d5d1 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/HousingCard.swift @@ -0,0 +1,123 @@ +// +// HousingCard.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/15/26. +// + +import SwiftUI + +private enum Constant { + static let cardWidth: CGFloat = 225 + static let cornerRadius: CGFloat = 6 + static let lineWidth: CGFloat = 3 + + enum Padding { + static let horizontal: CGFloat = 16 + static let top: CGFloat = 16 + static let bottom: CGFloat = 15 + static let textSpacing: CGFloat = 4 + static let titleSpacing: CGFloat = 15 + static let imageTop: CGFloat = 19 + static let buttonTop: CGFloat = 14 + } +} + +struct HousingCard: View { + let housing: Housing + let state: ItemState + let isSelected: Bool + let onTap: () -> Void + let onButtonTap: () -> Void + + var body: some View { + HousingCardContent( + housing: housing, + buttonTitle: state == .locked ? "장착중" : "이사하기", + isButtonEnabled: state == .available, + onButtonTap: onButtonTap + ) + .contentShape(RoundedRectangle(cornerRadius: Constant.cornerRadius)) + .overlay { + if isSelected { + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .stroke(Color.black, lineWidth: Constant.lineWidth) + } + } + .onTapGesture { onTap() } + } +} + +private struct HousingCardContent: View { + let housing: Housing + let buttonTitle: String + let isButtonEnabled: Bool + let onButtonTap: () -> Void + + var body: some View { + VStack(spacing: 0) { + VStack(spacing: 0) { + // 상단 텍스트 영역 + HStack(spacing: Constant.Padding.titleSpacing) { + Text(housing.displayTitle) + .textStyle(.callout) + + Text("₩\(housing.cost.gold.formatted)") + .textStyle(.caption2) + .frame(maxWidth: .infinity, alignment: .leading) + } + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.top, Constant.Padding.top) + + Text("초당 골드 획득량 \(housing.goldPerSecond.formatted)") + .textStyle(.label) + .frame(maxWidth: .infinity, alignment: .leading) + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.top, Constant.Padding.textSpacing) + + // 이미지 영역 + Image(housing.imageName) + .resizable() + .aspectRatio(contentMode: .fill) + .frame(width: Constant.cardWidth) + .frame(maxHeight: .infinity) + .clipped() + .padding(.top, Constant.Padding.imageTop) + .padding(.bottom, Constant.Padding.buttonTop) + } + .contentShape(Rectangle()) + + // 버튼 영역 + LargeButton( + title: buttonTitle, + isEnabled: isButtonEnabled + ) { + onButtonTap() + } + .padding(.bottom, Constant.Padding.bottom) + } + .foregroundStyle(.black) + .frame(maxHeight: .infinity) + .frame(width: Constant.cardWidth) + .background(AppColors.gray100) + .cornerRadius(Constant.cornerRadius) + } +} + +#Preview { + @Previewable @State var isSelected = false + @Previewable @State var state: ItemState = .available + + HousingCard( + housing: .init(tier: .villa), + state: state, + isSelected: isSelected, + onTap: { + isSelected.toggle() + }, + onButtonTap: { + state = .locked + } + ) + .frame(height: 500) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/ItemRow.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/ItemRow.swift new file mode 100644 index 00000000..5e8be5ae --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/ItemRow.swift @@ -0,0 +1,116 @@ +// +// ItemRow.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/15/26. +// + +import SwiftUI + +private enum Constant { + + static let imageSize: CGSize = .init(width: 38, height: 38) + static let cornerRadius: CGFloat = 4 + static let priceButtonWidth: CGFloat = 95 + + enum Spacing { + static let horizontal: CGFloat = 8 + static let vertical: CGFloat = 4 + } + + enum Padding { + static let horizontal: CGFloat = 16 + static let vertical: CGFloat = 8 + } +} + +struct ItemRow: View { + + let title: String + let description: String + let imageName: String + let cost: Cost + let state: ItemState + let action: () -> Void + let onLongPressAction: (() -> Bool)? + + init( + title: String, + description: String, + imageName: String, + cost: Cost, + state: ItemState, + action: @escaping () -> Void, + onLongPressAction: (() -> Bool)? = nil + ) { + self.title = title + self.description = description + self.imageName = imageName + self.cost = cost + self.state = state + self.action = action + self.onLongPressAction = onLongPressAction + } + + var body: some View { + HStack(spacing: Constant.Spacing.horizontal) { + Image(imageName) + .resizable() + .clipShape(RoundedRectangle(cornerRadius: Constant.cornerRadius)) + .frame(width: Constant.imageSize.width, height: Constant.imageSize.height) + VStack(alignment: .leading, spacing: Constant.Spacing.vertical) { + Text(title) + .foregroundStyle(.black) + .textStyle(.subheadline) + Text(description) + .foregroundStyle(.black) + .textStyle(.label) + } + + Spacer() + + PriceButton( + cost: cost, + state: state, + axis: .vertical, + width: Constant.priceButtonWidth, + action: action, + onLongPressRepeat: onLongPressAction + ) + } + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.vertical, Constant.Padding.vertical) + } +} + +#Preview { + VStack { + ItemRow( + title: "강화 / 아이템 이름 이름 이름", + description: "항목 설명 설명 설명", + imageName: "housing_street", + cost: .init(gold: 1_000_0000000000, diamond: 99), + state: .available + ) { + print("Tapped") + } + ItemRow( + title: "강화 / 아이템 이름 이름 이름", + description: "항목 설명 설명 설명", + imageName: "housing_street", + cost: .init(gold: 1_000_000), + state: .locked + ) { + print("Tapped") + } + ItemRow( + title: "강화 / 아이템 이름 이름 이름", + description: "항목 설명 설명 설명", + imageName: "housing_street", + cost: .init(gold: 1_000_000), + state: .insufficient + ) { + print("Tapped") + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LanguageButton.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LanguageButton.swift new file mode 100644 index 00000000..67b3bd0a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LanguageButton.swift @@ -0,0 +1,96 @@ +// +// LanguageButton.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/12/26. +// + +import SwiftUI + +private enum Constant { + static let vStackSpacing: CGFloat = 8 + static let cornerRadius: CGFloat = 12 + + enum Size { + static let imageWidth: CGFloat = 60 + static let imageHeight: CGFloat = 60 + static let buttonWidth: CGFloat = 77 + static let buttonHeight: CGFloat = 77 + } + + enum Offset { + static let pressedX: CGFloat = 2 + static let pressedY: CGFloat = 2 + static let shadowX: CGFloat = 3 + static let shadowY: CGFloat = 3 + } +} + +struct LanguageButton: View { + let languageType: LanguageType + let action: () -> Void + + @State private var isPressed: Bool = false + + var body: some View { + Button(action: action) { + ZStack(alignment: .bottomTrailing) { + VStack(spacing: Constant.vStackSpacing) { + ZStack { + // 그림자 + RoundedRectangle(cornerRadius: + Constant.cornerRadius) + .fill(AppColors.gray600) + .frame( + width: Constant.Size.imageWidth, + height: Constant.Size.imageHeight + ) + .offset( + x: Constant.Offset.shadowX, + y: Constant.Offset.shadowY + ) + + // 실제 이미지 + Image(languageType.imageName) + .resizable() + .scaledToFit() + .frame( + width: Constant.Size.imageWidth, + height: Constant.Size.imageHeight + ) + .offset( + x: isPressed ? Constant.Offset.pressedX : 0, + y: isPressed ? Constant.Offset.pressedY : 0 + ) + } + + Text(languageType.rawValue) + .textStyle(.caption) + } + .frame(width: Constant.Size.buttonWidth, height: Constant.Size.buttonHeight) + + .frame(width: Constant.Size.buttonWidth, height: Constant.Size.buttonHeight) + } + .animation(.none, value: isPressed) + } + .buttonStyle(.pressable(isPressed: $isPressed)) + } +} + +#Preview { + HStack(spacing: 20) { + LanguageButton(languageType: .swift) { + print("Swift 버튼이 눌렸습니다") + } + LanguageButton(languageType: .kotlin) { + print("Kotlin 버튼이 눌렸습니다") + } + LanguageButton(languageType: .dart) { + print("Dart 버튼이 눌렸습니다") + } + LanguageButton(languageType: .python) { + print("Python 버튼이 눌렸습니다") + } + } + .padding() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LanguageItem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LanguageItem.swift new file mode 100644 index 00000000..c06757de --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LanguageItem.swift @@ -0,0 +1,104 @@ +// +// LanguageItem.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/12/26. +// + +import SwiftUI + +private enum Constant { + static let vStackSpacing: CGFloat = 15 + + enum IconSize { + static let normal: CGFloat = 44 + static let active: CGFloat = 60 + } + + enum Opacity { + static let completed: Double = 0.5 + static let normal: Double = 1.0 + static let empty: Double = 0.0 + } +} + +struct LanguageItem: View { + let languageType: LanguageType + let state: LanguageItemState + + var body: some View { + VStack(alignment: .center, spacing: Constant.vStackSpacing) { + ZStack { + if state != .empty { + Image(languageType.imageName) + .resizable() + .scaledToFit() + .opacity(opacity) + } + } + .frame(width: iconSize, height: iconSize) + + Text(languageType.rawValue) + .textStyle(textStyle) + .fixedSize(horizontal: true, vertical: false) + .opacity(opacity) + } + } + + // MARK: - Computed Properties + + private var iconSize: CGFloat { + switch state { + case .completed, .upcoming, .empty: + return Constant.IconSize.normal + case .active: + return Constant.IconSize.active + } + } + + private var opacity: Double { + switch state { + case .completed: + return Constant.Opacity.completed + case .active, .upcoming: + return Constant.Opacity.normal + case .empty: + return Constant.Opacity.empty + } + } + + private var textStyle: TypographyStyle { + switch state { + case .completed, .upcoming, .empty: + return .caption + case .active: + return .title3 + } + } +} + +#Preview { + VStack(spacing: 20) { + HStack(spacing: 20) { + LanguageItem(languageType: .swift, state: .upcoming) + LanguageItem(languageType: .kotlin, state: .upcoming) + LanguageItem(languageType: .dart, state: .upcoming) + LanguageItem(languageType: .python, state: .upcoming) + } + + HStack(spacing: 20) { + LanguageItem(languageType: .swift, state: .active) + LanguageItem(languageType: .kotlin, state: .active) + LanguageItem(languageType: .dart, state: .active) + LanguageItem(languageType: .python, state: .active) + } + + HStack(spacing: 20) { + LanguageItem(languageType: .swift, state: .completed) + LanguageItem(languageType: .kotlin, state: .completed) + LanguageItem(languageType: .dart, state: .completed) + LanguageItem(languageType: .python, state: .completed) + } + } + .padding() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LargeButton.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LargeButton.swift new file mode 100644 index 00000000..f02f6d68 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/LargeButton.swift @@ -0,0 +1,82 @@ +// +// LargeButton.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-14. +// + +import SwiftUI + +private enum Constant { + static let radius: CGFloat = 8 + + enum Size { + static let buttonWidth: CGFloat = 170 + static let buttonHeight: CGFloat = 46 + static let badgeWidth: CGFloat = 30 + static let badgeHeight: CGFloat = 30 + } + + enum Offset { + static let badgeOffsetX: CGFloat = 17 + static let badgeOffsetY: CGFloat = -15 + } + + enum Opacity { + static let pressed: Double = 0.8 + static let unPressed: Double = 1.0 + } +} + +struct LargeButton: View { + @State private var isPressed: Bool = false + + let title: String + var hasBadge: Bool = false + var isEnabled: Bool = true + let action: () -> Void + + var body: some View { + Button(action: action) { + Text(title) + .textStyle(.body) + .foregroundColor(.white) + .frame(width: Constant.Size.buttonWidth, height: Constant.Size.buttonHeight) + .background(isEnabled ? .orange500 : .beige400) + .cornerRadius(Constant.radius) + .opacity(isPressed ? Constant.Opacity.pressed : Constant.Opacity.unPressed) + .overlay(alignment: .topTrailing) { + if hasBadge { badge } + } + } + .disabled(!isEnabled) + .buttonStyle(.pressable(isPressed: $isPressed)) + } + + private var badge: some View { + Image(.iconDiamondPlus) + .resizable() + .frame(width: Constant.Size.badgeWidth, height: Constant.Size.badgeHeight) + .offset(x: Constant.Offset.badgeOffsetX, y: Constant.Offset.badgeOffsetY) + } +} + +#Preview { + VStack(spacing: 20) { + LargeButton(title: "버튼 텍스트") { + print("버튼 클릭1") + } + + LargeButton(title: "버튼 텍스트", isEnabled: false) { + print("버튼 클릭2") + } + + LargeButton(title: "버튼 텍스트", hasBadge: true, isEnabled: false) { + print("버튼 클릭3") + } + + LargeButton(title: "버튼 텍스트", hasBadge: true) { + print("버튼 클릭4") + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MediumButton.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MediumButton.swift new file mode 100644 index 00000000..100ab2ca --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MediumButton.swift @@ -0,0 +1,113 @@ +// +// MediumButton.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-14. +// + +import SwiftUI + +private enum Constant { + static let radius: CGFloat = 8 + + enum Size { + static let buttonWidth: CGFloat = 89 + static let buttonHeight: CGFloat = 44 + static let badgeWidth: CGFloat = 30 + static let badgeHeight: CGFloat = 30 + } + + enum Offset { + static let badgeOffsetX: CGFloat = 17 + static let badgeOffsetY: CGFloat = -15 + } + + enum Opacity { + static let pressed: Double = 0.8 + static let unPressed: Double = 1.0 + } +} + +struct MediumButton: View { + @State private var isPressed: Bool = false + + let title: String + var isFilled: Bool = false + var hasBadge: Bool = false + var isEnabled: Bool = true + var isCancelButton: Bool = false + let action: () -> Void + + var body: some View { + Button(action: action) { + Text(title) + .textStyle(.caption) + .foregroundColor(isFilled ? .white : .black) + .frame(width: Constant.Size.buttonWidth, height: Constant.Size.buttonHeight) + .background(backgroundColor) + .cornerRadius(Constant.radius) + .opacity(isPressed ? Constant.Opacity.pressed : Constant.Opacity.unPressed) + .overlay { + if !isFilled { + RoundedRectangle(cornerRadius: Constant.radius) + .stroke() + } + } + .overlay(alignment: .topTrailing) { + if hasBadge { badge } + } + } + .disabled(!isEnabled) + .buttonStyle(.pressable(isPressed: $isPressed)) + } + + private var backgroundColor: Color { + if !isEnabled || isCancelButton { + return isFilled ? .gray200 : .white + } + return isFilled ? .orange500 : .white + } + + private var badge: some View { + Image(.iconDiamondPlus) + .resizable() + .frame(width: Constant.Size.badgeWidth, height: Constant.Size.badgeHeight) + .offset(x: Constant.Offset.badgeOffsetX, y: Constant.Offset.badgeOffsetY) + } +} + +#Preview { + VStack(spacing: 20) { + MediumButton(title: "버튼 텍스트", isFilled: true) { + print("버튼 클릭1") + } + + MediumButton(title: "버튼 텍스트", isFilled: true, isEnabled: false) { + print("버튼 클릭2") + } + + MediumButton(title: "버튼 텍스트", isFilled: true, hasBadge: true) { + print("버튼 클릭3") + } + + MediumButton(title: "버튼 텍스트", isFilled: true, hasBadge: true, isEnabled: false) { + print("버튼 클릭4") + } + + MediumButton(title: "버튼 텍스트", isFilled: false) { + print("버튼 클릭5") + } + + MediumButton(title: "버튼 텍스트", isFilled: false, hasBadge: true) { + print("버튼 클릭6") + } + + MediumButton(title: "취소", isFilled: true, isCancelButton: true) { + print("취소 버튼 클릭") + } + + MediumButton(title: "취소", isFilled: false, isCancelButton: true) { + print("취소 버튼 클릭") + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MissionCard.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MissionCard.swift new file mode 100644 index 00000000..10e3a81e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MissionCard.swift @@ -0,0 +1,169 @@ +// +// MissionCard.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/15/26. +// + +import SwiftUI + +private enum Constant { + static let cardColor = AppColors.beige100 + + enum Size { + static let icon: CGFloat = 16 + static let cardHeight: CGFloat = 180 + static let imageHeight: CGFloat = 72 + static let progressHeight: CGFloat = 14 + } + + enum Padding { + static let content: CGFloat = 10 + } + + enum Spacing { + static let rewardIconText: CGFloat = 2 + static let vertical: CGFloat = 4 + } + + enum Typography { + static let titleLineLimit: Int = 1 + static let conditionLineLimit: Int = 2 + } +} + +struct MissionCard: View { + let title: String + let reward: Cost + let imageName: String + let condition: String + let buttonState: MissionCardButton.ButtonState + var onButtonTap: (() -> Void)? + + init( + title: String, + reward: Cost, + imageName: String, + condition: String, + buttonState: MissionCardButton.ButtonState, + onButtonTap: (() -> Void)? = nil + ) { + self.title = title + self.reward = reward + self.imageName = imageName + self.condition = condition + self.buttonState = buttonState + self.onButtonTap = onButtonTap + } + + var body: some View { + MissionCardContentView( + title: title, + reward: reward, + imageName: imageName, + condition: condition, + buttonState: buttonState, + onButtonTap: onButtonTap + ) + .padding(.all, Constant.Padding.content) + .frame(height: Constant.Size.cardHeight) + .background { Rectangle().fill(Constant.cardColor) } + .onTapGesture { + onButtonTap?() + } + } +} + +private struct MissionCardContentView: View { + let title: String + let reward: Cost + let imageName: String + let condition: String + let buttonState: MissionCardButton.ButtonState + var onButtonTap: (() -> Void)? + + var body: some View { + VStack(alignment: .center, spacing: Constant.Spacing.vertical) { + // 타이틀과 보상 (왼쪽 정렬) + VStack(alignment: .leading, spacing: Constant.Spacing.vertical) { + Text(title) + .textStyle(.subheadline) + .foregroundStyle(.black) + .lineLimit(Constant.Typography.titleLineLimit) + HStack(spacing: Constant.Spacing.rewardIconText) { + if reward.gold > 0 { + CurrencyLabel( + axis: .horizontal, + icon: .gold, + textStyle: .caption, + value: reward.gold + ) + .foregroundStyle(.black) + } + if reward.diamond > 0 { + CurrencyLabel( + axis: .horizontal, + icon: .diamond, + textStyle: .caption, + value: reward.diamond + ) + .foregroundStyle(.black) + } + } + } + .frame(maxWidth: .infinity, alignment: .leading) + // 이미지 (가운데 정렬) + Image(imageName) + .resizable() + .scaledToFill() + .frame(height: Constant.Size.imageHeight) + .frame(maxWidth: .infinity) + .clipped() + // 보상 조건 (가운데 정렬) + Text(condition) + .textStyle(.label) + .foregroundStyle(.black) + .frame(maxWidth: .infinity) + .frame(height: 30) + .lineLimit(Constant.Typography.conditionLineLimit) + .multilineTextAlignment(.center) + + // MissionCardButton (가운데 정렬) + MissionCardButton(buttonState: buttonState) { + onButtonTap?() + } + } + } +} + +#Preview { + HStack(spacing: 12) { + MissionCard( + title: "탭따구리", + reward: .init(gold: 15), + imageName: "mission_trophy_gold", + condition: "탭 10,000회 달성", + buttonState: .claimable, + onButtonTap: { + print("미션 1 보상 획득") + } + ) + + MissionCard( + title: "정상이라는 착각", + reward: .init(diamond: 15), + imageName: "mission_trophy_silver", + condition: "버그 피하기 1000회달성", + buttonState: .claimed + ) + + MissionCard( + title: "명탐정의 돋보기", + reward: .init(gold: 15), + imageName: "mission_trophy_bronze", + condition: "탭 10,000회 달성", + buttonState: .inProgress(currentValue: 7356, totalValue: 10000) + ) + } + .padding() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MissionCardButton.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MissionCardButton.swift new file mode 100644 index 00000000..a06a6947 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/MissionCardButton.swift @@ -0,0 +1,139 @@ +// +// MissionCardButton.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/15/26. +// + +import SwiftUI + +private enum Constant { + static let buttonHeight: CGFloat = 16 + static let buttonWidth: CGFloat = .infinity + + enum Title { + static let claimable = "획득하기" + static let claimed = "달성 완료" + } + + enum BackgroundColor { + static let claimable = AppColors.accentGreen + static let claimed = AppColors.gray100 + static let progressTotal = Color.white + static let progressCurrent = AppColors.orange200 + } + + enum ForegroundColor { + static let claimable = Color.white + static let claimed = Color.black + static let progress = Color.black + } +} + +struct MissionCardButton: View { + let buttonState: ButtonState + var onTap: (() -> Void)? + + @State private var isPressed: Bool = false + + var body: some View { + Button { + onTap?() + } label: { + ZStack { + if case .inProgress(let currentValue, let totalValue) = buttonState { + Rectangle().fill(buttonState.backgroundColor) + GeometryReader { geometry in + let progress = min(Double(currentValue) / Double(totalValue), 1.0) + Rectangle() + .fill(Constant.BackgroundColor.progressCurrent) + .frame(width: geometry.size.width * progress) + } + } else { + Rectangle().fill(buttonState.backgroundColor) + } + + Text(buttonState.title) + .textStyle(.label) + .foregroundStyle(buttonState.foregroundColor) + } + .frame(height: Constant.buttonHeight) + .frame(maxWidth: Constant.buttonWidth) + .animation(.none, value: buttonState) + } + .disabled(!buttonState.isEnabled) + .buttonStyle(.pressable(isPressed: $isPressed)) + } +} + +// MARK: - ButtonState +extension MissionCardButton { + enum ButtonState: Equatable { + case claimable + case claimed + case inProgress(currentValue: Int, totalValue: Int) + + var title: String { + switch self { + case .claimable: + return Constant.Title.claimable + case .claimed: + return Constant.Title.claimed + case .inProgress(let currentValue, let totalValue): + let percent = Double(currentValue) / Double(totalValue) * 100 + return String(format: "%.2f%%", percent) + } + } + + var backgroundColor: Color { + switch self { + case .claimable: + return Constant.BackgroundColor.claimable + case .claimed: + return Constant.BackgroundColor.claimed + case .inProgress: + return Constant.BackgroundColor.progressTotal + } + } + + var foregroundColor: Color { + switch self { + case .claimable: + return Constant.ForegroundColor.claimable + case .claimed: + return Constant.ForegroundColor.claimed + case .inProgress: + return Constant.ForegroundColor.progress + } + } + + var isEnabled: Bool { + switch self { + case .claimable: + return true + case .claimed, .inProgress: + return false + } + } + } +} + +#Preview { + ZStack { + Rectangle() + .fill(AppColors.gray200) + VStack { + MissionCardButton( + buttonState: .inProgress(currentValue: 19, totalValue: 2000) + ) { + print("획득하기 버튼 클릭") + } + MissionCardButton(buttonState: .claimable) { + print("획득하기 버튼 클릭") + } + MissionCardButton(buttonState: .claimed) + } + .padding(.horizontal, 16) + } + .padding() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/Popup.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/Popup.swift new file mode 100644 index 00000000..f3448ef9 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/Popup.swift @@ -0,0 +1,135 @@ +// +// Popup.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/16/26. +// + +import SwiftUI + +private enum Constant { + static let cornerRadius: CGFloat = 8 + static let lineWidth: CGFloat = 2 + static let contentVerticalSpacing: CGFloat = 11 + + enum Padding { + static let titleTop: CGFloat = 20 + static let contentBottom: CGFloat = 20 + } +} + +struct PopupConfiguration { + let title: String + let maxHeight: CGFloat? + let content: AnyView + + init( + title: String, + maxHeight: CGFloat? = nil, + @ViewBuilder content: () -> some View + ) { + self.title = title + self.maxHeight = maxHeight + self.content = AnyView(content()) + } +} + +struct Popup: View { + let title: String + let contentView: ContentView + + // ViewBuilder를 통한 클로저 방식 + init( + title: String, + @ViewBuilder contentView: () -> ContentView + ) { + self.title = title + self.contentView = contentView() + } + + // View 인스턴스를 직접 전달하는 방식 + init( + title: String, + contentView: ContentView + ) { + self.title = title + self.contentView = contentView + } + + var body: some View { + VStack(spacing: Constant.contentVerticalSpacing) { + Text(title) + .textStyle(.title3) + .multilineTextAlignment(.center) + .frame(maxWidth: .infinity, alignment: .center) + .padding(.top, Constant.Padding.titleTop) + + contentView + .padding(.bottom, Constant.Padding.contentBottom) + } + .background(Color.white) + .cornerRadius(Constant.cornerRadius) + .overlay { + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .stroke(Color.black, lineWidth: Constant.lineWidth) + } + } +} + +#Preview { + VStack(spacing: 30) { + // ViewBuilder를 통한 클로저 방식 + Popup(title: "튜토리얼") { + VStack(alignment: .leading, spacing: 10) { + Text("당신은 취직에 실패한 개발자. 이대로 물러설 수는 없다. 나의 꿈은 1인 개발자로 성공하기 ~! 내 이름은!!") + .textStyle(.body) + .foregroundColor(.black) + + TextField("닉네임", text: .constant("")) + .textFieldStyle(.roundedBorder) + + HStack(spacing: 10) { + Spacer() + MediumButton(title: "바로 시작", isFilled: false) { + print("바로 시작") + } + + MediumButton(title: "튜토리얼", isFilled: true) { + print("튜토리얼") + } + Spacer() + } + } + } + + // View 인스턴스를 직접 전달하는 방식 + let customContentView = VStack(alignment: .leading, spacing: 10) { + Text("팝업 내용 내용 내용 내용 내용 내용 내용 내용 내용 내용 내용") + .textStyle(.body) + .foregroundColor(.black) + + HStack(spacing: 10) { + Spacer() + MediumButton(title: "취소", isFilled: false) { + print("취소") + } + + MediumButton(title: "확인", isFilled: true) { + print("확인") + } + Spacer() + } + } + Popup(title: "팝업 타이틀", contentView: customContentView) + + // 간단한 텍스트만 있는 팝업 + Popup(title: "알림") { + Text("간단한 메시지 팝업입니다.") + .textStyle(.body) + .foregroundColor(.black) + .frame(maxWidth: .infinity, alignment: .center) + } + } + .padding() + .background(Color.gray.opacity(0.1)) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/PriceButton.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/PriceButton.swift new file mode 100644 index 00000000..2ed3d5c1 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/PriceButton.swift @@ -0,0 +1,210 @@ +// +// PriceButton.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/14/26. +// + +import SwiftUI + +private enum Constant { + enum Layout { + static let cornerRadius: CGFloat = 5 + static let buttonHeight: CGFloat = 44 + static let contentSpacing: CGFloat = 3 + static let horizontalPadding: CGFloat = 8 + } + + enum Shadow { + static let offsetX: CGFloat = 2 + static let offsetY: CGFloat = 3 + } + + enum Icon { + static let coinSize = CGSize(width: 15, height: 15) + static let lockSize = CGSize(width: 18, height: 18) + } + + enum Overlay { + static let disabledOpacity: Double = 0.4 + } +} + +struct PriceButton: View { + + @GestureState private var isPressed: Bool = false + @State private var isLongPressing: Bool = false + + let cost: Cost + let state: ItemState + let axis: Axis + let width: CGFloat? + let action: () -> Void + let onLongPressRepeat: (() -> Bool)? + + init( + cost: Cost, + state: ItemState, + axis: Axis, + width: CGFloat? = nil, + action: @escaping () -> Void, + onLongPressRepeat: (() -> Bool)? = nil + ) { + self.cost = cost + self.state = state + self.axis = axis + self.width = width + self.action = action + self.onLongPressRepeat = onLongPressRepeat + } + + private var isDisabled: Bool { + state != .available + } + + var body: some View { + buttonContent + .overlay { + if isDisabled { + disabledOverlay + } + } + .animation(.none, value: isDisabled) + .background( + RoundedRectangle(cornerRadius: Constant.Layout.cornerRadius) + .foregroundStyle(isDisabled ? .gray400 : .black) + .offset( + x: Constant.Shadow.offsetX, + y: Constant.Shadow.offsetY + ) + ) + .animation(.none, value: isDisabled) + .contentShape(Rectangle()) + .longPressRepeat( + isLongPressing: $isLongPressing, + isDisabled: isDisabled, + onLongPressRepeat: onLongPressRepeat + ) + .onTapGesture { + if isLongPressing { + isLongPressing = false + return + } + if !isDisabled { + SoundService.shared.trigger(.buttonTap) + action() + } + } + .simultaneousGesture( + DragGesture(minimumDistance: 0) + .updating($isPressed) { _, state, _ in + if !isDisabled { + state = true + } + } + ) + .animation(.none, value: isPressed) + } + + @ViewBuilder + var buttonContent: some View { + Group { + if axis == .horizontal { + HStack(spacing: Constant.Layout.contentSpacing) { + contentViews + } + } else { + VStack(spacing: Constant.Layout.contentSpacing) { + contentViews + } + } + } + .frame(width: width ?? .none, height: Constant.Layout.buttonHeight) + .padding(.horizontal, Constant.Layout.horizontalPadding) + .background(isDisabled ? .gray300 : .orange500) + .clipShape(RoundedRectangle(cornerRadius: Constant.Layout.cornerRadius)) + .offset( + x: (isPressed && !isDisabled) ? Constant.Shadow.offsetX : 0, + y: (isPressed && !isDisabled) ? Constant.Shadow.offsetY : 0 + ) + .animation(.none, value: cost) + .animation(.none, value: state) + } + + @ViewBuilder + var contentViews: some View { + Group { + if state == .reachedMax { + Text("Max") + .textStyle(.caption) + .foregroundStyle(.white) + } else { + if cost.gold > 0 { + CurrencyLabel( + axis: .horizontal, + icon: .gold, + textStyle: .caption, + value: cost.gold + ) + .foregroundStyle(.white) + .fixedSize() + } + if cost.diamond > 0 { + CurrencyLabel( + axis: .horizontal, + icon: .diamond, + textStyle: .caption, + value: cost.diamond + ) + .foregroundStyle(.white) + .fixedSize() + } + } + } + .fixedSize() + .drawingGroup() + .minimumScaleFactor(0.7) + .lineLimit(1) + } + + var disabledOverlay: some View { + ZStack { + RoundedRectangle(cornerRadius: Constant.Layout.cornerRadius) + .foregroundStyle(Color.gray200.opacity(Constant.Overlay.disabledOpacity)) + + if state == .locked { + Image(.iconLock) + .resizable() + .frame( + width: Constant.Icon.lockSize.width, + height: Constant.Icon.lockSize.height + ) + } + } + } +} + +#Preview { + VStack(alignment: .center, spacing: 20) { + PriceButton( + cost: .init(gold: 1_000_000), + state: .available, + axis: .horizontal + ) {} + PriceButton( + cost: .init(gold: 100_000, diamond: 50), + state: .available, + axis: .vertical + ) {} + PriceButton( + cost: .init(gold: 1_000_000_000), + state: .locked, + axis: .horizontal + ) {} + PriceButton( + cost: .init(diamond: 1_000), + state: .insufficient, + axis: .vertical + ) {} + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/ProgressBar.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/ProgressBar.swift new file mode 100644 index 00000000..69e2ffbc --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/ProgressBar.swift @@ -0,0 +1,48 @@ +// +// ProgressBar.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-14. +// + +import SwiftUI + +private enum Constants { + static let height: CGFloat = 20 +} + +struct ProgressBar: View { + let maxValue: Double + let currentValue: Double + let text: String + + private var progress: Double { + return currentValue / maxValue + } + + var body: some View { + ZStack { + // 배경 + Rectangle() + .fill(.beige300) + + // 진행 바 + Rectangle() + .fill(.orange300) + .scaleEffect(x: progress, y: 1, anchor: .leading) + + Text(text) + .textStyle(.caption) + .foregroundColor(.black) + } + .frame(height: Constants.height) + } +} + +#Preview { + ProgressBar(maxValue: 60, currentValue: 10, text: "10 s") + ProgressBar(maxValue: 60, currentValue: 30, text: "30 s") + ProgressBar(maxValue: 60, currentValue: 40, text: "40 s") + ProgressBar(maxValue: 60, currentValue: 59, text: "59 s") + ProgressBar(maxValue: 60, currentValue: 60, text: "제한 시간 종료") +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/QuizButton.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/QuizButton.swift new file mode 100644 index 00000000..b8387b91 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/QuizButton.swift @@ -0,0 +1,172 @@ +// +// QuizButton.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/15/26. +// + +import SwiftUI + +private enum Constant { + static let height: CGFloat = 56 + static let cornerRadius: CGFloat = 10 + + enum Color { + static let foreground = SwiftUI.Color.white + static let selectedBackground = AppColors.lightOrange + static let defaultBackground = AppColors.gray700 + static let disabledBackground = AppColors.gray200 + } +} + +struct QuizButton: View { + @State private var isPressed: Bool = false + + let style: QuizButtonStyle + let isSelected: Bool + let isEnabled: Bool + let title: String + let action: () -> Void + + init( + style: QuizButtonStyle = .option, + isSelected: Bool = false, + isEnabled: Bool = true, + title: String, + action: @escaping () -> Void + ) { + self.style = style + self.isSelected = isSelected + self.isEnabled = isEnabled + self.title = title + self.action = action + } + + var body: some View { + Button { + if isEnabled { + action() + } + } label: { + Text(title) + .textStyle(textStyle) + .foregroundStyle(Constant.Color.foreground) + .padding(.leading, textLeadingPadding) + .frame(maxWidth: .infinity, alignment: textAlignment) + .frame(height: Constant.height) + .background(backgroundColor) + .cornerRadius(Constant.cornerRadius) + .animation(.none, value: isSelected) + .animation(.none, value: isEnabled) + } + .buttonStyle(.pressable(isPressed: $isPressed)) + .disabled(!isEnabled) + } +} + +// MARK: - Helper +extension QuizButton { + enum QuizButtonStyle { + case option // 선택지 버튼용 + case submit // 제출 버튼용 + } + + /// 버튼 텍스트 왼쪽 여백 + private var textLeadingPadding: CGFloat { + switch style { + case .option: + return 10 + case .submit: + return 0 + } + } + + /// 버튼 정렬 + private var textAlignment: Alignment { + switch style { + case .option: + return .leading + case .submit: + return .center + } + } + + /// 버튼 배경색 + private var backgroundColor: SwiftUI.Color { + guard isEnabled else { + return Constant.Color.disabledBackground + } + switch style { + case .option: + return isSelected ? Constant.Color.selectedBackground : Constant.Color.defaultBackground + case .submit: + return Constant.Color.defaultBackground + } + } + + /// 버튼 폰트 스타일 + var textStyle: TypographyStyle { + switch style { + case .option: + return .subheadline + case .submit: + return .body + } + } +} + +#Preview { + struct PreviewWrapper: View { + @State private var selectedIndex: Int? + + var body: some View { + VStack(spacing: 20) { + // 선택지 버튼들 (토글 가능) + QuizButton( + style: .option, + isSelected: selectedIndex == 0, + title: "1. (함께 코드를 보며) 이 부분 빨리 수정 가능할까요?" + ) { + // 토글: 이미 선택된 버튼이면 해제, 아니면 선택 + selectedIndex = selectedIndex == 0 ? nil : 0 + } + + QuizButton( + style: .option, + isSelected: selectedIndex == 1, + title: "2. (함께 코드를 보며) 이 부분 빨리 수정 가능할까요?" + ) { + selectedIndex = selectedIndex == 1 ? nil : 1 + } + + QuizButton( + style: .option, + isSelected: selectedIndex == 2, + title: "3. (함께 코드를 보며) 이 부분 빨리 수정 가능할까요?" + ) { + selectedIndex = selectedIndex == 2 ? nil : 2 + } + + QuizButton( + style: .option, + isSelected: selectedIndex == 3, + title: "4. (함께 코드를 보며) 이 부분 빨리 수정 가능할까요?" + ) { + selectedIndex = selectedIndex == 3 ? nil : 3 + } + + // 제출 버튼 + QuizButton( + style: .submit, + isEnabled: selectedIndex != nil, + title: "제출하기" + ) { + print("제출됨: \(selectedIndex ?? -1)") + } + } + .padding() + } + } + + return PreviewWrapper() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/RunningCharacter.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/RunningCharacter.swift new file mode 100644 index 00000000..a7d08f36 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/RunningCharacter.swift @@ -0,0 +1,88 @@ +// +// RunningCharacter.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-15. +// + +import SwiftUI + +private enum Constant { + static let animationSpeed: TimeInterval = 0.15 +} + +struct RunningCharacter: View { + @State private var currentFrame = 0 + @State private var animationTimer: Timer? + + private let frameImages: [ImageResource] = [ + .dodgeCharacter1, + .dodgeCharacter2, + .dodgeCharacter3, + .dodgeCharacter2 + ] + + let isFacingLeft: Bool + let isGamePaused: Bool + + /// 달리기 애니메이션 캐릭터 초기화 + /// - Parameters: + /// - isFacingLeft: 왼쪽을 향할지 여부 (기본값: false, 오른쪽 향함) + init(isFacingLeft: Bool = false, isGamePaused: Bool) { + self.isFacingLeft = isFacingLeft + self.isGamePaused = isGamePaused + } + + var body: some View { + Image(frameImages[currentFrame]) + .resizable() + .aspectRatio(contentMode: .fit) + .scaleEffect(x: isFacingLeft ? -1 : 1, y: 1) + .onAppear { + handleAnimation() + } + .onDisappear { + stopAnimation() + } + .onChange(of: isGamePaused) { _, _ in + handleAnimation() + } + } + + private func handleAnimation() { + if isGamePaused { + stopAnimation() + } else { + startAnimation() + } + } + + private func startAnimation() { + animationTimer = Timer.scheduledTimer(withTimeInterval: Constant.animationSpeed, repeats: true) { _ in + currentFrame = (currentFrame + 1) % frameImages.count + } + } + + private func stopAnimation() { + animationTimer?.invalidate() + animationTimer = nil + } +} + +#Preview { + HStack(spacing: 40) { + VStack { + RunningCharacter(isFacingLeft: false, isGamePaused: false) + .frame(width: 40, height: 40) + Text("오른쪽 →") + .font(.caption) + } + + VStack { + RunningCharacter(isFacingLeft: true, isGamePaused: false) + .frame(width: 40, height: 40) + Text("← 왼쪽") + .font(.caption) + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/SettingSlider.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/SettingSlider.swift new file mode 100644 index 00000000..1dc0e329 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/SettingSlider.swift @@ -0,0 +1,119 @@ +// +// SettingSlider.swift +// SoloDeveloperTraining +// + +import SwiftUI + +private enum Constant { + static let thumbDiameter: CGFloat = 24 + static let thumbRadius: CGFloat = thumbDiameter / 2 + static let trackHeight: CGFloat = 12 + static let cornerRadius: CGFloat = 4 + static let strokeLineWidth: CGFloat = 1 + static let strokeOpacityEnabled: Double = 0.15 + static let strokeOpacityDisabled: Double = 0.08 + static let dragMinimumDistance: CGFloat = 0 +} + +struct SettingSlider: View { + @Binding var value: Double + var range: ClosedRange = 0 ... 100 + var step: Double = 1 + var isEnabled: Bool = true + + private var progress: Double { + let span = range.upperBound - range.lowerBound + guard span > 0 else { return 0 } + return max(0, min(1, (value - range.lowerBound) / span)) + } + + var body: some View { + GeometryReader { geometry in + let width = geometry.size.width + let thumbCenterX = width * progress + + ZStack(alignment: .leading) { + // 트랙 배경 + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .fill(trackBackgroundColor) + .frame(height: Constant.trackHeight) + + // 진행 + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .fill(fillColor) + .frame(width: max(0, thumbCenterX), height: Constant.trackHeight) + + // 썸 + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .fill(thumbColor) + .frame(width: Constant.thumbDiameter, height: Constant.thumbDiameter) + .overlay { + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .stroke( + Color.black.opacity(isEnabled ? Constant.strokeOpacityEnabled : Constant.strokeOpacityDisabled), + lineWidth: Constant.strokeLineWidth + ) + } + .offset(x: thumbCenterX - Constant.thumbRadius) + } + .frame(height: Constant.thumbDiameter) + .frame(maxWidth: .infinity) + .contentShape(Rectangle()) + .allowsHitTesting(isEnabled) + .gesture( + DragGesture(minimumDistance: Constant.dragMinimumDistance) + .onChanged { gesture in + guard isEnabled else { return } + applyValue(from: gesture.location.x, trackWidth: width) + } + ) + .onTapGesture { location in + guard isEnabled else { return } + applyValue(from: location.x, trackWidth: width) + } + } + .frame(height: Constant.thumbDiameter) + } +} + +private extension SettingSlider { + var trackBackgroundColor: Color { + AppColors.beige300 + } + + var fillColor: Color { + isEnabled ? AppColors.orange300 : AppColors.gray300 + } + + var thumbColor: Color { + isEnabled ? AppColors.orange500 : AppColors.gray400 + } + + func applyValue(from locationX: CGFloat, trackWidth: CGFloat) { + guard trackWidth > 0 else { return } + let progressRatio = max(0, min(1, locationX / trackWidth)) + var rawValue = range.lowerBound + progressRatio * (range.upperBound - range.lowerBound) + if step > 0 { + rawValue = (rawValue / step).rounded() * step + } + rawValue = max(range.lowerBound, min(range.upperBound, rawValue)) + value = rawValue + } +} + +#Preview { + struct PreviewHolder: View { + @State private var volume: Double = 60 + var body: some View { + VStack(spacing: 24) { + SettingSlider(value: $volume, range: 0 ... 100, step: 1) + .padding(.horizontal) + Text("\(Int(volume))") + .textStyle(.body) + } + .padding() + } + } + return PreviewHolder() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/SmallButton.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/SmallButton.swift new file mode 100644 index 00000000..0316aa02 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/SmallButton.swift @@ -0,0 +1,128 @@ +// +// SmallButton.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-14. +// + +import SwiftUI + +private enum Constant { + static let radius: CGFloat = 8 + + enum Size { + static let buttonWidth: CGFloat = 44 + static let buttonHeight: CGFloat = 44 + static let badgeWidth: CGFloat = 30 + static let badgeHeight: CGFloat = 30 + } + + enum Offset { + static let badgeOffsetX: CGFloat = 17 + static let badgeOffsetY: CGFloat = -15 + static let pressedX: CGFloat = 2 + static let pressedY: CGFloat = 3 + static let shadowX: CGFloat = 2 + static let shadowY: CGFloat = 3 + } + + enum Opacity { + static let pressed: Double = 0.8 + static let unPressed: Double = 1.0 + } +} + +struct SmallButton: View { + @State private var isPressed: Bool = false + + let title: String + var image: Image? + var hasBadge: Bool = false + var isEnabled: Bool = true + let action: () -> Void + + var body: some View { + Button(action: action) { + ZStack { + buttonContent + .opacity(isPressed ? Constant.Opacity.pressed : Constant.Opacity.unPressed) + .frame(width: Constant.Size.buttonWidth, height: Constant.Size.buttonHeight) + .background(backgroundColor) + .cornerRadius(Constant.radius) + .overlay(alignment: .topTrailing) { + if hasBadge { badge } + } + .offset( + x: isPressed ? Constant.Offset.pressedX : 0, + y: isPressed ? Constant.Offset.pressedY : 0 + ) + .layoutPriority(1) + + Rectangle() + .fill(Color.black) + .frame(width: Constant.Size.buttonWidth, height: Constant.Size.buttonHeight) + .cornerRadius(Constant.radius) + .offset(x: Constant.Offset.shadowX, y: Constant.Offset.shadowY) + .zIndex(-1) + } + } + .disabled(!isEnabled) + .buttonStyle(.pressable(isPressed: $isPressed)) + + } + + @ViewBuilder + private var buttonContent: some View { + if let image { + image + .resizable() + .scaledToFill() + .frame(width: Constant.Size.buttonWidth, height: Constant.Size.buttonHeight) + .contentShape(Rectangle()) + } else { + Text(title) + .textStyle(.body) + .foregroundColor(.white) + } + } + + private var backgroundColor: Color { + if image != nil { + return .clear + } + if !isEnabled { + return .gray200 + } + return hasBadge ? AppColors.lightOrange : .orange500 + } + + private var badge: some View { + Image(.iconDiamondPlus) + .resizable() + .frame(width: Constant.Size.badgeWidth, height: Constant.Size.badgeHeight) + .offset(x: Constant.Offset.badgeOffsetX, y: Constant.Offset.badgeOffsetY) + } +} + +#Preview { + VStack(spacing: 20) { + SmallButton(title: "버튼") { + print("버튼 클릭1") + } + + SmallButton(title: "버튼", isEnabled: false) { + print("버튼 클릭2") + } + + SmallButton(title: "버튼", hasBadge: true, isEnabled: false) { + print("버튼 클릭3") + } + + SmallButton(title: "버튼", hasBadge: true) { + print("버튼 클릭4") + } + SmallButton(title: "버튼", image: Image(.iconSetting)) { + print("버튼 클릭4") + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/StatusBar.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/StatusBar.swift new file mode 100644 index 00000000..779f69f5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/StatusBar.swift @@ -0,0 +1,84 @@ +// +// StatusBar.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/8/26. +// + +import SwiftUI + +struct StatusBar: View { + let career: Career + let nickname: String + let careerProgress: Double + let gold: Int + let diamond: Int + + var body: some View { + VStack { + HStack(spacing: 16) { + // 왼쪽: 프로필 + 커리어 + 진행바 + HStack(spacing: 6) { + Image(career.imageName) + .resizable() + .aspectRatio(contentMode: .fill) + .frame(width: 30, height: 30) + .clipShape(RoundedRectangle(cornerRadius: 4)) + + VStack(alignment: .leading, spacing: 4) { + Text("\(career.rawValue) \(nickname)") + .textStyle(.caption) + .foregroundStyle(.primary) + .lineLimit(1) + .minimumScaleFactor(0.5) + + ProgressBarView(progress: careerProgress) + } + } + + Spacer() + + // 오른쪽: 재산 + 다이아 + HStack(spacing: 12) { + CurrencyLabel(axis: .horizontal, icon: .gold, value: gold) + CurrencyLabel(axis: .horizontal, icon: .diamond, value: diamond) + } + } + .padding(.horizontal, 16) + .padding(.vertical, 24) + } + .frame(maxHeight: .infinity, alignment: .bottom) + .frame(height: 113) + .background(Color.white.opacity(0.8)) + } +} + +private struct ProgressBarView: View { + let progress: Double + let height: CGFloat = 10 + + var body: some View { + GeometryReader { geometry in + ZStack(alignment: .leading) { + RoundedRectangle(cornerRadius: 4) + .fill(Color.gray200) + .frame(height: height) + + RoundedRectangle(cornerRadius: 4) + .fill(Color.lightOrange) + .frame(width: geometry.size.width * progress, height: height) + } + } + .frame(width: 129, height: height) + } +} + +#Preview("Status Bar") { + StatusBar( + career: .laptopOwner, + nickname: "소피아", + careerProgress: 0.42, + gold: 1234, + diamond: 56 + ) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/TabBar.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/TabBar.swift new file mode 100644 index 00000000..6f38b10f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/TabBar.swift @@ -0,0 +1,133 @@ +// +// TabBarView.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/8/26. +// + +import SwiftUI + +private enum Constant { + enum Spacing { + static let hStack: CGFloat = 4 + static let vStack: CGFloat = 4 + } + + enum Padding { + static let horizontal: CGFloat = 16 + static let vertical: CGFloat = 10 + static let buttonVertical: CGFloat = 4.5 + static let badgeTrailing: CGFloat = 6 + static let badgeBottom: CGFloat = 40 + static let imageHeight: CGFloat = 24 + } + + enum Size { + static let imageWidth: CGFloat = 24 + static let imageHeight: CGFloat = 24 + static let badge: CGFloat = 14 + } + + enum Offset { + static let pressedX: CGFloat = 2 + static let pressedY: CGFloat = 3 + static let shadowX: CGFloat = 2 + static let shadowY: CGFloat = 3 + } +} + +enum TabItem: String, CaseIterable { + case work = "업무" + case skill = "스킬" + case shop = "상점" + case mission = "미션" + + var imageName: String { + switch self { + case .work: return "icon_work" + case .skill: return "icon_skill" + case .shop: return "icon_shop" + case .mission: return "icon_mission" + } + } +} + +struct TabBar: View { + private var hasCompletedMisson: Bool + @Binding var selectedTab: TabItem + + init(selectedTab: Binding, hasCompletedMisson: Bool) { + self._selectedTab = selectedTab + self.hasCompletedMisson = hasCompletedMisson + } + + var body: some View { + HStack(alignment: .center, spacing: Constant.Spacing.hStack) { + ForEach(TabItem.allCases, id: \.self) { tab in + TabBarButton( + tab: tab, + isSelected: selectedTab == tab, + hasBadge: tab == .mission && hasCompletedMisson + ) { + selectedTab = tab + } + } + } + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.vertical, Constant.Padding.vertical) + .background(AppColors.beige200) + } +} + +private struct TabBarButton: View { + let tab: TabItem + let isSelected: Bool + let hasBadge: Bool + let action: () -> Void + + @State private var isPressed: Bool = false + + var body: some View { + Button(action: action) { + ZStack(alignment: .bottomTrailing) { + VStack(spacing: Constant.Spacing.vStack) { + Image(tab.imageName) + .resizable() + .frame(width: Constant.Size.imageWidth, height: Constant.Size.imageHeight) + + Text(tab.rawValue) + .textStyle(.caption) + } + .padding(.vertical, Constant.Padding.buttonVertical) + .foregroundStyle(isSelected ? .white : AppColors.orange500) + .frame(maxWidth: .infinity) + .background( + Rectangle() + .fill(isSelected ? AppColors.orange300 : AppColors.beige300) + ) + .offset(x: isPressed ? Constant.Offset.pressedX : 0, y: isPressed ? Constant.Offset.pressedY : 0) + .layoutPriority(1) + + Rectangle() + .fill(Color.black) + .offset(x: Constant.Offset.shadowX, y: Constant.Offset.shadowY) + .zIndex(-1) + /// 미션 탭 배지 + if hasBadge { + Image(.iconNewBadge) + .resizable() + .frame(width: Constant.Size.badge, height: Constant.Size.badge) + .padding(.trailing, Constant.Padding.badgeTrailing) + .padding(.bottom, Constant.Padding.badgeBottom) + } + } + .animation(.none, value: isSelected) + } + .buttonStyle(.pressable(isPressed: $isPressed)) + } +} + +#Preview { + @Previewable @State var selectedTab: TabItem = .work + TabBar(selectedTab: $selectedTab, hasCompletedMisson: true) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/Toast.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/Toast.swift new file mode 100644 index 00000000..e34a42dd --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/Toast.swift @@ -0,0 +1,124 @@ +// +// Toast.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/22/26. +// + +import SwiftUI + +private enum Constant { + static let cornerRadius: CGFloat = 6 + + enum Padding { + static let horizontal: CGFloat = 16 + static let vertical: CGFloat = 10 + static let bottom: CGFloat = 20 + } + + enum Width { + static let maxRatio: CGFloat = 0.9 + } + + enum Shadow { + static let color: Color = AppColors.gray400 + static let radius: CGFloat = 4 + static let yOffset: CGFloat = 3 + } + + enum Animation { + static let showDuration: CGFloat = 0.3 + static let hideDuration: CGFloat = 0.3 + } +} + +struct Toast: ViewModifier { + @Binding var isShowing: Bool + let message: String + let duration: Double + + @State private var showContent: Bool = false + @State private var opacity: Double = 0 + + func body(content: Content) -> some View { + ZStack { + content + + if showContent { + VStack { + Spacer() + Text(message) + .textStyle(.callout) + .multilineTextAlignment(.center) + .lineLimit(nil) + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.vertical, Constant.Padding.vertical) + .background(AppColors.orange300) + .foregroundColor(.white) + .cornerRadius(Constant.cornerRadius) + .shadow(color: Constant.Shadow.color, + radius: Constant.Shadow.radius, + y: Constant.Shadow.yOffset) + .frame(maxWidth: UIScreen.main.bounds.width * Constant.Width.maxRatio) + .padding(.bottom, Constant.Padding.bottom) + .opacity(opacity) + } + } + } + .onChange(of: isShowing) { _, newValue in + if newValue { + // 토스트 나타날 때 + showContent = true + withAnimation(.easeOut(duration: Constant.Animation.showDuration)) { + opacity = 1 + } + + // duration 후 사라지게 함 + DispatchQueue.main.asyncAfter(deadline: .now() + duration) { + // 부드럽게 사라지는 애니메이션 + withAnimation(.easeIn(duration: Constant.Animation.hideDuration)) { + opacity = 0 + } + + // 애니메이션 완료 후 뷰 제거 및 바인딩 리셋 + DispatchQueue.main.asyncAfter(deadline: .now() + Constant.Animation.hideDuration) { + showContent = false + isShowing = false + } + } + } + } + } +} + +#Preview { + struct ToastPreview: View { + @State private var shortToast: Bool = false + @State private var longToast: Bool = false + + var body: some View { + VStack(spacing: 50) { + VStack(spacing: 10) { + Button("짧은 메시지 표시") { + shortToast = true + } + .withTapSound() + Text("짧은 메시지 테스트") + .toast(isShowing: $shortToast, message: "짧은 토스트 메시지입니다.") + } + + VStack(spacing: 10) { + Button("긴 메시지 표시") { + longToast = true + } + .withTapSound() + Text("긴 메시지 테스트") + .toast(isShowing: $longToast, message: "긴 토스트 메시지입니다. 이 메시지는 길어서 줄바꿈과 최대 너비가 적용되는지 확인하는 테스트용입니다.") + } + } + .padding() + } + } + + return ToastPreview() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/WorkItemButton.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/WorkItemButton.swift new file mode 100644 index 00000000..566b6a49 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/WorkItemButton.swift @@ -0,0 +1,192 @@ +// +// WorkItemButton.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/14/26. +// + +import SwiftUI + +private enum Constant { + static let buttonHeight: CGFloat = 218 + static let borderWidth: CGFloat = 2 + static let cornerRadius: CGFloat = 4 + static let lockIconSize = CGSize(width: 24, height: 24) + static let disabledOverlayOpacity: Double = 0.5 + + enum Padding { + static let titleTop: CGFloat = 22 + static let descriptionTop: CGFloat = 4 + static let imageTop: CGFloat = 20 + static let imageHorizontal: CGFloat = 11 + static let imageBottom: CGFloat = 10 + } +} + +struct WorkItemButton: View { + let title: String + let imageName: String + @Binding var buttonState: ButtonState + var onTap: (() -> Void)? + + var body: some View { + buttonContent + .disabled(buttonState.isDisabled) + .onTapGesture { + guard !buttonState.isDisabled else { return } + SoundService.shared.trigger(.buttonTap) + if let onTap = onTap { + onTap() + } else { + changeState() + } + } + } +} + +// MARK: - Subviews +private extension WorkItemButton { + + var buttonContent: some View { + ZStack(alignment: .top) { + backgroundShape + contentStack + } + .frame(height: Constant.buttonHeight) + } + + var backgroundShape: some View { + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .foregroundStyle(buttonState.backgroundColor) + .overlay { + if let borderColor = buttonState.borderColor { + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .stroke(borderColor, lineWidth: Constant.borderWidth) + } + } + } + + var contentStack: some View { + VStack(spacing: 0) { + titleLabel + itemImage + } + } + + var titleLabel: some View { + Text(title) + .foregroundStyle(.black) + .textStyle(.headline) + .padding(.top, Constant.Padding.titleTop) + } + + var itemImage: some View { + GeometryReader { geometry in + Image(imageName) + .resizable() + .aspectRatio(contentMode: .fill) + .frame(width: geometry.size.width, height: geometry.size.height) + .clipped() + .overlay { + RoundedRectangle(cornerRadius: Constant.cornerRadius) + .stroke(.black, lineWidth: Constant.borderWidth) + } + } + .padding(.top, Constant.Padding.imageTop) + .padding(.horizontal, Constant.Padding.imageHorizontal) + .padding(.bottom, Constant.Padding.imageBottom) + .overlay { + if buttonState == .disabled { + disabledOverlay + } + } + } + + var disabledOverlay: some View { + ZStack { + Color.black.opacity(Constant.disabledOverlayOpacity) + .padding(.top, Constant.Padding.imageTop) + .padding(.horizontal, Constant.Padding.imageHorizontal) + .padding(.bottom, Constant.Padding.imageBottom) + + Image(.iconLock) + .resizable() + .frame( + width: Constant.lockIconSize.width, + height: Constant.lockIconSize.height + ) + } + } +} + +// MARK: - Helper +private extension WorkItemButton { + func changeState() { + switch buttonState { + case .normal: + buttonState = .focused + case .focused: + buttonState = .normal + case .disabled: + break + } + } +} + +// MARK: - ButtonState +extension WorkItemButton { + enum ButtonState { + case focused + case normal + case disabled + + var backgroundColor: Color { + switch self { + case .focused, .normal: + return .beige100 + case .disabled: + return .beige400 + } + } + + var borderColor: Color? { + switch self { + case .focused: + return .black + default: + return nil + } + } + + var isDisabled: Bool { + return self == .disabled + } + } +} + +#Preview { + @Previewable @State var buttonState1: WorkItemButton.ButtonState = .normal + @Previewable @State var buttonState2: WorkItemButton.ButtonState = .focused + @Previewable @State var buttonState3: WorkItemButton.ButtonState = .disabled + + VStack(spacing: 20) { + HStack { + WorkItemButton( + title: "타이틀", + imageName: "housing_street", + buttonState: $buttonState1 + ) + WorkItemButton( + title: "타이틀", + imageName: "housing_street", + buttonState: $buttonState2 + ) + WorkItemButton( + title: "타이틀", + imageName: "housing_street", + buttonState: $buttonState3 + ) + } + } + .padding(.horizontal) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/WorkSegmentControl.swift b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/WorkSegmentControl.swift new file mode 100644 index 00000000..90a49d5e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/DesignSystem/Components/WorkSegmentControl.swift @@ -0,0 +1,117 @@ +// +// WorkSegmentControl.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/14/26. +// + +import SwiftUI + +private enum Constant { + static let itemSpacing: CGFloat = 2 +} + +struct WorkSegmentControl: View { + let items: [WorkItem] + var onLockedTap: ((Career) -> Void)? + @Binding var selectedIndex: Int + + var body: some View { + HStack(spacing: Constant.itemSpacing) { + ForEach(items.indices, id: \.self) { index in + WorkItemButton( + title: items[index].title, + imageName: items[index].imageName, + buttonState: .constant(buttonState(for: index)), + onTap: { + handleTap(at: index) + } + ) + .overlay { + if items[index].isDisabled { + Color.clear + .contentShape(Rectangle()) + .onTapGesture { + handleLockedTap(at: index) + } + } + } + } + } + } +} + +private extension WorkSegmentControl { + func handleTap(at index: Int) { + guard !items[index].isDisabled else { return } + selectedIndex = index + } + + func handleLockedTap(at index: Int) { + guard let requiredCareer = items[index].requiredCareer else { return } + onLockedTap?(requiredCareer) + } + + func buttonState(for index: Int) -> WorkItemButton.ButtonState { + if items[index].isDisabled { + return .disabled + } else if selectedIndex == index { + return .focused + } else { + return .normal + } + } +} + +struct WorkItem { + let title: String + let imageName: String + let isDisabled: Bool + let requiredCareer: Career? + + init( + title: String, + imageName: String, + isDisabled: Bool = false, + requiredCareer: Career? = nil + ) { + self.title = title + self.imageName = imageName + self.isDisabled = isDisabled + self.requiredCareer = requiredCareer + } +} + +#Preview { + @Previewable @State var selectedIndex: Int = 0 + + VStack(spacing: 20) { + Text("Selected Index: \(selectedIndex)") + .textStyle(.headline) + + WorkSegmentControl( + items: [ + WorkItem( + title: "테스트", + imageName: GameType.tap.imageName + ), + WorkItem( + title: "테스트", + imageName: GameType.tap.imageName + ), + WorkItem( + title: "테스트", + imageName: GameType.tap.imageName, + isDisabled: true + ), + WorkItem( + title: "테스트", + imageName: GameType.tap.imageName, + isDisabled: false + ) + ], + selectedIndex: $selectedIndex + ) + } + .padding() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/CheatSystem/CheatManager.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/CheatSystem/CheatManager.swift new file mode 100644 index 00000000..e40a1de1 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/CheatSystem/CheatManager.swift @@ -0,0 +1,20 @@ +// +// CheatManager.swift +// SoloDeveloperTraining-Dev +// +// Created by sunjae on 1/9/26. +// + +import Foundation + +enum CheatManager { + static func performCheatingActions(game: TapGame, count: Int) async { + /// 백그라운드에서 수행합니다. + await Task.detached(priority: .userInitiated) { + for _ in 0.. 0 ? "+\(goldChange)" : "\(goldChange)") + .font(.title) + .bold() + .foregroundColor(goldChange > 0 ? .green : .red) + .position( + x: gameAreaWidth / 2 + game.motionSystem.characterX, + y: gameAreaHeight - gameAreaHeight * 0.375 + ) + .transition(.move(edge: .top).combined(with: .opacity)) + } + } + .frame(width: gameAreaWidth, height: gameAreaHeight) + + Spacer() + + // 시뮬레이터 테스트용 슬라이더 + VStack(spacing: 10) { + Text("🎮 Simulator Control") + .font(.headline) + HStack(spacing: 12) { + Text("←") + .font(.title2) + Slider(value: .init(get: { + sliderValue + }, set: { value in + sliderValue = value + game.motionSystem.characterX = value * game.motionSystem.screenLimit + }), in: -1...1) + .frame(width: max(gameAreaWidth - 80, 200)) + Text("→") + .font(.title2) + } + } + .padding() + + // 게임 정보 표시 + VStack(spacing: 15) { + // 현재 골드 + HStack { + Text("💰 Gold:") + .font(.headline) + Text("\(currentGold)") + .font(.title2) + .bold() + } + + // 액션당 획득 골드 + HStack { + Text("📈 Per Action:") + .font(.headline) + Text("\(goldPerAction)") + .font(.title3) + .bold() + .foregroundColor(.blue) + } + + // 버프 사용 시간 + HStack { + Text("⚡️ Buff Time:") + .font(.headline) + Text("\(buffDuration)s") + .font(.title3) + .bold() + .foregroundColor(buffDuration > 0 ? .orange : .gray) + } + + // Start/Stop 버튼 + Button( + action: { + if game.gameCore.isRunning == true { + game.stopGame() + } else { + game.startGame() + } + }, + label: { + Text(game.gameCore.isRunning == true ? "⏸ Stop" : "▶️ Start") + .font(.headline) + .padding(.horizontal, 40) + .padding(.vertical, 12) + .background(game.gameCore.isRunning == true ? Color.red : Color.green) + .foregroundColor(.white) + .cornerRadius(10) + } + ) + .withTapSound() + .padding(.top, 5) + } + .padding() + } + .onChange(of: geometry.size) { _, newSize in + updateGameArea(for: newSize) + } + .onAppear { + setupGame(with: geometry.size) + } + } + } + + private func updateGameArea(for size: CGSize) { + // 게임 영역 크기 계산 + let availableWidth = size.width - 32 // padding 고려 + let availableHeight = size.height * 0.5 // 화면의 50% 사용 + + gameAreaWidth = availableWidth + gameAreaHeight = availableHeight + + // 게임 시스템에 크기 전달 (gameCore와 motionSystem 모두 업데이트) + game.configure(gameAreaSize: CGSize(width: availableWidth, height: availableHeight)) + } + + private func setupGame(with size: CGSize) { + // 게임 영역 크기 설정 + updateGameArea(for: size) + + // 골드 변화 콜백 설정 + game.setGoldChangedHandler { goldDelta in + showGoldChangeAnimation(goldDelta) + } + + // 초기 값 업데이트 + Task { + await updateGold() + await updateItemCounts() + } + + // 상태 변화 감지를 위한 타이머 + Timer.scheduledTimer(withTimeInterval: 0.1, repeats: true) { _ in + Task { + await self.updateGold() + await self.updateItemCounts() + await self.updateGoldPerAction() + await MainActor.run { + self.buffDuration = game.buffSystem.duration + } + } + } + } + + private func updateGoldPerAction() async { + let perAction = Calculator.calculateGoldPerAction( + game: .dodge, + user: game.user, + feverMultiplier: game.feverSystem.feverMultiplier, + buffMultiplier: game.buffSystem.multiplier + ) + await MainActor.run { + goldPerAction = perAction + } + } + + private func showGoldChangeAnimation(_ goldDelta: Int) { + // 이전 애니메이션 작업 취소 + goldAnimationTask?.cancel() + + // 새 골드 변화 설정 + recentGoldChange = goldDelta + + withAnimation(.easeIn(duration: 0.2)) { + showGoldAnimation = true + } + + // 새 애니메이션 작업 생성 + let task = DispatchWorkItem { + withAnimation(.easeOut(duration: 0.3)) { + showGoldAnimation = false + } + } + goldAnimationTask = task + + DispatchQueue.main.asyncAfter(deadline: .now() + 1.0, execute: task) + } + + private func updateGold() async { + let gold = game.user.wallet.gold + await MainActor.run { + currentGold = gold + } + } + + private func updateItemCounts() async { + let coffee = game.user.inventory.count(.coffee) ?? 0 + let energyDrink = game.user.inventory.count(.energyDrink) ?? 0 + await MainActor.run { + coffeeCount = coffee + energyDrinkCount = energyDrink + } + } + + private func useCoffee() { + Task { + let success = game.user.inventory.drink(.coffee) + if success { + game.buffSystem.useConsumableItem(type: .coffee) + } + } + } + + private func useEnergyDrink() { + Task { + let success = game.user.inventory.drink(.energyDrink) + if success { + game.buffSystem.useConsumableItem(type: .energyDrink) + } + } + } +} + +#Preview { + let wallet = Wallet(gold: 1000, diamond: 0) + let inventory = Inventory( + equipmentItems: [], + consumableItems: [ + .init(type: .coffee, count: 5), + .init(type: .energyDrink, count: 5) + ], + housing: .init(tier: .street) + ) + let record = Record() + let user = User( + nickname: "TestUser", + wallet: wallet, + inventory: inventory, + record: record, + skills: [ + .init(key: SkillKey(game: .dodge, tier: .beginner), level: 1000) + ] + ) + DodgeGameTestView(user: user) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/LanguageGameTestView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/LanguageGameTestView.swift new file mode 100644 index 00000000..395d7e46 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/LanguageGameTestView.swift @@ -0,0 +1,108 @@ +// +// LanguageGameView.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/13/26. +// + +import SwiftUI + +private enum Constant { + enum Spacing { + static let vertical: CGFloat = 67 + static let itemHorizontal: CGFloat = 25 + static let buttonHorizontal: CGFloat = 17 + } +} + +struct LanguageGameTestView: View { + let user: User + let game: LanguageGame + let languageTypeList: [LanguageType] = [ + .swift, + .kotlin, + .dart, + .python + ] + + @State private var coffeeCount: Int + @State private var energyDrinkCount: Int + + init(user: User) { + self.user = user + coffeeCount = user.inventory.count(.coffee) ?? 0 + energyDrinkCount = user.inventory.count(.energyDrink) ?? 0 + + self.game = .init( + user: user, + feverSystem: .init( + decreaseInterval: 0.1, + decreasePercentPerTick: 10 + ), + buffSystem: .init(), + itemCount: 5 // 임시로 설정 + ) + self.game.startGame() + } + + var body: some View { + VStack(alignment: .center) { + GameToolBar( + closeButtonDidTapHandler: {}, + coffeeButtonDidTapHandler: { + useConsumableItem(.coffee) + }, + energyDrinkButtonDidTapHandler: { + useConsumableItem(.energyDrink) + }, + feverState: game.feverSystem, + buffSystem: game.buffSystem, + coffeeCount: $coffeeCount, + energyDrinkCount: $energyDrinkCount, + ) + + Text("총 재화: \(user.wallet.gold)") + Spacer() + + ScrollView(.horizontal) { + HStack(alignment: .center, spacing: Constant.Spacing.itemHorizontal) { + Spacer(minLength: 0) + ForEach(game.itemList.indices, id: \.self) { index in + let item = game.itemList[index] + LanguageItem( + languageType: item.languageType, + state: item.state + ) + } + Spacer(minLength: 0) + } + }.scrollIndicators(.never) + + Spacer() + + HStack(spacing: Constant.Spacing.buttonHorizontal) { + ForEach(languageTypeList, id: \.self) { type in + LanguageButton(languageType: type, action: { + Task { + _ = await game.didPerformAction(type) + } + }) + } + } + }.padding() + } +} + +private extension LanguageGameTestView { + func useConsumableItem(_ type: ConsumableType) { + let isSuccess = user.inventory.drink(.coffee) + if isSuccess { + self.updateConsumableItems() + } + } + + func updateConsumableItems() { + coffeeCount = user.inventory.count(.coffee) ?? 0 + energyDrinkCount = user.inventory.count(.energyDrink) ?? 0 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/MissionTestView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/MissionTestView.swift new file mode 100644 index 00000000..82124f8b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/MissionTestView.swift @@ -0,0 +1,822 @@ +// +// MissionTestView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-08. +// + +import SwiftUI + +// swiftlint:disable type_body_length + +struct MissionTestView: View { + // 시스템 객체들 + @State private var missionSystem: MissionSystem + @State private var record: Record + @State private var wallet: Wallet + + // UI 상태 + @State private var showAlert = false + @State private var alertMessage = "" + + init() { + // 팩토리를 사용하여 전체 미션 목록 생성 + let missions = MissionFactory.createAllMissions() + + _missionSystem = State(initialValue: MissionSystem(missions: missions)) + _record = State(initialValue: Record()) + _wallet = State(initialValue: Wallet()) + } + + var body: some View { + NavigationView { + ScrollView { + VStack(spacing: 20) { + // 상단: 재화 표시 + walletSection + + // 탭 카운터 섹션 + tapCounterSection + + // 언어 맞추기 섹션 + languageGameSection + + // 버그 피하기 섹션 + bugDodgeSection + + // 데이터 쌓기 섹션 + stackingGameSection + + // 소비 아이템 섹션 + consumableSection + + // 특수 달성 섹션 + specialAchievementSection + + // 미션 완료 알림 + if missionSystem.hasCompletedMission { + completionBanner + } + + // 미션 목록 + missionsSection + } + .padding() + } + .navigationTitle("미션 시스템 테스트") + .alert("알림", isPresented: $showAlert) { + Button("확인", role: .cancel) {} + .withTapSound() + } message: { + Text(alertMessage) + } + } + } + + // MARK: - Wallet Section + private var walletSection: some View { + HStack(spacing: 30) { + VStack { + Image(systemName: "dollarsign.circle.fill") + .font(.system(size: 40)) + .foregroundStyle(.yellow) + Text("골드") + .font(.caption) + Text("\(wallet.gold)") + .font(.title2) + .bold() + } + + VStack { + Image(systemName: "diamond.fill") + .font(.system(size: 40)) + .foregroundStyle(.cyan) + Text("다이아몬드") + .font(.caption) + Text("\(wallet.diamond)") + .font(.title2) + .bold() + } + } + .frame(maxWidth: .infinity) + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.background) + .shadow(color: .black.opacity(0.1), radius: 5) + ) + } + + // MARK: - Tap Counter Section + private var tapCounterSection: some View { + VStack(spacing: 15) { + Text("총 탭 횟수") + .font(.headline) + + Text("\(record.totalTapCount)") + .font(.system(size: 48, weight: .bold)) + .foregroundStyle(.blue) + + HStack(spacing: 15) { + Button { + performTap() + } label: { + Label("탭 +1", systemImage: "hand.tap.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.blue) + .foregroundStyle(.white) + .cornerRadius(10) + } + + Button { + performTap(count: 10) + } label: { + Label("탭 +10", systemImage: "hand.tap.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.purple) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + + HStack(spacing: 15) { + Button { + performTap(count: 100) + } label: { + Label("탭 +100", systemImage: "bolt.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.orange) + .foregroundStyle(.white) + .cornerRadius(10) + } + + Button { + performTap(count: 1000) + } label: { + Label("탭 +1000", systemImage: "bolt.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.red) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + + } + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.background) + .shadow(color: .black.opacity(0.1), radius: 5) + ) + } + + // MARK: - Language Game Section + private var languageGameSection: some View { + VStack(spacing: 15) { + Text("언어 맞추기 게임") + .font(.headline) + + HStack { + VStack { + Text("정답 횟수") + .font(.caption) + Text("\(record.languageCorrectCount)") + .font(.title3) + .bold() + .foregroundStyle(.green) + } + + Spacer() + + VStack { + Text("연속 정답") + .font(.caption) + Text("\(record.languageConsecutiveCorrect)") + .font(.title3) + .bold() + .foregroundStyle(.orange) + } + } + .padding(.vertical, 5) + + HStack(spacing: 15) { + Button { + performLanguageCorrect() + } label: { + Label("정답", systemImage: "checkmark.circle.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.green) + .foregroundStyle(.white) + .cornerRadius(10) + } + + Button { + performLanguageIncorrect() + } label: { + Label("오답 (리셋)", systemImage: "xmark.circle.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.red) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + } + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.background) + .shadow(color: .black.opacity(0.1), radius: 5) + ) + } + + // MARK: - Bug Dodge Section + private var bugDodgeSection: some View { + VStack(spacing: 15) { + Text("버그 피하기 게임") + .font(.headline) + + HStack { + VStack { + Text("골드 획득") + .font(.caption) + Text("\(record.dodgeGoldCollectedCount)") + .font(.title3) + .bold() + .foregroundStyle(.yellow) + } + + Spacer() + + VStack { + Text("최고 콤보") + .font(.caption) + Text("\(record.dodgeMaxCombo)") + .font(.title3) + .bold() + .foregroundStyle(.cyan) + } + } + .padding(.vertical, 5) + + HStack(spacing: 15) { + Button { + performDodgeGoldHit() + } label: { + Label("골드 획득 (성공)", systemImage: "checkmark.circle.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.yellow) + .foregroundStyle(.black) + .cornerRadius(10) + } + + Button { + performDodgeGoldHit(count: 100) + } label: { + Label("골드 +100", systemImage: "bolt.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.orange) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + + HStack(spacing: 15) { + Button { + performDodgeFail() + } label: { + Label("버그 충돌 (실패)", systemImage: "xmark.circle.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.red) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + } + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.background) + .shadow(color: .black.opacity(0.1), radius: 5) + ) + } + + // MARK: - Stacking Game Section + private var stackingGameSection: some View { + VStack(spacing: 15) { + Text("데이터 쌓기 게임") + .font(.headline) + + HStack { + VStack { + Text("성공 횟수") + .font(.caption) + Text("\(record.stackingSuccessCount)") + .font(.title3) + .bold() + .foregroundStyle(.purple) + } + + Spacer() + + VStack { + Text("연속 성공") + .font(.caption) + Text("\(record.stackConsecutiveSuccess)") + .font(.title3) + .bold() + .foregroundStyle(.indigo) + } + } + .padding(.vertical, 5) + + HStack(spacing: 15) { + Button { + performStackingSuccess() + } label: { + Label("성공 +1", systemImage: "checkmark.circle.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.purple) + .foregroundStyle(.white) + .cornerRadius(10) + } + + Button { + performStackingSuccess(count: 10) + } label: { + Label("성공 +10", systemImage: "bolt.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.indigo) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + + HStack(spacing: 15) { + Button { + performStackingFail() + } label: { + Label("실패 (리셋)", systemImage: "xmark.circle.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.red) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + } + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.background) + .shadow(color: .black.opacity(0.1), radius: 5) + ) + } + + // MARK: - Consumable Section + private var consumableSection: some View { + VStack(spacing: 15) { + Text("소비 아이템") + .font(.headline) + + HStack { + VStack { + Text("커피") + .font(.caption) + Text("\(record.coffeeUseCount)") + .font(.title3) + .bold() + .foregroundStyle(.brown) + } + + Spacer() + + VStack { + Text("박하스") + .font(.caption) + Text("\(record.energyDrinkUseCount)") + .font(.title3) + .bold() + .foregroundStyle(.mint) + } + } + .padding(.vertical, 5) + + HStack(spacing: 15) { + Button { + performCoffeeUse() + } label: { + Label("커피 +1", systemImage: "cup.and.saucer.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.brown) + .foregroundStyle(.white) + .cornerRadius(10) + } + + Button { + performCoffeeUse(count: 10) + } label: { + Label("커피 +10", systemImage: "bolt.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.brown.opacity(0.7)) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + + HStack(spacing: 15) { + Button { + performEnergyDrinkUse() + } label: { + Label("박하스 +1", systemImage: "bolt.circle.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.mint) + .foregroundStyle(.white) + .cornerRadius(10) + } + + Button { + performEnergyDrinkUse(count: 10) + } label: { + Label("박하스 +10", systemImage: "bolt.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.mint.opacity(0.7)) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + } + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.background) + .shadow(color: .black.opacity(0.1), radius: 5) + ) + } + + // MARK: - Special Achievement Section + private var specialAchievementSection: some View { + VStack(spacing: 15) { + Text("특수 달성") + .font(.headline) + + HStack { + VStack { + Text("튜토리얼") + .font(.caption) + Text(record.tutorialCompleted ? "완료" : "미완료") + .font(.caption) + .bold() + .foregroundStyle(record.tutorialCompleted ? .green : .gray) + } + + Spacer() + + VStack { + Text("하찮은 개발자") + .font(.caption) + Text(record.hasAchievedJuniorDeveloper ? "달성" : "미달성") + .font(.caption) + .bold() + .foregroundStyle(record.hasAchievedJuniorDeveloper ? .green : .gray) + } + } + .padding(.vertical, 5) + + HStack(spacing: 15) { + Button { + performTutorialComplete() + } label: { + Label("튜토리얼 완료", systemImage: "graduationcap.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.blue) + .foregroundStyle(.white) + .cornerRadius(10) + } + .disabled(record.tutorialCompleted) + + Button { + performCareerAchieve() + } label: { + Label("커리어 달성", systemImage: "star.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.purple) + .foregroundStyle(.white) + .cornerRadius(10) + } + .disabled(record.hasAchievedJuniorDeveloper) + } + + HStack(spacing: 15) { + Button { + addPlayTime(hours: 1) + } label: { + Label("플레이타임 +1시간", systemImage: "clock.fill") + .frame(maxWidth: .infinity) + .padding() + .background(Color.teal) + .foregroundStyle(.white) + .cornerRadius(10) + } + + Button { + addPlayTime(hours: 10) + } label: { + Label("플레이타임 +10시간", systemImage: "clock.arrow.circlepath") + .frame(maxWidth: .infinity) + .padding() + .background(Color.teal.opacity(0.7)) + .foregroundStyle(.white) + .cornerRadius(10) + } + } + + VStack(alignment: .leading, spacing: 4) { + Text("총 플레이 시간") + .font(.caption) + .foregroundStyle(.secondary) + Text("\(Int(record.totalPlayTime / 3600))시간") + .font(.title3) + .bold() + .foregroundStyle(.teal) + } + .frame(maxWidth: .infinity, alignment: .leading) + } + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.background) + .shadow(color: .black.opacity(0.1), radius: 5) + ) + } + + // MARK: - Completion Banner + private var completionBanner: some View { + HStack { + Image(systemName: "trophy.fill") + .font(.title) + .foregroundStyle(.yellow) + + Text("완료된 미션이 있습니다!") + .font(.headline) + + Spacer() + } + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.yellow.opacity(0.2)) + .overlay( + RoundedRectangle(cornerRadius: 12) + .stroke(.yellow, lineWidth: 2) + ) + ) + } + + // MARK: - Missions Section + private var missionsSection: some View { + VStack(alignment: .leading, spacing: 15) { + Text("미션 목록") + .font(.title2) + .bold() + + ForEach(missionSystem.missions, id: \.id) { mission in + MissionCardTest( + mission: mission, + onAcquire: { + acquireMission(mission) + } + ) + } + } + } + + // MARK: - Actions + private func performTap(count: Int = 1) { + record.record(.tap(count: count)) + missionSystem.updateCompletedMissions(record: record) + } + + private func acquireMission(_ mission: Mission) { + guard mission.state == .claimable else { + alertMessage = "수령할 수 없는 업적입니다" + showAlert = true + return + } + + missionSystem.claimMissionReward(mission: mission, wallet: wallet) + + var message = "미션 '\(mission.title)' 보상을 수령했습니다!\n" + if mission.reward.gold > 0 { + message += "골드 +\(mission.reward.gold)\n" + } + if mission.reward.diamond > 0 { + message += "다이아몬드 +\(mission.reward.diamond)" + } + + alertMessage = message + showAlert = true + } + + private func performLanguageCorrect() { + record.record(.languageCorrect) + missionSystem.updateCompletedMissions(record: record) + } + + private func performLanguageIncorrect() { + record.record(.languageIncorrect) + missionSystem.updateCompletedMissions(record: record) + } + + private func performDodgeGoldHit(count: Int = 1) { + for _ in 0.. Void + + private var stateColor: Color { + switch mission.state { + case .inProgress: return .gray + case .claimable: return .green + case .claimed: return .blue + } + } + + private var stateIcon: String { + switch mission.state { + case .inProgress: return "clock.fill" + case .claimable: return "gift.fill" + case .claimed: return "checkmark.circle.fill" + } + } + + private var stateText: String { + switch mission.state { + case .inProgress: return "진행 중" + case .claimable: return "수령 가능" + case .claimed: return "완료" + } + } + + var body: some View { + VStack(alignment: .leading, spacing: 12) { + HStack { + VStack(alignment: .leading, spacing: 4) { + Text(mission.title) + .font(.headline) + + Text(mission.description) + .font(.caption) + .foregroundStyle(.secondary) + } + + Spacer() + + VStack { + Image(systemName: stateIcon) + .font(.title2) + .foregroundStyle(stateColor) + + Text(stateText) + .font(.caption2) + .foregroundStyle(stateColor) + } + } + + // 진행도 바 + VStack(alignment: .leading, spacing: 4) { + HStack { + Text("\(mission.currentValue) / \(mission.targetValue)") + .font(.caption) + .foregroundStyle(.secondary) + + Spacer() + + Text("\(Int(mission.progress * 100))%") + .font(.caption) + .foregroundStyle(.secondary) + } + + ProgressView(value: mission.progress) + .tint(stateColor) + } + + // 보상 표시 + HStack { + if mission.reward.gold > 0 { + Label("\(mission.reward.gold)", systemImage: "dollarsign.circle.fill") + .font(.caption) + .foregroundStyle(.yellow) + } + + if mission.reward.diamond > 0 { + Label("\(mission.reward.diamond)", systemImage: "diamond.fill") + .font(.caption) + .foregroundStyle(.cyan) + } + + Spacer() + + if mission.state == .claimable { + Button { + onAcquire() + } label: { + Text("수령") + .font(.caption) + .padding(.horizontal, 12) + .padding(.vertical, 6) + .background(Color.green) + .foregroundStyle(.white) + .cornerRadius(8) + } + } + } + } + .padding() + .background( + RoundedRectangle(cornerRadius: 12) + .fill(.background) + .overlay( + RoundedRectangle(cornerRadius: 12) + .stroke(stateColor.opacity(0.3), lineWidth: 2) + ) + .shadow(color: .black.opacity(0.05), radius: 3) + ) + } +} + +// MARK: - Preview +#Preview { + MissionTestView() +} + diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/QuizGameTestContentView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/QuizGameTestContentView.swift new file mode 100644 index 00000000..78ef1132 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/QuizGameTestContentView.swift @@ -0,0 +1,327 @@ +// +// QuizGameTestContentView.swift +// SoloDeveloperTraining +// +// Created by Claude on 1/21/26. +// + +import SwiftUI + +private enum Constant { + enum Padding { + static let horizontal: CGFloat = 16 + static let vertical: CGFloat = 20 + static let questionTop: CGFloat = 40 + static let optionsSpacing: CGFloat = 12 + static let submitTop: CGFloat = 24 + } + + enum Timer { + static let height: CGFloat = 8 + static let cornerRadius: CGFloat = 4 + } +} + +struct QuizGameTestContentView: View { + // MARK: - Properties + @State var game: QuizGame + + @Binding var isGameStarted: Bool + + // MARK: - State + @State private var showCompletionAlert = false + @State private var hasStarted = false + + init(user: User, isGameStarted: Binding) { + self._isGameStarted = isGameStarted + self._game = State(initialValue: QuizGame(user: user)) + } + + var body: some View { + let state = game.state + + ZStack { + Color.white.ignoresSafeArea() + + VStack(spacing: 0) { + // 헤더 (닫기 버튼, 진행도) + headerSection + + // 타이머 프로그래스 바 + timerProgressBar + + // 문제 및 선택지 + if state.phase == .questionInProgress || state.phase == .showingExplanation { + questionSection + } else if state.phase == .completed { + EmptyView() + } + + Spacer() + } + } + .onAppear { + if !hasStarted { + print("👀 View appeared, starting game") + game.startGame() + hasStarted = true + } + } + .alert("퀴즈 완료!", isPresented: $showCompletionAlert) { + Button("확인") { + isGameStarted = false + } + .withTapSound() + } message: { + Text("정답: \(state.correctAnswersCount)/3개\n획득 다이아: \(state.totalDiamondsEarned)개") + } + .onChange(of: state.phase) { _, newPhase in + if newPhase == .completed { + showCompletionAlert = true + } + } + } +} + +// MARK: - View Components +private extension QuizGameTestContentView { + var headerSection: some View { + let state = game.state + + return VStack(spacing: 8) { + HStack { + Button { + game.stopGame() + isGameStarted = false + } label: { + Image(systemName: "xmark") + .resizable() + .frame(width: 20, height: 20) + .foregroundStyle(.black) + } + + Spacer() + + Text(state.progressText) + .textStyle(.title3) + .foregroundStyle(.black) + + Spacer() + + // 대칭을 위한 투명 공간 + Color.clear.frame(width: 20, height: 20) + } + + // 현재 다이아 표시 + HStack { + Image(systemName: "diamond.fill") + .foregroundStyle(.blue) + Text("다이아: \(game.user.wallet.diamond)") + .textStyle(.body) + .foregroundStyle(.black) + } + } + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.vertical, Constant.Padding.vertical) + } + + var timerProgressBar: some View { + let state = game.state + + return GeometryReader { geometry in + ZStack(alignment: .leading) { + // 배경 + Rectangle() + .fill(Color.gray.opacity(0.2)) + .frame(height: Constant.Timer.height) + + // 진행 바 + Rectangle() + .fill(timerColor) + .frame( + width: geometry.size.width * state.timerProgress, + height: Constant.Timer.height + ) + .animation(.linear(duration: 1.0), value: state.timerProgress) + } + .cornerRadius(Constant.Timer.cornerRadius) + } + .frame(height: Constant.Timer.height) + .padding(.horizontal, Constant.Padding.horizontal) + } + + var timerColor: Color { + let progress = game.state.timerProgress + if progress > 0.5 { + return .blue + } else if progress > 0.2 { + return .yellow + } else { + return .red + } + } + + var questionSection: some View { + let state = game.state + + return VStack(alignment: .leading, spacing: Constant.Padding.optionsSpacing) { + // 문제 텍스트 + if let question = state.currentQuestion { + Text(question.question) + .textStyle(.headline) + .foregroundStyle(.black) + .padding(.top, Constant.Padding.questionTop) + .padding(.bottom, Constant.Padding.vertical) + + // 답안 선택지 + ForEach(0.. some View { + let state = game.state + + return QuizButton( + isSelected: state.selectedAnswerIndex == index, + title: "\(index + 1). \(text)" + ) { + if state.phase == .questionInProgress { + game.selectAnswer(index) + } + } + .overlay { + // 제출 후 정답/오답 표시 + if state.phase == .showingExplanation { + correctnessOverlay(for: index) + } + } + .disabled(state.phase != .questionInProgress) + } + + @ViewBuilder + func correctnessOverlay(for index: Int) -> some View { + let state = game.state + + return Group { + if let question = state.currentQuestion { + if index == question.correctAnswerIndex { + // 정답에 체크 표시 + HStack { + Spacer() + Image(systemName: "checkmark.circle.fill") + .foregroundStyle(.green) + .font(.system(size: 24)) + .padding() + } + } else if index == state.selectedAnswerIndex && state.currentAnswerResult != .correct { + // 선택한 오답에 X 표시 + HStack { + Spacer() + Image(systemName: "xmark.circle.fill") + .foregroundStyle(.red) + .font(.system(size: 24)) + .padding() + } + } + } + } + } + + var submitButton: some View { + let state = game.state + + return LargeButton( + title: "제출", + isEnabled: state.isSubmitEnabled + ) { + game.submitSelectedAnswer() + } + .frame(maxWidth: .infinity) + .padding(.top, Constant.Padding.submitTop) + } + + var explanationSection: some View { + let state = game.state + + return VStack(alignment: .leading, spacing: 16) { + // 정답/오답 표시 + HStack { + Image(systemName: resultIcon) + .foregroundStyle(resultColor) + .font(.system(size: 24)) + Text(resultText) + .textStyle(.title3) + .foregroundStyle(resultColor) + } + + // 해설 + if let question = state.currentQuestion { + Text("해설") + .textStyle(.headline) + .foregroundStyle(.black) + + Text(question.explanation) + .textStyle(.body) + .foregroundStyle(.gray) + } + + // 다음 버튼 + LargeButton( + title: state.nextButtonTitle + ) { + game.proceedToNextQuestion() + } + .frame(maxWidth: .infinity) + .padding(.top, 8) + } + .padding(.top, Constant.Padding.submitTop) + } + + var resultIcon: String { + switch game.state.currentAnswerResult { + case .correct: return "checkmark.circle.fill" + case .incorrect, .timeout: return "xmark.circle.fill" + case .none: return "" + } + } + + var resultColor: Color { + switch game.state.currentAnswerResult { + case .correct: return .green + case .incorrect, .timeout: return .red + case .none: return .black + } + } + + var resultText: String { + switch game.state.currentAnswerResult { + case .correct: return "정답입니다!" + case .incorrect: return "오답입니다." + case .timeout: return "시간 초과!" + case .none: return "" + } + } +} + +#Preview { + struct PreviewWrapper: View { + @State private var isGameStarted = true + + var body: some View { + QuizGameTestView() + } + } + + return PreviewWrapper() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/QuizGameTestView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/QuizGameTestView.swift new file mode 100644 index 00000000..9a9012e6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/QuizGameTestView.swift @@ -0,0 +1,52 @@ +// +// QuizGameTestView.swift +// SoloDeveloperTraining +// +// Created by Claude on 1/21/26. +// + +import SwiftUI + +struct QuizGameTestView: View { + @State private var isGameStarted = false + @State private var user: User + + init() { + self._user = State(initialValue: User( + nickname: "퀴즈 테스터", + wallet: Wallet(gold: 0, diamond: 100), + inventory: Inventory(), + record: Record(), + skills: [] + )) + } + + var body: some View { + ZStack { + Color.white.ignoresSafeArea() + + if isGameStarted { + QuizGameTestContentView(user: user, isGameStarted: $isGameStarted) + } else { + VStack(spacing: 20) { + Text("퀴즈 게임 테스트") + .textStyle(.title) + .foregroundStyle(.black) + + Text("현재 다이아: \(user.wallet.diamond)") + .textStyle(.headline) + .foregroundStyle(.gray) + + LargeButton(title: "퀴즈 시작") { + isGameStarted = true + } + .padding() + } + } + } + } +} + +#Preview { + QuizGameTestView() +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/ShopTestView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/ShopTestView.swift new file mode 100644 index 00000000..73ad9930 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/ShopTestView.swift @@ -0,0 +1,65 @@ +// +// ShopTestView.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/7/26. +// + +import SwiftUI + +struct ShopTestView: View { + let user: User + let shopSystem: ShopSystem + + init(user: User) { + self.user = user + self.shopSystem = .init(user: user) + } + var body: some View { + VStack { + Text("보유 골드: \(user.wallet.gold)") + Text("보유 다이아: \(user.wallet.diamond)") + Text("초당 획득 골드: \(Calculator.calculateGoldPerSecond(user: user))") + Text("부동산: \(user.inventory.housing.displayTitle)") + + ScrollView { + LazyVStack(spacing: 0) { + ForEach(shopSystem.itemList(itemTypes: [.consumable, .equipment, .housing])) { item in + ItemRow( + title: item.displayTitle + "\(item.isEquipped ? "-착용중" : "")", + description: item.description, + imageName: item.imageName, + cost: item.cost, + state: item.isPurchasable ? .available : .insufficient + ) { + do { + try shopSystem.buy(item: item) + } catch { + print(error) + } + } + } + } + } + .scrollBounceBehavior(.basedOnSize) + } + } +} + +#Preview { + let user = User( + nickname: "user", + wallet: .init(gold: 1000000, diamond: 100), + inventory: .init(), + record: .init(), + skills: [ + .init(key: .init(game: .tap, tier: .beginner), level: 1000), + .init( + key: .init(game: .tap, tier: .intermediate), + level: 1000 + ), + .init(key: .init(game: .tap, tier: .advanced), level: 1000) + ] + ) + ShopTestView(user: user) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/SkillTestView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/SkillTestView.swift new file mode 100644 index 00000000..1b7b49d8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/SkillTestView.swift @@ -0,0 +1,112 @@ +// +// SkillTestView.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/20/26. +// + +import SwiftUI + +struct SkillTestView: View { + let user: User + let skillSystem: SkillSystem + + init(user: User) { + self.user = user + self.skillSystem = .init(user: user, careerSystem: nil) + } + var body: some View { + VStack { + Text("보유 골드: \(user.wallet.gold)") + Text("보유 다이아: \(user.wallet.diamond)") + Text("초당 획득 골드: \(Calculator.calculateGoldPerSecond(user: user))") + + List(skillSystem.skillList(), id: \.skill) { skillState in + itemRowView(skillState) + } + } + } +} + +private extension SkillTestView { + func itemRowView(_ skillState: SkillState) -> some View { + + return HStack { + VStack(alignment: .leading, spacing: 4) { + HStack { + Text( + "\(skillState.skill.key.game.displayTitle) " + + "\(skillState.skill.key.tier.displayTitle) " + + "Lv.\(skillState.skill.level)" + ) + if skillState.itemState == .locked { + Image(systemName: "lock.fill") + .font(.caption) + .foregroundStyle(.secondary) + } + } + + Text("획득 재화 \(skillState.skill.gainGold.formatted())") + .font(.caption) + .foregroundStyle(.secondary) + + if skillState.itemState == .locked { + Text(unlockConditionText(skillState.skill)) + .font(.caption2) + .foregroundStyle(.red) + } + } + + Spacer() + + Button { + do { + try skillSystem.upgrade(skill: skillState.skill) + } catch { + print(error.localizedDescription) + } + } label: { + VStack(alignment: .trailing) { + Text("💰 \(skillState.skill.upgradeCost.gold)") + Text("💎 \(skillState.skill.upgradeCost.diamond)") + } + .padding(6) + .background( + RoundedRectangle(cornerRadius: 6) + .stroke(skillState.itemState == .locked ? .gray : .black) + ) + } + .disabled(skillState.itemState != .available) + } + .padding(.vertical, 6) + .opacity(skillState.itemState == .locked ? 0.45 : 1.0) + } + + func unlockConditionText(_ skill: Skill) -> String { + switch skill.key.tier { + case .beginner: + return "" + case .intermediate: + return "초급 Lv.1000 필요" + case .advanced: + return "중급 Lv.1000 필요" + } + } +} + +#Preview { + let user = User( + nickname: "user", + wallet: .init(gold: 1000000000, diamond: 100), + inventory: .init(), + record: .init(), + skills: Set( + GameType.allCases.flatMap { game in + SkillTier.allCases.map { tier in + Skill(key: .init(game: game, tier: tier), level: tier.levelRange.minValue) + } + } + ) + ) + SkillTestView(user: user) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/StackGameScene.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/StackGameScene.swift new file mode 100644 index 00000000..247a5937 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/StackGameScene.swift @@ -0,0 +1,291 @@ +// +// StackGameScene.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-13. +// + +import SpriteKit + +private enum Constant { + enum Physics { + static let gravity = CGVector(dx: 0, dy: -9.8) + } + + enum Offset { + static let spawnYOffset: CGFloat = 100 + } + + enum Time { + // 블록 평가 체크 간격 (초) + static let evaluationCheckInterval: TimeInterval = 0.05 + // 폭탄 블록 제거 딜레이 (초) + static let bombRemovalDelay: TimeInterval = 0.8 + // 카메라 이동 애니메이션 시간 (초) + static let cameraMoveAnimationDuration: TimeInterval = 0.3 + // 다음 블록 생성 딜레이 (초) + static let nextBlockSpawnDelay: TimeInterval = 0.3 + // 실패한 블록 제거 딜레이 (초) + static let failedBlockRemovalDelay: TimeInterval = 1.0 + } +} + +final class StackGameScene: SKScene { + private let stackGame: StackGame + + private var currentBlockView: BlockItem? + private var blockViews: [BlockItem] = [] + private var currentHeight: CGFloat = 20 + /// 블록 배치 처리 중 여부 (UI 인터랙션 차단용) + private var isProcessing = false + + init(stackGame: StackGame) { + self.stackGame = stackGame + super.init(size: .zero) + self.scaleMode = .resizeFill + } + + required init?(coder aDecoder: NSCoder) { + return nil + } + + override func didMove(to view: SKView) { + // 화면 크기를 게임 코어에 전달 + stackGame.screenSize = size + setupScene() + startGame() + } + + override func touchesBegan(_ touches: Set, with event: UIEvent?) { + guard currentBlockView != nil, !isProcessing else { return } + dropBlock() + } + + /// 씬의 초기 설정을 수행합니다. + /// - 배경색을 흰색으로 설정 + /// - 물리 엔진의 중력 설정 + /// - 카메라 초기화 + private func setupScene() { + backgroundColor = .white + physicsWorld.gravity = Constant.Physics.gravity + + setupCamera() + } + + /// 카메라 노드를 생성하고 초기 위치를 설정합니다. + private func setupCamera() { + let cameraNode = SKCameraNode() + cameraNode.position = CGPoint(x: size.width / 2, y: size.height / 2) + addChild(cameraNode) + camera = cameraNode + } + + /// 게임을 시작하고 초기 상태로 설정합니다. + /// - 블록 배열 초기화 + /// - 게임 코어 시작 처리 + /// - 카메라 위치 리셋 + /// - 초기 블록 배치 및 첫 번째 블록 생성 + func startGame() { + blockViews = [] + currentHeight = 20 + isProcessing = false + + stackGame.startGame() + camera?.position = CGPoint(x: size.width / 2, y: size.height / 2) + + putInitialBlock() + spawnBlock() + } + + /// 게임을 중지하고 모든 진행중인 동작을 멈춥니다. + /// - 게임 코어 중지 처리 + /// - 현재 블록의 모든 액션 제거 + /// - 물리 엔진 정지 + func stopGame() { + stackGame.stopGame() + isProcessing = true + currentBlockView?.removeAllActions() + physicsWorld.speed = 0 + } + + /// 게임 시작 시 가장 아래에 배치되는 초기 블록을 생성합니다. + /// - 고정된 물리 바디를 가진 파란색 블록 생성 + /// - 게임 코어에 초기 블록 등록 + private func putInitialBlock() { + let firstBlockView = BlockItem(type: .blue) + firstBlockView.setupPhysicsBody() + firstBlockView.position = CGPoint(x: size.width / 2, y: currentHeight) + + // 게임 코어에 초기 블록 등록 + stackGame.addInitialBlock() + + addChild(firstBlockView) + blockViews.append(firstBlockView) + } + + /// 새로운 블록을 화면 상단에 생성하고 좌우로 움직이게 합니다. + /// - 랜덤한 타입의 블록 생성 + /// - 카메라 기준 상단 위치에서 생성 + /// - 좌우 이동 애니메이션 시작 + private func spawnBlock() { + isProcessing = false + + let blockType = BlockType.allCases.randomElement() ?? .blue + let blockView = BlockItem(type: blockType) + + // 게임 코어에 블록 생성 알림 + stackGame.spawnBlock(type: blockType) + + let spawnY = (camera?.position.y ?? size.height / 2) + size.height / 2 - Constant.Offset.spawnYOffset + let leftEdge = blockView.size.width / 2 + let rightEdge = size.width - blockView.size.width / 2 + + blockView.position = CGPoint(x: leftEdge, y: spawnY) + blockView.startMoving(distance: rightEdge - leftEdge) + + currentBlockView = blockView + addChild(blockView) + } + + /// 현재 블록의 이동을 멈추고 중력을 적용하여 떨어뜨립니다. + /// - 블록의 좌우 이동 중지 + /// - 중력 활성화 + /// - 블록 평가 시작 + private func dropBlock() { + guard let block = currentBlockView else { return } + + isProcessing = true + block.stopMoving() + block.enableGravity() + + // 블록 평가 시작 + evaluateBlock() + } + + /// 떨어지는 블록이 목표 위치에 도달했는지 재귀적으로 확인합니다. + /// - 목표 높이에 도달하면 정렬 체크 수행 + /// - 아직 도달하지 않았으면 일정 시간 후 재확인 + private func evaluateBlock() { + guard + let block = currentBlockView, + let previousBlock = stackGame.previousBlock + else { return } + + // StackGame의 previousBlock 정보를 사용해 목표 Y 계산 + let targetY = previousBlock.positionY + previousBlock.height + + if block.position.y <= targetY + block.size.height { + // 목표 위치에 도달했으므로 정렬 체크 + // 정렬 성공/실패에 따라 물리 처리를 다르게 적용 + checkAlignmentAndHandle(targetY: targetY) + } else { + // 아직 도달하지 않았으면 재확인 + DispatchQueue.global().asyncAfter(deadline: .now() + Constant.Time.evaluationCheckInterval) { [weak self] in + self?.evaluateBlock() + } + } + } + + /// 정렬을 체크하고 결과에 따라 물리 처리를 다르게 적용합니다. + /// - 성공: 블록 고정 후 배치 + /// - 실패: 물리를 유지하여 자연스럽게 떨어지도록 + private func checkAlignmentAndHandle(targetY: CGFloat) { + guard let block = currentBlockView else { return } + + // 현재 블록 위치를 게임 모델에 업데이트 + stackGame.updateCurrentBlockPosition(positionX: block.position.x, positionY: targetY) + + let isAligned = stackGame.checkAlignment() + + if isAligned { + // 정렬 성공: 블록 고정 + block.fixPosition() + block.physicsBody?.velocity = CGVector.zero + block.physicsBody?.angularVelocity = 0 + block.position = CGPoint(x: block.position.x, y: targetY) + isProcessing = false + + placeBlockSuccess() + } else { + // 정렬 실패: 물리를 유지하여 계속 떨어지도록 + isProcessing = false + placeBlockFail() + } + } + + /// 블록이 성공적으로 배치되었을 때의 처리를 수행합니다. + /// - 폭탄 블록: 패널티 적용 후 블록 제거 + /// - 일반 블록: 스택에 추가, 점수 증가, 보상 적용, 카메라 이동 + private func placeBlockSuccess() { + guard + let block = currentBlockView, + let currentBlock = stackGame.currentBlock, + let previousView = blockViews.last + else { return } + + // 이전 블록의 정확한 위치를 기준으로 배치 + let targetY = previousView.position.y + previousView.size.height + block.position = CGPoint(x: block.position.x, y: targetY) + + // currentHeight 업데이트 + currentHeight = targetY + block.size.height + + // 폭탄 블록 체크 + if currentBlock.type.isBomb { + stackGame.placeBombSuccess() + + DispatchQueue.main.asyncAfter(deadline: .now() + Constant.Time.bombRemovalDelay) { [weak self] in + block.removeFromParent() + self?.spawnBlock() + } + } else { + blockViews.append(block) + + // 높이 증가 (렌더링 정보) + currentHeight += block.size.height + + // 코어에 블록 배치 성공 알림 (위치는 이미 업데이트됨) + stackGame.placeBlockSuccess() + + // 카메라 이동 + if let camera = camera { + let newCameraY = camera.position.y + block.size.height + let moveCamera = SKAction.moveTo(y: newCameraY, duration: Constant.Time.cameraMoveAnimationDuration) + moveCamera.timingMode = .easeInEaseOut + camera.run(moveCamera) + } + + DispatchQueue.main.asyncAfter(deadline: .now() + Constant.Time.nextBlockSpawnDelay) { [weak self] in + self?.spawnBlock() + } + } + + currentBlockView = nil + } + + /// 블록 배치에 실패했을 때의 처리를 수행합니다. + /// - 폭탄 블록: 실패시 오히려 보상 적용 + /// - 일반 블록: 패널티 적용 + /// - 물리 효과로 떨어지고 화면 밖으로 나가면 제거 + private func placeBlockFail() { + guard + let block = currentBlockView, + let currentBlock = stackGame.currentBlock + else { return } + + // 폭탄 블록 실패 = 보상, 일반 블록 실패 = 패널티 + if currentBlock.type.isBomb { + stackGame.placeBombFail() + } else { + stackGame.placeBlockFail() + } + + // 일정 시간 후 블록 제거 및 다음 블록 생성 + DispatchQueue.main.asyncAfter(deadline: .now() + Constant.Time.failedBlockRemovalDelay) { [weak self] in + block.removeFromParent() + self?.spawnBlock() + } + + currentBlockView = nil + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/StackGameTestView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/StackGameTestView.swift new file mode 100644 index 00000000..502656c0 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/StackGameTestView.swift @@ -0,0 +1,62 @@ +// +// StackGameTestView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-13. +// + +import SpriteKit +import SwiftUI + +struct StackGameTestView: View { + @State private var game: StackGame + private let scene: StackGameScene + + init(user: User) { + let game = StackGame(user: user) + let scene = StackGameScene(stackGame: game) + + self.scene = scene + self._game = State(wrappedValue: game) + } + + var body: some View { + VStack { + VStack { + Text("골드: \(game.user.wallet.gold)") + Text("다이아몬드: \(game.user.wallet.diamond)") + } + .frame(maxHeight: .infinity) + + ZStack(alignment: .top) { + SpriteView(scene: scene) + + GameToolBar( + closeButtonDidTapHandler: stopGame, + coffeeButtonDidTapHandler: useCoffee, + energyDrinkButtonDidTapHandler: useEnergyDrink, + feverState: game.feverSystem, + buffSystem: game.buffSystem, + coffeeCount: .constant(game.user.inventory.count(.coffee) ?? 0), + energyDrinkCount: .constant(game.user.inventory.count(.energyDrink) ?? 0) + ) + .padding() + } + .frame(maxHeight: .infinity) + } + } + + private func stopGame() {} + + private func useCoffee() { + if game.user.inventory.drink(.coffee) { + game.buffSystem.useConsumableItem(type: .coffee) + } + } + + private func useEnergyDrink() { + if game.user.inventory.drink(.energyDrink) { + game.buffSystem.useConsumableItem(type: .energyDrink) + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/TabGameView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/TabGameView.swift new file mode 100644 index 00000000..47eb820a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Devlopment/Presentation/TabGameView.swift @@ -0,0 +1,76 @@ +// +// TabGameView.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import SwiftUI + +struct TabGameView: View { + let user: User + let game: TapGame + + init(user: User) { + self.user = user + self.game = .init( + user: user, + feverSystem: .init( + decreaseInterval: 0.1, + decreasePercentPerTick: 10 + ), + buffSystem: .init() + ) + self.game.startGame() + } + + var body: some View { + VStack(spacing: 40) { + VStack { + Text("gold: \(user.wallet.gold)") + Text("fever: \(game.feverSystem.feverPercent)") + Text("feverStage: \(game.feverSystem.feverStage)") + } + HStack(spacing: 30) { + Button { + Task { + let gainGold = await game.didPerformAction(()) + print(gainGold) + } + } label: { + Text("탭 1번 수행하기") + } + Button { + Task { + await CheatManager.performCheatingActions(game: game, count: 10000) + } + } label: { + Text("탭 10,000번 수행하기") + } + } + Button { + if user.inventory.drink(.coffee) { + game.buffSystem.useConsumableItem(type: .coffee) + } + } label: { + Text("☕️ Coffee \(user.inventory.count(.coffee) ?? 0)") + } + } + .padding() + } +} + +#Preview { + let user = User( + nickname: "user", + wallet: .init(), + inventory: .init(), + record: .init(), + skills: [ + .init(key: .init(game: .tap, tier: .beginner), level: 1000), + .init(key: .init(game: .tap, tier: .intermediate), level: 1000), + .init(key: .init(game: .tap, tier: .advanced), level: 1000) + ] + ) + TabGameView(user: user) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/ButtonStyle+.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/ButtonStyle+.swift new file mode 100644 index 00000000..057a4afb --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/ButtonStyle+.swift @@ -0,0 +1,43 @@ +// +// ButtonStyle+.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/12/26. +// + +import SwiftUI + +struct PressableButtonStyle: ButtonStyle { + @Binding var isPressed: Bool + + func makeBody(configuration: Configuration) -> some View { + configuration.label + .onChange(of: configuration.isPressed) { _, newValue in + isPressed = newValue + if newValue { + SoundService.shared.trigger(.buttonTap) + } + } + } +} + +struct SoundTapButtonStyle: ButtonStyle { + func makeBody(configuration: Configuration) -> some View { + configuration.label + .onChange(of: configuration.isPressed) { _, newValue in + if newValue { + SoundService.shared.trigger(.buttonTap) + } + } + } +} + +extension ButtonStyle where Self == PressableButtonStyle { + static func pressable(isPressed: Binding) -> PressableButtonStyle { + PressableButtonStyle(isPressed: isPressed) + } +} + +extension ButtonStyle where Self == SoundTapButtonStyle { + static var soundTap: SoundTapButtonStyle { SoundTapButtonStyle() } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/Int+.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/Int+.swift new file mode 100644 index 00000000..b16e9401 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/Int+.swift @@ -0,0 +1,48 @@ +// +// Int+.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/8/26. +// + +import Foundation + +extension Int { + var formatted: String { + let absValue = abs(self) + let sign = self < 0 ? "-" : "" + + switch absValue { + case 0..<1_000: + return "\(sign)\(absValue)" + case 1_000..<1_000_000: + let value = Double(absValue) / 1_000.0 + return "\(sign)\(formatDecimal(value))K" + case 1_000_000..<1_000_000_000: + let value = Double(absValue) / 1_000_000.0 + return "\(sign)\(formatDecimal(value))M" + case 1_000_000_000..<1_000_000_000_000: + let value = Double(absValue) / 1_000_000_000.0 + return "\(sign)\(formatDecimal(value))B" + default: + let value = Double(absValue) / 1_000_000_000_000.0 + return "\(sign)\(formatDecimal(value))T" + } + } + + /// 소수점을 적절히 포맷팅합니다 (불필요한 0 제거) + private func formatDecimal(_ value: Double) -> String { + // 소수점 이하 2자리까지 표시 + var formatted = String(format: "%.2f", value) + + // 끝의 0 제거 + while formatted.contains(".") && formatted.hasSuffix("0") { + formatted = String(formatted.dropLast()) + } + if formatted.hasSuffix(".") { + formatted = String(formatted.dropLast()) + } + + return formatted + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/LongPressRepeatModifier.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/LongPressRepeatModifier.swift new file mode 100644 index 00000000..a5924221 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/LongPressRepeatModifier.swift @@ -0,0 +1,64 @@ +// +// LongPressRepeatModifier.swift +// SoloDeveloperTraining +// + +import SwiftUI + +private enum Constant { + static let minimumDuration: Double = 0.5 + static let repeatInterval: TimeInterval = 0.1 +} + +struct LongPressRepeatModifier: ViewModifier { + @Binding var isLongPressing: Bool + @State private var repeatTimer: Timer? + + let isDisabled: Bool + let onLongPressRepeat: (() -> Bool)? + + func body(content: Content) -> some View { + content + .onLongPressGesture( + minimumDuration: Constant.minimumDuration, + pressing: handlePressingChange, + perform: startRepeating + ) + .onDisappear { + repeatTimer?.invalidate() + repeatTimer = nil + } + } +} + +private extension LongPressRepeatModifier { + func handlePressingChange(_ pressing: Bool) { + guard !isDisabled, onLongPressRepeat != nil else { return } + if !pressing { stopRepeating() } + } + + func startRepeating() { + guard let onLongPressRepeat, repeatTimer == nil else { return } + + isLongPressing = true + if onLongPressRepeat() { SoundService.shared.trigger(.buttonTap) } + + let timer = Timer.scheduledTimer(withTimeInterval: Constant.repeatInterval, repeats: true) { [onLongPressRepeat, isLongPressingBinding = $isLongPressing] timer in + if onLongPressRepeat() { + SoundService.shared.trigger(.buttonTap) + } else { + timer.invalidate() + repeatTimer = nil + isLongPressingBinding.wrappedValue = false + } + } + RunLoop.current.add(timer, forMode: .common) + repeatTimer = timer + } + + func stopRepeating() { + repeatTimer?.invalidate() + repeatTimer = nil + isLongPressing = false + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/Typography.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/Typography.swift new file mode 100644 index 00000000..324e5ab5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/Typography.swift @@ -0,0 +1,242 @@ +// +// Typography.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/8/26. +// +// 사용 예시: +// +// 1. Text (lineSpacing 자동 포함) +// Text("안녕하세요") +// .textStyle(.caption) +// +// 2. TextField (단일 라인 - lineSpacing 불필요) +// TextField("입력", text: $text) +// .font(.pfFont(.body)) +// +// 3. TextEditor (다중 라인 - lineSpacing 필요) +// TextEditor(text: $longText) +// .font(.pfFont(.body)) +// .lineSpacing(TypographyStyle.body.lineSpacing) +// +// 4. 커스텀 크기 +// Text("특별한 크기") +// .font(.pfCustom(.extraBold, size: 25)) + +import SwiftUI + +// MARK: - Font Weight +enum PFFontWeight { + case regular + case bold + case extraBold + + var fontName: String { + switch self { + case .regular: return "PFStardust" + case .bold: return "PFStardustBold" + case .extraBold: return "PFStardustExtraBold" + } + } +} + +// MARK: - Typography Style +struct PFTextStyle: ViewModifier { + let fontName: String + let size: CGFloat + let lineHeight: CGFloat + + func body(content: Content) -> some View { + content + .font(.custom(fontName, size: size)) + .lineSpacing(size * (lineHeight - 1)) + } +} + +// MARK: - Typography Preset +enum Typography { + + /// LargeTitle – ExtraBold, 34pt, 130% + static let largeTitle = PFTextStyle( + fontName: PFFontWeight.extraBold.fontName, + size: 34, + lineHeight: 1.3 + ) + + /// Title – Bold, 28pt, 140% + static let title = PFTextStyle( + fontName: PFFontWeight.bold.fontName, + size: 28, + lineHeight: 1.4 + ) + + /// Title2 – Bold, 22pt, 140% + static let title2 = PFTextStyle( + fontName: PFFontWeight.bold.fontName, + size: 22, + lineHeight: 1.4 + ) + + /// Title3 – Bold, 20pt, 140% + static let title3 = PFTextStyle( + fontName: PFFontWeight.bold.fontName, + size: 20, + lineHeight: 1.4 + ) + + /// Headline – ExtraBold, 17pt, 140% + static let headline = PFTextStyle( + fontName: PFFontWeight.extraBold.fontName, + size: 17, + lineHeight: 1.4 + ) + + /// Body – Bold, 17pt, 140% + static let body = PFTextStyle( + fontName: PFFontWeight.bold.fontName, + size: 17, + lineHeight: 1.4 + ) + + /// Callout – Bold, 16pt, 140% + static let callout = PFTextStyle( + fontName: PFFontWeight.bold.fontName, + size: 16, + lineHeight: 1.4 + ) + + /// Subheadline – ExtraBold, 15pt, 140% + static let subheadline = PFTextStyle( + fontName: PFFontWeight.extraBold.fontName, + size: 15, + lineHeight: 1.4 + ) + + /// Caption – ExtraBold, 12pt, 140% + static let caption = PFTextStyle( + fontName: PFFontWeight.extraBold.fontName, + size: 12, + lineHeight: 1.4 + ) + + /// Caption2 – Bold, 12pt, 140% + static let caption2 = PFTextStyle( + fontName: PFFontWeight.bold.fontName, + size: 12, + lineHeight: 1.4 + ) + + /// Label – Bold, 11pt, 140% + static let label = PFTextStyle( + fontName: PFFontWeight.bold.fontName, + size: 11, + lineHeight: 1.4 + ) + + /// LabelLined – Regular, 11pt, 140% + static let labelLined = PFTextStyle( + fontName: PFFontWeight.regular.fontName, + size: 11, + lineHeight: 1.4 + ) +} + +enum TypographyStyle { + case largeTitle + case title + case title2 + case title3 + case headline + case body + case callout + case subheadline + case caption + case caption2 + case label + case labelLined + + var lineSpacing: CGFloat { + switch self { + case .largeTitle: return 34 * (1.3 - 1) + case .title: return 28 * (1.4 - 1) + case .title2: return 22 * (1.4 - 1) + case .title3: return 20 * (1.4 - 1) + case .headline: return 17 * (1.4 - 1) + case .body: return 17 * (1.4 - 1) + case .callout: return 16 * (1.4 - 1) + case .subheadline: return 15 * (1.4 - 1) + case .caption: return 12 * (1.4 - 1) + case .caption2: return 12 * (1.4 - 1) + case .label: return 11 * (1.4 - 1) + case .labelLined: return 11 * (1.4 - 1) + } + } +} + +// MARK: - Text Extension +extension Text { + func textStyle(_ style: TypographyStyle) -> some View { + switch style { + case .largeTitle: + self.modifier(Typography.largeTitle) + case .title: + self.modifier(Typography.title) + case .title2: + self.modifier(Typography.title2) + case .title3: + self.modifier(Typography.title3) + case .headline: + self.modifier(Typography.headline) + case .body: + self.modifier(Typography.body) + case .callout: + self.modifier(Typography.callout) + case .subheadline: + self.modifier(Typography.subheadline) + case .caption: + self.modifier(Typography.caption) + case .caption2: + self.modifier(Typography.caption2) + case .label: + self.modifier(Typography.label) + case .labelLined: + self.modifier(Typography.labelLined) + } + } +} + +// MARK: - Font Extension +extension Font { + static func pfFont(_ style: TypographyStyle) -> Font { + switch style { + case .largeTitle: + return .custom(PFFontWeight.extraBold.fontName, size: 34) + case .title: + return .custom(PFFontWeight.bold.fontName, size: 28) + case .title2: + return .custom(PFFontWeight.bold.fontName, size: 22) + case .title3: + return .custom(PFFontWeight.bold.fontName, size: 20) + case .headline: + return .custom(PFFontWeight.extraBold.fontName, size: 17) + case .body: + return .custom(PFFontWeight.bold.fontName, size: 17) + case .callout: + return .custom(PFFontWeight.bold.fontName, size: 16) + case .subheadline: + return .custom(PFFontWeight.extraBold.fontName, size: 15) + case .caption: + return .custom(PFFontWeight.extraBold.fontName, size: 12) + case .caption2: + return .custom(PFFontWeight.bold.fontName, size: 12) + case .label: + return .custom(PFFontWeight.bold.fontName, size: 11) + case .labelLined: + return .custom(PFFontWeight.regular.fontName, size: 11) + } + } + + static func pfCustom(_ weight: PFFontWeight, size: CGFloat) -> Font { + return .custom(weight.fontName, size: size) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/View+.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/View+.swift new file mode 100644 index 00000000..7476e977 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Extensions/View+.swift @@ -0,0 +1,48 @@ +// +// View+.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/22/26. +// + +import SwiftUI + +extension View { + func withTapSound() -> some View { + buttonStyle(.soundTap) + } + + func toast(isShowing: Binding, message: String, duration: Double = 1.5) -> some View { + self.modifier(Toast(isShowing: isShowing, message: message, duration: duration)) + } + + func pauseGameStyle( + isGameViewDisappeared: Binding, + height: CGFloat, + onLeave: @escaping () -> Void, + onPause: @escaping () -> Void, + onResume: @escaping () -> Void + ) -> some View { + self.modifier( + GamePauseWrapper( + isGameViewDisappeared: isGameViewDisappeared, + height: height, + onLeave: onLeave, + onPause: onPause, + onResume: onResume + ) + ) + } + + func longPressRepeat( + isLongPressing: Binding, + isDisabled: Bool, + onLongPressRepeat: (() -> Bool)? + ) -> some View { + modifier(LongPressRepeatModifier( + isLongPressing: isLongPressing, + isDisabled: isDisabled, + onLongPressRepeat: onLongPressRepeat + )) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/DodgeGame.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/DodgeGame.swift new file mode 100644 index 00000000..59e4342f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/DodgeGame.swift @@ -0,0 +1,234 @@ +// +// DodgeGame.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +final class DodgeGame: Game { + typealias ActionInput = DropItem.DropItemType + + /// 게임 종류 + var kind: GameType = .dodge + /// 사용자 정보 + var user: User + /// 피버 시스템 + var feverSystem: FeverSystem = FeverSystem( + decreaseInterval: Policy.Fever.decreaseInterval, + decreasePercentPerTick: Policy.Fever.Dodge.decreasePercent + ) + /// 버프 시스템 + var buffSystem: BuffSystem = BuffSystem() + /// 캐릭터 애니메이션 시스템 + var animationSystem: CharacterAnimationSystem? + + // MARK: - Game Systems + /// 모션 시스템 (기기 기울기 감지 및 캐릭터 이동) + let motionSystem: MotionSystem + /// 게임 코어 시스템 (낙하물 생성, 충돌 감지) + let gameCore: DodgeGameCore + + // MARK: - Game State + /// 현재 연속 회피 콤보 + var currentCombo: Int = 0 + + /// 플레이어 위치 동기화 타이머 (120fps) + private var positionSyncTimer: Timer? + /// 골드 변화 시 호출되는 콜백 핸들러 + private var onGoldChangedHandler: (Int) -> Void + + init( + user: User, + gameAreaSize: CGSize, + onGoldChanged: @escaping (Int) -> Void, + animationSystem: CharacterAnimationSystem? = nil + ) { + self.user = user + self.onGoldChangedHandler = onGoldChanged + self.animationSystem = animationSystem + + // 게임 시스템 초기화 (크기 필수 전달) + self.motionSystem = MotionSystem(screenLimit: gameAreaSize.width / 2) + self.gameCore = DodgeGameCore(screenWidth: gameAreaSize.width, screenHeight: gameAreaSize.height) + + // 충돌 콜백 설정 + gameCore.onCollision = { [weak self] itemType in + guard let self = self else { return } + Task { + let goldDelta = await self.didPerformAction(itemType) + await MainActor.run { + self.onGoldChangedHandler(goldDelta) + } + } + } + + // 버그가 땅에 닿았을 때 콜백 설정 + gameCore.onBugReachedGround = { [weak self] in + guard let self = self else { return } + Task { + let goldDelta = self.didDodgeBug() + await MainActor.run { + self.onGoldChangedHandler(goldDelta) + } + } + } + + // 플레이어 위치 동기화 타이머 시작 (120fps) + positionSyncTimer = Timer.scheduledTimer(withTimeInterval: 1.0 / Policy.Game.Dodge.updateFPS, repeats: true) { [weak self] _ in + guard let self = self else { return } + self.gameCore.playerX = self.motionSystem.characterX + } + } + + deinit { + positionSyncTimer?.invalidate() + positionSyncTimer = nil + } + + /// 게임 시작 (피버 시스템 및 게임 코어 타이머 활성화) + func startGame() { + // 콤보 초기화 + currentCombo = 0 + feverSystem.start() + gameCore.start() + } + + /// 게임 중지 (모든 타이머 정지 및 낙하물 제거) + func stopGame() { + buffSystem.stop() + feverSystem.stop() + gameCore.stop() + } + + /// 게임 일시정지 (피버, 버프 시스템 보존) + func pauseGame() { + feverSystem.pause() + buffSystem.pause() + gameCore.stop() + } + + /// 게임 재개 + func resumeGame() { + feverSystem.resume() + buffSystem.resume() + gameCore.start() + } + + /// 게임 영역 크기 업데이트 (화면 크기 변경 시 호출) + func configure(gameAreaSize: CGSize) { + gameCore.screenWidth = gameAreaSize.width + gameCore.screenHeight = gameAreaSize.height + motionSystem.screenLimit = gameAreaSize.width / 2 + } + + /// 골드 변화 콜백 핸들러 설정 + func setGoldChangedHandler(_ handler: @escaping (Int) -> Void) { + self.onGoldChangedHandler = handler + } + + /// 버그 회피 성공 처리 + /// - Returns: 획득한 골드 + func didDodgeBug() -> Int { + // 콤보 증가 + currentCombo += 1 + + // 피버 증가 + feverSystem.gainFever(Policy.Fever.Dodge.gainPerBugDodge) + + // 기본 골드 계산 + let baseGold = getBaseGold() + + // 1.0배 획득 + let gainGold = Int(Double(baseGold) * Policy.Game.Dodge.bugDodgeGoldMultiplier) + user.wallet.addGold(gainGold) + /// 누적 재산 업데이트 + user.record.record(.earnMoney(gainGold)) + /// 버그 회피 기록 (콤보 전달) + user.record.record(.dodgeBugAvoid(currentCombo: currentCombo)) + + // 재화 획득 시 캐릭터 웃게 만들기 + animationSystem?.playSmile() + return gainGold + } + + /// 아이템 충돌 처리 + /// - Parameter type: 충돌한 아이템 타입 + /// - Returns: 획득/손실한 골드 (손실은 음수) + func didPerformAction(_ input: DropItem.DropItemType) async -> Int { + switch input { + case .smallGold: + // 피버 증가 + feverSystem.gainFever(Policy.Fever.Dodge.gainPerSmallGold) + + // 기본 골드 계산 + let baseGold = getBaseGold() + + // 0.8배 획득 + let gainGold = Int(Double(baseGold) * Policy.Game.Dodge.smallGoldMultiplier) + user.wallet.addGold(gainGold) + /// 골드 수집 기록 + user.record.record(.dodgeGoldCollect) + /// 누적 재산 업데이트 + user.record.record(.earnMoney(gainGold)) + + // 재화 획득 시 캐릭터 웃게 만들기 + animationSystem?.playSmile() + SoundService.shared.trigger(.coinCollect) + return gainGold + + case .largeGold: + // 피버 증가 + feverSystem.gainFever(Policy.Fever.Dodge.gainPerLargeGold) + + // 기본 골드 계산 + let baseGold = getBaseGold() + + // 1.2배 획득 + let gainGold = Int(Double(baseGold) * Policy.Game.Dodge.largeGoldMultiplier) + user.wallet.addGold(gainGold) + /// 골드 수집 기록 + user.record.record(.dodgeGoldCollect) + /// 누적 재산 업데이트 + user.record.record(.earnMoney(gainGold)) + + // 재화 획득 시 캐릭터 웃게 만들기 + animationSystem?.playSmile() + SoundService.shared.trigger(.coinCollect) + return gainGold + + case .bug: + // 콤보 리셋 + currentCombo = 0 + + // 피버 감소 + feverSystem.gainFever(Policy.Fever.Dodge.lossPerBugHit) + + // 기본 골드 계산 + let baseGold = getBaseGold() + + // 골드 손실 + let loseGold = Int(Double(baseGold) * Policy.Game.Dodge.bugHitLossGoldMultiplier) + user.wallet.spendGold(loseGold) + /// 실패 기록 + user.record.record(.dodgeFail) + + SoundService.shared.trigger(.bugHit) + HapticService.shared.trigger(.error) + return -loseGold + } + } +} + +private extension DodgeGame { + func getBaseGold() -> Int { + let baseGold = Calculator.calculateGoldPerAction( + game: kind, + user: user, + feverMultiplier: feverSystem.feverMultiplier, + buffMultiplier: buffSystem.multiplier + ) + return baseGold + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/DodgeGameCore.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/DodgeGameCore.swift new file mode 100644 index 00000000..01686301 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/DodgeGameCore.swift @@ -0,0 +1,173 @@ +// +// DodgeGameCore.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/13/26. +// + +import Foundation +import SwiftUI + +private enum Constant { + static let itemRemovalOffset: CGFloat = 50 +} + +@Observable +final class DodgeGameCore { + // MARK: - 게임 설정 + /// 게임 업데이트 주사율 (120fps) + private let updateInterval: TimeInterval = 1.0 / Policy.Game.Dodge.updateFPS + /// 낙하물 생성 간격 (초) + private let spawnInterval: TimeInterval = Policy.Game.Dodge.spawnInterval + /// 낙하 속도 (120fps 기준) + private let fallSpeed: CGFloat = Policy.Game.Dodge.fallSpeed + /// 플레이어 크기 + private let playerSize: CGSize = CGSize(width: 40, height: 40) + + // MARK: - 게임 상태 + /// 현재 화면에 떨어지고 있는 아이템 목록 + var fallingItems: [FallingItem] = [] + /// 게임 실행 여부 + var isRunning: Bool = false + + /// 낙하물 생성 타이머 + private var spawnTimer: Timer? + /// 게임 업데이트 타이머 + private var updateTimer: Timer? + + // MARK: - Public Properties + /// 충돌 발생 시 호출되는 콜백 + var onCollision: ((DropItem.DropItemType) -> Void)? + /// 버그가 땅에 닿았을 때 호출되는 콜백 + var onBugReachedGround: (() -> Void)? + /// 플레이어의 X 위치 (MotionSystem에서 동기화) + var playerX: CGFloat = 0 + /// 게임 영역 너비 + var screenWidth: CGFloat + /// 게임 영역 높이 + var screenHeight: CGFloat + + init(screenWidth: CGFloat, screenHeight: CGFloat) { + self.screenWidth = screenWidth + self.screenHeight = screenHeight + } + + deinit { + stop() + } + + /// 게임 시작 (낙하물 생성 및 업데이트 타이머 활성화) + func start() { + guard !isRunning else { return } + isRunning = true + + // 낙하물 생성 타이머 (0.5초 간격) + spawnTimer = Timer.scheduledTimer(withTimeInterval: spawnInterval, repeats: true) { [weak self] _ in + self?.spawnItem() + } + + // 낙하물 업데이트 타이머 (120fps) + updateTimer = Timer.scheduledTimer(withTimeInterval: updateInterval, repeats: true) { [weak self] _ in + self?.updateItems() + } + } + + /// 게임 중지 (모든 타이머 정지 및 낙하물 제거) + func stop() { + isRunning = false + spawnTimer?.invalidate() + spawnTimer = nil + updateTimer?.invalidate() + updateTimer = nil + fallingItems.removeAll() + } +} + +// MARK: - Private Methods +private extension DodgeGameCore { + /// 새로운 낙하물 생성 + func spawnItem() { + // 랜덤 타입 생성 + let randomValue = Int.random(in: 0..<100) + let type: DropItem.DropItemType + if randomValue < Policy.Game.Dodge.largeGoldSpawnRate { + type = .largeGold + } else if randomValue < Policy.Game.Dodge.largeGoldSpawnRate + Policy.Game.Dodge.smallGoldSpawnRate { + type = .smallGold + } else { + type = .bug + } + + // 랜덤 X 위치 생성 (게임 영역 내) + let randomX = CGFloat.random(in: -screenWidth/2...screenWidth/2) + let item = FallingItem( + type: type, + position: CGPoint(x: randomX, y: -screenHeight/2) + ) + fallingItems.append(item) + } + + func updateItems() { + var indicesToRemove: [Int] = [] + let removalThreshold = screenHeight/2 + Constant.itemRemovalOffset + + // 위치 업데이트 및 제거 대상 수집 + for index in fallingItems.indices { + fallingItems[index].updatePosition(by: fallSpeed) + + if fallingItems[index].position.y > removalThreshold { + indicesToRemove.append(index) + } + } + + // 충돌 감지 (제거 전에 먼저 체크) + checkCollisions() + + // 버그가 땅에 닿았는지 확인 (충돌하지 않고 화면을 벗어난 경우) + for index in indicesToRemove where fallingItems[index].type == .bug { + onBugReachedGround?() + } + + // 역순으로 제거 + for index in indicesToRemove.reversed() { + fallingItems.remove(at: index) + } + } + + /// 플레이어와 낙하물 간의 충돌 감지 + func checkCollisions() { + var collidedIndices: [Int] = [] + + // 플레이어 Y 위치 계산 (화면 하단에서 25%) + let playerYOffset = screenHeight * 0.25 + + for (index, item) in fallingItems.enumerated() { + // 플레이어 영역 계산 + let playerRect = CGRect( + x: playerX - playerSize.width / 2, + y: screenHeight/2 - playerYOffset - playerSize.height / 2, + width: playerSize.width, + height: playerSize.height + ) + + // 아이템 영역 계산 + let itemRect = CGRect( + x: item.position.x - item.size.width / 2, + y: item.position.y - item.size.height / 2, + width: item.size.width, + height: item.size.height + ) + + // 충돌 확인 + if playerRect.intersects(itemRect) { + collidedIndices.append(index) + onCollision?(item.type) + } + } + + // 충돌한 아이템 제거 (역순으로 제거하여 인덱스 오류 방지) + for index in collidedIndices.reversed() { + fallingItems.remove(at: index) + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/FallingItem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/FallingItem.swift new file mode 100644 index 00000000..ccab2e6f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/DodgeGame/FallingItem.swift @@ -0,0 +1,25 @@ +// +// FallingItem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/14/26. +// + +import Foundation + +struct FallingItem: Identifiable { + /// 고유 식별자 + let id = UUID() + /// 아이템 타입 (smallGold, largeGold, bug) + let type: DropItem.DropItemType + /// 화면 상의 위치 (중심점 기준) + var position: CGPoint + /// 아이템 크기 + let size: CGSize = CGSize(width: 24, height: 24) + + /// 아이템 위치를 업데이트 + /// - Parameter offset: Y축 이동 오프셋 + mutating func updatePosition(by offset: CGFloat) { + position.y += offset + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/Game.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/Game.swift new file mode 100644 index 00000000..ac6d987d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/Game.swift @@ -0,0 +1,68 @@ +// +// Game.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +/// 게임 프로토콜 +protocol Game { + /// action parameter + associatedtype ActionInput + /// 게임 종류 + var kind: GameType { get set } + /// 유저 + var user: User { get set } + /// 피버 시스템 + var feverSystem: FeverSystem { get set } + /// 버프 시스템 + var buffSystem: BuffSystem { get set } + /// 캐릭터 애니메이션 시스템 + var animationSystem: CharacterAnimationSystem? { get set } + + /// 게임 시작 + func startGame() + /// 게임 종료 + func stopGame() + /// 게임 일시정지 + func pauseGame() + /// 게임 재개 + func resumeGame() + /// action 수행 + @discardableResult + func didPerformAction(_ input: ActionInput) async -> Int +} + +/// 게임 타입 +enum GameType: Int, CaseIterable { + case tap = 0 + case language = 1 + case dodge = 2 + case stack = 3 + + /// 화면에 표시될 제목 + var displayTitle: String { + switch self { + case .tap: "코드짜기" + case .language: "언어 맞추기" + case .dodge: "버그 피하기" + case .stack: "데이터 쌓기" + } + } + + /// 업무 이미지 명 + var imageName: String { + switch self { + case .tap: + return "work_tap" + case .language: + return "work_language" + case .dodge: + return "work_dodge" + case .stack: + return "work_stack" + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/LanguageGame.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/LanguageGame.swift new file mode 100644 index 00000000..01d1d6fa --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/LanguageGame.swift @@ -0,0 +1,208 @@ +// +// LanguageGame.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +enum LanguageType: String, CaseIterable { + case swift = "Swift" + case kotlin = "Kotlin" + case dart = "Dart" + case python = "Python" + case empty = "" + + var imageName: String { + switch self { + case .swift: return "language_swift" + case .kotlin: return "language_kotlin" + case .dart: return "language_dart" + case .python: return "language_python" + case .empty: return "" + } + } + + var backgroundColorName: String { + switch self { + case .swift: return "PastelYellow" + case .kotlin: return "PastelPink" + case .dart: return "PastelBlue" + case .python: return "PastelGreen" + case .empty: return "" + } + } + + static func random() -> Self { + return LanguageType.allCases + .filter { $0 != .empty}.randomElement() ?? .swift + } +} + +enum LanguageItemState { + case completed + case active + case upcoming + case empty +} + +@Observable +final class LanguageGame: Game { + typealias ActionInput = LanguageType + var kind: GameType = .language + var user: User + var feverSystem: FeverSystem + var buffSystem: BuffSystem + var animationSystem: CharacterAnimationSystem? + + let itemCount: Int + + // 한 화면에 보여지는 아이템 리스트 + var itemList: [LanguageItem] = [] + + // 활성화 아이템 외에 양쪽에 보여지는 아이템의 개수 + var leadingAndTrailingItemCount: Int { + itemCount / 2 + } + + init( + user: User, + feverSystem: FeverSystem = FeverSystem( + decreaseInterval: Policy.Fever.decreaseInterval, + decreasePercentPerTick: Policy.Fever.Language.decreasePercent + ), + buffSystem: BuffSystem, + itemCount: Int, + animationSystem: CharacterAnimationSystem? = nil + ) { + self.user = user + self.feverSystem = feverSystem + self.buffSystem = buffSystem + self.itemCount = itemCount + self.animationSystem = animationSystem + self.itemList = makeInitialItemList() + } + + func startGame() { + feverSystem.start() + if itemList.isEmpty { + itemList = makeInitialItemList() + } + } + + func stopGame() { + feverSystem.stop() + buffSystem.stop() + itemList = [] + } + + /// 게임 일시정지 (피버, 버프 시스템 보존) + func pauseGame() { + feverSystem.pause() + buffSystem.pause() + } + + /// 게임 재개 + func resumeGame() { + feverSystem.resume() + buffSystem.resume() + } + + func didPerformAction(_ input: LanguageType) async -> Int { + // Task가 취소되었으면 즉시 종료 + guard !Task.isCancelled else { return 0 } + + // 게임 종료 후 버튼 탭 크래시 방지 + guard itemList.count > leadingAndTrailingItemCount else { return 0 } + + let isSuccess = languageButtonTapHandler(tappedItemType: input) + + // 비즈니스 로직 실행 전 다시 한 번 취소 확인 + guard !Task.isCancelled else { return 0 } + + feverSystem + .gainFever( + isSuccess ? Policy.Fever.Language.gainPerCorrect : Policy.Fever.Language.lossPerIncorrect + ) + let gainGold = Calculator.calculateGoldPerAction( + game: kind, + user: user, + feverMultiplier: feverSystem.feverMultiplier, + buffMultiplier: buffSystem.multiplier + ) + if isSuccess { + user.wallet.addGold(gainGold) + /// 정답 횟수 기록 + user.record.record(.languageCorrect) + /// 누적 재산 업데이트 + user.record.record(.earnMoney(gainGold)) + // 재화 획득 시 캐릭터 웃게 만들기 + animationSystem?.playSmile() + return gainGold + } + user.wallet.spendGold(Int(Double(gainGold) * Policy.Game.Language.incorrectGoldLossMultiplier)) + /// 오답 횟수 기록 + user.record.record(.languageIncorrect) + return Int(Double(gainGold) * Policy.Game.Language.incorrectGoldLossMultiplier) * -1 + } + + private func languageButtonTapHandler(tappedItemType: LanguageType) -> Bool { + let activeItem = itemList[leadingAndTrailingItemCount] + + guard activeItem.languageType == tappedItemType else { + return false + } + + // 1. 복제 후 처음 요소 제거 + var newItems = itemList + newItems.removeFirst() + // 2. 새 요소를 마지막에 추가 + newItems.append(.init(languageType: LanguageType.random(), state: .upcoming)) + // 3. 요소 업데이트 + self.itemList = newItems + // 4. 상태 업데이트 + updateLanguageItemList() + + return true + } + + private func makeInitialItemList() -> [LanguageItem] { + var items: [LanguageItem] = [] + let activeIndex = leadingAndTrailingItemCount // 중앙 인덱스 + + for index in 0.. [QuizQuestion] { + guard let url = Bundle.main.url(forResource: fileName, withExtension: "tsv") else { + throw QuizDataLoaderError.fileNotFound + } + + do { + let data = try String(contentsOf: url, encoding: .utf8) + + // TSV 파싱: 탭으로 구분 + let lines = data.components(separatedBy: .newlines) + + // 헤더 제거 및 빈 줄 필터링 + let dataLines = lines.dropFirst().filter { !$0.trimmingCharacters(in: .whitespaces).isEmpty } + + var questions: [QuizQuestion] = [] + + for line in dataLines { + // 탭으로 분리 + let columns = line.components(separatedBy: "\t") + let trimColumns = columns.map { $0.trimmingCharacters(in: .whitespaces) } + + // TSV 형식: 문제 번호\t문제명\t선지1\t선지2\t선지3\t선지4\t정답\t해설 + guard columns.count == 8, + let id = Int(trimColumns[0]), + let correctAnswer = Int(trimColumns[6]) + else { + continue + } + + let question = QuizQuestion( + id: id, + question: trimColumns[1], + options: [trimColumns[2], trimColumns[3], trimColumns[4], trimColumns[5]], + correctAnswerIndex: correctAnswer - 1, // 1-based -> 0-based + explanation: trimColumns[7] + ) + questions.append(question) + } + + guard !questions.isEmpty else { throw QuizDataLoaderError.invalidQuestions } + return questions + } catch let error as QuizDataLoaderError { + throw error + } catch { + throw QuizDataLoaderError.decodingError(error) + } + } + + /// 문제 풀에서 N개의 랜덤 문제를 선택합니다 + /// - Parameters: + /// - questions: 사용 가능한 문제 풀 + /// - count: 선택할 문제 개수 + /// - Returns: 랜덤하게 선택된 문제 배열 + static func selectRandomQuestions(from questions: [QuizQuestion], count: Int) -> [QuizQuestion] { + return Array(questions.shuffled().prefix(count)) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGame.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGame.swift new file mode 100644 index 00000000..336ba6c5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGame.swift @@ -0,0 +1,264 @@ +// +// QuizGame.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation +import Observation + +private enum Constant { + /// 퀴즈 데이터 파일명 (확장자 제외) + static let questionLoadFileName = "QuizData" +} + +@Observable +final class QuizGame { + /// 게임을 플레이하는 사용자 + var user: User + /// 리소스 파일에서 로드한 전체 문제 풀 + private var allQuestions: [QuizQuestion] = [] + /// 현재 게임 세션에서 출제된 문제들 (게임당 3문제) + private(set) var currentGameQuestions: [QuizQuestion] = [] + /// 현재 풀고 있는 문제의 인덱스 (0부터 시작, 0~2) + private(set) var currentQuestionIndex: Int = 0 + /// 현재 표시 중인 문제 + /// - currentQuestionIndex가 유효한 범위일 때만 반환 + var currentQuestion: QuizQuestion? { + guard currentQuestionIndex < currentGameQuestions.count else { return nil } + return currentGameQuestions[currentQuestionIndex] + } + /// 사용자가 선택한 답안의 인덱스 (0~3) + /// - nil이면 아직 답을 선택하지 않은 상태 + private(set) var selectedAnswerIndex: Int? + /// 현재 문제의 정답/오답/타임아웃 결과 + /// - 답안 제출 후에만 값이 설정됨 + private(set) var currentAnswerResult: QuizAnswerResult? + /// 문제당 타이머 + /// - 문제 시작 시 생성되고, 답안 제출 또는 타임아웃 시 중지됨 + private var questionTimer: Timer? + /// 현재 문제의 남은 시간 (초) + /// - 매 초마다 1씩 감소 + private(set) var remainingSeconds: Int = Policy.Game.Quiz.secondsPerQuestion + /// 게임의 현재 진행 단계 + private(set) var phase: QuizGamePhase = .loading + /// 현재 세션에서 맞춘 문제 개수 + private(set) var correctAnswersCount: Int = 0 + + /// 모든 상태 정보 + var state: QuizGameState { + QuizGameState( + totalDiamondsEarned: correctAnswersCount * Policy.Game.Quiz.diamondsPerCorrect, + progressText: "\(currentQuestionIndex + 1)/\(Policy.Game.Quiz.questionsPerGame)", + nextButtonTitle: currentQuestionIndex >= currentGameQuestions.count - 1 ? "보상받기" : "다음으로", + isSubmitEnabled: selectedAnswerIndex != nil && phase == .questionInProgress, + timerProgress: Double(remainingSeconds) / Double(Policy.Game.Quiz.secondsPerQuestion), + phase: phase, + currentQuestion: currentQuestion, + selectedAnswerIndex: selectedAnswerIndex, + currentAnswerResult: currentAnswerResult, + remainingSeconds: remainingSeconds, + correctAnswersCount: correctAnswersCount + ) + } + + init(user: User) { + self.user = user + loadQuestions() + } + + deinit { + stopTimer() + } + + /// 새로운 게임 세션 시작 + /// - 전체 문제 풀에서 랜덤하게 3문제 선택 + /// - 게임 상태 초기화 (점수, 인덱스 등) + /// - 첫 번째 문제 자동 시작 + func startGame() { + // 랜덤하게 3문제 선택 + currentGameQuestions = QuizDataLoader.selectRandomQuestions( + from: allQuestions, + count: Policy.Game.Quiz.questionsPerGame + ) + + // 게임 상태 초기화 + currentQuestionIndex = 0 + correctAnswersCount = 0 + phase = .ready + + // 첫 문제 시작 + startQuestion() + } + + /// 게임 강제 종료 + /// - 타이머를 중지하고 게임 상태를 완료로 변경 + /// - 사용자가 중도 포기하는 경우 호출 + func stopGame() { + stopTimer() + phase = .completed + } + + /// 답 선택 + /// - Parameter index: 선택한 답안의 인덱스 (0~3) + /// - Note: 문제 풀이 중일 때만 선택 가능 + func selectAnswer(_ index: Int) { + guard phase == .questionInProgress else { return } + guard (0...3).contains(index) else { return } + selectedAnswerIndex = index + } + + /// 선택한 답안 해제 + /// - Note: 문제 풀이 중일 때만 선택 해제 가능 + func deselectAnswer() { + guard phase == .questionInProgress else { return } + selectedAnswerIndex = nil + } + + /// 선택한 답안 제출 + /// - 문제 풀이 중이고, 답을 선택한 상태일 때만 제출 가능 + /// - 제출 후 타이머 중지 및 결과 표시 + func submitSelectedAnswer() { + guard phase == .questionInProgress else { return } + guard let answerIndex = selectedAnswerIndex else { return } + submitAnswer(answerIndex) + } + + /// 다음 문제로 진행 또는 게임 종료 + /// - 해설 표시 중일 때만 호출 가능 + /// - 마지막 문제가 아니면: 다음 문제 시작 + /// - 마지막 문제면: 게임 완료 및 보상 지급 + func proceedToNextQuestion() { + guard phase == .showingExplanation else { return } + + currentQuestionIndex += 1 + + if currentQuestionIndex >= currentGameQuestions.count { + // 모든 문제 완료 - 보상 지급 및 게임 종료 + completeGame() + } else { + // 다음 문제 시작 + startQuestion() + } + } +} + +private extension QuizGame { + + /// 리소스 파일에서 문제 데이터 로드 + /// - TSV 파일에서 모든 문제를 읽어와 allQuestions에 저장 + /// - 로드 성공 시 게임 상태를 ready로 변경 + /// - 실패 시 에러 메시지와 함께 error 상태로 변경 + func loadQuestions() { + do { + allQuestions = try QuizDataLoader.loadQuestions(from: Constant.questionLoadFileName) + phase = .ready + } catch { + phase = .error("문제를 불러오는데 실패했습니다: \(error.localizedDescription)") + } + } + + /// 새로운 문제 시작 + /// - 선택한 답안, 결과, 타이머를 초기화 + /// - 게임 상태를 questionInProgress로 변경 + /// - 타이머 시작 + func startQuestion() { + guard currentQuestion != nil else { + phase = .error("문제를 찾을 수 없습니다") + return + } + + // 상태 초기화 + selectedAnswerIndex = nil + currentAnswerResult = nil + remainingSeconds = Policy.Game.Quiz.secondsPerQuestion + phase = .questionInProgress + + // 타이머 시작 + startTimer() + } + + /// 답안 제출 처리 + /// - Parameter answerIndex: 제출할 답안의 인덱스 + /// - 타이머 중지 + /// - 정답 확인 후 결과 저장 + /// - 정답일 경우 correctAnswersCount 증가 + /// - 게임 상태를 showingExplanation으로 변경 + func submitAnswer(_ answerIndex: Int) { + stopTimer() + + guard let question = currentQuestion else { return } + + // 정답 확인 + let isCorrect = answerIndex == question.correctAnswerIndex + currentAnswerResult = isCorrect ? .correct : .incorrect + + if isCorrect { + correctAnswersCount += 1 + } + + SoundService.shared.trigger(isCorrect ? .languageCorrect : .languageWrong) + if !isCorrect { + HapticService.shared.trigger(.error) + } + phase = .showingExplanation + } + + /// 게임 완료 처리 + /// - 타이머 중지 + /// - 획득한 다이아를 사용자 지갑에 추가 + /// - 게임 상태를 completed로 변경 + func completeGame() { + stopTimer() + + // 다이아 보상 지급 (정답당 5개) + let diamondsToAward = correctAnswersCount * Policy.Game.Quiz.diamondsPerCorrect + user.wallet.addDiamond(diamondsToAward) + + phase = .completed + } + + /// 타이머 시작 + /// - 기존 타이머가 있으면 먼저 정지 + /// - 1초마다 timerTick() 호출하는 타이머 생성 + /// - weak self를 사용하여 메모리 누수 방지 + func startTimer() { + stopTimer() + + questionTimer = Timer.scheduledTimer(withTimeInterval: 1.0, repeats: true) { [weak self] _ in + self?.timerTick() + } + } + + /// 타이머 중지 및 정리 + /// - 타이머 무효화 + /// - 타이머 참조 해제 + func stopTimer() { + questionTimer?.invalidate() + questionTimer = nil + } + + /// 매 초마다 호출되는 타이머 틱 + /// - 남은 시간 1초 감소 + /// - 시간이 0이 되면 타임아웃 처리 + func timerTick() { + remainingSeconds -= 1 + + if remainingSeconds <= 0 { + handleTimeout() + } + } + + /// 시간 초과 처리 + /// - 타이머 중지 + /// - 결과를 timeout으로 설정 + /// - 해설 화면으로 전환 (오답과 동일한 처리) + func handleTimeout() { + stopTimer() + + currentAnswerResult = .timeout + HapticService.shared.trigger(.error) + phase = .showingExplanation + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGamePhase.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGamePhase.swift new file mode 100644 index 00000000..71ba9cb3 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGamePhase.swift @@ -0,0 +1,21 @@ +// +// QuizGamePhase.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/22/26. +// + +enum QuizGamePhase: Equatable { + /// 문제 로딩 중 + case loading + /// 게임 시작 준비 완료 + case ready + /// 문제 풀이 진행 중 (타이머 작동) + case questionInProgress + /// 해설 표시 중 + case showingExplanation + /// 게임 완료 (모든 문제 풀이 완료) + case completed + /// 에러 발생 + case error(String) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGameState.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGameState.swift new file mode 100644 index 00000000..c800460f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizGameState.swift @@ -0,0 +1,31 @@ +// +// QuizGameState.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/22/26. +// + +struct QuizGameState { + /// 현재 세션에서 획득한 총 다이아 개수 + let totalDiamondsEarned: Int + /// 진행도 텍스트 (예: "1/3", "2/3") + let progressText: String + /// 다음 버튼의 텍스트 ("다음" 또는 "결과 보기") + let nextButtonTitle: String + /// 제출 버튼 활성화 여부 + let isSubmitEnabled: Bool + /// 타이머 프로그래스 바 값 (0.0 ~ 1.0) + let timerProgress: Double + /// 게임 진행 단계 + let phase: QuizGamePhase + /// 현재 문제 + let currentQuestion: QuizQuestion? + /// 선택한 답안 인덱스 + let selectedAnswerIndex: Int? + /// 현재 답안 결과 + let currentAnswerResult: QuizAnswerResult? + /// 남은 시간 (초) + let remainingSeconds: Int + /// 맞춘 문제 개수 + let correctAnswersCount: Int +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizQuestion.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizQuestion.swift new file mode 100644 index 00000000..7809534d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/QuizGame/QuizQuestion.swift @@ -0,0 +1,25 @@ +// +// QuizQuestion.swift +// SoloDeveloperTraining +// +// Created by Claude on 1/21/26. +// + +import Foundation + +/// 퀴즈 문제 데이터 모델 +struct QuizQuestion: Codable, Identifiable { + let id: Int + let question: String + let options: [String] + let correctAnswerIndex: Int + let explanation: String + + enum CodingKeys: String, CodingKey { + case id + case question + case options + case correctAnswerIndex = "correct_answer" + case explanation + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/StackBlock.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/StackBlock.swift new file mode 100644 index 00000000..be28f6c7 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/StackBlock.swift @@ -0,0 +1,64 @@ +// +// StackBlock.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-14. +// + +import Foundation + +enum BlockType: CaseIterable { + case blue + case green + case orange + case purple + case red + case yellow + case bomb + case bomb2 + + var size: CGSize { + switch self { + case .blue: return CGSize(width: 60, height: 20) + case .green: return CGSize(width: 70, height: 22) + case .orange: return CGSize(width: 50, height: 17) + case .purple: return CGSize(width: 80, height: 25) + case .red: return CGSize(width: 40, height: 19) + case .yellow: return CGSize(width: 90, height: 21) + case .bomb: return CGSize(width: 40, height: 18) + case .bomb2: return CGSize(width: 30, height: 40) + } + } + + var imageName: String { + switch self { + case .blue: return "stack_block_blue" + case .green: return "stack_block_green" + case .orange: return "stack_block_orange" + case .purple: return "stack_block_purple" + case .red: return "stack_block_red" + case .yellow: return "stack_block_yellow" + case .bomb: return "stack_block_bomb" + case .bomb2: return "stack_block_bomb2" + } + } + + /// 폭탄 블록인지 확인 + var isBomb: Bool { + self == .bomb || self == .bomb2 + } +} + +struct StackBlock { + let type: BlockType + var positionX: CGFloat + var positionY: CGFloat + + var width: CGFloat { + type.size.width + } + + var height: CGFloat { + type.size.height + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/StackGame.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/StackGame.swift new file mode 100644 index 00000000..cff174d0 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/StackGame.swift @@ -0,0 +1,183 @@ +// +// StackGame.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 1/14/26. +// + +import Foundation + +private enum Constant { + enum Position { + static let initialBlockYPosition: CGFloat = 20 + } +} + +@Observable +final class StackGame: Game { + typealias ActionInput = BlockType + + var kind: GameType = .stack + var user: User + var feverSystem: FeverSystem = .init( + decreaseInterval: Policy.Fever.decreaseInterval, + decreasePercentPerTick: Policy.Fever.Stack.decreasePercent + ) + var buffSystem: BuffSystem = .init() + var animationSystem: CharacterAnimationSystem? + var screenSize: CGSize = .init(width: 0, height: 0) + + private(set) var score: Int = 0 + private(set) var blocks: [StackBlock] = [] + private(set) var currentBlock: StackBlock? + private(set) var previousBlock: StackBlock? + + init(user: User, animationSystem: CharacterAnimationSystem? = nil) { + self.user = user + self.animationSystem = animationSystem + } + + func startGame() { + feverSystem.start() + score = 0 + blocks = [] + currentBlock = nil + previousBlock = nil + } + + func stopGame() { + feverSystem.stop() + buffSystem.stop() + } + + /// 게임 일시정지 (피버, 버프 시스템 보존) + func pauseGame() { + feverSystem.pause() + buffSystem.pause() + } + + /// 게임 재개 + func resumeGame() { + feverSystem.resume() + buffSystem.resume() + } + + /// 액션 수행 (Game 프로토콜 요구사항) + @discardableResult + func didPerformAction(_ input: BlockType) async -> Int { 0 } + + /// 초기 블록을 추가합니다 + /// 화면 크기를 기반으로 화면 중앙 하단에 배치합니다 + func addInitialBlock() { + let blockType = BlockType.blue + let initialBlock = StackBlock( + type: blockType, + positionX: screenSize.width / 2, + positionY: Constant.Position.initialBlockYPosition + ) + + blocks.append(initialBlock) + previousBlock = initialBlock + } + + /// 떨어뜨릴 블록을 생성합니다 + func spawnBlock(type: BlockType) { + currentBlock = StackBlock( + type: type, + positionX: 0, + positionY: 0 + ) + } + + /// 블록 정렬 여부를 확인합니다 + func checkAlignment() -> Bool { + guard + let currentBlock = currentBlock, + let previousBlock = previousBlock + else { return false } + + let previousLeft = previousBlock.positionX - previousBlock.width / 2 + let previousRight = previousBlock.positionX + previousBlock.width / 2 + let previousRange = previousLeft...previousRight + + return previousRange.contains(currentBlock.positionX) + } + + /// 현재 블록의 위치를 업데이트합니다 + func updateCurrentBlockPosition(positionX: CGFloat, positionY: CGFloat) { + currentBlock?.positionX = positionX + currentBlock?.positionY = positionY + } + + /// 블록이 성공적으로 배치되었을 때 처리 + func placeBlockSuccess() -> Int { + guard let block = currentBlock else { return 0 } + + blocks.append(block) + previousBlock = block + currentBlock = nil + + score += 1 + return applyReward() + } + + /// 블록 배치에 실패했을 때 처리 + func placeBlockFail() -> Int { + currentBlock = nil + return applyPenalty() + } + + /// 폭탄 블록이 성공적으로 배치되었을 때 처리 (패널티) + func placeBombSuccess() -> Int { + currentBlock = nil + return applyPenalty() + } + + /// 폭탄 블록 배치에 실패했을 때 처리 (보상) + func placeBombFail() -> Int { + currentBlock = nil + return applyReward() + } + + /// 보상을 적용합니다 (골드 획득, 피버 증가) + private func applyReward() -> Int { + let goldEarned = calculateGold() + user.wallet.addGold(goldEarned) + /// 성공 수 기록 + user.record.record(.stackingSuccess) + /// 누적 재산 업데이트 + user.record.record(.earnMoney(goldEarned)) + feverSystem.gainFever(Policy.Fever.Stack.gainPerSuccess) + + // 재화 획득 시 캐릭터 웃게 만들기 + animationSystem?.playSmile() + + return goldEarned + #if DEV_BUILD + print("💰 골드 획득: \(goldEarned), 총액: \(user.wallet.gold)") + #endif + } + + /// 패널티를 적용합니다 (골드 손실, 피버 감소) + private func applyPenalty() -> Int { + let goldLost = calculateGold() + user.wallet.spendGold(goldLost) + /// 실패 수 기록 + user.record.record(.stackingFail) + feverSystem.gainFever(Policy.Fever.Stack.lossPerFailure) + return -goldLost + #if DEV_BUILD + print("💸 골드 손실: \(goldLost), 총액: \(user.wallet.gold)") + #endif + } + + /// 현재 상태에 따른 골드 획득량을 계산합니다 + private func calculateGold() -> Int { + return Calculator.calculateGoldPerAction( + game: .stack, + user: user, + feverMultiplier: feverSystem.feverMultiplier, + buffMultiplier: buffSystem.multiplier + ) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/TapGame.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/TapGame.swift new file mode 100644 index 00000000..568ed0a8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Games/TapGame.swift @@ -0,0 +1,89 @@ +// +// TapGame.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +/// 탭 게임 클래스 +final class TapGame: Game { + typealias ActionInput = Void + + /// 게임 종류 + var kind: GameType + /// 사용자 정보 + var user: User + /// 피버 시스템 + var feverSystem: FeverSystem + /// 버프 시스템 + var buffSystem: BuffSystem + /// 캐릭터 애니메이션 시스템 + var animationSystem: CharacterAnimationSystem? + /// 인벤토리 + let inventory: Inventory + /// 일시정지 여부 (일시정지 시 탭 사운드 미재생용) + private(set) var isPaused: Bool = false + + /// 탭 게임 초기화 + init( + user: User, + feverSystem: FeverSystem = FeverSystem( + decreaseInterval: Policy.Fever.decreaseInterval, + decreasePercentPerTick: Policy.Fever.Tap.decreasePercent + ), + buffSystem: BuffSystem, + animationSystem: CharacterAnimationSystem? = nil + ) { + self.kind = .tap + self.user = user + self.feverSystem = feverSystem + self.buffSystem = buffSystem + self.animationSystem = animationSystem + self.inventory = user.inventory + } + + /// 게임 시작 + func startGame() { + feverSystem.start() + } + + /// 게임 종료 + func stopGame() { + feverSystem.stop() + } + + /// 게임 일시정지 (피버, 버프 시스템 보존) + func pauseGame() { + isPaused = true + feverSystem.pause() + buffSystem.pause() + } + + /// 게임 재개 + func resumeGame() { + isPaused = false + feverSystem.resume() + buffSystem.resume() + } + + /// 탭 액션 수행 및 골드 획득 + @discardableResult + func didPerformAction(_ input: Void = ()) async -> Int { + feverSystem.gainFever(Policy.Fever.Tap.gainPerTap) + let gainGold = Calculator.calculateGoldPerAction( + game: kind, + user: user, + feverMultiplier: feverSystem.feverMultiplier, + buffMultiplier: buffSystem.multiplier + ) + user.wallet.addGold(gainGold) + /// 탭 횟수 기록 + user.record.record(.tap()) + /// 누적 재산 업데이트 + user.record.record(.earnMoney(gainGold)) + + // 재화 획득 시 캐릭터 웃게 만들기 + animationSystem?.playSmile() + return gainGold + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Consumable.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Consumable.swift new file mode 100644 index 00000000..ca75e8f9 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Consumable.swift @@ -0,0 +1,106 @@ +// +// Consumable.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/8/26. +// + +import Foundation + +/// 소비 아이템 클래스 +@Observable +final class Consumable: Item { + // MARK: - Item + var displayTitle: String { + return type.displayTitle + " (보유:\(count)개)" + } + var description: String { + return "사용시 골드 획득량 \(type.buffMultiplier)배 증가" + } + var cost: Cost { + return type.cost + } + var imageName: String { + return type.imageName + } + var category: ItemCategory = .consumable + + // MARK: - Consumable + /// 소비 아이템 타입 + let type: ConsumableType + /// 보유 개수 + private(set) var count: Int + + /// 소비 아이템 초기화 + init(type: ConsumableType, count: Int) { + self.type = type + self.count = count + } + + /// 아이템 갯수 1 증가 + func addItem() { + count += 1 + } + + /// 아이템 갯수 1 감소 + func spendItem() { + count -= 1 + } +} + +/// 소비 아이템 종류 +enum ConsumableType { + /// 커피 + case coffee + /// 에너지 드링크 + case energyDrink + + /// 화면에 표시될 제목 + var displayTitle: String { + switch self { + case .coffee: + return "커피" + case .energyDrink: + return "박하스" + } + } + + /// 버프 가중치 + var buffMultiplier: Double { + switch self { + case .coffee: + return Policy.Consumable.Coffee.buffMultiplier + case .energyDrink: + return Policy.Consumable.EnergyDrink.buffMultiplier + } + } + + /// 지속시간 + var duration: Int { + switch self { + case .coffee: + return Policy.Consumable.Coffee.duration + case .energyDrink: + return Policy.Consumable.EnergyDrink.duration + } + } + + /// 비용 + var cost: Cost { + switch self { + case .coffee: + return .init(diamond: Policy.Consumable.Coffee.priceDiamond) + case .energyDrink: + return .init(diamond: Policy.Consumable.EnergyDrink.priceDiamond) + } + } + + var imageName: String { + switch self { + case .coffee: + return "icon_coffee" + case .energyDrink: + return "icon_energy_drink" + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Cost.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Cost.swift new file mode 100644 index 00000000..d90af7b7 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Cost.swift @@ -0,0 +1,22 @@ +// +// Cost.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/7/26. +// + +import Foundation + +/// 아이템 비용 모델 +struct Cost: Equatable { + /// 골드 비용 + let gold: Int + /// 다이아몬드 비용 + let diamond: Int + + /// 비용 초기화 + init(gold: Int = 0, diamond: Int = 0) { + self.gold = gold + self.diamond = diamond + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Equipment.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Equipment.swift new file mode 100644 index 00000000..e49cfc2a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Equipment.swift @@ -0,0 +1,282 @@ +// +// Equipment.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +/// 장비 아이템 클래스 +final class Equipment: Item { + // MARK: - Item + var displayTitle: String { + return tier.displayTitle + " " + type.displayTitle + } + var description: String { + guard canUpgrade else { + return "초당 골드 획득량 \(goldPerSecond.formatted)" + } + let nextTier = EquipmentTier(rawValue: tier.rawValue + 1) ?? .nationalTreasure + let nextEquipment = Equipment(type: type, tier: nextTier) + return "강화시 초당 골드 획득량 \(goldPerSecond.formatted) -> \(nextEquipment.goldPerSecond.formatted)" + } + var cost: Cost { + return tier.cost + } + var imageName: String { + return "item_\(type.imageName)_\(tier.imageName)" + } + var category: ItemCategory = .equipment + + // MARK: - Equipment + /// 장비 타입 + var type: EquipmentType + /// 장비 등급 + var tier: EquipmentTier + + /// 장비 초기화 + init(type: EquipmentType, tier: EquipmentTier) { + self.type = type + self.tier = tier + } + + /// 업그레이드 가능 여부 + var canUpgrade: Bool { + return tier != .nationalTreasure + } + + /// 초 당 획득 골드 + var goldPerSecond: Int { + switch type { + case .keyboard: + switch tier { + case .broken: + return Policy.Equipment.Keyboard.brokenGoldPerSecond + case .cheap: + return Policy.Equipment.Keyboard.cheapGoldPerSecond + case .vintage: + return Policy.Equipment.Keyboard.vintageGoldPerSecond + case .decent: + return Policy.Equipment.Keyboard.decentGoldPerSecond + case .premium: + return Policy.Equipment.Keyboard.premiumGoldPerSecond + case .diamond: + return Policy.Equipment.Keyboard.diamondGoldPerSecond + case .limited: + return Policy.Equipment.Keyboard.limitedGoldPerSecond + case .nationalTreasure: + return Policy.Equipment.Keyboard.nationalTreasureGoldPerSecond + } + case .mouse: + switch tier { + case .broken: + return Policy.Equipment.Mouse.brokenGoldPerSecond + case .cheap: + return Policy.Equipment.Mouse.cheapGoldPerSecond + case .vintage: + return Policy.Equipment.Mouse.vintageGoldPerSecond + case .decent: + return Policy.Equipment.Mouse.decentGoldPerSecond + case .premium: + return Policy.Equipment.Mouse.premiumGoldPerSecond + case .diamond: + return Policy.Equipment.Mouse.diamondGoldPerSecond + case .limited: + return Policy.Equipment.Mouse.limitedGoldPerSecond + case .nationalTreasure: + return Policy.Equipment.Mouse.nationalTreasureGoldPerSecond + } + case .monitor: + switch tier { + case .broken: + return Policy.Equipment.Monitor.brokenGoldPerSecond + case .cheap: + return Policy.Equipment.Monitor.cheapGoldPerSecond + case .vintage: + return Policy.Equipment.Monitor.vintageGoldPerSecond + case .decent: + return Policy.Equipment.Monitor.decentGoldPerSecond + case .premium: + return Policy.Equipment.Monitor.premiumGoldPerSecond + case .diamond: + return Policy.Equipment.Monitor.diamondGoldPerSecond + case .limited: + return Policy.Equipment.Monitor.limitedGoldPerSecond + case .nationalTreasure: + return Policy.Equipment.Monitor.nationalTreasureGoldPerSecond + } + case .chair: + switch tier { + case .broken: + return Policy.Equipment.Chair.brokenGoldPerSecond + case .cheap: + return Policy.Equipment.Chair.cheapGoldPerSecond + case .vintage: + return Policy.Equipment.Chair.vintageGoldPerSecond + case .decent: + return Policy.Equipment.Chair.decentGoldPerSecond + case .premium: + return Policy.Equipment.Chair.premiumGoldPerSecond + case .diamond: + return Policy.Equipment.Chair.diamondGoldPerSecond + case .limited: + return Policy.Equipment.Chair.limitedGoldPerSecond + case .nationalTreasure: + return Policy.Equipment.Chair.nationalTreasureGoldPerSecond + } + } + } + + /// 강화 확률에 따라 업그레이드 + func upgraded() -> Bool { + guard canUpgrade else { return false } + + let randomValue = Double.random(in: 0...1) + + if randomValue <= tier.upgradeSuccessRate { + self.tier = EquipmentTier(rawValue: tier.rawValue + 1) ?? .nationalTreasure + return true + } else { + return false + } + } +} + +/// 장비 종류 +enum EquipmentType { + /// 키보드 + case keyboard + /// 마우스 + case mouse + /// 모니터 + case monitor + /// 의자 + case chair + + /// 화면에 표시될 제목 + var displayTitle: String { + switch self { + case .keyboard: + return "키보드" + case .mouse: + return "마우스" + case .monitor: + return "모니터" + case .chair: + return "의자" + } + } + + var imageName: String { + switch self { + case .keyboard: + return "keyboard" + case .mouse: + return "mouse" + case .monitor: + return "monitor" + case .chair: + return "chair" + } + } +} + +/// 장비 등급 +enum EquipmentTier: Int { + /// 고장난 + case broken = 0 + /// 싸구려 + case cheap = 1 + /// 빈티지 + case vintage = 2 + /// 쓸만한 + case decent = 3 + /// 고급 + case premium = 4 + /// 다이아 + case diamond = 5 + /// 한정판 + case limited = 6 + /// 국보급 + case nationalTreasure = 7 + + /// 화면에 표시될 제목 + var displayTitle: String { + switch self { + case .broken: "고장난" + case .cheap: "싸구려" + case .vintage: "빈티지" + case .decent: "쓸만한" + case .premium: "고오급" + case .diamond: "다이아" + case .limited: "한정판" + case .nationalTreasure: "국보급" + } + } + + /// 업그레이드 비용 (골드 + 다이아몬드) + var cost: Cost { + switch self { + case .broken: + return Cost(gold: Policy.Equipment.brokenUpgradeCost, diamond: Policy.Equipment.brokenUpgradeDiamond) + case .cheap: + return Cost(gold: Policy.Equipment.cheapUpgradeCost, diamond: Policy.Equipment.cheapUpgradeDiamond) + case .vintage: + return Cost(gold: Policy.Equipment.vintageUpgradeCost, diamond: Policy.Equipment.vintageUpgradeDiamond) + case .decent: + return Cost(gold: Policy.Equipment.decentUpgradeCost, diamond: Policy.Equipment.decentUpgradeDiamond) + case .premium: + return Cost(gold: Policy.Equipment.premiumUpgradeCost, diamond: Policy.Equipment.premiumUpgradeDiamond) + case .diamond: + return Cost(gold: Policy.Equipment.diamondUpgradeCost, diamond: Policy.Equipment.diamondUpgradeDiamond) + case .limited: + return Cost(gold: Policy.Equipment.limitedUpgradeCost, diamond: Policy.Equipment.limitedUpgradeDiamond) + case .nationalTreasure: + return Cost(gold: Policy.Equipment.nationalTreasureUpgradeCost, diamond: Policy.Equipment.nationalTreasureUpgradeDiamond) + } + } + + /// 업그레이드 성공 확률 (0.0 ~ 1.0) + var upgradeSuccessRate: Double { + switch self { + case .broken: + return Policy.Equipment.brokenSuccessRate + case .cheap: + return Policy.Equipment.cheapSuccessRate + case .vintage: + return Policy.Equipment.vintageSuccessRate + case .decent: + return Policy.Equipment.decentSuccessRate + case .premium: + return Policy.Equipment.premiumSuccessRate + case .diamond: + return Policy.Equipment.diamondSuccessRate + case .limited: + return Policy.Equipment.limitedSuccessRate + case .nationalTreasure: + return Policy.Equipment.nationalTreasureSuccessRate + } + } + + var imageName: String { + switch self { + case .broken: + return "broken" + case .cheap: + return "cheap" + case .vintage: + return "vintage" + case .decent: + return "decent" + case .premium: + return "premium" + case .diamond: + return "diamond" + case .limited: + return "limited" + case .nationalTreasure: + return "nationalTreasure" + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Housing.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Housing.swift new file mode 100644 index 00000000..180f6716 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Housing.swift @@ -0,0 +1,113 @@ +// +// Housing.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +struct Housing: Item { + // MARK: - Item + var displayTitle: String { + return tier.displayTitle + } + var description: String { + return "초당 골드 획득량 \(goldPerSecond)" + } + var cost: Cost { + return tier.cost + } + var imageName: String { + return tier.imageName + } + var category: ItemCategory = .housing + + // MARK: - Housing + /// 초 당 획득 골드 + var goldPerSecond: Int { + return tier.goldPerSecond + } + let tier: HousingTier +} + +enum HousingTier: Int, CaseIterable { + case street = 0 + case semiBasement = 1 + case rooftop = 2 + case villa = 3 + case apartment = 4 + case house = 5 + case pentHouse = 6 + + var imageName: String { + switch self { + case .street: return "housing_street" + case .semiBasement: return "housing_semiBasement" + case .rooftop: return "housing_rooftop" + case .villa: return "housing_villa" + case .apartment: return "housing_apartment" + case .house: return "housing_house" + case .pentHouse: return "housing_pentHouse" + } + } + + var displayTitle: String { + switch self { + case .street: + return "길바닥" + case .semiBasement: + return "반지하" + case .rooftop: + return "옥탑방" + case .villa: + return "빌라" + case .apartment: + return "아파트" + case .house: + return "단독주택" + case .pentHouse: + return "펜트 하우스" + } + } + + /// 구입 비용 + var cost: Cost { + switch self { + case .street: + return .init(gold: Policy.Housing.streetPurchaseCost) + case .semiBasement: + return .init(gold: Policy.Housing.semiBasementPurchaseCost) + case .rooftop: + return .init(gold: Policy.Housing.rooftopPurchaseCost) + case .villa: + return .init(gold: Policy.Housing.villaPurchaseCost) + case .apartment: + return .init(gold: Policy.Housing.apartmentPurchaseCost) + case .house: + return .init(gold: Policy.Housing.housePurchaseCost) + case .pentHouse: + return .init(gold: Policy.Housing.pentHousePurchaseCost) + } + } + + /// 초 당 획득 골드 + var goldPerSecond: Int { + switch self { + case .street: + return Policy.Housing.streetGoldPerSecond + case .semiBasement: + return Policy.Housing.semiBasementGoldPerSecond + case .rooftop: + return Policy.Housing.rooftopGoldPerSecond + case .villa: + return Policy.Housing.villaGoldPerSecond + case .apartment: + return Policy.Housing.apartmentGoldPerSecond + case .house: + return Policy.Housing.houseGoldPerSecond + case .pentHouse: + return Policy.Housing.pentHouseGoldPerSecond + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Item.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Item.swift new file mode 100644 index 00000000..5c4f7290 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/Item.swift @@ -0,0 +1,31 @@ +// +// Item.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/20/26. +// + +import Foundation + +protocol Item { + /// 화면에 표시될 제목 + var displayTitle: String { get } + /// 아이템 설명 + var description: String { get } + /// 아이템 구입 비용 + var cost: Cost { get } + /// 아이템 이미지 이름 + var imageName: String { get } + /// 아이템 카테고리 (소비/장비/부동산) + var category: ItemCategory { get } +} + +/// 아이템 카테고리 +enum ItemCategory { + /// 소비 아이템 (커피, 에너지 드링크 등) + case consumable + /// 장비 아이템 (노트북, 모니터 등) + case equipment + /// 부동산 (집, 사무실 등) + case housing +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/ItemState.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/ItemState.swift new file mode 100644 index 00000000..7f9f807a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Items/ItemState.swift @@ -0,0 +1,49 @@ +// +// ItemState.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-20. +// + +import Foundation + +enum ItemState { + case available // 구매 가능 + case locked // 잠김 + case insufficient // 비용 부족 + case reachedMax // 최고 레벨 + + init(item: DisplayItem) { + // 장비 최고 레벨 체크 + if let equipment = item.item as? Equipment, !equipment.canUpgrade { + self = .reachedMax + return + } + + // 주거 아이템이 이미 장착되어 있으면 잠김 + if item.isEquipped && item.category == .housing { + self = .locked + return + } + + // 구매 가능 여부 + if item.isPurchasable { + self = .available + } else { + self = .insufficient + } + } + + init(canUpgrade: Bool, canUnlock: Bool, canAfford: Bool) { + switch (canUpgrade, canUnlock, canAfford) { + case (false, _, _): + self = .reachedMax + case (true, false, _): + self = .locked + case (true, true, false): + self = .insufficient + case (true, true, true): + self = .available + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Policy.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Policy.swift new file mode 100644 index 00000000..b6e671f7 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Policy.swift @@ -0,0 +1,371 @@ +// +// Policy.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/26/26. +// + +import Foundation + +enum Policy { + // MARK: - 커리어 시스템 (기준점) + /// 단계별 필요 누적 재산 (밸런스에 맞춰 약 20× 상향) + enum Career { + static let unemployed = 0 + static let laptopOwner = 100_000 + static let aspiringDeveloper = 1_000_000 + static let juniorDeveloper = 20_000_000 + static let normalDeveloper = 2_000_000_000 + static let nightOwlDeveloper = 20_000_000_000 + static let skilledDeveloper = 200_000_000_000 + static let famousDeveloper = 2_000_000_000_000 + static let allRounderDeveloper = 100_000_000_000_000 // 100조 + static let worldClassDeveloper = 2_000_000_000_000_000 // 2000조: 만렙 + + /// 게임별 해금 조건 (2단계씩: 탭 0 → 언어 2 → 버그 4 → 데이터 6) + enum GameUnlock { + static let tap = unemployed // 0단계 + static let language = aspiringDeveloper // 2단계 + static let dodge = normalDeveloper // 4단계 + static let stack = skilledDeveloper // 6단계 + } + } + + // MARK: - 피버 시스템 (쾌감 증대) + enum Fever { + /// 공통 설정 + static let maxPercent: Double = 400.0 + static let decreaseInterval: TimeInterval = 0.05 + + /// 피버 단계 경계값 + enum StageThreshold { + static let stage0: Double = 0 + static let stage1: Double = 100 + static let stage2: Double = 200 + static let stage3: Double = 300 + } + + /// 피버 단계별 배수 (상향 조정: 피버 시 확실한 보상) + enum Multiplier { + static let stage0: Double = 1.0 // 0~100 + static let stage1: Double = 1.5 // 1.2 -> 1.5 + static let stage2: Double = 2.5 // 1.5 -> 2.5 + static let stage3: Double = 5.0 // 2.0 -> 5.0 + } + + /// 코드 짜기 (TapGame) — 피버 상승량 대폭 하향 + enum Tap { + static let decreasePercent: Double = 1.5 + static let gainPerTap: Double = 5.0 // 2.0 -> 5.0 + } + + /// 언어 맞추기 (LanguageGame) + enum Language { + static let decreasePercent: Double = 1.5 + static let gainPerCorrect: Double = 33.0 + static let lossPerIncorrect: Double = -33.0 + } + + /// 버그 피하기 (DodgeGame) + enum Dodge { + static let decreasePercent: Double = 1.2 + static let gainPerSmallGold: Double = 33.0 + static let gainPerLargeGold: Double = 50.0 + static let gainPerBugDodge: Double = 15.0 + static let lossPerBugHit: Double = -20.0 + } + + /// 데이터 쌓기 (StackGame) + enum Stack { + static let decreasePercent: Double = 1.0 + static let gainPerSuccess: Double = 80.0 + static let lossPerFailure: Double = -40.0 + } + } + + // MARK: - 게임별 상수 + enum Game { + /// 코드 짜기 (TapGame) + enum Tap {} // 특수 상수 없음 + + /// 언어 맞추기 (LanguageGame) + enum Language { + static let incorrectGoldLossMultiplier: Double = 0.5 // 오답시 골드 감소 (획득량의 0.5배) + } + + /// 버그 피하기 (DodgeGame) + enum Dodge { + // 골드 + static let smallGoldMultiplier: Double = 1.5 + static let largeGoldMultiplier: Double = 2.0 + static let bugHitLossGoldMultiplier: Double = 0.5 // 버그 맞으면 골드 감소 (획득량의 0.5배) + static let bugDodgeGoldMultiplier: Double = 0.5 // 버그 피하면 골드 획득 + + // GameCore 설정 + static let updateFPS: Double = 120.0 // 업데이트 주기: 120fps + static let spawnInterval: TimeInterval = 0.3 // 낙하물 생성 간격 + static let fallSpeed: CGFloat = 3.0 // 낙하 속도 + + // 생성 확률 (%) + static let smallGoldSpawnRate: Int = 7 + static let largeGoldSpawnRate: Int = 3 + static let bugSpawnRate: Int = 90 + + /// 모션 시스템 + enum Motion { + static let deadZoneThreshold: Double = 0.05 + static let maxSpeed: CGFloat = 2000.0 + static let minSpeed: CGFloat = 300.0 + } + } + + /// 데이터 쌓기 (StackGame) + enum Stack { + static let failureGoldLossMultiplier: Double = 0.5 // 실패시 골드 감소 (획득량의 0.5배) + } + + /// 퀴즈 게임 (QuizGame) + enum Quiz { + static let questionsPerGame: Int = 3 // 게임당 문제 수 + static let secondsPerQuestion: Int = 20 // 문제당 제한 시간 + static let diamondsPerCorrect: Int = 5 // 정답당 다이아 + } + } + + // MARK: - 스킬 시스템 (초반 성장 가속) + // *전략: 초반 스킬 비용을 낮추고 효율을 높여 '클리커'의 재미를 느끼게 함 + enum Skill { + // 공통 레벨 범위 (모든 게임 통일) + static let beginnerMinLevel: Int = 1 + static let beginnerMaxLevel: Int = 999 + static let intermediateMinLevel: Int = 0 + static let intermediateMaxLevel: Int = 999 + static let advancedMinLevel: Int = 0 + static let advancedMaxLevel: Int = 999 + + /// 코드 짜기 (TapGame) + enum Tap { + // 기본 골드 단위 (상향: 최소 1) + static let baseGold: Int = 1 + + // 티어별 골드 획득 증가량 + static let beginnerGoldMultiplier: Int = 1 + static let intermediateGoldMultiplier: Int = 10 + static let advancedGoldMultiplier: Int = 100 + + // 업그레이드 비용 증가량 + static let beginnerGoldCostMultiplier: Int = 10 + static let intermediateGoldCostMultiplier: Int = 150 + static let advancedGoldCostMultiplier: Int = 2500 + static let diamondCostDivider: Int = 100 + static let diamondCostMultiplier: Int = 10 + + // 스킬 해금 조건 + static let intermediateUnlockLevel: Int = 200 + static let advancedUnlockLevel: Int = 300 + } + + /// 언어 맞추기 (LanguageGame) + /// * 분당 골드 = 탭의 3배 (피버 2.5 기준): 40정답/분 → Lv1,0,0 합 180 + enum Language { + // 기본 골드 단위 + static let baseGold: Int = 45 + + // 티어별 골드 획득 증가량 + static let beginnerGoldMultiplier: Int = 45 + static let intermediateGoldMultiplier: Int = 450 + static let advancedGoldMultiplier: Int = 4500 + + // 업그레이드 비용 증가량 (Tap 대비 스킬 합 비율 45배) + static let beginnerGoldCostMultiplier: Int = 450 + static let intermediateGoldCostMultiplier: Int = 6750 + static let advancedGoldCostMultiplier: Int = 112_500 + static let diamondCostDivider: Int = 100 + static let diamondCostMultiplier: Int = 10 + + // 스킬 해금 조건 + static let intermediateUnlockLevel: Int = 200 + static let advancedUnlockLevel: Int = 300 + } + + /// 버그 피하기 (DodgeGame) + /// * 아무튼 개발자(20억) 해금 시 언어(초250·중300·고100) 분당 ~6천만 수준에 맞춤: Lv1,0,0 합 348,000 + enum Dodge { + // 기본 골드 단위 + static let baseGold: Int = 87_000 + + // 티어별 골드 획득 증가량 + static let beginnerGoldMultiplier: Int = 87_000 + static let intermediateGoldMultiplier: Int = 870_000 + static let advancedGoldMultiplier: Int = 8_700_000 + + // 업그레이드 비용 증가량 (스킬 합 비율에 맞춤) + static let beginnerGoldCostMultiplier: Int = 870_000 + static let intermediateGoldCostMultiplier: Int = 13_050_000 + static let advancedGoldCostMultiplier: Int = 217_500_000 + static let diamondCostDivider: Int = 100 + static let diamondCostMultiplier: Int = 10 + + // 스킬 해금 조건 + static let intermediateUnlockLevel: Int = 200 + static let advancedUnlockLevel: Int = 300 + } + + /// 데이터 쌓기 (StackGame) + /// * 유능한 개발자(2000억) 해금 시 버그 피하기 대비 보상 상향: Lv1,0,0 합 9,000,000 (순 8회/분 → 분당 ~1.8억) + enum Stack { + // 기본 골드 단위 + static let baseGold: Int = 2_250_000 + + // 티어별 골드 획득 증가량 + static let beginnerGoldMultiplier: Int = 2_250_000 + static let intermediateGoldMultiplier: Int = 22_500_000 + static let advancedGoldMultiplier: Int = 225_000_000 + + // 업그레이드 비용 증가량 (스킬 합 비율에 맞춤) + static let beginnerGoldCostMultiplier: Int = 22_500_000 + static let intermediateGoldCostMultiplier: Int = 3_375_000_000 + static let advancedGoldCostMultiplier: Int = 56_250_000_000 + static let diamondCostDivider: Int = 100 + static let diamondCostMultiplier: Int = 10 + + // 스킬 해금 조건 + static let intermediateUnlockLevel: Int = 200 + static let advancedUnlockLevel: Int = 300 + } + } + + // MARK: - 소비 아이템 (효과 강화) + enum Consumable { + /// 커피 (1초당 3씩 증가) + enum Coffee { + static let duration: Int = 15 + static let buffMultiplier: Double = 2.0 + static let priceDiamond: Int = 5 + } + + /// 박하스 (1초당 6씩 증가) + enum EnergyDrink { + static let duration: Int = 20 + static let buffMultiplier: Double = 3.0 + static let priceDiamond: Int = 10 + } + } + + // MARK: - 장비 아이템 + // *밸런스: 업그레이드 비용 20×, 초당 골드 5× (부동산과 동일 비율) + enum Equipment { + // 업그레이드 비용 (골드) + static let brokenUpgradeCost: Int = 100_000 + static let cheapUpgradeCost: Int = 2_000_000 + static let vintageUpgradeCost: Int = 40_000_000 + static let decentUpgradeCost: Int = 1_000_000_000 + static let premiumUpgradeCost: Int = 20_000_000_000 + static let diamondUpgradeCost: Int = 200_000_000_000 + static let limitedUpgradeCost: Int = 1_000_000_000_000 + static let nationalTreasureUpgradeCost: Int = 4_000_000_000_000 + + // 업그레이드 비용 (다이아몬드) — 강화 시 골드와 함께 소모 + static let brokenUpgradeDiamond: Int = 5 + static let cheapUpgradeDiamond: Int = 10 + static let vintageUpgradeDiamond: Int = 20 + static let decentUpgradeDiamond: Int = 35 + static let premiumUpgradeDiamond: Int = 50 + static let diamondUpgradeDiamond: Int = 80 + static let limitedUpgradeDiamond: Int = 120 + static let nationalTreasureUpgradeDiamond: Int = 0 + + // 업그레이드 성공 확률 (모든 장비 공통) + static let brokenSuccessRate: Double = 1.0 + static let cheapSuccessRate: Double = 0.8 + static let vintageSuccessRate: Double = 0.6 + static let decentSuccessRate: Double = 0.4 + static let premiumSuccessRate: Double = 0.3 + static let diamondSuccessRate: Double = 0.2 + static let limitedSuccessRate: Double = 0.1 + static let nationalTreasureSuccessRate: Double = 0.05 + + /// 초당 획득 골드량 (키보드·마우스·모니터·의자 동일) + enum Keyboard { + static let brokenGoldPerSecond: Int = 0 + static let cheapGoldPerSecond: Int = 1_250 + static let vintageGoldPerSecond: Int = 30_000 + static let decentGoldPerSecond: Int = 750_000 + static let premiumGoldPerSecond: Int = 17_500_000 + static let diamondGoldPerSecond: Int = 200_000_000 + static let limitedGoldPerSecond: Int = 1_250_000_000 + static let nationalTreasureGoldPerSecond: Int = 6_000_000_000 + } + + /// 마우스 + enum Mouse { + static let brokenGoldPerSecond: Int = 0 + static let cheapGoldPerSecond: Int = 1_250 + static let vintageGoldPerSecond: Int = 30_000 + static let decentGoldPerSecond: Int = 750_000 + static let premiumGoldPerSecond: Int = 17_500_000 + static let diamondGoldPerSecond: Int = 200_000_000 + static let limitedGoldPerSecond: Int = 1_250_000_000 + static let nationalTreasureGoldPerSecond: Int = 6_000_000_000 + } + + /// 모니터 + enum Monitor { + static let brokenGoldPerSecond: Int = 0 + static let cheapGoldPerSecond: Int = 1_250 + static let vintageGoldPerSecond: Int = 30_000 + static let decentGoldPerSecond: Int = 750_000 + static let premiumGoldPerSecond: Int = 17_500_000 + static let diamondGoldPerSecond: Int = 200_000_000 + static let limitedGoldPerSecond: Int = 1_250_000_000 + static let nationalTreasureGoldPerSecond: Int = 6_000_000_000 + } + + /// 의자 + enum Chair { + static let brokenGoldPerSecond: Int = 0 + static let cheapGoldPerSecond: Int = 1_250 + static let vintageGoldPerSecond: Int = 30_000 + static let decentGoldPerSecond: Int = 750_000 + static let premiumGoldPerSecond: Int = 17_500_000 + static let diamondGoldPerSecond: Int = 200_000_000 + static let limitedGoldPerSecond: Int = 1_250_000_000 + static let nationalTreasureGoldPerSecond: Int = 6_000_000_000 + } + } + + // MARK: - 부동산 아이템 (로망 실현 및 자동 사냥 기지) + // *밸런스: 가격·초당 골드를 분당 3배수 경제에 맞춤 (가격 20×, 초당 골드 5× → 회수 시간 약 4배) + enum Housing { + // 구입 비용 + static let streetPurchaseCost: Int = 0 + static let semiBasementPurchaseCost: Int = 10_000_000 + static let rooftopPurchaseCost: Int = 200_000_000 + static let villaPurchaseCost: Int = 10_000_000_000 + static let apartmentPurchaseCost: Int = 100_000_000_000 + static let housePurchaseCost: Int = 1_000_000_000_000 + static let pentHousePurchaseCost: Int = 4_000_000_000_000 + + // 초당 골드 획득 (분당 = ×60) + static let streetGoldPerSecond: Int = 0 + static let semiBasementGoldPerSecond: Int = 2_500 + static let rooftopGoldPerSecond: Int = 50_000 + static let villaGoldPerSecond: Int = 2_500_000 + static let apartmentGoldPerSecond: Int = 25_000_000 + static let houseGoldPerSecond: Int = 250_000_000 + static let pentHouseGoldPerSecond: Int = 1_000_000_000 + } + + // MARK: - 기타 시스템 + enum System { + /// 자동 획득 시스템 (AutoGainSystem) + enum AutoGain { + static let interval: TimeInterval = 1.0 // 자동 획득 주기 (초) + } + + /// 버프 시스템 (BuffSystem) + enum Buff { + static let decreaseInterval: TimeInterval = 1.0 // 버프 감소 주기 (초) + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Storages/KeyValueLocalStorage.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Storages/KeyValueLocalStorage.swift new file mode 100644 index 00000000..22767eab --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Storages/KeyValueLocalStorage.swift @@ -0,0 +1,16 @@ +// +// KeyValueLocalStorage.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/27/26. +// + +import Foundation + +protocol KeyValueLocalStorage { + func register(defaults: [String: Any]) + func set(_ value: Any, forKey: String) + func integer(key: String) -> Int + func bool(key: String) -> Bool + func any(key: String) -> Any? +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Storages/UserDefaultsStorage.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Storages/UserDefaultsStorage.swift new file mode 100644 index 00000000..8579b292 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Storages/UserDefaultsStorage.swift @@ -0,0 +1,36 @@ +// +// UserDefaultsStorage.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/27/26. +// + +import Foundation + +struct UserDefaultsStorage: KeyValueLocalStorage { + private let userDefaults: UserDefaults + + init(userDefaults: UserDefaults = .standard) { + self.userDefaults = userDefaults + } + + func register(defaults: [String: Any]) { + userDefaults.register(defaults: defaults) + } + + func set(_ value: Any, forKey: String) { + userDefaults.set(value, forKey: forKey) + } + + func integer(key: String) -> Int { + return userDefaults.integer(forKey: key) + } + + func bool(key: String) -> Bool { + return userDefaults.bool(forKey: key) + } + + func any(key: String) -> Any? { + return userDefaults.object(forKey: key) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/AutoGainSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/AutoGainSystem.swift new file mode 100644 index 00000000..f9cad2fb --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/AutoGainSystem.swift @@ -0,0 +1,51 @@ +// +// AutoGainSystem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/8/26. +// + +import Foundation + +/// 자동 골드 획득 시스템 +final class AutoGainSystem { + /// 사용자 정보 + private let user: User + /// 타이머 + private var timer: Timer? + + /// 자동 획득 시스템 초기화 + init(user: User) { + self.user = user + } + + /// 자동 획득 시스템 시작 + func startSystem() { + // 기존 타이머가 있으면 정리 + stopSystem() + + let timer = Timer.scheduledTimer(withTimeInterval: Policy.System.AutoGain.interval, repeats: true) { [weak self] _ in + self?.gainGold() + } + // 스크롤 중에도 타이머가 작동하도록 common 모드에 추가 + RunLoop.current.add(timer, forMode: .common) + self.timer = timer + } + + /// 자동 획득 시스템 중지 + func stopSystem() { + timer?.invalidate() + timer = nil + } + + /// 초당 골드 획득 처리 + private func gainGold() { + let goldPerSecond = Calculator.calculateGoldPerSecond(user: user) + user.wallet.addGold(goldPerSecond) + /// 누적 재산 업데이트 + user.record.record(.earnMoney(goldPerSecond)) + + // 플레이 타임 기록 + user.record.record(.playTime) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/BuffSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/BuffSystem.swift new file mode 100644 index 00000000..74bc4033 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/BuffSystem.swift @@ -0,0 +1,150 @@ +// +// BuffSystem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation +import Observation + +/// 버프 시스템 관리 클래스 +@Observable +final class BuffSystem { + /// 각 버프의 종료 시각 + private var endTimes: [ConsumableType: TimeInterval] = [:] + + /// 각 버프의 배수 + private var multipliers: [ConsumableType: Double] = [:] + + /// 버프 단일 타이머 + private var timer: Timer? + + /// 버프 시스템 일시정지 여부 + private(set) var isPaused: Bool = false + + /// pause 시작 시각 + private var pauseStartTime: TimeInterval? + + // MARK: - Computed Properties + + /// 하나라도 활성화된 버프가 있는지 + var isRunning: Bool { + !endTimes.isEmpty + } + + /// 전체 버프 배수 (모든 활성화된 버프의 배수를 곱한 값) + var multiplier: Double { + multipliers.values.reduce(1.0, *) + } + + /// 커피 버프 남은 시간 (초) + var coffeeDuration: Int { + remainingDuration(for: .coffee) + } + + /// 에너지 드링크 버프 남은 시간 (초) + var energyDrinkDuration: Int { + remainingDuration(for: .energyDrink) + } + + /// 가장 긴 버프 남은 시간 (하위 호환) + var duration: Int { + max(coffeeDuration, energyDrinkDuration) + } + + // MARK: - Public Methods + + /// 소비 아이템 사용 + func useConsumableItem(type: ConsumableType) { + let now = currentTime + endTimes[type] = now + TimeInterval(type.duration) + multipliers[type] = type.buffMultiplier + startTimer() + } + + /// 버프 시스템 종료 + func stop() { + stopTimer() + endTimes.removeAll() + multipliers.removeAll() + isPaused = false + pauseStartTime = nil + } + + /// 버프 시스템 일시정지 + func pause() { + guard isRunning, !isPaused else { return } + isPaused = true + pauseStartTime = currentTime + stopTimer() + } + + /// 버프 시스템 재개 + func resume() { + guard isRunning, isPaused, let pauseStartTime else { return } + + let now = currentTime + let pausedDuration = now - pauseStartTime + + // 모든 버프의 종료 시각을 pause된 시간만큼 뒤로 밀기 + for type in endTimes.keys { + endTimes[type]! += pausedDuration + } + + self.pauseStartTime = nil + isPaused = false + startTimer() + } + + // MARK: - Private Helpers + + /// 현재 시간 (절대) + private var currentTime: TimeInterval { + Date.timeIntervalSinceReferenceDate + } + + /// 남은 시간 계산 (초) + private func remainingDuration(for type: ConsumableType) -> Int { + guard let endTime = endTimes[type] else { return 0 } + let remaining = endTime - currentTime + return max(0, Int(ceil(remaining))) + } + + /// 타이머 시작 + private func startTimer() { + guard timer == nil else { return } + + timer = Timer.scheduledTimer( + withTimeInterval: 1, + repeats: true + ) { [weak self] _ in + self?.tick() + } + } + + /// 타이머 종료 + private func stopTimer() { + timer?.invalidate() + timer = nil + } + + /// 1초 주기로 종료된 버프 정리 + private func tick() { + let now = currentTime + + for (type, endTime) in endTimes where now >= endTime { + stopBuff(for: type) + } + + if endTimes.isEmpty { + stopTimer() + } + } + + /// 특정 버프 제거 + private func stopBuff(for type: ConsumableType) { + endTimes.removeValue(forKey: type) + multipliers.removeValue(forKey: type) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/Calculator.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/Calculator.swift new file mode 100644 index 00000000..2aac3718 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/Calculator.swift @@ -0,0 +1,25 @@ +// +// Calculator.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/7/26. +// + +import Foundation + +enum Calculator { + /// 게임 액션당 획득 골드 계산 + static func calculateGoldPerAction(game: GameType, user: User, feverMultiplier: Double, buffMultiplier: Double) -> Int { + let actionPerGainGold = user.skills.filter { $0.key.game == game }.map { $0.gainGold }.reduce(0, +) + let result = Double(actionPerGainGold) * feverMultiplier * buffMultiplier + return Int(result) + } + + /// 초당 획득 골드 계산 + static func calculateGoldPerSecond(user: User) -> Int { + let goldPerSecond = user.inventory.equipmentItems.map { $0.goldPerSecond }.reduce(0, +) + let housingGoldPerSecond = user.inventory.housing.goldPerSecond + // 장비 아이템 + 부동산 아이템 초당 골드 + return Int(goldPerSecond) + housingGoldPerSecond + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/CareerSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/CareerSystem.swift new file mode 100644 index 00000000..0841c5d8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/CareerSystem.swift @@ -0,0 +1,71 @@ +// +// CareerSystem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation +import Observation + +@Observable +final class CareerSystem { + private let user: User + var currentCareer: Career + var careerProgress: Double = 0.0 + var onCareerChanged: ((Career) -> Void)? + + init(user: User) async { + self.user = user + self.currentCareer = user.career + await updateProgress() + } + + /// 누적 재산을 기반으로 현재 달성한 커리어 계산 + func calculateCareer() async -> Career { + let totalWealth = user.record.totalEarnedMoney + + var achievedCareer: Career = .unemployed + for career in Career.allCases { + if totalWealth >= career.requiredWealth { + achievedCareer = career + } else { + break + } + } + return achievedCareer + } + + /// 커리어 업데이트 + func updateCareer() async { + let newCareer = await calculateCareer() + if currentCareer != newCareer { + currentCareer = newCareer + user.updateCareer(to: newCareer) + onCareerChanged?(newCareer) + + if newCareer == .juniorDeveloper { + user.record.record(.juniorDeveloperAchieve) + } + } + await updateProgress() + } +} + +// MARK: - Helper +private extension CareerSystem { + /// 현재 커리어의 진행도 계산 (0.0 ~ 1.0) + func updateProgress() async { + guard let nextCareer = currentCareer.nextCareer else { + careerProgress = 1.0 + return + } + + let totalWealth = user.record.totalEarnedMoney + let currentRequirement = currentCareer.requiredWealth + let nextRequirement = nextCareer.requiredWealth + + let progress = Double(totalWealth - currentRequirement) / Double(nextRequirement - currentRequirement) + careerProgress = max(0.0, min(1.0, progress)) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/CharacterAnimationSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/CharacterAnimationSystem.swift new file mode 100644 index 00000000..5c3b678a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/CharacterAnimationSystem.swift @@ -0,0 +1,19 @@ +// +// CharacterAnimationSystem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/21/26. +// + +final class CharacterAnimationSystem { + var onSmile: (() -> Void)? + var onIdle: (() -> Void)? + + func playSmile() { + onSmile?() + } + + func playIdle() { + onIdle?() + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/FeverSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/FeverSystem.swift new file mode 100644 index 00000000..ed937bff --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/FeverSystem.swift @@ -0,0 +1,144 @@ +// +// FeverSystem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation +import Observation + +protocol FeverState { + var feverStage: Int { get } + var feverPercent: Double { get } + var feverMultiplier: Double { get } +} + +@Observable +final class FeverSystem: FeverState { + + // MARK: - Properties + /// 피버 감소 주기 (초 단위) + private let decreaseInterval: TimeInterval + /// 피버 감소 퍼센트 (주기당) + private let decreasePercentPerTick: Double + /// 현재 피버 단계 (0: 일반, 1~3: 피버 단계) + private(set) var feverStage: Int = 0 + /// 현재 피버 퍼센트 (0 ~ 300) + private(set) var feverPercent: Double = 0.0 { + didSet { + updateFeverStage() + } + } + /// 피버 감소 타이머 + private var decreaseTimer: Timer? + /// 피버 시스템 자체의 실행 중 여부 + private(set) var isRunning: Bool = false + /// 피버 시스템 일시정지 여부 + private(set) var isPaused: Bool = false + /// 피버 단계별 배수 + var feverMultiplier: Double { + switch feverStage { + case 0: return Policy.Fever.Multiplier.stage0 + case 1: return Policy.Fever.Multiplier.stage1 + case 2: return Policy.Fever.Multiplier.stage2 + case 3: return Policy.Fever.Multiplier.stage3 + default: return Policy.Fever.Multiplier.stage0 + } + } + + // MARK: - Initialization + init(decreaseInterval: TimeInterval, decreasePercentPerTick: Double) { + self.decreaseInterval = decreaseInterval + self.decreasePercentPerTick = decreasePercentPerTick + } + + deinit { + stop() + } + + // MARK: - Public Methods + /// 피버 획득 + /// - Parameter amount: 획득할 피버량 + func gainFever(_ amount: Double) { + let sumPercent = feverPercent + amount + feverPercent = min(Policy.Fever.maxPercent, max(0, sumPercent)) + } + + /// 피버 시스템 시작 + func start() { + guard !isRunning else { return } + isRunning = true + startTimer() + } + + /// 피버 시스템 종료 + func stop() { + isRunning = false + stopTimer() + } + + /// 피버 시스템 일시정지 + func pause() { + guard isRunning && !isPaused else { return } + isPaused = true + stopTimer() + } + + /// 피버 시스템 재개 + func resume() { + guard isRunning && isPaused else { return } + isPaused = false + startTimer() + } + + // MARK: - Private Methods + + /// 타이머 생성 및 시작 + private func startTimer() { + decreaseTimer = Timer.scheduledTimer( + withTimeInterval: decreaseInterval, + repeats: true + ) { [weak self] _ in + self?.decreaseFever() + } + } + + /// 타이머 종료 + private func stopTimer() { + decreaseTimer?.invalidate() + decreaseTimer = nil + } + + /// 피버 감소 (매 주기마다 호출) + private func decreaseFever() { + guard feverPercent > 0 else { return } + if feverPercent - decreasePercentPerTick < 0 { + feverPercent = 0 + } else { + feverPercent -= decreasePercentPerTick + } + } + + /// 피버 단계 업데이트 + private func updateFeverStage() { + let newStage: Int + + switch feverPercent { + case Policy.Fever.StageThreshold.stage0.. Int { + switch self { + case .tap: + return { $0.totalTapCount } + case .languageMatch: + return { $0.languageCorrectCount } + case .bugDodge: + return { $0.dodgeGoldCollectedCount } + case .stackItem: + return { $0.stackingSuccessCount } + case .playTime: + return { Int($0.totalPlayTime) } + case .coffee: + return { $0.coffeeUseCount } + case .energyDrink: + return { $0.energyDrinkUseCount } + case .languageConsecutive: + return { $0.languageConsecutiveCorrect } + case .bugDodgeConsecutive: + return { $0.dodgeMaxCombo } + case .stackConsecutive: + return { $0.stackConsecutiveSuccess } + case .tutorial: + return { $0.tutorialCompleted ? 1 : 0 } + case .career: + return { $0.hasAchievedJuniorDeveloper ? 1 : 0 } + } + } + + /// 완료 조건 (nil이면 기본 체크: currentValue >= targetValue) + var completeCondition: ((Record) -> Bool)? { + switch self { + case .tutorial: + return { $0.tutorialCompleted } + case .career: + return { $0.hasAchievedJuniorDeveloper } + default: + return nil + } + } + + var level: MissionLevel { + switch self { + case .tap(let level), + .languageMatch(let level), + .bugDodge(let level), + .stackItem(let level), + .playTime(let level), + .coffee(let level), + .energyDrink(let level), + .languageConsecutive(let level), + .bugDodgeConsecutive(let level), + .stackConsecutive(let level), + .tutorial(let level), + .career(let level): + return level + } + } +} + +/// 미션 목록을 생성하는 팩토리 +struct MissionFactory { + + // MARK: - Mission Configuration + + /// 미션 생성을 위한 설정 구조체 + struct MissionConfig { + let id: Int + let title: String + let description: String + let targetValue: Int + let reward: Cost + let type: MissionType + } + + // MARK: - Factory Method + + /// 전체 미션 목록 생성 + /// - Returns: 모든 미션의 배열 (총 32개) + static func createAllMissions() -> [Mission] { + let configs: [MissionConfig] = [ + // MARK: - 코드짜기 (탭) + MissionConfig( + id: MissionConstants.CodeTap.id1, + title: MissionConstants.CodeTap.title1, + description: MissionConstants.CodeTap.description1, + targetValue: MissionConstants.CodeTap.target1, + reward: MissionConstants.CodeTap.reward1, + type: .tap(.bronze) + ), + MissionConfig( + id: MissionConstants.CodeTap.id2, + title: MissionConstants.CodeTap.title2, + description: MissionConstants.CodeTap.description2, + targetValue: MissionConstants.CodeTap.target2, + reward: MissionConstants.CodeTap.reward2, + type: .tap(.silver) + ), + MissionConfig( + id: MissionConstants.CodeTap.id3, + title: MissionConstants.CodeTap.title3, + description: MissionConstants.CodeTap.description3, + targetValue: MissionConstants.CodeTap.target3, + reward: MissionConstants.CodeTap.reward3, + type: .tap(.gold) + ), + + // MARK: - 언어맞추기 (맞춘 횟수) + MissionConfig( + id: MissionConstants.LanguageMatch.id1, + title: MissionConstants.LanguageMatch.title1, + description: MissionConstants.LanguageMatch.description1, + targetValue: MissionConstants.LanguageMatch.target1, + reward: MissionConstants.LanguageMatch.reward1, + type: .languageMatch(.bronze) + ), + MissionConfig( + id: MissionConstants.LanguageMatch.id2, + title: MissionConstants.LanguageMatch.title2, + description: MissionConstants.LanguageMatch.description2, + targetValue: MissionConstants.LanguageMatch.target2, + reward: MissionConstants.LanguageMatch.reward2, + type: .languageMatch(.silver) + ), + MissionConfig( + id: MissionConstants.LanguageMatch.id3, + title: MissionConstants.LanguageMatch.title3, + description: MissionConstants.LanguageMatch.description3, + targetValue: MissionConstants.LanguageMatch.target3, + reward: MissionConstants.LanguageMatch.reward3, + type: .languageMatch(.gold) + ), + + // MARK: - 버그피하기 (골드 획득) + MissionConfig( + id: MissionConstants.BugDodge.id1, + title: MissionConstants.BugDodge.title1, + description: MissionConstants.BugDodge.description1, + targetValue: MissionConstants.BugDodge.target1, + reward: MissionConstants.BugDodge.reward1, + type: .bugDodge(.bronze) + ), + MissionConfig( + id: MissionConstants.BugDodge.id2, + title: MissionConstants.BugDodge.title2, + description: MissionConstants.BugDodge.description2, + targetValue: MissionConstants.BugDodge.target2, + reward: MissionConstants.BugDodge.reward2, + type: .bugDodge(.silver) + ), + MissionConfig( + id: MissionConstants.BugDodge.id3, + title: MissionConstants.BugDodge.title3, + description: MissionConstants.BugDodge.description3, + targetValue: MissionConstants.BugDodge.target3, + reward: MissionConstants.BugDodge.reward3, + type: .bugDodge(.gold) + ), + + // MARK: - 데이터쌓기 + MissionConfig( + id: MissionConstants.StackItem.id1, + title: MissionConstants.StackItem.title1, + description: MissionConstants.StackItem.description1, + targetValue: MissionConstants.StackItem.target1, + reward: MissionConstants.StackItem.reward1, + type: .stackItem(.bronze) + ), + MissionConfig( + id: MissionConstants.StackItem.id2, + title: MissionConstants.StackItem.title2, + description: MissionConstants.StackItem.description2, + targetValue: MissionConstants.StackItem.target2, + reward: MissionConstants.StackItem.reward2, + type: .stackItem(.silver) + ), + MissionConfig( + id: MissionConstants.StackItem.id3, + title: MissionConstants.StackItem.title3, + description: MissionConstants.StackItem.description3, + targetValue: MissionConstants.StackItem.target3, + reward: MissionConstants.StackItem.reward3, + type: .stackItem(.gold) + ), + + // MARK: - 플레이타임 + MissionConfig( + id: MissionConstants.PlayTime.id1, + title: MissionConstants.PlayTime.title1, + description: MissionConstants.PlayTime.description1, + targetValue: MissionConstants.PlayTime.targetHours1 * 3600, + reward: MissionConstants.PlayTime.reward1, + type: .playTime(.bronze) + ), + MissionConfig( + id: MissionConstants.PlayTime.id2, + title: MissionConstants.PlayTime.title2, + description: MissionConstants.PlayTime.description2, + targetValue: MissionConstants.PlayTime.targetHours2 * 3600, + reward: MissionConstants.PlayTime.reward2, + type: .playTime(.silver) + ), + MissionConfig( + id: MissionConstants.PlayTime.id3, + title: MissionConstants.PlayTime.title3, + description: MissionConstants.PlayTime.description3, + targetValue: MissionConstants.PlayTime.targetHours3 * 3600, + reward: MissionConstants.PlayTime.reward3, + type: .playTime(.gold) + ), + + // MARK: - 커피 + MissionConfig( + id: MissionConstants.Coffee.id1, + title: MissionConstants.Coffee.title1, + description: MissionConstants.Coffee.description1, + targetValue: MissionConstants.Coffee.target1, + reward: MissionConstants.Coffee.reward1, + type: .coffee(.bronze) + ), + MissionConfig( + id: MissionConstants.Coffee.id2, + title: MissionConstants.Coffee.title2, + description: MissionConstants.Coffee.description2, + targetValue: MissionConstants.Coffee.target2, + reward: MissionConstants.Coffee.reward2, + type: .coffee(.silver) + ), + MissionConfig( + id: MissionConstants.Coffee.id3, + title: MissionConstants.Coffee.title3, + description: MissionConstants.Coffee.description3, + targetValue: MissionConstants.Coffee.target3, + reward: MissionConstants.Coffee.reward3, + type: .coffee(.gold) + ), + + // MARK: - 박하스 + MissionConfig( + id: MissionConstants.EnergyDrink.id1, + title: MissionConstants.EnergyDrink.title1, + description: MissionConstants.EnergyDrink.description1, + targetValue: MissionConstants.EnergyDrink.target1, + reward: MissionConstants.EnergyDrink.reward1, + type: .energyDrink(.bronze) + ), + MissionConfig( + id: MissionConstants.EnergyDrink.id2, + title: MissionConstants.EnergyDrink.title2, + description: MissionConstants.EnergyDrink.description2, + targetValue: MissionConstants.EnergyDrink.target2, + reward: MissionConstants.EnergyDrink.reward2, + type: .energyDrink(.silver) + ), + MissionConfig( + id: MissionConstants.EnergyDrink.id3, + title: MissionConstants.EnergyDrink.title3, + description: MissionConstants.EnergyDrink.description3, + targetValue: MissionConstants.EnergyDrink.target3, + reward: MissionConstants.EnergyDrink.reward3, + type: .energyDrink(.gold) + ), + + // MARK: - 언어맞추기 (연속 성공) + MissionConfig( + id: MissionConstants.LanguageConsecutive.id1, + title: MissionConstants.LanguageConsecutive.title1, + description: MissionConstants.LanguageConsecutive.description1, + targetValue: MissionConstants.LanguageConsecutive.target1, + reward: MissionConstants.LanguageConsecutive.reward1, + type: .languageConsecutive(.bronze) + ), + MissionConfig( + id: MissionConstants.LanguageConsecutive.id2, + title: MissionConstants.LanguageConsecutive.title2, + description: MissionConstants.LanguageConsecutive.description2, + targetValue: MissionConstants.LanguageConsecutive.target2, + reward: MissionConstants.LanguageConsecutive.reward2, + type: .languageConsecutive(.silver) + ), + MissionConfig( + id: MissionConstants.LanguageConsecutive.id3, + title: MissionConstants.LanguageConsecutive.title3, + description: MissionConstants.LanguageConsecutive.description3, + targetValue: MissionConstants.LanguageConsecutive.target3, + reward: MissionConstants.LanguageConsecutive.reward3, + type: .languageConsecutive(.gold) + ), + + // MARK: - 버그피하기 (연속 성공) + MissionConfig( + id: MissionConstants.BugDodgeConsecutive.id1, + title: MissionConstants.BugDodgeConsecutive.title1, + description: MissionConstants.BugDodgeConsecutive.description1, + targetValue: MissionConstants.BugDodgeConsecutive.target1, + reward: MissionConstants.BugDodgeConsecutive.reward1, + type: .bugDodgeConsecutive(.bronze) + ), + MissionConfig( + id: MissionConstants.BugDodgeConsecutive.id2, + title: MissionConstants.BugDodgeConsecutive.title2, + description: MissionConstants.BugDodgeConsecutive.description2, + targetValue: MissionConstants.BugDodgeConsecutive.target2, + reward: MissionConstants.BugDodgeConsecutive.reward2, + type: .bugDodgeConsecutive(.silver) + ), + MissionConfig( + id: MissionConstants.BugDodgeConsecutive.id3, + title: MissionConstants.BugDodgeConsecutive.title3, + description: MissionConstants.BugDodgeConsecutive.description3, + targetValue: MissionConstants.BugDodgeConsecutive.target3, + reward: MissionConstants.BugDodgeConsecutive.reward3, + type: .bugDodgeConsecutive(.gold) + ), + + // MARK: - 데이터 쌓기 (연속 성공) + MissionConfig( + id: MissionConstants.StackConsecutive.id1, + title: MissionConstants.StackConsecutive.title1, + description: MissionConstants.StackConsecutive.description1, + targetValue: MissionConstants.StackConsecutive.target1, + reward: MissionConstants.StackConsecutive.reward1, + type: .stackConsecutive(.bronze) + ), + MissionConfig( + id: MissionConstants.StackConsecutive.id2, + title: MissionConstants.StackConsecutive.title2, + description: MissionConstants.StackConsecutive.description2, + targetValue: MissionConstants.StackConsecutive.target2, + reward: MissionConstants.StackConsecutive.reward2, + type: .stackConsecutive(.silver) + ), + MissionConfig( + id: MissionConstants.StackConsecutive.id3, + title: MissionConstants.StackConsecutive.title3, + description: MissionConstants.StackConsecutive.description3, + targetValue: MissionConstants.StackConsecutive.target3, + reward: MissionConstants.StackConsecutive.reward3, + type: .stackConsecutive(.gold) + ), + + // MARK: - 커리어 + MissionConfig( + id: MissionConstants.Career.id, + title: MissionConstants.Career.title, + description: MissionConstants.Career.description, + targetValue: 1, + reward: MissionConstants.Career.reward, + type: .career(.special) + ), + + // MARK: - 튜토리얼 + MissionConfig( + id: MissionConstants.Tutorial.id, + title: MissionConstants.Tutorial.title, + description: MissionConstants.Tutorial.description, + targetValue: 1, + reward: MissionConstants.Tutorial.reward, + type: .tutorial(.special) + ) + ] + + return configs.map { createMission(from: $0) } + } + + // MARK: - Private Helper + + /// MissionConfig로부터 Mission 인스턴스를 생성합니다. + private static func createMission(from config: MissionConfig) -> Mission { + Mission( + id: config.id, + type: config.type, + title: config.title, + description: config.description, + targetValue: config.targetValue, + updateCondition: config.type.currentValue, + completeCondition: config.type.completeCondition, + reward: config.reward + ) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/MissionSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/MissionSystem.swift new file mode 100644 index 00000000..41e1749d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/MissionSystem.swift @@ -0,0 +1,57 @@ +// +// MissionSystem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +@Observable +final class MissionSystem { + /// 모든 미션 + private(set) var missions: [Mission] + /// 완료된 미션 유무 플래그 + private(set) var hasCompletedMission: Bool = false + + /// 총 미션 수 + var allCount: Int { + missions.count + } + /// 획득한 미션 수 + var claimedCount: Int { + missions.count { $0.state == .claimed } + } + + init(missions: [Mission]) { + self.missions = missions + sortMissions() + } + + /// 기록을 통하여 현재 미션의 상태들을 업데이트 합니다. + func updateCompletedMissions(record: Record) { + missions + .filter { $0.state == .inProgress } + .forEach { $0.update(record: record) } + + sortMissions() + checkHasCompletedMission() + } + + func claimMissionReward(mission: Mission, wallet: Wallet) { + let reward = mission.claim() + wallet.addGold(reward.gold) + wallet.addDiamond(reward.diamond) + + sortMissions() + checkHasCompletedMission() + } + + private func checkHasCompletedMission() { + hasCompletedMission = missions.contains { $0.state == .claimable } + } + + private func sortMissions() { + missions.sort { $0.state.rawValue < $1.state.rawValue } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/MotionSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/MotionSystem.swift new file mode 100644 index 00000000..47a8a8f3 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/MotionSystem.swift @@ -0,0 +1,72 @@ +// +// MotionSystem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/13/26. +// + +import CoreMotion +import Observation + +@Observable +final class MotionSystem { + // MARK: - Properties + /// CoreMotion 매니저 + private let motionManager = CMMotionManager() + + // MARK: - 설정값 + /// 모션 업데이트 주사율 (120fps) + private let updateInterval = 1.0 / Policy.Game.Dodge.updateFPS + /// 데드존 임계값 (이 값 이하의 기울기는 무시) + private let threshold: Double = Policy.Game.Dodge.Motion.deadZoneThreshold + /// 최대 이동 속도 (기울기 1.0일 때) + private let maxSpeed: CGFloat = Policy.Game.Dodge.Motion.maxSpeed + /// 최소 이동 속도 (기울기 threshold일 때) + private let minSpeed: CGFloat = Policy.Game.Dodge.Motion.minSpeed + + /// 캐릭터의 X 위치 + var characterX: CGFloat = 0 + /// 보정된 X축 기울기 값 (-1.0 ~ 1.0) + var calibratedGravityX: Double = 0 + /// 화면 제한 범위 (게임 영역 너비의 절반) + var screenLimit: CGFloat + + init(screenLimit: CGFloat) { + self.screenLimit = screenLimit + startMotionUpdates() + } + + deinit { + motionManager.stopDeviceMotionUpdates() + } + + /// 모션 업데이트 시작 (120fps) + /// - 기울기 강도에 따라 이동 속도가 비선형적으로 증가 + /// - 화면 밖으로 나가지 않도록 제한 + func startMotionUpdates() { + guard motionManager.isDeviceMotionAvailable else { return } + motionManager.deviceMotionUpdateInterval = updateInterval + + motionManager.startDeviceMotionUpdates(to: .main) { [weak self] motion, _ in + guard let self = self, let motion = motion else { return } + + // 기울기 값 클램핑 (-1.0 ~ 1.0) + let clampedInput = max(-1.0, min(1.0, motion.gravity.x)) + self.calibratedGravityX = clampedInput + + // 캐릭터 이동 (데드존 이상일 때만) + if abs(clampedInput) > self.threshold { + // 기울기 강도에 따라 속도 계산 (비선형적으로 증가) + let tiltStrength = abs(clampedInput) + let speedMultiplier = self.minSpeed + (self.maxSpeed - self.minSpeed) * CGFloat(pow(tiltStrength, 1.5)) + let direction = clampedInput > 0 ? 1.0 : -1.0 + + // 위치 업데이트 + self.characterX += CGFloat(direction) * speedMultiplier * CGFloat(self.updateInterval) + + // 화면 밖 방지 + self.characterX = max(-self.screenLimit, min(self.screenLimit, self.characterX)) + } + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/PurchasingError.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/PurchasingError.swift new file mode 100644 index 00000000..3e345eae --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/PurchasingError.swift @@ -0,0 +1,20 @@ +// +// ShopSystemError.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/19/26. +// + +import Foundation + +/// 구매 기능 관련 에러 +enum PurchasingError: Error { + /// 잠김 + case locked + /// 골드 부족 + case insufficientGold + /// 다이아몬드 부족 + case insufficientDiamond + /// 구매 실패 + case purchaseFailed +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/ShopSystem/DisplayItem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/ShopSystem/DisplayItem.swift new file mode 100644 index 00000000..2fb1370d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/ShopSystem/DisplayItem.swift @@ -0,0 +1,33 @@ +// +// DisplayItem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/19/26. +// + +import Foundation + +struct DisplayItem: Identifiable, Item { + // MARK: - Item + var displayTitle: String { + return item.displayTitle + } + var description: String { + return item.description + } + var cost: Cost { + return item.cost + } + var imageName: String { + return item.imageName + } + var category: ItemCategory { + return item.category + } + + // MARK: - DisplayItem + let id = UUID() + let item: Item + let isEquipped: Bool + let isPurchasable: Bool +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/ShopSystem/ShopSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/ShopSystem/ShopSystem.swift new file mode 100644 index 00000000..4c6db79e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/ShopSystem/ShopSystem.swift @@ -0,0 +1,207 @@ +// +// ShopSystem.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +/// 상점 시스템 관리 클래스 +final class ShopSystem { + /// 사용자 정보 + private let user: User + + /// 상점 시스템 초기화 + init(user: User) { + self.user = user + } + + /// 상점에 표시될 아이템 목록 생성 + /// - Parameter itemTypes: 표시할 아이템 카테고리 목록 + /// - Returns: 화면에 표시할 DisplayItem 배열 + func itemList(itemTypes: [ItemCategory]) -> [DisplayItem] { + return itemTypes.map { makeDisplayItems(for: $0) }.flatMap { $0 } + } + + /// 부동산 실제 비용 계산 + /// - Parameter item: 부동산 아이템 + /// - Returns: 실제 지불/환불 금액 (양수: 지불, 음수: 환불) + func calculateHousingNetCost(for item: DisplayItem) -> Int { + let refundAmount = user.inventory.housing.cost.gold / 2 + return item.cost.gold - refundAmount + } + + /// 아이템 구매 + /// - Parameter item: 구매할 아이템 + /// - Returns: 구매 성공 여부 (장비의 경우 강화 성공/실패, 다른 아이템은 항상 true) + /// - Throws: + /// - ShopSystemError.insufficientGold: 골드 부족 + /// - ShopSystemError.insufficientDiamond: 다이아몬드 부족 + /// - ShopSystemError.purchaseFailed: 구매 처리 실패 + func buy(item: DisplayItem) throws -> Bool { + // 부동산의 경우 실제 지불 금액으로 구매 가능 여부 확인 + if item.category == .housing { + let netGoldCost = calculateHousingNetCost(for: item) + + // 실제 지불 금액이 양수일 때만 구매 가능 여부 확인 (음수면 환불이므로 항상 가능) + if netGoldCost > 0 { + guard user.wallet.gold >= netGoldCost else { + throw PurchasingError.insufficientGold + } + } + + // 다이아몬드 비용 확인 + if item.cost.diamond > 0 { + guard user.wallet.diamond >= item.cost.diamond else { + throw PurchasingError.insufficientDiamond + } + } + } else { + // 장비 및 소비품: 일반 구매 가능 여부 확인 + guard user.wallet.canAfford(item.cost) else { + if item.cost.gold > 0 { + throw PurchasingError.insufficientGold + } else { + throw PurchasingError.insufficientDiamond + } + } + } + + // 아이템 카테고리에 따른 구매 로직 실행 + switch item.category { + case .consumable: + return try buyConsumable(displayItem: item) + case .equipment: + return try buyEquipment(displayItem: item) + case .housing: + return try buyHousing(displayItem: item) + } + } +} + +private extension ShopSystem { + /// 아이템 카테고리에 따른 DisplayItem 목록 생성 + /// - Parameter category: 아이템 카테고리 + /// - Returns: 해당 카테고리의 DisplayItem 배열 + func makeDisplayItems(for category: ItemCategory) -> [DisplayItem] { + switch category { + case .consumable: + // 소비 아이템 목록 생성 + let consumables = user.inventory.consumableItems + return consumables.map { consumable in + DisplayItem( + item: consumable, + isEquipped: true, + isPurchasable: user.wallet.canAfford(consumable.cost) + ) + } + + case .equipment: + // 장비 아이템 목록 생성 + let equipments = user.inventory.equipmentItems + return equipments.map { equipment in + DisplayItem( + item: equipment, + isEquipped: true, + isPurchasable: user.wallet.canAfford(equipment.cost) + ) + } + + case .housing: + // 부동산 아이템 목록 생성 (모든 티어 표시) + let currentHousingTier = user.inventory.housing.tier.rawValue + let allHousings = HousingTier.allCases.map { Housing(tier: $0) } + return allHousings.map { housing in + DisplayItem( + item: housing, + isEquipped: housing.tier.rawValue == currentHousingTier, + isPurchasable: user.wallet.canAfford(housing.cost) + ) + } + .sorted { $0.isEquipped && !$1.isEquipped } + } + } + + /// 소비 아이템 구매 (개수 증가) + /// - Parameter displayItem: 구매할 소비 아이템 + /// - Returns: 항상 true (구매 성공) + /// - Throws: ShopSystemError.purchaseFailed - 아이템 타입 변환 실패 + func buyConsumable(displayItem: DisplayItem) throws -> Bool { + // DisplayItem을 Consumable로 변환 + guard let consumable = displayItem.item as? Consumable else { + throw PurchasingError.purchaseFailed + } + + // 재화 지불 (골드/다이아몬드) + let cost = displayItem.cost + if cost.gold > 0 { + user.wallet.spendGold(cost.gold) + } + if cost.diamond > 0 { + user.wallet.spendDiamond(cost.diamond) + } + + // 인벤토리에 소비 아이템 개수 증가 + user.inventory.gain(consumable: consumable.type) + + return true + } + + /// 장비 아이템 구매 (강화 시도) + /// - Parameter displayItem: 구매할 장비 아이템 + /// - Returns: 강화 성공 여부 (true: 성공, false: 실패) + /// - Throws: ShopSystemError.purchaseFailed - 아이템 타입 변환 실패 + /// - Note: 비용 지불 후 강화를 시도하며, 성공 여부는 티어의 강화 확률에 따라 결정됨 + func buyEquipment(displayItem: DisplayItem) throws -> Bool { + // DisplayItem을 Equipment로 변환 + guard let equipment = displayItem.item as? Equipment else { + throw PurchasingError.purchaseFailed + } + + // 재화 지불 (골드/다이아몬드) + let cost = displayItem.cost + if cost.gold > 0 { + user.wallet.spendGold(cost.gold) + } + if cost.diamond > 0 { + user.wallet.spendDiamond(cost.diamond) + } + + // 장비 강화 시도 (성공 확률은 티어마다 다름) + return equipment.upgraded() + } + + /// 부동산 아이템 구매 (교체 및 환불) + /// - Parameter displayItem: 구매할 부동산 아이템 + /// - Returns: 항상 true (구매 성공) + /// - Throws: ShopSystemError.purchaseFailed - 아이템 타입 변환 실패 + /// - Note: 기존 부동산 구입 금액의 50%를 환불받고, 실제 비용만 지불함 + func buyHousing(displayItem: DisplayItem) throws -> Bool { + // DisplayItem을 Housing으로 변환 + guard let newHousing = displayItem.item as? Housing else { + throw PurchasingError.purchaseFailed + } + + // 실제 지불 금액 계산 + let netGoldCost = calculateHousingNetCost(for: displayItem) + + // 골드 지불 또는 환불 + if netGoldCost > 0 { + user.wallet.spendGold(netGoldCost) + } else { + // 환불액이 더 큰 경우 골드 추가 (다운그레이드 시) + user.wallet.addGold(-netGoldCost) + } + + // 다이아몬드 비용 지불 + if displayItem.cost.diamond > 0 { + user.wallet.spendDiamond(displayItem.cost.diamond) + } + + // 인벤토리의 부동산을 새 부동산으로 교체 + user.inventory.housing = newHousing + + return true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/SkillSystem.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/SkillSystem.swift new file mode 100644 index 00000000..b813ab0a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/Systems/SkillSystem.swift @@ -0,0 +1,143 @@ +// +// SkillSystem.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/20/26. +// + +import Foundation + +struct SkillState { + let skill: Skill + let itemState: ItemState +} + +final class SkillSystem { + private let user: User + private let careerSystem: CareerSystem? + + init( + user: User, + careerSystem: CareerSystem? + ) { + self.user = user + self.careerSystem = careerSystem + } + + /// 스킬 리스트를 정렬하여 반환 + func skillList() -> [SkillState] { + let gameTypeCount = GameType.allCases.count + let skillTierCount = SkillTier.allCases.count + + var buckets: [Skill?] = Array( + repeating: nil, + count: gameTypeCount * skillTierCount + ) + for skill in user.skills { + let gameIndex = skill.key.game.rawValue + let tierIndex = skill.key.tier.rawValue + let bucketIndex = gameIndex * skillTierCount + tierIndex + buckets[bucketIndex] = skill + } + return buckets + .compactMap { $0 } + .map { skill in SkillState( + skill: skill, + itemState: getItemState(for: skill)) + } + } + + /// 스킬 항목을 구매하여 레벨 업그레이드 + func upgrade(skill: Skill) throws { + guard canUnlock(skill: skill) else { + throw PurchasingError.locked + } + guard skill.upgradeCost.gold <= user.wallet.gold else { + throw PurchasingError.insufficientGold + } + guard skill.upgradeCost.diamond <= user.wallet.diamond else { + throw PurchasingError.insufficientDiamond + } + + let costBeforeUpgrade = skill.upgradeCost + try skill.upgrade() + pay(cost: costBeforeUpgrade) + } +} + +private extension SkillSystem { + func pay(cost: Cost) { + user.wallet.spendGold(cost.gold) + user.wallet.spendDiamond(cost.diamond) + } + + func getItemState(for skill: Skill) -> ItemState { + let canUpgrade = skill.level < skill.key.tier.levelRange.maxValue + let canUnlock = canUnlock(skill: skill) + let cost = skill.upgradeCost + let canAfford = cost.gold <= user.wallet.gold && cost.diamond <= user.wallet.diamond + + return ItemState(canUpgrade: canUpgrade, canUnlock: canUnlock, canAfford: canAfford) + } + + func canUnlock(skill: Skill) -> Bool { + let unlockLevel: Int + + switch skill.key.game { + case .tap: + switch skill.key.tier { + case .beginner: + guard let careerSystem = careerSystem else { return false } + return careerSystem.currentCareer.requiredWealth >= Policy.Career.GameUnlock.tap + case .intermediate: + unlockLevel = Policy.Skill.Tap.intermediateUnlockLevel + case .advanced: + unlockLevel = Policy.Skill.Tap.advancedUnlockLevel + } + case .language: + switch skill.key.tier { + case .beginner: + guard let careerSystem = careerSystem else { return false } + return careerSystem.currentCareer.requiredWealth >= Policy.Career.GameUnlock.language + case .intermediate: + unlockLevel = Policy.Skill.Language.intermediateUnlockLevel + case .advanced: + unlockLevel = Policy.Skill.Language.advancedUnlockLevel + } + case .dodge: + switch skill.key.tier { + case .beginner: + guard let careerSystem = careerSystem else { return false } + return careerSystem.currentCareer.requiredWealth >= Policy.Career.GameUnlock.dodge + case .intermediate: + unlockLevel = Policy.Skill.Dodge.intermediateUnlockLevel + case .advanced: + unlockLevel = Policy.Skill.Dodge.advancedUnlockLevel + } + case .stack: + switch skill.key.tier { + case .beginner: + guard let careerSystem = careerSystem else { return false } + return careerSystem.currentCareer.requiredWealth >= Policy.Career.GameUnlock.stack + case .intermediate: + unlockLevel = Policy.Skill.Stack.intermediateUnlockLevel + case .advanced: + unlockLevel = Policy.Skill.Stack.advancedUnlockLevel + } + } + + // intermediate나 advanced인 경우 + let previousTier: SkillTier = skill.key.tier == .intermediate ? .beginner : .intermediate + guard let previousSkill = getPreviousTierSkill(key: .init(game: skill.key.game, tier: previousTier)) else { + return false + } + return previousSkill.level >= unlockLevel + } + + func getPreviousTierSkill(key: SkillKey) -> Skill? { + guard let previousTierSkill = user.skills.first(where: {$0.key == key}) else { + return nil + } + return previousTierSkill + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Career.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Career.swift new file mode 100644 index 00000000..ad131604 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Career.swift @@ -0,0 +1,118 @@ +// +// Career.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +enum Career: String, CaseIterable { + case unemployed = "백수" + case laptopOwner = "노트북 보유자" + case aspiringDeveloper = "개발자 지망생" + case juniorDeveloper = "하찮은 개발자" + case normalDeveloper = "아무튼 개발자" + case nightOwlDeveloper = "밤 새는 개발자" + case skilledDeveloper = "유능한 개발자" + case famousDeveloper = "유명한 개발자" + case allRounderDeveloper = "올라운더 개발자" + case worldClassDeveloper = "월드클래스 개발자" + + var description: String { + switch self { + case .unemployed: + return "아직 아무것도 시작하지 않았지만, 시간은 가장 많다" + case .laptopOwner: + return "별다방 입장권 획득" + case .aspiringDeveloper: + return "헬로 월드(Hello World) 장인" + case .juniorDeveloper: + return "에러는 많고 자신감은 적다" + case .normalDeveloper: + return "이유는 몰라도 코드는 돌아간다" + case .nightOwlDeveloper: + return "해 뜨는게 퇴근 신호" + case .skilledDeveloper: + return "복붙의 신" + case .famousDeveloper: + return "개발자들의 연예인" + case .allRounderDeveloper: + return "맡다 보니 전부 다 하게 됐다" + case .worldClassDeveloper: + return "0과 1로 대화 가능" + } + } + + var imageName: String { + switch self { + case .unemployed: return "profile_unemployed" + case .laptopOwner: return "profile_laptop_owner" + case .aspiringDeveloper: return "profile_aspiring_developer" + case .juniorDeveloper: return "profile_junior_developer" + case .normalDeveloper: return "profile_normal_developer" + case .nightOwlDeveloper: return "profile_night_owl_developer" + case .skilledDeveloper: return "profile_skilled_developer" + case .famousDeveloper: return "profile_famous_developer" + case .allRounderDeveloper: return "profile_all_rounder_developer" + case .worldClassDeveloper: return "profile_world_class_developer" + } + } + + var characterImagePrefix: String { + switch self { + case .unemployed: return "character_unemployed" + case .laptopOwner: return "character_laptop_owner" + case .aspiringDeveloper: return "character_aspiring_developer" + case .juniorDeveloper: return "character_junior_developer" + case .normalDeveloper: return "character_normal_developer" + case .nightOwlDeveloper: return "character_night_owl_developer" + case .skilledDeveloper: return "character_skilled_developer" + case .famousDeveloper: return "character_famous_developer" + case .allRounderDeveloper: return "character_all_rounder_developer" + case .worldClassDeveloper: return "character_world_class_developer" + } + } + + /// 다음 단계로 업그레이드하기 위해 필요한 누적 재산 + var requiredWealth: Int { + switch self { + case .unemployed: + return Policy.Career.unemployed + case .laptopOwner: + return Policy.Career.laptopOwner + case .aspiringDeveloper: + return Policy.Career.aspiringDeveloper + case .juniorDeveloper: + return Policy.Career.juniorDeveloper + case .normalDeveloper: + return Policy.Career.normalDeveloper + case .nightOwlDeveloper: + return Policy.Career.nightOwlDeveloper + case .skilledDeveloper: + return Policy.Career.skilledDeveloper + case .famousDeveloper: + return Policy.Career.famousDeveloper + case .allRounderDeveloper: + return Policy.Career.allRounderDeveloper + case .worldClassDeveloper: + return Policy.Career.worldClassDeveloper + } + } + + /// 다음 Career 단계 + var nextCareer: Career? { + guard let currentIndex = Career.allCases.firstIndex(of: self), + currentIndex < Career.allCases.count - 1 + else { + return nil + } + return Career.allCases[currentIndex + 1] + } + + /// 현재 누적재산으로 업그레이드 가능한지 확인 + func canUpgrade(currentWealth: Int) -> Bool { + guard let next = nextCareer else { return false } + return currentWealth >= next.requiredWealth + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Inventory.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Inventory.swift new file mode 100644 index 00000000..120b4793 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Inventory.swift @@ -0,0 +1,60 @@ +// +// Inventory.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation +import Observation + +@MainActor +@Observable +final class Inventory { + + /// 장비 아이템 + let equipmentItems: [Equipment] + /// 소비 아이템 + let consumableItems: [Consumable] + /// 부동산 + var housing: Housing + + init( + equipmentItems: [Equipment] = [ + .init(type: .keyboard, tier: .broken), + .init(type: .mouse, tier: .broken), + .init(type: .monitor, tier: .broken), + .init(type: .chair, tier: .broken) + ], + consumableItems: [Consumable] = [ + .init(type: .coffee, count: 5), + .init(type: .energyDrink, count: 5) + ], + housing: Housing = .init(tier: .street) + ) { + self.equipmentItems = equipmentItems + self.consumableItems = consumableItems + self.housing = housing + } + + /// 소비 아이템을 사용합니다. + /// - Returns: 사용 성공 여부 + func drink(_ type: ConsumableType) -> Bool { + guard let targetItem = consumableItems.filter({ $0.type == type }).first else { return false } + guard targetItem.count > 0 else { return false } + targetItem.spendItem() + return true + } + + /// 소비 아이템 갯수 + /// - Parameter type: 소비 아이템 타입 + /// - Returns: 소비 아이템 갯수 + func count(_ type: ConsumableType) -> Int? { + guard let targetItem = consumableItems.filter({ $0.type == type }).first else { return nil } + return targetItem.count + } + + func gain(consumable: ConsumableType) { + consumableItems.filter { $0.type == consumable }.first?.addItem() + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Mission.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Mission.swift new file mode 100644 index 00000000..eca60e5c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Mission.swift @@ -0,0 +1,135 @@ +// +// Mission.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation +import Observation + +enum MissionCardState: Equatable { + case claimed + case claimable + case inProgress(currentValue: Int, totalValue: Int) +} + +enum MissionLevel { + /// 금 + case gold + /// 은 + case silver + /// 동 + case bronze + /// 특수 + case special + + var imageName: String { + switch self { + case .bronze: return "mission_trophy_bronze" + case .silver: return "mission_trophy_silver" + case .gold: return "mission_trophy_gold" + case .special: return "mission_trophy_special" + } + } +} + +@MainActor +@Observable +final class Mission { + enum State: Int { + /// "획득하기" + case claimable + /// "진행중" + case inProgress + /// "달성 완료" + case claimed + } + + /// 미션 아이디 + var id: Int + /// 미션 타입 + var type: MissionType + /// 업적 제목 + var title: String + /// 업적 설명 + var description: String + /// 목표 수치 + var targetValue: Int + /// 현재 수치 + var currentValue: Int + /// 진행 상태 + var progress: Double { + min(Double(currentValue) / Double(targetValue), 1.0) + } + /// 현재 미션 상태 + var state: State = .inProgress + /// 업데이트 조건 + var updateCondition: (Record) -> Int + /// 달성 조건 + var completeCondition: ((Record) -> Bool)? + /// 보상 + var reward: Cost + + /// 미션 카드 상태로 매핑 + var missionCardState: MissionCardState { + switch state { + case .claimed: + return .claimed + case .claimable: + return .claimable + case .inProgress: + return .inProgress( + currentValue: currentValue, + totalValue: targetValue + ) + } + } + + init( + id: Int, + type: MissionType, + title: String, + description: String, + targetValue: Int, + currentValue: Int = 0, + state: State = .inProgress, + updateCondition: @escaping (Record) -> Int, + completeCondition: ((Record) -> Bool)? = nil, + reward: Cost + ) { + self.id = id + self.type = type + self.title = title + self.description = description + self.targetValue = targetValue + self.currentValue = currentValue + self.state = state + self.updateCondition = updateCondition + self.completeCondition = completeCondition + self.reward = reward + } + + /// 최신 기록으로 업데이트하고, 완료 조건을 체크 + func update(record: Record) { + guard state == .inProgress else { return } + + // 현재 값 업데이트 + currentValue = updateCondition(record) + + // 완료 조건 체크 + let isComplete = completeCondition?(record) ?? (currentValue >= targetValue) + + if isComplete { + state = .claimable + } + } + + /// 미션을 완료하고 보상을 리턴합니다. + func claim() -> Cost { + guard state == .claimable else { return Cost() } + state = .claimed + + return reward + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Record.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Record.swift new file mode 100644 index 00000000..40853913 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Record.swift @@ -0,0 +1,185 @@ +// +// Record.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation +import Observation + +@MainActor +@Observable +final class Record { + /// 미션 시스템을 소유해서 기록을 반영합니다. + let missionSystem: MissionSystem + + init(missionSystem: MissionSystem = MissionSystem(missions: MissionFactory.createAllMissions())) { + self.missionSystem = missionSystem + } + + // MARK: - Financial Records + /// 누적 획득 재산 + var totalEarnedMoney: Int = 0 + /// 누적 소비 재산 + var totalSpentMoney: Int = 0 + /// 누적 스킬 업그레이드 비용 + var totalSkillUpgradeCost: Int = 0 + /// 누적 장비 강화 비용 + var totalEquipmentEnhancementCost: Int = 0 + /// 누적 소비 아이템 구입 비용 + var totalConsumablePurchaseCost: Int = 0 + /// 누적 부동산 이사 비용 + var totalHousingMoveCost: Int = 0 + + // MARK: - Tap Records + /// 총 탭 횟수 + var totalTapCount: Int = 0 + + // MARK: - Language Game Records + /// 언어 맞추기 성공 횟수 + var languageCorrectCount: Int = 0 + /// 언어 맞추기 연속 성공 횟수 + var languageConsecutiveCorrect: Int = 0 + + // MARK: - Bug Dodging Records + /// 버그피하기 골드 수집 횟수 + var dodgeGoldCollectedCount: Int = 0 + /// 버그피하기 최고 콤보 (역대 최고 연속 회피 횟수) + var dodgeMaxCombo: Int = 0 + /// 버그피하기 총 버그 회피 횟수 + var dodgeBugAvoidedCount: Int = 0 + /// 버그피하기 버그 수집 횟수 + var dodgeBugCollectCount: Int = 0 + + // MARK: - Stacking Game Records + /// 데이터 쌓기 성공 횟수 + var stackingSuccessCount: Int = 0 + /// 데이터 쌓기 연속 성공 횟수 + var stackConsecutiveSuccess: Int = 0 + + // MARK: - Consumable Usage Records + /// 커피 사용 횟수 + var coffeeUseCount: Int = 0 + /// 에너지 드링크 사용 횟수 + var energyDrinkUseCount: Int = 0 + + // MARK: - Play Time Records + /// 총 플레이 시간 + var totalPlayTime: TimeInterval = 0 + + // MARK: - Tutorial Records + /// 튜토리얼 클리어 여부 + var tutorialCompleted: Bool = false + + // MARK: - Career Records + /// 하찮은 개발자 달성 여부 + var hasAchievedJuniorDeveloper: Bool = false +} + +// MARK: - Record Event +extension Record { + enum Event { + // Tap + case tap(count: Int = 1) + + // Language Game + case languageCorrect + case languageIncorrect + + // Bug Dodging + case dodgeGoldCollect + case dodgeBugAvoid(currentCombo: Int) + case dodgeFail + + // Stacking Game + case stackingSuccess + case stackingFail + + // Consumables + case coffeeUse + case energyDrinkUse + + // Play Time + case playTime + + // Financial + case earnMoney(Int) + case spendMoney(Int) + case skillUpgrade(cost: Int) + case equipmentEnhancement(cost: Int) + case consumablePurchase(cost: Int) + case housingMove(cost: Int) + + // Achievements + case tutorialComplete + case juniorDeveloperAchieve + } + + /// Record Event 기록 + func record(_ event: Event) { + switch event { + case .tap(let count): + totalTapCount += count + + case .languageCorrect: + languageCorrectCount += 1 + languageConsecutiveCorrect += 1 + + case .languageIncorrect: + languageConsecutiveCorrect = 0 + + case .dodgeGoldCollect: + dodgeGoldCollectedCount += 1 + + case .dodgeBugAvoid(let currentCombo): + dodgeBugAvoidedCount += 1 + dodgeMaxCombo = max(currentCombo, dodgeMaxCombo) + + case .dodgeFail: + dodgeBugCollectCount += 1 + + case .stackingSuccess: + stackingSuccessCount += 1 + stackConsecutiveSuccess += 1 + + case .stackingFail: + stackConsecutiveSuccess = 0 + + case .coffeeUse: + coffeeUseCount += 1 + + case .energyDrinkUse: + energyDrinkUseCount += 1 + + case .playTime: + totalPlayTime += 1 + + case .earnMoney(let amount): + totalEarnedMoney += amount + + case .spendMoney(let amount): + totalSpentMoney += amount + + case .skillUpgrade(let cost): + totalSkillUpgradeCost += cost + + case .equipmentEnhancement(let cost): + totalEquipmentEnhancementCost += cost + + case .consumablePurchase(let cost): + totalConsumablePurchaseCost += cost + + case .housingMove(let cost): + totalHousingMoveCost += cost + + case .tutorialComplete: + tutorialCompleted = true + + case .juniorDeveloperAchieve: + hasAchievedJuniorDeveloper = true + } + /// 미션 상태 업데이트 + missionSystem.updateCompletedMissions(record: self) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/Skill.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/Skill.swift new file mode 100644 index 00000000..2bdc3423 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/Skill.swift @@ -0,0 +1,180 @@ +// +// Skill.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +/// 스킬 정보를 키로 식별하여 관리 +struct SkillKey: Hashable { + let game: GameType + let tier: SkillTier +} + +final class Skill: Hashable { + /// 고유 스킬 정보 (게임 종류, 스킬 티어) + let key: SkillKey + + /// 스킬 레벨 + private(set) var level: Int + + /// 획득 재화량 + var gainGold: Double { + let baseGold: Int + let multiplier: Int + + switch key.game { + case .tap: + baseGold = Policy.Skill.Tap.baseGold + switch key.tier { + case .beginner: + multiplier = Policy.Skill.Tap.beginnerGoldMultiplier + case .intermediate: + multiplier = Policy.Skill.Tap.intermediateGoldMultiplier + case .advanced: + multiplier = Policy.Skill.Tap.advancedGoldMultiplier + } + case .language: + baseGold = Policy.Skill.Language.baseGold + switch key.tier { + case .beginner: + multiplier = Policy.Skill.Language.beginnerGoldMultiplier + case .intermediate: + multiplier = Policy.Skill.Language.intermediateGoldMultiplier + case .advanced: + multiplier = Policy.Skill.Language.advancedGoldMultiplier + } + case .dodge: + baseGold = Policy.Skill.Dodge.baseGold + switch key.tier { + case .beginner: + multiplier = Policy.Skill.Dodge.beginnerGoldMultiplier + case .intermediate: + multiplier = Policy.Skill.Dodge.intermediateGoldMultiplier + case .advanced: + multiplier = Policy.Skill.Dodge.advancedGoldMultiplier + } + case .stack: + baseGold = Policy.Skill.Stack.baseGold + switch key.tier { + case .beginner: + multiplier = Policy.Skill.Stack.beginnerGoldMultiplier + case .intermediate: + multiplier = Policy.Skill.Stack.intermediateGoldMultiplier + case .advanced: + multiplier = Policy.Skill.Stack.advancedGoldMultiplier + } + } + + return Double(baseGold + multiplier * level) + } + + /// 이미지 리소스 + var imageName: String { + let gameName: String = { + switch key.game { + case .tap: return "tap" + case .language: return "language" + case .dodge: return "dodge" + case .stack: return "stack" + } + }() + + let tierNumber: Int = { + switch key.tier { + case .beginner: return 1 + case .intermediate: return 2 + case .advanced: return 3 + } + }() + + return "skill_\(gameName)_\(tierNumber)" + } + + /// 스킬 타이틀 + var title: String { + return "\(key.game.displayTitle) \(key.tier.displayTitle) Lv.\(level)" + } + + /// 업그레이드 비용 + var upgradeCost: Cost { + let goldCostMultiplier: Int + let diamondCostDivider: Int + let diamondCostMultiplier: Int + + switch key.game { + case .tap: + switch key.tier { + case .beginner: + goldCostMultiplier = Policy.Skill.Tap.beginnerGoldCostMultiplier + case .intermediate: + goldCostMultiplier = Policy.Skill.Tap.intermediateGoldCostMultiplier + case .advanced: + goldCostMultiplier = Policy.Skill.Tap.advancedGoldCostMultiplier + } + diamondCostDivider = Policy.Skill.Tap.diamondCostDivider + diamondCostMultiplier = Policy.Skill.Tap.diamondCostMultiplier + case .language: + switch key.tier { + case .beginner: + goldCostMultiplier = Policy.Skill.Language.beginnerGoldCostMultiplier + case .intermediate: + goldCostMultiplier = Policy.Skill.Language.intermediateGoldCostMultiplier + case .advanced: + goldCostMultiplier = Policy.Skill.Language.advancedGoldCostMultiplier + } + diamondCostDivider = Policy.Skill.Language.diamondCostDivider + diamondCostMultiplier = Policy.Skill.Language.diamondCostMultiplier + case .dodge: + switch key.tier { + case .beginner: + goldCostMultiplier = Policy.Skill.Dodge.beginnerGoldCostMultiplier + case .intermediate: + goldCostMultiplier = Policy.Skill.Dodge.intermediateGoldCostMultiplier + case .advanced: + goldCostMultiplier = Policy.Skill.Dodge.advancedGoldCostMultiplier + } + diamondCostDivider = Policy.Skill.Dodge.diamondCostDivider + diamondCostMultiplier = Policy.Skill.Dodge.diamondCostMultiplier + case .stack: + switch key.tier { + case .beginner: + goldCostMultiplier = Policy.Skill.Stack.beginnerGoldCostMultiplier + case .intermediate: + goldCostMultiplier = Policy.Skill.Stack.intermediateGoldCostMultiplier + case .advanced: + goldCostMultiplier = Policy.Skill.Stack.advancedGoldCostMultiplier + } + diamondCostDivider = Policy.Skill.Stack.diamondCostDivider + diamondCostMultiplier = Policy.Skill.Stack.diamondCostMultiplier + } + + return .init( + gold: goldCostMultiplier * level, + diamond: level == 0 || ((level+1) % diamondCostDivider == 0) ? diamondCostMultiplier : 0 + ) + } + + init(key: SkillKey, level: Int? = nil) { + self.key = key + self.level = key.tier.levelRange.clamped(level ?? key.tier.levelRange.minValue) + } + + /// 해당 스킬의 레벨을 1 상승 시킵니다. + func upgrade() throws { + guard key.tier.levelRange.canUpgrade(from: level) else { + throw SkillError.levelExceeded + } + level += 1 + } + + static func == (lhs: Skill, rhs: Skill) -> Bool { + lhs.key == rhs.key + } + + func hash(into hasher: inout Hasher) { + hasher.combine(key) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/SkillError.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/SkillError.swift new file mode 100644 index 00000000..f3cd11f5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/SkillError.swift @@ -0,0 +1,11 @@ +// +// SkillError.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/20/26. +// + +enum SkillError: Error { + /// 레벨 한도 초과 + case levelExceeded +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/SkillTier.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/SkillTier.swift new file mode 100644 index 00000000..18665cc3 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Skill/SkillTier.swift @@ -0,0 +1,55 @@ +// +// SkillTier.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/20/26. +// + +/// 스킬 등급 +enum SkillTier: Int, CaseIterable { + case beginner = 0 + case intermediate = 1 + case advanced = 2 + + var displayTitle: String { + switch self { + case .beginner: "초급" + case .intermediate: "중급" + case .advanced: "고급" + } + } + + var levelRange: LevelRange { + switch self { + case .beginner: + LevelRange( + minValue: Policy.Skill.beginnerMinLevel, + maxValue: Policy.Skill.beginnerMaxLevel + ) + case .intermediate: + LevelRange( + minValue: Policy.Skill.intermediateMinLevel, + maxValue: Policy.Skill.intermediateMaxLevel + ) + case .advanced: + LevelRange( + minValue: Policy.Skill.advancedMinLevel, + maxValue: Policy.Skill.advancedMaxLevel + ) + } + } +} + +/// 레벨 범위 표현을 위한 타입 +struct LevelRange { + let minValue: Int + let maxValue: Int + + func clamped(_ level: Int) -> Int { + max(minValue, min(maxValue, level)) + } + + func canUpgrade(from level: Int) -> Bool { + level < maxValue + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/User.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/User.swift new file mode 100644 index 00000000..24688128 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/User.swift @@ -0,0 +1,80 @@ +// +// User.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation + +@MainActor +final class User { + /// 유저 식별용 ID + let id: UUID + /// 유저 닉네임 + let nickname: String + /// 커리어 + var career: Career + /// 지갑 [재산, 다이아] + let wallet: Wallet + /// 인벤토리 [장비, 소비, 부동산] 아이템 + let inventory: Inventory + /// 게임 기록 + let record: Record + + /// 보유 스킬 + let skills: Set + + init( + id: UUID = UUID(), + nickname: String, + career: Career = .unemployed, + wallet: Wallet, + inventory: Inventory, + record: Record, + skills: Set = [] + ) { + self.id = id + self.nickname = nickname + self.career = career + self.wallet = wallet + self.inventory = inventory + self.record = record + self.skills = skills + } + + func updateCareer(to newCareer: Career) { + career = newCareer + } + + convenience init(nickname: String) { + self.init( + nickname: nickname, + wallet: .init(), + inventory: .init( + equipmentItems: [ + .init(type: .chair, tier: .broken), + .init(type: .keyboard, tier: .broken), + .init(type: .monitor, tier: .broken), + .init(type: .mouse, tier: .broken) + ], + housing: .init(tier: .street) + ), + record: .init(), + skills: [ + .init(key: SkillKey(game: .tap, tier: .beginner)), + .init(key: SkillKey(game: .tap, tier: .intermediate)), + .init(key: SkillKey(game: .tap, tier: .advanced)), + .init(key: SkillKey(game: .language, tier: .beginner)), + .init(key: SkillKey(game: .language, tier: .intermediate)), + .init(key: SkillKey(game: .language, tier: .advanced)), + .init(key: SkillKey(game: .dodge, tier: .beginner)), + .init(key: SkillKey(game: .dodge, tier: .intermediate)), + .init(key: SkillKey(game: .dodge, tier: .advanced)), + .init(key: SkillKey(game: .stack, tier: .beginner)), + .init(key: SkillKey(game: .stack, tier: .intermediate)), + .init(key: SkillKey(game: .stack, tier: .advanced)) + ] + ) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Wallet.swift b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Wallet.swift new file mode 100644 index 00000000..036cffa2 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/GameCore/Models/User/Wallet.swift @@ -0,0 +1,65 @@ +// +// Wallet.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/6/26. +// + +import Foundation +import Observation + +/// 게임 내 재화 관리 클래스 +@MainActor +@Observable +final class Wallet { + + // MARK: - Properties + /// 현재 보유 골드 + private(set) var gold: Int + /// 현재 보유 다이아몬드 + private(set) var diamond: Int + + // MARK: - Initialization + /// 지갑 초기화 + init(gold: Int = 0, diamond: Int = 0) { + self.gold = gold + self.diamond = diamond + } + + /// 비용 지불 가능 여부 확인 + func canAfford(_ cost: Cost) -> Bool { + return gold >= cost.gold && diamond >= cost.diamond + } + + /// 골드 획득 + func addGold(_ amount: Int) { + guard amount > 0 else { return } + gold += amount + } + + /// 골드 소모 + /// - Returns: 성공 여부 + @discardableResult + func spendGold(_ amount: Int) -> Bool { + guard amount > 0 else { return false } + guard gold >= amount else { return false } + gold -= amount + return true + } + + /// 다이아몬드 획득 + func addDiamond(_ amount: Int) { + guard amount > 0 else { return } + diamond += amount + } + + /// 다이아몬드 소모 + /// - Returns: 성공 여부 + @discardableResult + func spendDiamond(_ amount: Int) -> Bool { + guard amount > 0 else { return false } + guard diamond >= amount else { return false } + diamond -= amount + return true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/CareerDTO.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/CareerDTO.swift new file mode 100644 index 00000000..7aa87e67 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/CareerDTO.swift @@ -0,0 +1,20 @@ +// +// CareerDTO.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-26. +// + +import Foundation + +struct CareerDTO: Codable { + let rawValue: String + + init(from career: Career) { + self.rawValue = career.rawValue + } + + func toCareer() -> Career { + Career(rawValue: rawValue) ?? .unemployed + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/InventoryDTO.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/InventoryDTO.swift new file mode 100644 index 00000000..de79be08 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/InventoryDTO.swift @@ -0,0 +1,134 @@ +// +// InventoryDTO.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-26. +// + +import Foundation + +struct InventoryDTO: Codable { + let equipmentItems: [EquipmentDTO] + let consumableItems: [ConsumableDTO] + let housing: HousingDTO + + init(from inventory: Inventory) { + self.equipmentItems = inventory.equipmentItems.map { EquipmentDTO(from: $0) } + self.consumableItems = inventory.consumableItems.map { ConsumableDTO(from: $0) } + self.housing = HousingDTO(from: inventory.housing) + } + + func toInventory() -> Inventory { + Inventory( + equipmentItems: equipmentItems.map { $0.toEquipment() }, + consumableItems: consumableItems.map { $0.toConsumable() }, + housing: housing.toHousing() + ) + } +} + +struct EquipmentDTO: Codable { + let type: EquipmentTypeDTO + let tier: EquipmentTierDTO + + init(from equipment: Equipment) { + self.type = EquipmentTypeDTO(from: equipment.type) + self.tier = EquipmentTierDTO(from: equipment.tier) + } + + func toEquipment() -> Equipment { + Equipment(type: type.toEquipmentType(), tier: tier.toEquipmentTier()) + } +} + +struct EquipmentTypeDTO: Codable { + let rawValue: String + + init(from type: EquipmentType) { + switch type { + case .keyboard: self.rawValue = "keyboard" + case .mouse: self.rawValue = "mouse" + case .monitor: self.rawValue = "monitor" + case .chair: self.rawValue = "chair" + } + } + + func toEquipmentType() -> EquipmentType { + switch rawValue { + case "keyboard": return .keyboard + case "mouse": return .mouse + case "monitor": return .monitor + case "chair": return .chair + default: return .keyboard + } + } +} + +struct EquipmentTierDTO: Codable { + let rawValue: Int + + init(from tier: EquipmentTier) { + self.rawValue = tier.rawValue + } + + func toEquipmentTier() -> EquipmentTier { + EquipmentTier(rawValue: rawValue) ?? .broken + } +} + +struct ConsumableDTO: Codable { + let type: ConsumableTypeDTO + let count: Int + + init(from consumable: Consumable) { + self.type = ConsumableTypeDTO(from: consumable.type) + self.count = consumable.count + } + + func toConsumable() -> Consumable { + Consumable(type: type.toConsumableType(), count: count) + } +} + +struct ConsumableTypeDTO: Codable { + let rawValue: String + + init(from type: ConsumableType) { + switch type { + case .coffee: self.rawValue = "coffee" + case .energyDrink: self.rawValue = "energyDrink" + } + } + + func toConsumableType() -> ConsumableType { + switch rawValue { + case "coffee": return .coffee + case "energyDrink": return .energyDrink + default: return .coffee + } + } +} + +struct HousingDTO: Codable { + let tier: HousingTierDTO + + init(from housing: Housing) { + self.tier = HousingTierDTO(from: housing.tier) + } + + func toHousing() -> Housing { + Housing(tier: tier.toHousingTier()) + } +} + +struct HousingTierDTO: Codable { + let rawValue: Int + + init(from tier: HousingTier) { + self.rawValue = tier.rawValue + } + + func toHousingTier() -> HousingTier { + HousingTier(rawValue: rawValue) ?? .street + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/RecordDTO.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/RecordDTO.swift new file mode 100644 index 00000000..3f19970c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/RecordDTO.swift @@ -0,0 +1,151 @@ +// +// RecordDTO.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-26. +// + +import Foundation + +struct RecordDTO: Codable { + // Financial Records + let totalEarnedMoney: Int + let totalSpentMoney: Int + let totalSkillUpgradeCost: Int + let totalEquipmentEnhancementCost: Int + let totalConsumablePurchaseCost: Int + let totalHousingMoveCost: Int + + // Tap Records + let totalTapCount: Int + + // Language Game Records + let languageCorrectCount: Int + let languageConsecutiveCorrect: Int + + // Bug Dodging Records + let dodgeGoldCollectedCount: Int + let dodgeMaxCombo: Int + let dodgeBugAvoidedCount: Int + let dodgeBugCollectCount: Int + + // Stacking Game Records + let stackingSuccessCount: Int + let stackConsecutiveSuccess: Int + + // Consumable Usage Records + let coffeeUseCount: Int + let energyDrinkUseCount: Int + + // Play Time Records + let totalPlayTime: TimeInterval + + // Tutorial Records + let tutorialCompleted: Bool + + // Career Records + let hasAchievedJuniorDeveloper: Bool + + // Mission States + let missionStates: [MissionStateDTO] + + init(from record: Record) { + self.totalEarnedMoney = record.totalEarnedMoney + self.totalSpentMoney = record.totalSpentMoney + self.totalSkillUpgradeCost = record.totalSkillUpgradeCost + self.totalEquipmentEnhancementCost = record.totalEquipmentEnhancementCost + self.totalConsumablePurchaseCost = record.totalConsumablePurchaseCost + self.totalHousingMoveCost = record.totalHousingMoveCost + self.totalTapCount = record.totalTapCount + self.languageCorrectCount = record.languageCorrectCount + self.languageConsecutiveCorrect = record.languageConsecutiveCorrect + self.dodgeGoldCollectedCount = record.dodgeGoldCollectedCount + self.dodgeMaxCombo = record.dodgeMaxCombo + self.dodgeBugAvoidedCount = record.dodgeBugAvoidedCount + self.dodgeBugCollectCount = record.dodgeBugCollectCount + self.stackingSuccessCount = record.stackingSuccessCount + self.stackConsecutiveSuccess = record.stackConsecutiveSuccess + self.coffeeUseCount = record.coffeeUseCount + self.energyDrinkUseCount = record.energyDrinkUseCount + self.totalPlayTime = record.totalPlayTime + self.tutorialCompleted = record.tutorialCompleted + self.hasAchievedJuniorDeveloper = record.hasAchievedJuniorDeveloper + self.missionStates = record.missionSystem.missions.map { MissionStateDTO(from: $0) } + } + + func toRecord() -> Record { + let record = Record(missionSystem: MissionSystem(missions: MissionFactory.createAllMissions())) + + // Financial Records + record.totalEarnedMoney = totalEarnedMoney + record.totalSpentMoney = totalSpentMoney + record.totalSkillUpgradeCost = totalSkillUpgradeCost + record.totalEquipmentEnhancementCost = totalEquipmentEnhancementCost + record.totalConsumablePurchaseCost = totalConsumablePurchaseCost + record.totalHousingMoveCost = totalHousingMoveCost + + // Tap Records + record.totalTapCount = totalTapCount + + // Language Game Records + record.languageCorrectCount = languageCorrectCount + record.languageConsecutiveCorrect = languageConsecutiveCorrect + + // Bug Dodging Records + record.dodgeGoldCollectedCount = dodgeGoldCollectedCount + record.dodgeMaxCombo = dodgeMaxCombo + record.dodgeBugAvoidedCount = dodgeBugAvoidedCount + record.dodgeBugCollectCount = dodgeBugCollectCount + + // Stacking Game Records + record.stackingSuccessCount = stackingSuccessCount + record.stackConsecutiveSuccess = stackConsecutiveSuccess + + // Consumable Usage Records + record.coffeeUseCount = coffeeUseCount + record.energyDrinkUseCount = energyDrinkUseCount + + // Play Time Records + record.totalPlayTime = totalPlayTime + + // Tutorial Records + record.tutorialCompleted = tutorialCompleted + + // Career Records + record.hasAchievedJuniorDeveloper = hasAchievedJuniorDeveloper + + // Mission States 복원 + for missionState in missionStates { + if let mission = record.missionSystem.missions.first(where: { $0.id == missionState.id }) { + mission.currentValue = missionState.currentValue + mission.state = missionState.state.toMissionState() + } + } + + return record + } +} + +struct MissionStateDTO: Codable { + let id: Int + let currentValue: Int + let state: MissionStateRawDTO + + init(from mission: Mission) { + self.id = mission.id + self.currentValue = mission.currentValue + self.state = MissionStateRawDTO(from: mission.state) + } +} + +struct MissionStateRawDTO: Codable { + let rawValue: Int + + init(from state: Mission.State) { + self.rawValue = state.rawValue + } + + func toMissionState() -> Mission.State { + Mission.State(rawValue: rawValue) ?? .inProgress + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/SkillDTO.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/SkillDTO.swift new file mode 100644 index 00000000..93b50442 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/SkillDTO.swift @@ -0,0 +1,48 @@ +// +// SkillDTO.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-26. +// + +import Foundation + +struct SkillDTO: Codable { + let game: GameTypeDTO + let tier: SkillTierDTO + let level: Int + + init(from skill: Skill) { + self.game = GameTypeDTO(from: skill.key.game) + self.tier = SkillTierDTO(from: skill.key.tier) + self.level = skill.level + } + + func toSkill() -> Skill { + Skill(key: SkillKey(game: game.toGameType(), tier: tier.toSkillTier()), level: level) + } +} + +struct GameTypeDTO: Codable { + let rawValue: Int + + init(from type: GameType) { + self.rawValue = type.rawValue + } + + func toGameType() -> GameType { + GameType(rawValue: rawValue) ?? .tap + } +} + +struct SkillTierDTO: Codable { + let rawValue: Int + + init(from tier: SkillTier) { + self.rawValue = tier.rawValue + } + + func toSkillTier() -> SkillTier { + SkillTier(rawValue: rawValue) ?? .beginner + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/UserDTO.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/UserDTO.swift new file mode 100644 index 00000000..76b900ce --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/UserDTO.swift @@ -0,0 +1,18 @@ +// +// UserDTO.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-26. +// + +import Foundation + +struct UserDTO: Codable { + let id: UUID + let nickname: String + let career: CareerDTO + let wallet: WalletDTO + let inventory: InventoryDTO + let record: RecordDTO + let skills: [SkillDTO] +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/WalletDTO.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/WalletDTO.swift new file mode 100644 index 00000000..2d628c24 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/DTOs/WalletDTO.swift @@ -0,0 +1,22 @@ +// +// WalletDTO.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-26. +// + +import Foundation + +struct WalletDTO: Codable { + let gold: Int + let diamond: Int + + init(from wallet: Wallet) { + self.gold = wallet.gold + self.diamond = wallet.diamond + } + + func toWallet() -> Wallet { + Wallet(gold: gold, diamond: diamond) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/Repository/UserRepository.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/Repository/UserRepository.swift new file mode 100644 index 00000000..766671be --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Data/Repository/UserRepository.swift @@ -0,0 +1,67 @@ +// +// UserRepository.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 1/26/26. +// + +import Foundation + +protocol UserRepository { + func save(_ user: User) async throws + func load() async throws -> User? +} + +final class FileManagerUserRepository: UserRepository { + private let fileManager = FileManager() + private let fileName = "user_data.json" + + private var fileURL: URL { + let documentsDirectory = fileManager.urls(for: .documentDirectory, in: .userDomainMask)[0] + return documentsDirectory.appendingPathComponent(fileName) + } + + func save(_ user: User) async throws { + let id = user.id + let nickname = user.nickname + let career = user.career + let wallet = user.wallet + let inventory = user.inventory + let record = user.record + let skills = Array(user.skills) + + let userDTO = UserDTO( + id: id, + nickname: nickname, + career: CareerDTO(from: career), + wallet: WalletDTO(from: wallet), + inventory: InventoryDTO(from: inventory), + record: RecordDTO(from: record), + skills: skills.map { SkillDTO(from: $0) } + ) + + let url = fileURL + let data = try JSONEncoder().encode(userDTO) + + Task.detached { + try data.write(to: url, options: [.atomic]) + } + } + + func load() async throws -> User? { + guard fileManager.fileExists(atPath: fileURL.path) else { return nil } + + let data = try Data(contentsOf: fileURL) + let userDTO = try JSONDecoder().decode(UserDTO.self, from: data) + + return User( + id: userDTO.id, + nickname: userDTO.nickname, + career: userDTO.career.toCareer(), + wallet: userDTO.wallet.toWallet(), + inventory: userDTO.inventory.toInventory(), + record: userDTO.record.toRecord(), + skills: Set(userDTO.skills.map { $0.toSkill() }) + ) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/PurchasingError+UserReadableError.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/PurchasingError+UserReadableError.swift new file mode 100644 index 00000000..005ec355 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/PurchasingError+UserReadableError.swift @@ -0,0 +1,17 @@ +// +// PurchasingError+UserReadableError.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/20/26. +// + +extension PurchasingError: UserReadableError { + var message: String { + switch self { + case .insufficientGold: "골드가 부족합니다." + case .insufficientDiamond: "다이아가 부족합니다." + case .locked: "해당 항목은 잠겨있습니다." + case .purchaseFailed: "구매에 실패했습니다." + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/SkillError+UserReadableError.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/SkillError+UserReadableError.swift new file mode 100644 index 00000000..52cf8107 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/SkillError+UserReadableError.swift @@ -0,0 +1,14 @@ +// +// SkillError+UserReadableError.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/20/26. +// + +extension SkillError: UserReadableError { + var message: String { + switch self { + case .levelExceeded: "이미 최대 레벨입니다." + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/UserReadableError.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/UserReadableError.swift new file mode 100644 index 00000000..29549f73 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Error/UserReadableError.swift @@ -0,0 +1,10 @@ +// +// UserReadableError.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/20/26. +// + +protocol UserReadableError { + var message: String { get } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/HapticService.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/HapticService.swift new file mode 100644 index 00000000..a2cda69c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/HapticService.swift @@ -0,0 +1,44 @@ +// +// HapticService.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/22/26. +// + +import SwiftUI + +private enum Constant { + static let hapticEnabledKey: String = "isHapticEnabled" +} + +@Observable +final class HapticService { + static let shared = HapticService() + private let localStorage: KeyValueLocalStorage = UserDefaultsStorage() + + var isEnabled: Bool { + didSet { + localStorage.set(isEnabled, forKey: Constant.hapticEnabledKey) + } + } + + private init() { + // 저장된 키가 없을 경우 기본값 등록 + localStorage.register(defaults: [Constant.hapticEnabledKey: true]) + + self.isEnabled = localStorage.bool(key: Constant.hapticEnabledKey) + } + + func toggle() { + isEnabled.toggle() + // 활성화 되었음을 알리기 위해 햅틱 트리거 + if isEnabled { + HapticType.medium.trigger() + } + } + + func trigger(_ type: HapticType) { + guard isEnabled else { return } + type.trigger() + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/HapticType.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/HapticType.swift new file mode 100644 index 00000000..2f751ebf --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/HapticType.swift @@ -0,0 +1,60 @@ +// +// HapticType.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/22/26. +// + +import UIKit + +enum HapticType { + // Impact + /// 가볍고 작게 1번 + case light + /// 중간 1번 + case medium + /// 둔탁하게 1번 + case heavy + // Notification + /// 빠르게 2번, 점점 세기 강해짐 + case success + /// 빠르게 4번 + case warning + /// 빠르게 2번, 점점 세기 약해짐 + case error + + // 모든 타입을 실제 피드백 발생으로 매핑 + func trigger() { + switch self { + // MARK: - Impact + case .light, .medium, .heavy: + let generator = UIImpactFeedbackGenerator(style: style) + generator.prepare() + generator.impactOccurred() + + // MARK: - Notification + case .success: + let generator = UINotificationFeedbackGenerator() + generator.prepare() + generator.notificationOccurred(.success) + case .warning: + let generator = UINotificationFeedbackGenerator() + generator.prepare() + generator.notificationOccurred(.warning) + case .error: + let generator = UINotificationFeedbackGenerator() + generator.prepare() + generator.notificationOccurred(.error) + } + } + + /// Impact 스타일 매핑 + private var style: UIImpactFeedbackGenerator.FeedbackStyle { + switch self { + case .light: return .light + case .medium: return .medium + case .heavy: return .heavy + default: return .medium + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/SoundService.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/SoundService.swift new file mode 100644 index 00000000..2c5f3268 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/SoundService.swift @@ -0,0 +1,144 @@ +// +// HapticService.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/22/26. +// + +import SwiftUI +import AVFoundation + +private enum Constant { + static let sfxEnabledKey: String = "isSFXEnabled" + static let bgmEnabledKey: String = "isBGMEnabled" + static let bgmVolumeKey: String = "bgmVolume" + static let sfxVolumeKey: String = "sfxVolume" + static let volumeRange: ClosedRange = 0 ... 100 + static let defaultVolume: Int = 100 + /// 효과음 동시 재생 상한 (중첩 허용) + static let maxConcurrentSFX: Int = 10 +} + +@Observable +final class SoundService { + static let shared = SoundService() + + private var sfxPlayers: [AVAudioPlayer] = [] + private var bgmPlayer: AVAudioPlayer? + private let sfxDelegate = SoundPlayerDelegate() + private let localStorage: KeyValueLocalStorage = UserDefaultsStorage() + + var isSFXEnabled: Bool { + didSet { + localStorage.set(isSFXEnabled, forKey: Constant.sfxEnabledKey) + } + } + + var isBGMEnabled: Bool { + didSet { + localStorage.set(isBGMEnabled, forKey: Constant.bgmEnabledKey) + if isBGMEnabled { + playBGM() + } else { + stopBGM() + } + } + } + + var bgmVolume: Int { + didSet { + localStorage.set(bgmVolume, forKey: Constant.bgmVolumeKey) + bgmPlayer?.volume = Float(bgmVolume) / 100 + } + } + + var sfxVolume: Int { + didSet { + localStorage.set(sfxVolume, forKey: Constant.sfxVolumeKey) + } + } + + private init() { + localStorage.register(defaults: [ + Constant.sfxEnabledKey: true, + Constant.bgmEnabledKey: true, + Constant.bgmVolumeKey: Constant.defaultVolume, + Constant.sfxVolumeKey: Constant.defaultVolume + ]) + + self.isSFXEnabled = localStorage.bool(key: Constant.sfxEnabledKey) + self.isBGMEnabled = localStorage.bool(key: Constant.bgmEnabledKey) + let storedBgm = localStorage.integer(key: Constant.bgmVolumeKey) + let storedSfx = localStorage.integer(key: Constant.sfxVolumeKey) + self.bgmVolume = Constant.volumeRange.contains(storedBgm) ? storedBgm : Constant.defaultVolume + self.sfxVolume = Constant.volumeRange.contains(storedSfx) ? storedSfx : Constant.defaultVolume + + try? AVAudioSession.sharedInstance().setCategory( + .playback, // 무음모드 무시 + options: [.mixWithOthers] + ) + sfxDelegate.onFinish = { [weak self] player in + self?.removeFinishedSFXPlayer(player) + } + } + + func removeFinishedSFXPlayer(_ player: AVAudioPlayer) { + sfxPlayers.removeAll { $0 === player } + } + + func trigger(_ sound: SoundType) { + guard isSFXEnabled else { return } + guard let url = sound.url else { return } + if sfxPlayers.count >= Constant.maxConcurrentSFX { return } + do { + let player = try AVAudioPlayer(contentsOf: url) + player.volume = Float(sfxVolume) / 100 + player.delegate = sfxDelegate + player.prepareToPlay() + player.play() + sfxPlayers.append(player) + } catch { + print("소리를 재생할 수 없음", error) + } + } + + /// 재생 중인 효과음 전부 정지 (게임 일시정지·뷰 이탈 시 등) + func stopAllSFX() { + sfxPlayers.forEach { $0.stop() } + sfxPlayers.removeAll() + } + + // MARK: - BGM + + func playBGM() { + guard isBGMEnabled else { return } + guard let url = SoundType.bgm.url else { return } + stopBGM() + do { + let player = try AVAudioPlayer(contentsOf: url) + player.numberOfLoops = -1 + player.volume = Float(bgmVolume) / 100 + player.prepareToPlay() + player.play() + bgmPlayer = player + } catch { + print("BGM 재생 실패", error) + } + } + + func stopBGM() { + bgmPlayer?.stop() + bgmPlayer = nil + } +} + +// MARK: - SFX 재생 완료 처리 (중첩 재생용) +private final class SoundPlayerDelegate: NSObject, AVAudioPlayerDelegate { + var onFinish: ((AVAudioPlayer) -> Void)? + + func audioPlayerDidFinishPlaying(_ player: AVAudioPlayer, successfully flag: Bool) { + DispatchQueue.main.async { [weak self] in + self?.onFinish?(player) + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/SoundType.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/SoundType.swift new file mode 100644 index 00000000..607e022d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/FeedbackSystem/SoundType.swift @@ -0,0 +1,68 @@ +// +// SoundType.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/22/26. +// + +import Foundation + +enum SoundType: String { + /// 전체 버튼 클릭음 + case buttonTap + + // MARK: - BGM + case bgm + + // MARK: - 언어 맞추기 + /// 맞았을 때 + case languageCorrect + /// 틀렸을 때 + case languageWrong + + // MARK: - 버그 피하기 + /// 코인 먹는 소리 + case coinCollect + /// 버그 맞는 소리 + case bugHit + + // MARK: - 데이터 쌓기 + /// 블록 쌓기 + case blockStack + /// 블록 떨굼 + case blockDrop + /// 폭탄 쌓기 + case bombStack + + // MARK: - 퀴즈 + /// 끝나기 3초 전 째깍 + case quizCountdown + /// 퀴즈 시간 초과 + case quizTimeOver + /// 퀴즈 정답 + case quizCorrect + /// 퀴즈 오답 + case quizWrong + + // MARK: - 아이템 소비 + /// 커피/박하스 클릭 시 + case itemConsume + + // MARK: - 장비 강화 + /// 강화 성공 (팝업 뜰 때) + case upgradeSuccess + /// 강화 실패 (팝업 뜰 때) + case upgradeFailure + + // MARK: - 미션 + /// 미션 획득 시 + case missionAcquired + + /// 탭게임 탭 시 + case tapGameTyping + + /// wav 우선, 없으면 mp3 로드 + var url: URL? { + Bundle.main.url(forResource: rawValue, withExtension: "wav") + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/CareerPopupView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/CareerPopupView.swift new file mode 100644 index 00000000..0463dca5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/CareerPopupView.swift @@ -0,0 +1,52 @@ +// +// CareerPopupView.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/22/26. +// + +import SwiftUI + +private enum Constant { + static let contentHorizontalPadding: CGFloat = 16 + static let progressBarTopPadding: CGFloat = 18 + static let progressBarBottomPadding: CGFloat = 18 + static let careerRowSpacing: CGFloat = 10 + static let scrollViewBottomPadding: CGFloat = 45 +} + +struct CareerPopupView: View { + let careerSystem: CareerSystem + let user: User + let onClose: () -> Void + + var body: some View { + VStack(alignment: .center, spacing: 0) { + CareerProgressBar( + career: careerSystem.currentCareer, + totalEarnedMoney: user.record.totalEarnedMoney, + progress: careerSystem.careerProgress + ) + .padding(.bottom, Constant.progressBarBottomPadding) + .padding(.top, Constant.progressBarTopPadding) + + ScrollView { + VStack(spacing: Constant.careerRowSpacing) { + ForEach(Career.allCases, id: \.self) { career in + CareerRow( + career: career, + userCareer: careerSystem.currentCareer + ) + } + } + } + .scrollIndicators(.never) + .padding(.bottom, Constant.scrollViewBottomPadding) + + MediumButton(title: "닫기", isFilled: true) { + onClose() + } + } + .padding(.horizontal, Constant.contentHorizontalPadding) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/CharacterScene.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/CharacterScene.swift new file mode 100644 index 00000000..b85b8eff --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/CharacterScene.swift @@ -0,0 +1,117 @@ +// +// CharacterScene.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/20/26. +// + +import SpriteKit +import SwiftUI + +private enum Constant { + static let characterSize = CGSize(width: 100, height: 100) + static let blinkDuration: TimeInterval = 0.1 + static let blinkInterval: TimeInterval = 3.5 + static let smileDuration: TimeInterval = 0.5 +} + +final class CharacterScene: SKScene { + + // MARK: - Properties + private var characterSprite: SKSpriteNode? + private var user: User + + init(size: CGSize, user: User) { + self.user = user + super.init(size: size) + } + + required init?(coder aDecoder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + // MARK: - Computed Properties (텍스처) + private var idleTexture: SKTexture { + SKTexture(imageNamed: "\(user.career.characterImagePrefix)_default") + } + + private var blinkTexture: SKTexture { + SKTexture(imageNamed: "\(user.career.characterImagePrefix)_close") + } + + private var smileTexture: SKTexture { + SKTexture(imageNamed: "\(user.career.characterImagePrefix)_smile") + } + + private enum AnimationKey { + static let blink = "blink" + static let smile = "smile" + } + + override func didMove(to view: SKView) { + backgroundColor = .clear + let sprite = SKSpriteNode(texture: idleTexture) + sprite.position = CGPoint(x: size.width / 2, y: size.height / 2) + sprite.size = Constant.characterSize + addChild(sprite) + characterSprite = sprite + startBlinking() + } + + /// 캐릭터를 웃게 만들기 + func playSmile() { + guard let sprite = characterSprite else { return } + // 깜빡임 애니메이션 일시 중지 + sprite.removeAction(forKey: AnimationKey.blink) + // 웃는 애니메이션 + let smile = SKAction.sequence([ + SKAction.setTexture(smileTexture), + SKAction.wait(forDuration: Constant.smileDuration), + SKAction.setTexture(idleTexture), + SKAction.run { [weak self] in + // 웃음 애니메이션 끝나면 다시 깜빡임 시작 + self?.startBlinking() + } + ]) + sprite.run(smile, withKey: AnimationKey.smile) + } + + /// 기본 상태로 돌아가기 + func playIdle() { + guard let sprite = characterSprite else { return } + sprite.removeAction(forKey: AnimationKey.smile) + sprite.texture = idleTexture + startBlinking() + } + + /// 커리어 변경 시 캐릭터 이미지 업데이트 + func updateCareerAppearance(to newCareer: Career) { + guard let sprite = characterSprite else { return } + + // 새로운 텍스처로 업데이트 + let newIdleTexture = SKTexture(imageNamed: "\(newCareer.characterImagePrefix)_default") + sprite.texture = newIdleTexture + + // 애니메이션 재시작 + sprite.removeAllActions() + startBlinking() + } + + // MARK: - Private Methods + private func startBlinking() { + guard let sprite = characterSprite else { return } + // 이미 깜빡이는 중이면 무시 + if sprite.action(forKey: AnimationKey.blink) != nil { return } + let blink = SKAction.sequence([ + // 대기 + SKAction.wait(forDuration: Constant.blinkInterval), + // 눈 감기 + SKAction.setTexture(blinkTexture), + SKAction.wait(forDuration: Constant.blinkDuration), + // 눈 뜨기 + SKAction.setTexture(idleTexture), + SKAction.wait(forDuration: Constant.blinkDuration) + ]) + sprite.run(.repeatForever(blink), withKey: AnimationKey.blink) + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/DodgeGameView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/DodgeGameView.swift new file mode 100644 index 00000000..cc569db6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/DodgeGameView.swift @@ -0,0 +1,270 @@ +// +// DodgeGameView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-15. +// + +import SwiftUI + +private enum Constant { + enum Size { + static let ground: CGFloat = 60 + static let character: CGSize = CGSize(width: 40, height: 40) + } + + enum Position { + /// 캐릭터의 Y 위치 비율 (화면 하단 기준) + static let characterYRatio: CGFloat = 0.25 + /// 이펙트가 캐릭터 위로 올라가는 오프셋 + static let effectOffset: CGFloat = 30 + } + + enum Threshold { + /// 캐릭터 방향 전환을 위한 최소 이동 거리 + static let directionChange: CGFloat = 0.1 + } + + enum Padding { + static let horizontal: CGFloat = 16 + static let toolBarBottom: CGFloat = 10 + } +} + +struct DodgeGameView: View { + @State private var game: DodgeGame + @State private var gameAreaWidth: CGFloat = 0 + @State private var gameAreaHeight: CGFloat = 0 + @State private var isFacingLeft: Bool = false + @State private var goldEffects: [EffectLabelData] = [] + // 일시정지 상태 추가 + @State private var isGamePaused: Bool = false + + @Binding var isGameStarted: Bool + @Binding var isGameViewDisappeared: Bool + + init( + user: User, + isGameStarted: Binding, + isGameViewDisappeared: Binding, + animationSystem: CharacterAnimationSystem? = nil + ) { + self._isGameStarted = isGameStarted + self._isGameViewDisappeared = isGameViewDisappeared + self.game = DodgeGame( + user: user, + gameAreaSize: CGSize.zero, + onGoldChanged: { _ in }, + animationSystem: animationSystem + ) + } + + var body: some View { + GeometryReader { geometry in + VStack(spacing: 0) { + // 상단 툴바 (닫기, 아이템 버튼, 피버 게이지) + toolbarSection + // 게임 영역 (바닥, 플레이어, 낙하물, 골드 이펙트) + gameAreaSection + } + .onAppear { + setupGame(with: geometry.size) + } + .onDisappear { + game.stopGame() + } + .pauseGameStyle( + isGameViewDisappeared: $isGameViewDisappeared, + height: geometry.size.height, + onLeave: { handleCloseButton() }, + onPause: { + isGamePaused = true + game.pauseGame() + }, + onResume: { + isGamePaused = false + game.resumeGame() + } + ) + } + } +} + +// MARK: - View Components +private extension DodgeGameView { + /// 상단 툴바 + var toolbarSection: some View { + GameToolBar( + closeButtonDidTapHandler: handleCloseButton, + coffeeButtonDidTapHandler: { useConsumableItem(.coffee) }, + energyDrinkButtonDidTapHandler: { useConsumableItem(.energyDrink) }, + feverState: game.feverSystem, + buffSystem: game.buffSystem, + coffeeCount: .constant(game.user.inventory.count(.coffee) ?? 0), + energyDrinkCount: .constant(game.user.inventory.count(.energyDrink) ?? 0) + ) + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.bottom, Constant.Padding.toolBarBottom) + } + + /// 게임 영역 + var gameAreaSection: some View { + ZStack(alignment: .bottom) { + // 바닥 + groundView + // 플레이어 + playerView + // 낙하물 + fallingItemsView + // 골드 변화 이펙트 + goldEffectsView + } + } + + /// 바닥 + var groundView: some View { + Image(.dodgeGround) + .resizable() + .frame(height: Constant.Size.ground) + } + + /// 플레이어 + var playerView: some View { + RunningCharacter(isFacingLeft: isFacingLeft, isGamePaused: isGamePaused) + .frame( + width: Constant.Size.character.width, + height: Constant.Size.character.height + ) + .position( + x: gameAreaWidth / 2 + (isGamePaused ? 0 : game.motionSystem.characterX), + y: gameAreaHeight * (1 - Constant.Position.characterYRatio) + ) + .onChange(of: game.motionSystem.characterX) { oldPositionX, newPositionX in + updateCharacterDirection(oldPositionX: oldPositionX, newPositionX: newPositionX) + } + } + + /// 낙하물 + var fallingItemsView: some View { + ForEach(game.gameCore.fallingItems) { item in + DropItem(type: item.type) + .position( + x: gameAreaWidth / 2 + item.position.x, + y: gameAreaHeight / 2 + item.position.y + ) + } + } + + /// 골드 변화 이펙트 + var goldEffectsView: some View { + ForEach(goldEffects) { effect in + EffectLabel( + value: effect.value, + onComplete: { removeEffectLabel(id: effect.id) } + ) + .position(effect.position) + } + } +} + +// MARK: - Actions +private extension DodgeGameView { + /// 닫기 버튼 클릭 처리 + func handleCloseButton() { + game.stopGame() + isGameStarted = false + } + + /// 소비 아이템 사용 처리 + func useConsumableItem(_ type: ConsumableType) { + if game.user.inventory.drink(type) { + SoundService.shared.trigger(.itemConsume) + HapticService.shared.trigger(.success) + game.buffSystem.useConsumableItem(type: type) + game.user.record.record(type == .coffee ? .coffeeUse : .energyDrinkUse) + } + } +} + +// MARK: - Helper Methods +private extension DodgeGameView { + /// 게임 초기 설정 + func setupGame(with size: CGSize) { + gameAreaWidth = size.width + gameAreaHeight = size.height + + game.setGoldChangedHandler(showGoldChangeEffect) + + game.configure(gameAreaSize: CGSize(width: size.width, height: size.height)) + game.startGame() + } + + /// 골드 변화 이펙트 표시 + func showGoldChangeEffect(_ goldDelta: Int) { + let effect = EffectLabelData( + id: UUID(), + position: CGPoint( + x: gameAreaWidth / 2 + game.motionSystem.characterX, + y: gameAreaHeight * (1 - Constant.Position.characterYRatio) - Constant.Position.effectOffset + ), + value: goldDelta + ) + + goldEffects.append(effect) + } + + /// 효과 라벨 제거 (애니메이션 완료 시 콜백으로 호출) + /// - Parameter id: 제거할 효과 라벨의 ID + func removeEffectLabel(id: UUID) { + goldEffects.removeAll { $0.id == id } + } + + /// 캐릭터의 진행 방향을 업데이트 합니다. + func updateCharacterDirection(oldPositionX: CGFloat, newPositionX: CGFloat) { + guard !isGamePaused else { return } + if abs(newPositionX - oldPositionX) > Constant.Threshold.directionChange { + isFacingLeft = newPositionX < oldPositionX + } + } +} + +#Preview { + @Previewable @State var isGameStarted = true + @Previewable @State var isGameViewDisappeared = true + + let wallet = Wallet(gold: 1000, diamond: 0) + let inventory = Inventory( + equipmentItems: [], + consumableItems: [ + .init(type: .coffee, count: 5), + .init(type: .energyDrink, count: 5) + ], + housing: .init(tier: .street) + ) + let record = Record() + let user = User( + nickname: "TestUser", + wallet: wallet, + inventory: inventory, + record: record, + skills: [ + .init(key: SkillKey(game: .dodge, tier: .beginner), level: 1000) + ] + ) + + GeometryReader { geometry in + VStack(spacing: 0) { + Spacer() + .frame(maxHeight: .infinity) + .background(Color.gray.opacity(0.2)) + + DodgeGameView( + user: user, + isGameStarted: $isGameStarted, + isGameViewDisappeared: $isGameViewDisappeared + ) + .ignoresSafeArea() + .frame(height: geometry.size.height / 2 - Constant.Size.ground) + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/FeedbackSettingView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/FeedbackSettingView.swift new file mode 100644 index 00000000..e7890073 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/FeedbackSettingView.swift @@ -0,0 +1,108 @@ +// +// FeedbackSettingView.swift +// SoloDeveloperTraining +// + +import SwiftUI + +private enum Constant { + static let title: String = "설정" + static let rowSpacing: CGFloat = 30 + static let horizontalPadding: CGFloat = 20 + static let volumeRange: ClosedRange = 0 ... 100 + static let volumeStep: Double = 1 +} + +struct FeedbackSettingView: View { + let onClose: (() -> Void)? + + var body: some View { + Popup(title: "") { + Text("설정") + .textStyle(.largeTitle) + VStack(alignment: .leading, spacing: Constant.rowSpacing) { + soundSettingSection( + title: "배경음", + isOn: SoundService.shared.isBGMEnabled, + setOn: { SoundService.shared.isBGMEnabled = $0 }, + volume: bgmVolumeBinding + ) + soundSettingSection( + title: "효과음", + isOn: SoundService.shared.isSFXEnabled, + setOn: { SoundService.shared.isSFXEnabled = $0 }, + volume: sfxVolumeBinding + ) + settingRow( + title: "햅틱", + isOn: HapticService.shared.isEnabled, + setOn: { HapticService.shared.isEnabled = $0 } + ) + closeButton + } + .frame(maxWidth: .infinity, alignment: .leading) + .padding(.horizontal, Constant.horizontalPadding) + } + } +} + +private extension FeedbackSettingView { + var bgmVolumeBinding: Binding { + Binding( + get: { Double(SoundService.shared.bgmVolume) }, + set: { SoundService.shared.bgmVolume = min(max(Int($0), 0), 100) } + ) + } + + var sfxVolumeBinding: Binding { + Binding( + get: { Double(SoundService.shared.sfxVolume) }, + set: { SoundService.shared.sfxVolume = min(max(Int($0), 0), 100) } + ) + } + + var closeButton: some View { + HStack { + Spacer() + MediumButton(title: "닫기", isFilled: true) { + onClose?() + } + Spacer() + } + } + + func soundSettingSection( + title: String, + isOn: Bool, + setOn: @escaping (Bool) -> Void, + volume: Binding + ) -> some View { + VStack(alignment: .leading, spacing: 15) { + settingRow(title: title, isOn: isOn, setOn: setOn) + SettingSlider(value: volume, range: Constant.volumeRange, step: Constant.volumeStep, isEnabled: isOn) + } + } + + func settingRow(title: String, isOn: Bool, setOn: @escaping (Bool) -> Void) -> some View { + HStack { + Text(title) + .textStyle(.title2) + Spacer() + MediumButton(title: isOn ? "ON" : "OFF", isFilled: isOn) { + var transaction = Transaction() + transaction.animation = nil + withTransaction(transaction) { + setOn(!isOn) + } + } + .transaction { $0.animation = nil } + } + } +} + +#Preview { + FeedbackSettingView { + + } + .padding(25) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/IntroView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/IntroView.swift new file mode 100644 index 00000000..dd2d53d9 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/IntroView.swift @@ -0,0 +1,88 @@ +// +// IntroView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-21. +// + +import SwiftUI + +private enum Constant { + enum Animation { + static let transitionDuration: Double = 0.5 // 화면 전환 + static let blinkingDuration: Double = 1.0 // 깜빡임 + } + + enum Layout { + static let bottomPadding: CGFloat = 100 + } + + enum Opacity { + static let blinking: Double = 0.3 + static let normal: Double = 1.0 + } + + enum Text { + static let touchPrompt = "화면을 터치해 주세요" + } +} + +struct IntroView: View { + @State private var isBlinking = true + @Binding var hasSeenIntro: Bool + @Binding var showNicknameSetup: Bool + let user: User? + + var body: some View { + ZStack { + backgroundImage + touchPromptView + } + .onTapGesture { + if user == nil { + showNicknameSetup = true + } else { + withAnimation(.easeOut(duration: Constant.Animation.transitionDuration)) { + hasSeenIntro = true + } + } + } + .ignoresSafeArea() + .onAppear { + isBlinking = false + } + } +} + +private extension IntroView { + var backgroundImage: some View { + GeometryReader { geometry in + Image(.appLaunchScreen) + .resizable() + .aspectRatio(contentMode: .fill) + .frame(width: geometry.size.width, height: geometry.size.height) + .clipped() + } + } + + var touchPromptView: some View { + VStack { + Spacer() + + Text(Constant.Text.touchPrompt) + .textStyle(.title2) + .foregroundColor(.white) + .opacity(isBlinking ? Constant.Opacity.blinking : Constant.Opacity.normal) + .animation(.easeInOut(duration: Constant.Animation.blinkingDuration).repeatForever(autoreverses: true), value: isBlinking) + .padding(.bottom, Constant.Layout.bottomPadding) + } + } +} + +#Preview { + IntroView( + hasSeenIntro: .constant(false), + showNicknameSetup: .constant(false), + user: nil + ) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/NicknameSetupView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/NicknameSetupView.swift new file mode 100644 index 00000000..8b4941e4 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/NicknameSetupView.swift @@ -0,0 +1,149 @@ +// +// NicknameSetupView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-21. +// + +import SwiftUI + +private enum Constant { + enum Spacing { + static let content: CGFloat = 10 + static let textGroup: CGFloat = 4 + static let button: CGFloat = 15 + } + + enum Size { + static let textFieldHeight: CGFloat = 40 + static let cornerRadius: CGFloat = 10 + static let strokeLineWidth: CGFloat = 1 + static let errorTextMinHeight: CGFloat = 15 + } + + enum Padding { + static let textFieldHorizontalPadding: CGFloat = 17 + static let textFieldBottomPadding: CGFloat = 9 + static let contentPadding: CGFloat = 20 + } + + enum Opacity { + static let background: Double = 0.3 + static let stroke: Double = 0.3 + } + + enum Text { + static let popupTitle = "닉네임 설정" + static let story = "당신은 취직에 실패한 개발자.\n... 이대로 물러설 수는 없다.\n나의 꿈은 1인 개발자로 성공하기 ~!\n\n내 이름은!!" + static let nicknamePlaceholder = "닉네임" + static let startButton = "바로 시작" + static let tutorialButton = "튜토리얼" + } +} + +struct NicknameSetupView: View { + @State private var nickname: String = "" + @State private var errorMessage: String = "" + private let validator = Validator() + let onStart: (String) -> Void + let onTutorial: (String) -> Void + + var body: some View { + Popup(title: Constant.Text.popupTitle) { + VStack(alignment: .leading, spacing: Constant.Spacing.content) { + storyTexts + nicknameTextField + errorText + buttons + } + .padding(Constant.Padding.contentPadding) + } + } +} + +private extension NicknameSetupView { + var storyTexts: some View { + Text(Constant.Text.story) + .textStyle(.body) + .foregroundColor(.black) + } + + var nicknameTextField: some View { + TextField(Constant.Text.nicknamePlaceholder, text: $nickname) + .font(.pfFont(.body)) + .padding(.horizontal, Constant.Padding.textFieldHorizontalPadding) + .frame(height: Constant.Size.textFieldHeight) + .background(AppColors.gray100.opacity(Constant.Opacity.background)) + .cornerRadius(Constant.Size.cornerRadius) + .foregroundColor(.black) + .overlay { + RoundedRectangle(cornerRadius: Constant.Size.cornerRadius) + .stroke( + errorMessage.isEmpty + ? Color.gray.opacity(Constant.Opacity.stroke) + : Color.red.opacity(0.7), + lineWidth: Constant.Size.strokeLineWidth + ) + } + .padding(.bottom, Constant.Padding.textFieldBottomPadding) + .onChange(of: nickname) { _, newValue in + updateValidationState(for: newValue) + } + } + + var errorText: some View { + Text(errorMessage) + .font(.pfFont(.caption)) + .foregroundColor(.red) + .frame(minHeight: Constant.Size.errorTextMinHeight, alignment: .leading) + } + + var buttons: some View { + HStack(spacing: Constant.Spacing.button) { + MediumButton( + title: Constant.Text.startButton, + isFilled: !validator.isValid(nickname), + isEnabled: validator.isValid(nickname), + isCancelButton: true + ) { + onStart(nickname) + } + + MediumButton( + title: Constant.Text.tutorialButton, + isFilled: true, + hasBadge: true, + isEnabled: validator.isValid(nickname) + ) { + onTutorial(nickname) + } + } + .frame(maxWidth: .infinity) + } + + func updateValidationState(for value: String) { + switch validator.validate(value) { + case .empty, .valid: + errorMessage = "" + case .invalid(let message): + errorMessage = message + } + } +} + +#Preview { + ZStack { + Color.gray.opacity(0.5) + .ignoresSafeArea() + + NicknameSetupView( + onStart: { nickname in + print("시작: \(nickname)") + }, + onTutorial: { nickname in + print("튜토리얼: \(nickname)") + } + ) + .padding() + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/TutorialPageView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/TutorialPageView.swift new file mode 100644 index 00000000..acebdb4f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/TutorialPageView.swift @@ -0,0 +1,75 @@ +// +// TutorialPageView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-22. +// + +import SwiftUI + +private enum Constant { + enum Spacing { + static let content: CGFloat = 30 + static let textGroup: CGFloat = 16 + } + + enum Size { + static let imageMaxWidth: CGFloat = 300 + static let imageMaxHeight: CGFloat = 300 + static let placeholderWidth: CGFloat = 300 + static let placeholderHeight: CGFloat = 300 + static let cornerRadius: CGFloat = 4 + static let strokeWidth: CGFloat = 2 + } + + enum Layout { + static let textHorizontalPadding: CGFloat = 40 + } +} + +struct TutorialPage { + let title: String + let description: String + let imageName: ImageResource +} + +struct TutorialPageView: View { + let page: TutorialPage + + var body: some View { + VStack(spacing: Constant.Spacing.content) { + textGroup + imageView + .frame(maxWidth: .infinity, maxHeight: .infinity) + .padding(.bottom, Constant.Size.strokeWidth) + } + } +} + +private extension TutorialPageView { + var imageView: some View { + Image(page.imageName) + .resizable() + .scaledToFit() + .frame(maxHeight: .infinity) + .overlay { + RoundedRectangle(cornerRadius: Constant.Size.cornerRadius) + .stroke(.black, lineWidth: Constant.Size.strokeWidth) + } + } + + var textGroup: some View { + VStack(spacing: Constant.Spacing.textGroup) { + Text(page.title) + .textStyle(.largeTitle) + .foregroundColor(.black) + .multilineTextAlignment(.center) + + Text(page.description) + .textStyle(.body) + .foregroundColor(AppColors.gray600) + .multilineTextAlignment(.center) + .padding(.horizontal, Constant.Layout.textHorizontalPadding) + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/TutorialView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/TutorialView.swift new file mode 100644 index 00000000..28da90b7 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/Intro/TutorialView.swift @@ -0,0 +1,156 @@ +// +// TutorialView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-21. +// + +import SwiftUI + +private enum Constant { + enum Spacing { + static let content: CGFloat = 10 + static let indicator: CGFloat = 8 + static let button: CGFloat = 15 + } + + enum Size { + static let indicatorCircle: CGFloat = 8 + static let placeholderButtonWidth: CGFloat = 89 + static let placeholderButtonHeight: CGFloat = 44 + } + + enum Padding { + static let buttonHorizontal: CGFloat = 25 + static let buttonVertical: CGFloat = 25 + } +} + +struct TutorialView: View { + @State private var currentPage: Int = 0 + @Binding var isPresented: Bool + + let onComplete: () -> Void + + private let tutorialPages: [TutorialPage] = [ + TutorialPage( + title: "게임 플레이", + description: "다양한 미니게임을 통해 골드를 획득하세요.", + imageName: .tutorialWork + ), + TutorialPage( + title: "스킬 업그레이드", + description: "스킬을 업그레이드하여 게임 효율을 높이세요.", + imageName: .tutorialSkill + ), + TutorialPage( + title: "장비 강화", + description: "장비를 강화하여 더 많은 보상을 받으세요.", + imageName: .tutorialItem + ), + TutorialPage( + title: "부동산 구매", + description: "부동산을 구매하여 더 많은 보상을 받으세요.", + imageName: .tutorialHousing + ), + TutorialPage( + title: "퀴즈 풀기", + description: "퀴즈를 풀고 다이아를 획득하세요.", + imageName: .tutorialQuiz + ), + TutorialPage( + title: "미션 완료", + description: "미션을 완료하여 추가 보상을 받으세요.", + imageName: .tutorialMission + ), + TutorialPage( + title: "커리어 성장", + description: "커리어를 발전시켜 더 높은 직급에 도전하세요.", + imageName: .tutorialCareer + ) + ] + + var body: some View { + ZStack { + AppTheme.backgroundColor + .ignoresSafeArea() + + VStack(spacing: Constant.Spacing.content) { + tutorialContent + buttonGroup + } + .ignoresSafeArea() + } + } +} + +private extension TutorialView { + var indicator: some View { + HStack(spacing: Constant.Spacing.indicator) { + ForEach(0.. 0 { + MediumButton(title: "이전", isFilled: true, isCancelButton: true) { + withAnimation { + currentPage -= 1 + } + } + } else { + Color.clear + .frame( + width: Constant.Size.placeholderButtonWidth, + height: Constant.Size.placeholderButtonHeight + ) + } + + Spacer() + + indicator + + Spacer() + + if currentPage < tutorialPages.count - 1 { + MediumButton(title: "다음", isFilled: true) { + withAnimation { + currentPage += 1 + } + } + } else { + MediumButton(title: "시작하기", isFilled: true) { + onComplete() + } + } + } + .padding(.horizontal, Constant.Padding.buttonHorizontal) + .padding(.vertical, Constant.Padding.buttonVertical) + } +} + +#Preview { + TutorialView( + isPresented: .constant(true), + onComplete: { + print("튜토리얼 완료") + } + ) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/LanguageGameView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/LanguageGameView.swift new file mode 100644 index 00000000..ff4ba899 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/LanguageGameView.swift @@ -0,0 +1,232 @@ +// +// LanguageGameView.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/15/26. +// + +import SwiftUI + +private enum Constant { + enum Padding { + static let horizontal: CGFloat = 16 + static let toolBarBottom: CGFloat = 10 + } + + enum Spacing { + static let itemHorizontal: CGFloat = 25 + static let buttonHorizontal: CGFloat = 17 + } + + enum Game { + static let itemCount: Int = 5 + static let feverDecreaseInterval: Double = 0.1 + static let feverDecreasePercentPerTick: Double = 5 + } + + enum EffectLabel { + static let offsetY: CGFloat = -34 + } +} + +struct LanguageGameView: View { + // MARK: Properties + let user: User + + /// 게임에 사용되는 언어 타입 목록 + private let languageTypeList: [LanguageType] = [ + .swift, + .kotlin, + .dart, + .python + ] + + // MARK: State Properties + /// 게임 시작 상태 (부모 뷰와 바인딩) + @Binding var isGameStarted: Bool + @Binding var isGameViewDisappeared: Bool + + /// 상태를 유지 + @State private var game: LanguageGame + + /// 획득한 골드를 표시하기 위한 효과 라벨 배열 + @State private var effectValues: [(id: UUID, value: Int)] = [] + + /// 현재 진행 중인 언어 버튼 탭 Task + @State private var currentActionTask: Task? + + init( + user: User, + isGameStarted: Binding, + isGameViewDisappeared: Binding, + animationSystem: CharacterAnimationSystem? = nil + ) { + self._isGameStarted = isGameStarted + self._isGameViewDisappeared = isGameViewDisappeared + self.user = user + + // 게임 초기화 + let game = LanguageGame( + user: user, + feverSystem: .init( + decreaseInterval: Constant.Game.feverDecreaseInterval, + decreasePercentPerTick: Constant.Game.feverDecreasePercentPerTick + ), + buffSystem: .init(), + itemCount: Constant.Game.itemCount, + animationSystem: animationSystem + ) + self._game = State(initialValue: game) + self.game.startGame() + } + + var body: some View { + GeometryReader { geometry in + VStack(alignment: .center, spacing: 0) { + // 상단 툴바 (닫기, 아이템 버튼, 피버 게이지) + toolbarSection + Spacer() + // 중앙 언어 아이템 영역 (획득 골드 효과 포함) + languageItemsSection + Spacer() + // 하단 언어 선택 버튼 영역 + languageButtonsSection + Spacer() + } + .pauseGameStyle( + isGameViewDisappeared: $isGameViewDisappeared, + height: geometry.size.height, + onLeave: { handleCloseButton() }, + onPause: { game.pauseGame() }, + onResume: { game.resumeGame() } + ) + } + } +} + +// MARK: - View Components +private extension LanguageGameView { + /// 상단 툴바 + var toolbarSection: some View { + GameToolBar( + closeButtonDidTapHandler: handleCloseButton, + coffeeButtonDidTapHandler: { useConsumableItem(.coffee) }, + energyDrinkButtonDidTapHandler: { useConsumableItem(.energyDrink) }, + feverState: game.feverSystem, + buffSystem: game.buffSystem, + coffeeCount: .constant(game.user.inventory.count(.coffee) ?? 0), + energyDrinkCount: .constant(game.user.inventory.count(.energyDrink) ?? 0) + ) + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.bottom, Constant.Padding.toolBarBottom) + } + + /// 중앙 언어 아이템 영역 + var languageItemsSection: some View { + HStack(alignment: .bottom, spacing: Constant.Spacing.itemHorizontal) { + ForEach(Array(game.itemList.enumerated()), id: \.offset) { _, item in + LanguageItem( + languageType: item.languageType, + state: item.state + ) + } + } + .frame(maxWidth: .infinity) + .overlay(alignment: .top) { + // 획득한 골드를 표시하는 효과 라벨 + ZStack { + ForEach(effectValues, id: \.id) { effect in + EffectLabel(value: effect.value) { + removeEffectLabel(id: effect.id) + } + } + } + .offset(y: Constant.EffectLabel.offsetY) + } + } + + /// 하단 언어 선택 버튼 영역 + var languageButtonsSection: some View { + HStack(spacing: Constant.Spacing.buttonHorizontal) { + ForEach(languageTypeList, id: \.self) { type in + LanguageButton(languageType: type) { + handleLanguageButtonTap(type) + } + } + } + } +} + +// MARK: - Actions +private extension LanguageGameView { + /// 닫기 버튼 클릭 처리 + func handleCloseButton() { + // 진행 중인 액션 Task 취소 + currentActionTask?.cancel() + currentActionTask = nil + + game.stopGame() + isGameStarted = false + } + + /// 언어 버튼 클릭 처리 + func handleLanguageButtonTap(_ type: LanguageType) { + // 이전 액션이 진행 중이면 취소 + currentActionTask?.cancel() + + currentActionTask = Task { + let gainedGold = await game.didPerformAction(type) + + // Task가 취소되었으면 UI 업데이트 생략 + guard !Task.isCancelled else { return } + + SoundService.shared.trigger(gainedGold > 0 ? .languageCorrect : .languageWrong) + if gainedGold <= 0 { + HapticService.shared.trigger(.error) + } + showEffectLabel(gainedGold: gainedGold) + } + } + + /// 소비 아이템 사용 처리 + func useConsumableItem(_ type: ConsumableType) { + if game.user.inventory.drink(type) { + SoundService.shared.trigger(.itemConsume) + HapticService.shared.trigger(.success) + game.buffSystem.useConsumableItem(type: type) + game.user.record.record(type == .coffee ? .coffeeUse : .energyDrinkUse) + } + } +} + +// MARK: - Helper Methods +private extension LanguageGameView { + /// 획득한 골드를 표시하는 효과 라벨 추가 + /// - Parameter gainedGold: 획득한 골드 (음수일 경우 손실) + func showEffectLabel(gainedGold: Int) { + let effectId = UUID() + effectValues.append((id: effectId, value: gainedGold)) + } + + /// 효과 라벨 제거 (애니메이션 완료 시 콜백으로 호출) + /// - Parameter id: 제거할 효과 라벨의 ID + func removeEffectLabel(id: UUID) { + effectValues.removeAll { $0.id == id } + } +} + +#Preview { + @Previewable @State var isGameStarted = true + @Previewable @State var isGameViewDisappeared = true + let user = User( + nickname: "Test", + wallet: .init(), + inventory: .init(), + record: .init(), + skills: [ + .init(key: SkillKey(game: .language, tier: .beginner), level: 1000) + ] + ) + + LanguageGameView(user: user, isGameStarted: $isGameStarted, isGameViewDisappeared: $isGameViewDisappeared, animationSystem: nil) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MainView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MainView.swift new file mode 100644 index 00000000..1a4cb98b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MainView.swift @@ -0,0 +1,300 @@ +// +// MainView.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/8/26. +// + +import SwiftUI +import SpriteKit + +enum AppTheme { + static let backgroundColor: Color = AppColors.beige200 +} + +private enum Constant { + static let characterSceneSize = CGSize(width: 100, height: 100) + static let spriteViewSize = CGSize(width: 200, height: 200) + static let topAreaHeightRatio: CGFloat = 0.5 + + enum Padding { + static let horizontalPadding: CGFloat = 25 + } + + enum Color { + static let overlay = SwiftUI.Color.black.opacity(0.3) + } + + enum CareerPopup { + static let title: String = "커리어" + static let maxHeight: CGFloat = 650 + } + + enum TopButton { + static let top: CGFloat = 128 + static let horizontal: CGFloat = 16 + } +} + +struct MainView: View { + @Environment(\.scenePhase) var scenePhase + @State private var selectedTab: TabItem = .work + @State private var popupContent: PopupConfiguration? + @State private var careerSystem: CareerSystem? + @State private var isWorkGameInProgress: Bool = false + @State private var showQuizView: Bool = false + @State private var showSettingsView: Bool = false + + private var autoGainSystem: AutoGainSystem + private let user: User + private let scene: CharacterScene + private let animationSystem: CharacterAnimationSystem + + init(user: User) { + self.autoGainSystem = AutoGainSystem(user: user) + self.user = user + + self.scene = CharacterScene(size: Constant.characterSceneSize, user: user) + self.scene.scaleMode = .aspectFit + self.scene.playIdle() + + // 애니메이션 시스템 생성 및 클로저 연결 + self.animationSystem = CharacterAnimationSystem() + self.animationSystem.onSmile = { [weak scene] in + scene?.playSmile() + } + self.animationSystem.onIdle = { [weak scene] in + scene?.playIdle() + } + } + + var body: some View { + GeometryReader { geometry in + VStack(spacing: 0) { + topAreaContent + .frame(height: geometry.size.height * Constant.topAreaHeightRatio) + .background(housingBackgroundView) + tabBar + tabContentView + .frame(maxWidth: .infinity, maxHeight: .infinity, alignment: .top) + } + .ignoresSafeArea(edges: [.top, .bottom]) + .background(AppTheme.backgroundColor) + .onAppear(perform: setupOnAppear) + .onDisappear { SoundService.shared.stopBGM() } + .onChange(of: scenePhase, handleScenePhaseChange) + .task(id: user.record.totalEarnedMoney) { + await careerSystem?.updateCareer() + } + .overlay { popupOverlayView } + .overlay { settingsOverlayView } + .fullScreenCover(isPresented: $showQuizView) { + QuizGameView(user: user) + } + } + } +} + +private extension MainView { + var topAreaContent: some View { + ZStack(alignment: .top) { + topAreaMainContent + topButtonOverlay + } + } + + var topAreaMainContent: some View { + VStack(spacing: 0) { + StatusBar( + career: careerSystem?.currentCareer ?? .unemployed, + nickname: user.nickname, + careerProgress: careerSystem?.careerProgress ?? 0.0, + gold: user.wallet.gold, + diamond: user.wallet.diamond + ) + .onTapGesture { showCareerPopup() } + Spacer() + characterSceneView + } + } + + var tabBar: some View { + TabBar( + selectedTab: $selectedTab, + hasCompletedMisson: user.record + .missionSystem.hasCompletedMission + ) + } + + var characterSceneView: some View { + SpriteView(scene: scene, options: [.allowsTransparency]) + .frame(width: Constant.spriteViewSize.width, height: Constant.spriteViewSize.height) + .background(Color.clear) + } + + var housingBackgroundView: some View { + Image(user.inventory.housing.imageName) + .resizable() + .aspectRatio(contentMode: .fill) + } + + var topButtonOverlay: some View { + VStack { + HStack { + SmallButton(title: "설정", image: Image(.iconSetting)) { + showSettingsView = true + } + Spacer() + SmallButton(title: "퀴즈", hasBadge: true) { + showQuizView = true + } + } + .padding(.top, Constant.TopButton.top) + .padding(.horizontal, Constant.TopButton.horizontal) + Spacer() + } + } + + var tabContentView: some View { + ZStack { + if isWorkGameInProgress { + workGameOverlayView + } + if !isWorkGameInProgress || selectedTab != .work { + tabContentSwitchView + } + } + } + + var workGameOverlayView: some View { + WorkSelectedView( + user: user, + animationSystem: animationSystem, + isGameStarted: $isWorkGameInProgress, + isGameViewDisappeared: Binding( + get: { selectedTab != .work || showQuizView }, + set: { _ in } + ), + careerSystem: $careerSystem + ) + .opacity(selectedTab == .work ? 1 : 0) + .allowsHitTesting(selectedTab == .work) + } + + @ViewBuilder + var tabContentSwitchView: some View { + switch selectedTab { + case .work: + if !isWorkGameInProgress { + WorkSelectedView( + user: user, + animationSystem: animationSystem, + isGameStarted: $isWorkGameInProgress, + isGameViewDisappeared: Binding( + get: { selectedTab != .work || showQuizView }, + set: { _ in } + ), + careerSystem: $careerSystem + ) + } + case .skill: + SkillView(user: user, careerSystem: careerSystem, popupContent: $popupContent) + case .shop: + ShopView(user: user, popupContent: $popupContent) + case .mission: + MissionView(user: user) + } + } + + @ViewBuilder + var popupOverlayView: some View { + if let popupContent { + ZStack { + Constant.Color.overlay + .ignoresSafeArea() + .onTapGesture { self.popupContent = nil } + + Popup(title: popupContent.title, contentView: popupContent.content) + .frame(maxHeight: popupContent.maxHeight) + .padding(.horizontal, Constant.Padding.horizontalPadding) + } + } + } + + @ViewBuilder + var settingsOverlayView: some View { + if showSettingsView { + ZStack { + Constant.Color.overlay + .ignoresSafeArea() + .onTapGesture { showSettingsView = false } + + FeedbackSettingView(onClose: { showSettingsView = false }) + .padding(.horizontal, Constant.Padding.horizontalPadding) + } + } + } + + func setupOnAppear() { + SoundService.shared.playBGM() + autoGainSystem.startSystem() + Task { + if careerSystem == nil { + careerSystem = await CareerSystem(user: user) + careerSystem?.onCareerChanged = { [weak scene] newCareer in + scene?.updateCareerAppearance(to: newCareer) + } + } + } + } + + func handleScenePhaseChange(_ oldValue: ScenePhase, _ newValue: ScenePhase) { + if newValue == .active { + autoGainSystem.startSystem() + } else if newValue == .inactive || newValue == .background { + autoGainSystem.stopSystem() + } + } + + func showCareerPopup() { + guard let careerSystem else { return } + + popupContent = PopupConfiguration( + title: Constant.CareerPopup.title, + maxHeight: Constant.CareerPopup.maxHeight + ) { + CareerPopupView( + careerSystem: careerSystem, + user: user, + onClose: { + popupContent = nil + } + ) + } + } +} + +#Preview { + let user = User( + nickname: "소피아", + career: .unemployed, + wallet: .init(), + inventory: Inventory( + equipmentItems: [ + .init(type: .chair, tier: .broken), + .init(type: .keyboard, tier: .broken), + .init(type: .monitor, tier: .broken), + .init(type: .mouse, tier: .broken) + ], + housing: .init(tier: .street) + ), + record: .init(), + skills: [ + .init(key: SkillKey(game: .tap, tier: .beginner), level: 10), + .init(key: SkillKey(game: .language, tier: .beginner), level: 1), + .init(key: SkillKey(game: .dodge, tier: .beginner), level: 1), + .init(key: SkillKey(game: .stack, tier: .beginner), level: 1) + ] + ) + MainView(user: user) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MissionView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MissionView.swift new file mode 100644 index 00000000..eca9e860 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MissionView.swift @@ -0,0 +1,98 @@ +// +// MissionView.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/21/26. +// + +import SwiftUI + +private enum Constant { + static let vertical: CGFloat = 15 + static let gridVeticalSpacing: CGFloat = 14 + static let minWidth: CGFloat = 115 +} + +struct MissionView: View { + private let user: User + private let missionSystem: MissionSystem + + @State private var showToast: Bool = false + @State private var toastMessage: String = "" + + init(user: User) { + self.user = user + self.missionSystem = user.record.missionSystem + } + + var body: some View { + VStack(spacing: Constant.vertical) { + ProgressBar( + maxValue: Double(missionSystem.allCount), + currentValue: Double(missionSystem.claimedCount), + text: "\(missionSystem.claimedCount) / \(missionSystem.allCount)" + ) + ScrollView { + LazyVGrid( + columns: [GridItem(.adaptive(minimum: Constant.minWidth))], + spacing: Constant.gridVeticalSpacing + ) { + ForEach(missionSystem.missions, id: \.id) { mission in + MissionCard( + title: mission.title, + reward: mission.reward, + imageName: mission.type.level.imageName, + condition: mission.description, + buttonState: toButtonState( + missionCardState: mission + .missionCardState), + onButtonTap: { + missionCardDidTapHandler( + mission: mission + ) + } + ) + } + } + } + .scrollIndicators(.never) + } + .padding(.horizontal) + .toast(isShowing: $showToast, message: toastMessage) + + } +} + +private extension MissionView { + func toButtonState(missionCardState: MissionCardState) -> MissionCardButton.ButtonState { + switch missionCardState { + case .claimed: + .claimed + case .claimable: + .claimable + case .inProgress(let currentValue, let totalValue): + .inProgress(currentValue: currentValue, totalValue: totalValue) + } + } + + func missionCardDidTapHandler(mission: Mission) { + if mission.missionCardState == .claimable { + missionSystem.claimMissionReward(mission: mission, wallet: user.wallet) + SoundService.shared.trigger(.missionAcquired) + showToast = false + let reward = mission.reward + if reward.gold > 0 && reward.diamond > 0 { + toastMessage = "미션을 달성했습니다.\n보상: \(reward.gold.formatted) 골드, \(reward.diamond.formatted) 다이아" + } else if reward.gold > 0 { + toastMessage = "미션을 달성했습니다.\n보상: \(reward.gold.formatted) 골드" + } else { + toastMessage = "미션을 달성했습니다.\n보상: \(reward.diamond.formatted) 다이아" + } + showToast = true + } else { + showToast = false + toastMessage = mission.missionCardState == .claimed ? "이미 보유한 미션입니다." : "아직 달성하지 못한 미션입니다." + showToast = true + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MultiTouchView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MultiTouchView.swift new file mode 100644 index 00000000..2a6928d5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/MultiTouchView.swift @@ -0,0 +1,51 @@ +// +// MultiTouchView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-23. +// + +import SwiftUI +import UIKit + +struct MultiTouchView: UIViewRepresentable { + let onTap: (CGPoint) -> Void + + func makeUIView(context: Context) -> MultiTouchUIView { + let view = MultiTouchUIView() + view.onTap = onTap + return view + } + + func updateUIView(_ uiView: MultiTouchUIView, context: Context) { + uiView.onTap = onTap + } +} + +final class MultiTouchUIView: UIView { + var onTap: ((CGPoint) -> Void)? + + override init(frame: CGRect) { + super.init(frame: frame) + isMultipleTouchEnabled = true + isUserInteractionEnabled = true + backgroundColor = .clear + } + + required init?(coder: NSCoder) { nil } + + override func touchesBegan(_ touches: Set, with event: UIEvent?) { + super.touchesBegan(touches, with: event) + + // 모든 터치에 대해 처리 + for touch in touches { + let location = touch.location(in: self) + let safeAreaBottom = safeAreaInsets.bottom + let maxY = bounds.height - safeAreaBottom + + if location.y <= maxY { + onTap?(location) + } + } + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/QuizGameView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/QuizGameView.swift new file mode 100644 index 00000000..9ff01503 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/QuizGameView.swift @@ -0,0 +1,314 @@ +// +// QuizGameView.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/22/26. +// + +import SwiftUI + +private enum Constant { + static let totalQuizCount: Int = 3 + static let rewardCount: Int = 5 + + enum Padding { + static let horizontal: CGFloat = 16 + static let top: CGFloat = 15 + static let titleBottom: CGFloat = 14 + static let quizCountBottom: CGFloat = 6 + static let remainSecondsBottom: CGFloat = 30 + static let questionTitleBottom: CGFloat = 8 + static let rewardBottom: CGFloat = 8 + static let optionsBottom: CGFloat = 30 + static let submitBottom: CGFloat = 15 + } + + enum Spacing { + static let title: CGFloat = 2 + static let quizButton: CGFloat = 16 + } + + enum Size { + static let closeButtonWidth: CGFloat = 28 + static let closeButtonHeight: CGFloat = 28 + } +} + +struct QuizGameView: View { + @Environment(\.dismiss) private var dismiss + @State private var quizGame: QuizGame + @State private var showRewardPopup: Bool = false + + init(user: User) { + _quizGame = State(initialValue: QuizGame(user: user)) + } + + var body: some View { + let state = quizGame.state + + VStack(spacing: 0) { + + /// 문제 헤더 영역 + QuizHeaderView( + currentQuizNumber: state.currentQuestion != nil ? quizGame.currentQuestionIndex + 1 : 0, + totalQuizCount: Constant.totalQuizCount, + remainingSeconds: state.remainingSeconds, + quizTitle: state.currentQuestion?.question ?? "", + rewardCount: Constant.rewardCount, + onClose: { + dismiss() + } + ) + + /// 해설 영역 + QuizExplanationView( + isSubmitted: state.phase == .showingExplanation, + explanation: state.currentQuestion?.explanation ?? "", + isCorrect: state.currentAnswerResult?.isCorrect ?? false, + correctAnswerIndex: state.currentQuestion?.correctAnswerIndex + ) + + /// 선지, 제출버튼 영역 + QuizOptionsView( + options: state.currentQuestion?.options ?? [], + selectedIndex: state.selectedAnswerIndex, + isShowingExplanation: state.phase == .showingExplanation, + submitButtonTitle: state.phase == .showingExplanation ? state.nextButtonTitle : "제출하기", + onSelect: { index in + if state.selectedAnswerIndex == index { + quizGame.deselectAnswer() + } else { + quizGame.selectAnswer(index) + } + }, + onSubmit: { + if state.phase == .showingExplanation { + if state.nextButtonTitle == "보상받기" { + showRewardPopup = true + } else { + quizGame.proceedToNextQuestion() + } + } else { + quizGame.submitSelectedAnswer() + } + } + ) + } + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.top, Constant.Padding.top) + .background(AppColors.beige100) + .onAppear { + if quizGame.state.phase == .ready { + quizGame.startGame() + } + } + .onChange(of: state.remainingSeconds) { _, newValue in + if newValue == 3 { + SoundService.shared.trigger(.quizCountdown) + } else if newValue == 0 { + SoundService.shared.trigger(.quizTimeOver) + } + } + .onDisappear { SoundService.shared.stopAllSFX() } + .overlay { + if showRewardPopup { + ZStack { + Color.black.opacity(0.4) + .ignoresSafeArea() + + RewardPopupView( + totalDiamondsEarned: state.totalDiamondsEarned, + onClose: { + quizGame.proceedToNextQuestion() + showRewardPopup = false + dismiss() + } + ) + } + } + } + } +} + +// MARK: - 퀴즈 헤더 뷰 +private struct QuizHeaderView: View { + let currentQuizNumber: Int + let totalQuizCount: Int + let remainingSeconds: Int + let quizTitle: String + let rewardCount: Int + let onClose: () -> Void + + var body: some View { + VStack(alignment: .leading, spacing: 0) { + // 타이틀 + HStack(spacing: Constant.Spacing.title) { + Image(.quizDogFace) + Image(.quizDogFoot) + Text("개발 퀴즈") + .textStyle(.largeTitle) + Spacer() + Button { + onClose() + } label: { + Image(systemName: "xmark.square.fill") + .resizable() + .foregroundStyle(.black) + .frame( + width: Constant.Size.closeButtonWidth, + height: Constant.Size.closeButtonHeight + ) + } + .buttonStyle(.soundTap) + } + .padding(.bottom, Constant.Padding.titleBottom) + + // 문제 개수 + HStack { + Spacer() + Text("\(currentQuizNumber) / \(totalQuizCount)") + .textStyle(.headline) + } + .padding(.bottom, Constant.Padding.quizCountBottom) + + // Progress + ProgressBar( + maxValue: Double(Policy.Game.Quiz.secondsPerQuestion), + currentValue: Double(remainingSeconds), + text: remainingSeconds > 0 ? "\(remainingSeconds)s" : "제한 시간 종료" + ) + .padding(.bottom, Constant.Padding.remainSecondsBottom) + + // 문제 + Text(quizTitle) + .textStyle(.title2) + .padding(.bottom, Constant.Padding.questionTitleBottom) + .fixedSize(horizontal: false, vertical: true) + + // 보상 + HStack { + Spacer() + CurrencyLabel(axis: .horizontal, icon: .diamond, textStyle: .body, value: rewardCount) + } + .padding(.bottom, Constant.Padding.rewardBottom) + } + } +} + +// MARK: - 해설 뷰 +private struct QuizExplanationView: View { + let isSubmitted: Bool + let explanation: String + let isCorrect: Bool + let correctAnswerIndex: Int? + + var body: some View { + VStack(spacing: 0) { + if isSubmitted { + Text(resultPrefix+explanation) + .textStyle(.callout) + .foregroundColor( + isCorrect ? AppColors.accentGreen : AppColors.accentRed + ) + .frame(maxWidth: .infinity, alignment: .leading) + .fixedSize(horizontal: false, vertical: true) + .minimumScaleFactor(0.8) + } + + Spacer(minLength: 0) + } + .frame(maxWidth: .infinity, maxHeight: .infinity) + } + + private var resultPrefix: String { + if isCorrect { + return "정답\n" + } else { + let answerNumber = (correctAnswerIndex ?? 0) + 1 // 0-based → 1-based + return "오답 / 정답은 \(answerNumber)번이다.\n" + } + } +} + +// MARK: - 선지, 제출버튼 뷰 +private struct QuizOptionsView: View { + let options: [String] + let selectedIndex: Int? + let isShowingExplanation: Bool + let submitButtonTitle: String + let onSelect: (Int) -> Void + let onSubmit: () -> Void + + var body: some View { + VStack(spacing: 0) { + + // 선지 + VStack(spacing: Constant.Spacing.quizButton) { + ForEach(options.indices, id: \.self) { index in + QuizButton( + isSelected: selectedIndex == index, + title: "\(index + 1). \(options[index])" + ) { + onSelect(index) + } + .disabled(isShowingExplanation) + } + } + .padding(.bottom, Constant.Padding.optionsBottom) + + // 제출 버튼 + QuizButton( + style: .submit, + isEnabled: isShowingExplanation ? true : selectedIndex != nil, + title: submitButtonTitle + ) { + onSubmit() + } + .padding(.bottom, Constant.Padding.submitBottom) + } + } +} + +// MARK: - 보상 팝업 뷰 +private struct RewardPopupView: View { + let totalDiamondsEarned: Int + let onClose: () -> Void + + var body: some View { + Popup(title: "보상 획득") { + VStack(alignment: .center, spacing: 0) { + VStack(alignment: .leading, spacing: 0) { + Text("퀴즈 풀이를 완료했습니다!\n진정한 개발자에 한걸음 더 가까워졌습니다.") + .textStyle(.body) + .padding(.top, 11) + .padding(.bottom, 20) + + HStack(spacing: 4) { + Text("획득한 다이아: ") + .textStyle(.body) + CurrencyLabel( + axis: .horizontal, + icon: .diamond, + textStyle: .body, + value: totalDiamondsEarned + ) + } + .padding(.bottom, 20) + } + MediumButton(title: "종료하기", isFilled: true) { + onClose() + } + } + } + .padding(.horizontal, 40) + } +} + +#Preview { + QuizGameView(user: User( + nickname: "Preview User", + wallet: Wallet(), + inventory: Inventory(), + record: Record() + )) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/ShopPurchaseHelper.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/ShopPurchaseHelper.swift new file mode 100644 index 00000000..0c06d48f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/ShopPurchaseHelper.swift @@ -0,0 +1,119 @@ +// +// ShopPurchaseHelper.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-21. +// + +import Foundation +import SwiftUI + +private enum Constant { + enum Spacing { + static let popupContent: CGFloat = 11 + static let popupButton: CGFloat = 15 + } + + enum Padding { + static let popupTop: CGFloat = 11 + static let popupHorizontal: CGFloat = 25 + } +} + +enum ShopPurchaseHelper { + /// 확인 팝업 표시 (취소/확인 버튼) + static func showConfirm( + popupContent: Binding, + title: String, + message: String, + confirmTitle: String, + onConfirm: @escaping () -> Void + ) { + var didPurchasingButtonTapped = false + + popupContent.wrappedValue = PopupConfiguration( + title: title, + maxHeight: nil + ) { + VStack(spacing: Constant.Spacing.popupContent) { + Text(message) + .textStyle(.body) + .foregroundColor(.black) + .multilineTextAlignment(.center) + + HStack(spacing: Constant.Spacing.popupButton) { + MediumButton(title: "취소", isFilled: true, isCancelButton: true) { + popupContent.wrappedValue = nil + } + MediumButton(title: confirmTitle, isFilled: true) { + guard !didPurchasingButtonTapped else { return } + didPurchasingButtonTapped = true + onConfirm() + popupContent.wrappedValue = nil + } + } + } + .padding(.top, Constant.Padding.popupTop) + } + } + + /// 알림 팝업 표시 (확인 버튼만) + static func showAlert( + popupContent: Binding, + title: String, + message: String + ) { + popupContent.wrappedValue = PopupConfiguration( + title: title, + maxHeight: nil + ) { + VStack(spacing: Constant.Spacing.popupContent) { + Text(message) + .textStyle(.body) + .foregroundColor(.black) + .multilineTextAlignment(.center) + + MediumButton(title: "확인", isFilled: true) { + popupContent.wrappedValue = nil + } + } + .padding(.top, Constant.Padding.popupTop) + } + } + + /// 구매 정보 생성 + static func purchaseInfo(for item: DisplayItem) -> (title: String, message: String, buttonTitle: String) { + switch item.category { + case .equipment: + let successRate = (item.item as? Equipment).map { Int($0.tier.upgradeSuccessRate * 100) } + let message = successRate.map { "강화하시겠습니까?\n(성공 확률: \($0)%)" } ?? "강화하시겠습니까?" + return ("장비 강화", message, "강화") + case .housing: + return ("부동산 구매", "구매하시겠습니까?", "구매") + case .consumable: + return ("아이템 구매", "구매하시겠습니까?", "구매") + } + } + + /// 구매 메시지 생성 + static func createPurchaseMessage(item: DisplayItem, baseMessage: String, shopSystem: ShopSystem) -> String { + let priceText = createPriceText(for: item, shopSystem: shopSystem) + let prefix = item.category == .housing && shopSystem.calculateHousingNetCost(for: item) < 0 ? "를 환불받고" : "를 사용하여" + return "\(priceText)\(prefix)\n\(baseMessage)" + } + + /// 가격 텍스트 생성 + static func createPriceText(for item: DisplayItem, shopSystem: ShopSystem) -> String { + var components: [String] = [] + + if item.category == .housing { + let netCost = shopSystem.calculateHousingNetCost(for: item) + components.append("\(abs(netCost).formatted) 골드") + } else { + if item.cost.gold > 0 { components.append("\(item.cost.gold.formatted) 골드") } + if item.cost.diamond > 0 { components.append("\(item.cost.diamond.formatted) 다이아") } + } + + return "[\(components.joined(separator: ", "))]" + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/ShopView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/ShopView.swift new file mode 100644 index 00000000..f33c3261 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/ShopView.swift @@ -0,0 +1,190 @@ +// +// ShopView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-21. +// + +import SwiftUI + +private enum Constant { + enum Text { + static let itemSegment = "아이템" + static let housingSegment = "부동산" + + static let enhanceSuccessTitle = "강화 성공" + static let enhanceFailureTitle = "강화 실패" + static let enhanceSuccessMessage = "강화에 성공했습니다!" + static let enhanceFailureMessage = "강화에 실패했습니다.\n비용은 소모되었습니다." + + static let purchaseFailureTitle = "구매 실패" + static let purchaseFailureMessage = "구매에 실패했습니다." + } + + enum ID { + static let housingScrollStart = "housingScrollStart" + } + + enum Spacing { + static let itemCard: CGFloat = 12 + } + + enum Padding { + static let horizontal: CGFloat = 16 + static let housingTop: CGFloat = 15 + static let housingBottom: CGFloat = 23 + } +} + +struct ShopView: View { + private let user: User + private let shopSystem: ShopSystem + + @State private var selectedCategoryIndex: Int = 0 + @State private var selectedHousingTier: HousingTier? + + @Binding var popupContent: PopupConfiguration? + + init(user: User, popupContent: Binding) { + self.user = user + self.shopSystem = ShopSystem(user: user) + self._popupContent = popupContent + } + + var body: some View { + VStack { + DefaultSegmentControl( + selection: $selectedCategoryIndex, + segments: [Constant.Text.itemSegment, Constant.Text.housingSegment] + ) + .padding(.horizontal, Constant.Padding.horizontal) + + if selectedCategoryIndex == 0 { + itemView + } else { + housingView + } + } + } +} + +private extension ShopView { + var displayItems: [DisplayItem] { + if selectedCategoryIndex == 0 { + return shopSystem.itemList(itemTypes: [.consumable, .equipment]) + } else { + return shopSystem.itemList(itemTypes: [.housing]) + } + } + + var itemView: some View { + ScrollView { + LazyVStack(spacing: Constant.Spacing.itemCard) { + ForEach(displayItems) { item in + ItemRow( + title: item.displayTitle, + description: item.description, + imageName: item.imageName, + cost: item.cost, + state: ItemState(item: item) + ) { + purchase(item: item) + } + } + } + .padding(.bottom) + } + .scrollIndicators(.never) + } + + var housingView: some View { + ScrollViewReader { proxy in + ScrollView(.horizontal) { + LazyHStack(spacing: Constant.Spacing.itemCard) { + ForEach(displayItems) { item in + if let housing = item.item as? Housing { + HousingCard( + housing: housing, + state: ItemState(item: item), + isSelected: selectedHousingTier == housing.tier, + onTap: { + selectedHousingTier = housing.tier + }, + onButtonTap: { + selectedHousingTier = housing.tier + purchase(item: item, scrollProxy: proxy) + } + ) + .id(item.id) + } + } + } + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.top, Constant.Padding.housingTop) + .padding(.bottom, Constant.Padding.housingBottom) + .id(Constant.ID.housingScrollStart) + } + .scrollIndicators(.never) + } + } + + /// 아이템 구매 확인 팝업 표시 + func purchase(item: DisplayItem, scrollProxy: ScrollViewProxy? = nil) { + let (title, message, buttonTitle) = ShopPurchaseHelper.purchaseInfo(for: item) + let fullMessage = ShopPurchaseHelper.createPurchaseMessage(item: item, baseMessage: message, shopSystem: shopSystem) + + ShopPurchaseHelper.showConfirm( + popupContent: $popupContent, + title: title, + message: fullMessage, + confirmTitle: buttonTitle + ) { + executePurchase(item: item, scrollProxy: scrollProxy) + } + } + + /// 실제 구매 실행 + func executePurchase(item: DisplayItem, scrollProxy: ScrollViewProxy? = nil) { + do { + let isSuccess = try shopSystem.buy(item: item) + if isSuccess { + // 성공 시 가로 스크롤을 맨 처음으로 이동 + if let proxy = scrollProxy, selectedCategoryIndex == 1 { + withAnimation { + proxy.scrollTo(Constant.ID.housingScrollStart, anchor: .leading) + } + } + } + if item.category == .equipment { + SoundService.shared.trigger(isSuccess ? .upgradeSuccess : .upgradeFailure) + if !isSuccess { + HapticService.shared.trigger(.error) + } + let title = isSuccess ? Constant.Text.enhanceSuccessTitle : Constant.Text.enhanceFailureTitle + let message = isSuccess ? Constant.Text.enhanceSuccessMessage : Constant.Text.enhanceFailureMessage + ShopPurchaseHelper.showAlert(popupContent: $popupContent, title: title, message: message) + } + } catch let error as PurchasingError { + HapticService.shared.trigger(.error) + ShopPurchaseHelper.showAlert(popupContent: $popupContent, title: Constant.Text.purchaseFailureTitle, message: error.message) + } catch { + HapticService.shared.trigger(.error) + ShopPurchaseHelper.showAlert(popupContent: $popupContent, title: Constant.Text.purchaseFailureTitle, message: Constant.Text.purchaseFailureMessage) + } + } +} + +#Preview { + let user = User( + nickname: "테스트", + wallet: .init(gold: 1_000_000, diamond: 100), + inventory: .init(), + record: .init(), + skills: [ + .init(key: SkillKey(game: .tap, tier: .beginner), level: 1) + ] + ) + Spacer() + .frame(height: 500) + ShopView(user: user, popupContent: .constant(nil)) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/SkillView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/SkillView.swift new file mode 100644 index 00000000..fd9c2433 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/SkillView.swift @@ -0,0 +1,129 @@ +// +// SkillView.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-20. +// + +import SwiftUI + +private enum Constant { + static let horizontalPadding: CGFloat = 16 + static let popupHorizontalPadding: CGFloat = 25 + static let itemCardSpacing: CGFloat = 12 + static let popupContentSpacing: CGFloat = 20 +} + +struct SkillView: View { + private let user: User + private let careerSystem: CareerSystem? + private let skillSystem: SkillSystem + + @Binding var popupContent: PopupConfiguration? + + init( + user: User, + careerSystem: CareerSystem?, + popupContent: Binding + ) { + self.user = user + self.careerSystem = careerSystem + self.skillSystem = SkillSystem(user: user, careerSystem: careerSystem) + self._popupContent = popupContent + } + + var body: some View { + ScrollView { + LazyVStack(spacing: Constant.itemCardSpacing) { + ForEach(skillSystem.skillList(), id: \.skill) { skillState in + ItemRow( + title: skillState.skill.title, + description: "액션당 \(Int(skillState.skill.gainGold).formatted()) 골드 획득", + imageName: skillState.skill.imageName, + cost: skillState.skill.upgradeCost, + state: skillState.itemState, + action: { upgrade(skill: skillState.skill) }, + onLongPressAction: { upgradeRepeating(skill: skillState.skill) } + ) + } + } + } + .padding(.bottom) + .scrollIndicators(.never) + } +} + +private extension SkillView { + func upgrade(skill: Skill) { + do { + try skillSystem.upgrade(skill: skill) + } catch let error as UserReadableError { + popupContent = PopupConfiguration(title: "스킬") { + VStack(spacing: Constant.popupContentSpacing) { + Text(error.message) + .textStyle(.body) + .foregroundColor(.black) + .multilineTextAlignment(.center) + + MediumButton(title: "확인", isFilled: true) { + popupContent = nil + } + } + } + } catch { + // UserReadableError를 채택하지 않은 예상치 못한 에러 + // 실제로는 발생하지 않지만 Swift 컴파일러 요구사항 + popupContent = PopupConfiguration(title: "스킬") { + VStack(spacing: Constant.popupContentSpacing) { + Text(error.localizedDescription) + .textStyle(.body) + .foregroundColor(.black) + .multilineTextAlignment(.center) + + MediumButton(title: "확인", isFilled: true) { + popupContent = nil + } + } + } + } + } + + /// 롱프레스 연속 구매용. 성공 시 `true`, 실패(재화 부족 등) 시 `false` 반환해 연속 호출 중단. + func upgradeRepeating(skill: Skill) -> Bool { + do { + try skillSystem.upgrade(skill: skill) + return true + } catch { + return false + } + } +} + +#Preview { + let user = User( + nickname: "테스트", + wallet: .init(gold: 110, diamond: 100), + inventory: .init(), + record: .init(), + skills: [ + // 코드짜기 + .init(key: SkillKey(game: .tap, tier: .beginner), level: 1), + .init(key: SkillKey(game: .tap, tier: .intermediate), level: 1), + .init(key: SkillKey(game: .tap, tier: .advanced), level: 1), + // 언어 맞추기 + .init(key: SkillKey(game: .language, tier: .beginner), level: 1), + .init(key: SkillKey(game: .language, tier: .intermediate), level: 1), + .init(key: SkillKey(game: .language, tier: .advanced), level: 1), + // 버그 피하기 + .init(key: SkillKey(game: .dodge, tier: .beginner), level: 1), + .init(key: SkillKey(game: .dodge, tier: .intermediate), level: 1), + .init(key: SkillKey(game: .dodge, tier: .advanced), level: 1), + // 데이터 쌓기 + .init(key: SkillKey(game: .stack, tier: .beginner), level: 1), + .init(key: SkillKey(game: .stack, tier: .intermediate), level: 1), + .init(key: SkillKey(game: .stack, tier: .advanced), level: 1) + ] + ) + + SkillView(user: user, careerSystem: nil, popupContent: .constant(nil)) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/StackGameScene.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/StackGameScene.swift new file mode 100644 index 00000000..8680ced1 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/StackGameScene.swift @@ -0,0 +1,323 @@ +// +// StackGameScene.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/15/26. +// + +import SwiftUI +import SpriteKit + +private enum Constant { + enum Physics { + static let gravity = CGVector(dx: 0, dy: -9.8) + } + + enum Offset { + // 최상단에서 블록을 살짝 안쪽으로 내려서 생성하기 위한 여백 + static let spawnYOffset: CGFloat = 24 + } + + enum Time { + // 블록 평가 체크 간격 (초) + static let evaluationCheckInterval: TimeInterval = 0.05 + // 폭탄 블록 제거 딜레이 (초) + static let bombRemovalDelay: TimeInterval = 0.8 + // 카메라 이동 애니메이션 시간 (초) + static let cameraMoveAnimationDuration: TimeInterval = 0.3 + // 다음 블록 생성 딜레이 (초) + static let nextBlockSpawnDelay: TimeInterval = 0.3 + // 실패한 블록 제거 딜레이 (초) + static let failedBlockRemovalDelay: TimeInterval = 1.0 + } +} + +final class StackGameScene: SKScene { + private let stackGame: StackGame + + private var currentBlockView: BlockItem? + private var blockViews: [BlockItem] = [] + /// 첫 블록의 바닥 기준 높이 + private var currentHeight: CGFloat = 0 + /// 블록 배치 처리 중 여부 (UI 인터랙션 차단용) + private var isProcessing = false + /// 자체 게임 상태 관리 변수 + private var isGamePaused = false + + var onBlockDropped: ((Int) -> Void) + + init(stackGame: StackGame, onBlockDropped: @escaping ((Int) -> Void)) { + self.stackGame = stackGame + self.onBlockDropped = onBlockDropped + super.init(size: .zero) + self.scaleMode = .resizeFill + } + + required init?(coder aDecoder: NSCoder) { + return nil + } + + override func didMove(to view: SKView) { + // 화면 크기를 게임 코어에 전달 + stackGame.screenSize = size + setupScene() + startGame() + } + + override func touchesBegan(_ touches: Set, with event: UIEvent?) { + guard currentBlockView != nil, !isProcessing, !isGamePaused else { return } + dropBlock() + } + + /// 씬의 초기 설정을 수행합니다. + /// - 배경색을 앱 테마 배경색으로 설정 + /// - 물리 엔진의 중력 설정 + /// - 카메라 초기화 + private func setupScene() { + backgroundColor = UIColor(AppTheme.backgroundColor) + physicsWorld.gravity = Constant.Physics.gravity + + setupCamera() + } + + /// 카메라 노드를 생성하고 초기 위치를 설정합니다. + private func setupCamera() { + let cameraNode = SKCameraNode() + cameraNode.position = CGPoint(x: size.width / 2, y: size.height / 2) + addChild(cameraNode) + camera = cameraNode + } + + /// 게임을 시작하고 초기 상태로 설정합니다. + /// - 블록 배열 초기화 + /// - 게임 코어 시작 처리 + /// - 카메라 위치 리셋 + /// - 초기 블록 배치 및 첫 번째 블록 생성 + func startGame() { + blockViews = [] + currentHeight = 0 + isProcessing = false + + stackGame.startGame() + camera?.position = CGPoint(x: size.width / 2, y: size.height / 2) + + putInitialBlock() + spawnBlock() + } + + /// 게임을 중지하고 모든 진행중인 동작을 멈춥니다. + /// - 게임 코어 중지 처리 + /// - 현재 블록의 모든 액션 제거 + /// - 물리 엔진 정지 + func stopGame() { + stackGame.stopGame() + isProcessing = true + currentBlockView?.removeAllActions() + physicsWorld.speed = 0 + } + + /// 게임 Scene 일시정지 + func pauseGame() { + stackGame.pauseGame() + isGamePaused = true + physicsWorld.speed = 0 + currentBlockView?.removeAllActions() + currentBlockView?.removeFromParent() + currentBlockView = nil + } + + /// 게임 Scene 재개 + func resumeGame() { + stackGame.resumeGame() + isGamePaused = false + physicsWorld.speed = 1 + if currentBlockView == nil { + spawnBlock() + } + } + + /// 게임 시작 시 가장 아래에 배치되는 초기 블록을 생성합니다. + /// - 고정된 물리 바디를 가진 파란색 블록 생성 + /// - 게임 코어에 초기 블록 등록 + private func putInitialBlock() { + let firstBlockView = BlockItem(type: .blue) + firstBlockView.setupPhysicsBody() + firstBlockView.position = CGPoint(x: size.width / 2, y: currentHeight) + + // 게임 코어에 초기 블록 등록 + stackGame.addInitialBlock() + + addChild(firstBlockView) + blockViews.append(firstBlockView) + } + + /// 새로운 블록을 화면 상단에 생성하고 좌우로 움직이게 합니다. + /// - 랜덤한 타입의 블록 생성 + /// - 카메라 기준 상단 위치에서 생성 + /// - 좌우 이동 애니메이션 시작 + private func spawnBlock() { + // 일시정지 상태에서는 동작을 막음 + guard !isGamePaused else { return } + + isProcessing = false + + let blockType = BlockType.allCases.randomElement() ?? .blue + let blockView = BlockItem(type: blockType) + + // 게임 코어에 블록 생성 알림 + stackGame.spawnBlock(type: blockType) + + let spawnY = (camera?.position.y ?? size.height / 2) + size.height / 2 - ( + Constant.Offset.spawnYOffset + blockView.size.height / 2) + let leftEdge = blockView.size.width / 2 + let rightEdge = size.width - blockView.size.width / 2 + + blockView.position = CGPoint(x: leftEdge, y: spawnY) + blockView.startMoving(distance: rightEdge - leftEdge) + + currentBlockView = blockView + addChild(blockView) + } + + /// 현재 블록의 이동을 멈추고 중력을 적용하여 떨어뜨립니다. + /// - 블록의 좌우 이동 중지 + /// - 중력 활성화 + /// - 블록 평가 시작 + private func dropBlock() { + guard let block = currentBlockView else { return } + + isProcessing = true + block.stopMoving() + block.enableGravity() + + // 블록 평가 시작 + evaluateBlock() + } + + /// 떨어지는 블록이 목표 위치에 도달했는지 재귀적으로 확인합니다. + /// - 목표 높이에 도달하면 정렬 체크 수행 + /// - 아직 도달하지 않았으면 일정 시간 후 재확인 + private func evaluateBlock() { + guard + let block = currentBlockView, + let previousBlock = stackGame.previousBlock, + !isPaused + else { return } + + // StackGame의 previousBlock 정보를 사용해 목표 Y 계산 + let targetY = previousBlock.positionY + previousBlock.height + + if block.position.y <= targetY { + // 목표 위치에 도달했으므로 정렬 체크 + // 정렬 성공/실패에 따라 물리 처리를 다르게 적용 + checkAlignmentAndHandle(targetY: targetY) + } else { + // 아직 도달하지 않았으면 재확인 + DispatchQueue.main.asyncAfter(deadline: .now() + Constant.Time.evaluationCheckInterval) { [weak self] in + self?.evaluateBlock() + } + } + } + + /// 정렬을 체크하고 결과에 따라 물리 처리를 다르게 적용합니다. + /// - 성공: 블록 고정 후 배치 + /// - 실패: 물리를 유지하여 자연스럽게 떨어지도록 + private func checkAlignmentAndHandle(targetY: CGFloat) { + guard let block = currentBlockView else { return } + + // 현재 블록 위치를 게임 모델에 업데이트 + stackGame.updateCurrentBlockPosition(positionX: block.position.x, positionY: targetY) + + let isAligned = stackGame.checkAlignment() + + if isAligned { + // 정렬 성공: 블록 고정 + block.fixPosition() + block.physicsBody?.velocity = CGVector.zero + block.physicsBody?.angularVelocity = 0 + block.position = CGPoint(x: block.position.x, y: targetY) + isProcessing = false + + placeBlockSuccess() + } else { + // 정렬 실패: 물리를 유지하여 계속 떨어지도록 + isProcessing = false + placeBlockFail() + } + } + + /// 블록이 성공적으로 배치되었을 때의 처리를 수행합니다. + /// - 폭탄 블록: 패널티 적용 후 블록 제거 + /// - 일반 블록: 스택에 추가, 점수 증가, 보상 적용, 카메라 이동 + private func placeBlockSuccess() { + guard + let block = currentBlockView, + let currentBlock = stackGame.currentBlock, + let previousView = blockViews.last + else { return } + + // 이전 블록의 정확한 위치를 기준으로 배치 + let targetY = previousView.position.y + previousView.size.height + block.position = CGPoint(x: block.position.x, y: targetY) + + // currentHeight 업데이트 + currentHeight = targetY + block.size.height + + // 폭탄 블록 체크 + if currentBlock.type.isBomb { + onBlockDropped(stackGame.placeBombSuccess()) + SoundService.shared.trigger(.bombStack) + HapticService.shared.trigger(.error) + DispatchQueue.main.asyncAfter(deadline: .now() + Constant.Time.bombRemovalDelay) { [weak self] in + block.removeFromParent() + self?.spawnBlock() + } + } else { + blockViews.append(block) + // 코어에 블록 배치 성공 알림 (위치는 이미 업데이트됨) + onBlockDropped(stackGame.placeBlockSuccess()) + SoundService.shared.trigger(.blockStack) + // 카메라 이동 + if let camera = camera { + let newCameraY = camera.position.y + block.size.height + let moveCamera = SKAction.moveTo(y: newCameraY, duration: Constant.Time.cameraMoveAnimationDuration) + moveCamera.timingMode = .easeInEaseOut + camera.run(moveCamera) + } + + DispatchQueue.main.asyncAfter(deadline: .now() + Constant.Time.nextBlockSpawnDelay) { [weak self] in + self?.spawnBlock() + } + } + currentBlockView = nil + } + + /// 블록 배치에 실패했을 때의 처리를 수행합니다. + /// - 폭탄 블록: 실패시 오히려 보상 적용 + /// - 일반 블록: 패널티 적용 + /// - 물리 효과로 떨어지고 화면 밖으로 나가면 제거 + private func placeBlockFail() { + guard + let block = currentBlockView, + let currentBlock = stackGame.currentBlock + else { return } + + // 폭탄 블록 실패 = 보상, 일반 블록 실패 = 패널티 + if currentBlock.type.isBomb { + onBlockDropped(stackGame.placeBombFail()) + SoundService.shared.trigger(.blockDrop) + } else { + onBlockDropped(stackGame.placeBlockFail()) + SoundService.shared.trigger(.blockDrop) + HapticService.shared.trigger(.error) + } + + // 일정 시간 후 블록 제거 및 다음 블록 생성 + DispatchQueue.main.asyncAfter(deadline: .now() + Constant.Time.failedBlockRemovalDelay) { [weak self] in + block.removeFromParent() + self?.spawnBlock() + } + + currentBlockView = nil + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/StackGameView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/StackGameView.swift new file mode 100644 index 00000000..00990a7c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/StackGameView.swift @@ -0,0 +1,169 @@ +// +// StackGameView.swift +// SoloDeveloperTraining +// +// Created by sunjae on 1/15/26. +// + +import SwiftUI +import SpriteKit + +private enum Constant { + static let effectLabelXRatios: [CGFloat] = [0.3, 0.4, 0.7] + static let effectLabelYPositions: [CGFloat] = [150, 200, 250] + + enum Padding { + static let horizontal: CGFloat = 16 + static let toolBarBottom: CGFloat = 10 + } +} + +struct StackGameView: View { + @State private var stackGame: StackGame + @State private var scene: StackGameScene + @State private var effectLabels: [EffectLabelData] = [] + + /// 게임 시작 상태 (부모 뷰와 바인딩) + @Binding var isGameStarted: Bool + @Binding var isGameViewDisappeared: Bool + + init( + user: User, + isGameStarted: Binding, + isGameViewDisappeared: Binding, + animationSystem: CharacterAnimationSystem? = nil + ) { + let stackGame = StackGame(user: user, animationSystem: animationSystem) + self._stackGame = State(initialValue: stackGame) + self._isGameStarted = isGameStarted + self._isGameViewDisappeared = isGameViewDisappeared + + let initialScene = StackGameScene( + stackGame: stackGame, + onBlockDropped: { _ in } + ) + self._isGameStarted = isGameStarted + self._stackGame = State(initialValue: stackGame) + self._scene = State(initialValue: initialScene) + } + + var body: some View { + GeometryReader { geometry in + VStack(spacing: 0) { + // 상단 툴바 (닫기, 아이템 버튼, 피버 게이지) + toolbarSection + // 게임 영역 (SpriteKit 씬, 골드 이펙트) + gameAreaSection + } + .background(AppTheme.backgroundColor) + .navigationBarBackButtonHidden(true) // 임시로 숨김 + .onAppear { + setupGameCallbacks(with: geometry) + } + .pauseGameStyle( + isGameViewDisappeared: $isGameViewDisappeared, + height: geometry.size.height, + onLeave: { handleCloseButton() }, + onPause: { scene.pauseGame() }, + onResume: { scene.resumeGame() } + ) + } + } +} + +// MARK: - View Components +private extension StackGameView { + /// 상단 툴바 + var toolbarSection: some View { + GameToolBar( + closeButtonDidTapHandler: handleCloseButton, + coffeeButtonDidTapHandler: { useConsumableItem(.coffee) }, + energyDrinkButtonDidTapHandler: { useConsumableItem(.energyDrink) }, + feverState: stackGame.feverSystem, + buffSystem: stackGame.buffSystem, + coffeeCount: .constant(stackGame.user.inventory.count(.coffee) ?? 0), + energyDrinkCount: .constant(stackGame.user.inventory.count(.energyDrink) ?? 0) + ) + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.bottom, Constant.Padding.toolBarBottom) + } + + /// 게임 영역 + var gameAreaSection: some View { + ZStack { + SpriteView(scene: scene) + + ForEach(effectLabels) { effectLabel in + EffectLabel( + value: effectLabel.value, + onComplete: { removeEffectLabel(id: effectLabel.id) } + ) + .position(effectLabel.position) + } + } + } +} + +// MARK: - Actions +private extension StackGameView { + /// 닫기 버튼 클릭 처리 + func handleCloseButton() { + stackGame.stopGame() + isGameStarted = false + } + + /// 소비 아이템 사용 처리 + func useConsumableItem(_ type: ConsumableType) { + if stackGame.user.inventory.drink(type) { + SoundService.shared.trigger(.itemConsume) + HapticService.shared.trigger(.success) + stackGame.buffSystem.useConsumableItem(type: type) + stackGame.user.record.record(type == .coffee ? .coffeeUse : .energyDrinkUse) + } + } +} + +// MARK: - Helper Methods +private extension StackGameView { + /// 게임 콜백 설정 + func setupGameCallbacks(with geometry: GeometryProxy) { + scene.onBlockDropped = { gold in + showEffectLabel( + at: CGPoint( + x: geometry.size.width * randomEffectXRatio, + y: randomEffectYOffset + ), + value: gold + ) + } + } + + /// 효과 라벨 추가 + /// - Parameters: + /// - location: 표시할 위치 + /// - value: 표시할 값 + func showEffectLabel(at location: CGPoint, value: Int) { + let labelData = EffectLabelData( + id: UUID(), + position: location, + value: value + ) + effectLabels.append(labelData) + } + + /// 효과 라벨 제거 (애니메이션 완료 시 콜백으로 호출) + /// - Parameter id: 제거할 효과 라벨의 ID + func removeEffectLabel(id: UUID) { + effectLabels.removeAll { $0.id == id } + } + + /// 랜덤 효과 라벨 X 위치 비율 + var randomEffectXRatio: CGFloat { + Constant.effectLabelXRatios.randomElement() ?? 0.4 + } + + /// 랜덤 효과 라벨 Y 오프셋 + var randomEffectYOffset: CGFloat { + Constant.effectLabelYPositions.randomElement() ?? 200 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/TapGameView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/TapGameView.swift new file mode 100644 index 00000000..862deffa --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/TapGameView.swift @@ -0,0 +1,202 @@ +// +// TapGameView.swift +// SoloDeveloperTraining +// +// Created by 김성훈 on 1/14/26. +// + +import SwiftUI + +private enum Constant { + enum Padding { + static let horizontal: CGFloat = 16 + static let toolBarBottom: CGFloat = 10 + } + /// 탭 사운드 최소 재생 간격 (초) + static let tapSoundThrottleInterval: TimeInterval = 0.05 +} + +struct TapGameView: View { + // MARK: - Properties + /// 게임 시작 상태 (부모 뷰와 바인딩) + @Binding var isGameStarted: Bool + @Binding var isGameViewDisappeared: Bool + + // MARK: - State + /// 의존 게임 + @State private var tapGame: TapGame + /// 터치한 위치에 표시될 EffectLabel들의 위치와 값 + @State private var effectLabels: [EffectLabelData] = [] + /// 탭 사운드 쓰로틀용 마지막 재생 시각 + @State private var lastTapSoundTime: Date = .distantPast + + init( + user: User, + isGameStarted: Binding, + isGameViewDisappeared: Binding, + animationSystem: CharacterAnimationSystem? + ) { + let tapGame = TapGame( + user: user, + buffSystem: BuffSystem(), + animationSystem: animationSystem + ) + self._tapGame = State(initialValue: tapGame) + self._isGameStarted = isGameStarted + self._isGameViewDisappeared = isGameViewDisappeared + self.tapGame.startGame() + } + + var body: some View { + GeometryReader { geometry in + VStack(spacing: 0) { + // 상단 툴바 (닫기, 아이템 버튼, 피버 게이지) + toolbarSection + // 터치 가능한 게임 영역 + tapAreaSection(geometry: geometry) + } + .pauseGameStyle( + isGameViewDisappeared: $isGameViewDisappeared, + height: geometry.size.height, + onLeave: { handleCloseButton() }, + onPause: { + tapGame.pauseGame() + SoundService.shared.stopAllSFX() + lastTapSoundTime = .distantPast + }, + onResume: { tapGame.resumeGame() } + ) + } + } +} + +// MARK: - View Components +private extension TapGameView { + /// 상단 툴바 + var toolbarSection: some View { + GameToolBar( + closeButtonDidTapHandler: handleCloseButton, + coffeeButtonDidTapHandler: { useConsumableItem(.coffee) }, + energyDrinkButtonDidTapHandler: { useConsumableItem(.energyDrink) }, + feverState: tapGame.feverSystem, + buffSystem: tapGame.buffSystem, + coffeeCount: Binding( + get: { tapGame.inventory.count(.coffee) ?? 0 }, + set: { _ in } + ), + energyDrinkCount: Binding( + get: { tapGame.inventory.count(.energyDrink) ?? 0 }, + set: { _ in } + ) + ) + .padding(.horizontal, Constant.Padding.horizontal) + .padding(.bottom, Constant.Padding.toolBarBottom) + } + + /// 터치 가능한 게임 영역 + func tapAreaSection(geometry: GeometryProxy) -> some View { + ZStack { + // 배경 이미지 + Image(.tapBackground) + .resizable() + .aspectRatio(contentMode: .fill) + + // 효과 라벨들 + ForEach(effectLabels) { effectLabel in + EffectLabel( + value: effectLabel.value, + onComplete: { removeEffectLabel(id: effectLabel.id) } + ) + .position(effectLabel.position) + } + + // 멀티터치 뷰 + MultiTouchView { location in + Task { await handleTap(at: location) } + } + } + } +} + +// MARK: - Actions +private extension TapGameView { + /// 닫기 버튼 클릭 처리 + func handleCloseButton() { + tapGame.stopGame() + isGameStarted = false + } + + /// 터치 이벤트 처리 + /// - Parameter location: 터치한 위치 + @MainActor + func handleTap(at location: CGPoint) async { + let now = Date() + if !tapGame.isPaused, + now.timeIntervalSince(lastTapSoundTime) >= Constant.tapSoundThrottleInterval { + SoundService.shared.trigger(.tapGameTyping) + lastTapSoundTime = now + } + let gainGold = await tapGame.didPerformAction() + showEffectLabel(at: location, value: gainGold) + } + + /// 소비 아이템 사용 처리 + func useConsumableItem(_ type: ConsumableType) { + if tapGame.inventory.drink(type) { + SoundService.shared.trigger(.itemConsume) + HapticService.shared.trigger(.success) + tapGame.buffSystem.useConsumableItem(type: type) + tapGame.user.record.record(type == .coffee ? .coffeeUse : .energyDrinkUse) + } + } +} + +// MARK: - Helper Methods +private extension TapGameView { + /// 터치 위치에 효과 라벨 추가 + /// - Parameters: + /// - location: 터치한 위치 + /// - value: 표시할 값 + func showEffectLabel(at location: CGPoint, value: Int) { + let labelData = EffectLabelData( + id: UUID(), + position: location, + value: value + ) + effectLabels.append(labelData) + } + + /// 효과 라벨 제거 (애니메이션 완료 시 콜백으로 호출) + /// - Parameter id: 제거할 효과 라벨의 ID + func removeEffectLabel(id: UUID) { + effectLabels.removeAll { $0.id == id } + } +} + +#Preview { + @Previewable @State var isGameStarted: Bool = true + @Previewable @State var isGameViewDisappeared: Bool = true + + let user = User( + nickname: "Preview User", + wallet: Wallet(gold: 10000, diamond: 50), + inventory: Inventory( + consumableItems: [ + Consumable(type: .coffee, count: 5), + Consumable(type: .energyDrink, count: 3) + ] + ), + record: Record(), + skills: [ + Skill(key: SkillKey(game: .tap, tier: .beginner), level: 10), + Skill(key: SkillKey(game: .tap, tier: .intermediate), level: 5) + ] + ) + + TapGameView( + user: user, + isGameStarted: $isGameStarted, + isGameViewDisappeared: $isGameViewDisappeared, + animationSystem: nil + ) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/WorkSelectedView.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/WorkSelectedView.swift new file mode 100644 index 00000000..a6cd2209 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Presentation/WorkSelectedView.swift @@ -0,0 +1,251 @@ +// +// WorkSelectedView.swift +// SoloDeveloperTraining +// +// Created by SeoJunYoung on 1/15/26. +// + +import SwiftUI + +private enum Constant { + enum Padding { + static let horizontal: CGFloat = 16 + static let selectionViewBottom: CGFloat = 30 + } + + enum UserDefaults { + static let lastSelectedWorkIndexKey = "lastSelectedWorkIndex" + } + + enum Description { + static let tapGame = "모니터를 최대한 많이 누르세요." + static let languageGame = "올바른 버튼을 누르세요." + static let dodgeGame = "기기를 기울여 버그를 피하고 골드를 획득하세요." + static let stackGame = "최대한 높은 데이터를 쌓으세요." + } + + static let contentSpacing: CGFloat = 17 + static let descriptionSpacing: CGFloat = 10 +} + +struct WorkSelectedView: View { + + let user: User + let animationSystem: CharacterAnimationSystem? + @State var selectedIndex: Int = 0 + @State var workItems: [WorkItem] = [] + @State private var showToast: Bool = false + @State private var toastMessage: String = "" + @Binding var isGameStarted: Bool + @Binding var isGameViewDisappeared: Bool + @Binding var careerSystem: CareerSystem? + + private let localStorage: KeyValueLocalStorage = UserDefaultsStorage() + + init( + user: User, + animationSystem: CharacterAnimationSystem?, + isGameStarted: Binding, + isGameViewDisappeared: Binding, + careerSystem: Binding + ) { + self.user = user + self.animationSystem = animationSystem + self._isGameStarted = isGameStarted + self._isGameViewDisappeared = isGameViewDisappeared + self._careerSystem = careerSystem + } + + var body: some View { + Group { + if isGameStarted { + gameView(for: selectedIndex) + } else { + selectionView + } + } + .onAppear { + workItems = createWorkItems(career: careerSystem?.currentCareer) + loadLastSelectedIndex() + } + .onChange(of: selectedIndex) { _, newValue in + saveLastSelectedIndex(newValue) + } + .onChange(of: careerSystem?.currentCareer) { _, newValue in + workItems = createWorkItems(career: newValue) + } + } +} + +// MARK: - Subviews +private extension WorkSelectedView { + + var selectionView: some View { + VStack(spacing: Constant.contentSpacing) { + workSegmentControl + descriptionStack + Spacer() + startButton + } + .padding(.horizontal, Constant.Padding.horizontal) + .toast(isShowing: $showToast, message: toastMessage) + } + + var workSegmentControl: some View { + WorkSegmentControl( + items: workItems, + onLockedTap: { requiredCareer in + toastMessage = "\(requiredCareer.rawValue)부터 플레이할 수 있습니다." + showToast = true + } + , selectedIndex: $selectedIndex + ) + } + + var descriptionStack: some View { + VStack(alignment: .leading, spacing: Constant.descriptionSpacing) { + Text(actionDescription(for: selectedIndex)) + .foregroundStyle(.gray300) + .textStyle(.subheadline) + .frame(maxWidth: .infinity, alignment: .leading) + } + } + + var startButton: some View { + LargeButton(title: "시작하기") { + isGameStarted = true + } + .frame(maxWidth: .infinity, alignment: .bottom) + .padding(.bottom, Constant.Padding.selectionViewBottom) + } +} + +// MARK: - Helper +private extension WorkSelectedView { + + func createWorkItems(career: Career?) -> [WorkItem] { + let currentCareer = career ?? .unemployed + let currentWealth = currentCareer.requiredWealth + + let tapUnlocked = currentWealth >= Policy.Career.GameUnlock.tap + let languageUnlocked = currentWealth >= Policy.Career.GameUnlock.language + let dodgeUnlocked = currentWealth >= Policy.Career.GameUnlock.dodge + let stackUnlocked = currentWealth >= Policy.Career.GameUnlock.stack + + return [ + .init( + title: "코드짜기", + imageName: GameType.tap.imageName, + isDisabled: !tapUnlocked, + requiredCareer: findCareer(for: Policy.Career.GameUnlock.tap) + ), + .init( + title: "언어 맞추기", + imageName: GameType.language.imageName, + isDisabled: !languageUnlocked, + requiredCareer: findCareer(for: Policy.Career.GameUnlock.language) + ), + .init( + title: "버그 피하기", + imageName: GameType.dodge.imageName, + isDisabled: !dodgeUnlocked, + requiredCareer: findCareer(for: Policy.Career.GameUnlock.dodge) + ), + .init( + title: "데이터 쌓기", + imageName: GameType.stack.imageName, + isDisabled: !stackUnlocked, + requiredCareer: findCareer(for: Policy.Career.GameUnlock.stack) + ) + ] + } + + func findCareer(for requiredWealth: Int) -> Career? { + return Career.allCases.first { $0.requiredWealth == requiredWealth } + } + + @ViewBuilder + func gameView(for index: Int) -> some View { + switch index { + case 0: + TapGameView( + user: user, + isGameStarted: $isGameStarted, + isGameViewDisappeared: $isGameViewDisappeared, + animationSystem: animationSystem + ) + case 1: + LanguageGameView( + user: user, + isGameStarted: $isGameStarted, + isGameViewDisappeared: $isGameViewDisappeared, + animationSystem: animationSystem + ) + case 2: + DodgeGameView( + user: user, + isGameStarted: $isGameStarted, + isGameViewDisappeared: $isGameViewDisappeared, + animationSystem: animationSystem + ) + case 3: + StackGameView( + user: user, + isGameStarted: $isGameStarted, + isGameViewDisappeared: $isGameViewDisappeared, + animationSystem: animationSystem + ) + default: + EmptyView() + } + } + + func actionDescription(for index: Int) -> String { + switch index { + case 0: + return Constant.Description.tapGame + case 1: + return Constant.Description.languageGame + case 2: + return Constant.Description.dodgeGame + case 3: + return Constant.Description.stackGame + default: + return "" + } + } + + func loadLastSelectedIndex() { + let savedIndex = localStorage.integer(key: Constant.UserDefaults.lastSelectedWorkIndexKey) + if savedIndex >= 0 && savedIndex < workItems.count { + selectedIndex = savedIndex + } else { + selectedIndex = 0 + } + } + + func saveLastSelectedIndex(_ index: Int) { + localStorage.set(index, forKey: Constant.UserDefaults.lastSelectedWorkIndexKey) + } +} + +#Preview { + @Previewable @State var isGameStarted = false + @Previewable @State var isGameViewDisappeared = false + @Previewable @State var careerSystem: CareerSystem? = nil + + let user = User( + nickname: "Test", + wallet: .init(), + inventory: .init(), + record: .init() + ) + + WorkSelectedView( + user: user, + animationSystem: nil, + isGameStarted: $isGameStarted, + isGameViewDisappeared: $isGameViewDisappeared, + careerSystem: $careerSystem + ) +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Production/Utility/Validator.swift b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Utility/Validator.swift new file mode 100644 index 00000000..0d692afd --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Production/Utility/Validator.swift @@ -0,0 +1,71 @@ +// +// Validator.swift +// SoloDeveloperTraining +// +// Created by 최범수 on 2026-01-21. +// + +import Foundation + +enum ValidationResult { + case empty + case valid + case invalid(String) +} + +private enum Constant { + static let pattern: String = "^[가-힣a-zA-Z0-9]+$" + + enum Length { + static let min: Int = 2 + static let max: Int = 7 + } + + enum Text { + static let invalidCharacter = "한글, 숫자, 영어만 사용 가능하며 공백은 사용할 수 없습니다." + static let tooShort = "닉네임은 최소 \(Length.min)자 이상이어야 합니다." + static let tooLong = "닉네임은 최대 \(Length.max)자까지 입력 가능합니다." + } +} + +final class Validator { + private var nicknameRegex: NSRegularExpression? { + try? NSRegularExpression(pattern: Constant.pattern, options: []) + } + + /// 닉네임을 검증하고 결과를 반환합니다. + func validate(_ nickname: String) -> ValidationResult { + if nickname.isEmpty { + return .empty + } + + // 한글, 숫자, 영어만 허용 (공백 불가) + guard let regex = nicknameRegex, + regex.firstMatch(in: nickname, options: [], range: NSRange(location: 0, length: nickname.utf16.count)) != nil + else { + return .invalid(Constant.Text.invalidCharacter) + } + + if nickname.count < Constant.Length.min { + return .invalid(Constant.Text.tooShort) + } + + if nickname.count > Constant.Length.max { + return .invalid(Constant.Text.tooLong) + } + + return .valid + } + + /// 닉네임이 유효한지 확인합니다. + func isValid(_ nickname: String) -> Bool { + // 한글, 숫자, 영어만 허용 (공백 불가) + guard let regex = nicknameRegex, + regex.firstMatch(in: nickname, options: [], range: NSRange(location: 0, length: nickname.utf16.count)) != nil + else { + return false + } + + return nickname.count >= Constant.Length.min && nickname.count <= Constant.Length.max && !nickname.isEmpty + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentColor.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentColor.colorset/Contents.json new file mode 100644 index 00000000..274babba --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentColor.colorset/Contents.json @@ -0,0 +1,20 @@ +{ + "colors" : [ + { + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentGreen.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentGreen.colorset/Contents.json new file mode 100644 index 00000000..6996c6e7 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentGreen.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x55", + "green" : "0xB7", + "red" : "0x59" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x55", + "green" : "0xB7", + "red" : "0x59" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentRed.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentRed.colorset/Contents.json new file mode 100644 index 00000000..5b16437d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentRed.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x2F", + "green" : "0x2F", + "red" : "0xD3" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x2F", + "green" : "0x2F", + "red" : "0xD3" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentYellow.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentYellow.colorset/Contents.json new file mode 100644 index 00000000..e6938a59 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/AccentYellow.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x2D", + "green" : "0xC0", + "red" : "0xFB" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x2D", + "green" : "0xC0", + "red" : "0xFB" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige100.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige100.colorset/Contents.json new file mode 100644 index 00000000..c1b9a9d4 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige100.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xF0", + "green" : "0xF9", + "red" : "0xFF" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xF0", + "green" : "0xF9", + "red" : "0xFF" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige200.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige200.colorset/Contents.json new file mode 100644 index 00000000..09de815f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige200.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xE3", + "green" : "0xED", + "red" : "0xF4" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xE3", + "green" : "0xED", + "red" : "0xF4" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige300.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige300.colorset/Contents.json new file mode 100644 index 00000000..ae777ee4 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige300.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xD5", + "green" : "0xE1", + "red" : "0xEE" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xD5", + "green" : "0xE1", + "red" : "0xEE" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige400.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige400.colorset/Contents.json new file mode 100644 index 00000000..fbf34f2a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Beige400.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xA3", + "green" : "0xAE", + "red" : "0xBC" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xA3", + "green" : "0xAE", + "red" : "0xBC" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray100.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray100.colorset/Contents.json new file mode 100644 index 00000000..18d09c3b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray100.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xDC", + "green" : "0xDC", + "red" : "0xDC" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xDC", + "green" : "0xDC", + "red" : "0xDC" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray200.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray200.colorset/Contents.json new file mode 100644 index 00000000..3ab4aeb1 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray200.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xB4", + "green" : "0xB4", + "red" : "0xB4" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xB4", + "green" : "0xB4", + "red" : "0xB4" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray300.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray300.colorset/Contents.json new file mode 100644 index 00000000..7ebc6ad1 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray300.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x8E", + "green" : "0x8E", + "red" : "0x8E" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x8E", + "green" : "0x8E", + "red" : "0x8E" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray400.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray400.colorset/Contents.json new file mode 100644 index 00000000..af3f890a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray400.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x6A", + "green" : "0x6A", + "red" : "0x6A" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x6A", + "green" : "0x6A", + "red" : "0x6A" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray500.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray500.colorset/Contents.json new file mode 100644 index 00000000..09168b3a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray500.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x48", + "green" : "0x48", + "red" : "0x48" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x48", + "green" : "0x48", + "red" : "0x48" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray600.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray600.colorset/Contents.json new file mode 100644 index 00000000..71930475 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray600.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x28", + "green" : "0x28", + "red" : "0x28" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x28", + "green" : "0x28", + "red" : "0x28" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray700.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray700.colorset/Contents.json new file mode 100644 index 00000000..f9d5e17f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Gray700.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x11", + "green" : "0x11", + "red" : "0x11" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x11", + "green" : "0x11", + "red" : "0x11" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/LightGreen.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/LightGreen.colorset/Contents.json new file mode 100644 index 00000000..54587806 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/LightGreen.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x48", + "green" : "0xFF", + "red" : "0x4E" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x48", + "green" : "0xFF", + "red" : "0x4E" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/LightOrange.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/LightOrange.colorset/Contents.json new file mode 100644 index 00000000..f01c8e45 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/LightOrange.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x00", + "green" : "0x7C", + "red" : "0xF5" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x00", + "green" : "0x7C", + "red" : "0xF5" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange100.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange100.colorset/Contents.json new file mode 100644 index 00000000..e94797aa --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange100.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xCF", + "green" : "0xD8", + "red" : "0xF9" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xCF", + "green" : "0xD8", + "red" : "0xF9" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange200.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange200.colorset/Contents.json new file mode 100644 index 00000000..e3e0255e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange200.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x87", + "green" : "0xA4", + "red" : "0xF3" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x87", + "green" : "0xA4", + "red" : "0xF3" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange300.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange300.colorset/Contents.json new file mode 100644 index 00000000..346cf4c0 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange300.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x42", + "green" : "0x77", + "red" : "0xD9" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x42", + "green" : "0x77", + "red" : "0xD9" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange400.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange400.colorset/Contents.json new file mode 100644 index 00000000..4361175e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange400.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x30", + "green" : "0x59", + "red" : "0xA4" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x30", + "green" : "0x59", + "red" : "0xA4" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange500.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange500.colorset/Contents.json new file mode 100644 index 00000000..ea9d1af5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange500.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x1E", + "green" : "0x3C", + "red" : "0x72" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x1E", + "green" : "0x3C", + "red" : "0x72" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange600.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange600.colorset/Contents.json new file mode 100644 index 00000000..522b0e97 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange600.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x00", + "green" : "0x00", + "red" : "0x00" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x0E", + "green" : "0x21", + "red" : "0x44" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange700.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange700.colorset/Contents.json new file mode 100644 index 00000000..55c35845 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/Orange700.colorset/Contents.json @@ -0,0 +1,41 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x04", + "green" : "0x0C", + "red" : "0x1F" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x04", + "green" : "0x0C", + "red" : "0x1F" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "localizable" : true + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelBlue.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelBlue.colorset/Contents.json new file mode 100644 index 00000000..e5102683 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelBlue.colorset/Contents.json @@ -0,0 +1,38 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xF8", + "green" : "0xCF", + "red" : "0xC7" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xF8", + "green" : "0xCF", + "red" : "0xC7" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelGreen.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelGreen.colorset/Contents.json new file mode 100644 index 00000000..5bc8e2f8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelGreen.colorset/Contents.json @@ -0,0 +1,38 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xBC", + "green" : "0xF8", + "red" : "0xB0" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xBC", + "green" : "0xF8", + "red" : "0xB0" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelPink.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelPink.colorset/Contents.json new file mode 100644 index 00000000..ca7ee4ab --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelPink.colorset/Contents.json @@ -0,0 +1,38 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xFA", + "green" : "0x9A", + "red" : "0xF6" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xFA", + "green" : "0x9A", + "red" : "0xF6" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelYellow.colorset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelYellow.colorset/Contents.json new file mode 100644 index 00000000..f4ebbd1e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/AppColors/PastelYellow.colorset/Contents.json @@ -0,0 +1,38 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x9A", + "green" : "0xDF", + "red" : "0xFA" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x9A", + "green" : "0xDF", + "red" : "0xFA" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character1.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character1.imageset/Contents.json new file mode 100644 index 00000000..061375da --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character1.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "dodge_character1.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character1.imageset/dodge_character1.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character1.imageset/dodge_character1.png new file mode 100644 index 00000000..1105375f Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character1.imageset/dodge_character1.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character2.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character2.imageset/Contents.json new file mode 100644 index 00000000..86f6b032 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character2.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "dodge_character2.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character2.imageset/dodge_character2.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character2.imageset/dodge_character2.png new file mode 100644 index 00000000..5917214e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character2.imageset/dodge_character2.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character3.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character3.imageset/Contents.json new file mode 100644 index 00000000..fc496672 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character3.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "dodge_character3.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character3.imageset/dodge_character3.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character3.imageset/dodge_character3.png new file mode 100644 index 00000000..96670027 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_character3.imageset/dodge_character3.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_bug.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_bug.imageset/Contents.json new file mode 100644 index 00000000..cd25626b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_bug.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "dropItemError.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_bug.imageset/dropItemError.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_bug.imageset/dropItemError.png new file mode 100644 index 00000000..17d681a5 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_bug.imageset/dropItemError.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_large_gold.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_large_gold.imageset/Contents.json new file mode 100644 index 00000000..a4c41a18 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_large_gold.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "dropItemLargeGold.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_large_gold.imageset/dropItemLargeGold.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_large_gold.imageset/dropItemLargeGold.png new file mode 100644 index 00000000..39d70c7c Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_large_gold.imageset/dropItemLargeGold.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_small_gold.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_small_gold.imageset/Contents.json new file mode 100644 index 00000000..5c5d7d59 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_small_gold.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "dropItemSmallGold.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_small_gold.imageset/dropItemSmallGold.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_small_gold.imageset/dropItemSmallGold.png new file mode 100644 index 00000000..0594f736 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_drop_small_gold.imageset/dropItemSmallGold.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_ground.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_ground.imageset/Contents.json new file mode 100644 index 00000000..bda8fab8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_ground.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "dodge_ground.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_ground.imageset/dodge_ground.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_ground.imageset/dodge_ground.png new file mode 100644 index 00000000..990e1342 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Dodge/dodge_ground.imageset/dodge_ground.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_dart.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_dart.imageset/Contents.json new file mode 100644 index 00000000..420745f0 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_dart.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "language_dart@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_dart.imageset/language_dart@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_dart.imageset/language_dart@2x.png new file mode 100644 index 00000000..7691b743 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_dart.imageset/language_dart@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_kotlin.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_kotlin.imageset/Contents.json new file mode 100644 index 00000000..2c4a0bbd --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_kotlin.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "language_kotlin@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_kotlin.imageset/language_kotlin@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_kotlin.imageset/language_kotlin@2x.png new file mode 100644 index 00000000..c37e2310 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_kotlin.imageset/language_kotlin@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_python.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_python.imageset/Contents.json new file mode 100644 index 00000000..5dae1ac0 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_python.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "language_python@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_python.imageset/language_python@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_python.imageset/language_python@2x.png new file mode 100644 index 00000000..35117422 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_python.imageset/language_python@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_swift.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_swift.imageset/Contents.json new file mode 100644 index 00000000..c9407c60 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_swift.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "language_swift@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_swift.imageset/language_swift@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_swift.imageset/language_swift@2x.png new file mode 100644 index 00000000..c960b57d Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Language/language_swift.imageset/language_swift@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_bronze.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_bronze.imageset/Contents.json new file mode 100644 index 00000000..2338ebb2 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_bronze.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "mission_trophy_bronze@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_bronze.imageset/mission_trophy_bronze@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_bronze.imageset/mission_trophy_bronze@2x.png new file mode 100644 index 00000000..ff779b4f Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_bronze.imageset/mission_trophy_bronze@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_gold.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_gold.imageset/Contents.json new file mode 100644 index 00000000..d5811d97 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_gold.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "mission_trophy_gold@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_gold.imageset/mission_trophy_gold@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_gold.imageset/mission_trophy_gold@2x.png new file mode 100644 index 00000000..8f2ba60c Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_gold.imageset/mission_trophy_gold@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_silver.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_silver.imageset/Contents.json new file mode 100644 index 00000000..f2231c3e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_silver.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "mission_trophy_silver@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_silver.imageset/mission_trophy_silver@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_silver.imageset/mission_trophy_silver@2x.png new file mode 100644 index 00000000..b7f76371 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_silver.imageset/mission_trophy_silver@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_special.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_special.imageset/Contents.json new file mode 100644 index 00000000..a4a0ff0a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_special.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "mission_trophy_special@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_special.imageset/mission_trophy_special@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_special.imageset/mission_trophy_special@2x.png new file mode 100644 index 00000000..46f78e71 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Mission/mission_trophy_special.imageset/mission_trophy_special@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_face.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_face.imageset/Contents.json new file mode 100644 index 00000000..fe47524b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_face.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "quiz_dog_face@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_face.imageset/quiz_dog_face@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_face.imageset/quiz_dog_face@2x.png new file mode 100644 index 00000000..9f681523 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_face.imageset/quiz_dog_face@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_foot.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_foot.imageset/Contents.json new file mode 100644 index 00000000..155f4c8d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_foot.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "quiz_dog_foot@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_foot.imageset/quiz_dog_foot@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_foot.imageset/quiz_dog_foot@2x.png new file mode 100644 index 00000000..9e468513 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Quiz/quiz_dog_foot.imageset/quiz_dog_foot@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_blue.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_blue.imageset/Contents.json new file mode 100644 index 00000000..a90b2269 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_blue.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "block_blue.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_blue.imageset/block_blue.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_blue.imageset/block_blue.png new file mode 100644 index 00000000..be673954 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_blue.imageset/block_blue.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb.imageset/Contents.json new file mode 100644 index 00000000..d0132c92 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "block_bomb.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb.imageset/block_bomb.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb.imageset/block_bomb.png new file mode 100644 index 00000000..1d5940c5 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb.imageset/block_bomb.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb2.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb2.imageset/Contents.json new file mode 100644 index 00000000..7a827890 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb2.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "block_bomb2.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb2.imageset/block_bomb2.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb2.imageset/block_bomb2.png new file mode 100644 index 00000000..24d09174 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_bomb2.imageset/block_bomb2.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_green.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_green.imageset/Contents.json new file mode 100644 index 00000000..53113fd5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_green.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "block_green.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_green.imageset/block_green.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_green.imageset/block_green.png new file mode 100644 index 00000000..a4e76e10 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_green.imageset/block_green.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_orange.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_orange.imageset/Contents.json new file mode 100644 index 00000000..32c3996a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_orange.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "block_orange.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_orange.imageset/block_orange.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_orange.imageset/block_orange.png new file mode 100644 index 00000000..9b71b589 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_orange.imageset/block_orange.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_purple.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_purple.imageset/Contents.json new file mode 100644 index 00000000..fcdef825 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_purple.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "block_purple.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_purple.imageset/block_purple.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_purple.imageset/block_purple.png new file mode 100644 index 00000000..63d3d376 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_purple.imageset/block_purple.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_red.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_red.imageset/Contents.json new file mode 100644 index 00000000..966e3a26 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_red.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "block_red.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_red.imageset/block_red.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_red.imageset/block_red.png new file mode 100644 index 00000000..4e7ef8d8 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_red.imageset/block_red.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_yellow.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_yellow.imageset/Contents.json new file mode 100644 index 00000000..7c59ed1c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_yellow.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "block_yellow.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_yellow.imageset/block_yellow.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_yellow.imageset/block_yellow.png new file mode 100644 index 00000000..fc99beda Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Stack/stack_block_yellow.imageset/block_yellow.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/tap_background.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/tap_background.imageset/Contents.json new file mode 100644 index 00000000..0f4265b6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/tap_background.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "background_tapGame@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/tap_background.imageset/background_tapGame@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/tap_background.imageset/background_tapGame@2x.png new file mode 100644 index 00000000..a489a7bb Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Game/Tap/tap_background.imageset/background_tapGame@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_apartment.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_apartment.imageset/Contents.json new file mode 100644 index 00000000..19784b23 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_apartment.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "housing_apartment.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_apartment.imageset/housing_apartment.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_apartment.imageset/housing_apartment.png new file mode 100644 index 00000000..cd98d8f4 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_apartment.imageset/housing_apartment.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_house.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_house.imageset/Contents.json new file mode 100644 index 00000000..435981bf --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_house.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "housing_house.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_house.imageset/housing_house.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_house.imageset/housing_house.png new file mode 100644 index 00000000..2ba67833 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_house.imageset/housing_house.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_pentHouse.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_pentHouse.imageset/Contents.json new file mode 100644 index 00000000..24c153e3 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_pentHouse.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "housing_pentHouse.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_pentHouse.imageset/housing_pentHouse.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_pentHouse.imageset/housing_pentHouse.png new file mode 100644 index 00000000..2288ebfa Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_pentHouse.imageset/housing_pentHouse.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_rooftop.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_rooftop.imageset/Contents.json new file mode 100644 index 00000000..984787b7 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_rooftop.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "housing_rooftop.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_rooftop.imageset/housing_rooftop.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_rooftop.imageset/housing_rooftop.png new file mode 100644 index 00000000..7b9ffec6 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_rooftop.imageset/housing_rooftop.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_semiBasement.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_semiBasement.imageset/Contents.json new file mode 100644 index 00000000..e8da3e2f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_semiBasement.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "housing_semiBasement.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_semiBasement.imageset/housing_semiBasement.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_semiBasement.imageset/housing_semiBasement.png new file mode 100644 index 00000000..3a97fed0 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_semiBasement.imageset/housing_semiBasement.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_street.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_street.imageset/Contents.json new file mode 100644 index 00000000..bb227808 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_street.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "housing_street.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_street.imageset/housing_street.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_street.imageset/housing_street.png new file mode 100644 index 00000000..b35c32e2 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_street.imageset/housing_street.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_villa.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_villa.imageset/Contents.json new file mode 100644 index 00000000..15ec5ec4 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_villa.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "housing_villa.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_villa.imageset/housing_villa.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_villa.imageset/housing_villa.png new file mode 100644 index 00000000..69fcfab5 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Housing/housing_villa.imageset/housing_villa.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_cancel.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_cancel.imageset/Contents.json new file mode 100644 index 00000000..6aa30ea2 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_cancel.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "Gemini_Generated_Image_ppwnj4ppwnj4ppwn-Photoroom 1@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_cancel.imageset/Gemini_Generated_Image_ppwnj4ppwnj4ppwn-Photoroom 1@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_cancel.imageset/Gemini_Generated_Image_ppwnj4ppwnj4ppwn-Photoroom 1@2x.png new file mode 100644 index 00000000..717ff453 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_cancel.imageset/Gemini_Generated_Image_ppwnj4ppwnj4ppwn-Photoroom 1@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_close.imageset/Contents.json new file mode 100644 index 00000000..90432ad5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_close.imageset/close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_close.imageset/close.png new file mode 100644 index 00000000..2e444534 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_close.imageset/close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coffee.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coffee.imageset/Contents.json new file mode 100644 index 00000000..0ff63661 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coffee.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "coffee.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coffee.imageset/coffee.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coffee.imageset/coffee.png new file mode 100644 index 00000000..958c8f77 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coffee.imageset/coffee.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_bag.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_bag.imageset/Contents.json new file mode 100644 index 00000000..81a13d6f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_bag.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "icon_coin_bag@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_bag.imageset/icon_coin_bag@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_bag.imageset/icon_coin_bag@2x.png new file mode 100644 index 00000000..70d40c79 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_bag.imageset/icon_coin_bag@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_stack.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_stack.imageset/Contents.json new file mode 100644 index 00000000..7de9e89a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_stack.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "icon_coin_stack.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_stack.imageset/icon_coin_stack.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_stack.imageset/icon_coin_stack.png new file mode 100644 index 00000000..74a15da9 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_coin_stack.imageset/icon_coin_stack.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_green.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_green.imageset/Contents.json new file mode 100644 index 00000000..125d5a9a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_green.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "icon_diamond_green@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_green.imageset/icon_diamond_green@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_green.imageset/icon_diamond_green@2x.png new file mode 100644 index 00000000..6ac17e4b Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_green.imageset/icon_diamond_green@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_plus.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_plus.imageset/Contents.json new file mode 100644 index 00000000..a7c214e6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_plus.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "icon_diamond_plus.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_plus.imageset/icon_diamond_plus.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_plus.imageset/icon_diamond_plus.png new file mode 100644 index 00000000..66e593d2 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_diamond_plus.imageset/icon_diamond_plus.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_energy_drink.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_energy_drink.imageset/Contents.json new file mode 100644 index 00000000..ac827ca6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_energy_drink.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "energyDrink.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_energy_drink.imageset/energyDrink.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_energy_drink.imageset/energyDrink.png new file mode 100644 index 00000000..2661371d Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_energy_drink.imageset/energyDrink.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_lock.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_lock.imageset/Contents.json new file mode 100644 index 00000000..06b999aa --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_lock.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "lock.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_lock.imageset/lock.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_lock.imageset/lock.png new file mode 100644 index 00000000..f18eb180 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_lock.imageset/lock.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_minus.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_minus.imageset/Contents.json new file mode 100644 index 00000000..84a4bae8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_minus.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "minus.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_minus.imageset/minus.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_minus.imageset/minus.png new file mode 100644 index 00000000..6ca92687 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_minus.imageset/minus.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_mission.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_mission.imageset/Contents.json new file mode 100644 index 00000000..a19280c3 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_mission.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "Rectangle 245-3.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_mission.imageset/Rectangle 245-3.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_mission.imageset/Rectangle 245-3.png new file mode 100644 index 00000000..e150f213 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_mission.imageset/Rectangle 245-3.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_new_badge.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_new_badge.imageset/Contents.json new file mode 100644 index 00000000..ef2f6dde --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_new_badge.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "icon_new_badge@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_new_badge.imageset/icon_new_badge@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_new_badge.imageset/icon_new_badge@2x.png new file mode 100644 index 00000000..fa193a6a Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_new_badge.imageset/icon_new_badge@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_play.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_play.imageset/Contents.json new file mode 100644 index 00000000..fa6ee137 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_play.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "Gemini_Generated_Image_ppwnj4ppwnj4ppwn-Photoroom 2@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_play.imageset/Gemini_Generated_Image_ppwnj4ppwnj4ppwn-Photoroom 2@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_play.imageset/Gemini_Generated_Image_ppwnj4ppwnj4ppwn-Photoroom 2@2x.png new file mode 100644 index 00000000..57fdfc3d Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_play.imageset/Gemini_Generated_Image_ppwnj4ppwnj4ppwn-Photoroom 2@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_plus.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_plus.imageset/Contents.json new file mode 100644 index 00000000..52e243f5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_plus.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "add.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_plus.imageset/add.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_plus.imageset/add.png new file mode 100644 index 00000000..11d0a648 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_plus.imageset/add.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_setting.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_setting.imageset/Contents.json new file mode 100644 index 00000000..5ff478a0 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_setting.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "icon_setting.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_setting.imageset/icon_setting.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_setting.imageset/icon_setting.png new file mode 100644 index 00000000..5f66f3fb Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_setting.imageset/icon_setting.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_shop.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_shop.imageset/Contents.json new file mode 100644 index 00000000..06f184ff --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_shop.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "Rectangle 245-2.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_shop.imageset/Rectangle 245-2.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_shop.imageset/Rectangle 245-2.png new file mode 100644 index 00000000..55837b28 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_shop.imageset/Rectangle 245-2.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_skill.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_skill.imageset/Contents.json new file mode 100644 index 00000000..b4aabe09 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_skill.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "Rectangle 245-1.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_skill.imageset/Rectangle 245-1.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_skill.imageset/Rectangle 245-1.png new file mode 100644 index 00000000..e2b23a84 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_skill.imageset/Rectangle 245-1.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_work.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_work.imageset/Contents.json new file mode 100644 index 00000000..f0b85321 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_work.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "Rectangle 245.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_work.imageset/Rectangle 245.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_work.imageset/Rectangle 245.png new file mode 100644 index 00000000..3f0b83c6 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Icons/icon_work.imageset/Rectangle 245.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_broken.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_broken.imageset/Contents.json new file mode 100644 index 00000000..d791335e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_broken.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_chair_broken.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_broken.imageset/item_chair_broken.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_broken.imageset/item_chair_broken.png new file mode 100644 index 00000000..bfc63adb Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_broken.imageset/item_chair_broken.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_cheap.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_cheap.imageset/Contents.json new file mode 100644 index 00000000..53107657 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_cheap.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_chair_cheap.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_cheap.imageset/item_chair_cheap.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_cheap.imageset/item_chair_cheap.png new file mode 100644 index 00000000..5d9ef74e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_cheap.imageset/item_chair_cheap.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_decent.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_decent.imageset/Contents.json new file mode 100644 index 00000000..c4f155b1 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_decent.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_chair_decent.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_decent.imageset/item_chair_decent.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_decent.imageset/item_chair_decent.png new file mode 100644 index 00000000..212550f4 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_decent.imageset/item_chair_decent.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_diamond.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_diamond.imageset/Contents.json new file mode 100644 index 00000000..990f6257 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_diamond.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_chair_diamond.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_diamond.imageset/item_chair_diamond.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_diamond.imageset/item_chair_diamond.png new file mode 100644 index 00000000..d2094524 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_diamond.imageset/item_chair_diamond.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_limited.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_limited.imageset/Contents.json new file mode 100644 index 00000000..9626801b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_limited.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_chair_limited.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_limited.imageset/item_chair_limited.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_limited.imageset/item_chair_limited.png new file mode 100644 index 00000000..0042927a Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_limited.imageset/item_chair_limited.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_nationalTreasure.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_nationalTreasure.imageset/Contents.json new file mode 100644 index 00000000..294996a6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_nationalTreasure.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_chair_nationalTreasure.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_nationalTreasure.imageset/item_chair_nationalTreasure.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_nationalTreasure.imageset/item_chair_nationalTreasure.png new file mode 100644 index 00000000..29497f43 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_nationalTreasure.imageset/item_chair_nationalTreasure.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_premium.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_premium.imageset/Contents.json new file mode 100644 index 00000000..80c17cf6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_premium.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_chair_premium.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_premium.imageset/item_chair_premium.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_premium.imageset/item_chair_premium.png new file mode 100644 index 00000000..e3121855 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_premium.imageset/item_chair_premium.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_vintage.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_vintage.imageset/Contents.json new file mode 100644 index 00000000..ee3ed275 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_vintage.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_chair_vintage.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_vintage.imageset/item_chair_vintage.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_vintage.imageset/item_chair_vintage.png new file mode 100644 index 00000000..2bd38fa4 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/chair/item_chair_vintage.imageset/item_chair_vintage.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_broken.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_broken.imageset/Contents.json new file mode 100644 index 00000000..73046ef2 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_broken.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_keyboard_broken.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_broken.imageset/item_keyboard_broken.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_broken.imageset/item_keyboard_broken.png new file mode 100644 index 00000000..737bac78 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_broken.imageset/item_keyboard_broken.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_cheap.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_cheap.imageset/Contents.json new file mode 100644 index 00000000..84086187 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_cheap.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_keyboard_limited.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_cheap.imageset/item_keyboard_limited.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_cheap.imageset/item_keyboard_limited.png new file mode 100644 index 00000000..bff03939 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_cheap.imageset/item_keyboard_limited.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_decent.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_decent.imageset/Contents.json new file mode 100644 index 00000000..c5aeb80e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_decent.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_keyboard_decent.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_decent.imageset/item_keyboard_decent.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_decent.imageset/item_keyboard_decent.png new file mode 100644 index 00000000..e4fab853 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_decent.imageset/item_keyboard_decent.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_diamond.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_diamond.imageset/Contents.json new file mode 100644 index 00000000..6d82f01b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_diamond.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_keyboard_diamond.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_diamond.imageset/item_keyboard_diamond.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_diamond.imageset/item_keyboard_diamond.png new file mode 100644 index 00000000..d16f97b5 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_diamond.imageset/item_keyboard_diamond.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_limited.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_limited.imageset/Contents.json new file mode 100644 index 00000000..1c54a97d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_limited.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_keyboard_vintage.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_limited.imageset/item_keyboard_vintage.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_limited.imageset/item_keyboard_vintage.png new file mode 100644 index 00000000..87e30c7e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_limited.imageset/item_keyboard_vintage.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_nationalTreasure.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_nationalTreasure.imageset/Contents.json new file mode 100644 index 00000000..67997b1d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_nationalTreasure.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_keyboard_nationalTreasure.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_nationalTreasure.imageset/item_keyboard_nationalTreasure.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_nationalTreasure.imageset/item_keyboard_nationalTreasure.png new file mode 100644 index 00000000..9b6eb701 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_nationalTreasure.imageset/item_keyboard_nationalTreasure.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_premium.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_premium.imageset/Contents.json new file mode 100644 index 00000000..bd4535ee --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_premium.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_keyboard_premium.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_premium.imageset/item_keyboard_premium.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_premium.imageset/item_keyboard_premium.png new file mode 100644 index 00000000..cf9f4556 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_premium.imageset/item_keyboard_premium.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_vintage.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_vintage.imageset/Contents.json new file mode 100644 index 00000000..823e0778 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_vintage.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_keyboard_cheap.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_vintage.imageset/item_keyboard_cheap.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_vintage.imageset/item_keyboard_cheap.png new file mode 100644 index 00000000..c0cc6260 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/keyboard/item_keyboard_vintage.imageset/item_keyboard_cheap.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_broken.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_broken.imageset/Contents.json new file mode 100644 index 00000000..a0fb8123 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_broken.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_monitor_broken.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_broken.imageset/item_monitor_broken.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_broken.imageset/item_monitor_broken.png new file mode 100644 index 00000000..2c89d49c Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_broken.imageset/item_monitor_broken.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_cheap.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_cheap.imageset/Contents.json new file mode 100644 index 00000000..656a1e75 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_cheap.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_monitor_cheap.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_cheap.imageset/item_monitor_cheap.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_cheap.imageset/item_monitor_cheap.png new file mode 100644 index 00000000..0cbff4ab Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_cheap.imageset/item_monitor_cheap.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_decent.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_decent.imageset/Contents.json new file mode 100644 index 00000000..cc7dab47 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_decent.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_monitor_decent.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_decent.imageset/item_monitor_decent.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_decent.imageset/item_monitor_decent.png new file mode 100644 index 00000000..d28b86bb Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_decent.imageset/item_monitor_decent.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_diamond.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_diamond.imageset/Contents.json new file mode 100644 index 00000000..0f658aa8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_diamond.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_monitor_diamond.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_diamond.imageset/item_monitor_diamond.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_diamond.imageset/item_monitor_diamond.png new file mode 100644 index 00000000..fa0f8b6b Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_diamond.imageset/item_monitor_diamond.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_limited.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_limited.imageset/Contents.json new file mode 100644 index 00000000..cf88eb78 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_limited.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_monitor_limited.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_limited.imageset/item_monitor_limited.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_limited.imageset/item_monitor_limited.png new file mode 100644 index 00000000..ce7af1bc Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_limited.imageset/item_monitor_limited.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_nationalTreasure.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_nationalTreasure.imageset/Contents.json new file mode 100644 index 00000000..8f6cf330 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_nationalTreasure.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_monitor_nationalTreasure.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_nationalTreasure.imageset/item_monitor_nationalTreasure.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_nationalTreasure.imageset/item_monitor_nationalTreasure.png new file mode 100644 index 00000000..c95690bc Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_nationalTreasure.imageset/item_monitor_nationalTreasure.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_premium.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_premium.imageset/Contents.json new file mode 100644 index 00000000..b6a82799 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_premium.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_monitor_premium.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_premium.imageset/item_monitor_premium.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_premium.imageset/item_monitor_premium.png new file mode 100644 index 00000000..f113dbe9 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_premium.imageset/item_monitor_premium.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_vintage.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_vintage.imageset/Contents.json new file mode 100644 index 00000000..61a30163 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_vintage.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_monitor_vintage.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_vintage.imageset/item_monitor_vintage.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_vintage.imageset/item_monitor_vintage.png new file mode 100644 index 00000000..b3cd9ba2 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/monitor/item_monitor_vintage.imageset/item_monitor_vintage.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_broken.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_broken.imageset/Contents.json new file mode 100644 index 00000000..ab38e460 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_broken.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_mouse_broken.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_broken.imageset/item_mouse_broken.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_broken.imageset/item_mouse_broken.png new file mode 100644 index 00000000..cb7cbe32 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_broken.imageset/item_mouse_broken.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_cheap.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_cheap.imageset/Contents.json new file mode 100644 index 00000000..a967cd05 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_cheap.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_mouse_cheap.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_cheap.imageset/item_mouse_cheap.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_cheap.imageset/item_mouse_cheap.png new file mode 100644 index 00000000..38fdb790 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_cheap.imageset/item_mouse_cheap.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_decent.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_decent.imageset/Contents.json new file mode 100644 index 00000000..93869765 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_decent.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_mouse_decent.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_decent.imageset/item_mouse_decent.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_decent.imageset/item_mouse_decent.png new file mode 100644 index 00000000..0c19e084 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_decent.imageset/item_mouse_decent.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_diamond.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_diamond.imageset/Contents.json new file mode 100644 index 00000000..726284d1 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_diamond.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_mouse_diamond.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_diamond.imageset/item_mouse_diamond.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_diamond.imageset/item_mouse_diamond.png new file mode 100644 index 00000000..74633ac9 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_diamond.imageset/item_mouse_diamond.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_limited.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_limited.imageset/Contents.json new file mode 100644 index 00000000..bc708abc --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_limited.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_mouse_limited.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_limited.imageset/item_mouse_limited.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_limited.imageset/item_mouse_limited.png new file mode 100644 index 00000000..19d45726 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_limited.imageset/item_mouse_limited.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_nationalTreasure.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_nationalTreasure.imageset/Contents.json new file mode 100644 index 00000000..861e8f30 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_nationalTreasure.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "nationalTreasure.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_nationalTreasure.imageset/nationalTreasure.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_nationalTreasure.imageset/nationalTreasure.png new file mode 100644 index 00000000..19980578 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_nationalTreasure.imageset/nationalTreasure.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_premium.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_premium.imageset/Contents.json new file mode 100644 index 00000000..071fa110 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_premium.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_mouse_premium.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_premium.imageset/item_mouse_premium.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_premium.imageset/item_mouse_premium.png new file mode 100644 index 00000000..883d6466 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_premium.imageset/item_mouse_premium.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_vintage.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_vintage.imageset/Contents.json new file mode 100644 index 00000000..6778f89e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_vintage.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "item_mouse_vintage.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_vintage.imageset/item_mouse_vintage.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_vintage.imageset/item_mouse_vintage.png new file mode 100644 index 00000000..e7bc33b4 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Item/mouse/item_mouse_vintage.imageset/item_mouse_vintage.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_all_rounder_developer.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_all_rounder_developer.imageset/Contents.json new file mode 100644 index 00000000..dc72569b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_all_rounder_developer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_allRounderDeveloper.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_all_rounder_developer.imageset/profile_allRounderDeveloper.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_all_rounder_developer.imageset/profile_allRounderDeveloper.png new file mode 100644 index 00000000..72937834 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_all_rounder_developer.imageset/profile_allRounderDeveloper.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_aspiring_developer.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_aspiring_developer.imageset/Contents.json new file mode 100644 index 00000000..847672e9 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_aspiring_developer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_aspiringDeveloper.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_aspiring_developer.imageset/profile_aspiringDeveloper.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_aspiring_developer.imageset/profile_aspiringDeveloper.png new file mode 100644 index 00000000..52162f43 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_aspiring_developer.imageset/profile_aspiringDeveloper.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_famous_developer.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_famous_developer.imageset/Contents.json new file mode 100644 index 00000000..a5ddcc42 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_famous_developer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_famousDeveloper.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_famous_developer.imageset/profile_famousDeveloper.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_famous_developer.imageset/profile_famousDeveloper.png new file mode 100644 index 00000000..93e501fe Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_famous_developer.imageset/profile_famousDeveloper.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_junior_developer.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_junior_developer.imageset/Contents.json new file mode 100644 index 00000000..16facf01 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_junior_developer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_juniorDeveloper.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_junior_developer.imageset/profile_juniorDeveloper.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_junior_developer.imageset/profile_juniorDeveloper.png new file mode 100644 index 00000000..9e9ed8df Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_junior_developer.imageset/profile_juniorDeveloper.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_laptop_owner.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_laptop_owner.imageset/Contents.json new file mode 100644 index 00000000..8462b57c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_laptop_owner.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_laptopOwner.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_laptop_owner.imageset/profile_laptopOwner.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_laptop_owner.imageset/profile_laptopOwner.png new file mode 100644 index 00000000..36df91d0 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_laptop_owner.imageset/profile_laptopOwner.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_locked.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_locked.imageset/Contents.json new file mode 100644 index 00000000..09dfffc8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_locked.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_locked.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_locked.imageset/profile_locked.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_locked.imageset/profile_locked.png new file mode 100644 index 00000000..70326e62 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_locked.imageset/profile_locked.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_night_owl_developer.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_night_owl_developer.imageset/Contents.json new file mode 100644 index 00000000..8ac41250 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_night_owl_developer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_nightOwlDeveloper.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_night_owl_developer.imageset/profile_nightOwlDeveloper.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_night_owl_developer.imageset/profile_nightOwlDeveloper.png new file mode 100644 index 00000000..049d2d83 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_night_owl_developer.imageset/profile_nightOwlDeveloper.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_normal_developer.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_normal_developer.imageset/Contents.json new file mode 100644 index 00000000..98df45aa --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_normal_developer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_normalDeveloper.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_normal_developer.imageset/profile_normalDeveloper.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_normal_developer.imageset/profile_normalDeveloper.png new file mode 100644 index 00000000..1c8908cc Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_normal_developer.imageset/profile_normalDeveloper.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_skilled_developer.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_skilled_developer.imageset/Contents.json new file mode 100644 index 00000000..2df39e02 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_skilled_developer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_skilledDeveloper.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_skilled_developer.imageset/profile_skilledDeveloper.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_skilled_developer.imageset/profile_skilledDeveloper.png new file mode 100644 index 00000000..d42c896b Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_skilled_developer.imageset/profile_skilledDeveloper.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_unemployed.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_unemployed.imageset/Contents.json new file mode 100644 index 00000000..c001c728 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_unemployed.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_unemployed.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_unemployed.imageset/profile_unemployed.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_unemployed.imageset/profile_unemployed.png new file mode 100644 index 00000000..1289387f Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_unemployed.imageset/profile_unemployed.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_world_class_developer.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_world_class_developer.imageset/Contents.json new file mode 100644 index 00000000..11d0d3b6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_world_class_developer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "profile_worldClassDeveloper.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_world_class_developer.imageset/profile_worldClassDeveloper.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_world_class_developer.imageset/profile_worldClassDeveloper.png new file mode 100644 index 00000000..4686a578 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Profiles/profile_world_class_developer.imageset/profile_worldClassDeveloper.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-1.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-1.png new file mode 100644 index 00000000..e2b23a84 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-1.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-2.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-2.png new file mode 100644 index 00000000..55837b28 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-2.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-3.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-3.png new file mode 100644 index 00000000..e150f213 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245-3.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245.png new file mode 100644 index 00000000..3f0b83c6 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Rectangle 245.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_1.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_1.imageset/Contents.json new file mode 100644 index 00000000..24995c7f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_1.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_dodge_1.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_1.imageset/skill_dodge_1.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_1.imageset/skill_dodge_1.png new file mode 100644 index 00000000..8019f5d1 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_1.imageset/skill_dodge_1.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_2.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_2.imageset/Contents.json new file mode 100644 index 00000000..c7c8a3df --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_2.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_dodge_2.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_2.imageset/skill_dodge_2.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_2.imageset/skill_dodge_2.png new file mode 100644 index 00000000..efb16c02 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_2.imageset/skill_dodge_2.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_3.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_3.imageset/Contents.json new file mode 100644 index 00000000..91875a90 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_3.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_dodge_3.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_3.imageset/skill_dodge_3.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_3.imageset/skill_dodge_3.png new file mode 100644 index 00000000..5a126eb8 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_dodge_3.imageset/skill_dodge_3.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_1.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_1.imageset/Contents.json new file mode 100644 index 00000000..bcc58d16 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_1.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_language_1.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_1.imageset/skill_language_1.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_1.imageset/skill_language_1.png new file mode 100644 index 00000000..3848a4c6 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_1.imageset/skill_language_1.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_2.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_2.imageset/Contents.json new file mode 100644 index 00000000..81598679 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_2.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_language_2.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_2.imageset/skill_language_2.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_2.imageset/skill_language_2.png new file mode 100644 index 00000000..e3200d48 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_2.imageset/skill_language_2.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_3.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_3.imageset/Contents.json new file mode 100644 index 00000000..11eb05cf --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_3.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_language_3.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_3.imageset/skill_language_3.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_3.imageset/skill_language_3.png new file mode 100644 index 00000000..b0191f06 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_language_3.imageset/skill_language_3.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_1.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_1.imageset/Contents.json new file mode 100644 index 00000000..20fbb971 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_1.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_stack_1.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_1.imageset/skill_stack_1.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_1.imageset/skill_stack_1.png new file mode 100644 index 00000000..52e27754 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_1.imageset/skill_stack_1.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_2.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_2.imageset/Contents.json new file mode 100644 index 00000000..9c354157 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_2.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_stack_2.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_2.imageset/skill_stack_2.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_2.imageset/skill_stack_2.png new file mode 100644 index 00000000..73986b05 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_2.imageset/skill_stack_2.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_3.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_3.imageset/Contents.json new file mode 100644 index 00000000..994caecf --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_3.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_stack_3.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_3.imageset/skill_stack_3.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_3.imageset/skill_stack_3.png new file mode 100644 index 00000000..112e3649 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_stack_3.imageset/skill_stack_3.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_1.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_1.imageset/Contents.json new file mode 100644 index 00000000..bbef6a5c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_1.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_tap_1.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_1.imageset/skill_tap_1.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_1.imageset/skill_tap_1.png new file mode 100644 index 00000000..a97ce954 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_1.imageset/skill_tap_1.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_2.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_2.imageset/Contents.json new file mode 100644 index 00000000..73c006c2 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_2.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_tap_2.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_2.imageset/skill_tap_2.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_2.imageset/skill_tap_2.png new file mode 100644 index 00000000..20231af4 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_2.imageset/skill_tap_2.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_3.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_3.imageset/Contents.json new file mode 100644 index 00000000..317ac0b3 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_3.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "skill_tap_3.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_3.imageset/skill_tap_3.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_3.imageset/skill_tap_3.png new file mode 100644 index 00000000..7e4d47dd Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Skill/skill_tap_3.imageset/skill_tap_3.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_career.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_career.imageset/Contents.json new file mode 100644 index 00000000..480cb66e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_career.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "tutorial_career.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_career.imageset/tutorial_career.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_career.imageset/tutorial_career.png new file mode 100644 index 00000000..1dc8349e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_career.imageset/tutorial_career.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_housing.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_housing.imageset/Contents.json new file mode 100644 index 00000000..6dcc84a9 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_housing.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "tutorial_housing.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_housing.imageset/tutorial_housing.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_housing.imageset/tutorial_housing.png new file mode 100644 index 00000000..462d42ea Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_housing.imageset/tutorial_housing.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_item.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_item.imageset/Contents.json new file mode 100644 index 00000000..d0a09aa5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_item.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "tutorial_item.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_item.imageset/tutorial_item.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_item.imageset/tutorial_item.png new file mode 100644 index 00000000..e8044062 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_item.imageset/tutorial_item.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_mission.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_mission.imageset/Contents.json new file mode 100644 index 00000000..c52729d5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_mission.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "tutorial_mission.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_mission.imageset/tutorial_mission.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_mission.imageset/tutorial_mission.png new file mode 100644 index 00000000..2d91a8da Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_mission.imageset/tutorial_mission.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_quiz.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_quiz.imageset/Contents.json new file mode 100644 index 00000000..ed2ccfc2 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_quiz.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "tutorial_quiz.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_quiz.imageset/tutorial_quiz.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_quiz.imageset/tutorial_quiz.png new file mode 100644 index 00000000..ad7772be Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_quiz.imageset/tutorial_quiz.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_skill.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_skill.imageset/Contents.json new file mode 100644 index 00000000..6117cb62 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_skill.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "tutorial_skill.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_skill.imageset/tutorial_skill.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_skill.imageset/tutorial_skill.png new file mode 100644 index 00000000..65900c5c Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_skill.imageset/tutorial_skill.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_work.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_work.imageset/Contents.json new file mode 100644 index 00000000..0df250d0 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_work.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "tutorial_work.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_work.imageset/tutorial_work.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_work.imageset/tutorial_work.png new file mode 100644 index 00000000..810df77a Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/Tutorial/tutorial_work.imageset/tutorial_work.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_dev.appiconset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_dev.appiconset/Contents.json new file mode 100644 index 00000000..6ba9ffdd --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_dev.appiconset/Contents.json @@ -0,0 +1,36 @@ +{ + "images" : [ + { + "filename" : "app_icon_dev.png", + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "tinted" + } + ], + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_dev.appiconset/app_icon_dev.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_dev.appiconset/app_icon_dev.png new file mode 100644 index 00000000..4f91f5f9 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_dev.appiconset/app_icon_dev.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_release.appiconset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_release.appiconset/Contents.json new file mode 100644 index 00000000..b9a1c351 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_release.appiconset/Contents.json @@ -0,0 +1,36 @@ +{ + "images" : [ + { + "filename" : "app_icon_release.png", + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "tinted" + } + ], + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_release.appiconset/app_icon_release.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_release.appiconset/app_icon_release.png new file mode 100644 index 00000000..04e6cff5 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_icon_release.appiconset/app_icon_release.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_launch_screen.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_launch_screen.imageset/Contents.json new file mode 100644 index 00000000..ece3083b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_launch_screen.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "app_launch_screen@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_launch_screen.imageset/app_launch_screen@2x.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_launch_screen.imageset/app_launch_screen@2x.png new file mode 100644 index 00000000..d41f267e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/app_launch_screen.imageset/app_launch_screen@2x.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_close.imageset/Contents.json new file mode 100644 index 00000000..7e437a54 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_all_rounder_developer_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_close.imageset/character_all_rounder_developer_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_close.imageset/character_all_rounder_developer_close.png new file mode 100644 index 00000000..1f6fc1b9 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_close.imageset/character_all_rounder_developer_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_default.imageset/Contents.json new file mode 100644 index 00000000..15176cd2 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_all_rounder_developer_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_default.imageset/character_all_rounder_developer_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_default.imageset/character_all_rounder_developer_default.png new file mode 100644 index 00000000..98ffd096 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_default.imageset/character_all_rounder_developer_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_smile.imageset/Contents.json new file mode 100644 index 00000000..853252a5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_all_rounder_developer_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_smile.imageset/character_all_rounder_developer_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_smile.imageset/character_all_rounder_developer_smile.png new file mode 100644 index 00000000..16b4bc12 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_all_rounder_developer_smile.imageset/character_all_rounder_developer_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_close.imageset/Contents.json new file mode 100644 index 00000000..d4b554cc --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_aspiring_developer_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_close.imageset/character_aspiring_developer_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_close.imageset/character_aspiring_developer_close.png new file mode 100644 index 00000000..ce09dc9b Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_close.imageset/character_aspiring_developer_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_default.imageset/Contents.json new file mode 100644 index 00000000..903259df --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_aspiring_developer_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_default.imageset/character_aspiring_developer_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_default.imageset/character_aspiring_developer_default.png new file mode 100644 index 00000000..a908ef62 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_default.imageset/character_aspiring_developer_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_smile.imageset/Contents.json new file mode 100644 index 00000000..71022938 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_aspiring_developer_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_smile.imageset/character_aspiring_developer_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_smile.imageset/character_aspiring_developer_smile.png new file mode 100644 index 00000000..47b565ef Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_aspiring_developer_smile.imageset/character_aspiring_developer_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_close.imageset/Contents.json new file mode 100644 index 00000000..54fb09a2 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_famous_developer_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_close.imageset/character_famous_developer_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_close.imageset/character_famous_developer_close.png new file mode 100644 index 00000000..cc0509a5 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_close.imageset/character_famous_developer_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_default.imageset/Contents.json new file mode 100644 index 00000000..5c118bb8 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_famous_developer_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_default.imageset/character_famous_developer_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_default.imageset/character_famous_developer_default.png new file mode 100644 index 00000000..ced12e66 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_default.imageset/character_famous_developer_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_smile.imageset/Contents.json new file mode 100644 index 00000000..a31604cd --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_famous_developer_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_smile.imageset/character_famous_developer_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_smile.imageset/character_famous_developer_smile.png new file mode 100644 index 00000000..d82db87e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_famous_developer_smile.imageset/character_famous_developer_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_close.imageset/Contents.json new file mode 100644 index 00000000..d85ffcba --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_junior_developer_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_close.imageset/character_junior_developer_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_close.imageset/character_junior_developer_close.png new file mode 100644 index 00000000..d3f9bca0 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_close.imageset/character_junior_developer_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_default.imageset/Contents.json new file mode 100644 index 00000000..45f8fc00 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_junior_developer_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_default.imageset/character_junior_developer_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_default.imageset/character_junior_developer_default.png new file mode 100644 index 00000000..65a2cc7e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_default.imageset/character_junior_developer_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_smile.imageset/Contents.json new file mode 100644 index 00000000..43198a1c --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_junior_developer_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_smile.imageset/character_junior_developer_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_smile.imageset/character_junior_developer_smile.png new file mode 100644 index 00000000..30710aa8 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_junior_developer_smile.imageset/character_junior_developer_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_close.imageset/Contents.json new file mode 100644 index 00000000..c0b33307 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_laptop_owner_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_close.imageset/character_laptop_owner_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_close.imageset/character_laptop_owner_close.png new file mode 100644 index 00000000..989275bd Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_close.imageset/character_laptop_owner_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_default.imageset/Contents.json new file mode 100644 index 00000000..e7dffb5a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_laptop_owner_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_default.imageset/character_laptop_owner_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_default.imageset/character_laptop_owner_default.png new file mode 100644 index 00000000..d1bc1716 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_default.imageset/character_laptop_owner_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_smile.imageset/Contents.json new file mode 100644 index 00000000..8f3e0922 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_laptop_owner_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_smile.imageset/character_laptop_owner_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_smile.imageset/character_laptop_owner_smile.png new file mode 100644 index 00000000..8ecd875a Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_laptop_owner_smile.imageset/character_laptop_owner_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_close.imageset/Contents.json new file mode 100644 index 00000000..3d37474a --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_night_owl_developer_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_close.imageset/character_night_owl_developer_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_close.imageset/character_night_owl_developer_close.png new file mode 100644 index 00000000..9b18b468 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_close.imageset/character_night_owl_developer_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_default.imageset/Contents.json new file mode 100644 index 00000000..914c051b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_night_owl_developer_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_default.imageset/character_night_owl_developer_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_default.imageset/character_night_owl_developer_default.png new file mode 100644 index 00000000..daeca1a2 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_default.imageset/character_night_owl_developer_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_smile.imageset/Contents.json new file mode 100644 index 00000000..0776abd0 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_night_owl_developer_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_smile.imageset/character_night_owl_developer_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_smile.imageset/character_night_owl_developer_smile.png new file mode 100644 index 00000000..ca12101d Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_night_owl_developer_smile.imageset/character_night_owl_developer_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_close.imageset/Contents.json new file mode 100644 index 00000000..38a04268 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_normal_developer_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_close.imageset/character_normal_developer_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_close.imageset/character_normal_developer_close.png new file mode 100644 index 00000000..336ba4d7 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_close.imageset/character_normal_developer_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_default.imageset/Contents.json new file mode 100644 index 00000000..d5175cf3 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_normal_developer_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_default.imageset/character_normal_developer_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_default.imageset/character_normal_developer_default.png new file mode 100644 index 00000000..be2fa743 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_default.imageset/character_normal_developer_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_smile.imageset/Contents.json new file mode 100644 index 00000000..ce827ead --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_normal_developer_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_smile.imageset/character_normal_developer_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_smile.imageset/character_normal_developer_smile.png new file mode 100644 index 00000000..8b0e942f Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_normal_developer_smile.imageset/character_normal_developer_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_close.imageset/Contents.json new file mode 100644 index 00000000..f5beabe5 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_skilled_developer_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_close.imageset/character_skilled_developer_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_close.imageset/character_skilled_developer_close.png new file mode 100644 index 00000000..63817316 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_close.imageset/character_skilled_developer_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_default.imageset/Contents.json new file mode 100644 index 00000000..f6bb9afb --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_skilled_developer_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_default.imageset/character_skilled_developer_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_default.imageset/character_skilled_developer_default.png new file mode 100644 index 00000000..fbe75b2c Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_default.imageset/character_skilled_developer_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_smile.imageset/Contents.json new file mode 100644 index 00000000..973093de --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_skilled_developer_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_smile.imageset/character_skilled_developer_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_smile.imageset/character_skilled_developer_smile.png new file mode 100644 index 00000000..aed04068 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_skilled_developer_smile.imageset/character_skilled_developer_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_close.imageset/Contents.json new file mode 100644 index 00000000..48d39f82 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_unemployed_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_close.imageset/character_unemployed_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_close.imageset/character_unemployed_close.png new file mode 100644 index 00000000..7e3ddc31 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_close.imageset/character_unemployed_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_default.imageset/Contents.json new file mode 100644 index 00000000..dba4843d --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_unemployed_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_default.imageset/character_unemployed_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_default.imageset/character_unemployed_default.png new file mode 100644 index 00000000..26264c69 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_default.imageset/character_unemployed_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_smile.imageset/Contents.json new file mode 100644 index 00000000..ab3a1170 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_unemployed_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_smile.imageset/character_unemployed_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_smile.imageset/character_unemployed_smile.png new file mode 100644 index 00000000..638350e1 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_unemployed_smile.imageset/character_unemployed_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_close.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_close.imageset/Contents.json new file mode 100644 index 00000000..79e261c6 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_close.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_world_class_developer_close.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_close.imageset/character_world_class_developer_close.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_close.imageset/character_world_class_developer_close.png new file mode 100644 index 00000000..3b7e9728 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_close.imageset/character_world_class_developer_close.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_default.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_default.imageset/Contents.json new file mode 100644 index 00000000..b55d7e0f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_default.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_world_class_developer_default.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_default.imageset/character_world_class_developer_default.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_default.imageset/character_world_class_developer_default.png new file mode 100644 index 00000000..18ed6d28 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_default.imageset/character_world_class_developer_default.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_smile.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_smile.imageset/Contents.json new file mode 100644 index 00000000..00b6b2ee --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_smile.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "character_world_class_developer_smile.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_smile.imageset/character_world_class_developer_smile.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_smile.imageset/character_world_class_developer_smile.png new file mode 100644 index 00000000..36ac14ba Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/character/character_world_class_developer_smile.imageset/character_world_class_developer_smile.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_dodge.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_dodge.imageset/Contents.json new file mode 100644 index 00000000..99c5fbcc --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_dodge.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "work_dodge.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_dodge.imageset/work_dodge.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_dodge.imageset/work_dodge.png new file mode 100644 index 00000000..09f7a66f Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_dodge.imageset/work_dodge.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_language.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_language.imageset/Contents.json new file mode 100644 index 00000000..ac14400b --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_language.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "work_language.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_language.imageset/work_language.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_language.imageset/work_language.png new file mode 100644 index 00000000..9a72a04e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_language.imageset/work_language.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_stack.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_stack.imageset/Contents.json new file mode 100644 index 00000000..5ca176b4 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_stack.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "work_stack.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_stack.imageset/work_stack.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_stack.imageset/work_stack.png new file mode 100644 index 00000000..4e922911 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_stack.imageset/work_stack.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_tap.imageset/Contents.json b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_tap.imageset/Contents.json new file mode 100644 index 00000000..af88405e --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_tap.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "work_tap.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_tap.imageset/work_tap.png b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_tap.imageset/work_tap.png new file mode 100644 index 00000000..2c2b568c Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Assets.xcassets/work/work_tap.imageset/work_tap.png differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bgm.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bgm.wav new file mode 100644 index 00000000..ed74e0ab Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bgm.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/blockDrop.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/blockDrop.wav new file mode 100644 index 00000000..8351f1de Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/blockDrop.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/blockStack.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/blockStack.wav new file mode 100644 index 00000000..31c7c2ec Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/blockStack.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bombStack.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bombStack.wav new file mode 100644 index 00000000..4cf958c3 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bombStack.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bugHit.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bugHit.wav new file mode 100644 index 00000000..7a1eebbe Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/bugHit.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/buttonTap.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/buttonTap.wav new file mode 100644 index 00000000..03532701 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/buttonTap.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/coinCollect.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/coinCollect.wav new file mode 100644 index 00000000..aae8bbeb Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/coinCollect.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/itemConsume.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/itemConsume.wav new file mode 100644 index 00000000..b93d194e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/itemConsume.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/languageCorrect.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/languageCorrect.wav new file mode 100644 index 00000000..f8a0b826 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/languageCorrect.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/languageWrong.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/languageWrong.wav new file mode 100644 index 00000000..166fe4a1 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/languageWrong.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/missionAcquired.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/missionAcquired.wav new file mode 100644 index 00000000..0a8f2375 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/missionAcquired.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizCorrect.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizCorrect.wav new file mode 100644 index 00000000..afefb6a5 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizCorrect.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizCountdown.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizCountdown.wav new file mode 100644 index 00000000..df84a5e3 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizCountdown.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizTimeOver.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizTimeOver.wav new file mode 100644 index 00000000..fa0791bf Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizTimeOver.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizWrong.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizWrong.wav new file mode 100644 index 00000000..cb1cfe07 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/quizWrong.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/tapGameTyping.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/tapGameTyping.wav new file mode 100644 index 00000000..d6807bd7 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/tapGameTyping.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/upgradeFailure.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/upgradeFailure.wav new file mode 100644 index 00000000..e791409e Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/upgradeFailure.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/upgradeSuccess.wav b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/upgradeSuccess.wav new file mode 100644 index 00000000..0d27eb30 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Audio/upgradeSuccess.wav differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-Bold.ttf b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-Bold.ttf new file mode 100644 index 00000000..808da50b Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-Bold.ttf differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-ExtraBold.ttf b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-ExtraBold.ttf new file mode 100644 index 00000000..e4e3f924 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-ExtraBold.ttf differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-Regular.ttf b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-Regular.ttf new file mode 100644 index 00000000..a1312352 Binary files /dev/null and b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/Fonts/PFStardust-Regular.ttf differ diff --git a/SoloDeveloperTraining/SoloDeveloperTraining/Resources/QuizData.tsv b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/QuizData.tsv new file mode 100644 index 00000000..fe3dd773 --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTraining/Resources/QuizData.tsv @@ -0,0 +1,137 @@ +문제 번호 문제명 선지1 선지2 선지3 선지4 정답 해설 +0 오늘은 중요한 기능을 배포하는 날이다. 다음 중 개발자가 가장 들으면 안되는 말은? (함께 코드를 보며) 이 부분 빨리 수정 가능할까요? (깜깜한 새벽, 혼자 모니터링을 하며) 어? 음...어떡하죠? (퇴근 후 슬랙 멘션이 울린다) @님 지금 어디세요? 4 이 일의 담당자인 개발자가 퇴근 후 업무 슬랙으로 호출당했다는 것만큼 급한 상황은 없다. +1 개발자가 갑자기 모니터를 껐다 켰다 한다. 가장 그럴듯한 이유는? 전기를 아끼려는 환경운동 새로 산 모니터를 구경하려고 이러면 왠지 될 것 같아서 눈이 피로해서 3 개발자 밈 국룰: “껐다 켜봤어?” +2 개발자가 “이건 진짜 금방 끝나요”라고 말했다. 실제로 걸리는 시간은? 30분 오늘 퇴근 전에 이번 주 금요일 퇴근 전에 아무도 모른다. 4 “금방”은 개발자 세계에서 상대적인 시간 단위다. +3 개발자를 괴롭히던 에러가 갑자기 사라졌다. 개발자의 반응은? 문제를 정확히 분석한다. 재현 테스트를 작성한다. 상사에게 정리해 상세히 보고한다. 다음 작업을 진행한다. 4 “됐으면 됐다…” +4 “이건 진짜 간단해요”라는 말을 믿으면 안 되는 이유는? 문제의 크기를 아직 몰라서 담당자의 실력이 없어서 설명하기 귀찮아서 농담을 좋아해서 1 간단한 줄 알았던 일이 대형 프로젝트로 진화하는 순간 +5 개발자가 커피를 마시는 이유는? 맛있어서 밤샘으로 지친 컨디션을 회복하기 위해 분위기 때문에 코드가 돌아가길 기원하며 4 커피 == 부적 +6 “어제까지 잘 됐는데요?” 이 말을 들은 사람의 직업은? 마술사 의사 개발자 에어컨 수리 기사 3 개발자 세계 3대 미스터리 중 하나 +7 아무것도 안 바꿨는데 문제가 생겼다. 이때 개발자의 머릿속은? 시스템 오류다. 두 달 전의 내가 저질렀다. 동료의 잘못이다. AI 잘못이다. 2 과거의 내가 쓴 코드는 나도 모른다. +8 개발자가 에러를 보고 가장 먼저 하는 행동은? 에러 메시지를 그대로 복사한다. 매뉴얼을 찾아본다. 코드를 처음부터 다시 짠다. 동료에게 도움을 구한다. 1 구글은 개발자의 두 번째 뇌. 첫 번째는 기억 안 남 +9 개발자가 코드에 주석을 안 남긴 이유는? 천재라서 바빠서 미래의 누군가 고생하길 바라서 귀찮아서 3 “이 코드 누가 짰어” -> 과거의 나 +10 개발자가 가장 신뢰하는 테스트 방법은? 자동 테스트 QA 팀 검증 시뮬레이터 테스트 내 컴퓨터에서 되면 OK 4 “내 컴퓨터에선 되는데요?” 개발자 방어 주문 1단계 +11 개발자가 가장 공포를 느끼는 색깔은? 검은색 초록색 빨간색 파란색 3 화면 가득 오류가 뜨는 빨간 줄은 개발자의 심박수를 급격히 올립니다. +12 개발자가 퇴근하려는데 상사가 "이거 간단한 건데"라고 잡았다. 이때 개발자의 속마음은? 아싸, 빨리 끝내고 가야지 진짜 간단하겠지? 오늘 퇴근은 글렀구나 ~ 상사가 날 아끼는구나 3 개발자 사전에 "간단한건데" == "구조를 다 뜯어고쳐야 하는 복잡한 일" +13 숫자를 셀 때 개발자는 몇 부터 시작할까? 0 1 2 3 1 배열의 인덱스는 0부터 시작하는 것이 국룰 +14 개발자가 코드를 짤 때 가장 많이 하는 행동은? 수학 공식 계산하기 타자 연습 복사 붙여넣기 모니터 청소하기 3 우리의 코드는 AI가 작성하고 이제 복사 붙여넣기 해야죠 +15 신입 개발자가 들어왔을 때 사수가 가장 먼저 알려주는 것은? 회사의 비전 탕비실 위치 복잡한 서버 구조 코딩 스타일 가이드 2 카페인과 당 충전은 코드 품질과 직결되기 때문입니다. +16 경고 메시지를 본 개발자의 반응은? 심각하게 분석한다 즉시 고친다 못본 척 무시한다 동료에게 알린다 3 에러만 아니면 됩니다. 노란색 경고 따위는 가볍게 무시! +17 코드 리뷰에서 LGTM은 무슨 뜻일까요? Let's Go To Market (쇼핑 ㄱㄱ) Looks Good To Me (좋아 보이네요) Let's Get This Money (돈 벌자) Love Good Tears Money 2 문제가 없어보일 때 많이 사용되는 줄임말입니다 +18 다음 중 개발자가 "라이브러리"라고 부르는 것은? 책 읽는 도서관 미리 작성된 코드 모음집 회사 휴게실 사내 식당 2 남이 잘 짜준 코드를 가져와서 조립하는 것, 이게 바로 현대 개발 +19 AI가 코드를 짜주기 시작했다. 개발자의 반응은? 내 밥그릇 ㅠㅠ 오 개꿀 (복붙) 기획자로 전향한다 퇴사한다 2 AI는 훌륭한 부사수입니다 +20 개발자가 "이게 왜 돌아가지"라고 말하는 상황은? 코드가 완벽할 때 버그가 있어야 하는데 정상 동작할 때 컴퓨터가 회전할 때 의자가 돌아갈 때 2 내가 짠 논리대로라면 에러가 나야 하는데 잘 되면 더 불안합니다. +21 개발자가 “이건 설정 문제예요”라고 말할 때 가장 가까운 의미는? 옵션 하나만 바꾸면 된다 환경마다 다르게 터진다 코드가 잘못됐다 다시 설치하면 된다 2 내 컴퓨터에선 되는데 다른 환경에선 안 되는 그 문제. +22 개발자가 “이건 리팩터링이에요”라고 말하면 실제로는? 성능 개선 코드 정리 기능 변경 다 뜯어고침 4 리팩터링은 종종 이사 + 인테리어 + 철거다. +23 개발자가 가장 오래 붙잡고 있는 파일은? 최신 파일 가장 큰 파일 이름이 애매한 파일 테스트 파일 3 util, temp, final_final 같은 이름은 정체 불명이다. +24 개발자가 기능을 만들고 “이건 쓰지 마세요”라고 말하는 이유는? 미완성이라서 위험해서 언제 터질지 몰라서 본인도 이해 못 해서 4 만든 사람도 설명 못 하면 금지 구역이 된다. +25 개발자가 회의에서 가장 자주 쓰는 말은? 네 일단 해볼게요 가능은 한데요 다 맞습니다 3 “가능은 한데요” 뒤에는 조건·시간·위험이 숨어 있다. +26 개발자가 기능 설명을 하다 갑자기 말이 느려지는 순간은? 중요한 부분이라서 긴장해서 기억이 안 나서 아직 구현 안 해서 4 설명하면서 구현 안 한 게 떠오르는 순간이다. +27 개발자가 파일 이름에 숫자를 붙이기 시작했다면? 버전 관리를 잘한다 실험 중이다 이전 버전이 망했다 정리가 끝났다 3 final2, final3은 희생의 흔적이다. +28 개발자가 코드를 보며 가장 먼저 확인하는 것은? 디자인 주석 함수 이름 누가 썼는지 4 작성자를 알면 마음의 준비를 할 수 있다. +29 개발자가 “이건 임시로 넣어둔 거예요”라고 말하면? 곧 삭제된다 테스트용이다 중요하지 않다 계속 사용된다 4 임시는 개발 세계에서 가장 영구적인 상태다. +30 개발자가 가장 안심하는 순간은 언제일까? 코드가 컴파일될 때 테스트가 통과할 때 에러가 안 날 때 하루 동안 아무 연락이 없을 때 4 진짜 평화는 아무도 연락 안 오는 날이다. +31 USB를 한 번에 꽂을 확률은? 50% 100% 0% (반드시 3번 돌려야 들어감) 99% 3 앞면으로 꽂음(안 들어감) -> 뒷면으로 돌림(안 들어감) -> 다시 앞면(들어감). 과학으로 설명할 수 없는 현상입니다. +32 개발자가 스트레스를 풀기 위해 장만하는 대표적인 '장비빨' 아이템은? 기계식 키보드 5천원짜리 마우스 노트와 펜 계산기 1 타건음(딸깍딸깍)은 개발자의 심신 안정에 도움을 줍니다. +33 개발자가 가장 어려워하고 머리 쥐어짜는 난제는? 복잡한 수학 계산 1000줄 코딩하기 변수 이름 짓기 야식 메뉴 고르기 3 data, temp, a... 짓다가 하루가 다 갑니다. +34 명절에 친척들이 개발자 조카에게 가장 많이 부탁하는 것은? 최신 앱 개발 인공지능 알고리즘 설명 프린터 연결 및 컴퓨터 수리 서버 구축 3 "컴퓨터 전공이니까 컴퓨터 잘 고치지?"는 개발자의 숙명입니다. +35 프로그래밍 언어를 배울 때 가장 먼저 출력해보는 문구는? I am God Hello World Show me the money Error 404 2 개발자의 탄생을 알리는 국룰 인사법입니다. +36 전 세계 개발자들의 코딩 선생님이자 구세주인 사이트는? 싸이월드 스택 오버플로우 (Stack Overflow) 배달의 민족 넷플릭스 2 개발의 절반은 구글링과 복사+붙여넣기(Ctrl+C, V)입니다. +37 개발자들의 눈 건강을 지켜주는, 간지의 상징인 화면 모드는? 다크 모드 (Dark Mode) 화이트 모드 핑크 모드 투명 모드 1 흰 배경에 검은 글씨는 개발자의 눈을 공격합니다. +38 코드가 안 풀릴 때, 책상 위 인형에게 코드를 설명하며 답을 찾는 방법을 무엇이라 할까요? 독백 디버깅 러버덕(고무오리) 디버깅 인형 놀이 혼잣말 대잔치 2 남에게 설명하다 보면 스스로 답을 찾게 되는 마법입니다. +39 개발자가 버그를 발견했을 때 우겨보는 말은? "이건 버그가 아니라 OOO입니다." 바이러스 의도 기능 (Feature) 실수 3 "It's not a bug, it's a feature." (버그가 아니라 의도된 기능입니다.) +40 개발자가 가장 싫어하는 문서는? 월급명세서 휴가 신청서 변경된 기획서 합격 통지서 3 개발 다 끝났는데 기획이 바뀌는 순간... (절망) +41 웹사이트를 찾을 수 없을 때 발생하는 '404 Not Found' 에러는 현실 세계의 '이것'에서 유래했다는 설이 있습니다. (사실 여부는 불분명하지만 유명한 이야기입니다) 서버실의 방 번호 (Room 404) 404번 버스 4월 4일 개발자의 몸무게 1 CERN의 404호 방에 데이터베이스가 있었다는 설이 있지만, 실제로는 4층 자체가 없었다는 반박도 있습니다. 그럼에도 가장 유명한 전설입니다. +42 개발자가 가장 싫어하는, 얽히고설킨 난해한 코드를 음식에 비유하면? 피자 코드 스파게티 코드 햄버거 코드 김밥 코드 2 "이 변수가 어디서 바뀌었지?"를 찾으려면 스파게티 면발(로직)을 하나하나 다 따라가야 합니다. +43 프로그래밍 언어 커뮤니티에서 "이것은 프로그래밍 언어입니까?"라는 논쟁으로 개발자를 가장 화나게 할 수 있는 것은? C++ Python HTML Swift 3 컴퓨터 공학적 관점에서 엄밀히 따지면 HTML은 '마크업 언어'이지 '프로그래밍 언어'가 아닙니다. +44 키보드에서 개발자의 지문이 가장 많이 묻어있는 세 개의 키 조합은? Ctrl, Alt, Del Ctrl, C, V W, A, S, D F1, F2, F3 2 실제로 개발자들을 위한 선물로 키가 딱 3개(Ctrl, C, V)만 있는 미니 키보드가 판매되기도 할 정도로 유명한 자학 개그 +45 개발자가 에러를 고치고 가장 먼저 하는 생각은? 왜 이런 실수를 했지 다시는 이런 일 없게 하자 기록으로 남겨야지 다른 데서 터지진 않겠지 4 하나 고치면 어딘가 하나는 흔들린다. +46 개발자가 갑자기 한숨을 쉰 이유는? 피곤해서 방금 뭔가 잘못 눌렀다는 것을 깨달아서 회의 생각나서 퇴근하고 싶어서 2 클릭 한 번에 하루를 날린다. +47 개발자가 가장 싫어하는 성격 유형은? 급한 사람 잘난 체하는 사람 변덕쟁이 답답한 사람 3 요구 사항이 계속 바뀜 == 지옥 +48 개발자가 코드에서 TODO 주석을 발견했다. 가장 가능성 높은 상황은? 곧바로 처리한다 이슈를 생성한다 해당 줄을 조심히 피해서 지나간다 주석을 지운다 3 TODO는 미래의 나에게 떠넘기는 책임이다. +49 “이거 왜 이렇게 짰어요?”라는 질문을 들은 개발자의 반응은? 즉시 리팩터링을 약속한다 조용히 커밋 히스토리를 본다 설명을 시작하다가 말이 느려진다 웃으면서 넘긴다 3 말이 느려질수록 본인도 확신이 없다. +50 장애 공지 문구를 작성할 때 개발자가 가장 고민하는 것은? 정확한 원인 재발 방지 대책 언제까지 복구될지 누가 썼는지 안 들키는 표현 4 “시스템 이슈”라는 말에 수많은 감정이 담겨 있다. +51 개발자가 배포 직전에 가장 자주 하는 말은? 이건 안전해요 테스트 다 했어요 설마 문제 있겠어? 기도하자 4 배포는 과학이 아니라 신앙의 영역입니다. 테스트를 다 해도 기도는 필수 +52 개발자가 가장 무서워하는 문장은? 이 기능 언제 돼요? 간단한 수정이에요 급한 건 아닌데요 기획이 조금 바뀌었어요 4 조금 바뀐 기획은 보통 전체 구조 변경을 의미한다 +53 개발자가 가장 자주 미루는 작업은? 핵심 기능 구현 코드 리뷰 문서 작성 배포 3 문서는 항상 나중에 쓴다. 그 나중은 거의 오지 않는다. +54 개발자가 가장 믿지 않는 말은? 공식 문서 에러 메시지 QA 결과 금방 끝나요 4 이 말은 시공간을 왜곡한다. +55 개발자에게 "여자(남자)친구 있어요?"라고 물었을 때 가장 개발자스러운 답변은? "있는데 안 만나요." "404 Not Found" "Null Pointer Exception" "지금 빌드 중입니다." 2 요청하신 리소스를 찾을 수 없습니다. +56 다음 중 가장 믿을 수 없는 '시간 단위'는? 전자레인지 30초 컵라면 3분 군대의 국방부 시계 개발자의 "5분이면 수정해요." 4 개발자의 '5분'은 지구의 시간이 아닐 수 있습니다. (인터스텔라급 왜곡 발생) +57 3시간 동안 버그를 찾지 못해 헤매던 중, 알고 보니 원인이 '세미콜론(;)' 하나 빠진 것이었을 때 느끼는 감정은? "아, 다행이다. 해결했네." "다음부터는 조심해야지." "이 멍청한 컴퓨터는 융통성이라곤 없구나." (허탈한 웃음과 함께) 키보드를 반으로 접고 싶은 충동 4 고작 점 하나 때문에 내 3시간이 증발했다는 사실을 받아들이는 데는 시간이 필요합니다. +58 동료의 코드를 리뷰(Code Review)할 때, 코드가 너무 길고 복잡해서 읽기 귀찮을 때 남기는 마법의 약어는? WIP (Work In Progress) ASAP (As Soon As Possible) LGTM (Looks Good To Me) FYI (For Your Information) 3 "제 눈엔 좋아 보이네요"라는 뜻이지만, 사실 "에라 모르겠다 일단 합치자(Merge)"라는 뜻일 확률이 높습니다. +59 개발자가 "이번엔 제대로 짰어요"라고 자신있게 말한 후 가장 먼저 일어나는 일은? 칭찬받는다 승진한다 프로덕션에서 500 에러가 터진다 동료들이 박수친다 3 자신감 넘치는 배포 후엔 항상 뭔가 터진다. 머피의 법칙은 개발자를 비켜가지 않는다. +60 개발자가 "이 코드 누가 짰어?"라고 분노하며 Git Blame을 돌렸을 때 가장 자주 나오는 결과는? 퇴사한 전임자 인턴 6개월 전의 나 AI 3 과거의 나는 남이다. 그것도 원수 같은 남. +61 개발자가 주말에 가장 하고 싶지 않은 알림은? "배포 롤백했습니다" "서버 다운됐어요" "긴급 회의 소집" "아 그거 테스트 안 해봤는데..." 2 일요일 저녁 서버 장애 알림은 개발자의 영혼을 탈탈 털어갑니다. 주말은 이미 끝났다. +62 개발자가 가장 믿는 기록 방식은? 메모 녹음 로그 기억 3 기억은 지워져도 로그는 남는다. +63 개발자가 가장 좋아하는 정리 상태는? 깔끔함 미니멀 정돈됨 정렬됨 4 순서만 맞아도 마음이 편안 +64 다음 중 버그가 가장 생기기 쉬운 상황은? 코드가 짧을때 기능이 단순할때 테스트가 많을때 조건문이 많을때 4 if가 늘어날수록 경우의 수는 폭발한다. +65 다음 중 “잘 돌아가는 코드”의 기준으로 가장 중요한 것은? 멋있는 코드 이해 가능한 코드 짧은 코드 빠른 코드 2 나중에 고칠 사람은 거의 항상 "나"다. +66 리팩토링(Refactoring) 이라는 말의 느낌과 가장 가까운 것은? 성능을 폭발시킨다 기능을 추가한다 모양을 다시 다듬는다 완전히 새로 만든다 3 Re(다시) + Factor(구조, 요소) +67 디버그(Debug) 라는 말의 어원에 가장 가까운 설명은? 벌레를 잡는다 벌레랑 싸운다 벌레를 먹는다 벌레를 살려준다 1 Debug = De(제거) + Bug(벌레) +68 다음 중 print("여기")가 가장 자주 사용되는 목적은? 성능 측정 기도 로그 수집 디버깅 1 사실상 “여기까지는 실행됐겠지?”라는 존재 확인 의식. 논리적 디버깅이라기보다 “제발 여기까지만이라도 와줘…” 라는 마음가짐이 핵심이다. +69 다음 중 컴공생의 시간 감각을 가장 왜곡시키는 것은? 알고리즘 문제 리팩터링 환경설정 디버깅 3 “이거 5분이면 돼~” → 3시간. +70 컴공생이 가장 많이 하는 거짓말은? 금방 끝나요 이해했습니다 테스트 했어요 주석에 써놨어요 4 주석은 있다. 다만 오래됨/틀림/코드와 불일치/혹은 아무 말도 안 함 그래도 그래도 “써놨다”고 말하게 된다. +71 개발자의 하루 일과 중 가장 많은 시간을 차지하는 것은? 코딩 회의 구글링 왜 안되는지 고민 4 코드 짜는 시간보다 "왜 안 돼?"를 외치는 시간이 3배 😭 +72 "코드 리팩토링 하겠습니다"의 진짜 의미는? 코드를 개선하겠다 전부 다시 짜겠다 이해가 안 된다 시간이 남았다 2 리팩토링 = 갈아엎기 🔥 결국 새로 짜는 거죠 +73 테스트 코드를 작성하는 시점은? 개발 전 개발 중 개발 후 배포 후 버그 터졌을 때 4 "이제 테스트 코드 짜야지" (다음 생에) +74 이거 누가 짠 거야?"라고 물었을 때의 답은? 제가 짰어요 전임자요 모르겠는데요 git blame 돌려보세요 4 전임자에게 모든 걸 떠넘기기 🏃 (본인이 짠 거라도) +75 프로그래머가 운동을 시작하는 이유는? 건강을 위해 스트레스 해소 허리가 아파서 아직 안 시작함 4 "내일부터 할 거야..." (이미 3년째) 🏋️ +76 주말에 갑자기 서버가 다운되면? 즉시 복구한다 월요일까지 기다린다 전화를 못 본 척한다 "연락 못 받았어요" 3 주말엔 핸드폰이 고장 나는 법 📱🔇 +77 개발자의 변명 중 가장 많이 쓰는 것은? 시간이 부족했어요 요구사항이 애매했어요 테스트 환경이 달랐어요 위의 모든 것 4 변명의 성삼위일체 ✨ 상황에 따라 픽 +78 "이건 나중에 정리할게요"라는 말의 실제 의미는? 이번 주 안에 처리한다 다음 주 안에 처리한다 시간 날 때 처리한다 영원히 처리하지 않는다 4 나중에 = 미래의 나에게 떠넘기기. 그 미래는 오지 않는다 +79 개발자가 갑자기 불안해지는 순간은? 에러가 발생했을 때 테스트가 실패했을 때 갑자기 모든 게 잘 될 때 코드 리뷰 요청이 왔을 때 3 에러가 나야 정상인데 너무 잘 돌아가면 오히려 무섭다 +80 컴퓨터 화면에 빨간 글씨가 잔뜩 뜰 때 첫 번째 행동은? 침착하게 읽어본다 하나씩 해결해본다 모니터를 멀리한다 아무 일 없었던 척한다 4 본능적으로 거리부터 확보한다. +81 어제까지 잘 되던 게 오늘 갑자기 안 될 때 가장 많이 나오는 말은? "내가 잘못했나?" "컴퓨터가 이상해" "누가 건드렸어?" "원래 이런 거야" 2 사람은 자신의 실수보다는 기계를 먼저 의심한다 +82 갑자기 컴퓨터를 껐다 켜는 사람의 이유는? 전기를 아끼려고 고장이 난 것 같아서 왠지 이러면 될 것 같아서 화면이 지겨워서 3 논리보다 믿음이 앞서는 순간입니다. +83 퇴근 후 단체 메신저에서 이름이 불렸을 때의 기분은? 반갑다 별일 아니겠지 심장이 철렁한다 행복하다 3 퇴근 후 호출은 대부분 긴급 상황이다. +84 잘 안 되던 일이 갑자기 잘 되기 시작했을 때 반응은? 이유를 분석한다 다시 한 번 확인한다 주변에 자랑한다 넘어간다 4 "됐으면 됐다" 더 건드리면 다시 망가질 수 있다. +85 가장 효과적인 오류 해결 방법은? 원하는 대답이 나올 때까지 AI를 괴롭힌다 지우고 처음부터 다시 한다 비슷한 사례를 분석한다 동료에게 토스한다 4 기존 코드의 정신적 부채를 전부 날려버리는 전략. “왜 안 되지?” → “아 처음부터 이렇게 하면 되네” +86 어렵기로 소문난 전공 수업을 들은 후 가장 먼저 드는 생각은? 컴퓨터가 더 친숙해졌다 세상이 논리적으로 보인다 이게 왜 돌아가는지 모르겠다 내가 똑똑해진 것 같다 3 이해는 못 했지만 컴퓨터는 여전히 잘 돌아간다. → 그래서 더 무섭다. +87 “이 코드는 이해되면 안 된다”의 의미는? 코드가 너무 난해하다 작성자가 없다 이미 너무 오래 돌아가고 있다 관련 문서가 없다 3 이해하려는 순간 책임이 생긴다. +88 개발자가 가장 행복해 보이는 순간은? 배포 직후 에러가 사라졌을 때 회의가 취소됐을 때 아무 알림도 없을 때 4 진짜 평화는 "아무 일도 안 일어나는 날" +89 개발자가 회의에서 고개를 끄덕일 때 실제 의미는? 완전히 이해했다 일단 해결 방법이 있다 일단 듣고는 있다 동의한다 3 끄덕이지만 뇌는 아직 로딩 중이다. +90 개발자가 문제를 설명하다가 갑자기 말이 빨라지는 이유는? 자신 있어서 시간이 없어서 이미 여러 번 겪어서 대충 넘기고 싶어서 4 말하면 더 큰 얘기가 나온다. +91 개발자가 가장 무서워하는 테스트 결과는? 실패 경고 성공 아무 결과도 안 나옴 4 아무 반응이 없다는 건 뭔가 더 큰 문제가 숨어 있다는 뜻이다. +92 다음 중 개발자가 "버그"에 대해 느끼는 가장 가까운 의미는? 없을 수록 좋다 없으면 좋지만 있어도 괜찮다 있어도 안되고 없어도 안된다 무조건 없어야 한다 3 있다? 고쳐야 됨. 없다? 언제 문제가 터질지 모름. +93 컴공생이 말하는 “시간 복잡도”의 실제 의미는? 알고리즘 효율 실행 시간 입력 크기 이거 커지면 망한다 4 아슬아슬한 시간 복잡도를 보면 불안감이 먼저 온다. +94 ‘추상화(abstraction)’가 필요한 가장 솔직한 이유는? 유연성을 위해 확장성을 위해 재사용성을 위해 다 생각하기 싫어서 4 인간의 뇌는 디테일을 버텨내지 못한다. +95 개발자가 “일단 이렇게 가죠”라고 말하는 순간은? 합리적인 판단 최선의 선택 안정적인 방향 지쳤다 4 “일단”은 에너지 고갈 신호다. +96 개발자가 “이 로직은 건드리면 안 돼요”라고 한다. 이유는? 최적의 로직이라서 자기도 몰라서 모두 검증되어서 권한이 없어서 2 이 코드는 지뢰다. +97 개발자가 갑자기 조용해졌을 때 가장 가능성 높은 상황은? 에러 메시지 읽는 중 어려운 설계 중 코드에 몰입 중 자는 중 1 에러 메시지는 말이 많아서 읽느라 조용해진다. +98 개발자가 “이건 구조상 어쩔 수 없어요”라고 말하는 순간 가장 가까운 진실은? 최적의 설계다 이미 너무 깊이 들어왔다 회사 정책 때문이다 처음부터 잘못됐다 2 지금 고치면 다 무너진다. 이미 돌아가고 있으니 신성불가침이 됐다. +99 개발자가 버그를 재현하지 못했을 때 가장 자주 드는 생각은? 사용자 환경 문제다 로그를 더 심어야겠다 다시 시도해보자 내가 미쳤나? 4 방금 전까진 분명 터졌는데… 나만 봤나? +100 개발자가 “이거 리스크는 있어요”라고 말했는데도 일이 진행됐다. 가장 높은 확률로 일어나는 일은? 아무 문제 없이 끝난다 예상한 문제가 그대로 터진다 전혀 다른 문제가 터진다 아무도 책임지지 않는다 3 예상한 문제는 빙산의 일각. 진짜는 수면 아래 있다. +101 개발자가 가장 자주 까먹는 것은? 변수명 함수명 Git 커밋 자신이 한 일 4 자신이 짠 코드는 3일만 지나면 낮설다. +102 개발자가 자주 사용하는 채팅 앱은? Slack Discord Telegram KakaoTalk 1 Slack = 회사생활 +103 개발자가 새로 배운 기술을 실제 프로젝트에 적용하고 싶어하는 시점은? 바로 적용 테스트 후 적용 신중하게 고민 후 적용 새 프로젝트가 생기면 4 기존 프로젝트에 신기술 = 리스크, 새 프로젝트가 안전 +104 개발자가 코드를 짜다가 "이건 너무 복잡한데 일단 동작하게 하자"라고 생각하는 순간은? 초기 설계 단계 중간 개발 단계 마지막 단계 항상 4 모든 코드는 "일단 동작하게"에서 시작한다 +105 개발자가 문서를 읽기 전에 먼저 하는 일은? 예제 코드 찾기 직접 실행해보기 스택오버플로우 검색 포기하기 2 문서보다 직접 해보는게 확실하다. +106 개발자가 코드를 완성한 후 느끼는 감정은? 성취감 자신감 의심 뭔가 놓친게 있다 4 100%는 거짓말이다. 항상 놓치는 게 있다 +107 개발자가 데드라인 3일 전에 느끼는 감정은? 여유롭다 걱정된다 당황한다 이미 늦었다 1 아직 3일이나 남았네? +108 개발자가 동료의 코드를 보면서 가장 자주 하는 생각은? 이거 개선하면 좋을 것 같아요 나쁘지 않네요 왜 이렇게 했을까? 고치고 싶지만 말하면 귀찮아질 것 같다 4 남의 코드를 만지면 일이 더 늘어난다. +109 개발자가 에러 핸들링을 제대로 안 했을 때의 변명은? 나중에 추가할 예정이에요 에러가 안 날 것 같아서 try-catch만 할면 될 것 같아서 에러? 그게 뭐에요? 4 에러 핸들링은 미래의 나에게 맡긴다 +110 개발자가 새로운 IDE나 에디터를 배울 때 하는 것은? 단축키 확인 플러그인 추천받기 튜토리얼 다시 원래로 들어가기 4 새 에디터는 며칠 쓰다가 기존 에디터로 복귀 +111 개발자가 새로 온 동료에게 프로젝트를 설명할 때 가장 어려운 부분은? 전체 구조 왜 이렇게 설계했는지 각 모듈의 역할 전체 4 설명하다 보면 내가 왜 이렇게 짰는지도 모르겠다 +112 개발자가 "이건 표준을 따라 작성했어요"라고 말하는 실제 의미는? 업계 표준 회사 표준 내가 생각하는 표준 표준이 뭐에요? 3 표준은 주관적이다. +113 개발자가 코드를 작성하다가 갑자기 "이건 너무 과한 것 같은데?"라고 생각할 때의 반응은? 간소화한다 일단 둔다 다시 작성한다 과한게 뭐 어때서 4 과한것도 동작하면 OK! +114 개발자가 프로젝트 마감일이 다가올 때 가장 많이 삭제하는 것은? 불필요한 기능 주석 테스트 코드 모두 3 마감일 = 테스트 코드의 생명줄이 끊기는 날 +115 개발자가 메모리 누수를 발견했을 때 가장 먼저 하는 행동은? 즉시 수정한다 로그를 남긴다 무시한다 나중에 수정한다 3 메모리? 그건 가상의 개념이야 (무시) +116 개발자가 갑자기 키보드를 빠르게 치기 시작했다. 가장 그럴듯한 이유는? 타자 연습 중이다. 중요한 아이디어가 떠올랐다. 코드가 안 돌아간다. 카톡 중이다. 4 빠르게 키보드를 치는 것은 메신저를 하고 있을 때 뿐이다. +117 개발자가 가장 자주 저장하는 파일 이름은? final final_real final_final final_final_진짜마지막 4 숫자와 수식어가 늘어날수록 프로젝트의 고통도 함께 올라간다. +118 개발자가 문제를 설명하다가 갑자기 "어.. 잠깐만요"라고 말한다. 무슨 상황일까? 설명이 너무 길어졌다. 질문을 이해 못했다. 말하다 보니 문제를 발견했다. 배가 고프다. 3 남에게 설명하다 보면 본인이 놓친 버그를 스스로 발견하는 경우가 많다. +119 개발자가 가장 안심하는 문장은? 고생 많으셨어요 이거 언제 돼요? 아무 일 없었어요 기획이 조금 바뀌었어요 3 진짜 평화는 칭찬이 아니라 아무 소식도 없는 상태이다. +120 개발자가 "일단 이렇게 두죠"라고 말하는 이유는? 최고의 선택이라서 시간이 부족해서 더 생각하기 싫어서 이미 지쳤기 때문에 4 "일단"은 체력과 멘탈이 동시에 바닥났다는 신호이다. +121 개발자가 가장 진지해지는 순간은? 일정 조율 네이밍 새로운 기능 제안 디자인 논의 2 이름은 한 번 정하면 평생 간다. +122 “이거 구현은 쉬운데…”라는 말 뒤에 가장 자주 오는 것은? 조건 설명 20분 테스트 코드 작성 금방 끝남 1 구현은 쉽지만 조건이 지옥이다. +123 개발자가 문서를 안 믿게되는 순간은? 문서가 길다 문서가 영어다 문서대로 안 된다 문서가 없다 3 한 번 배신하면 다신 안 믿는다. +124 “이건 나중에 정리할게요”가 의미하는 일정은? 오늘 이번 주 다음 스프린트 프로젝트 종료 직전 4 정리는 항상 마지막이다. 그리고 마지막은 잘 오지 않는다. +125 개발자가 만든 앱에 성능 문제가 없는 이유는? AI를 써서 구현에 흠잡을 데가 없어서 테스트로 검증이 완료되어서 사용자가 없어서 4 지금은 요청이 거의 없어서 문제가 안 보일 뿐이다. 유저가 늘어나는 순간, 이 말은 회고록이 된다. +126 잘 돌아가던 프로그램이 갑자기 돌아가지 않는다. 가장 효과적인 해결책은? AI에게 질문한다 가장 최근에 짠 코드부터 의심한다 옆에 있는 동료에게 떠넘긴다 처음부터 다시 한다 2 코드는 거짓말 안 한다. 방금 손댄 사람이 문제다. 그리고 대부분 그 사람이 나다. +127 개발자가 회의에서 가장 자주 쓰는 방어 스킬은? 침묵 "케이스 바이 케이스죠" 질문 고개 끄덕임 2 정답을 미루는 만능 주문. +128 개발자가 가장 안심하는 순간은? 코드가 깔끔할 때 테스트에 성공했을 때 아무 변화 없을 때 리뷰에 통과했을 때 3 아무 변화 없음 == 사고 없음. +129 개발자가 갑자기 웃을 때 가장 위험한 이유는? 포기해서 해결해서 아이디어가 떠올라서 농담이 생각나서 1 웃음은 정신적 마감 신호다. +130 금요일 오후 5시, 아주 사소한 코드 수정을 마쳤다. 이때 개발자가 절대 해서는 안 될 행동은? 코드 리뷰 요청하기 로컬에서 테스트하기 운영 서버에 바로 배포하고 퇴근하기 다음 주 할 일 목록 작성하기 3 금요일 오후 배포 금지"는 개발계의 성경과 같습니다. 월화수목 잘 돌아가던 서버가 내가 퇴근하는 순간 멈추는 마법을 경험하고 싶지 않다면 말이죠. +131 개발자가 코드를 작성하는 시간보다 더 많이 쓰는 시간은? 기획안 읽기 변수 이름 고민하기 키보드 청소하기 탕비실에서 간식 먹기 2 data1, temp_list, final_result... 적절한 이름을 짓는 것은 알고리즘 설계보다 더 큰 창작의 고통을 수반합니다. +132 시니어 개발자가 코드를 삭제할 때 짓는 표정은? 아까워하며 눈물을 흘린다. 분노하며 키보드를 친다. 그 어느 때보다 인자하고 행복해 보인다. 아무 생각이 없다. 3 코드를 100줄 추가하는 것보다, 불필요한 코드 100줄을 지우고 똑같이 동작하게 만드는 것이 진정한 고수의 희열입니다. +133 개발자가 기능을 지우는 데 죄책감을 느끼지 않는 순간은? 아무도 안 쓸 때 버그일 때 성능 문제 기획 변경 1 코드는 애착이 아니라 도구다. +134 “이 방식도 나쁘지 않아요”의 숨은 의미는? 좋은 선택 합리적 판단 대안 인정 딱히 더 나은게 없다 4 찬성이 아닌 포기다. +135 “이 정도면 충분하죠”가 나오는 정확한 타이밍은? 완벽할 때 요구사항 충족 시 더 하면 끝이 없을 때 태스트 통과 후 3 \ No newline at end of file diff --git a/SoloDeveloperTraining/SoloDeveloperTrainingTests/ConcurrencyIssueTests.swift b/SoloDeveloperTraining/SoloDeveloperTrainingTests/ConcurrencyIssueTests.swift new file mode 100644 index 00000000..625b6c6f --- /dev/null +++ b/SoloDeveloperTraining/SoloDeveloperTrainingTests/ConcurrencyIssueTests.swift @@ -0,0 +1,35 @@ +// +// ConcurrencyIssueTests.swift +// SoloDeveloperTrainingTests +// +// Created by SeoJunYoung on 2/2/26. +// + +import Testing +import Foundation +@testable import SoloDeveloperTraining + +struct ConcurrencyIssueTests { + + @Test("Wallet Task.detached Concurrent - race condition 재현") + func walletTaskDetachedRaceCondition() async throws { + let user = await User(nickname: "test") + let wallet = await user.wallet + let iterations = 10000 + + await withTaskGroup(of: Void.self) { group in + for _ in 0..