[
  {
    "path": "LICENSE",
    "content": "                                 Apache License\n                           Version 2.0, January 2004\n                        http://www.apache.org/licenses/\n\n   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION\n\n   1. Definitions.\n\n      \"License\" shall mean the terms and conditions for use, reproduction,\n      and distribution as defined by Sections 1 through 9 of this document.\n\n      \"Licensor\" shall mean the copyright owner or entity authorized by\n      the copyright owner that is granting the License.\n\n      \"Legal Entity\" shall mean the union of the acting entity and all\n      other entities that control, are controlled by, or are under common\n      control with that entity. For the purposes of this definition,\n      \"control\" means (i) the power, direct or indirect, to cause the\n      direction or management of such entity, whether by contract or\n      otherwise, or (ii) ownership of fifty percent (50%) or more of the\n      outstanding shares, or (iii) beneficial ownership of such entity.\n\n      \"You\" (or \"Your\") shall mean an individual or Legal Entity\n      exercising permissions granted by this License.\n\n      \"Source\" form shall mean the preferred form for making modifications,\n      including but not limited to software source code, documentation\n      source, and configuration files.\n\n      \"Object\" form shall mean any form resulting from mechanical\n      transformation or translation of a Source form, including but\n      not limited to compiled object code, generated documentation,\n      and conversions to other media types.\n\n      \"Work\" shall mean the work of authorship, whether in Source or\n      Object form, made available under the License, as indicated by a\n      copyright notice that is included in or attached to the work\n      (an example is provided in the Appendix below).\n\n      \"Derivative Works\" shall mean any work, whether in Source or Object\n      form, that is based on (or derived from) the Work and for which the\n      editorial revisions, annotations, elaborations, or other modifications\n      represent, as a whole, an original work of authorship. For the purposes\n      of this License, Derivative Works shall not include works that remain\n      separable from, or merely link (or bind by name) to the interfaces of,\n      the Work and Derivative Works thereof.\n\n      \"Contribution\" shall mean any work of authorship, including\n      the original version of the Work and any modifications or additions\n      to that Work or Derivative Works thereof, that is intentionally\n      submitted to Licensor for inclusion in the Work by the copyright owner\n      or by an individual or Legal Entity authorized to submit on behalf of\n      the copyright owner. For the purposes of this definition, \"submitted\"\n      means any form of electronic, verbal, or written communication sent\n      to the Licensor or its representatives, including but not limited to\n      communication on electronic mailing lists, source code control systems,\n      and issue tracking systems that are managed by, or on behalf of, the\n      Licensor for the purpose of discussing and improving the Work, but\n      excluding communication that is conspicuously marked or otherwise\n      designated in writing by the copyright owner as \"Not a Contribution.\"\n\n      \"Contributor\" shall mean Licensor and any individual or Legal Entity\n      on behalf of whom a Contribution has been received by Licensor and\n      subsequently incorporated within the Work.\n\n   2. Grant of Copyright License. Subject to the terms and conditions of\n      this License, each Contributor hereby grants to You a perpetual,\n      worldwide, non-exclusive, no-charge, royalty-free, irrevocable\n      copyright license to reproduce, prepare Derivative Works of,\n      publicly display, publicly perform, sublicense, and distribute the\n      Work and such Derivative Works in Source or Object form.\n\n   3. Grant of Patent License. Subject to the terms and conditions of\n      this License, each Contributor hereby grants to You a perpetual,\n      worldwide, non-exclusive, no-charge, royalty-free, irrevocable\n      (except as stated in this section) patent license to make, have made,\n      use, offer to sell, sell, import, and otherwise transfer the Work,\n      where such license applies only to those patent claims licensable\n      by such Contributor that are necessarily infringed by their\n      Contribution(s) alone or by combination of their Contribution(s)\n      with the Work to which such Contribution(s) was submitted. If You\n      institute patent litigation against any entity (including a\n      cross-claim or counterclaim in a lawsuit) alleging that the Work\n      or a Contribution incorporated within the Work constitutes direct\n      or contributory patent infringement, then any patent licenses\n      granted to You under this License for that Work shall terminate\n      as of the date such litigation is filed.\n\n   4. Redistribution. You may reproduce and distribute copies of the\n      Work or Derivative Works thereof in any medium, with or without\n      modifications, and in Source or Object form, provided that You\n      meet the following conditions:\n\n      (a) You must give any other recipients of the Work or\n          Derivative Works a copy of this License; and\n\n      (b) You must cause any modified files to carry prominent notices\n          stating that You changed the files; and\n\n      (c) You must retain, in the Source form of any Derivative Works\n          that You distribute, all copyright, patent, trademark, and\n          attribution notices from the Source form of the Work,\n          excluding those notices that do not pertain to any part of\n          the Derivative Works; and\n\n      (d) If the Work includes a \"NOTICE\" text file as part of its\n          distribution, then any Derivative Works that You distribute must\n          include a readable copy of the attribution notices contained\n          within such NOTICE file, excluding those notices that do not\n          pertain to any part of the Derivative Works, in at least one\n          of the following places: within a NOTICE text file distributed\n          as part of the Derivative Works; within the Source form or\n          documentation, if provided along with the Derivative Works; or,\n          within a display generated by the Derivative Works, if and\n          wherever such third-party notices normally appear. The contents\n          of the NOTICE file are for informational purposes only and\n          do not modify the License. You may add Your own attribution\n          notices within Derivative Works that You distribute, alongside\n          or as an addendum to the NOTICE text from the Work, provided\n          that such additional attribution notices cannot be construed\n          as modifying the License.\n\n      You may add Your own copyright statement to Your modifications and\n      may provide additional or different license terms and conditions\n      for use, reproduction, or distribution of Your modifications, or\n      for any such Derivative Works as a whole, provided Your use,\n      reproduction, and distribution of the Work otherwise complies with\n      the conditions stated in this License.\n\n   5. Submission of Contributions. Unless You explicitly state otherwise,\n      any Contribution intentionally submitted for inclusion in the Work\n      by You to the Licensor shall be under the terms and conditions of\n      this License, without any additional terms or conditions.\n      Notwithstanding the above, nothing herein shall supersede or modify\n      the terms of any separate license agreement you may have executed\n      with Licensor regarding such Contributions.\n\n   6. Trademarks. This License does not grant permission to use the trade\n      names, trademarks, service marks, or product names of the Licensor,\n      except as required for reasonable and customary use in describing the\n      origin of the Work and reproducing the content of the NOTICE file.\n\n   7. Disclaimer of Warranty. Unless required by applicable law or\n      agreed to in writing, Licensor provides the Work (and each\n      Contributor provides its Contributions) on an \"AS IS\" BASIS,\n      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or\n      implied, including, without limitation, any warranties or conditions\n      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A\n      PARTICULAR PURPOSE. You are solely responsible for determining the\n      appropriateness of using or redistributing the Work and assume any\n      risks associated with Your exercise of permissions under this License.\n\n   8. Limitation of Liability. In no event and under no legal theory,\n      whether in tort (including negligence), contract, or otherwise,\n      unless required by applicable law (such as deliberate and grossly\n      negligent acts) or agreed to in writing, shall any Contributor be\n      liable to You for damages, including any direct, indirect, special,\n      incidental, or consequential damages of any character arising as a\n      result of this License or out of the use or inability to use the\n      Work (including but not limited to damages for loss of goodwill,\n      work stoppage, computer failure or malfunction, or any and all\n      other commercial damages or losses), even if such Contributor\n      has been advised of the possibility of such damages.\n\n   9. Accepting Warranty or Additional Liability. While redistributing\n      the Work or Derivative Works thereof, You may choose to offer,\n      and charge a fee for, acceptance of support, warranty, indemnity,\n      or other liability obligations and/or rights consistent with this\n      License. However, in accepting such obligations, You may act only\n      on Your own behalf and on Your sole responsibility, not on behalf\n      of any other Contributor, and only if You agree to indemnify,\n      defend, and hold each Contributor harmless for any liability\n      incurred by, or claims asserted against, such Contributor by reason\n      of your accepting any such warranty or additional liability.\n\n   END OF TERMS AND CONDITIONS\n\n   APPENDIX: How to apply the Apache License to your work.\n\n      To apply the Apache License to your work, attach the following\n      boilerplate notice, with the fields enclosed by brackets \"[]\"\n      replaced with your own identifying information. (Don't include\n      the brackets!)  The text should be enclosed in the appropriate\n      comment syntax for the file format. We also recommend that a\n      file or class name and description of purpose be included on the\n      same \"printed page\" as the copyright notice for easier\n      identification within third-party archives.\n\n   Copyright [yyyy] [name of copyright owner]\n\n   Licensed under the Apache License, Version 2.0 (the \"License\");\n   you may not use this file except in compliance with the License.\n   You may obtain a copy of the License at\n\n       http://www.apache.org/licenses/LICENSE-2.0\n\n   Unless required by applicable law or agreed to in writing, software\n   distributed under the License is distributed on an \"AS IS\" BASIS,\n   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n   See the License for the specific language governing permissions and\n   limitations under the License."
  },
  {
    "path": "MotionScape/Assets.xcassets/AccentColor.colorset/Contents.json",
    "content": "{\n  \"colors\" : [\n    {\n      \"idiom\" : \"universal\"\n    }\n  ],\n  \"info\" : {\n    \"author\" : \"xcode\",\n    \"version\" : 1\n  }\n}\n"
  },
  {
    "path": "MotionScape/Assets.xcassets/AppIcon.appiconset/Contents.json",
    "content": "{\n  \"images\" : [\n    {\n      \"filename\" : \"16.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"1x\",\n      \"size\" : \"16x16\"\n    },\n    {\n      \"filename\" : \"16@2x.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"2x\",\n      \"size\" : \"16x16\"\n    },\n    {\n      \"filename\" : \"32.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"1x\",\n      \"size\" : \"32x32\"\n    },\n    {\n      \"filename\" : \"32@2x.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"2x\",\n      \"size\" : \"32x32\"\n    },\n    {\n      \"filename\" : \"128.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"1x\",\n      \"size\" : \"128x128\"\n    },\n    {\n      \"filename\" : \"128@2x.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"2x\",\n      \"size\" : \"128x128\"\n    },\n    {\n      \"filename\" : \"256.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"1x\",\n      \"size\" : \"256x256\"\n    },\n    {\n      \"filename\" : \"256@2x.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"2x\",\n      \"size\" : \"256x256\"\n    },\n    {\n      \"filename\" : \"512.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"1x\",\n      \"size\" : \"512x512\"\n    },\n    {\n      \"filename\" : \"512@2x.png\",\n      \"idiom\" : \"mac\",\n      \"scale\" : \"2x\",\n      \"size\" : \"512x512\"\n    }\n  ],\n  \"info\" : {\n    \"author\" : \"xcode\",\n    \"version\" : 1\n  }\n}\n"
  },
  {
    "path": "MotionScape/Assets.xcassets/Contents.json",
    "content": "{\n  \"info\" : {\n    \"author\" : \"xcode\",\n    \"version\" : 1\n  }\n}\n"
  },
  {
    "path": "MotionScape/Assets.xcassets/codePreviewBackground.colorset/Contents.json",
    "content": "{\n  \"colors\" : [\n    {\n      \"color\" : {\n        \"color-space\" : \"srgb\",\n        \"components\" : {\n          \"alpha\" : \"1.000\",\n          \"blue\" : \"0xFF\",\n          \"green\" : \"0xFF\",\n          \"red\" : \"0xFF\"\n        }\n      },\n      \"idiom\" : \"universal\"\n    },\n    {\n      \"appearances\" : [\n        {\n          \"appearance\" : \"luminosity\",\n          \"value\" : \"dark\"\n        }\n      ],\n      \"color\" : {\n        \"color-space\" : \"srgb\",\n        \"components\" : {\n          \"alpha\" : \"1.000\",\n          \"blue\" : \"0x21\",\n          \"green\" : \"0x1F\",\n          \"red\" : \"0x1D\"\n        }\n      },\n      \"idiom\" : \"universal\"\n    }\n  ],\n  \"info\" : {\n    \"author\" : \"xcode\",\n    \"version\" : 1\n  }\n}\n"
  },
  {
    "path": "MotionScape/Assets.xcassets/copyPopupBackgroundColor.colorset/Contents.json",
    "content": "{\n  \"colors\" : [\n    {\n      \"color\" : {\n        \"color-space\" : \"srgb\",\n        \"components\" : {\n          \"alpha\" : \"1.000\",\n          \"blue\" : \"1.000\",\n          \"green\" : \"1.000\",\n          \"red\" : \"1.000\"\n        }\n      },\n      \"idiom\" : \"universal\"\n    },\n    {\n      \"appearances\" : [\n        {\n          \"appearance\" : \"luminosity\",\n          \"value\" : \"dark\"\n        }\n      ],\n      \"color\" : {\n        \"color-space\" : \"srgb\",\n        \"components\" : {\n          \"alpha\" : \"1.000\",\n          \"blue\" : \"0.000\",\n          \"green\" : \"0.000\",\n          \"red\" : \"0.000\"\n        }\n      },\n      \"idiom\" : \"universal\"\n    }\n  ],\n  \"info\" : {\n    \"author\" : \"xcode\",\n    \"version\" : 1\n  }\n}\n"
  },
  {
    "path": "MotionScape/Assets.xcassets/stream-logo.imageset/Contents.json",
    "content": "{\n  \"images\" : [\n    {\n      \"filename\" : \"stream-logo.svg\",\n      \"idiom\" : \"universal\",\n      \"scale\" : \"1x\"\n    },\n    {\n      \"idiom\" : \"universal\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"universal\",\n      \"scale\" : \"3x\"\n    }\n  ],\n  \"info\" : {\n    \"author\" : \"xcode\",\n    \"version\" : 1\n  }\n}\n"
  },
  {
    "path": "MotionScape/Model/AnimationControlOption.swift",
    "content": "//\n//  AnimationControlOption.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 25.03.22.\n//\n\nimport Foundation\n\nenum AnimationControlOption: String, CaseIterable, Identifiable {\n    case parameters, options\n    var id: Self { self }\n}\n"
  },
  {
    "path": "MotionScape/Model/AnimationExample.swift",
    "content": "//\n//  AnimationExample.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 18.03.22.\n//\n\nimport Foundation\n\nenum AnimationExample: String, CaseIterable, Identifiable {\n    case circles, chains, emojis, gradientCircle, text\n    \n    var id: Self { self }\n}\n"
  },
  {
    "path": "MotionScape/Model/AnimationOption.swift",
    "content": "//\n//  AnimationOption.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 24.03.22.\n//\n\nimport SwiftUI\n\nstruct AnimationOption: Identifiable, Equatable, Documentable {\n    \n    var id = UUID()\n    var type: AnimationOptionType\n    var active: Bool = false\n    var value: Double\n    var description: String\n    var defaultValueDescription: String\n    var rangeDescription: String\n    \n    \n    var name: String {\n        return type.rawValue.capitalized\n    }\n    \n    func createCodeSnippet() -> String {\n        switch type {\n        case .delay:\n            return \"\\n\\t.delay(\\(value.stringWith(places: codePreviewDecimalPlaces)))\"\n        case .speed:\n            return \"\\n\\t.speed(\\(value.stringWith(places: codePreviewDecimalPlaces)))\"\n        }\n    }\n    \n    static func createDelay() -> AnimationOption {\n        return AnimationOption(\n            type: .delay,\n            value: 1.0,\n            description: \"Delays the animation by the specified amount. Unit is seconds and the value needs to be a `Double`.\",\n            defaultValueDescription: \"0.0\",\n            rangeDescription: \"0 ... 10\"\n        )\n    }\n    \n    static func createSpeed() -> AnimationOption {\n        return AnimationOption(\n            type: .speed,\n            value: 0.25,\n            description: \"This is an option to speed up or slow down the animation. It is of type `Double`. You can make it a value below 1 to slow it down and increase it to a value larger than 1 to speed it up.\",\n            defaultValueDescription: \"1\",\n            rangeDescription: \"0 ... 10\")\n    }\n}\n"
  },
  {
    "path": "MotionScape/Model/AnimationOptionType.swift",
    "content": "//\n//  AnimationOptionType.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 25.03.22.\n//\n\nimport Foundation\n\nenum AnimationOptionType: String {\n    case delay, speed\n}\n"
  },
  {
    "path": "MotionScape/Model/AnimationParameter.swift",
    "content": "//\n//  AnimationParameter.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport Foundation\n\nstruct AnimationParameter: Equatable, Documentable {\n    var name: String\n    var description: String\n    var defaultValue: Double?\n    var range: ClosedRange<Double>\n    \n    var defaultValueDescription: String {\n        defaultValue != nil ? \"\\(defaultValue!)\" : \"Not available\"\n    }\n    \n    var rangeDescription: String {\n        range.toString\n    }\n}\n"
  },
  {
    "path": "MotionScape/Model/AnimationType.swift",
    "content": "//\n//  AnimationType.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport Foundation\n\nenum AnimationType: String, Equatable {\n    case interpolatingSpring = \"Interpolating Spring\"\n    case interactiveSpring = \"Interactive Spring\"\n    case spring = \"Spring\"\n    case linear = \"Linear\"\n    case defaultAnimation = \"Default\"\n    case easeIn = \"Ease In\"\n    case easeOut = \"Ease Out\"\n    case easeInOut = \"Ease In Out\"\n    case timingCurve = \"Custom Timing Curve\"\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/AllAnimations.swift",
    "content": "//\n//  AllAnimations.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 18.03.22.\n//\n\nimport Foundation\n\nstruct AllAnimations {\n    var interpolatingSpring = InterpolatingSpring()\n    var interactiveSpring = InteractiveSpring()\n    var spring = Spring()\n    var linear = Linear()\n    var defaultAnimation = Default()\n    var easeIn = EaseIn()\n    var easeOut = EaseOut()\n    var easeInOut = EaseInOut()\n    var timingCurve = TimingCurve()\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/Default.swift",
    "content": "//\n//  Default.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 29.03.22.\n//\n\nimport SwiftUI\n\nstruct Default: Equatable {\n    var name = \"Default\"\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay(),\n        .createSpeed()\n    ]\n}\n\nextension Default: MyAnimation {\n    \n    func createAnimation() -> Animation {\n        return .default\n            .speed(getSpeed())\n            .delay(getDelay())\n    }\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.default\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n    \n    \n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/EaseIn.swift",
    "content": "//\n//  EaseIn.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct EaseIn: Equatable {\n    var duration: Double = 1\n    \n    var name = \"EaseIn\"\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay(),\n        .createSpeed()\n    ]\n}\n\nextension EaseIn: MyAnimation {\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.easeIn(\n    duration: \\(duration.stringWith(places: codePreviewDecimalPlaces))\n)\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n    \n    func createAnimation() -> Animation {\n        return .easeIn(\n            duration: duration\n        )\n        .speed(getSpeed())\n        .delay(getDelay())\n    }\n}\n\nextension EaseIn {\n    \n    // Parameter\n    static let durationParameter = AnimationParameter(\n        name: \"Duration\",\n        description: \"The duration of the animation.\",\n        range: 0 ... 10)\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/EaseInOut.swift",
    "content": "//\n//  EaseInOut.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct EaseInOut: Equatable {\n    var duration: Double = 1\n    \n    var name = \"EaseInOut\"\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay(),\n        .createSpeed()\n    ]\n}\n\nextension EaseInOut: MyAnimation {\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.easeInOut(\n    duration: \\(duration.stringWith(places: codePreviewDecimalPlaces))\n)\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n    \n    func createAnimation() -> Animation {\n        return .easeInOut(\n            duration: duration\n        )\n        .speed(getSpeed())\n        .delay(getDelay())\n    }\n}\n\nextension EaseInOut {\n    \n    // Parameter\n    static let durationParameter = AnimationParameter(\n        name: \"Duration\",\n        description: \"The duration of the animation.\",\n        range: 0 ... 10)\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/EaseOut.swift",
    "content": "//\n//  EaseOut.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct EaseOut: Equatable {\n    var duration: Double = 1\n    \n    var name = \"EaseOut\"\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay(),\n        .createSpeed()\n    ]\n}\n\nextension EaseOut: MyAnimation {\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.easeOut(\n    duration: \\(duration.stringWith(places: codePreviewDecimalPlaces))\n)\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n    \n    func createAnimation() -> Animation {\n        return .easeOut(\n            duration: duration\n        )\n        .speed(getSpeed())\n        .delay(getDelay())\n    }\n}\n\nextension EaseOut {\n    \n    // Parameter\n    static let durationParameter = AnimationParameter(\n        name: \"Duration\",\n        description: \"The duration of the animation.\",\n        range: 0 ... 10)\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/InteractiveSpring.swift",
    "content": "//\n//  InteractiveSpring.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport SwiftUI\n\nstruct InteractiveSpring: Equatable {\n    \n    var name = \"Interactive Spring\"\n    \n    var response: Double = 0.15\n    var dampingFraction: Double = 0.86\n    var blendDuration: Double = 0.25\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay()\n    ]\n}\n\nextension InteractiveSpring: MyAnimation {\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.interactiveSpring(\n    response: \\(response.stringWith(places: codePreviewDecimalPlaces)),\n    dampingFraction: \\(dampingFraction.stringWith(places: codePreviewDecimalPlaces)),\n    blendDuration: \\(blendDuration.stringWith(places: codePreviewDecimalPlaces))\n)\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n    \n    \n    func createAnimation() -> Animation {\n        return .interactiveSpring(\n            response: response,\n            dampingFraction: dampingFraction,\n            blendDuration: blendDuration\n        )\n        .speed(getSpeed())\n        .delay(getDelay())\n    }\n}\n\nextension InteractiveSpring {\n    \n    // Parameters\n    static let responseParameter = AnimationParameter(\n        name: \"Response\",\n        description: \"\"\"\nResponse controls how quickly an animating property value will try and get to a target. You can use the response to create an infinitely-stiff spring by setting its value to 0.\n\nA sensible range is from 0.15 - 0.55.\n\"\"\",\n        defaultValue: 0.15,\n        range: 0.01 ... 1\n    )\n    \n    static let dampingFractionParameter = AnimationParameter(\n        name: \"Damping fraction\",\n        description: \"\"\"\nIt is defined as the ratio at which an oscillating view stops over time. You can use it to causes a gradual reduction in the spring's oscillation.\n\nExperiment with the range of 0.25 - 0.55 to build animations with higher bounciness.\n\"\"\",\n        defaultValue: 0.86,\n        range: 0.01 ... 1\n    )\n    \n    static let blendDurationParameter = AnimationParameter(\n        name: \"Blend duration\",\n        description: \"\"\"\nBlend duration defines how previous animation transitions to the next. It works with stacked or chained animations only, helping to create a smooth transition from the previous animation to the next. It cannot be observed when you have only one spring animation on the view.\n\n\"\"\",\n        defaultValue: 0.25,\n        range: 0.01 ... 1\n    )\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/InterpolatingSpring.swift",
    "content": "//\n//  InterpolatingSpring.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport SwiftUI\n\nstruct InterpolatingSpring {\n    \n    var name: String = \"Interpolating Spring\"\n    var mass: Double = 1\n    var stiffness: Double = 170\n    var damping: Double = 15\n    var initialVelocity: Double = 0.0\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay()\n    ]\n}\n\nextension InterpolatingSpring: MyAnimation {\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.interpolatingSpring(\n    mass: \\(mass.stringWith(places: codePreviewDecimalPlaces)),\n    stiffness: \\(stiffness.stringWith(places: codePreviewDecimalPlaces)),\n    damping: \\(damping.stringWith(places: codePreviewDecimalPlaces)),\n    initialVelocity: \\(initialVelocity.stringWith(places: codePreviewDecimalPlaces))\n)\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n \n    func createAnimation() -> Animation {\n        return .interpolatingSpring(\n            mass: mass,\n            stiffness: stiffness,\n            damping: damping,\n            initialVelocity: initialVelocity\n        )\n        .speed(getSpeed())\n        .delay(getDelay())\n    }\n}\n\nextension InterpolatingSpring {\n    \n    // Parameters\n    static let stiffnessParamter = AnimationParameter(\n        name: \"Stiffness\",\n        description: \"\"\"\nStiffness is the tensile strength of the spring. It changes how quickly the object moves towards its target. A higher stiffness will make the animation snappier.\n\nUse a range of 5 - 400 depending on the mass and damping value. For example, using the stiffness of 170 and the damping of 15 creates a gentle bounce.\n\"\"\",\n        range: 1 ... 400)\n    \n    static let dampingParameter = AnimationParameter(\n        name: \"Damping\",\n        description: \"\"\"\nIt is defined as the back-drag frictional force of the surface the animating object is resting on. The operating principle of damping is much like the braking system of a car. Its purpose is to stop the animating view over time.  It also affects the ability to overshoot the object.  Use damping of 0 to make the animating view oscillate forever.\n\nStart with 100, which means no overshoot. Using, for example, damping of 5 will create more overshoot.\n\"\"\",\n        range: 1 ... 400\n    )\n    \n    static let massParameter = AnimationParameter(\n        name: \"Mass\",\n        description: \"\"\"\nThis is the weight of the object attached to the spring. It changes the willingness of the object to move or stop moving. A larger mass makes the animating view difficult to move, speed up, slow down.\n\nA sensible range goes from 1 to 10.\n\"\"\",\n        defaultValue: 1,\n        range: 1 ... 10\n    )\n    \n    static let initialVelocityParameter = AnimationParameter(\n        name: \"Initial Velocity\",\n        description: \"\"\"\nIt is defined as the speed at which the animation object changes at the beginning of the animation. Using a higher value will speed up the animation quickly in the begin-time.\n\nA sensible range goes from 1 to 4.\n\"\"\",\n        defaultValue: 0,\n        range: 0 ... 10\n    )\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/Linear.swift",
    "content": "//\n//  Linear.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct Linear: Equatable {\n    var duration: Double = 1\n    \n    var name = \"Linear\"\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay(),\n        .createSpeed()\n    ]\n}\n\nextension Linear: MyAnimation {\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.linear(\n    duration: \\(duration.stringWith(places: codePreviewDecimalPlaces))\n)\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n \n    func createAnimation() -> Animation {\n        return .linear(\n            duration: duration\n        )\n        .speed(getSpeed())\n        .delay(getDelay())\n    }\n    \n}\n\nextension Linear {\n    \n    // Parameter\n    static let durationParameter = AnimationParameter(\n        name: \"Duration\",\n        description: \"The duration of the animation.\",\n        range: 0 ... 10)\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/Spring.swift",
    "content": "//\n//  Spring.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct Spring: Equatable {\n    var response: Double = 0.55\n    var dampingFraction: Double = 0.825\n    var blendDuration: Double = 0\n    \n    var name = \"Spring\"\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay()\n    ]\n}\n\nextension Spring: MyAnimation {\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.spring(\n    response: \\(response.stringWith(places: codePreviewDecimalPlaces)),\n    dampingFraction: \\(dampingFraction.stringWith(places: codePreviewDecimalPlaces)),\n    blendDuration: \\(blendDuration.stringWith(places: codePreviewDecimalPlaces))\n)\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n    \n    \n    func createAnimation() -> Animation {\n        return .spring(\n            response: response,\n            dampingFraction: dampingFraction,\n            blendDuration: blendDuration\n        )\n        .speed(getSpeed())\n        .delay(getDelay())\n    }\n}\n\nextension Spring {\n    \n    // Parameter\n    static let responseParameter = AnimationParameter(\n        name: \"Response\",\n        description: \"\"\"\nResponse controls how quickly an animating property value will try and get to a target. You can use the response to create an infinitely-stiff spring by setting its value to 0.\n\nA sensible range is from 0 - 1.\n\"\"\",\n        defaultValue: 0.55,\n        range: 0.01 ... 1\n    )\n    \n    static let dampingFractionParameter = AnimationParameter(\n        name: \"Damping fraction\",\n        description: \"\"\"\nThe amount of drag applied to the value being animated, as a fraction of an estimate of the amount needed to produce critical damping. It is defined as the ratio at which an oscillating view stops over time. You can use it to causes a gradual reduction in the spring's oscillation.\n\nA sensible range is from 0 - 1.\n\"\"\",\n        defaultValue: 0.825,\n        range: 0.01 ... 1\n    )\n    \n    static let blendDurationParameter = AnimationParameter(\n        name: \"Blend duration\",\n        description: \"\"\"\nBlend duration defines how previous animation transitions to the next. It works with stacked or chained animations only, helping to create a smooth transition from the previous animation to the next. It cannot be observed when you have only one spring animation on the view.\n\n\"\"\",\n        defaultValue: 0,\n        range: 0.01 ... 1\n    )\n}\n"
  },
  {
    "path": "MotionScape/Model/Animations/TimingCurve.swift",
    "content": "//\n//  TimingCurve.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 21.03.22.\n//\n\nimport SwiftUI\n\nstruct TimingCurve {\n    var x0: Double = 0.13\n    var y0: Double = 0.65\n    var x1: Double = 0.81\n    var y1: Double = 0.18\n    \n    var name = \"Custom Timing Curve\"\n    \n    var animationOptions: [AnimationOption] = [\n        .createDelay(),\n        .createSpeed()\n    ]\n}\n\nextension TimingCurve: MyAnimation {\n    \n    func createCodeSnippet() -> String {\n        let animationString = \"\"\"\n.timingCurve(\n    \\(x0.stringWith(places: codePreviewDecimalPlaces)),\n    \\(y0.stringWith(places: codePreviewDecimalPlaces)),\n    \\(x1.stringWith(places: codePreviewDecimalPlaces)),\n    \\(y1.stringWith(places: codePreviewDecimalPlaces))\n)\n\"\"\"\n        \n        return addAnimationOptions(to: animationString)\n    }\n \n    func createAnimation() -> Animation {\n        return .timingCurve(\n            x0,\n            y0,\n            x1,\n            y1\n        )\n        .speed(getSpeed())\n        .delay(getDelay())\n    }\n    \n}\n\nextension TimingCurve {\n    \n    // Parameters\n    static let firstControlPointX = AnimationParameter(name: \"x0\", description: \"The x value of the first control point for the timing curve.\", defaultValue: 0.13, range: 0 ... 1)\n    \n    static let firstControlPointY = AnimationParameter(name: \"y0\", description: \"The y value of the first control point for the timing curve.\", defaultValue: 0.65, range: 0 ... 1)\n    \n    static let secondControlPointX = AnimationParameter(name: \"x1\", description: \"The x value of the second control point for the timing curve.\", defaultValue: 0.81, range: 0 ... 1)\n    \n    static let secondControlPointY = AnimationParameter(name: \"y1\", description: \"The y value of the second control point for the timing curve.\", defaultValue: 0.18, range: 0 ... 1)\n}\n"
  },
  {
    "path": "MotionScape/Model/DefaultValues.swift",
    "content": "//\n//  DefaultValues.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 30.03.22.\n//\n\nlet codePreviewDecimalPlaces = 5\n"
  },
  {
    "path": "MotionScape/Model/Documentable.swift",
    "content": "//\n//  Documentable.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 25.03.22.\n//\n\nimport Foundation\n\nprotocol Documentable {\n    var description: String { get }\n    var defaultValueDescription: String { get }\n    var rangeDescription: String { get }\n}\n"
  },
  {
    "path": "MotionScape/Model/Extensions/Animation+Create.swift",
    "content": "//\n//  Animation+Create.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport SwiftUI\n\nextension Animation {\n    static func create(from viewModel: AnimationsViewModel) -> Animation {\n        switch viewModel.selectedAnimation {\n        case .interpolatingSpring:\n            return viewModel.animations.interpolatingSpring.createAnimation()\n        case .interactiveSpring:\n            return viewModel.animations.interactiveSpring.createAnimation()\n        case .spring:\n            return viewModel.animations.spring.createAnimation()\n        case .linear:\n            return viewModel.animations.linear.createAnimation()\n        case .defaultAnimation:\n            return viewModel.animations.defaultAnimation.createAnimation()\n        case .easeIn:\n            return viewModel.animations.easeIn.createAnimation()\n        case .easeOut:\n            return viewModel.animations.easeOut.createAnimation()\n        case .easeInOut:\n            return viewModel.animations.easeInOut.createAnimation()\n        case .timingCurve:\n            return viewModel.animations.timingCurve.createAnimation()\n        case .none:\n            fatalError(\"No animation was selected and that's not a valid use-case!\")\n        }\n    }\n}\n"
  },
  {
    "path": "MotionScape/Model/Extensions/CGFloat+Constants.swift",
    "content": "//\n//  CGFloat+Constants.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 04.04.22.\n//\n\nimport Foundation\n\nextension CGFloat {\n    static let MIN_WIDTH_CONTROL_VIEW: CGFloat = 500\n    static let MIN_WIDTH_ANIMATION_VIEW: CGFloat = 450\n    static let MIN_WIDTH_SIDEBAR: CGFloat = 200\n}\n"
  },
  {
    "path": "MotionScape/Model/Extensions/ClosedRange+toString.swift",
    "content": "//\n//  ClosedRange+toString.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 24.03.22.\n//\n\nimport Foundation\n\nextension ClosedRange {\n    var toString: String {\n        \"\\(lowerBound) ... \\(upperBound)\"\n    }\n}\n"
  },
  {
    "path": "MotionScape/Model/Extensions/Double+DecimalPlaces.swift",
    "content": "//\n//  Double+DecimalPlaces.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport Foundation\n\nextension Double {\n    func stringWith(places decimals: Int) -> String {\n        let formatter = NumberFormatter()\n        formatter.numberStyle = .decimal\n        formatter.maximumFractionDigits = decimals\n        formatter.decimalSeparator = \".\"\n        return formatter.string(from: self as NSNumber) ?? String(format: \"%.\\(decimals)f\", self)\n    }\n}\n"
  },
  {
    "path": "MotionScape/Model/MyAnimation.swift",
    "content": "//\n//  MyAnimation.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 25.03.22.\n//\n\nimport SwiftUI\n\nprotocol MyAnimation {\n    var name: String { get }\n    var animationOptions: [AnimationOption] { get }\n    \n    func createAnimation() -> Animation\n    func createCodeSnippet() -> String\n    func addAnimationOptions(to animationString: String) -> String\n    \n    func getDelay() -> Double\n    func getSpeed() -> Double\n}\n\nextension MyAnimation {\n    \n    func addAnimationOptions(to animationString: String) -> String {\n        var copiedAnimationString = animationString\n        \n        // get all active animation options for that Animation\n        let activeAnimationOptions = animationOptions\n            .filter { $0.active }\n        \n        // add each animation option to the copied String\n        for animation in activeAnimationOptions {\n            copiedAnimationString += animation.createCodeSnippet()\n        }\n        \n        return copiedAnimationString\n    }\n    \n    func getDelay() -> Double {\n        return animationOptions\n            .filter { $0.type == .delay && $0.active }\n            .map { $0.value }\n            .first ?? 0.0\n    }\n    \n    func getSpeed() -> Double {\n        return animationOptions\n            .filter { $0.type == .speed && $0.active }\n            .map { $0.value }\n            .first ?? 1\n    }\n    \n}\n"
  },
  {
    "path": "MotionScape/Model/PreviewType.swift",
    "content": "//\n//  PreviewType.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 24.03.22.\n//\n\nimport Foundation\n\nenum PreviewType: String, CaseIterable, Identifiable {\n    case animation, code\n    var id: Self { self }\n}\n"
  },
  {
    "path": "MotionScape/MotionScape.entitlements",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<!DOCTYPE plist PUBLIC \"-//Apple//DTD PLIST 1.0//EN\" \"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">\n<plist version=\"1.0\">\n<dict>\n    <key>com.apple.security.app-sandbox</key>\n    <true/>\n    <key>com.apple.security.files.user-selected.read-only</key>\n    <true/>\n</dict>\n</plist>\n"
  },
  {
    "path": "MotionScape/MotionScapeApp.swift",
    "content": "//\n//  MotionScapeApp.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 07.03.22.\n//\n\nimport SwiftUI\n\n@main\nstruct MotionScapeApp: App {\n    var body: some Scene {\n        WindowGroup {\n            ContentView()\n        }\n    }\n}\n"
  },
  {
    "path": "MotionScape/Preview Content/Preview Assets.xcassets/Contents.json",
    "content": "{\n  \"info\" : {\n    \"author\" : \"xcode\",\n    \"version\" : 1\n  }\n}\n"
  },
  {
    "path": "MotionScape/View/AnimationViews/AnimationsContainerView.swift",
    "content": "//\n//  AnimationsContainerView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 18.03.22.\n//\n\nimport SwiftUI\n\nstruct AnimationsContainerView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    \n    @StateObject var exampleViewModel = AnimationsExampleViewModel()\n    @State private var id = 0\n    @State private var selectedPreview: PreviewType = .animation\n    @State private var copyTextOpacity: CGFloat = 0.0\n    \n    var body: some View {\n        VStack(spacing: 20) {\n            HStack {\n                Spacer()\n                \n                Picker(\"\", selection: $selectedPreview) {\n                    ForEach(PreviewType.allCases) { previewType in\n                        Text(previewType.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                Spacer()\n            }\n            \n            Spacer()\n            \n            switch selectedPreview {\n            case .animation:\n                switch exampleViewModel.selectedAnimationExample {\n                case .circles:\n                    CirclesView(viewModel: viewModel)\n                        .id(viewModel.id)\n                case .chains:\n                    ChainsView(viewModel: viewModel)\n                        .id(viewModel.id)\n                case .emojis:\n                    EmojisView(viewModel: viewModel)\n                        .id(viewModel.id)\n                case .gradientCircle:\n                    GradientCircleView(viewModel: viewModel)\n                        .id(viewModel.id)\n                case .text:\n                    TextAnimationContainer(viewModel: viewModel)\n                }\n            case .code:\n                CodePreviewView(code: viewModel.createAnimationCode()) {\n                    viewModel.copyAnimationCodeToClipboard()\n                }\n            }\n            \n            Spacer()\n            \n            Divider()\n            \n            HStack {\n                \n                Picker(\"Animation:\", selection: $exampleViewModel.selectedAnimationExample) {\n                    ForEach(AnimationExample.allCases) { animation in\n                        Text(animation.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.radioGroup)\n                .padding()\n                \n                \n                Button(action: {\n                    viewModel.copyAnimationCodeToClipboard()\n                    \n                    withAnimation(.easeInOut(duration: 0.5)) {\n                        copyTextOpacity = 1\n                    }\n                    withAnimation(.easeInOut(duration: 0.5).delay(2)) {\n                        copyTextOpacity = 0\n                    }\n                }, label: {\n                    Label(\"Copy animation code\", systemImage: \"doc.on.doc.fill\")\n                        .padding(20)\n                })\n                .overlay {\n                    Text(\"Copied\")\n                        .foregroundColor(.primary)\n                        .padding(.vertical, 2)\n                        .padding(.horizontal, 8)\n                        .background(Color(\"copyPopupBackgroundColor\"))\n                        .clipShape(\n                            RoundedRectangle(cornerRadius: 4, style: .continuous)\n                        )\n                        .overlay(\n                            RoundedRectangle(cornerRadius: 4, style: .continuous)\n                                .stroke(Color.primary.opacity(0.4), lineWidth: 1)\n                        )\n                        .offset(y: -30)\n                        .frame(width: 60)\n                        .opacity(copyTextOpacity)\n                        .edgesIgnoringSafeArea(.all)\n                }\n                .padding()\n            }\n        }\n    }\n}\n\nstruct AnimationsContainerView_Previews: PreviewProvider {\n    static var previews: some View {\n        AnimationsContainerView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/AnimationViews/ChainsView.swift",
    "content": "//\n//  ChainsView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 18.03.22.\n//\n\nimport SwiftUI\n\nstruct ChainsView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    \n    let letters = Array(repeating: \"🔗\", count: 25)\n    \n    @State private var enabled = false\n    @State private var dragAmount = CGSize.zero\n    @State private var textOpacity: CGFloat = 1\n    \n    var body: some View {\n        VStack(spacing: 0) {\n            ForEach(0 ..< 25) { num in\n                Text(String(letters[num]))\n                    .padding(-10)\n                    .font(.largeTitle)\n                    .rotationEffect(.degrees(enabled ? 0 : 360), anchor: .bottom)\n                    .offset(dragAmount)\n                    .animation(\n                        .create(from: viewModel)\n                            .delay(Double(num) / 20),\n                        value: dragAmount\n                    )\n            }\n            \n            Text(\"Drag chain to animate\")\n                .foregroundColor(.secondary)\n                .padding(.top)\n                .opacity(textOpacity)\n                .animation(.easeInOut, value: textOpacity)\n        }\n        .padding()\n        .gesture(\n            DragGesture()\n                .onChanged {\n                    dragAmount = $0.translation\n                    textOpacity = 0\n                }\n            // _ ignore the value coming in this time\n                .onEnded { _ in\n                    withAnimation{\n                        dragAmount = .zero\n                        enabled.toggle()\n                    }\n                    \n                    textOpacity = 1\n                }\n        )\n    }\n}\n\nstruct ChainsView_Previews: PreviewProvider {\n    static var previews: some View {\n        ChainsView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/AnimationViews/CirclesView.swift",
    "content": "//\n//  CirclesView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 10.03.22.\n//\n\nimport SwiftUI\n\nstruct CirclesView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    \n    @State private var moving = false\n    \n    let yUpper: CGFloat = 100\n    let yLower: CGFloat = -120\n    \n    var body: some View {\n        ZStack{\n            ForEach (0 ..< 8, id: \\.self) { i in\n                Circle()\n                    .stroke(lineWidth: 5)\n                    .frame(width: CGFloat(20 + (i * 30)), height: CGFloat(20 + (i * 30)))\n                    .rotation3DEffect(.degrees(75), axis: (x: 1, y: 0, z: 0))\n                    .offset(y: moving ? yUpper : yLower)\n                    .animation(\n                        .create(from: viewModel)\n                            .repeatForever(autoreverses: true)\n                            .delay(Double(i) * 0.05),\n                        value: moving\n                    )\n            }\n        }\n        .onAppear {\n            moving.toggle()\n        }\n    }\n}\n\nstruct CirclesView_Previews: PreviewProvider {\n    static var previews: some View {\n        CirclesView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/AnimationViews/CodePreviewView.swift",
    "content": "//\n//  CodePreviewView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 24.03.22.\n//\n\nimport SwiftUI\n\nstruct CodePreviewView: View {\n    \n    var code: String\n    var copyAnimationCodeToClipboard: () -> Void\n    \n    @State private var copyButtonHovered = false\n    @State private var copyTextOpacity = 0.0\n    \n    var body: some View {\n        ZStack(alignment: .topTrailing) {\n            Text(\"```\\(code)```\")\n                .padding(40)\n                .textSelection(.enabled)\n                .background(Color(\"codePreviewBackground\"))\n                .clipShape(\n                    RoundedRectangle(cornerRadius: 10, style: .continuous)\n                )\n                .overlay(\n                    RoundedRectangle(cornerRadius: 10, style: .continuous)\n                        .stroke(Color.primary.opacity(0.5), lineWidth: 1)\n                )\n                .animation(.spring(), value: code)\n            \n            \n            Image(systemName: \"doc.on.doc\")\n                .foregroundColor(\n                    copyButtonHovered ? .primary : .secondary.opacity(0.6)\n                )\n                .padding(4)\n                .overlay(\n                    RoundedRectangle(cornerRadius: 4, style: .continuous)\n                        .stroke(\n                            copyButtonHovered ? .primary : Color.secondary.opacity(0.6),\n                            lineWidth: 1\n                        )\n                )\n                .animation(.easeInOut.speed(1.5), value: copyButtonHovered)\n                .padding(10)\n                .overlay {\n                    Text(\"Copied\")\n                        .foregroundColor(.primary)\n                        .padding(.vertical, 2)\n                        .padding(.horizontal, 8)\n                        .background(Color(\"copyPopupBackgroundColor\"))\n                        .clipShape(\n                            RoundedRectangle(cornerRadius: 4, style: .continuous)\n                        )\n                        .overlay(\n                            RoundedRectangle(cornerRadius: 4, style: .continuous)\n                                .stroke(Color.primary.opacity(0.4), lineWidth: 1)\n                        )\n                        .offset(y: -30)\n                        .frame(width: 60)\n                        .opacity(copyTextOpacity)\n                        .edgesIgnoringSafeArea(.all)\n                }\n                .onHover { hovering in\n                    copyButtonHovered = hovering\n                }\n                .onTapGesture {\n                    copyAnimationCodeToClipboard()\n                    \n                    withAnimation(.easeInOut(duration: 0.5)) {\n                        copyTextOpacity = 1\n                    }\n                    withAnimation(.easeInOut(duration: 0.5).delay(2)) {\n                        copyTextOpacity = 0\n                    }\n                }\n        }\n    }\n}\n\nstruct CodePreviewView_Previews: PreviewProvider {\n    static var previews: some View {\n        CodePreviewView(code: \"I'm a code preview. Oink, oink\") {\n            // Nothing to do here\n        }\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/AnimationViews/EmojisView.swift",
    "content": "//\n//  EmojisView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 18.03.22.\n//\n\nimport SwiftUI\n\nstruct EmojisView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    \n    let letters = Array(\"😇🥰😝😂😍🤓😜🤭😰🥰😇\")\n    let lettersAmount = 11\n    \n    @State private var enabled = false\n    @State private var dragAmount = CGSize.zero\n    @State private var textOpacity: CGFloat = 1\n    \n    var body: some View {\n        ZStack {\n            ForEach(0 ..< 11) { num in\n                Text(String(letters[num]))\n                    .font(.largeTitle)\n                    .offset(dragAmount)\n                    .animation(\n                        .create(from: viewModel)\n                            .delay(Double(num) / 10),\n                        value: dragAmount\n                    )\n            }\n            .scaleEffect(3)\n            \n            Text(\"Drag emojis to animate\")\n                .foregroundColor(.secondary)\n                .padding(.top)\n                .font(.body)\n                .offset(y: 60)\n                .opacity(textOpacity)\n                .animation(.easeInOut, value: textOpacity)\n        }\n        \n        .padding()\n        .gesture(\n            DragGesture()\n                .onChanged {\n                    dragAmount = $0.translation\n                    textOpacity = 0\n                }\n            // _ ignore the value coming in this time\n                .onEnded { _ in\n                    withAnimation{\n                        dragAmount = .zero\n                        enabled.toggle()\n                    }\n                    \n                    textOpacity = 1\n                }\n        )\n    }\n}\n\nstruct EmojisView_Previews: PreviewProvider {\n    static var previews: some View {\n        EmojisView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/AnimationViews/GradientCircleView.swift",
    "content": "//\n//  GradientCircleView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 21.03.22.\n//\n\nimport SwiftUI\n\nstruct GradientCircleView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    \n    @State private var rotationAmount: Double = -90\n    @State private var scale: CGFloat = 1\n    \n    var body: some View {\n        Circle()\n            .fill(\n                AngularGradient(\n                    colors: [.blue, .green, .orange, .red, .blue],\n                    center: .center\n                )\n            )\n            .scaleEffect(scale)\n            .animation(\n                .create(from: viewModel)\n                    .repeatForever(autoreverses: true),\n                value: scale\n            )\n            .rotationEffect(.degrees(rotationAmount))\n            .animation(\n                .create(from: viewModel)\n                    .repeatForever(autoreverses: true),\n                value: rotationAmount\n            )\n            .overlay(Circle().stroke(Color.primary.opacity(0.6), lineWidth: 3))\n            .padding(40)\n            .onAppear {\n                rotationAmount += 360\n                scale = 0.5\n            }\n    }\n}\n\nstruct GradientCircleView_Previews: PreviewProvider {\n    static var previews: some View {\n        GradientCircleView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/AnimationViews/TextAnimationView.swift",
    "content": "//\n//  TextAnimationView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 18.04.23.\n//\n\nimport SwiftUI\n\nstruct TextAnimationContainer: View {\n    @ObservedObject var viewModel: AnimationsViewModel\n    \n    @State private var text = \"Motionscape👀\"\n    \n    @FocusState private var textFieldFocused: Bool\n    \n    var body: some View {\n        VStack {\n            HStack(spacing: 8) {\n                TextField(\"Your custom text:\", text: $text)\n                    .textFieldStyle(CustomTextFieldStyle(isFocused: $textFieldFocused.wrappedValue))\n                    .focused($textFieldFocused)\n                \n                Spacer()\n                \n                Label(\"Click to change text\", systemImage: \"lightbulb\")\n            }\n            .padding(40)\n            \n            Spacer()\n            \n            TextAnimationView(text: text, animation: .create(from: viewModel))\n                .id(viewModel.id)\n            \n            Spacer()\n        }\n        .onChange(of: text) { _ in\n            viewModel.id = UUID()\n        }\n        \n    }\n}\n\nstruct TextAnimationView: View {\n    \n    @State private var offset: CGFloat = 0\n    \n    var text: String\n    var animation: Animation\n    \n    var body: some View {\n        HStack {\n            ForEach(Array(text.enumerated()), id: \\.offset) { (index, character) in\n                Text(String(character))\n                    .font(.system(size: 60, weight: .bold, design: .rounded))\n                    .offset(x: 0, y: offset)\n                    .animation(\n                        animation\n                        .repeatForever(autoreverses: true)\n                        .delay(Double(index) / 30),\n                        value: offset\n                    )\n            }\n        }\n        \n        .onAppear {\n            withAnimation {\n                offset = 100\n            }\n        }\n    }\n}\n\nstruct TextAnimationView_Previews: PreviewProvider {\n    static var previews: some View {\n        TextAnimationView(text: \"Motionscape\", animation: .easeInOut)\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/AnimationOptionsView.swift",
    "content": "//\n//  AnimationOptionsView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 25.03.22.\n//\n\nimport SwiftUI\n\nstruct AnimationOptionsView: View {\n    \n    @Binding var animationOptions: [AnimationOption]\n    \n    var body: some View {\n        VStack(alignment: .leading, spacing: 20) {\n            Text(\"Options\")\n                .font(.headline)\n            \n            ForEach($animationOptions) { $option in\n                AnimationOptionView(option: $option)\n            }\n        }\n        .padding()\n    }\n}\n\nstruct AnimationOptionsView_Previews: PreviewProvider {\n    static var previews: some View {\n        AnimationOptionsView(animationOptions: .constant([]))\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/ControlContainerView.swift",
    "content": "//\n//  ControlContainerView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport SwiftUI\n\nstruct ControlContainerView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    \n    var body: some View {\n        switch viewModel.selectedAnimation {\n        case .interpolatingSpring:\n            InterpolatingSpringControlView(viewModel: viewModel)\n        case .interactiveSpring:\n            InteractiveSpringControlView(viewModel: viewModel)\n        case .spring:\n            SpringControlView(viewModel: viewModel)\n        case .linear:\n            LinearControlView(viewModel: viewModel)\n        case .defaultAnimation:\n            DefaultControlView(viewModel: viewModel)\n        case .easeIn:\n            EaseInControlView(viewModel: viewModel)\n        case .easeOut:\n            EaseOutControlView(viewModel: viewModel)\n        case .easeInOut:\n            EaseInOutControlView(viewModel: viewModel)\n        case .timingCurve:\n            TimingCurveControlView(viewModel: viewModel)\n        case .none:\n            Text(\"No animation type selected.\")\n        }\n    }\n}\n\nstruct ControlContainerView_Previews: PreviewProvider {\n    static var previews: some View {\n        ControlContainerView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/DefaultControlView.swift",
    "content": "//\n//  DefaultControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 29.03.22.\n//\n\nimport SwiftUI\n\nstruct DefaultControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"Default\",\n                    description: \"The default animation that is provided is in fact an easeInOut animation. It has the control points (0.25, 0.1) and (0.25, 1.0).\\nUnder the parameters tab below you can see the timing curve for it. Other than that it has no other parameters you can customize.\",\n                    timingCurve: TimingCurve(x0: 0.25, y0: 0.1, x1: 0.25, y1: 1.0)\n                )\n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    Text(\"The default animation does not have any parameters to customize. Its goal is to mimic most of the system's animations as close as possible.\")\n                        .padding()\n                    \n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.defaultAnimation.animationOptions)\n                }\n                \n                HStack {\n                    Spacer()\n                    \n                    Button {\n                        // Let's see about that\n                        viewModel.resetCurrentAnimation()\n                    } label: {\n                        Text(\"Reset to default\")\n                    }\n                    \n                    Spacer()\n                }\n                .padding(.bottom)\n            }\n        }\n    }\n}\n\nstruct DefaultControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        DefaultControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/EaseInControlView.swift",
    "content": "//\n//  EaseInControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct EaseInControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"EaseIn\",\n                    description: \"This pacing causes the animation to start slowly and stop abruptly at the end. It is greater for something that exits the screen. It has the control points (0.42, 0.0) and (1.0, 1.0).\",\n                    timingCurve: TimingCurve(x0: 0.42, y0: 0.0, x1: 1.0, y1: 1.0)\n                )\n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    SliderControlView(value: $viewModel.animations.easeIn.duration, parameter: EaseIn.durationParameter)\n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.easeIn.animationOptions)\n                }\n            }\n            \n            HStack {\n                Spacer()\n                \n                Button {\n                    // Let's see about that\n                    viewModel.resetCurrentAnimation()\n                } label: {\n                    Text(\"Reset to default\")\n                }\n                \n                Spacer()\n            }\n            .padding(.bottom)\n        }\n    }\n}\n\nstruct EaseInControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        EaseInControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/EaseInOutControlView.swift",
    "content": "//\n//  EaseInOutControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct EaseInOutControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"EaseInOut\",\n                    description: \"It combines ease-in and ease-out pacing. This pacing causes an animation to start slowly, speed up in the middle and slow down again before it completes. Make it your great choice for point-to-point (back-and-forth) movement on the screen. It has the control points (0.42, 0.0) and (0.58, 1.0).\",\n                    timingCurve: TimingCurve(x0: 0.42, y0: 0.0, x1: 0.58, y1: 1.0)\n                )\n                \n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    SliderControlView(value: $viewModel.animations.easeInOut.duration, parameter: EaseInOut.durationParameter)\n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.easeInOut.animationOptions)\n                }\n                \n                HStack {\n                    Spacer()\n                    \n                    Button {\n                        // Let's see about that\n                        viewModel.resetCurrentAnimation()\n                    } label: {\n                        Text(\"Reset to default\")\n                    }\n                    \n                    Spacer()\n                }\n                .padding(.bottom)\n            }\n        }\n    }\n}\n\nstruct EaseInOutControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        EaseInOutControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/EaseOutControlView.swift",
    "content": "//\n//  EaseOutControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct EaseOutControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"EaseOut\",\n                    description: \"This pacing is the inverse of ease-in. It speeds up in the beginning and slows down at the end. It is suitable for entrance animations. Think of ease-out in the real world like when a ball is rolled on the floor towards you. You expect the ball’s movement to slow down before it gets to you. It has the control points (0.0, 0.0) and (0.58, 1.0).\",\n                    timingCurve: TimingCurve(x0: 0.0, y0: 0.0, x1: 0.58, y1: 1.0)\n                )\n                \n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    SliderControlView(value: $viewModel.animations.easeOut.duration, parameter: EaseOut.durationParameter)\n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.easeOut.animationOptions)\n                }\n                \n                HStack {\n                    Spacer()\n                    \n                    Button {\n                        // Let's see about that\n                        viewModel.resetCurrentAnimation()\n                    } label: {\n                        Text(\"Reset to default\")\n                    }\n                    \n                    Spacer()\n                }\n                .padding(.bottom)\n            }\n        }\n    }\n}\n\nstruct EaseOutControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        EaseOutControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/InteractiveSpringControlView.swift",
    "content": "//\n//  InteractiveSpringControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport SwiftUI\n\nstruct InteractiveSpringControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"Interactive spring\",\n                    description: \"A convenience for a spring() animation with a lower response value, intended for driving interactive animations.\"\n                )\n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    SliderControlView(value: $viewModel.animations.interactiveSpring.response, parameter: InteractiveSpring.responseParameter)\n                    \n                    SliderControlView(value: $viewModel.animations.interactiveSpring.dampingFraction, parameter: InteractiveSpring.dampingFractionParameter)\n                    \n                    SliderControlView(value: $viewModel.animations.interactiveSpring.blendDuration, parameter: InteractiveSpring.blendDurationParameter)\n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.interactiveSpring.animationOptions)\n                }\n                \n                HStack {\n                    Spacer()\n                    \n                    Button {\n                        // Let's see about that\n                        viewModel.resetCurrentAnimation()\n                    } label: {\n                        Text(\"Reset to default\")\n                    }\n                    \n                    Spacer()\n                }\n                .padding(.bottom)\n            }\n        }\n    }\n}\n\nstruct InteractiveSpringControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        InteractiveSpringControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/InterpolatingSpringControlView.swift",
    "content": "//\n//  InterpolatingSpringControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 10.03.22.\n//\n\nimport SwiftUI\n\nstruct InterpolatingSpringControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"Interpolating spring\",\n                    description: \"An interpolating spring animation that uses a damped spring model to produce values in the range [0, 1] that are then used to interpolate within the [from, to] range of the animated property.\\nPreserves velocity across overlapping animations by adding the effects of each animation.\"\n                )\n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    SliderControlView(value: $viewModel.animations.interpolatingSpring.stiffness, parameter: InterpolatingSpring.stiffnessParamter)\n                    \n                    SliderControlView(value: $viewModel.animations.interpolatingSpring.damping, parameter: InterpolatingSpring.dampingParameter)\n                    \n                    SliderControlView(value: $viewModel.animations.interpolatingSpring.mass, parameter: InterpolatingSpring.massParameter)\n                    \n                    SliderControlView(value: $viewModel.animations.interpolatingSpring.initialVelocity, parameter: InterpolatingSpring.initialVelocityParameter)\n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.interpolatingSpring.animationOptions)\n                }\n                \n                HStack {\n                    Spacer()\n                    \n                    Button {\n                        // Let's see about that\n                        viewModel.resetCurrentAnimation()\n                    } label: {\n                        Text(\"Reset to default\")\n                    }\n                    \n                    Spacer()\n                }\n                .padding(.bottom)\n            }\n        }\n    }\n}\n\nstruct CirclesControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        InterpolatingSpringControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/LinearControlView.swift",
    "content": "//\n//  LinearControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct LinearControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"Linear\",\n                    description: \"Linear has no speed changes over the course of the animation. There is nothing like speeding up and slowing down because the animating view moves at a constant speed. It may feel mechanical or robotic, but in some cases, it is the most suitable easing to use over ease in and ease out. It has the control points (0.0, 0.0) and (1.0, 1.0).\"\n                )\n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    SliderControlView(value: $viewModel.animations.linear.duration, parameter: Linear.durationParameter)\n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.linear.animationOptions)\n                }\n                \n                HStack {\n                    Spacer()\n                    \n                    Button {\n                        // Let's see about that\n                        viewModel.resetCurrentAnimation()\n                    } label: {\n                        Text(\"Reset to default\")\n                    }\n                    \n                    Spacer()\n                }\n                .padding(.bottom)\n            }\n        }\n    }\n}\n\nstruct LinearControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        LinearControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/SpringControlView.swift",
    "content": "//\n//  SpringControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 16.03.22.\n//\n\nimport SwiftUI\n\nstruct SpringControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"Spring\",\n                    description: \"A persistent spring animation. When mixed with other spring() or interactiveSpring() animations on the same property, each animation will be replaced by their successor, preserving velocity from one animation to the next. Optionally blends the response values between springs over a time period.\"\n                )\n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    SliderControlView(value: $viewModel.animations.spring.response, parameter: Spring.responseParameter)\n                    \n                    SliderControlView(value: $viewModel.animations.spring.dampingFraction, parameter: Spring.dampingFractionParameter)\n                    \n                    SliderControlView(value: $viewModel.animations.spring.blendDuration, parameter: Spring.blendDurationParameter)\n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.spring.animationOptions)\n                }\n                \n                HStack {\n                    Spacer()\n                    \n                    Button {\n                        // Let's see about that\n                        viewModel.resetCurrentAnimation()\n                    } label: {\n                        Text(\"Reset to default\")\n                    }\n                    \n                    Spacer()\n                }\n                .padding(.bottom)\n            }\n        }\n    }\n}\n\nstruct SpringControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        SpringControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/ControlViews/TimingCurveControlView.swift",
    "content": "//\n//  TimingCurveControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 21.03.22.\n//\n\nimport SwiftUI\n\nstruct TimingCurveControlView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    @State private var selectedOption: AnimationControlOption = .parameters\n    \n    let textFieldWidth: CGFloat = 100\n    \n    var body: some View {\n        ScrollView {\n            VStack(alignment: .leading, spacing: 20) {\n                HeadlineView(\n                    headline: \"Custom Timing Curve\",\n                    description: \"Manually create the timing curve as shown in the preview below. It is defined via two control points (as depicted in the preview) which are used to draw the curve.\")\n                \n                Picker(\"\", selection: $selectedOption) {\n                    ForEach(AnimationControlOption.allCases) { option in\n                        Text(option.rawValue.capitalized)\n                    }\n                }\n                .pickerStyle(.segmented)\n                .padding()\n                \n                switch selectedOption {\n                case .parameters:\n                    Text(\"Preview\")\n                        .font(.headline)\n                        .padding(.horizontal)\n                    HStack {\n                        Spacer()\n                    TimingCurveView(\n                        timingCurve: viewModel.animations.timingCurve\n                    )\n                    .frame(width: 200, height: 200)\n                        \n                        Spacer()\n                    }\n                    \n                    HStack(spacing: 10) {\n                        Spacer()\n                        \n                        Text(\"Control Point 1:\")\n                            .foregroundColor(.secondary)\n                        \n                        TextFieldControlView(value: $viewModel.animations.timingCurve.x0, parameter: TimingCurve.firstControlPointX)\n                            .frame(width: textFieldWidth)\n                        \n                        TextFieldControlView(value: $viewModel.animations.timingCurve.y0, parameter: TimingCurve.firstControlPointY)\n                            .frame(width: textFieldWidth)\n                        \n                        Spacer()\n                    }\n                    .padding(.top)\n                    \n                    HStack(spacing: 10) {\n                        Spacer()\n                        \n                        Text(\"Control Point 2:\")\n                            .foregroundColor(.secondary)\n                        \n                        TextFieldControlView(value: $viewModel.animations.timingCurve.x1, parameter: TimingCurve.secondControlPointX)\n                            .frame(width: textFieldWidth)\n                        \n                        TextFieldControlView(value: $viewModel.animations.timingCurve.y1, parameter: TimingCurve.secondControlPointY)\n                            .frame(width: textFieldWidth)\n                        \n                        Spacer()\n                    }\n                    \n                case .options:\n                    AnimationOptionsView(animationOptions: $viewModel.animations.timingCurve.animationOptions)\n                }\n                \n                HStack {\n                    Spacer()\n                    \n                    Button {\n                        // Let's see about that\n                        viewModel.resetCurrentAnimation()\n                    } label: {\n                        Text(\"Reset to default\")\n                    }\n                    \n                    Spacer()\n                }\n                .padding(.bottom)\n            }\n        }\n    }\n}\n\nstruct TimingCurveControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        TimingCurveControlView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/AnimationOptionView.swift",
    "content": "//\n//  AnimationOptionView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 25.03.22.\n//\n\nimport SwiftUI\n\nstruct AnimationOptionView: View {\n    \n    @Binding var option: AnimationOption\n    @State private var numberPopoverShowing = false\n    \n    var body: some View {\n        HStack(spacing: 40) {\n            Toggle(isOn: $option.active) {\n                Text(option.name)\n                    .frame(minWidth: 60)\n            }\n                .toggleStyle(.checkbox)\n            \n            Group {\n                \n                Slider(value: $option.value, in: 0...10) {\n                    Text(option.value.stringWith(places: 2))\n                        .valueLabel()\n                        .opacity(option.active ? 1.0 : 0.3)\n                        .padding(.trailing)\n                        .popover(isPresented: $numberPopoverShowing) {\n                            VStack(alignment: .leading, spacing: 20) {\n                                HStack(spacing: 20) {\n                                    Text(\"Set value:\")\n                                        .font(.headline)\n                                        .foregroundColor(.secondary)\n                                        .layoutPriority(1)\n                                    \n                                    Spacer()\n                                    \n                                    TextField(\n                                        \"Value\",\n                                        value: $option.value,\n                                        format: .number\n                                    )\n                                    .textFieldStyle(.roundedBorder)\n                                }\n                                \n                                switch option.type {\n                                case .speed:\n                                    (\n                                        Text(Image(systemName: \"info.circle\"))\n                                        +\n                                        Text(\" Values lower than 1 will slow down and larger than 1 speed up the animation.\")\n                                    )\n                                    .frame(height: 40)\n                                case .delay:\n                                    (\n                                        Text(Image(systemName: \"info.circle\"))\n                                        +\n                                        Text(\" Specifies the delay in seconds before your animation starts initially.\")\n                                    )\n                                    .frame(height: 40)\n                                }\n                                    \n                                \n                                HStack {\n                                    Spacer()\n                                    \n                                    Button(\"Ok\") {\n                                        numberPopoverShowing = false\n                                    }\n                                }\n                            }\n                            .padding()\n                            .frame(maxWidth: 300)\n                        }\n                } minimumValueLabel: {\n                    Text(\"0\")\n                        .foregroundColor(.secondary)\n                        .font(.callout)\n                        .opacity(option.active ? 1.0 : 0.3)\n                } maximumValueLabel: {\n                    Text(\"10\")\n                        .foregroundColor(.secondary)\n                        .font(.callout)\n                        .opacity(option.active ? 1.0 : 0.3)\n                }\n                \n                EditValueButton()\n                    .disabled(!option.active)\n                    .opacity(option.active ? 1.0 : 0.3)\n                    .onTapGesture {\n                        numberPopoverShowing = true\n                    }\n            }\n            .disabled(!option.active)\n        }\n        \n        ParameterDescriptionView(parameter: option)\n    }\n}\n\nstruct AnimationOptionView_Previews: PreviewProvider {\n    static var previews: some View {\n        AnimationOptionView(option: .constant(AnimationOption(type: .delay, active: false, value: 1.0, description: \"\", defaultValueDescription: \"\" , rangeDescription: \"\")))\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/CustomModifiers.swift",
    "content": "//\n//  CustomModifiers.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 24.03.22.\n//\n\nimport SwiftUI\n\nstruct ValueLabel: ViewModifier {\n    func body(content: Content) -> some View {\n        content\n            .frame(width: 60)\n            .font(.title2)\n            .padding(.horizontal, 8)\n            .padding(.vertical, 4)\n            .foregroundColor(.white)\n            .background(Color.accentColor)\n            .clipShape(RoundedRectangle(cornerRadius: 8, style: .continuous))\n            \n    }\n}\n\nstruct SidebarHeadline: ViewModifier {\n    func body(content: Content) -> some View {\n        content\n            .foregroundColor(.secondary)\n            .font(.headline)\n    }\n}\n\nextension View {\n    func valueLabel() -> some View {\n        modifier(ValueLabel())\n    }\n    \n    func sidebarHeadline() -> some View {\n        modifier(SidebarHeadline())\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/CustomTextFieldStyle.swift",
    "content": "//\n//  CustomTextFieldStyle.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 02.05.23.\n//\n\nimport SwiftUI\n\nstruct CustomTextFieldStyle: TextFieldStyle {\n    \n    var isFocused: Bool\n    \n    func _body(configuration: TextField<Self._Label>) -> some View {\n        ZStack {\n            Color(.blue)\n                .clipShape(Capsule())\n                \n            \n            configuration\n                .textFieldStyle(.plain)\n                .font(.title2)\n                .foregroundColor(.white)\n                .padding(EdgeInsets(top: 12, leading: 20, bottom: 12, trailing: 20))\n                .background(.thinMaterial, in: Capsule())\n            \n            if isFocused {\n                Capsule()\n                    .stroke(Color.white.opacity(0.7), lineWidth: 4)\n            }\n        }\n        .frame(height: 40)\n        .shadow(color: .black.opacity(0.2), radius: 8, x: 0, y: 4)\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/EditValueButton.swift",
    "content": "//\n//  EditValueButton.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 30.03.22.\n//\n\nimport SwiftUI\n\nstruct EditValueButton: View {\n    var body: some View {\n        Image(systemName: \"square.and.pencil\")\n            .resizable()\n            .frame(width: 16, height: 16)\n            .foregroundColor(.primary)\n            .accessibilityLabel(\"Set value\")\n    }\n}\n\nstruct EditValueButton_Previews: PreviewProvider {\n    static var previews: some View {\n        EditValueButton()\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/HeadlineView.swift",
    "content": "//\n//  HeadlineView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 21.03.22.\n//\n\nimport SwiftUI\n\nstruct HeadlineView: View {\n    \n    var headline: String\n    var description: String\n    var timingCurve: TimingCurve?\n    \n    var body: some View {\n        VStack(alignment: .leading) {\n            Text(headline)\n                .font(.largeTitle)\n                .foregroundColor(.secondary)\n                .bold()\n                .padding([.horizontal, .top])\n            \n            Text(description)\n                .padding()\n            \n            if let unwrappedTimingCurve = timingCurve {\n                Text(\"Timing curve\")\n                    .font(.headline)\n                    .foregroundColor(.secondary)\n                    .padding([.horizontal, .bottom])\n                \n                HStack {\n                    Spacer()\n                    \n                    TimingCurveView(timingCurve: unwrappedTimingCurve)\n                        .frame(width: 200, height: 200)\n                    \n                    Spacer()\n                }\n            }\n            \n            Divider()\n        }\n    }\n}\n\nstruct HeadlineView_Previews: PreviewProvider {\n    static var previews: some View {\n        HeadlineView(headline: \"Interpolating Spring\", description: \"An interpolating spring animation that uses a damped spring model to produce values in the range [0, 1] that are then used to interpolate within the [from, to] range of the animated property.\\nPreserves velocity across overlapping animations by adding the effects of each animation.\")\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/InfoText.swift",
    "content": "//\n//  InfoText.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport SwiftUI\n\nstruct InfoText: View {\n    \n    var text: String\n    \n    var body: some View {\n        HStack(alignment: .firstTextBaseline, spacing: 10) {\n            Image(systemName: \"info.circle\")\n            \n            Text(\"\\(text)\")\n        }\n        .padding()\n        .background(Color(.textBackgroundColor))\n        .clipShape(RoundedRectangle(cornerRadius: 10, style: .continuous))\n        .overlay(\n            RoundedRectangle(cornerRadius: 10, style: .continuous)\n                .stroke(Color.secondary.opacity(0.3), lineWidth: 2)\n        )\n    }\n}\n\nstruct InfoText_Previews: PreviewProvider {\n    static var previews: some View {\n        InfoText(text: \"The stiffness of the spring.\")\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/InfoView.swift",
    "content": "//\n//  InfoView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 29.03.22.\n//\n\nimport SwiftUI\n\nstruct InfoView: View {\n    @State private var textOpacity = 0.0\n    @State private var logoOffsetX: CGFloat = -500\n    \n    var body: some View {\n        VStack(alignment: .leading, spacing: 20) {\n            HStack {\n                VStack(alignment: .leading) {\n                    Text(\"Brought to you by\".uppercased())\n                        .font(.subheadline)\n                        .foregroundColor(.secondary)\n                    \n                    Text(\"Stream\")\n                        .font(.largeTitle)\n                }\n                .opacity(textOpacity)\n                \n                Spacer()\n                \n                Image(\"stream-logo\")\n                    .offset(x: logoOffsetX)\n            }\n            \n            Text(\"Why?\")\n                .font(.headline)\n                .foregroundColor(.secondary)\n            \n            Text(\"We love creating helpful things for developers. And we love animations. Playing around to finetune them and find just the right setting to make it awesome is what we really enjoy.\")\n                .multilineTextAlignment(.leading)\n                .frame(minHeight: 50, maxHeight: .infinity)\n            \n            Text(\"So we thought that an application to do just that would be perfect. We hope you like MotionScape as well as we do.\")\n                .multilineTextAlignment(.leading)\n                .frame(minHeight: 40, maxHeight: .infinity)\n            \n            Text(\"If you enjoy the app then please consider giving it a review on the App Store. You should also go to the repo once it's open-source and give it a star.\")\n                .multilineTextAlignment(.leading)\n                .frame(minHeight: 50, maxHeight: .infinity)\n            \n        }\n        .padding()\n        .frame(maxWidth: 500)\n        .onAppear {\n            withAnimation(.easeInOut(duration: 0.8)) {\n                textOpacity = 1\n            }\n            \n            withAnimation(.spring().speed(0.7)) {\n                logoOffsetX = 0\n            }\n        }\n    }\n}\n\nstruct InfoView_Previews: PreviewProvider {\n    static var previews: some View {\n        InfoView()\n            .environment(\\.sizeCategory, .accessibilityExtraLarge)\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/ParameterDescriptionView.swift",
    "content": "//\n//  ParameterDescriptionView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 18.03.22.\n//\n\nimport SwiftUI\n\nstruct ParameterDescriptionView: View {\n    \n    var parameter: Documentable\n    \n    @State private var detailExpanded = false\n    \n    var body: some View {\n        DisclosureGroup(isExpanded: $detailExpanded) {\n            VStack(alignment: .leading, spacing: 10) {\n                createElement(with: \"Description\", text: parameter.description)\n                \n                createElement(with: \"Default value\", text: parameter.defaultValueDescription)\n                \n                createElement(with: \"Range\", text: parameter.rangeDescription)\n            }        .padding(.top)\n        } label: {\n            HStack(spacing: 10) {\n                Text(\"Show details\")\n                \n                Image(systemName: \"info.circle\")\n            }\n            .padding(.horizontal)\n            .onTapGesture {\n                withAnimation {\n                    detailExpanded.toggle()\n                }\n            }\n        }\n    }\n    \n    func createElement(with title: String, text: String) -> some View {\n        VStack(alignment: .leading) {\n            Text(title.uppercased())\n                .font(.system(.body, design: .monospaced))\n                .bold()\n                .foregroundColor(.secondary)\n            \n            HStack(spacing: 0) {\n                Rectangle()\n                    .fill(Color.secondary)\n                    .frame(minWidth: 4, maxWidth: 4, maxHeight: .infinity)\n                    .padding(.horizontal)\n                \n                Text(text)\n                    .fixedSize(horizontal: false, vertical: true)\n            }\n        }\n        \n    }\n}\n\nstruct ParameterDescriptionView_Previews: PreviewProvider {\n    static var previews: some View {\n        ParameterDescriptionView(parameter: InterpolatingSpring.stiffnessParamter)\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/SliderControlView.swift",
    "content": "//\n//  SliderControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 14.03.22.\n//\n\nimport SwiftUI\n\nstruct SliderControlView: View {\n    \n    @Binding var value: Double\n    var parameter: AnimationParameter\n    \n    @State private var sheetShowing = false\n    @State private var numberPopoverShowing = false\n    \n    var body: some View {\n        VStack(alignment: .leading) {\n            HStack(spacing: 20) {\n                Text(parameter.name)\n                \n                Slider(value: $value, in: parameter.range)\n                \n                Text(value.stringWith(places: 2))\n                    .valueLabel()\n                    .popover(isPresented: $numberPopoverShowing) {\n                        VStack(alignment: .trailing, spacing: 20) {\n                            HStack(spacing: 20) {\n                                Text(\"Set value:\")\n                                    .font(.headline)\n                                    .foregroundColor(.secondary)\n                                    .layoutPriority(1)\n                                \n                                Spacer()\n                                \n                                TextField(\n                                    \"Value\",\n                                    value: $value,\n                                    format: .number\n                                )\n                                .textFieldStyle(.roundedBorder)\n                            }\n                            \n                            HStack(spacing: 20) {\n                                Text(\"Sensitive range:\")\n                                    .font(.headline)\n                                    .foregroundColor(.secondary)\n                                    .layoutPriority(1)\n                                \n                                Spacer()\n                                \n                                Text(parameter.range.toString)\n                            }\n                            \n                            Button(\"Ok\") {\n                                numberPopoverShowing = false\n                            }\n                        }\n                        .padding()\n                        .frame(minWidth: 300)\n                    }\n                    \n                EditValueButton()\n                    .onTapGesture {\n                        numberPopoverShowing = true\n                    }\n            }\n            \n            ParameterDescriptionView(parameter: parameter)\n                .padding(.top)\n        }\n        .padding()\n    }\n}\n\nstruct SliderControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        SliderControlView(value: .constant(4), parameter: InterpolatingSpring.stiffnessParamter)\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/TextFieldControlView.swift",
    "content": "//\n//  TextFieldControlView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 29.03.22.\n//\n\nimport SwiftUI\n\nstruct TextFieldControlView: View {\n    \n    @Binding var value: Double\n    var parameter: AnimationParameter\n    \n    let formatter: NumberFormatter = {\n        let formatter = NumberFormatter()\n        formatter.numberStyle = .decimal\n        return formatter\n    }()\n    \n    var body: some View {\n        HStack {\n            Text(parameter.name)\n            \n            TextField(parameter.name, value: $value, formatter: formatter)\n                .frame(width: 70)\n        }\n    }\n}\n\nstruct TextFieldControlView_Previews: PreviewProvider {\n    static var previews: some View {\n        TextFieldControlView(value: .constant(0.3), parameter: TimingCurve.firstControlPointX)\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/Helpers/TimingCurveView.swift",
    "content": "//\n//  TimingCurveView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 21.03.22.\n//\n\nimport SwiftUI\n\nstruct TimingCurveView: View {\n    \n    var timingCurve: TimingCurve\n    \n    let pointRadius: CGFloat = 20\n    \n    var body: some View {\n        GeometryReader { reader in\n            ZStack {\n                // Background\n                Path { path in\n                    path.move(to: CGPoint(x: 0, y: reader.size.height / 2))\n                    path.addLine(to: CGPoint(x: reader.size.width, y: reader.size.height / 2))\n                    \n                    path.move(to: CGPoint(x: reader.size.width / 2, y: 0))\n                    path.addLine(to: CGPoint(x: reader.size.width / 2, y: reader.size.height))\n                }\n                .stroke(Color.gray.opacity(0.4), lineWidth: 1)\n                \n                // Curve\n                Path { path in\n                    path.move(to: CGPoint(x: 0, y: reader.size.height))\n                    let firstPoint = createFirstControlPoint(for: timingCurve, in: reader.size)\n                    let secondPoint = createSecondControlPoint(for: timingCurve, in: reader.size)\n                    \n                    path.addCurve(\n                        to: CGPoint(\n                            x: reader.size.width,\n                            y: 0\n                        ),\n                        control1: firstPoint,\n                        control2: secondPoint\n                    )\n                }\n                .stroke(Color.accentColor, lineWidth: 6)\n                .clipShape(Rectangle())\n                \n                \n                // Control points\n                Path { path in\n                    let firstPoint = createFirstControlPoint(for: timingCurve, in: reader.size)\n                    let secondPoint = createSecondControlPoint(for: timingCurve, in: reader.size)\n                    \n                    path.addEllipse(in: CGRect(x: firstPoint.x - pointRadius / 2, y: firstPoint.y - pointRadius / 2, width: pointRadius, height: pointRadius))\n                    \n                    path.addEllipse(in: CGRect(x: secondPoint.x - pointRadius / 2, y: secondPoint.y - pointRadius / 2, width: pointRadius, height: pointRadius))\n                }\n                .fill(Color.primary)\n\n            }\n            .overlay(Rectangle().stroke(Color.secondary, lineWidth: 2))\n            \n        }\n    }\n    \n    func createFirstControlPoint(for curve: TimingCurve, in size: CGSize) -> CGPoint {\n        return CGPoint(\n            x: min(curve.x0 * size.width, size.width),\n            y: min(size.height - curve.y0 * size.height, size.height)\n        )\n    }\n    \n    func createSecondControlPoint(for curve: TimingCurve, in size: CGSize) -> CGPoint {\n        return CGPoint(\n            x: min(curve.x1 * size.width, size.width),\n            y: min(size.height - curve.y1 * size.height, size.height)\n        )\n    }\n}\n\nstruct TimingCurveView_Previews: PreviewProvider {\n    static var previews: some View {\n        Group {\n            TimingCurveView(timingCurve: TimingCurve(x0: 0.17, y0: 0.67, x1: 0.83, y1: 0.67))\n                .previewLayout(.fixed(width: 300, height: 300))\n            TimingCurveView(timingCurve: TimingCurve(x0: 0.17, y0: 0.67, x1: 0.83, y1: 0.67))\n                .previewLayout(.fixed(width: 300, height: 200))\n            TimingCurveView(timingCurve: TimingCurve(x0: 0.17, y0: 0.67, x1: 0.83, y1: 0.67))\n                .previewLayout(.fixed(width: 200, height: 300))\n        }\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/MenuViews/ContentView.swift",
    "content": "//\n//  ContentView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 07.03.22.\n//\n\nimport SwiftUI\n\nstruct ContentView: View {\n    \n    @StateObject var viewModel = AnimationsViewModel()\n    @State private var id = 0\n    \n    var body: some View {\n        NavigationView {\n            // Sidebar with the selectable items\n            SidebarView(viewModel: viewModel)\n                .frame(minWidth: .MIN_WIDTH_SIDEBAR)\n            \n            // Default view when nothing is selected\n            // (not shown if we have a default option set in the SidebarView\n            ControlContainerView(viewModel: viewModel)\n                .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n            \n            // Pane number 3, if we don't want that we can simply not provide a third view here\n            AnimationsContainerView(viewModel: viewModel)\n                .frame(minWidth: .MIN_WIDTH_ANIMATION_VIEW, maxWidth: .infinity)\n        }\n    }\n}\n\nstruct ContentView_Previews: PreviewProvider {\n    static var previews: some View {\n        ContentView()\n    }\n}\n"
  },
  {
    "path": "MotionScape/View/MenuViews/SidebarView.swift",
    "content": "//\n//  SidebarView.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 07.03.22.\n//\n\nimport SwiftUI\n\nstruct SidebarView: View {\n    \n    @ObservedObject var viewModel: AnimationsViewModel\n    \n    @State private var isInfoScreenShown = false\n    \n    var body: some View {\n        List {\n            Text(\"Spring animations\")\n                .sidebarHeadline()\n            \n            Group {\n                NavigationLink(tag: AnimationType.interpolatingSpring, selection: $viewModel.selectedAnimation, destination: {\n                    InterpolatingSpringControlView(viewModel: viewModel)\n                        .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Interpolating Spring\", systemImage: \"1.circle\")\n                })\n                \n                NavigationLink(tag: AnimationType.interactiveSpring, selection: $viewModel.selectedAnimation, destination: {\n                    InteractiveSpringControlView(viewModel: viewModel)\n                        .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Interactive Spring\", systemImage: \"2.circle\")\n                })\n                \n                NavigationLink(tag: AnimationType.spring, selection: $viewModel.selectedAnimation, destination: {\n                    SpringControlView(viewModel: viewModel)\n                        .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Spring\", systemImage: \"3.circle\")\n                })\n            }\n            \n            Text(\"Linear animation\")\n                .sidebarHeadline()\n                .padding(.top)\n            \n            Group {\n                NavigationLink(tag: AnimationType.linear, selection: $viewModel.selectedAnimation, destination: {\n                    LinearControlView(viewModel: viewModel)\n                        .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Linear\", systemImage: \"1.circle\")\n                })\n            }\n            \n            Text(\"Easing animations\")\n                .sidebarHeadline()\n                .padding(.top)\n            \n            Group{\n                \n                NavigationLink(tag: AnimationType.defaultAnimation, selection: $viewModel.selectedAnimation, destination: {\n                    DefaultControlView(viewModel: viewModel)\n                        .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Default\", systemImage: \"1.circle\")\n                })\n                \n                NavigationLink(tag: AnimationType.easeIn, selection: $viewModel.selectedAnimation, destination: {\n                    EaseInControlView(viewModel: viewModel)\n                        .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Ease In\", systemImage: \"2.circle\")\n                })\n                \n                NavigationLink(tag: AnimationType.easeOut, selection: $viewModel.selectedAnimation, destination: {\n                    EaseOutControlView(viewModel: viewModel)\n                        .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Ease Out\", systemImage: \"3.circle\")\n                })\n                \n                NavigationLink(tag: AnimationType.easeInOut, selection: $viewModel.selectedAnimation, destination: {\n                    EaseInOutControlView(viewModel: viewModel)\n                        .frame(minWidth: .MIN_WIDTH_CONTROL_VIEW, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Ease In Out\", systemImage: \"4.circle\")\n                })\n            }\n            \n            Text(\"Timing curves\")\n                .sidebarHeadline()\n                .padding(.top)\n            \n            Group {\n                NavigationLink(tag: AnimationType.timingCurve, selection: $viewModel.selectedAnimation, destination: {\n                    TimingCurveControlView(viewModel: viewModel)\n                        .frame(minWidth: 500, maxWidth: .infinity)\n                }, label: {\n                    Label(\"Custom Timing Curve\", systemImage: \"1.circle\")\n                })\n            }\n        }\n        .listStyle(SidebarListStyle())\n        .toolbar {\n            Button(action: toggleSidebar) {\n                Image(systemName: \"sidebar.left\")\n                    .help(\"Toggle Sidebar\")\n            }\n            \n            Button(action: showInfo) {\n                Image(systemName: \"info.circle\")\n                    .help(\"Show Info View\")\n            }\n            .popover(isPresented: $isInfoScreenShown, content: {\n                InfoView()\n            })\n        }\n    }\n    \n    func toggleSidebar() {\n        NSApp.keyWindow?.firstResponder?\n                .tryToPerform(#selector(NSSplitViewController.toggleSidebar(_:)), with: nil)\n    }\n    \n    func showInfo() {\n        isInfoScreenShown = true\n    }\n}\n\nstruct SidebarView_Previews: PreviewProvider {\n    static var previews: some View {\n        SidebarView(viewModel: AnimationsViewModel())\n    }\n}\n"
  },
  {
    "path": "MotionScape/ViewModel/AnimationsExampleViewModel.swift",
    "content": "//\n//  AnimationsExampleViewModel.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 18.03.22.\n//\n\nimport Foundation\n\nclass AnimationsExampleViewModel: ObservableObject {\n    \n    @Published var selectedAnimationExample: AnimationExample = .text\n    \n}\n"
  },
  {
    "path": "MotionScape/ViewModel/AnimationsViewModel.swift",
    "content": "//\n//  AnimationsViewModel.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 10.03.22.\n//\n\nimport Foundation\nimport AppKit\n\nclass AnimationsViewModel: ObservableObject {\n    \n    @Published var animations = AllAnimations() {\n        didSet {\n            id = UUID()\n        }\n    }\n    \n    @Published var id = UUID()\n    \n    @Published var selectedAnimation: AnimationType? = AnimationType.interpolatingSpring {\n        didSet {\n            id = UUID()\n        }\n    }\n    \n    \n    func copyAnimationCodeToClipboard() {\n        let pasteboard = NSPasteboard.general\n        pasteboard.declareTypes([.string], owner: nil)\n        pasteboard.setString(createAnimationCode(), forType: .string)\n    }\n    \n    func createAnimationCode() -> String {\n        switch selectedAnimation {\n        case .interpolatingSpring:\n            return animations.interpolatingSpring.createCodeSnippet()\n        case .interactiveSpring:\n            return animations.interactiveSpring.createCodeSnippet()\n        case .spring:\n            return animations.spring.createCodeSnippet()\n        case .linear:\n            return animations.linear.createCodeSnippet()\n        case .defaultAnimation:\n            return animations.defaultAnimation.createCodeSnippet()\n        case .easeIn:\n            return animations.easeIn.createCodeSnippet()\n        case .easeOut:\n            return animations.easeOut.createCodeSnippet()\n        case .easeInOut:\n            return animations.easeInOut.createCodeSnippet()\n        case .timingCurve:\n            return animations.timingCurve.createCodeSnippet()\n        case .none:\n            return \"No animation selected.\"\n        }\n    }\n    \n    func resetCurrentAnimation() {\n        switch selectedAnimation {\n        case .interpolatingSpring:\n            animations.interpolatingSpring = InterpolatingSpring()\n        case .interactiveSpring:\n            animations.interactiveSpring = InteractiveSpring()\n        case .spring:\n            animations.spring = Spring()\n        case .linear:\n            animations.linear = Linear()\n        case .defaultAnimation:\n            animations.defaultAnimation = Default()\n        case .easeIn:\n            animations.easeIn = EaseIn()\n        case .easeOut:\n            animations.easeOut = EaseOut()\n        case .easeInOut:\n            animations.easeInOut = EaseInOut()\n        case .timingCurve:\n            animations.timingCurve = TimingCurve()\n        case .none:\n            print(\"Nothing to do here\")\n        }\n    }\n}\n"
  },
  {
    "path": "MotionScape/ViewModel/CirclesViewModel.swift",
    "content": "//\n//  CirclesViewModel.swift\n//  MotionScape\n//\n//  Created by Stefan Blos on 10.03.22.\n//\n\nimport Foundation\nimport AppKit\n\nclass AnimationsViewModel: ObservableObject {\n    \n    @Published var animations = AllAnimations() {\n        didSet {\n            id = UUID()\n        }\n    }\n    \n    @Published var id = UUID()\n    \n    @Published var selectedAnimation: AnimationType? = AnimationType.interpolatingSpring\n    \n    func copyAnimationCodeToClipboard() {\n        let pasteboard = NSPasteboard.general\n        pasteboard.declareTypes([.string], owner: nil)\n        pasteboard.setString(createAnimationCode(), forType: .string)\n    }\n    \n    private func createAnimationCode() -> String {\n        switch selectedAnimation {\n        case .interpolatingSpring:\n            return \"\"\"\n.interpolatingSpring(\n    mass: \\(animations.interpolatingSpring.mass),\n    stiffness: \\(animations.interpolatingSpring.stiffness),\n    damping: \\(animations.interpolatingSpring.damping),\n    initialVelocity: \\(animations.interpolatingSpring.initialVelocity)\n)\n\"\"\"\n        case .interactiveSpring:\n            return \"\"\"\n.interactiveSpring(\n    response: \\(animations.interactiveSpring.response),\n    dampingFraction: \\(animations.interactiveSpring.dampingFraction),\n    blendDuration: \\(animations.interactiveSpring.blendDuration)\n)\n\"\"\"\n        case .spring:\n            return \"\"\"\n.spring(\n    response: \\(animations.interactiveSpring.response),\n    dampingFraction: \\(animations.interactiveSpring.dampingFraction),\n    blendDuration: \\(animations.interactiveSpring.blendDuration)\n)\n\"\"\"\n        case .linear:\n            return \"\"\"\n.linear(\n    duration: \\(animations.linear.duration)\n)\n\"\"\"\n        case .easeIn:\n            return \"\"\"\n.easeIn(\n    duration: \\(animations.easeIn.duration)\n)\n\"\"\"\n        case .easeOut:\n            return \"\"\"\n.easeOut(\n    duration: \\(animations.easeOut.duration)\n)\n\"\"\"\n        case .easeInOut:\n            return \"\"\"\n.easeInOut(\n    duration: \\(animations.easeInOut.duration)\n)\n\"\"\"\n        case .none:\n            return \"No animation selected.\"\n        }\n    }\n}\n"
  },
  {
    "path": "MotionScape.xcodeproj/project.pbxproj",
    "content": "// !$*UTF8*$!\n{\n\tarchiveVersion = 1;\n\tclasses = {\n\t};\n\tobjectVersion = 55;\n\tobjects = {\n\n/* Begin PBXBuildFile section */\n\t\tAC25E78E27E48455003AF1EE /* AnimationsContainerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC25E78D27E48455003AF1EE /* AnimationsContainerView.swift */; };\n\t\tAC25E79027E484A1003AF1EE /* AnimationExample.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC25E78F27E484A1003AF1EE /* AnimationExample.swift */; };\n\t\tAC25E79227E499F3003AF1EE /* AnimationsExampleViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC25E79127E499F3003AF1EE /* AnimationsExampleViewModel.swift */; };\n\t\tAC25E79427E49ADD003AF1EE /* AllAnimations.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC25E79327E49ADD003AF1EE /* AllAnimations.swift */; };\n\t\tAC25E79827E4A44A003AF1EE /* ParameterDescriptionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC25E79727E4A44A003AF1EE /* ParameterDescriptionView.swift */; };\n\t\tAC25E79A27E4B9E4003AF1EE /* ChainsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC25E79927E4B9E4003AF1EE /* ChainsView.swift */; };\n\t\tAC25E79C27E4BD1C003AF1EE /* EmojisView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC25E79B27E4BD1C003AF1EE /* EmojisView.swift */; };\n\t\tAC2E6AD427D6756900AEE63E /* MotionScapeApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC2E6AD327D6756900AEE63E /* MotionScapeApp.swift */; };\n\t\tAC2E6AD627D6756900AEE63E /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC2E6AD527D6756900AEE63E /* ContentView.swift */; };\n\t\tAC2E6AD827D6756A00AEE63E /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = AC2E6AD727D6756A00AEE63E /* Assets.xcassets */; };\n\t\tAC2E6ADB27D6756A00AEE63E /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = AC2E6ADA27D6756A00AEE63E /* Preview Assets.xcassets */; };\n\t\tAC2E6AE327D6791C00AEE63E /* SidebarView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC2E6AE227D6791C00AEE63E /* SidebarView.swift */; };\n\t\tAC3E898A29EE9B0500F96A6D /* TextAnimationView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC3E898929EE9B0500F96A6D /* TextAnimationView.swift */; };\n\t\tAC43367927E886F900F57ABD /* GradientCircleView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43367827E886F900F57ABD /* GradientCircleView.swift */; };\n\t\tAC43367B27E88C1900F57ABD /* TimingCurveView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43367A27E88C1900F57ABD /* TimingCurveView.swift */; };\n\t\tAC43367D27E88C2700F57ABD /* TimingCurve.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43367C27E88C2700F57ABD /* TimingCurve.swift */; };\n\t\tAC43368127E8A94D00F57ABD /* HeadlineView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43368027E8A94D00F57ABD /* HeadlineView.swift */; };\n\t\tAC43368327E8AA9600F57ABD /* TimingCurveControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43368227E8AA9600F57ABD /* TimingCurveControlView.swift */; };\n\t\tAC43368D27EC804E00F57ABD /* ClosedRange+toString.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43368C27EC804E00F57ABD /* ClosedRange+toString.swift */; };\n\t\tAC43368F27ECA7D200F57ABD /* CodePreviewView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43368E27ECA7D200F57ABD /* CodePreviewView.swift */; };\n\t\tAC43369127ECA9CD00F57ABD /* PreviewType.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43369027ECA9CD00F57ABD /* PreviewType.swift */; };\n\t\tAC43369527ECC97B00F57ABD /* AnimationOption.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43369427ECC97B00F57ABD /* AnimationOption.swift */; };\n\t\tAC43369727ECF87600F57ABD /* CustomModifiers.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC43369627ECF87600F57ABD /* CustomModifiers.swift */; };\n\t\tAC55658127E2264100279DC7 /* Spring.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55658027E2264100279DC7 /* Spring.swift */; };\n\t\tAC55658327E2275500279DC7 /* SpringControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55658227E2275500279DC7 /* SpringControlView.swift */; };\n\t\tAC55658527E2327500279DC7 /* AnimationParameter.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55658427E2327500279DC7 /* AnimationParameter.swift */; };\n\t\tAC55658727E2341A00279DC7 /* Linear.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55658627E2341A00279DC7 /* Linear.swift */; };\n\t\tAC55658927E2343700279DC7 /* EaseIn.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55658827E2343700279DC7 /* EaseIn.swift */; };\n\t\tAC55658B27E2344D00279DC7 /* EaseOut.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55658A27E2344D00279DC7 /* EaseOut.swift */; };\n\t\tAC55658D27E2346300279DC7 /* EaseInOut.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55658C27E2346300279DC7 /* EaseInOut.swift */; };\n\t\tAC55658F27E2361C00279DC7 /* LinearControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55658E27E2361C00279DC7 /* LinearControlView.swift */; };\n\t\tAC55659127E2367E00279DC7 /* EaseInControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55659027E2367E00279DC7 /* EaseInControlView.swift */; };\n\t\tAC55659327E236B900279DC7 /* EaseOutControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55659227E236B900279DC7 /* EaseOutControlView.swift */; };\n\t\tAC55659527E236EC00279DC7 /* EaseInOutControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC55659427E236EC00279DC7 /* EaseInOutControlView.swift */; };\n\t\tAC5FCF7F27FB0ACA00D32FD8 /* CGFloat+Constants.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC5FCF7E27FB0ACA00D32FD8 /* CGFloat+Constants.swift */; };\n\t\tAC7BA47427DA59B700B65EC6 /* CirclesView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC7BA47327DA59B700B65EC6 /* CirclesView.swift */; };\n\t\tAC7BA47627DA5A4C00B65EC6 /* AnimationsViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC7BA47527DA5A4C00B65EC6 /* AnimationsViewModel.swift */; };\n\t\tAC7BA47827DA5D2500B65EC6 /* InterpolatingSpringControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC7BA47727DA5D2500B65EC6 /* InterpolatingSpringControlView.swift */; };\n\t\tACA1801B27DF5B3300133D54 /* InfoText.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA1801A27DF5B3300133D54 /* InfoText.swift */; };\n\t\tACA1801D27DF620900133D54 /* SliderControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA1801C27DF620900133D54 /* SliderControlView.swift */; };\n\t\tACA1801F27DF820000133D54 /* ControlContainerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA1801E27DF820000133D54 /* ControlContainerView.swift */; };\n\t\tACA1802427DF9EBC00133D54 /* InterpolatingSpring.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA1802327DF9EBC00133D54 /* InterpolatingSpring.swift */; };\n\t\tACA1802627DF9ED100133D54 /* AnimationType.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA1802527DF9ED100133D54 /* AnimationType.swift */; };\n\t\tACA1802C27DF9F6200133D54 /* Animation+Create.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA1802B27DF9F6200133D54 /* Animation+Create.swift */; };\n\t\tACA1802F27DFA0F700133D54 /* InteractiveSpring.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA1802E27DFA0F700133D54 /* InteractiveSpring.swift */; };\n\t\tACA1803127DFA1BF00133D54 /* InteractiveSpringControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA1803027DFA1BF00133D54 /* InteractiveSpringControlView.swift */; };\n\t\tACB77E3927EDC0A70005F9EE /* AnimationOptionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACB77E3827EDC0A70005F9EE /* AnimationOptionView.swift */; };\n\t\tACC877142A010AD400FAB01A /* CustomTextFieldStyle.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACC877132A010AD400FAB01A /* CustomTextFieldStyle.swift */; };\n\t\tACD067FA27DFE20B00BDEE44 /* Double+DecimalPlaces.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACD067F927DFE20B00BDEE44 /* Double+DecimalPlaces.swift */; };\n\t\tACE1A70527F3476900CB9683 /* TextFieldControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACE1A70427F3476900CB9683 /* TextFieldControlView.swift */; };\n\t\tACE1A70727F3556200CB9683 /* DefaultControlView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACE1A70627F3556200CB9683 /* DefaultControlView.swift */; };\n\t\tACE1A70927F355A900CB9683 /* Default.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACE1A70827F355A900CB9683 /* Default.swift */; };\n\t\tACE1A70B27F363D800CB9683 /* InfoView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACE1A70A27F363D800CB9683 /* InfoView.swift */; };\n\t\tACE1A70D27F4434900CB9683 /* DefaultValues.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACE1A70C27F4434900CB9683 /* DefaultValues.swift */; };\n\t\tACE1A70F27F44AAA00CB9683 /* EditValueButton.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACE1A70E27F44AAA00CB9683 /* EditValueButton.swift */; };\n\t\tACFD222127EDCE3C005DE1CF /* MyAnimation.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACFD222027EDCE3C005DE1CF /* MyAnimation.swift */; };\n\t\tACFD222327EDCE77005DE1CF /* AnimationOptionType.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACFD222227EDCE77005DE1CF /* AnimationOptionType.swift */; };\n\t\tACFD222527EDCF55005DE1CF /* AnimationControlOption.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACFD222427EDCF55005DE1CF /* AnimationControlOption.swift */; };\n\t\tACFD222727EDD473005DE1CF /* Documentable.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACFD222627EDD473005DE1CF /* Documentable.swift */; };\n\t\tACFD222927EDDBC3005DE1CF /* AnimationOptionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACFD222827EDDBC3005DE1CF /* AnimationOptionsView.swift */; };\n/* End PBXBuildFile section */\n\n/* Begin PBXFileReference section */\n\t\tAC25E78D27E48455003AF1EE /* AnimationsContainerView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationsContainerView.swift; sourceTree = \"<group>\"; };\n\t\tAC25E78F27E484A1003AF1EE /* AnimationExample.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationExample.swift; sourceTree = \"<group>\"; };\n\t\tAC25E79127E499F3003AF1EE /* AnimationsExampleViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationsExampleViewModel.swift; sourceTree = \"<group>\"; };\n\t\tAC25E79327E49ADD003AF1EE /* AllAnimations.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AllAnimations.swift; sourceTree = \"<group>\"; };\n\t\tAC25E79727E4A44A003AF1EE /* ParameterDescriptionView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ParameterDescriptionView.swift; sourceTree = \"<group>\"; };\n\t\tAC25E79927E4B9E4003AF1EE /* ChainsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ChainsView.swift; sourceTree = \"<group>\"; };\n\t\tAC25E79B27E4BD1C003AF1EE /* EmojisView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EmojisView.swift; sourceTree = \"<group>\"; };\n\t\tAC2E6AD027D6756900AEE63E /* MotionScape.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = MotionScape.app; sourceTree = BUILT_PRODUCTS_DIR; };\n\t\tAC2E6AD327D6756900AEE63E /* MotionScapeApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MotionScapeApp.swift; sourceTree = \"<group>\"; };\n\t\tAC2E6AD527D6756900AEE63E /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = \"<group>\"; };\n\t\tAC2E6AD727D6756A00AEE63E /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = \"<group>\"; };\n\t\tAC2E6ADA27D6756A00AEE63E /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = \"Preview Assets.xcassets\"; sourceTree = \"<group>\"; };\n\t\tAC2E6ADC27D6756A00AEE63E /* MotionScape.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = MotionScape.entitlements; sourceTree = \"<group>\"; };\n\t\tAC2E6AE227D6791C00AEE63E /* SidebarView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SidebarView.swift; sourceTree = \"<group>\"; };\n\t\tAC3E898929EE9B0500F96A6D /* TextAnimationView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TextAnimationView.swift; sourceTree = \"<group>\"; };\n\t\tAC43367827E886F900F57ABD /* GradientCircleView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = GradientCircleView.swift; sourceTree = \"<group>\"; };\n\t\tAC43367A27E88C1900F57ABD /* TimingCurveView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TimingCurveView.swift; sourceTree = \"<group>\"; };\n\t\tAC43367C27E88C2700F57ABD /* TimingCurve.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TimingCurve.swift; sourceTree = \"<group>\"; };\n\t\tAC43368027E8A94D00F57ABD /* HeadlineView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = HeadlineView.swift; sourceTree = \"<group>\"; };\n\t\tAC43368227E8AA9600F57ABD /* TimingCurveControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TimingCurveControlView.swift; sourceTree = \"<group>\"; };\n\t\tAC43368C27EC804E00F57ABD /* ClosedRange+toString.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = \"ClosedRange+toString.swift\"; sourceTree = \"<group>\"; };\n\t\tAC43368E27ECA7D200F57ABD /* CodePreviewView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CodePreviewView.swift; sourceTree = \"<group>\"; };\n\t\tAC43369027ECA9CD00F57ABD /* PreviewType.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PreviewType.swift; sourceTree = \"<group>\"; };\n\t\tAC43369427ECC97B00F57ABD /* AnimationOption.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationOption.swift; sourceTree = \"<group>\"; };\n\t\tAC43369627ECF87600F57ABD /* CustomModifiers.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CustomModifiers.swift; sourceTree = \"<group>\"; };\n\t\tAC55658027E2264100279DC7 /* Spring.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Spring.swift; sourceTree = \"<group>\"; };\n\t\tAC55658227E2275500279DC7 /* SpringControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SpringControlView.swift; sourceTree = \"<group>\"; };\n\t\tAC55658427E2327500279DC7 /* AnimationParameter.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationParameter.swift; sourceTree = \"<group>\"; };\n\t\tAC55658627E2341A00279DC7 /* Linear.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Linear.swift; sourceTree = \"<group>\"; };\n\t\tAC55658827E2343700279DC7 /* EaseIn.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EaseIn.swift; sourceTree = \"<group>\"; };\n\t\tAC55658A27E2344D00279DC7 /* EaseOut.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EaseOut.swift; sourceTree = \"<group>\"; };\n\t\tAC55658C27E2346300279DC7 /* EaseInOut.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EaseInOut.swift; sourceTree = \"<group>\"; };\n\t\tAC55658E27E2361C00279DC7 /* LinearControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinearControlView.swift; sourceTree = \"<group>\"; };\n\t\tAC55659027E2367E00279DC7 /* EaseInControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EaseInControlView.swift; sourceTree = \"<group>\"; };\n\t\tAC55659227E236B900279DC7 /* EaseOutControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EaseOutControlView.swift; sourceTree = \"<group>\"; };\n\t\tAC55659427E236EC00279DC7 /* EaseInOutControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EaseInOutControlView.swift; sourceTree = \"<group>\"; };\n\t\tAC5FCF7E27FB0ACA00D32FD8 /* CGFloat+Constants.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = \"CGFloat+Constants.swift\"; sourceTree = \"<group>\"; };\n\t\tAC7BA47327DA59B700B65EC6 /* CirclesView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CirclesView.swift; sourceTree = \"<group>\"; };\n\t\tAC7BA47527DA5A4C00B65EC6 /* AnimationsViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationsViewModel.swift; sourceTree = \"<group>\"; };\n\t\tAC7BA47727DA5D2500B65EC6 /* InterpolatingSpringControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InterpolatingSpringControlView.swift; sourceTree = \"<group>\"; };\n\t\tACA1801A27DF5B3300133D54 /* InfoText.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InfoText.swift; sourceTree = \"<group>\"; };\n\t\tACA1801C27DF620900133D54 /* SliderControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SliderControlView.swift; sourceTree = \"<group>\"; };\n\t\tACA1801E27DF820000133D54 /* ControlContainerView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ControlContainerView.swift; sourceTree = \"<group>\"; };\n\t\tACA1802327DF9EBC00133D54 /* InterpolatingSpring.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InterpolatingSpring.swift; sourceTree = \"<group>\"; };\n\t\tACA1802527DF9ED100133D54 /* AnimationType.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationType.swift; sourceTree = \"<group>\"; };\n\t\tACA1802B27DF9F6200133D54 /* Animation+Create.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = \"Animation+Create.swift\"; sourceTree = \"<group>\"; };\n\t\tACA1802E27DFA0F700133D54 /* InteractiveSpring.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InteractiveSpring.swift; sourceTree = \"<group>\"; };\n\t\tACA1803027DFA1BF00133D54 /* InteractiveSpringControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InteractiveSpringControlView.swift; sourceTree = \"<group>\"; };\n\t\tACB77E3827EDC0A70005F9EE /* AnimationOptionView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationOptionView.swift; sourceTree = \"<group>\"; };\n\t\tACC877132A010AD400FAB01A /* CustomTextFieldStyle.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CustomTextFieldStyle.swift; sourceTree = \"<group>\"; };\n\t\tACD067F927DFE20B00BDEE44 /* Double+DecimalPlaces.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = \"Double+DecimalPlaces.swift\"; sourceTree = \"<group>\"; };\n\t\tACE1A70427F3476900CB9683 /* TextFieldControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TextFieldControlView.swift; sourceTree = \"<group>\"; };\n\t\tACE1A70627F3556200CB9683 /* DefaultControlView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DefaultControlView.swift; sourceTree = \"<group>\"; };\n\t\tACE1A70827F355A900CB9683 /* Default.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Default.swift; sourceTree = \"<group>\"; };\n\t\tACE1A70A27F363D800CB9683 /* InfoView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InfoView.swift; sourceTree = \"<group>\"; };\n\t\tACE1A70C27F4434900CB9683 /* DefaultValues.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DefaultValues.swift; sourceTree = \"<group>\"; };\n\t\tACE1A70E27F44AAA00CB9683 /* EditValueButton.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EditValueButton.swift; sourceTree = \"<group>\"; };\n\t\tACFD222027EDCE3C005DE1CF /* MyAnimation.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MyAnimation.swift; sourceTree = \"<group>\"; };\n\t\tACFD222227EDCE77005DE1CF /* AnimationOptionType.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationOptionType.swift; sourceTree = \"<group>\"; };\n\t\tACFD222427EDCF55005DE1CF /* AnimationControlOption.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationControlOption.swift; sourceTree = \"<group>\"; };\n\t\tACFD222627EDD473005DE1CF /* Documentable.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Documentable.swift; sourceTree = \"<group>\"; };\n\t\tACFD222827EDDBC3005DE1CF /* AnimationOptionsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnimationOptionsView.swift; sourceTree = \"<group>\"; };\n/* End PBXFileReference section */\n\n/* Begin PBXFrameworksBuildPhase section */\n\t\tAC2E6ACD27D6756900AEE63E /* Frameworks */ = {\n\t\t\tisa = PBXFrameworksBuildPhase;\n\t\t\tbuildActionMask = 2147483647;\n\t\t\tfiles = (\n\t\t\t);\n\t\t\trunOnlyForDeploymentPostprocessing = 0;\n\t\t};\n/* End PBXFrameworksBuildPhase section */\n\n/* Begin PBXGroup section */\n\t\tAC25E78C27E48353003AF1EE /* AnimationViews */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tAC7BA47327DA59B700B65EC6 /* CirclesView.swift */,\n\t\t\t\tAC25E78D27E48455003AF1EE /* AnimationsContainerView.swift */,\n\t\t\t\tAC25E79927E4B9E4003AF1EE /* ChainsView.swift */,\n\t\t\t\tAC25E79B27E4BD1C003AF1EE /* EmojisView.swift */,\n\t\t\t\tAC43367827E886F900F57ABD /* GradientCircleView.swift */,\n\t\t\t\tAC43368E27ECA7D200F57ABD /* CodePreviewView.swift */,\n\t\t\t\tAC3E898929EE9B0500F96A6D /* TextAnimationView.swift */,\n\t\t\t);\n\t\t\tpath = AnimationViews;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tAC2E6AC727D6756900AEE63E = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tAC2E6AD227D6756900AEE63E /* MotionScape */,\n\t\t\t\tAC2E6AD127D6756900AEE63E /* Products */,\n\t\t\t);\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tAC2E6AD127D6756900AEE63E /* Products */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tAC2E6AD027D6756900AEE63E /* MotionScape.app */,\n\t\t\t);\n\t\t\tname = Products;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tAC2E6AD227D6756900AEE63E /* MotionScape */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tACA1802027DF9E9800133D54 /* Model */,\n\t\t\t\tACA1802127DF9E9D00133D54 /* View */,\n\t\t\t\tACA1802227DF9EA400133D54 /* ViewModel */,\n\t\t\t\tAC2E6AD327D6756900AEE63E /* MotionScapeApp.swift */,\n\t\t\t\tAC2E6AD727D6756A00AEE63E /* Assets.xcassets */,\n\t\t\t\tAC2E6ADC27D6756A00AEE63E /* MotionScape.entitlements */,\n\t\t\t\tAC2E6AD927D6756A00AEE63E /* Preview Content */,\n\t\t\t);\n\t\t\tpath = MotionScape;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tAC2E6AD927D6756A00AEE63E /* Preview Content */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tAC2E6ADA27D6756A00AEE63E /* Preview Assets.xcassets */,\n\t\t\t);\n\t\t\tpath = \"Preview Content\";\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tACA1802027DF9E9800133D54 /* Model */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tACA1802D27DFA0CD00133D54 /* Animations */,\n\t\t\t\tACA1802A27DF9F5100133D54 /* Extensions */,\n\t\t\t\tACA1802527DF9ED100133D54 /* AnimationType.swift */,\n\t\t\t\tAC55658427E2327500279DC7 /* AnimationParameter.swift */,\n\t\t\t\tAC25E78F27E484A1003AF1EE /* AnimationExample.swift */,\n\t\t\t\tAC43369027ECA9CD00F57ABD /* PreviewType.swift */,\n\t\t\t\tAC43369427ECC97B00F57ABD /* AnimationOption.swift */,\n\t\t\t\tACFD222027EDCE3C005DE1CF /* MyAnimation.swift */,\n\t\t\t\tACFD222227EDCE77005DE1CF /* AnimationOptionType.swift */,\n\t\t\t\tACFD222427EDCF55005DE1CF /* AnimationControlOption.swift */,\n\t\t\t\tACFD222627EDD473005DE1CF /* Documentable.swift */,\n\t\t\t\tACE1A70C27F4434900CB9683 /* DefaultValues.swift */,\n\t\t\t);\n\t\t\tpath = Model;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tACA1802127DF9E9D00133D54 /* View */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tAC25E78C27E48353003AF1EE /* AnimationViews */,\n\t\t\t\tACA1802927DF9F2100133D54 /* MenuViews */,\n\t\t\t\tACA1802827DF9F0D00133D54 /* ControlViews */,\n\t\t\t\tACA1802727DF9EEF00133D54 /* Helpers */,\n\t\t\t);\n\t\t\tpath = View;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tACA1802227DF9EA400133D54 /* ViewModel */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tAC7BA47527DA5A4C00B65EC6 /* AnimationsViewModel.swift */,\n\t\t\t\tAC25E79127E499F3003AF1EE /* AnimationsExampleViewModel.swift */,\n\t\t\t);\n\t\t\tpath = ViewModel;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tACA1802727DF9EEF00133D54 /* Helpers */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tACB77E3827EDC0A70005F9EE /* AnimationOptionView.swift */,\n\t\t\t\tAC43368027E8A94D00F57ABD /* HeadlineView.swift */,\n\t\t\t\tACA1801A27DF5B3300133D54 /* InfoText.swift */,\n\t\t\t\tAC25E79727E4A44A003AF1EE /* ParameterDescriptionView.swift */,\n\t\t\t\tACA1801C27DF620900133D54 /* SliderControlView.swift */,\n\t\t\t\tAC43367A27E88C1900F57ABD /* TimingCurveView.swift */,\n\t\t\t\tAC43369627ECF87600F57ABD /* CustomModifiers.swift */,\n\t\t\t\tACE1A70427F3476900CB9683 /* TextFieldControlView.swift */,\n\t\t\t\tACE1A70A27F363D800CB9683 /* InfoView.swift */,\n\t\t\t\tACE1A70E27F44AAA00CB9683 /* EditValueButton.swift */,\n\t\t\t\tACC877132A010AD400FAB01A /* CustomTextFieldStyle.swift */,\n\t\t\t);\n\t\t\tpath = Helpers;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tACA1802827DF9F0D00133D54 /* ControlViews */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tACA1801E27DF820000133D54 /* ControlContainerView.swift */,\n\t\t\t\tAC7BA47727DA5D2500B65EC6 /* InterpolatingSpringControlView.swift */,\n\t\t\t\tACA1803027DFA1BF00133D54 /* InteractiveSpringControlView.swift */,\n\t\t\t\tAC55658227E2275500279DC7 /* SpringControlView.swift */,\n\t\t\t\tAC55658E27E2361C00279DC7 /* LinearControlView.swift */,\n\t\t\t\tAC55659027E2367E00279DC7 /* EaseInControlView.swift */,\n\t\t\t\tAC55659227E236B900279DC7 /* EaseOutControlView.swift */,\n\t\t\t\tAC55659427E236EC00279DC7 /* EaseInOutControlView.swift */,\n\t\t\t\tAC43368227E8AA9600F57ABD /* TimingCurveControlView.swift */,\n\t\t\t\tACFD222827EDDBC3005DE1CF /* AnimationOptionsView.swift */,\n\t\t\t\tACE1A70627F3556200CB9683 /* DefaultControlView.swift */,\n\t\t\t);\n\t\t\tpath = ControlViews;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tACA1802927DF9F2100133D54 /* MenuViews */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tAC2E6AD527D6756900AEE63E /* ContentView.swift */,\n\t\t\t\tAC2E6AE227D6791C00AEE63E /* SidebarView.swift */,\n\t\t\t);\n\t\t\tpath = MenuViews;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tACA1802A27DF9F5100133D54 /* Extensions */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tACA1802B27DF9F6200133D54 /* Animation+Create.swift */,\n\t\t\t\tACD067F927DFE20B00BDEE44 /* Double+DecimalPlaces.swift */,\n\t\t\t\tAC43368C27EC804E00F57ABD /* ClosedRange+toString.swift */,\n\t\t\t\tAC5FCF7E27FB0ACA00D32FD8 /* CGFloat+Constants.swift */,\n\t\t\t);\n\t\t\tpath = Extensions;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tACA1802D27DFA0CD00133D54 /* Animations */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tACA1802327DF9EBC00133D54 /* InterpolatingSpring.swift */,\n\t\t\t\tACA1802E27DFA0F700133D54 /* InteractiveSpring.swift */,\n\t\t\t\tAC55658027E2264100279DC7 /* Spring.swift */,\n\t\t\t\tAC55658627E2341A00279DC7 /* Linear.swift */,\n\t\t\t\tAC55658827E2343700279DC7 /* EaseIn.swift */,\n\t\t\t\tAC55658A27E2344D00279DC7 /* EaseOut.swift */,\n\t\t\t\tAC55658C27E2346300279DC7 /* EaseInOut.swift */,\n\t\t\t\tAC43367C27E88C2700F57ABD /* TimingCurve.swift */,\n\t\t\t\tAC25E79327E49ADD003AF1EE /* AllAnimations.swift */,\n\t\t\t\tACE1A70827F355A900CB9683 /* Default.swift */,\n\t\t\t);\n\t\t\tpath = Animations;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n/* End PBXGroup section */\n\n/* Begin PBXNativeTarget section */\n\t\tAC2E6ACF27D6756900AEE63E /* MotionScape */ = {\n\t\t\tisa = PBXNativeTarget;\n\t\t\tbuildConfigurationList = AC2E6ADF27D6756A00AEE63E /* Build configuration list for PBXNativeTarget \"MotionScape\" */;\n\t\t\tbuildPhases = (\n\t\t\t\tAC2E6ACC27D6756900AEE63E /* Sources */,\n\t\t\t\tAC2E6ACD27D6756900AEE63E /* Frameworks */,\n\t\t\t\tAC2E6ACE27D6756900AEE63E /* Resources */,\n\t\t\t);\n\t\t\tbuildRules = (\n\t\t\t);\n\t\t\tdependencies = (\n\t\t\t);\n\t\t\tname = MotionScape;\n\t\t\tproductName = MotionScape;\n\t\t\tproductReference = AC2E6AD027D6756900AEE63E /* MotionScape.app */;\n\t\t\tproductType = \"com.apple.product-type.application\";\n\t\t};\n/* End PBXNativeTarget section */\n\n/* Begin PBXProject section */\n\t\tAC2E6AC827D6756900AEE63E /* Project object */ = {\n\t\t\tisa = PBXProject;\n\t\t\tattributes = {\n\t\t\t\tBuildIndependentTargetsInParallel = 1;\n\t\t\t\tLastSwiftUpdateCheck = 1320;\n\t\t\t\tLastUpgradeCheck = 1330;\n\t\t\t\tTargetAttributes = {\n\t\t\t\t\tAC2E6ACF27D6756900AEE63E = {\n\t\t\t\t\t\tCreatedOnToolsVersion = 13.2;\n\t\t\t\t\t};\n\t\t\t\t};\n\t\t\t};\n\t\t\tbuildConfigurationList = AC2E6ACB27D6756900AEE63E /* Build configuration list for PBXProject \"MotionScape\" */;\n\t\t\tcompatibilityVersion = \"Xcode 13.0\";\n\t\t\tdevelopmentRegion = en;\n\t\t\thasScannedForEncodings = 0;\n\t\t\tknownRegions = (\n\t\t\t\ten,\n\t\t\t\tBase,\n\t\t\t);\n\t\t\tmainGroup = AC2E6AC727D6756900AEE63E;\n\t\t\tproductRefGroup = AC2E6AD127D6756900AEE63E /* Products */;\n\t\t\tprojectDirPath = \"\";\n\t\t\tprojectRoot = \"\";\n\t\t\ttargets = (\n\t\t\t\tAC2E6ACF27D6756900AEE63E /* MotionScape */,\n\t\t\t);\n\t\t};\n/* End PBXProject section */\n\n/* Begin PBXResourcesBuildPhase section */\n\t\tAC2E6ACE27D6756900AEE63E /* Resources */ = {\n\t\t\tisa = PBXResourcesBuildPhase;\n\t\t\tbuildActionMask = 2147483647;\n\t\t\tfiles = (\n\t\t\t\tAC2E6ADB27D6756A00AEE63E /* Preview Assets.xcassets in Resources */,\n\t\t\t\tAC2E6AD827D6756A00AEE63E /* Assets.xcassets in Resources */,\n\t\t\t);\n\t\t\trunOnlyForDeploymentPostprocessing = 0;\n\t\t};\n/* End PBXResourcesBuildPhase section */\n\n/* Begin PBXSourcesBuildPhase section */\n\t\tAC2E6ACC27D6756900AEE63E /* Sources */ = {\n\t\t\tisa = PBXSourcesBuildPhase;\n\t\t\tbuildActionMask = 2147483647;\n\t\t\tfiles = (\n\t\t\t\tAC3E898A29EE9B0500F96A6D /* TextAnimationView.swift in Sources */,\n\t\t\t\tACFD222727EDD473005DE1CF /* Documentable.swift in Sources */,\n\t\t\t\tACB77E3927EDC0A70005F9EE /* AnimationOptionView.swift in Sources */,\n\t\t\t\tACA1803127DFA1BF00133D54 /* InteractiveSpringControlView.swift in Sources */,\n\t\t\t\tACFD222927EDDBC3005DE1CF /* AnimationOptionsView.swift in Sources */,\n\t\t\t\tACE1A70F27F44AAA00CB9683 /* EditValueButton.swift in Sources */,\n\t\t\t\tAC55659327E236B900279DC7 /* EaseOutControlView.swift in Sources */,\n\t\t\t\tACE1A70927F355A900CB9683 /* Default.swift in Sources */,\n\t\t\t\tACC877142A010AD400FAB01A /* CustomTextFieldStyle.swift in Sources */,\n\t\t\t\tAC7BA47427DA59B700B65EC6 /* CirclesView.swift in Sources */,\n\t\t\t\tACA1802F27DFA0F700133D54 /* InteractiveSpring.swift in Sources */,\n\t\t\t\tAC43369727ECF87600F57ABD /* CustomModifiers.swift in Sources */,\n\t\t\t\tACA1801D27DF620900133D54 /* SliderControlView.swift in Sources */,\n\t\t\t\tACFD222327EDCE77005DE1CF /* AnimationOptionType.swift in Sources */,\n\t\t\t\tAC43368D27EC804E00F57ABD /* ClosedRange+toString.swift in Sources */,\n\t\t\t\tAC25E79227E499F3003AF1EE /* AnimationsExampleViewModel.swift in Sources */,\n\t\t\t\tACE1A70D27F4434900CB9683 /* DefaultValues.swift in Sources */,\n\t\t\t\tAC7BA47627DA5A4C00B65EC6 /* AnimationsViewModel.swift in Sources */,\n\t\t\t\tAC55659527E236EC00279DC7 /* EaseInOutControlView.swift in Sources */,\n\t\t\t\tAC25E79427E49ADD003AF1EE /* AllAnimations.swift in Sources */,\n\t\t\t\tAC5FCF7F27FB0ACA00D32FD8 /* CGFloat+Constants.swift in Sources */,\n\t\t\t\tACFD222527EDCF55005DE1CF /* AnimationControlOption.swift in Sources */,\n\t\t\t\tAC43368127E8A94D00F57ABD /* HeadlineView.swift in Sources */,\n\t\t\t\tAC25E79C27E4BD1C003AF1EE /* EmojisView.swift in Sources */,\n\t\t\t\tACE1A70727F3556200CB9683 /* DefaultControlView.swift in Sources */,\n\t\t\t\tACA1802627DF9ED100133D54 /* AnimationType.swift in Sources */,\n\t\t\t\tAC55658727E2341A00279DC7 /* Linear.swift in Sources */,\n\t\t\t\tAC55658D27E2346300279DC7 /* EaseInOut.swift in Sources */,\n\t\t\t\tAC43368327E8AA9600F57ABD /* TimingCurveControlView.swift in Sources */,\n\t\t\t\tAC55658527E2327500279DC7 /* AnimationParameter.swift in Sources */,\n\t\t\t\tAC43367B27E88C1900F57ABD /* TimingCurveView.swift in Sources */,\n\t\t\t\tACE1A70527F3476900CB9683 /* TextFieldControlView.swift in Sources */,\n\t\t\t\tAC55658927E2343700279DC7 /* EaseIn.swift in Sources */,\n\t\t\t\tAC7BA47827DA5D2500B65EC6 /* InterpolatingSpringControlView.swift in Sources */,\n\t\t\t\tACD067FA27DFE20B00BDEE44 /* Double+DecimalPlaces.swift in Sources */,\n\t\t\t\tACA1801F27DF820000133D54 /* ControlContainerView.swift in Sources */,\n\t\t\t\tAC43369527ECC97B00F57ABD /* AnimationOption.swift in Sources */,\n\t\t\t\tACA1802427DF9EBC00133D54 /* InterpolatingSpring.swift in Sources */,\n\t\t\t\tAC55658127E2264100279DC7 /* Spring.swift in Sources */,\n\t\t\t\tAC43368F27ECA7D200F57ABD /* CodePreviewView.swift in Sources */,\n\t\t\t\tAC25E79027E484A1003AF1EE /* AnimationExample.swift in Sources */,\n\t\t\t\tACFD222127EDCE3C005DE1CF /* MyAnimation.swift in Sources */,\n\t\t\t\tAC55658B27E2344D00279DC7 /* EaseOut.swift in Sources */,\n\t\t\t\tACA1801B27DF5B3300133D54 /* InfoText.swift in Sources */,\n\t\t\t\tACA1802C27DF9F6200133D54 /* Animation+Create.swift in Sources */,\n\t\t\t\tAC2E6AD627D6756900AEE63E /* ContentView.swift in Sources */,\n\t\t\t\tAC55659127E2367E00279DC7 /* EaseInControlView.swift in Sources */,\n\t\t\t\tAC2E6AD427D6756900AEE63E /* MotionScapeApp.swift in Sources */,\n\t\t\t\tAC25E78E27E48455003AF1EE /* AnimationsContainerView.swift in Sources */,\n\t\t\t\tACE1A70B27F363D800CB9683 /* InfoView.swift in Sources */,\n\t\t\t\tAC43367927E886F900F57ABD /* GradientCircleView.swift in Sources */,\n\t\t\t\tAC2E6AE327D6791C00AEE63E /* SidebarView.swift in Sources */,\n\t\t\t\tAC25E79A27E4B9E4003AF1EE /* ChainsView.swift in Sources */,\n\t\t\t\tAC55658F27E2361C00279DC7 /* LinearControlView.swift in Sources */,\n\t\t\t\tAC43369127ECA9CD00F57ABD /* PreviewType.swift in Sources */,\n\t\t\t\tAC43367D27E88C2700F57ABD /* TimingCurve.swift in Sources */,\n\t\t\t\tAC25E79827E4A44A003AF1EE /* ParameterDescriptionView.swift in Sources */,\n\t\t\t\tAC55658327E2275500279DC7 /* SpringControlView.swift in Sources */,\n\t\t\t);\n\t\t\trunOnlyForDeploymentPostprocessing = 0;\n\t\t};\n/* End PBXSourcesBuildPhase section */\n\n/* Begin XCBuildConfiguration section */\n\t\tAC2E6ADD27D6756A00AEE63E /* Debug */ = {\n\t\t\tisa = XCBuildConfiguration;\n\t\t\tbuildSettings = {\n\t\t\t\tALWAYS_SEARCH_USER_PATHS = NO;\n\t\t\t\tCLANG_ANALYZER_NONNULL = YES;\n\t\t\t\tCLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;\n\t\t\t\tCLANG_CXX_LANGUAGE_STANDARD = \"gnu++17\";\n\t\t\t\tCLANG_CXX_LIBRARY = \"libc++\";\n\t\t\t\tCLANG_ENABLE_MODULES = YES;\n\t\t\t\tCLANG_ENABLE_OBJC_ARC = YES;\n\t\t\t\tCLANG_ENABLE_OBJC_WEAK = YES;\n\t\t\t\tCLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;\n\t\t\t\tCLANG_WARN_BOOL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_COMMA = YES;\n\t\t\t\tCLANG_WARN_CONSTANT_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;\n\t\t\t\tCLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;\n\t\t\t\tCLANG_WARN_DOCUMENTATION_COMMENTS = YES;\n\t\t\t\tCLANG_WARN_EMPTY_BODY = YES;\n\t\t\t\tCLANG_WARN_ENUM_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_INFINITE_RECURSION = YES;\n\t\t\t\tCLANG_WARN_INT_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;\n\t\t\t\tCLANG_WARN_OBJC_LITERAL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;\n\t\t\t\tCLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;\n\t\t\t\tCLANG_WARN_RANGE_LOOP_ANALYSIS = YES;\n\t\t\t\tCLANG_WARN_STRICT_PROTOTYPES = YES;\n\t\t\t\tCLANG_WARN_SUSPICIOUS_MOVE = YES;\n\t\t\t\tCLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;\n\t\t\t\tCLANG_WARN_UNREACHABLE_CODE = YES;\n\t\t\t\tCLANG_WARN__DUPLICATE_METHOD_MATCH = YES;\n\t\t\t\tCOPY_PHASE_STRIP = NO;\n\t\t\t\tDEBUG_INFORMATION_FORMAT = dwarf;\n\t\t\t\tENABLE_STRICT_OBJC_MSGSEND = YES;\n\t\t\t\tENABLE_TESTABILITY = YES;\n\t\t\t\tGCC_C_LANGUAGE_STANDARD = gnu11;\n\t\t\t\tGCC_DYNAMIC_NO_PIC = NO;\n\t\t\t\tGCC_NO_COMMON_BLOCKS = YES;\n\t\t\t\tGCC_OPTIMIZATION_LEVEL = 0;\n\t\t\t\tGCC_PREPROCESSOR_DEFINITIONS = (\n\t\t\t\t\t\"DEBUG=1\",\n\t\t\t\t\t\"$(inherited)\",\n\t\t\t\t);\n\t\t\t\tGCC_WARN_64_TO_32_BIT_CONVERSION = YES;\n\t\t\t\tGCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;\n\t\t\t\tGCC_WARN_UNDECLARED_SELECTOR = YES;\n\t\t\t\tGCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;\n\t\t\t\tGCC_WARN_UNUSED_FUNCTION = YES;\n\t\t\t\tGCC_WARN_UNUSED_VARIABLE = YES;\n\t\t\t\tMACOSX_DEPLOYMENT_TARGET = 12.1;\n\t\t\t\tMTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE;\n\t\t\t\tMTL_FAST_MATH = YES;\n\t\t\t\tONLY_ACTIVE_ARCH = YES;\n\t\t\t\tSDKROOT = macosx;\n\t\t\t\tSWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG;\n\t\t\t\tSWIFT_OPTIMIZATION_LEVEL = \"-Onone\";\n\t\t\t};\n\t\t\tname = Debug;\n\t\t};\n\t\tAC2E6ADE27D6756A00AEE63E /* Release */ = {\n\t\t\tisa = XCBuildConfiguration;\n\t\t\tbuildSettings = {\n\t\t\t\tALWAYS_SEARCH_USER_PATHS = NO;\n\t\t\t\tCLANG_ANALYZER_NONNULL = YES;\n\t\t\t\tCLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;\n\t\t\t\tCLANG_CXX_LANGUAGE_STANDARD = \"gnu++17\";\n\t\t\t\tCLANG_CXX_LIBRARY = \"libc++\";\n\t\t\t\tCLANG_ENABLE_MODULES = YES;\n\t\t\t\tCLANG_ENABLE_OBJC_ARC = YES;\n\t\t\t\tCLANG_ENABLE_OBJC_WEAK = YES;\n\t\t\t\tCLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;\n\t\t\t\tCLANG_WARN_BOOL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_COMMA = YES;\n\t\t\t\tCLANG_WARN_CONSTANT_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;\n\t\t\t\tCLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;\n\t\t\t\tCLANG_WARN_DOCUMENTATION_COMMENTS = YES;\n\t\t\t\tCLANG_WARN_EMPTY_BODY = YES;\n\t\t\t\tCLANG_WARN_ENUM_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_INFINITE_RECURSION = YES;\n\t\t\t\tCLANG_WARN_INT_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;\n\t\t\t\tCLANG_WARN_OBJC_LITERAL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;\n\t\t\t\tCLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;\n\t\t\t\tCLANG_WARN_RANGE_LOOP_ANALYSIS = YES;\n\t\t\t\tCLANG_WARN_STRICT_PROTOTYPES = YES;\n\t\t\t\tCLANG_WARN_SUSPICIOUS_MOVE = YES;\n\t\t\t\tCLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;\n\t\t\t\tCLANG_WARN_UNREACHABLE_CODE = YES;\n\t\t\t\tCLANG_WARN__DUPLICATE_METHOD_MATCH = YES;\n\t\t\t\tCOPY_PHASE_STRIP = NO;\n\t\t\t\tDEBUG_INFORMATION_FORMAT = \"dwarf-with-dsym\";\n\t\t\t\tENABLE_NS_ASSERTIONS = NO;\n\t\t\t\tENABLE_STRICT_OBJC_MSGSEND = YES;\n\t\t\t\tGCC_C_LANGUAGE_STANDARD = gnu11;\n\t\t\t\tGCC_NO_COMMON_BLOCKS = YES;\n\t\t\t\tGCC_WARN_64_TO_32_BIT_CONVERSION = YES;\n\t\t\t\tGCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;\n\t\t\t\tGCC_WARN_UNDECLARED_SELECTOR = YES;\n\t\t\t\tGCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;\n\t\t\t\tGCC_WARN_UNUSED_FUNCTION = YES;\n\t\t\t\tGCC_WARN_UNUSED_VARIABLE = YES;\n\t\t\t\tMACOSX_DEPLOYMENT_TARGET = 12.1;\n\t\t\t\tMTL_ENABLE_DEBUG_INFO = NO;\n\t\t\t\tMTL_FAST_MATH = YES;\n\t\t\t\tSDKROOT = macosx;\n\t\t\t\tSWIFT_COMPILATION_MODE = wholemodule;\n\t\t\t\tSWIFT_OPTIMIZATION_LEVEL = \"-O\";\n\t\t\t};\n\t\t\tname = Release;\n\t\t};\n\t\tAC2E6AE027D6756A00AEE63E /* Debug */ = {\n\t\t\tisa = XCBuildConfiguration;\n\t\t\tbuildSettings = {\n\t\t\t\tASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;\n\t\t\t\tASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor;\n\t\t\t\tCODE_SIGN_ENTITLEMENTS = MotionScape/MotionScape.entitlements;\n\t\t\t\tCODE_SIGN_IDENTITY = \"Apple Development\";\n\t\t\t\tCODE_SIGN_STYLE = Automatic;\n\t\t\t\tCOMBINE_HIDPI_IMAGES = YES;\n\t\t\t\tCURRENT_PROJECT_VERSION = 2;\n\t\t\t\tDEVELOPMENT_ASSET_PATHS = \"\\\"MotionScape/Preview Content\\\"\";\n\t\t\t\tDEVELOPMENT_TEAM = EHV7XZLAHA;\n\t\t\t\tENABLE_HARDENED_RUNTIME = YES;\n\t\t\t\tENABLE_PREVIEWS = YES;\n\t\t\t\tGENERATE_INFOPLIST_FILE = YES;\n\t\t\t\tINFOPLIST_KEY_LSApplicationCategoryType = \"public.app-category.developer-tools\";\n\t\t\t\tINFOPLIST_KEY_NSHumanReadableCopyright = \"\";\n\t\t\t\tLD_RUNPATH_SEARCH_PATHS = (\n\t\t\t\t\t\"$(inherited)\",\n\t\t\t\t\t\"@executable_path/../Frameworks\",\n\t\t\t\t);\n\t\t\t\tMACOSX_DEPLOYMENT_TARGET = 12.0;\n\t\t\t\tMARKETING_VERSION = 1.0;\n\t\t\t\tPRODUCT_BUNDLE_IDENTIFIER = io.getstream.MotionScape;\n\t\t\t\tPRODUCT_NAME = \"$(TARGET_NAME)\";\n\t\t\t\tSWIFT_EMIT_LOC_STRINGS = YES;\n\t\t\t\tSWIFT_VERSION = 5.0;\n\t\t\t};\n\t\t\tname = Debug;\n\t\t};\n\t\tAC2E6AE127D6756A00AEE63E /* Release */ = {\n\t\t\tisa = XCBuildConfiguration;\n\t\t\tbuildSettings = {\n\t\t\t\tASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;\n\t\t\t\tASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor;\n\t\t\t\tCODE_SIGN_ENTITLEMENTS = MotionScape/MotionScape.entitlements;\n\t\t\t\tCODE_SIGN_IDENTITY = \"Apple Development\";\n\t\t\t\tCODE_SIGN_STYLE = Automatic;\n\t\t\t\tCOMBINE_HIDPI_IMAGES = YES;\n\t\t\t\tCURRENT_PROJECT_VERSION = 2;\n\t\t\t\tDEVELOPMENT_ASSET_PATHS = \"\\\"MotionScape/Preview Content\\\"\";\n\t\t\t\tDEVELOPMENT_TEAM = EHV7XZLAHA;\n\t\t\t\tENABLE_HARDENED_RUNTIME = YES;\n\t\t\t\tENABLE_PREVIEWS = YES;\n\t\t\t\tGENERATE_INFOPLIST_FILE = YES;\n\t\t\t\tINFOPLIST_KEY_LSApplicationCategoryType = \"public.app-category.developer-tools\";\n\t\t\t\tINFOPLIST_KEY_NSHumanReadableCopyright = \"\";\n\t\t\t\tLD_RUNPATH_SEARCH_PATHS = (\n\t\t\t\t\t\"$(inherited)\",\n\t\t\t\t\t\"@executable_path/../Frameworks\",\n\t\t\t\t);\n\t\t\t\tMACOSX_DEPLOYMENT_TARGET = 12.0;\n\t\t\t\tMARKETING_VERSION = 1.0;\n\t\t\t\tPRODUCT_BUNDLE_IDENTIFIER = io.getstream.MotionScape;\n\t\t\t\tPRODUCT_NAME = \"$(TARGET_NAME)\";\n\t\t\t\tSWIFT_EMIT_LOC_STRINGS = YES;\n\t\t\t\tSWIFT_VERSION = 5.0;\n\t\t\t};\n\t\t\tname = Release;\n\t\t};\n/* End XCBuildConfiguration section */\n\n/* Begin XCConfigurationList section */\n\t\tAC2E6ACB27D6756900AEE63E /* Build configuration list for PBXProject \"MotionScape\" */ = {\n\t\t\tisa = XCConfigurationList;\n\t\t\tbuildConfigurations = (\n\t\t\t\tAC2E6ADD27D6756A00AEE63E /* Debug */,\n\t\t\t\tAC2E6ADE27D6756A00AEE63E /* Release */,\n\t\t\t);\n\t\t\tdefaultConfigurationIsVisible = 0;\n\t\t\tdefaultConfigurationName = Release;\n\t\t};\n\t\tAC2E6ADF27D6756A00AEE63E /* Build configuration list for PBXNativeTarget \"MotionScape\" */ = {\n\t\t\tisa = XCConfigurationList;\n\t\t\tbuildConfigurations = (\n\t\t\t\tAC2E6AE027D6756A00AEE63E /* Debug */,\n\t\t\t\tAC2E6AE127D6756A00AEE63E /* Release */,\n\t\t\t);\n\t\t\tdefaultConfigurationIsVisible = 0;\n\t\t\tdefaultConfigurationName = Release;\n\t\t};\n/* End XCConfigurationList section */\n\t};\n\trootObject = AC2E6AC827D6756900AEE63E /* Project object */;\n}\n"
  },
  {
    "path": "MotionScape.xcodeproj/project.xcworkspace/contents.xcworkspacedata",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<Workspace\n   version = \"1.0\">\n   <FileRef\n      location = \"self:\">\n   </FileRef>\n</Workspace>\n"
  },
  {
    "path": "MotionScape.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<!DOCTYPE plist PUBLIC \"-//Apple//DTD PLIST 1.0//EN\" \"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">\n<plist version=\"1.0\">\n<dict>\n\t<key>IDEDidComputeMac32BitWarning</key>\n\t<true/>\n</dict>\n</plist>\n"
  },
  {
    "path": "MotionScape.xcodeproj/xcuserdata/stefanblos.xcuserdatad/xcdebugger/Breakpoints_v2.xcbkptlist",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<Bucket\n   uuid = \"88AAA6CA-73F1-4594-9246-CF76850080C1\"\n   type = \"1\"\n   version = \"2.0\">\n</Bucket>\n"
  },
  {
    "path": "MotionScape.xcodeproj/xcuserdata/stefanblos.xcuserdatad/xcschemes/xcschememanagement.plist",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<!DOCTYPE plist PUBLIC \"-//Apple//DTD PLIST 1.0//EN\" \"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">\n<plist version=\"1.0\">\n<dict>\n\t<key>SchemeUserState</key>\n\t<dict>\n\t\t<key>MotionScape.xcscheme_^#shared#^_</key>\n\t\t<dict>\n\t\t\t<key>orderHint</key>\n\t\t\t<integer>0</integer>\n\t\t</dict>\n\t</dict>\n</dict>\n</plist>\n"
  },
  {
    "path": "README.md",
    "content": "# MotionScape\n\n<section style=\"display:flex;justify-content:center;max-width:75ch;margin-left:auto;margin-right:auto\">\n    <img style=\"border-radius:2rem;width:12rem;height:12rem\" src=\"resources/logo.png\" alt=\"Logo of Motionscape\">\n    <div style=\"margin-left:2rem\">\n        <h2 style=\"border-bottom:none\">MotionScape</h2>\n        <a href=\"https://apps.apple.com/us/app/motionscape-animation-studio/id1616840951\">\n        <img src=\"resources/mac-store-download.svg\" alt=\"Go to the Mac App Store page of the app\">\n        </a>\n    </div>\n</section>\n\nMotionScape is your animations playground as a developer. You can see all animations and their parameters in effect with beautifully designed and handcrafted animation examples.\n\nGet help with custom descriptions of all parameters that help you bring light into the dark tunnel of confusing namings and concepts.\n\nBest of all: directly preview and export your settings as production-ready SwiftUI code that you can use in your apps as-is.\n\nSupercharge your apps with animations and get to know how to use them - with MotionScape!\n\n## Animation Examples\n\n<img style=\"border-radius:0.5rem\" src=\"resources/AllPreviews.gif\" alt=\"Circle, chains, emojis, gradient circle\">\n\n## Screenshots\n\n<img style=\"border-radius:0.5rem\" src=\"resources/preview-1.png\" alt=\"First preview of the app\">\n\n<img style=\"border-radius:0.5rem\" src=\"resources/preview-2.png\" alt=\"Second preview of the app\">\n\n<img style=\"border-radius:0.5rem\" src=\"resources/preview-3.png\" alt=\"Third preview of the app\">\n\n<img style=\"border-radius:0.5rem\" src=\"resources/preview-4.png\" alt=\"Fourth preview of the app\">\n\n"
  },
  {
    "path": "index.md",
    "content": "<section style=\"display:flex;justify-content:center;max-width:75ch;margin-left:auto;margin-right:auto\">\n    <img style=\"border-radius:2rem;width:12rem;height:12rem\" src=\"resources/logo.png\" alt=\"Logo of Motionscape\">\n    <div style=\"margin-left:2rem\">\n        <h2 style=\"border-bottom:none\">MotionScape</h2>\n        <a href=\"https://apps.apple.com/us/app/motionscape-animation-studio/id1616840951\">\n        <img src=\"resources/mac-store-download.svg\" alt=\"Go to the Mac App Store page of the app\">\n        </a>\n    </div>\n</section>\n\n<section style=\"max-width:75ch;margin-left:auto;margin-right:auto\">\n    <h2>Preview videos</h2>\n    <div style=\"display: grid;overflow-x: scroll;grid-gap: 1rem;grid-template-columns: repeat(4, 90%);scroll-snap-type: x mandatory;\">\n        <img style=\"border-radius:0.5rem;scroll-snap-align: center;\" src=\"resources/circlesPreview.gif\" alt=\"First preview of the app\">\n        <img style=\"border-radius:0.5rem;scroll-snap-align: center;\" src=\"resources/gradientCirclePreview.gif\" alt=\"Second preview of the app\">\n        <img style=\"border-radius:0.5rem;scroll-snap-align: center;\" src=\"resources/emojisPreview.gif\" alt=\"Third preview of the app\">\n        <img style=\"border-radius:0.5rem;scroll-snap-align: center;\" src=\"resources/chainPreview.gif\" alt=\"Fourth preview of the app\">\n    </div>\n</section>\n\n<section style=\"max-width:75ch;margin-left:auto;margin-right:auto;\">\n    <h2>What is MotionScape?</h2>\n    <p>\n    MotionScape is your animations playground as a developer. You can see\n    all animations and their parameters in effect with beautifully designed\n    and handcrafted animation examples.\n    </p>\n    <p>\n    Get help with custom descriptions of all parameters that help you bring\n    light into the dark tunnel of confusing namings and concepts.\n    </p>\n    <p>\n    Best of all: directly preview and export your settings as\n    production-ready SwiftUI code that you can use in your apps as-is.\n    </p>\n    <p>\n    Supercharge your apps with animations and get to know how to use them -\n    with MotionScape!\n    </p>\n</section>\n\n<section style=\"max-width:75ch;margin-left:auto;margin-right:auto\">\n    <h2>Screenshots</h2>\n    <div style=\"display: grid;overflow-x: scroll;grid-gap: 1rem;grid-template-columns: repeat(4, 90%);scroll-snap-type: x mandatory;\">\n        <img style=\"border-radius:0.5rem;scroll-snap-align: center;\" src=\"resources/preview-1.png\" alt=\"First preview of the app\">\n        <img style=\"border-radius:0.5rem;scroll-snap-align: center;\" src=\"resources/preview-2.png\" alt=\"Second preview of the app\">\n        <img style=\"border-radius:0.5rem;scroll-snap-align: center;\" src=\"resources/preview-3.png\" alt=\"Third preview of the app\">\n        <img style=\"border-radius:0.5rem;scroll-snap-align: center;\" src=\"resources/preview-4.png\" alt=\"Fourth preview of the app\">\n    </div>\n</section>\n\n<section style=\"max-width:75ch;margin-left:auto;margin-right:auto;\" id=\"#privacy\">\n    <h2>Privacy policy</h2>\n    <p>We do not collect, use, save, or have access to any of your personal data in MotionScape.</p>\n    <p>Also, we do not use any kind of tracking software or analytics that might lead to the collection of data from third parties.</p>\n    <!-- <p>In order to make this transparent, the entire source code of the app is open-source and <a href=\"https://github.com/GetStream/motionscape-app\">you can find it on Github to inspect.</a></p> -->\n</section>\n"
  }
]