├── Hook.coffee ├── LICENSE ├── README.md ├── hook-example-gravity.framer ├── .gitignore ├── .preview.html ├── .viewer.html ├── app.coffee ├── framer │ ├── coffee-script.js │ ├── config.json │ ├── framer.generated.js │ ├── framer.init.js │ ├── framer.js │ ├── framer.js.map │ ├── framer.modules.js │ ├── images │ │ ├── cursor-active.png │ │ ├── cursor-active@2x.png │ │ ├── cursor.png │ │ ├── cursor@2x.png │ │ ├── icon-120.png │ │ ├── icon-152.png │ │ ├── icon-180.png │ │ ├── icon-192.png │ │ └── icon-76.png │ ├── manifest.txt │ ├── metadata.json │ ├── preview.png │ ├── style.css │ └── version ├── images │ └── .gitkeep ├── index.html └── modules │ └── Hook.coffee ├── hook-example-modulator.framer ├── .gitignore ├── .viewer.html ├── app.coffee ├── framer │ ├── coffee-script.js │ ├── config.json │ ├── framer.generated.js │ ├── framer.init.js │ ├── framer.js │ ├── framer.js.map │ ├── framer.modules.js │ ├── images │ │ ├── cursor-active.png │ │ ├── cursor-active@2x.png │ │ ├── cursor.png │ │ ├── cursor@2x.png │ │ ├── icon-120.png │ │ ├── icon-152.png │ │ ├── icon-180.png │ │ ├── icon-192.png │ │ └── icon-76.png │ ├── manifest.txt │ ├── preview.png │ ├── style.css │ └── version ├── images │ ├── .gitkeep │ └── spinner.svg ├── imported │ └── app@2x │ │ ├── images │ │ └── Layer-page-meneqzew.png │ │ ├── layers.json │ │ └── layers.json.js ├── index.html └── modules │ └── Hook.coffee └── hook-example-spring.framer ├── .gitignore ├── .viewer.html ├── app.coffee ├── framer ├── coffee-script.js ├── config.json ├── framer.generated.js ├── framer.init.js ├── framer.js ├── framer.js.map ├── framer.modules.js ├── images │ ├── cursor-active.png │ ├── cursor-active@2x.png │ ├── cursor.png │ ├── cursor@2x.png │ ├── icon-120.png │ ├── icon-152.png │ ├── icon-180.png │ ├── icon-192.png │ └── icon-76.png ├── manifest.txt ├── metadata.json ├── preview.png ├── style.css └── version ├── images └── .gitkeep ├── index.html ├── modules └── Hook.coffee ├── spring-example-720.gif └── spring-example.gif /Hook.coffee: -------------------------------------------------------------------------------- 1 | ### 2 | -------------------------------------------------------------------------------- 3 | Hook module for Framer 4 | -------------------------------------------------------------------------------- 5 | 6 | by: Sigurd Mannsåker 7 | github: https://github.com/sigtm/framer-hook 8 | 9 | ················································································ 10 | 11 | 12 | The Hook module simply expands the Layer prototype, and lets you make any 13 | numeric Layer property follow another property - either its own or another 14 | object's - via a spring or gravity attraction. 15 | 16 | 17 | -------------------------------------------------------------------------------- 18 | Example: Layered animation (eased + spring) 19 | -------------------------------------------------------------------------------- 20 | 21 | myLayer = new Layer 22 | 23 | # Make our own custom property for the x property to follow 24 | myLayer.easedX = 0 25 | 26 | # Hook x to easedX via a spring 27 | myLayer.hook 28 | property: "x" 29 | targetProperty: "easedX" 30 | type: "spring(150, 15)" 31 | 32 | # Animate easedX 33 | myLayer.animate 34 | properties: 35 | easedX: 200 36 | time: 0.15 37 | curve: "cubic-bezier(0.2, 0, 0.4, 1)" 38 | 39 | NOTE: 40 | To attach both the x and y position, use "pos", "midPos" or "maxPos" as the 41 | property/targetProperty. 42 | 43 | 44 | -------------------------------------------------------------------------------- 45 | Example: Hooking property to another layer 46 | -------------------------------------------------------------------------------- 47 | 48 | target = new Layer 49 | hooked = new Layer 50 | 51 | hooked.hook 52 | property: "scale" 53 | to: target 54 | type: "spring(150, 15)" 55 | 56 | The "hooked" layer's scale will now continuously follow the target layer's scale 57 | with a spring animation. 58 | 59 | 60 | -------------------------------------------------------------------------------- 61 | layer.hook(options) 62 | -------------------------------------------------------------------------------- 63 | 64 | Options are passed as a single object, like you would for a new Layer. 65 | The options object takes the following properties: 66 | 67 | 68 | property [String] 69 | ----------------- 70 | The property you'd like to hook onto another object's property 71 | 72 | 73 | type [String] 74 | ------------- 75 | Either "spring(strength, friction)" or "gravity(strength, drag)". Only the last 76 | specified drag value is used for each property, since it is only applied to 77 | each property once (and only if it has a gravity hook applied to it.) 78 | 79 | 80 | to [Object] (Optional) 81 | ---------------------- 82 | The object to attach it to. Defaults to itself. 83 | 84 | 85 | targetProperty [String] (Optional) 86 | ---------------------------------- 87 | Specify the target object's property to follow, if you don't want to follow 88 | the same property that the hook is applied to. 89 | 90 | 91 | modulator [Function] (Optional) 92 | ------------------------------- 93 | The modulator function receives the target property's value, and lets you 94 | modify it before it is fed into the physics calculations. Useful for anything 95 | from standard Utils.modulate() type stuff to snapping and conditional values. 96 | 97 | 98 | zoom [Number] (Optional) 99 | ------------------------ 100 | This factor defines the distance that 1px represents in regards to gravity and 101 | drag calculations. Only one value is stored per layer, so specifying it 102 | overwrites its existing value. Default is 100. 103 | 104 | 105 | -------------------------------------------------------------------------------- 106 | layer.unHook(property, object) 107 | -------------------------------------------------------------------------------- 108 | 109 | This removes all hooks for a given property and target object. Example: 110 | 111 | # Hook it 112 | layer.hook 113 | property: "x" 114 | to: "otherlayer" 115 | targetProperty: "y" 116 | type: "spring(200,20)" 117 | 118 | # Unhook it 119 | layer.unHook "x", otherlayer 120 | 121 | 122 | -------------------------------------------------------------------------------- 123 | layer.onHookUpdate(delta) 124 | -------------------------------------------------------------------------------- 125 | 126 | After a layer is done applying accelerations to its hooked properties, it calls 127 | onHookUpdate() at the end of each frame, if it is defined. This is an easy way 128 | to animate or trigger other stuff, perhaps based on your layer's updated 129 | properties or velocities. 130 | 131 | The delta value from the Framer loop is passed on to onHookUpdate() as well, 132 | which is the time in seconds since the last animation frame. 133 | 134 | Note that if you unhook all your hooks, onHookUpdate() will of course no longer 135 | be called for that layer. 136 | 137 | ### 138 | 139 | 140 | # Since older versions of Safari seem to be missing String.prototype.includes() 141 | 142 | unless String.prototype.includes 143 | String::includes = (search, start) -> 144 | 'use strict' 145 | start = 0 if typeof start is 'number' 146 | 147 | if start + search.length > this.length 148 | return false; 149 | else 150 | return @indexOf(search, start) isnt -1 151 | 152 | # Expand layer 153 | 154 | Layer::hook = (config) -> 155 | 156 | throw new Error 'layer.hook() needs a property, a hook type and either a target object or target property to work' unless config.property and config.type and (config.to or config.targetProperty) 157 | 158 | # Single array for all hooks, as opposed to nested arrays per property, because performance 159 | @hooks ?= 160 | hooks: [] 161 | velocities: {} 162 | defs: 163 | zoom: 100 164 | getDrag: (velocity, drag, zoom) => 165 | velocity /= zoom 166 | # Dividing by 10 is unscientific, but it means a value of 2 equals roughly a 100g ball with 15cm radius in air 167 | drag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity) 168 | if _.isNaN(drag) then return 0 else return drag 169 | getGravity: (strength, distance, zoom) => 170 | dist = Math.max(1, distance / zoom) 171 | return strength * zoom / (dist * dist) 172 | 173 | # Update the zoom value if given 174 | @hooks.zoom = config.zoom if config.zoom 175 | 176 | # Parse physics config string 177 | f = Utils.parseFunction config.type 178 | config.type = f.name 179 | config.strength = f.args[0] 180 | config.friction = f.args[1] or 0 181 | 182 | # Default to same targetProperty on same object (hopefully you've set at least one of these to something else) 183 | config.targetProperty ?= config.property 184 | config.to ?= @ 185 | 186 | # All position accelerations are added to a single 'pos' velocity. Store actual properties so we don't have to do it again every frame 187 | 188 | if config.property.toLowerCase().includes 'pos' 189 | config.prop = 'pos' 190 | 191 | if config.property.toLowerCase().includes 'mid' 192 | config.thisX = 'midX' 193 | config.thisY = 'midY' 194 | 195 | else if config.property.toLowerCase().includes 'max' 196 | config.thisX = 'maxX' 197 | config.thisY = 'maxY' 198 | 199 | else 200 | config.thisX = 'x' 201 | config.thisY = 'y' 202 | 203 | if config.targetProperty.toLowerCase().includes 'mid' 204 | config.toX = 'midX' 205 | config.toY = 'midY' 206 | 207 | else if config.targetProperty.toLowerCase().includes 'max' 208 | config.toX = 'maxX' 209 | config.toY = 'maxY' 210 | else 211 | config.toX = 'x' 212 | config.toY = 'y' 213 | 214 | else 215 | config.prop = config.property 216 | 217 | # Save hook to @hooks array 218 | @hooks.hooks.push(config) 219 | 220 | # Create velocity property if necessary 221 | @hooks.velocities[config.prop] ?= if config.prop is 'pos' then { x: 0, y: 0 } else 0 222 | 223 | # Use Framer's animation loop, slightly more robust than requestAnimationFrame directly 224 | # Save the returned AnimationLoop reference to make sure @hookLoop isn't added multiple times per layer 225 | @hooks.emitter ?= Framer.Loop.on('render', @hookLoop, this) 226 | 227 | Layer::unHook = (property, object) -> 228 | 229 | return unless @hooks 230 | 231 | prop = if property.toLowerCase().includes 'pos' then 'pos' else property 232 | 233 | # Remove all matches 234 | @hooks.hooks = @hooks.hooks.filter (hook) -> 235 | hook.to isnt object or hook.property isnt property 236 | 237 | # If there are no hooks left, shut it down 238 | if @hooks.hooks.length is 0 239 | delete @hooks 240 | Framer.Loop.removeListener 'render', @hookLoop 241 | return 242 | 243 | # Still here? Check if there are any remaining hooks affecting same velocity 244 | remaining = @hooks.hooks.filter (hook) -> 245 | prop is hook.prop 246 | 247 | # If not, delete velocity (otherwise it won't be reset if you make new hook for same property) 248 | delete @hooks.velocities[prop] if remaining.length is 0 249 | 250 | Layer::hookLoop = (delta) -> 251 | 252 | if @hooks 253 | 254 | # Multiple hooks can affect the same property. Add accelerations to temporary object so the property's velocity is the same for all calculations within the same animation frame 255 | acceleration = {} 256 | 257 | # Save drag for each property to this object, since only most recently specified value is used for each property 258 | drag = {} 259 | 260 | # Add accelerations 261 | for hook in @hooks.hooks 262 | 263 | if hook.prop is 'pos' 264 | 265 | acceleration.pos ?= { x: 0, y: 0 } 266 | 267 | target = { x: hook.to[hook.toX], y: hook.to[hook.toY] } 268 | 269 | target = hook.modulator(target) if hook.modulator 270 | 271 | vector = 272 | x: target.x - @[hook.thisX] 273 | y: target.y - @[hook.thisY] 274 | 275 | vLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y)) 276 | 277 | if hook.type is 'spring' 278 | 279 | damper = 280 | x: -hook.friction * @hooks.velocities.pos.x 281 | y: -hook.friction * @hooks.velocities.pos.y 282 | 283 | vector.x *= hook.strength 284 | vector.y *= hook.strength 285 | 286 | acceleration.pos.x += (vector.x + damper.x) * delta 287 | acceleration.pos.y += (vector.y + damper.y) * delta 288 | 289 | else if hook.type is 'gravity' 290 | 291 | drag.pos = hook.friction 292 | 293 | gravity = @hooks.defs.getGravity(hook.strength, vLength, @hooks.defs.zoom) 294 | 295 | vector.x *= gravity / vLength 296 | vector.y *= gravity / vLength 297 | 298 | acceleration.pos.x += vector.x * delta 299 | acceleration.pos.y += vector.y * delta 300 | 301 | else 302 | 303 | acceleration[hook.prop] ?= 0 304 | 305 | target = hook.to[hook.targetProperty] 306 | 307 | target = hook.modulator(target) if hook.modulator 308 | 309 | vector = target - @[hook.prop] 310 | 311 | if hook.type is 'spring' 312 | 313 | force = vector * hook.strength 314 | damper = -hook.friction * @hooks.velocities[hook.prop] 315 | 316 | acceleration[hook.prop] += (force + damper) * delta 317 | 318 | 319 | else if hook.type is 'gravity' 320 | 321 | drag[hook.prop] = hook.friction 322 | 323 | force = @hooks.defs.getGravity(hook.strength, vector, @hooks.defs.zoom) 324 | 325 | acceleration[hook.prop] += force * delta 326 | 327 | 328 | # Add velocities to properties. Doing this at the end in case there are multiple hooks affecting the same velocity 329 | for prop, velocity of @hooks.velocities 330 | 331 | if prop is 'pos' 332 | 333 | # Add drag, if it exists 334 | if drag.pos 335 | velocity.x += @hooks.defs.getDrag(velocity.x, drag.pos, @hooks.defs.zoom) 336 | velocity.y += @hooks.defs.getDrag(velocity.y, drag.pos, @hooks.defs.zoom) 337 | 338 | # Add acceleration to velocity 339 | velocity.x += acceleration.pos.x 340 | velocity.y += acceleration.pos.y 341 | 342 | # Add velocity to position 343 | @x += velocity.x * delta 344 | @y += velocity.y * delta 345 | 346 | else 347 | 348 | # Add drag, if it exists 349 | if drag[prop] 350 | @hooks.velocities[prop] += @hooks.defs.getDrag(@hooks.velocities[prop], drag[prop], @hooks.defs.zoom) 351 | 352 | # Add acceleration to velocity 353 | @hooks.velocities[prop] += acceleration[prop] 354 | 355 | # Add velocity to property 356 | @[prop] += @hooks.velocities[prop] * delta 357 | 358 | @onHookUpdate?(delta) -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2016 sigtm 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Hook module for Framer 2 | ![Spring example](http://www.sigurd.io/framer-hook/hook-example-spring.framer/spring-example-720.gif) 3 | 4 | The Hook module expands Framer's Layer prototype, and lets you make any numeric Layer property follow another property - either its own or another object's - via a spring or gravity attraction. 5 | 6 | Enough chat. Examples: 7 | 8 | [Example 1: Easing + spring](http://www.sigurd.io/framer-hook/hook-example-spring.framer/) 9 | 10 | [Example 2: Modulation from one property type to another](http://www.sigurd.io/framer-hook/hook-example-modulator.framer/) 11 | 12 | [Example 3: Gravity, too](http://www.sigurd.io/framer-hook/hook-example-gravity.framer/) 13 | 14 | The original use case was to layer a spring animation on top of an eased animation to give more control over the timing and feel of a transition, as seen in Example 1. You do not need two layers to achieve this though, as shown in the first code snippet below. 15 | 16 | I'm not a developer nor a mathematician, so much of this is improvised, particularly the physics. Please do let me know, or create a pull request, if you have any suggestions for improvements. 17 | 18 | For a more detailed documentation, check the comments at the top of Hook.coffee. 19 | 20 | 21 | ### Example: Layered animation (eased + spring) 22 | 23 | ``` 24 | myLayer = new Layer 25 | 26 | # Make our own custom property for the x property to follow 27 | myLayer.easedX = 0 28 | 29 | # Hook x to easedX via a spring 30 | myLayer.hook 31 | property: "x" 32 | targetProperty: "easedX" 33 | type: "spring(150, 15)" 34 | 35 | # Animate easedX 36 | myLayer.animate 37 | properties: 38 | easedX: 200 39 | time: 0.15 40 | curve: "cubic-bezier(0.2, 0, 0.4, 1)" 41 | ``` 42 | 43 | NOTE: 44 | To attach both the x and y position, use "pos", "midPos" or "maxPos" as the 45 | property/targetProperty. 46 | 47 | 48 | ### Example: Hooking property to another layer 49 | 50 | ``` 51 | target = new Layer 52 | hooked = new Layer 53 | 54 | hooked.hook 55 | property: "scale" 56 | to: target 57 | type: "spring(150, 15)" 58 | ``` 59 | 60 | The "hooked" layer's scale will now continuously follow the target layer's scale 61 | with a spring animation. 62 | 63 | 64 | ### Example: Adding a modulator function 65 | 66 | The modulator function allows you to do anything you want with the target property's value before it is applied. As a very basic example, let's say you want to convert one layer's y position into a corresponding scale value for another layer: 67 | 68 | ``` 69 | target = new Layer 70 | hooked = new Layer 71 | 72 | hooked.hook 73 | property: "scale" 74 | to: target 75 | targetProperty: "y" 76 | type: "spring(200,20)" 77 | modulator: (input) -> Utils.modulate(input, [0, 400], [0.5, 1]) 78 | ``` 79 | 80 | ### Documentation 81 | 82 | A more thorough documentation is included in the comments at the top of Hook.coffee. -------------------------------------------------------------------------------- /hook-example-gravity.framer/.gitignore: -------------------------------------------------------------------------------- 1 | # Framer Git Ignore 2 | 3 | # General OSX 4 | 5 | .DS_Store 6 | .AppleDouble 7 | .LSOverride 8 | 9 | # Icon must end with two \r 10 | Icon 11 | 12 | 13 | # Thumbnails 14 | ._* 15 | 16 | # Files that might appear in the root of a volume 17 | .DocumentRevisions-V100 18 | .fseventsd 19 | .Spotlight-V100 20 | .TemporaryItems 21 | .Trashes 22 | .VolumeIcon.icns 23 | 24 | # Directories potentially created on remote AFP share 25 | .AppleDB 26 | .AppleDesktop 27 | Network Trash Folder 28 | Temporary Items 29 | .apdisk 30 | 31 | 32 | # Framer Specific 33 | .temp.html 34 | framer/*.old* 35 | framer/backup.coffee 36 | framer/backups/* 37 | framer/.*.hash 38 | -------------------------------------------------------------------------------- /hook-example-gravity.framer/.preview.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | -------------------------------------------------------------------------------- /hook-example-gravity.framer/.viewer.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | -------------------------------------------------------------------------------- /hook-example-gravity.framer/app.coffee: -------------------------------------------------------------------------------- 1 | ### 2 | 3 | Hook example 4 | -------------- 5 | 6 | The Hook module simply expands the Layer prototype, and lets you make any 7 | numeric Layer property follow a property on another object via a spring or a 8 | gravity attraction. Check the comments in modules/Hook.coffee for a more thorough 9 | documentation. 10 | 11 | ### 12 | 13 | 14 | 15 | # Require Hook. No exports, it just adds methods to the Layer prototype 16 | # -------------------------------------------------------------------------------- 17 | 18 | require "Hook" 19 | 20 | 21 | 22 | # Settings 23 | # -------------------------------------------------------------------------------- 24 | 25 | gravity = 200 26 | drag = 2 27 | burstSize = 20 28 | 29 | colors = 30 | background: "#260355" 31 | orange: "#f90" 32 | blue: "#63ffff" 33 | primary: "white" 34 | secondary: "rgba(255,255,255,0.1)" 35 | 36 | easeInOut = "cubic-bezier(0.2, 0, 0.4, 1)" 37 | 38 | 39 | 40 | # Set background and defaults 41 | # -------------------------------------------------------------------------------- 42 | 43 | Framer.Device.viewport.backgroundColor = colors.background 44 | 45 | Framer.Defaults.Animation = 46 | curve: easeInOut 47 | time: 0.05 48 | 49 | 50 | 51 | # Set up example 52 | # -------------------------------------------------------------------------------- 53 | 54 | magnet = new Layer 55 | borderRadius: 100 56 | backgroundColor: colors.primary 57 | width: 80 58 | height: 80 59 | 60 | magnet.center() 61 | 62 | magnet.draggable.enabled = true 63 | 64 | magnet.onTouchStart -> 65 | 66 | @animate 67 | properties: 68 | scale: 0.95 69 | 70 | for i in [0...burstSize] 71 | 72 | bubble = new Layer 73 | x: @midX 74 | y: @midY 75 | index: 0 76 | borderRadius: 5 77 | backgroundColor: if (i % 2) then colors.blue else colors.orange 78 | width: 10 79 | height: 10 80 | 81 | bubble.hook 82 | property: 'midPos' 83 | to: magnet 84 | type: 'gravity(' + gravity + ',' + drag + ')' 85 | 86 | initialAngle = Math.random() * Math.PI * 2 87 | 88 | bubble.hooks.velocities.pos.x += Math.cos(initialAngle) * 3000 89 | bubble.hooks.velocities.pos.y += Math.sin(initialAngle) * 3000 90 | 91 | 92 | magnet.onTouchEnd -> 93 | @animate 94 | properties: 95 | scale: 1 96 | -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/config.json: -------------------------------------------------------------------------------- 1 | { 2 | "propertyPanelToggleStates" : { 3 | 4 | }, 5 | "deviceOrientation" : 0, 6 | "sharedPrototype" : 0, 7 | "contentScale" : 1, 8 | "deviceType" : "apple-iphone-6s-silver", 9 | "selectedHand" : "", 10 | "updateDelay" : 0.3, 11 | "deviceScale" : "fit", 12 | "foldedCodeRanges" : [ 13 | 14 | ], 15 | "orientation" : 0, 16 | "fullScreen" : false 17 | } -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/framer.generated.js: -------------------------------------------------------------------------------- 1 | // This is autogenerated by Framer 2 | 3 | 4 | if (!window.Framer && window._bridge) {window._bridge('runtime.error', {message:'[framer.js] Framer library missing or corrupt. Select File → Update Framer Library.'})} 5 | if (DeviceComponent) {DeviceComponent.Devices["iphone-6-silver"].deviceImageJP2 = false}; 6 | if (window.Framer) {window.Framer.Defaults.DeviceView = {"deviceScale":"fit","selectedHand":"","deviceType":"apple-iphone-6s-silver","contentScale":1,"orientation":0}; 7 | } 8 | if (window.Framer) {window.Framer.Defaults.DeviceComponent = {"deviceScale":"fit","selectedHand":"","deviceType":"apple-iphone-6s-silver","contentScale":1,"orientation":0}; 9 | } 10 | window.FramerStudioInfo = {"deviceImagesUrl":"\/_server\/resources\/DeviceImages","documentTitle":"hook-example-gravity.framer"}; 11 | 12 | Framer.Device = new Framer.DeviceView(); 13 | Framer.Device.setupContext(); -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/framer.init.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | 3 | function isFileLoadingAllowed() { 4 | return (window.location.protocol.indexOf("file") == -1) 5 | } 6 | 7 | function isHomeScreened() { 8 | return ("standalone" in window.navigator) && window.navigator.standalone == true 9 | } 10 | 11 | function isCompatibleBrowser() { 12 | return Utils.isWebKit() 13 | } 14 | 15 | var alertNode; 16 | 17 | function dismissAlert() { 18 | alertNode.parentElement.removeChild(alertNode) 19 | loadProject() 20 | } 21 | 22 | function showAlert(html) { 23 | 24 | alertNode = document.createElement("div") 25 | 26 | alertNode.classList.add("framerAlertBackground") 27 | alertNode.innerHTML = html 28 | 29 | document.addEventListener("DOMContentLoaded", function(event) { 30 | document.body.appendChild(alertNode) 31 | }) 32 | 33 | window.dismissAlert = dismissAlert; 34 | } 35 | 36 | function showBrowserAlert() { 37 | var html = "" 38 | html += "
" 39 | html += "Error: Not A WebKit Browser" 40 | html += "Your browser is not supported.
Please use Safari or Chrome.
" 41 | html += "Try anyway" 42 | html += "
" 43 | 44 | showAlert(html) 45 | } 46 | 47 | function showFileLoadingAlert() { 48 | var html = "" 49 | html += "
" 50 | html += "Error: Local File Restrictions" 51 | html += "Preview this prototype with Framer Mirror or learn more about " 52 | html += "file restrictions.
" 53 | html += "Try anyway" 54 | html += "
" 55 | 56 | showAlert(html) 57 | } 58 | 59 | function loadProject() { 60 | CoffeeScript.load("app.coffee") 61 | } 62 | 63 | function setDefaultPageTitle() { 64 | // If no title was set we set it to the project folder name so 65 | // you get a nice name on iOS if you bookmark to desktop. 66 | document.addEventListener("DOMContentLoaded", function() { 67 | if (document.title == "") { 68 | if (window.FramerStudioInfo && window.FramerStudioInfo.documentTitle) { 69 | document.title = window.FramerStudioInfo.documentTitle 70 | } else { 71 | document.title = window.location.pathname.replace(/\//g, "") 72 | } 73 | } 74 | }) 75 | } 76 | 77 | function init() { 78 | 79 | if (Utils.isFramerStudio()) { 80 | return 81 | } 82 | 83 | setDefaultPageTitle() 84 | 85 | if (!isCompatibleBrowser()) { 86 | return showBrowserAlert() 87 | } 88 | 89 | if (!isFileLoadingAllowed()) { 90 | return showFileLoadingAlert() 91 | } 92 | 93 | loadProject() 94 | 95 | } 96 | 97 | init() 98 | 99 | })() 100 | -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/framer.modules.js: -------------------------------------------------------------------------------- 1 | require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o this.length) { 146 | return false; 147 | } else { 148 | return this.indexOf(search, start) !== -1; 149 | } 150 | }; 151 | } 152 | 153 | Layer.prototype.hook = function(config) { 154 | var base, base1, f, name; 155 | if (!(config.property && config.type && (config.to || config.targetProperty))) { 156 | throw new Error('layer.hook() needs a property, a hook type and either a target object or target property to work'); 157 | } 158 | if (this.hooks == null) { 159 | this.hooks = { 160 | hooks: [], 161 | velocities: {}, 162 | defs: { 163 | zoom: 100, 164 | getDrag: (function(_this) { 165 | return function(velocity, drag, zoom) { 166 | velocity /= zoom; 167 | drag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity); 168 | if (_.isNaN(drag)) { 169 | return 0; 170 | } else { 171 | return drag; 172 | } 173 | }; 174 | })(this), 175 | getGravity: (function(_this) { 176 | return function(strength, distance, zoom) { 177 | var dist; 178 | dist = Math.max(1, distance / zoom); 179 | return strength * zoom / (dist * dist); 180 | }; 181 | })(this) 182 | } 183 | }; 184 | } 185 | if (config.zoom) { 186 | this.hooks.zoom = config.zoom; 187 | } 188 | f = Utils.parseFunction(config.type); 189 | config.type = f.name; 190 | config.strength = f.args[0]; 191 | config.friction = f.args[1] || 0; 192 | if (config.targetProperty == null) { 193 | config.targetProperty = config.property; 194 | } 195 | if (config.to == null) { 196 | config.to = this; 197 | } 198 | if (config.property.toLowerCase().includes('pos')) { 199 | config.prop = 'pos'; 200 | if (config.property.toLowerCase().includes('mid')) { 201 | config.thisX = 'midX'; 202 | config.thisY = 'midY'; 203 | } else if (config.property.toLowerCase().includes('max')) { 204 | config.thisX = 'maxX'; 205 | config.thisY = 'maxY'; 206 | } else { 207 | config.thisX = 'x'; 208 | config.thisY = 'y'; 209 | } 210 | if (config.targetProperty.toLowerCase().includes('mid')) { 211 | config.toX = 'midX'; 212 | config.toY = 'midY'; 213 | } else if (config.targetProperty.toLowerCase().includes('max')) { 214 | config.toX = 'maxX'; 215 | config.toY = 'maxY'; 216 | } else { 217 | config.toX = 'x'; 218 | config.toY = 'y'; 219 | } 220 | } else { 221 | config.prop = config.property; 222 | } 223 | this.hooks.hooks.push(config); 224 | if ((base = this.hooks.velocities)[name = config.prop] == null) { 225 | base[name] = config.prop === 'pos' ? { 226 | x: 0, 227 | y: 0 228 | } : 0; 229 | } 230 | return (base1 = this.hooks).emitter != null ? base1.emitter : base1.emitter = Framer.Loop.on('render', this.hookLoop, this); 231 | }; 232 | 233 | Layer.prototype.unHook = function(property, object) { 234 | var prop, remaining; 235 | if (!this.hooks) { 236 | return; 237 | } 238 | prop = property.toLowerCase().includes('pos') ? 'pos' : property; 239 | this.hooks.hooks = this.hooks.hooks.filter(function(hook) { 240 | return hook.to !== object || hook.property !== property; 241 | }); 242 | if (this.hooks.hooks.length === 0) { 243 | delete this.hooks; 244 | Framer.Loop.removeListener('render', this.hookLoop); 245 | return; 246 | } 247 | remaining = this.hooks.hooks.filter(function(hook) { 248 | return prop === hook.prop; 249 | }); 250 | if (remaining.length === 0) { 251 | return delete this.hooks.velocities[prop]; 252 | } 253 | }; 254 | 255 | Layer.prototype.hookLoop = function(delta) { 256 | var acceleration, damper, drag, force, gravity, hook, i, len, name, prop, ref, ref1, target, vLength, vector, velocity; 257 | if (this.hooks) { 258 | acceleration = {}; 259 | drag = {}; 260 | ref = this.hooks.hooks; 261 | for (i = 0, len = ref.length; i < len; i++) { 262 | hook = ref[i]; 263 | if (hook.prop === 'pos') { 264 | if (acceleration.pos == null) { 265 | acceleration.pos = { 266 | x: 0, 267 | y: 0 268 | }; 269 | } 270 | target = { 271 | x: hook.to[hook.toX], 272 | y: hook.to[hook.toY] 273 | }; 274 | if (hook.modulator) { 275 | target = hook.modulator(target); 276 | } 277 | vector = { 278 | x: target.x - this[hook.thisX], 279 | y: target.y - this[hook.thisY] 280 | }; 281 | vLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y)); 282 | if (hook.type === 'spring') { 283 | damper = { 284 | x: -hook.friction * this.hooks.velocities.pos.x, 285 | y: -hook.friction * this.hooks.velocities.pos.y 286 | }; 287 | vector.x *= hook.strength; 288 | vector.y *= hook.strength; 289 | acceleration.pos.x += (vector.x + damper.x) * delta; 290 | acceleration.pos.y += (vector.y + damper.y) * delta; 291 | } else if (hook.type === 'gravity') { 292 | drag.pos = hook.friction; 293 | gravity = this.hooks.defs.getGravity(hook.strength, vLength, this.hooks.defs.zoom); 294 | vector.x *= gravity / vLength; 295 | vector.y *= gravity / vLength; 296 | acceleration.pos.x += vector.x * delta; 297 | acceleration.pos.y += vector.y * delta; 298 | } 299 | } else { 300 | if (acceleration[name = hook.prop] == null) { 301 | acceleration[name] = 0; 302 | } 303 | target = hook.to[hook.targetProperty]; 304 | if (hook.modulator) { 305 | target = hook.modulator(target); 306 | } 307 | vector = target - this[hook.prop]; 308 | if (hook.type === 'spring') { 309 | force = vector * hook.strength; 310 | damper = -hook.friction * this.hooks.velocities[hook.prop]; 311 | acceleration[hook.prop] += (force + damper) * delta; 312 | } else if (hook.type === 'gravity') { 313 | drag[hook.prop] = hook.friction; 314 | force = this.hooks.defs.getGravity(hook.strength, vector, this.hooks.defs.zoom); 315 | acceleration[hook.prop] += force * delta; 316 | } 317 | } 318 | } 319 | ref1 = this.hooks.velocities; 320 | for (prop in ref1) { 321 | velocity = ref1[prop]; 322 | if (prop === 'pos') { 323 | if (drag.pos) { 324 | velocity.x += this.hooks.defs.getDrag(velocity.x, drag.pos, this.hooks.defs.zoom); 325 | velocity.y += this.hooks.defs.getDrag(velocity.y, drag.pos, this.hooks.defs.zoom); 326 | } 327 | velocity.x += acceleration.pos.x; 328 | velocity.y += acceleration.pos.y; 329 | this.x += velocity.x * delta; 330 | this.y += velocity.y * delta; 331 | } else { 332 | if (drag[prop]) { 333 | this.hooks.velocities[prop] += this.hooks.defs.getDrag(this.hooks.velocities[prop], drag[prop], this.hooks.defs.zoom); 334 | } 335 | this.hooks.velocities[prop] += acceleration[prop]; 336 | this[prop] += this.hooks.velocities[prop] * delta; 337 | } 338 | } 339 | return typeof this.onHookUpdate === "function" ? this.onHookUpdate(delta) : void 0; 340 | } 341 | }; 342 | 343 | 344 | },{}]},{},[]) 345 | //# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/sigurd/Repos/framer-hook/hook-example-gravity.framer/modules/Hook.coffee"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","###\n--------------------------------------------------------------------------------\nHook module for Framer\n--------------------------------------------------------------------------------\n\nby:      Sigurd Mannsåker\ngithub:  https://github.com/sigtm/framer-hook\n\n················································································\n\n\nThe Hook module simply expands the Layer prototype, and lets you make any\nnumeric Layer property follow another property - either its own or another\nobject's - via a spring or gravity attraction.\n\n\n--------------------------------------------------------------------------------\nExample: Layered animation (eased + spring)\n--------------------------------------------------------------------------------\n\nmyLayer = new Layer\n\n# Make our own custom property for the x property to follow\nmyLayer.easedX = 0\n\n# Hook x to easedX via a spring\nmyLayer.hook\n\tproperty: \"x\"\n\ttargetProperty: \"easedX\"\n\ttype: \"spring(150, 15)\"\n\n# Animate easedX\nmyLayer.animate\n\tproperties:\n\t\teasedX: 200\n\ttime: 0.15\n\tcurve: \"cubic-bezier(0.2, 0, 0.4, 1)\"\n\nNOTE: \nTo attach both the x and y position, use \"pos\", \"midPos\" or \"maxPos\" as the\nproperty/targetProperty.\n\n\n--------------------------------------------------------------------------------\nExample: Hooking property to another layer\n--------------------------------------------------------------------------------\n\ntarget = new Layer\nhooked = new Layer\n\nhooked.hook\n\tproperty: \"scale\"\n\tto: target\n\ttype: \"spring(150, 15)\"\n\nThe \"hooked\" layer's scale will now continuously follow the target layer's scale\nwith a spring animation.\n\n\n--------------------------------------------------------------------------------\nlayer.hook(options)\n--------------------------------------------------------------------------------\n\nOptions are passed as a single object, like you would for a new Layer.\nThe options object takes the following properties:\n\n\nproperty [String]\n-----------------\nThe property you'd like to hook onto another object's property\n\n\ntype [String]\n-------------\nEither \"spring(strength, friction)\" or \"gravity(strength, drag)\". Only the last\nspecified drag value is used for each property, since it is only applied to\neach property once (and only if it has a gravity hook applied to it.)\n\n\nto [Object] (Optional)\n----------------------\nThe object to attach it to. Defaults to itself.\n\n\ntargetProperty [String] (Optional)\n----------------------------------\nSpecify the target object's property to follow, if you don't want to follow\nthe same property that the hook is applied to.\n\n\nmodulator [Function] (Optional)\n-------------------------------\nThe modulator function receives the target property's value, and lets you\nmodify it before it is fed into the physics calculations. Useful for anything\nfrom standard Utils.modulate() type stuff to snapping and conditional values.\n\n\nzoom [Number] (Optional)\n------------------------\nThis factor defines the distance that 1px represents in regards to gravity and\ndrag calculations. Only one value is stored per layer, so specifying it\noverwrites its existing value. Default is 100.\n\n\n--------------------------------------------------------------------------------\nlayer.unHook(property, object)\n--------------------------------------------------------------------------------\n\nThis removes all hooks for a given property and target object. Example:\n\n# Hook it\nlayer.hook\n\tproperty: \"x\"\n\tto: \"otherlayer\"\n\ttargetProperty: \"y\"\n\ttype: \"spring(200,20)\"\n\n# Unhook it\nlayer.unHook \"x\", otherlayer\n\n\n--------------------------------------------------------------------------------\nlayer.onHookUpdate(delta)\n--------------------------------------------------------------------------------\n\nAfter a layer is done applying accelerations to its hooked properties, it calls\nonHookUpdate() at the end of each frame, if it is defined. This is an easy way\nto animate or trigger other stuff, perhaps based on your layer's updated\nproperties or velocities.\n\nThe delta value from the Framer loop is passed on to onHookUpdate() as well,\nwhich is the time in seconds since the last animation frame.\n\nNote that if you unhook all your hooks, onHookUpdate() will of course no longer\nbe called for that layer.\n\n###\n\n\n# Since older versions of Safari seem to be missing String.prototype.includes()\n\nunless String.prototype.includes\n\tString::includes = (search, start) ->\n\t\t'use strict'\n\t\tstart = 0 if typeof start is 'number'\n\n\t\tif start + search.length > this.length\n\t\t\treturn false;\n\t\telse\n\t\t\treturn @indexOf(search, start) isnt -1\n\n# Expand layer\n\nLayer::hook = (config) ->\n\n\tthrow new Error 'layer.hook() needs a property, a hook type and either a target object or target property to work' unless config.property and config.type and (config.to or config.targetProperty)\n\n\t# Single array for all hooks, as opposed to nested arrays per property, because performance\n\t@hooks ?=\n\t\thooks: []\n\t\tvelocities: {}\n\t\tdefs:\n\t\t\tzoom: 100\n\t\t\tgetDrag: (velocity, drag, zoom) =>\n\t\t\t\tvelocity /= zoom\n\t\t\t\t# Dividing by 10 is unscientific, but it means a value of 2 equals roughly a 100g ball with 15cm radius in air\n\t\t\t\tdrag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity)\n\t\t\t\tif _.isNaN(drag) then return 0 else return drag\n\t\t\tgetGravity: (strength, distance, zoom) =>\n\t\t\t\tdist = Math.max(1, distance / zoom)\n\t\t\t\treturn strength * zoom / (dist * dist)\n\n\t# Update the zoom value if given\n\t@hooks.zoom = config.zoom if config.zoom\n\n\t# Parse physics config string\n\tf = Utils.parseFunction config.type\n\tconfig.type = f.name\n\tconfig.strength = f.args[0]\n\tconfig.friction = f.args[1] or 0\n\n\t# Default to same targetProperty on same object (hopefully you've set at least one of these to something else)\n\tconfig.targetProperty ?= config.property\n\tconfig.to ?= @\n\n\t# All position accelerations are added to a single 'pos' velocity. Store actual properties so we don't have to do it again every frame\n\n\tif config.property.toLowerCase().includes 'pos'\n\t\tconfig.prop = 'pos'\n\t\t\n\t\tif config.property.toLowerCase().includes 'mid'\n\t\t\tconfig.thisX = 'midX'\n\t\t\tconfig.thisY = 'midY'\n\t\t\n\t\telse if config.property.toLowerCase().includes 'max'\n\t\t\tconfig.thisX = 'maxX'\n\t\t\tconfig.thisY = 'maxY'\n\t\t\n\t\telse\n\t\t\tconfig.thisX = 'x'\n\t\t\tconfig.thisY = 'y'\n\t\t\n\t\tif config.targetProperty.toLowerCase().includes 'mid'\n\t\t\tconfig.toX = 'midX'\n\t\t\tconfig.toY = 'midY'\n\t\t\n\t\telse if config.targetProperty.toLowerCase().includes 'max'\n\t\t\tconfig.toX = 'maxX'\n\t\t\tconfig.toY = 'maxY'\t\t\n\t\telse\n\t\t\tconfig.toX = 'x'\n\t\t\tconfig.toY = 'y'\n\t\t\n\telse\n\t\tconfig.prop = config.property\n\n\t# Save hook to @hooks array\t\n\t@hooks.hooks.push(config)\n\n\t# Create velocity property if necessary\n\t@hooks.velocities[config.prop] ?= if config.prop is 'pos' then { x: 0, y: 0 } else 0\n\n\t# Use Framer's animation loop, slightly more robust than requestAnimationFrame directly\n\t# Save the returned AnimationLoop reference to make sure @hookLoop isn't added multiple times per layer\n\t@hooks.emitter ?= Framer.Loop.on('render', @hookLoop, this)\n\nLayer::unHook = (property, object) ->\n\t\n\treturn unless @hooks\n\n\tprop = if property.toLowerCase().includes 'pos' then 'pos' else property\n\n\t# Remove all matches\n\t@hooks.hooks = @hooks.hooks.filter (hook) ->\n\t\thook.to isnt object or hook.property isnt property\n\n\t# If there are no hooks left, shut it down\n\tif @hooks.hooks.length is 0\n\t\tdelete @hooks\n\t\tFramer.Loop.removeListener 'render', @hookLoop\n\t\treturn\n\n\t# Still here? Check if there are any remaining hooks affecting same velocity\n\tremaining = @hooks.hooks.filter (hook) ->\n\t\tprop is hook.prop\n\t\t\n\t# If not, delete velocity (otherwise it won't be reset if you make new hook for same property)\n\tdelete @hooks.velocities[prop] if remaining.length is 0\n\nLayer::hookLoop = (delta) ->\n\n\tif @hooks\n\n\t\t# Multiple hooks can affect the same property. Add accelerations to temporary object so the property's velocity is the same for all calculations within the same animation frame\n\t\tacceleration = {}\n\t\t\n\t\t# Save drag for each property to this object, since only most recently specified value is used for each property\n\t\tdrag = {}\n\t\t\n\t\t# Add accelerations\n\t\tfor hook in @hooks.hooks\n\t\t\n\t\t\tif hook.prop is 'pos'\n\n\t\t\t\tacceleration.pos ?= { x: 0, y: 0 }\n\n\t\t\t\ttarget = { x: hook.to[hook.toX], y: hook.to[hook.toY] }\n\n\t\t\t\ttarget = hook.modulator(target) if hook.modulator\n\n\t\t\t\tvector =\n\t\t\t\t\tx: target.x - @[hook.thisX]\n\t\t\t\t\ty: target.y - @[hook.thisY]\n\t\t\t\t\n\t\t\t\tvLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y))\n\n\t\t\t\tif hook.type is 'spring'\n\n\t\t\t\t\tdamper =\n\t\t\t\t\t\tx: -hook.friction * @hooks.velocities.pos.x\n\t\t\t\t\t\ty: -hook.friction * @hooks.velocities.pos.y\n\n\t\t\t\t\tvector.x *= hook.strength\n\t\t\t\t\tvector.y *= hook.strength\n\t\t\t\t\t\n\t\t\t\t\tacceleration.pos.x += (vector.x + damper.x) * delta\n\t\t\t\t\tacceleration.pos.y += (vector.y + damper.y) * delta\n\t\t\t\t\n\t\t\t\telse if hook.type is 'gravity'\n\t\t\t\t\n\t\t\t\t\tdrag.pos = hook.friction\n\t\t\t\t\t\n\t\t\t\t\tgravity = @hooks.defs.getGravity(hook.strength, vLength, @hooks.defs.zoom)\n\n\t\t\t\t\tvector.x *= gravity / vLength\n\t\t\t\t\tvector.y *= gravity / vLength\n\t\t\t\t\t\n\t\t\t\t\tacceleration.pos.x += vector.x * delta\n\t\t\t\t\tacceleration.pos.y += vector.y * delta\n\t\t\t\t\t\t\t\t\t\n\t\t\telse\n\t\t\t\t\n\t\t\t\tacceleration[hook.prop] ?= 0\n\n\t\t\t\ttarget = hook.to[hook.targetProperty]\n\n\t\t\t\ttarget = hook.modulator(target) if hook.modulator\n\n\t\t\t\tvector = target - @[hook.prop]\n\t\t\t\t\n\t\t\t\tif hook.type is 'spring'\n\n\t\t\t\t\tforce = vector * hook.strength\n\t\t\t\t\tdamper = -hook.friction * @hooks.velocities[hook.prop]\n\n\t\t\t\t\tacceleration[hook.prop] += (force + damper) * delta\n\n\t\t\t\t\n\t\t\t\telse if hook.type is 'gravity'\n\t\n\t\t\t\t\tdrag[hook.prop] = hook.friction\n\t\t\t\t\t\n\t\t\t\t\tforce = @hooks.defs.getGravity(hook.strength, vector, @hooks.defs.zoom)\n\t\t\t\t\t\n\t\t\t\t\tacceleration[hook.prop] += force * delta\n\t\t\n\t\t\n\t\t# Add velocities to properties. Doing this at the end in case there are multiple hooks affecting the same velocity\n\t\tfor prop, velocity of @hooks.velocities\n\t\t\n\t\t\tif prop is 'pos'\n\n\t\t\t\t# Add drag, if it exists\n\t\t\t\tif drag.pos\n\t\t\t\t\tvelocity.x += @hooks.defs.getDrag(velocity.x, drag.pos, @hooks.defs.zoom)\n\t\t\t\t\tvelocity.y += @hooks.defs.getDrag(velocity.y, drag.pos, @hooks.defs.zoom)\n\t\t\t\t\t\n\t\t\t\t# Add acceleration to velocity\n\t\t\t\tvelocity.x += acceleration.pos.x\n\t\t\t\tvelocity.y += acceleration.pos.y\n\t\t\t\t\n\t\t\t\t# Add velocity to position\n\t\t\t\t@x += velocity.x * delta\n\t\t\t\t@y += velocity.y * delta\n\t\t\t\n\t\t\telse\n\t\t\t\n\t\t\t\t# Add drag, if it exists\n\t\t\t\tif drag[prop]\n\t\t\t\t\t@hooks.velocities[prop] += @hooks.defs.getDrag(@hooks.velocities[prop], drag[prop], @hooks.defs.zoom)\n\t\t\t\t\n\t\t\t\t# Add acceleration to velocity\n\t\t\t\t@hooks.velocities[prop] += acceleration[prop]\n\t\t\t\t\n\t\t\t\t# Add velocity to property\n\t\t\t\t@[prop] += @hooks.velocities[prop] * delta\n\n\t\t@onHookUpdate?(delta)"]} 346 | -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/cursor-active.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/cursor-active.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/cursor-active@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/cursor-active@2x.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/cursor.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/cursor.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/cursor@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/cursor@2x.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/icon-120.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/icon-120.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/icon-152.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/icon-152.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/icon-180.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/icon-180.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/icon-192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/icon-192.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/images/icon-76.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/images/icon-76.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/manifest.txt: -------------------------------------------------------------------------------- 1 | app.coffee 2 | framer/coffee-script.js 3 | framer/config.json 4 | framer/framer.generated.js 5 | framer/framer.init.js 6 | framer/framer.js 7 | framer/framer.js.map 8 | framer/framer.modules.js 9 | framer/images/cursor-active.png 10 | framer/images/cursor-active@2x.png 11 | framer/images/cursor.png 12 | framer/images/cursor@2x.png 13 | framer/images/icon-120.png 14 | framer/images/icon-152.png 15 | framer/images/icon-180.png 16 | framer/images/icon-192.png 17 | framer/images/icon-76.png 18 | framer/manifest.txt 19 | framer/metadata.json 20 | framer/preview.png 21 | framer/style.css 22 | framer/version 23 | index.html 24 | modules/Hook.coffee -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/metadata.json: -------------------------------------------------------------------------------- 1 | {"title":"hook-example-gravity","date":"2016-09-09T12:28:41Z"} -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/preview.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/framer/preview.png -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/style.css: -------------------------------------------------------------------------------- 1 | * { 2 | margin: 0; 3 | padding: 0; 4 | border: none; 5 | -webkit-user-select: none; 6 | -webkit-tap-highlight-color: rgba(0,0,0,0); 7 | } 8 | 9 | body { 10 | background-color: #fff; 11 | font: 28px/1em "Helvetica"; 12 | color: gray; 13 | overflow: hidden; 14 | } 15 | 16 | a { 17 | color: gray; 18 | } 19 | 20 | body { 21 | cursor: url('images/cursor.png') 32 32, auto; 22 | cursor: -webkit-image-set( 23 | url('images/cursor.png') 1x, 24 | url('images/cursor@2x.png') 2x 25 | ) 32 32, auto; 26 | } 27 | 28 | body:active { 29 | cursor: url('images/cursor-active.png') 32 32, auto; 30 | cursor: -webkit-image-set( 31 | url('images/cursor-active.png') 1x, 32 | url('images/cursor-active@2x.png') 2x 33 | ) 32 32, auto; 34 | } 35 | 36 | .framerAlertBackground { 37 | position: absolute; top:0px; left:0px; right:0px; bottom:0px; 38 | z-index: 1000; 39 | background-color: #fff; 40 | } 41 | 42 | .framerAlert { 43 | font:400 14px/1.4 "Helvetica Neue", Helvetica, Arial, sans-serif; 44 | -webkit-font-smoothing:antialiased; 45 | color:#616367; text-align:center; 46 | position: absolute; top:40%; left:50%; width:260px; margin-left:-130px; 47 | } 48 | .framerAlert strong { font-weight:500; color:#000; margin-bottom:8px; display:block; } 49 | .framerAlert a { color:#28AFFA; } 50 | .framerAlert .btn { 51 | font-weight:500; text-decoration:none; line-height:1; 52 | display:inline-block; padding:6px 12px 7px 12px; 53 | border-radius:3px; margin-top:12px; 54 | background:#28AFFA; color:#fff; 55 | } 56 | 57 | ::-webkit-scrollbar { 58 | display: none; 59 | } -------------------------------------------------------------------------------- /hook-example-gravity.framer/framer/version: -------------------------------------------------------------------------------- 1 | 4 -------------------------------------------------------------------------------- /hook-example-gravity.framer/images/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-gravity.framer/images/.gitkeep -------------------------------------------------------------------------------- /hook-example-gravity.framer/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | -------------------------------------------------------------------------------- /hook-example-gravity.framer/modules/Hook.coffee: -------------------------------------------------------------------------------- 1 | ### 2 | -------------------------------------------------------------------------------- 3 | Hook module for Framer 4 | -------------------------------------------------------------------------------- 5 | 6 | by: Sigurd Mannsåker 7 | github: https://github.com/sigtm/framer-hook 8 | 9 | ················································································ 10 | 11 | 12 | The Hook module simply expands the Layer prototype, and lets you make any 13 | numeric Layer property follow another property - either its own or another 14 | object's - via a spring or gravity attraction. 15 | 16 | 17 | -------------------------------------------------------------------------------- 18 | Example: Layered animation (eased + spring) 19 | -------------------------------------------------------------------------------- 20 | 21 | myLayer = new Layer 22 | 23 | # Make our own custom property for the x property to follow 24 | myLayer.easedX = 0 25 | 26 | # Hook x to easedX via a spring 27 | myLayer.hook 28 | property: "x" 29 | targetProperty: "easedX" 30 | type: "spring(150, 15)" 31 | 32 | # Animate easedX 33 | myLayer.animate 34 | properties: 35 | easedX: 200 36 | time: 0.15 37 | curve: "cubic-bezier(0.2, 0, 0.4, 1)" 38 | 39 | NOTE: 40 | To attach both the x and y position, use "pos", "midPos" or "maxPos" as the 41 | property/targetProperty. 42 | 43 | 44 | -------------------------------------------------------------------------------- 45 | Example: Hooking property to another layer 46 | -------------------------------------------------------------------------------- 47 | 48 | target = new Layer 49 | hooked = new Layer 50 | 51 | hooked.hook 52 | property: "scale" 53 | to: target 54 | type: "spring(150, 15)" 55 | 56 | The "hooked" layer's scale will now continuously follow the target layer's scale 57 | with a spring animation. 58 | 59 | 60 | -------------------------------------------------------------------------------- 61 | layer.hook(options) 62 | -------------------------------------------------------------------------------- 63 | 64 | Options are passed as a single object, like you would for a new Layer. 65 | The options object takes the following properties: 66 | 67 | 68 | property [String] 69 | ----------------- 70 | The property you'd like to hook onto another object's property 71 | 72 | 73 | type [String] 74 | ------------- 75 | Either "spring(strength, friction)" or "gravity(strength, drag)". Only the last 76 | specified drag value is used for each property, since it is only applied to 77 | each property once (and only if it has a gravity hook applied to it.) 78 | 79 | 80 | to [Object] (Optional) 81 | ---------------------- 82 | The object to attach it to. Defaults to itself. 83 | 84 | 85 | targetProperty [String] (Optional) 86 | ---------------------------------- 87 | Specify the target object's property to follow, if you don't want to follow 88 | the same property that the hook is applied to. 89 | 90 | 91 | modulator [Function] (Optional) 92 | ------------------------------- 93 | The modulator function receives the target property's value, and lets you 94 | modify it before it is fed into the physics calculations. Useful for anything 95 | from standard Utils.modulate() type stuff to snapping and conditional values. 96 | 97 | 98 | zoom [Number] (Optional) 99 | ------------------------ 100 | This factor defines the distance that 1px represents in regards to gravity and 101 | drag calculations. Only one value is stored per layer, so specifying it 102 | overwrites its existing value. Default is 100. 103 | 104 | 105 | -------------------------------------------------------------------------------- 106 | layer.unHook(property, object) 107 | -------------------------------------------------------------------------------- 108 | 109 | This removes all hooks for a given property and target object. Example: 110 | 111 | # Hook it 112 | layer.hook 113 | property: "x" 114 | to: "otherlayer" 115 | targetProperty: "y" 116 | type: "spring(200,20)" 117 | 118 | # Unhook it 119 | layer.unHook "x", otherlayer 120 | 121 | 122 | -------------------------------------------------------------------------------- 123 | layer.onHookUpdate(delta) 124 | -------------------------------------------------------------------------------- 125 | 126 | After a layer is done applying accelerations to its hooked properties, it calls 127 | onHookUpdate() at the end of each frame, if it is defined. This is an easy way 128 | to animate or trigger other stuff, perhaps based on your layer's updated 129 | properties or velocities. 130 | 131 | The delta value from the Framer loop is passed on to onHookUpdate() as well, 132 | which is the time in seconds since the last animation frame. 133 | 134 | Note that if you unhook all your hooks, onHookUpdate() will of course no longer 135 | be called for that layer. 136 | 137 | ### 138 | 139 | 140 | # Since older versions of Safari seem to be missing String.prototype.includes() 141 | 142 | unless String.prototype.includes 143 | String::includes = (search, start) -> 144 | 'use strict' 145 | start = 0 if typeof start is 'number' 146 | 147 | if start + search.length > this.length 148 | return false; 149 | else 150 | return @indexOf(search, start) isnt -1 151 | 152 | # Expand layer 153 | 154 | Layer::hook = (config) -> 155 | 156 | throw new Error 'layer.hook() needs a property, a hook type and either a target object or target property to work' unless config.property and config.type and (config.to or config.targetProperty) 157 | 158 | # Single array for all hooks, as opposed to nested arrays per property, because performance 159 | @hooks ?= 160 | hooks: [] 161 | velocities: {} 162 | defs: 163 | zoom: 100 164 | getDrag: (velocity, drag, zoom) => 165 | velocity /= zoom 166 | # Dividing by 10 is unscientific, but it means a value of 2 equals roughly a 100g ball with 15cm radius in air 167 | drag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity) 168 | if _.isNaN(drag) then return 0 else return drag 169 | getGravity: (strength, distance, zoom) => 170 | dist = Math.max(1, distance / zoom) 171 | return strength * zoom / (dist * dist) 172 | 173 | # Update the zoom value if given 174 | @hooks.zoom = config.zoom if config.zoom 175 | 176 | # Parse physics config string 177 | f = Utils.parseFunction config.type 178 | config.type = f.name 179 | config.strength = f.args[0] 180 | config.friction = f.args[1] or 0 181 | 182 | # Default to same targetProperty on same object (hopefully you've set at least one of these to something else) 183 | config.targetProperty ?= config.property 184 | config.to ?= @ 185 | 186 | # All position accelerations are added to a single 'pos' velocity. Store actual properties so we don't have to do it again every frame 187 | 188 | if config.property.toLowerCase().includes 'pos' 189 | config.prop = 'pos' 190 | 191 | if config.property.toLowerCase().includes 'mid' 192 | config.thisX = 'midX' 193 | config.thisY = 'midY' 194 | 195 | else if config.property.toLowerCase().includes 'max' 196 | config.thisX = 'maxX' 197 | config.thisY = 'maxY' 198 | 199 | else 200 | config.thisX = 'x' 201 | config.thisY = 'y' 202 | 203 | if config.targetProperty.toLowerCase().includes 'mid' 204 | config.toX = 'midX' 205 | config.toY = 'midY' 206 | 207 | else if config.targetProperty.toLowerCase().includes 'max' 208 | config.toX = 'maxX' 209 | config.toY = 'maxY' 210 | else 211 | config.toX = 'x' 212 | config.toY = 'y' 213 | 214 | else 215 | config.prop = config.property 216 | 217 | # Save hook to @hooks array 218 | @hooks.hooks.push(config) 219 | 220 | # Create velocity property if necessary 221 | @hooks.velocities[config.prop] ?= if config.prop is 'pos' then { x: 0, y: 0 } else 0 222 | 223 | # Use Framer's animation loop, slightly more robust than requestAnimationFrame directly 224 | # Save the returned AnimationLoop reference to make sure @hookLoop isn't added multiple times per layer 225 | @hooks.emitter ?= Framer.Loop.on('render', @hookLoop, this) 226 | 227 | Layer::unHook = (property, object) -> 228 | 229 | return unless @hooks 230 | 231 | prop = if property.toLowerCase().includes 'pos' then 'pos' else property 232 | 233 | # Remove all matches 234 | @hooks.hooks = @hooks.hooks.filter (hook) -> 235 | hook.to isnt object or hook.property isnt property 236 | 237 | # If there are no hooks left, shut it down 238 | if @hooks.hooks.length is 0 239 | delete @hooks 240 | Framer.Loop.removeListener 'render', @hookLoop 241 | return 242 | 243 | # Still here? Check if there are any remaining hooks affecting same velocity 244 | remaining = @hooks.hooks.filter (hook) -> 245 | prop is hook.prop 246 | 247 | # If not, delete velocity (otherwise it won't be reset if you make new hook for same property) 248 | delete @hooks.velocities[prop] if remaining.length is 0 249 | 250 | Layer::hookLoop = (delta) -> 251 | 252 | if @hooks 253 | 254 | # Multiple hooks can affect the same property. Add accelerations to temporary object so the property's velocity is the same for all calculations within the same animation frame 255 | acceleration = {} 256 | 257 | # Save drag for each property to this object, since only most recently specified value is used for each property 258 | drag = {} 259 | 260 | # Add accelerations 261 | for hook in @hooks.hooks 262 | 263 | if hook.prop is 'pos' 264 | 265 | acceleration.pos ?= { x: 0, y: 0 } 266 | 267 | target = { x: hook.to[hook.toX], y: hook.to[hook.toY] } 268 | 269 | target = hook.modulator(target) if hook.modulator 270 | 271 | vector = 272 | x: target.x - @[hook.thisX] 273 | y: target.y - @[hook.thisY] 274 | 275 | vLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y)) 276 | 277 | if hook.type is 'spring' 278 | 279 | damper = 280 | x: -hook.friction * @hooks.velocities.pos.x 281 | y: -hook.friction * @hooks.velocities.pos.y 282 | 283 | vector.x *= hook.strength 284 | vector.y *= hook.strength 285 | 286 | acceleration.pos.x += (vector.x + damper.x) * delta 287 | acceleration.pos.y += (vector.y + damper.y) * delta 288 | 289 | else if hook.type is 'gravity' 290 | 291 | drag.pos = hook.friction 292 | 293 | gravity = @hooks.defs.getGravity(hook.strength, vLength, @hooks.defs.zoom) 294 | 295 | vector.x *= gravity / vLength 296 | vector.y *= gravity / vLength 297 | 298 | acceleration.pos.x += vector.x * delta 299 | acceleration.pos.y += vector.y * delta 300 | 301 | else 302 | 303 | acceleration[hook.prop] ?= 0 304 | 305 | target = hook.to[hook.targetProperty] 306 | 307 | target = hook.modulator(target) if hook.modulator 308 | 309 | vector = target - @[hook.prop] 310 | 311 | if hook.type is 'spring' 312 | 313 | force = vector * hook.strength 314 | damper = -hook.friction * @hooks.velocities[hook.prop] 315 | 316 | acceleration[hook.prop] += (force + damper) * delta 317 | 318 | 319 | else if hook.type is 'gravity' 320 | 321 | drag[hook.prop] = hook.friction 322 | 323 | force = @hooks.defs.getGravity(hook.strength, vector, @hooks.defs.zoom) 324 | 325 | acceleration[hook.prop] += force * delta 326 | 327 | 328 | # Add velocities to properties. Doing this at the end in case there are multiple hooks affecting the same velocity 329 | for prop, velocity of @hooks.velocities 330 | 331 | if prop is 'pos' 332 | 333 | # Add drag, if it exists 334 | if drag.pos 335 | velocity.x += @hooks.defs.getDrag(velocity.x, drag.pos, @hooks.defs.zoom) 336 | velocity.y += @hooks.defs.getDrag(velocity.y, drag.pos, @hooks.defs.zoom) 337 | 338 | # Add acceleration to velocity 339 | velocity.x += acceleration.pos.x 340 | velocity.y += acceleration.pos.y 341 | 342 | # Add velocity to position 343 | @x += velocity.x * delta 344 | @y += velocity.y * delta 345 | 346 | else 347 | 348 | # Add drag, if it exists 349 | if drag[prop] 350 | @hooks.velocities[prop] += @hooks.defs.getDrag(@hooks.velocities[prop], drag[prop], @hooks.defs.zoom) 351 | 352 | # Add acceleration to velocity 353 | @hooks.velocities[prop] += acceleration[prop] 354 | 355 | # Add velocity to property 356 | @[prop] += @hooks.velocities[prop] * delta 357 | 358 | @onHookUpdate?(delta) -------------------------------------------------------------------------------- /hook-example-modulator.framer/.gitignore: -------------------------------------------------------------------------------- 1 | # Framer Git Ignore 2 | 3 | # General OSX 4 | 5 | .DS_Store 6 | .AppleDouble 7 | .LSOverride 8 | 9 | # Icon must end with two \r 10 | Icon 11 | 12 | 13 | # Thumbnails 14 | ._* 15 | 16 | # Files that might appear in the root of a volume 17 | .DocumentRevisions-V100 18 | .fseventsd 19 | .Spotlight-V100 20 | .TemporaryItems 21 | .Trashes 22 | .VolumeIcon.icns 23 | 24 | # Directories potentially created on remote AFP share 25 | .AppleDB 26 | .AppleDesktop 27 | Network Trash Folder 28 | Temporary Items 29 | .apdisk 30 | 31 | 32 | # Framer Specific 33 | .temp.html 34 | framer/*.old* 35 | framer/backup.coffee 36 | framer/backups/* 37 | framer/.*.hash 38 | -------------------------------------------------------------------------------- /hook-example-modulator.framer/.viewer.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | -------------------------------------------------------------------------------- /hook-example-modulator.framer/app.coffee: -------------------------------------------------------------------------------- 1 | ### 2 | 3 | Hook example 4 | -------------- 5 | 6 | The Hook module simply expands the Layer prototype, and lets you make any 7 | numeric Layer property follow a property on another object via a spring or a 8 | gravity attraction. Check the comments in modules/Hook.coffee for a more thorough 9 | documentation. 10 | 11 | ### 12 | 13 | 14 | 15 | # Require Hook. No exports, it just adds methods to the Layer prototype 16 | # -------------------------------------------------------------------------------- 17 | 18 | require "Hook" 19 | 20 | 21 | 22 | # Import sketch file 23 | # -------------------------------------------------------------------------------- 24 | sketch = Framer.Importer.load("imported/app@2x") 25 | Utils.globalLayers(sketch) 26 | 27 | 28 | 29 | # Settings 30 | # -------------------------------------------------------------------------------- 31 | 32 | colors = 33 | background: "#260355" 34 | orange: "#f90" 35 | blue: "#63ffff" 36 | primary: "white" 37 | secondary: "rgba(255,255,255,0.1)" 38 | 39 | margin = 60 40 | 41 | easeInOut = "cubic-bezier(0.2, 0, 0.4, 1)" 42 | 43 | 44 | 45 | # Set background and defaults 46 | # -------------------------------------------------------------------------------- 47 | 48 | Framer.Device.viewport.backgroundColor = colors.background 49 | 50 | Framer.Defaults.Animation = 51 | curve: easeInOut 52 | time: 0.15 53 | 54 | 55 | 56 | # Set up example 57 | # -------------------------------------------------------------------------------- 58 | 59 | # Make the page layer draggable 60 | page.draggable.enabled = true 61 | page.draggable.horizontal = false 62 | page.draggable.constraints = { height: Screen.height } 63 | page.draggable.bounceOptions.tension = 400 64 | 65 | # Make the spinner layer 66 | spinner = new Layer 67 | index: 0 68 | x: Align.center 69 | midY: 40 70 | width: 96 71 | height: 96 72 | borderRadius: 40 73 | image: "images/spinner.svg" 74 | 75 | # Attach the spinner's scale to the page's y property 76 | spinner.hook 77 | property: "scale" 78 | to: page 79 | targetProperty: "y" 80 | type: "spring(200,10)" 81 | 82 | # Add a modulator function, since we don't want the scale to be the same as page.y 83 | modulator: (inputvalue) -> 84 | 85 | # If page.y is under 300, translate to a scale between 0.5 and 0.75 86 | if inputvalue < 300 87 | return Utils.modulate(inputvalue, [0, 400], [0.5, 0.75]) 88 | 89 | # Otherwise, snap to 1 90 | else 91 | return 1 92 | 93 | # Now attach spinner.midY to page.y with similar conditional modulation 94 | spinner.hook 95 | property: "midY" 96 | to: page 97 | targetProperty: "y" 98 | type: "spring(100,8)" 99 | modulator: (inputvalue) -> 100 | if inputvalue < 300 101 | inputvalue / 6 + 40 102 | else 103 | inputvalue / 2 104 | 105 | # onHookUpdate() is called at the end of each frame, after all of the layer's hooks are applied 106 | # It receives the delta value from the Framer loop, which is the time in seconds since the last frame 107 | spinner.onHookUpdate = (delta) -> 108 | if page.y < 300 109 | @doSpin = false 110 | @opacity = 0.2 111 | @rotation = page.y / 2 112 | else 113 | @doSpin = true 114 | @opacity = 1 115 | @rotation += delta * 300 if spinner.doSpin -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/config.json: -------------------------------------------------------------------------------- 1 | { 2 | "propertyPanelToggleStates" : { 3 | "Filters" : false 4 | }, 5 | "deviceOrientation" : 0, 6 | "sharedPrototype" : 0, 7 | "contentScale" : 1, 8 | "deviceType" : "apple-iphone-6s-silver", 9 | "selectedHand" : "", 10 | "updateDelay" : 0.3, 11 | "deviceScale" : "fit", 12 | "foldedCodeRanges" : [ 13 | 14 | ], 15 | "orientation" : 0, 16 | "fullScreen" : false 17 | } -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/framer.generated.js: -------------------------------------------------------------------------------- 1 | // This is autogenerated by Framer 2 | 3 | 4 | if (!window.Framer && window._bridge) {window._bridge('runtime.error', {message:'[framer.js] Framer library missing or corrupt. Select File → Update Framer Library.'})} 5 | window.__imported__ = window.__imported__ || {}; 6 | window.__imported__["app@2x/layers.json.js"] = [ 7 | { 8 | "objectId": "0CDC1067-9F1A-4B22-8CF3-3B8FEE516032", 9 | "kind": "group", 10 | "name": "page", 11 | "maskFrame": null, 12 | "layerFrame": { 13 | "x": 0, 14 | "y": 0, 15 | "width": 375, 16 | "height": 667 17 | }, 18 | "visible": true, 19 | "metadata": { 20 | "opacity": 1 21 | }, 22 | "image": { 23 | "path": "images/Layer-page-meneqzew.png", 24 | "frame": { 25 | "x": 0, 26 | "y": 0, 27 | "width": 375, 28 | "height": 667 29 | } 30 | }, 31 | "children": [], 32 | "time": 123 33 | } 34 | ] 35 | if (DeviceComponent) {DeviceComponent.Devices["iphone-6-silver"].deviceImageJP2 = false}; 36 | if (window.Framer) {window.Framer.Defaults.DeviceView = {"deviceScale":"fit","selectedHand":"","deviceType":"apple-iphone-6s-silver","contentScale":1,"orientation":0}; 37 | } 38 | if (window.Framer) {window.Framer.Defaults.DeviceComponent = {"deviceScale":"fit","selectedHand":"","deviceType":"apple-iphone-6s-silver","contentScale":1,"orientation":0}; 39 | } 40 | window.FramerStudioInfo = {"deviceImagesUrl":"\/_server\/resources\/DeviceImages","documentTitle":"hook-example-modulator.framer"}; 41 | 42 | Framer.Device = new Framer.DeviceView(); 43 | Framer.Device.setupContext(); -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/framer.init.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | 3 | function isFileLoadingAllowed() { 4 | return (window.location.protocol.indexOf("file") == -1) 5 | } 6 | 7 | function isHomeScreened() { 8 | return ("standalone" in window.navigator) && window.navigator.standalone == true 9 | } 10 | 11 | function isCompatibleBrowser() { 12 | return Utils.isWebKit() 13 | } 14 | 15 | var alertNode; 16 | 17 | function dismissAlert() { 18 | alertNode.parentElement.removeChild(alertNode) 19 | loadProject() 20 | } 21 | 22 | function showAlert(html) { 23 | 24 | alertNode = document.createElement("div") 25 | 26 | alertNode.classList.add("framerAlertBackground") 27 | alertNode.innerHTML = html 28 | 29 | document.addEventListener("DOMContentLoaded", function(event) { 30 | document.body.appendChild(alertNode) 31 | }) 32 | 33 | window.dismissAlert = dismissAlert; 34 | } 35 | 36 | function showBrowserAlert() { 37 | var html = "" 38 | html += "
" 39 | html += "Error: Not A WebKit Browser" 40 | html += "Your browser is not supported.
Please use Safari or Chrome.
" 41 | html += "Try anyway" 42 | html += "
" 43 | 44 | showAlert(html) 45 | } 46 | 47 | function showFileLoadingAlert() { 48 | var html = "" 49 | html += "
" 50 | html += "Error: Local File Restrictions" 51 | html += "Preview this prototype with Framer Mirror or learn more about " 52 | html += "file restrictions.
" 53 | html += "Try anyway" 54 | html += "
" 55 | 56 | showAlert(html) 57 | } 58 | 59 | function loadProject() { 60 | CoffeeScript.load("app.coffee") 61 | } 62 | 63 | function setDefaultPageTitle() { 64 | // If no title was set we set it to the project folder name so 65 | // you get a nice name on iOS if you bookmark to desktop. 66 | document.addEventListener("DOMContentLoaded", function() { 67 | if (document.title == "") { 68 | if (window.FramerStudioInfo && window.FramerStudioInfo.documentTitle) { 69 | document.title = window.FramerStudioInfo.documentTitle 70 | } else { 71 | document.title = window.location.pathname.replace(/\//g, "") 72 | } 73 | } 74 | }) 75 | } 76 | 77 | function init() { 78 | 79 | if (Utils.isFramerStudio()) { 80 | return 81 | } 82 | 83 | setDefaultPageTitle() 84 | 85 | if (!isCompatibleBrowser()) { 86 | return showBrowserAlert() 87 | } 88 | 89 | if (!isFileLoadingAllowed()) { 90 | return showFileLoadingAlert() 91 | } 92 | 93 | loadProject() 94 | 95 | } 96 | 97 | init() 98 | 99 | })() 100 | -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/framer.modules.js: -------------------------------------------------------------------------------- 1 | require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o this.length) { 146 | return false; 147 | } else { 148 | return this.indexOf(search, start) !== -1; 149 | } 150 | }; 151 | } 152 | 153 | Layer.prototype.hook = function(config) { 154 | var base, base1, f, name; 155 | if (!(config.property && config.type && (config.to || config.targetProperty))) { 156 | throw new Error('layer.hook() needs a property, a hook type and either a target object or target property to work'); 157 | } 158 | if (this.hooks == null) { 159 | this.hooks = { 160 | hooks: [], 161 | velocities: {}, 162 | defs: { 163 | zoom: 100, 164 | getDrag: (function(_this) { 165 | return function(velocity, drag, zoom) { 166 | velocity /= zoom; 167 | drag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity); 168 | if (_.isNaN(drag)) { 169 | return 0; 170 | } else { 171 | return drag; 172 | } 173 | }; 174 | })(this), 175 | getGravity: (function(_this) { 176 | return function(strength, distance, zoom) { 177 | var dist; 178 | dist = Math.max(1, distance / zoom); 179 | return strength * zoom / (dist * dist); 180 | }; 181 | })(this) 182 | } 183 | }; 184 | } 185 | if (config.zoom) { 186 | this.hooks.zoom = config.zoom; 187 | } 188 | f = Utils.parseFunction(config.type); 189 | config.type = f.name; 190 | config.strength = f.args[0]; 191 | config.friction = f.args[1] || 0; 192 | if (config.targetProperty == null) { 193 | config.targetProperty = config.property; 194 | } 195 | if (config.to == null) { 196 | config.to = this; 197 | } 198 | if (config.property.toLowerCase().includes('pos')) { 199 | config.prop = 'pos'; 200 | if (config.property.toLowerCase().includes('mid')) { 201 | config.thisX = 'midX'; 202 | config.thisY = 'midY'; 203 | } else if (config.property.toLowerCase().includes('max')) { 204 | config.thisX = 'maxX'; 205 | config.thisY = 'maxY'; 206 | } else { 207 | config.thisX = 'x'; 208 | config.thisY = 'y'; 209 | } 210 | if (config.targetProperty.toLowerCase().includes('mid')) { 211 | config.toX = 'midX'; 212 | config.toY = 'midY'; 213 | } else if (config.targetProperty.toLowerCase().includes('max')) { 214 | config.toX = 'maxX'; 215 | config.toY = 'maxY'; 216 | } else { 217 | config.toX = 'x'; 218 | config.toY = 'y'; 219 | } 220 | } else { 221 | config.prop = config.property; 222 | } 223 | this.hooks.hooks.push(config); 224 | if ((base = this.hooks.velocities)[name = config.prop] == null) { 225 | base[name] = config.prop === 'pos' ? { 226 | x: 0, 227 | y: 0 228 | } : 0; 229 | } 230 | return (base1 = this.hooks).emitter != null ? base1.emitter : base1.emitter = Framer.Loop.on('render', this.hookLoop, this); 231 | }; 232 | 233 | Layer.prototype.unHook = function(property, object) { 234 | var prop, remaining; 235 | if (!this.hooks) { 236 | return; 237 | } 238 | prop = property.toLowerCase().includes('pos') ? 'pos' : property; 239 | this.hooks.hooks = this.hooks.hooks.filter(function(hook) { 240 | return hook.to !== object || hook.property !== property; 241 | }); 242 | if (this.hooks.hooks.length === 0) { 243 | delete this.hooks; 244 | Framer.Loop.removeListener('render', this.hookLoop); 245 | return; 246 | } 247 | remaining = this.hooks.hooks.filter(function(hook) { 248 | return prop === hook.prop; 249 | }); 250 | if (remaining.length === 0) { 251 | return delete this.hooks.velocities[prop]; 252 | } 253 | }; 254 | 255 | Layer.prototype.hookLoop = function(delta) { 256 | var acceleration, damper, drag, force, gravity, hook, i, len, name, prop, ref, ref1, target, vLength, vector, velocity; 257 | if (this.hooks) { 258 | acceleration = {}; 259 | drag = {}; 260 | ref = this.hooks.hooks; 261 | for (i = 0, len = ref.length; i < len; i++) { 262 | hook = ref[i]; 263 | if (hook.prop === 'pos') { 264 | if (acceleration.pos == null) { 265 | acceleration.pos = { 266 | x: 0, 267 | y: 0 268 | }; 269 | } 270 | target = { 271 | x: hook.to[hook.toX], 272 | y: hook.to[hook.toY] 273 | }; 274 | if (hook.modulator) { 275 | target = hook.modulator(target); 276 | } 277 | vector = { 278 | x: target.x - this[hook.thisX], 279 | y: target.y - this[hook.thisY] 280 | }; 281 | vLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y)); 282 | if (hook.type === 'spring') { 283 | damper = { 284 | x: -hook.friction * this.hooks.velocities.pos.x, 285 | y: -hook.friction * this.hooks.velocities.pos.y 286 | }; 287 | vector.x *= hook.strength; 288 | vector.y *= hook.strength; 289 | acceleration.pos.x += (vector.x + damper.x) * delta; 290 | acceleration.pos.y += (vector.y + damper.y) * delta; 291 | } else if (hook.type === 'gravity') { 292 | drag.pos = hook.friction; 293 | gravity = this.hooks.defs.getGravity(hook.strength, vLength, this.hooks.defs.zoom); 294 | vector.x *= gravity / vLength; 295 | vector.y *= gravity / vLength; 296 | acceleration.pos.x += vector.x * delta; 297 | acceleration.pos.y += vector.y * delta; 298 | } 299 | } else { 300 | if (acceleration[name = hook.prop] == null) { 301 | acceleration[name] = 0; 302 | } 303 | target = hook.to[hook.targetProperty]; 304 | if (hook.modulator) { 305 | target = hook.modulator(target); 306 | } 307 | vector = target - this[hook.prop]; 308 | if (hook.type === 'spring') { 309 | force = vector * hook.strength; 310 | damper = -hook.friction * this.hooks.velocities[hook.prop]; 311 | acceleration[hook.prop] += (force + damper) * delta; 312 | } else if (hook.type === 'gravity') { 313 | drag[hook.prop] = hook.friction; 314 | force = this.hooks.defs.getGravity(hook.strength, vector, this.hooks.defs.zoom); 315 | acceleration[hook.prop] += force * delta; 316 | } 317 | } 318 | } 319 | ref1 = this.hooks.velocities; 320 | for (prop in ref1) { 321 | velocity = ref1[prop]; 322 | if (prop === 'pos') { 323 | if (drag.pos) { 324 | velocity.x += this.hooks.defs.getDrag(velocity.x, drag.pos, this.hooks.defs.zoom); 325 | velocity.y += this.hooks.defs.getDrag(velocity.y, drag.pos, this.hooks.defs.zoom); 326 | } 327 | velocity.x += acceleration.pos.x; 328 | velocity.y += acceleration.pos.y; 329 | this.x += velocity.x * delta; 330 | this.y += velocity.y * delta; 331 | } else { 332 | if (drag[prop]) { 333 | this.hooks.velocities[prop] += this.hooks.defs.getDrag(this.hooks.velocities[prop], drag[prop], this.hooks.defs.zoom); 334 | } 335 | this.hooks.velocities[prop] += acceleration[prop]; 336 | this[prop] += this.hooks.velocities[prop] * delta; 337 | } 338 | } 339 | return typeof this.onHookUpdate === "function" ? this.onHookUpdate(delta) : void 0; 340 | } 341 | }; 342 | 343 | 344 | },{}]},{},[]) 345 | //# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/sigurd/Repos/framer-hook/hook-example-modulator.framer/modules/Hook.coffee"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","###\n--------------------------------------------------------------------------------\nHook module for Framer\n--------------------------------------------------------------------------------\n\nby:      Sigurd Mannsåker\ngithub:  https://github.com/sigtm/framer-hook\n\n················································································\n\n\nThe Hook module simply expands the Layer prototype, and lets you make any\nnumeric Layer property follow another property - either its own or another\nobject's - via a spring or gravity attraction.\n\n\n--------------------------------------------------------------------------------\nExample: Layered animation (eased + spring)\n--------------------------------------------------------------------------------\n\nmyLayer = new Layer\n\n# Make our own custom property for the x property to follow\nmyLayer.easedX = 0\n\n# Hook x to easedX via a spring\nmyLayer.hook\n\tproperty: \"x\"\n\ttargetProperty: \"easedX\"\n\ttype: \"spring(150, 15)\"\n\n# Animate easedX\nmyLayer.animate\n\tproperties:\n\t\teasedX: 200\n\ttime: 0.15\n\tcurve: \"cubic-bezier(0.2, 0, 0.4, 1)\"\n\nNOTE: \nTo attach both the x and y position, use \"pos\", \"midPos\" or \"maxPos\" as the\nproperty/targetProperty.\n\n\n--------------------------------------------------------------------------------\nExample: Hooking property to another layer\n--------------------------------------------------------------------------------\n\ntarget = new Layer\nhooked = new Layer\n\nhooked.hook\n\tproperty: \"scale\"\n\tto: target\n\ttype: \"spring(150, 15)\"\n\nThe \"hooked\" layer's scale will now continuously follow the target layer's scale\nwith a spring animation.\n\n\n--------------------------------------------------------------------------------\nlayer.hook(options)\n--------------------------------------------------------------------------------\n\nOptions are passed as a single object, like you would for a new Layer.\nThe options object takes the following properties:\n\n\nproperty [String]\n-----------------\nThe property you'd like to hook onto another object's property\n\n\ntype [String]\n-------------\nEither \"spring(strength, friction)\" or \"gravity(strength, drag)\". Only the last\nspecified drag value is used for each property, since it is only applied to\neach property once (and only if it has a gravity hook applied to it.)\n\n\nto [Object] (Optional)\n----------------------\nThe object to attach it to. Defaults to itself.\n\n\ntargetProperty [String] (Optional)\n----------------------------------\nSpecify the target object's property to follow, if you don't want to follow\nthe same property that the hook is applied to.\n\n\nmodulator [Function] (Optional)\n-------------------------------\nThe modulator function receives the target property's value, and lets you\nmodify it before it is fed into the physics calculations. Useful for anything\nfrom standard Utils.modulate() type stuff to snapping and conditional values.\n\n\nzoom [Number] (Optional)\n------------------------\nThis factor defines the distance that 1px represents in regards to gravity and\ndrag calculations. Only one value is stored per layer, so specifying it\noverwrites its existing value. Default is 100.\n\n\n--------------------------------------------------------------------------------\nlayer.unHook(property, object)\n--------------------------------------------------------------------------------\n\nThis removes all hooks for a given property and target object. Example:\n\n# Hook it\nlayer.hook\n\tproperty: \"x\"\n\tto: \"otherlayer\"\n\ttargetProperty: \"y\"\n\ttype: \"spring(200,20)\"\n\n# Unhook it\nlayer.unHook \"x\", otherlayer\n\n\n--------------------------------------------------------------------------------\nlayer.onHookUpdate(delta)\n--------------------------------------------------------------------------------\n\nAfter a layer is done applying accelerations to its hooked properties, it calls\nonHookUpdate() at the end of each frame, if it is defined. This is an easy way\nto animate or trigger other stuff, perhaps based on your layer's updated\nproperties or velocities.\n\nThe delta value from the Framer loop is passed on to onHookUpdate() as well,\nwhich is the time in seconds since the last animation frame.\n\nNote that if you unhook all your hooks, onHookUpdate() will of course no longer\nbe called for that layer.\n\n###\n\n\n# Since older versions of Safari seem to be missing String.prototype.includes()\n\nunless String.prototype.includes\n\tString::includes = (search, start) ->\n\t\t'use strict'\n\t\tstart = 0 if typeof start is 'number'\n\n\t\tif start + search.length > this.length\n\t\t\treturn false;\n\t\telse\n\t\t\treturn @indexOf(search, start) isnt -1\n\n# Expand layer\n\nLayer::hook = (config) ->\n\n\tthrow new Error 'layer.hook() needs a property, a hook type and either a target object or target property to work' unless config.property and config.type and (config.to or config.targetProperty)\n\n\t# Single array for all hooks, as opposed to nested arrays per property, because performance\n\t@hooks ?=\n\t\thooks: []\n\t\tvelocities: {}\n\t\tdefs:\n\t\t\tzoom: 100\n\t\t\tgetDrag: (velocity, drag, zoom) =>\n\t\t\t\tvelocity /= zoom\n\t\t\t\t# Dividing by 10 is unscientific, but it means a value of 2 equals roughly a 100g ball with 15cm radius in air\n\t\t\t\tdrag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity)\n\t\t\t\tif _.isNaN(drag) then return 0 else return drag\n\t\t\tgetGravity: (strength, distance, zoom) =>\n\t\t\t\tdist = Math.max(1, distance / zoom)\n\t\t\t\treturn strength * zoom / (dist * dist)\n\n\t# Update the zoom value if given\n\t@hooks.zoom = config.zoom if config.zoom\n\n\t# Parse physics config string\n\tf = Utils.parseFunction config.type\n\tconfig.type = f.name\n\tconfig.strength = f.args[0]\n\tconfig.friction = f.args[1] or 0\n\n\t# Default to same targetProperty on same object (hopefully you've set at least one of these to something else)\n\tconfig.targetProperty ?= config.property\n\tconfig.to ?= @\n\n\t# All position accelerations are added to a single 'pos' velocity. Store actual properties so we don't have to do it again every frame\n\n\tif config.property.toLowerCase().includes 'pos'\n\t\tconfig.prop = 'pos'\n\t\t\n\t\tif config.property.toLowerCase().includes 'mid'\n\t\t\tconfig.thisX = 'midX'\n\t\t\tconfig.thisY = 'midY'\n\t\t\n\t\telse if config.property.toLowerCase().includes 'max'\n\t\t\tconfig.thisX = 'maxX'\n\t\t\tconfig.thisY = 'maxY'\n\t\t\n\t\telse\n\t\t\tconfig.thisX = 'x'\n\t\t\tconfig.thisY = 'y'\n\t\t\n\t\tif config.targetProperty.toLowerCase().includes 'mid'\n\t\t\tconfig.toX = 'midX'\n\t\t\tconfig.toY = 'midY'\n\t\t\n\t\telse if config.targetProperty.toLowerCase().includes 'max'\n\t\t\tconfig.toX = 'maxX'\n\t\t\tconfig.toY = 'maxY'\t\t\n\t\telse\n\t\t\tconfig.toX = 'x'\n\t\t\tconfig.toY = 'y'\n\t\t\n\telse\n\t\tconfig.prop = config.property\n\n\t# Save hook to @hooks array\t\n\t@hooks.hooks.push(config)\n\n\t# Create velocity property if necessary\n\t@hooks.velocities[config.prop] ?= if config.prop is 'pos' then { x: 0, y: 0 } else 0\n\n\t# Use Framer's animation loop, slightly more robust than requestAnimationFrame directly\n\t# Save the returned AnimationLoop reference to make sure @hookLoop isn't added multiple times per layer\n\t@hooks.emitter ?= Framer.Loop.on('render', @hookLoop, this)\n\nLayer::unHook = (property, object) ->\n\t\n\treturn unless @hooks\n\n\tprop = if property.toLowerCase().includes 'pos' then 'pos' else property\n\n\t# Remove all matches\n\t@hooks.hooks = @hooks.hooks.filter (hook) ->\n\t\thook.to isnt object or hook.property isnt property\n\n\t# If there are no hooks left, shut it down\n\tif @hooks.hooks.length is 0\n\t\tdelete @hooks\n\t\tFramer.Loop.removeListener 'render', @hookLoop\n\t\treturn\n\n\t# Still here? Check if there are any remaining hooks affecting same velocity\n\tremaining = @hooks.hooks.filter (hook) ->\n\t\tprop is hook.prop\n\t\t\n\t# If not, delete velocity (otherwise it won't be reset if you make new hook for same property)\n\tdelete @hooks.velocities[prop] if remaining.length is 0\n\nLayer::hookLoop = (delta) ->\n\n\tif @hooks\n\n\t\t# Multiple hooks can affect the same property. Add accelerations to temporary object so the property's velocity is the same for all calculations within the same animation frame\n\t\tacceleration = {}\n\t\t\n\t\t# Save drag for each property to this object, since only most recently specified value is used for each property\n\t\tdrag = {}\n\t\t\n\t\t# Add accelerations\n\t\tfor hook in @hooks.hooks\n\t\t\n\t\t\tif hook.prop is 'pos'\n\n\t\t\t\tacceleration.pos ?= { x: 0, y: 0 }\n\n\t\t\t\ttarget = { x: hook.to[hook.toX], y: hook.to[hook.toY] }\n\n\t\t\t\ttarget = hook.modulator(target) if hook.modulator\n\n\t\t\t\tvector =\n\t\t\t\t\tx: target.x - @[hook.thisX]\n\t\t\t\t\ty: target.y - @[hook.thisY]\n\t\t\t\t\n\t\t\t\tvLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y))\n\n\t\t\t\tif hook.type is 'spring'\n\n\t\t\t\t\tdamper =\n\t\t\t\t\t\tx: -hook.friction * @hooks.velocities.pos.x\n\t\t\t\t\t\ty: -hook.friction * @hooks.velocities.pos.y\n\n\t\t\t\t\tvector.x *= hook.strength\n\t\t\t\t\tvector.y *= hook.strength\n\t\t\t\t\t\n\t\t\t\t\tacceleration.pos.x += (vector.x + damper.x) * delta\n\t\t\t\t\tacceleration.pos.y += (vector.y + damper.y) * delta\n\t\t\t\t\n\t\t\t\telse if hook.type is 'gravity'\n\t\t\t\t\n\t\t\t\t\tdrag.pos = hook.friction\n\t\t\t\t\t\n\t\t\t\t\tgravity = @hooks.defs.getGravity(hook.strength, vLength, @hooks.defs.zoom)\n\n\t\t\t\t\tvector.x *= gravity / vLength\n\t\t\t\t\tvector.y *= gravity / vLength\n\t\t\t\t\t\n\t\t\t\t\tacceleration.pos.x += vector.x * delta\n\t\t\t\t\tacceleration.pos.y += vector.y * delta\n\t\t\t\t\t\t\t\t\t\n\t\t\telse\n\t\t\t\t\n\t\t\t\tacceleration[hook.prop] ?= 0\n\n\t\t\t\ttarget = hook.to[hook.targetProperty]\n\n\t\t\t\ttarget = hook.modulator(target) if hook.modulator\n\n\t\t\t\tvector = target - @[hook.prop]\n\t\t\t\t\n\t\t\t\tif hook.type is 'spring'\n\n\t\t\t\t\tforce = vector * hook.strength\n\t\t\t\t\tdamper = -hook.friction * @hooks.velocities[hook.prop]\n\n\t\t\t\t\tacceleration[hook.prop] += (force + damper) * delta\n\n\t\t\t\t\n\t\t\t\telse if hook.type is 'gravity'\n\t\n\t\t\t\t\tdrag[hook.prop] = hook.friction\n\t\t\t\t\t\n\t\t\t\t\tforce = @hooks.defs.getGravity(hook.strength, vector, @hooks.defs.zoom)\n\t\t\t\t\t\n\t\t\t\t\tacceleration[hook.prop] += force * delta\n\t\t\n\t\t\n\t\t# Add velocities to properties. Doing this at the end in case there are multiple hooks affecting the same velocity\n\t\tfor prop, velocity of @hooks.velocities\n\t\t\n\t\t\tif prop is 'pos'\n\n\t\t\t\t# Add drag, if it exists\n\t\t\t\tif drag.pos\n\t\t\t\t\tvelocity.x += @hooks.defs.getDrag(velocity.x, drag.pos, @hooks.defs.zoom)\n\t\t\t\t\tvelocity.y += @hooks.defs.getDrag(velocity.y, drag.pos, @hooks.defs.zoom)\n\t\t\t\t\t\n\t\t\t\t# Add acceleration to velocity\n\t\t\t\tvelocity.x += acceleration.pos.x\n\t\t\t\tvelocity.y += acceleration.pos.y\n\t\t\t\t\n\t\t\t\t# Add velocity to position\n\t\t\t\t@x += velocity.x * delta\n\t\t\t\t@y += velocity.y * delta\n\t\t\t\n\t\t\telse\n\t\t\t\n\t\t\t\t# Add drag, if it exists\n\t\t\t\tif drag[prop]\n\t\t\t\t\t@hooks.velocities[prop] += @hooks.defs.getDrag(@hooks.velocities[prop], drag[prop], @hooks.defs.zoom)\n\t\t\t\t\n\t\t\t\t# Add acceleration to velocity\n\t\t\t\t@hooks.velocities[prop] += acceleration[prop]\n\t\t\t\t\n\t\t\t\t# Add velocity to property\n\t\t\t\t@[prop] += @hooks.velocities[prop] * delta\n\n\t\t@onHookUpdate?(delta)"]} 346 | -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/cursor-active.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/cursor-active.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/cursor-active@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/cursor-active@2x.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/cursor.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/cursor.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/cursor@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/cursor@2x.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/icon-120.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/icon-120.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/icon-152.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/icon-152.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/icon-180.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/icon-180.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/icon-192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/icon-192.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/images/icon-76.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/images/icon-76.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/manifest.txt: -------------------------------------------------------------------------------- 1 | app.coffee 2 | framer/coffee-script.js 3 | framer/config.json 4 | framer/framer.generated.js 5 | framer/framer.init.js 6 | framer/framer.js 7 | framer/framer.js.map 8 | framer/framer.modules.js 9 | framer/images/cursor-active.png 10 | framer/images/cursor-active@2x.png 11 | framer/images/cursor.png 12 | framer/images/cursor@2x.png 13 | framer/images/icon-120.png 14 | framer/images/icon-152.png 15 | framer/images/icon-180.png 16 | framer/images/icon-192.png 17 | framer/images/icon-76.png 18 | framer/preview.png 19 | framer/style.css 20 | framer/version 21 | images/spinner.svg 22 | imported/app@2x/images/Layer-page-meneqzew.png 23 | imported/app@2x/layers.json 24 | imported/app@2x/layers.json.js 25 | index.html 26 | modules/Hook.coffee -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/preview.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/framer/preview.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/style.css: -------------------------------------------------------------------------------- 1 | * { 2 | margin: 0; 3 | padding: 0; 4 | border: none; 5 | -webkit-user-select: none; 6 | -webkit-tap-highlight-color: rgba(0,0,0,0); 7 | } 8 | 9 | body { 10 | background-color: #fff; 11 | font: 28px/1em "Helvetica"; 12 | color: gray; 13 | overflow: hidden; 14 | } 15 | 16 | a { 17 | color: gray; 18 | } 19 | 20 | body { 21 | cursor: url('images/cursor.png') 32 32, auto; 22 | cursor: -webkit-image-set( 23 | url('images/cursor.png') 1x, 24 | url('images/cursor@2x.png') 2x 25 | ) 32 32, auto; 26 | } 27 | 28 | body:active { 29 | cursor: url('images/cursor-active.png') 32 32, auto; 30 | cursor: -webkit-image-set( 31 | url('images/cursor-active.png') 1x, 32 | url('images/cursor-active@2x.png') 2x 33 | ) 32 32, auto; 34 | } 35 | 36 | .framerAlertBackground { 37 | position: absolute; top:0px; left:0px; right:0px; bottom:0px; 38 | z-index: 1000; 39 | background-color: #fff; 40 | } 41 | 42 | .framerAlert { 43 | font:400 14px/1.4 "Helvetica Neue", Helvetica, Arial, sans-serif; 44 | -webkit-font-smoothing:antialiased; 45 | color:#616367; text-align:center; 46 | position: absolute; top:40%; left:50%; width:260px; margin-left:-130px; 47 | } 48 | .framerAlert strong { font-weight:500; color:#000; margin-bottom:8px; display:block; } 49 | .framerAlert a { color:#28AFFA; } 50 | .framerAlert .btn { 51 | font-weight:500; text-decoration:none; line-height:1; 52 | display:inline-block; padding:6px 12px 7px 12px; 53 | border-radius:3px; margin-top:12px; 54 | background:#28AFFA; color:#fff; 55 | } 56 | 57 | ::-webkit-scrollbar { 58 | display: none; 59 | } -------------------------------------------------------------------------------- /hook-example-modulator.framer/framer/version: -------------------------------------------------------------------------------- 1 | 4 -------------------------------------------------------------------------------- /hook-example-modulator.framer/images/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/images/.gitkeep -------------------------------------------------------------------------------- /hook-example-modulator.framer/images/spinner.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /hook-example-modulator.framer/imported/app@2x/images/Layer-page-meneqzew.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-modulator.framer/imported/app@2x/images/Layer-page-meneqzew.png -------------------------------------------------------------------------------- /hook-example-modulator.framer/imported/app@2x/layers.json: -------------------------------------------------------------------------------- 1 | [ 2 | { 3 | "objectId": "0CDC1067-9F1A-4B22-8CF3-3B8FEE516032", 4 | "kind": "group", 5 | "name": "page", 6 | "maskFrame": null, 7 | "layerFrame": { 8 | "x": 0, 9 | "y": 0, 10 | "width": 375, 11 | "height": 667 12 | }, 13 | "visible": true, 14 | "metadata": { 15 | "opacity": 1 16 | }, 17 | "image": { 18 | "path": "images/Layer-page-meneqzew.png", 19 | "frame": { 20 | "x": 0, 21 | "y": 0, 22 | "width": 375, 23 | "height": 667 24 | } 25 | }, 26 | "children": [], 27 | "time": 123 28 | } 29 | ] -------------------------------------------------------------------------------- /hook-example-modulator.framer/imported/app@2x/layers.json.js: -------------------------------------------------------------------------------- 1 | window.__imported__ = window.__imported__ || {}; 2 | window.__imported__["app@2x/layers.json.js"] = [ 3 | { 4 | "objectId": "0CDC1067-9F1A-4B22-8CF3-3B8FEE516032", 5 | "kind": "group", 6 | "name": "page", 7 | "maskFrame": null, 8 | "layerFrame": { 9 | "x": 0, 10 | "y": 0, 11 | "width": 375, 12 | "height": 667 13 | }, 14 | "visible": true, 15 | "metadata": { 16 | "opacity": 1 17 | }, 18 | "image": { 19 | "path": "images/Layer-page-meneqzew.png", 20 | "frame": { 21 | "x": 0, 22 | "y": 0, 23 | "width": 375, 24 | "height": 667 25 | } 26 | }, 27 | "children": [], 28 | "time": 123 29 | } 30 | ] -------------------------------------------------------------------------------- /hook-example-modulator.framer/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | -------------------------------------------------------------------------------- /hook-example-modulator.framer/modules/Hook.coffee: -------------------------------------------------------------------------------- 1 | ### 2 | -------------------------------------------------------------------------------- 3 | Hook module for Framer 4 | -------------------------------------------------------------------------------- 5 | 6 | by: Sigurd Mannsåker 7 | github: https://github.com/sigtm/framer-hook 8 | 9 | ················································································ 10 | 11 | 12 | The Hook module simply expands the Layer prototype, and lets you make any 13 | numeric Layer property follow another property - either its own or another 14 | object's - via a spring or gravity attraction. 15 | 16 | 17 | -------------------------------------------------------------------------------- 18 | Example: Layered animation (eased + spring) 19 | -------------------------------------------------------------------------------- 20 | 21 | myLayer = new Layer 22 | 23 | # Make our own custom property for the x property to follow 24 | myLayer.easedX = 0 25 | 26 | # Hook x to easedX via a spring 27 | myLayer.hook 28 | property: "x" 29 | targetProperty: "easedX" 30 | type: "spring(150, 15)" 31 | 32 | # Animate easedX 33 | myLayer.animate 34 | properties: 35 | easedX: 200 36 | time: 0.15 37 | curve: "cubic-bezier(0.2, 0, 0.4, 1)" 38 | 39 | NOTE: 40 | To attach both the x and y position, use "pos", "midPos" or "maxPos" as the 41 | property/targetProperty. 42 | 43 | 44 | -------------------------------------------------------------------------------- 45 | Example: Hooking property to another layer 46 | -------------------------------------------------------------------------------- 47 | 48 | target = new Layer 49 | hooked = new Layer 50 | 51 | hooked.hook 52 | property: "scale" 53 | to: target 54 | type: "spring(150, 15)" 55 | 56 | The "hooked" layer's scale will now continuously follow the target layer's scale 57 | with a spring animation. 58 | 59 | 60 | -------------------------------------------------------------------------------- 61 | layer.hook(options) 62 | -------------------------------------------------------------------------------- 63 | 64 | Options are passed as a single object, like you would for a new Layer. 65 | The options object takes the following properties: 66 | 67 | 68 | property [String] 69 | ----------------- 70 | The property you'd like to hook onto another object's property 71 | 72 | 73 | type [String] 74 | ------------- 75 | Either "spring(strength, friction)" or "gravity(strength, drag)". Only the last 76 | specified drag value is used for each property, since it is only applied to 77 | each property once (and only if it has a gravity hook applied to it.) 78 | 79 | 80 | to [Object] (Optional) 81 | ---------------------- 82 | The object to attach it to. Defaults to itself. 83 | 84 | 85 | targetProperty [String] (Optional) 86 | ---------------------------------- 87 | Specify the target object's property to follow, if you don't want to follow 88 | the same property that the hook is applied to. 89 | 90 | 91 | modulator [Function] (Optional) 92 | ------------------------------- 93 | The modulator function receives the target property's value, and lets you 94 | modify it before it is fed into the physics calculations. Useful for anything 95 | from standard Utils.modulate() type stuff to snapping and conditional values. 96 | 97 | 98 | zoom [Number] (Optional) 99 | ------------------------ 100 | This factor defines the distance that 1px represents in regards to gravity and 101 | drag calculations. Only one value is stored per layer, so specifying it 102 | overwrites its existing value. Default is 100. 103 | 104 | 105 | -------------------------------------------------------------------------------- 106 | layer.unHook(property, object) 107 | -------------------------------------------------------------------------------- 108 | 109 | This removes all hooks for a given property and target object. Example: 110 | 111 | # Hook it 112 | layer.hook 113 | property: "x" 114 | to: "otherlayer" 115 | targetProperty: "y" 116 | type: "spring(200,20)" 117 | 118 | # Unhook it 119 | layer.unHook "x", otherlayer 120 | 121 | 122 | -------------------------------------------------------------------------------- 123 | layer.onHookUpdate(delta) 124 | -------------------------------------------------------------------------------- 125 | 126 | After a layer is done applying accelerations to its hooked properties, it calls 127 | onHookUpdate() at the end of each frame, if it is defined. This is an easy way 128 | to animate or trigger other stuff, perhaps based on your layer's updated 129 | properties or velocities. 130 | 131 | The delta value from the Framer loop is passed on to onHookUpdate() as well, 132 | which is the time in seconds since the last animation frame. 133 | 134 | Note that if you unhook all your hooks, onHookUpdate() will of course no longer 135 | be called for that layer. 136 | 137 | ### 138 | 139 | 140 | # Since older versions of Safari seem to be missing String.prototype.includes() 141 | 142 | unless String.prototype.includes 143 | String::includes = (search, start) -> 144 | 'use strict' 145 | start = 0 if typeof start is 'number' 146 | 147 | if start + search.length > this.length 148 | return false; 149 | else 150 | return @indexOf(search, start) isnt -1 151 | 152 | # Expand layer 153 | 154 | Layer::hook = (config) -> 155 | 156 | throw new Error 'layer.hook() needs a property, a hook type and either a target object or target property to work' unless config.property and config.type and (config.to or config.targetProperty) 157 | 158 | # Single array for all hooks, as opposed to nested arrays per property, because performance 159 | @hooks ?= 160 | hooks: [] 161 | velocities: {} 162 | defs: 163 | zoom: 100 164 | getDrag: (velocity, drag, zoom) => 165 | velocity /= zoom 166 | # Dividing by 10 is unscientific, but it means a value of 2 equals roughly a 100g ball with 15cm radius in air 167 | drag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity) 168 | if _.isNaN(drag) then return 0 else return drag 169 | getGravity: (strength, distance, zoom) => 170 | dist = Math.max(1, distance / zoom) 171 | return strength * zoom / (dist * dist) 172 | 173 | # Update the zoom value if given 174 | @hooks.zoom = config.zoom if config.zoom 175 | 176 | # Parse physics config string 177 | f = Utils.parseFunction config.type 178 | config.type = f.name 179 | config.strength = f.args[0] 180 | config.friction = f.args[1] or 0 181 | 182 | # Default to same targetProperty on same object (hopefully you've set at least one of these to something else) 183 | config.targetProperty ?= config.property 184 | config.to ?= @ 185 | 186 | # All position accelerations are added to a single 'pos' velocity. Store actual properties so we don't have to do it again every frame 187 | 188 | if config.property.toLowerCase().includes 'pos' 189 | config.prop = 'pos' 190 | 191 | if config.property.toLowerCase().includes 'mid' 192 | config.thisX = 'midX' 193 | config.thisY = 'midY' 194 | 195 | else if config.property.toLowerCase().includes 'max' 196 | config.thisX = 'maxX' 197 | config.thisY = 'maxY' 198 | 199 | else 200 | config.thisX = 'x' 201 | config.thisY = 'y' 202 | 203 | if config.targetProperty.toLowerCase().includes 'mid' 204 | config.toX = 'midX' 205 | config.toY = 'midY' 206 | 207 | else if config.targetProperty.toLowerCase().includes 'max' 208 | config.toX = 'maxX' 209 | config.toY = 'maxY' 210 | else 211 | config.toX = 'x' 212 | config.toY = 'y' 213 | 214 | else 215 | config.prop = config.property 216 | 217 | # Save hook to @hooks array 218 | @hooks.hooks.push(config) 219 | 220 | # Create velocity property if necessary 221 | @hooks.velocities[config.prop] ?= if config.prop is 'pos' then { x: 0, y: 0 } else 0 222 | 223 | # Use Framer's animation loop, slightly more robust than requestAnimationFrame directly 224 | # Save the returned AnimationLoop reference to make sure @hookLoop isn't added multiple times per layer 225 | @hooks.emitter ?= Framer.Loop.on('render', @hookLoop, this) 226 | 227 | Layer::unHook = (property, object) -> 228 | 229 | return unless @hooks 230 | 231 | prop = if property.toLowerCase().includes 'pos' then 'pos' else property 232 | 233 | # Remove all matches 234 | @hooks.hooks = @hooks.hooks.filter (hook) -> 235 | hook.to isnt object or hook.property isnt property 236 | 237 | # If there are no hooks left, shut it down 238 | if @hooks.hooks.length is 0 239 | delete @hooks 240 | Framer.Loop.removeListener 'render', @hookLoop 241 | return 242 | 243 | # Still here? Check if there are any remaining hooks affecting same velocity 244 | remaining = @hooks.hooks.filter (hook) -> 245 | prop is hook.prop 246 | 247 | # If not, delete velocity (otherwise it won't be reset if you make new hook for same property) 248 | delete @hooks.velocities[prop] if remaining.length is 0 249 | 250 | Layer::hookLoop = (delta) -> 251 | 252 | if @hooks 253 | 254 | # Multiple hooks can affect the same property. Add accelerations to temporary object so the property's velocity is the same for all calculations within the same animation frame 255 | acceleration = {} 256 | 257 | # Save drag for each property to this object, since only most recently specified value is used for each property 258 | drag = {} 259 | 260 | # Add accelerations 261 | for hook in @hooks.hooks 262 | 263 | if hook.prop is 'pos' 264 | 265 | acceleration.pos ?= { x: 0, y: 0 } 266 | 267 | target = { x: hook.to[hook.toX], y: hook.to[hook.toY] } 268 | 269 | target = hook.modulator(target) if hook.modulator 270 | 271 | vector = 272 | x: target.x - @[hook.thisX] 273 | y: target.y - @[hook.thisY] 274 | 275 | vLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y)) 276 | 277 | if hook.type is 'spring' 278 | 279 | damper = 280 | x: -hook.friction * @hooks.velocities.pos.x 281 | y: -hook.friction * @hooks.velocities.pos.y 282 | 283 | vector.x *= hook.strength 284 | vector.y *= hook.strength 285 | 286 | acceleration.pos.x += (vector.x + damper.x) * delta 287 | acceleration.pos.y += (vector.y + damper.y) * delta 288 | 289 | else if hook.type is 'gravity' 290 | 291 | drag.pos = hook.friction 292 | 293 | gravity = @hooks.defs.getGravity(hook.strength, vLength, @hooks.defs.zoom) 294 | 295 | vector.x *= gravity / vLength 296 | vector.y *= gravity / vLength 297 | 298 | acceleration.pos.x += vector.x * delta 299 | acceleration.pos.y += vector.y * delta 300 | 301 | else 302 | 303 | acceleration[hook.prop] ?= 0 304 | 305 | target = hook.to[hook.targetProperty] 306 | 307 | target = hook.modulator(target) if hook.modulator 308 | 309 | vector = target - @[hook.prop] 310 | 311 | if hook.type is 'spring' 312 | 313 | force = vector * hook.strength 314 | damper = -hook.friction * @hooks.velocities[hook.prop] 315 | 316 | acceleration[hook.prop] += (force + damper) * delta 317 | 318 | 319 | else if hook.type is 'gravity' 320 | 321 | drag[hook.prop] = hook.friction 322 | 323 | force = @hooks.defs.getGravity(hook.strength, vector, @hooks.defs.zoom) 324 | 325 | acceleration[hook.prop] += force * delta 326 | 327 | 328 | # Add velocities to properties. Doing this at the end in case there are multiple hooks affecting the same velocity 329 | for prop, velocity of @hooks.velocities 330 | 331 | if prop is 'pos' 332 | 333 | # Add drag, if it exists 334 | if drag.pos 335 | velocity.x += @hooks.defs.getDrag(velocity.x, drag.pos, @hooks.defs.zoom) 336 | velocity.y += @hooks.defs.getDrag(velocity.y, drag.pos, @hooks.defs.zoom) 337 | 338 | # Add acceleration to velocity 339 | velocity.x += acceleration.pos.x 340 | velocity.y += acceleration.pos.y 341 | 342 | # Add velocity to position 343 | @x += velocity.x * delta 344 | @y += velocity.y * delta 345 | 346 | else 347 | 348 | # Add drag, if it exists 349 | if drag[prop] 350 | @hooks.velocities[prop] += @hooks.defs.getDrag(@hooks.velocities[prop], drag[prop], @hooks.defs.zoom) 351 | 352 | # Add acceleration to velocity 353 | @hooks.velocities[prop] += acceleration[prop] 354 | 355 | # Add velocity to property 356 | @[prop] += @hooks.velocities[prop] * delta 357 | 358 | @onHookUpdate?(delta) -------------------------------------------------------------------------------- /hook-example-spring.framer/.gitignore: -------------------------------------------------------------------------------- 1 | # Framer Git Ignore 2 | 3 | # General OSX 4 | 5 | .DS_Store 6 | .AppleDouble 7 | .LSOverride 8 | 9 | # Icon must end with two \r 10 | Icon 11 | 12 | 13 | # Thumbnails 14 | ._* 15 | 16 | # Files that might appear in the root of a volume 17 | .DocumentRevisions-V100 18 | .fseventsd 19 | .Spotlight-V100 20 | .TemporaryItems 21 | .Trashes 22 | .VolumeIcon.icns 23 | 24 | # Directories potentially created on remote AFP share 25 | .AppleDB 26 | .AppleDesktop 27 | Network Trash Folder 28 | Temporary Items 29 | .apdisk 30 | 31 | 32 | # Framer Specific 33 | .temp.html 34 | framer/*.old* 35 | framer/backup.coffee 36 | framer/backups/* 37 | framer/.*.hash 38 | -------------------------------------------------------------------------------- /hook-example-spring.framer/.viewer.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | -------------------------------------------------------------------------------- /hook-example-spring.framer/app.coffee: -------------------------------------------------------------------------------- 1 | ### 2 | 3 | Hook example 4 | -------------- 5 | 6 | Note: This example uses two layers to illustrate the concept more clearly, but 7 | the separate layer for the eased animation is unnecessary. We could just as easily 8 | have created another property, say layer.easedX, and hooked the x property to that. 9 | Then you'd just run layer.animate() on the easedX property like you would any 10 | other property. 11 | 12 | The Hook module simply expands the Layer prototype, and lets you make any 13 | numeric Layer property follow a property on another object via a spring or a 14 | gravity attraction. Check the comments in modules/Hook.coffee for a more thorough 15 | documentation. 16 | 17 | ### 18 | 19 | 20 | 21 | # Require Hook. No exports, it just adds methods to the Layer prototype 22 | # -------------------------------------------------------------------------------- 23 | 24 | require "Hook" 25 | 26 | 27 | 28 | # Settings 29 | # -------------------------------------------------------------------------------- 30 | 31 | colors = 32 | background: "#260355" 33 | orange: "#f90" 34 | blue: "#63ffff" 35 | primary: "white" 36 | secondary: "rgba(255,255,255,0.1)" 37 | 38 | margin = 60 39 | 40 | easeInOut = "cubic-bezier(0.2, 0, 0.4, 1)" 41 | 42 | 43 | 44 | # Set background and defaults 45 | # -------------------------------------------------------------------------------- 46 | 47 | Framer.Device.viewport.backgroundColor = colors.background 48 | 49 | Framer.Defaults.Animation = 50 | curve: easeInOut 51 | time: 0.15 52 | 53 | 54 | 55 | # Set up example 56 | # -------------------------------------------------------------------------------- 57 | 58 | eased = new Layer 59 | x: margin 60 | y: 300 61 | width: 100 62 | height: 100 63 | backgroundColor: colors.blue 64 | borderRadius: 12 65 | 66 | eased.states.add 67 | right: 68 | maxX: Screen.width - margin 69 | 70 | hooked = new Layer 71 | x: eased.x 72 | y: eased.maxY + margin 73 | width: 100 74 | height: 100 75 | backgroundColor: colors.orange 76 | borderRadius: 12 77 | 78 | 79 | # We run hook() in the slider change callback 80 | runAnim = () -> 81 | 82 | # Remove any existing hooks, since we want updated spring values 83 | hooked.unHook 'x', eased 84 | 85 | # Attach the x property to the eased layer with the sliders' spring values 86 | hooked.hook 87 | property: 'x' 88 | to: eased 89 | type: 'spring(' + spring.value + ', ' + friction.value + ')' 90 | 91 | # Animate the eased layer 92 | eased.states.animationOptions = 93 | curve: easeInOut 94 | time: speed.value / 1000 95 | 96 | eased.states.next() 97 | 98 | 99 | 100 | # Styled slider class 101 | # -------------------------------------------------------------------------------- 102 | 103 | class Slider extends SliderComponent 104 | 105 | constructor: (config) -> 106 | super config 107 | 108 | # Basics 109 | @x = margin 110 | @width = Screen.width - margin * 2 111 | @height = 4 112 | @fill.backgroundColor = colors.primary 113 | @backgroundColor = colors.secondary 114 | @value = config.value 115 | @unitName = config.unit or '' 116 | 117 | @baseStyle = 118 | fontFamily: "Roboto" 119 | fontWeight: 500 120 | fontSize: "26px" 121 | 122 | # Knob 123 | @knobSize = 24 124 | @knob.backgroundColor = colors.primary 125 | @knob.borderRadius = 30 126 | @knob.shadowColor = colors.background 127 | @knob.shadowBlur = 0 128 | @knob.shadowSpread = 6 129 | 130 | # Name label 131 | @txtLabel = new Layer 132 | parent: @ 133 | name: "label" 134 | y: -50 135 | backgroundColor: "" 136 | html: config.label 137 | color: colors.primary 138 | style: @baseStyle 139 | 140 | @txtLabel.style.fontStyle = "italic" 141 | 142 | # Value label 143 | @txtValue = new Layer 144 | parent: @ 145 | name: "value" 146 | maxX: @width 147 | y: -50 148 | backgroundColor: "" 149 | html: Math.round(@value) + ' ' + @unitName 150 | color: config.valueColor or colors.orange 151 | style: @baseStyle 152 | 153 | @txtValue.style.textAlign = "right" 154 | 155 | # Events 156 | 157 | @onValueChange -> 158 | @txtValue.html = Math.round(@value) + ' ' + @unitName 159 | 160 | @onTouchStart -> 161 | @knob.animate 162 | properties: 163 | scale: 1.4 164 | 165 | @onTouchEnd -> 166 | runAnim() 167 | @knob.animate 168 | properties: 169 | scale: 1 170 | 171 | 172 | 173 | # Instantiate sliders 174 | # -------------------------------------------------------------------------------- 175 | 176 | friction = new Slider 177 | label: "Friction" 178 | maxY: Screen.height - margin * 1.5 179 | min: 0 180 | max: 50 181 | value: 18 182 | 183 | spring = new Slider 184 | label: "Spring" 185 | y: friction.y - margin * 2 186 | min: 0 187 | max: 200 188 | value: 150 189 | 190 | speed = new Slider 191 | label: "Time" 192 | unit: "ms" 193 | valueColor: colors.blue 194 | y: spring.y - margin * 2 195 | min: 0 196 | max: 1000 197 | value: 200 198 | 199 | 200 | -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/config.json: -------------------------------------------------------------------------------- 1 | { 2 | "propertyPanelToggleStates" : { 3 | 4 | }, 5 | "deviceOrientation" : 0, 6 | "sharedPrototype" : 0, 7 | "contentScale" : 1, 8 | "deviceType" : "apple-iphone-6s-silver", 9 | "selectedHand" : "", 10 | "updateDelay" : 0.3, 11 | "deviceScale" : "fit", 12 | "foldedCodeRanges" : [ 13 | 14 | ], 15 | "orientation" : 0, 16 | "fullScreen" : false 17 | } -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/framer.generated.js: -------------------------------------------------------------------------------- 1 | // This is autogenerated by Framer 2 | 3 | 4 | if (!window.Framer && window._bridge) {window._bridge('runtime.error', {message:'[framer.js] Framer library missing or corrupt. Select File → Update Framer Library.'})} 5 | if (DeviceComponent) {DeviceComponent.Devices["iphone-6-silver"].deviceImageJP2 = false}; 6 | if (window.Framer) {window.Framer.Defaults.DeviceView = {"deviceScale":"fit","selectedHand":"","deviceType":"apple-iphone-6s-silver","contentScale":1,"orientation":0}; 7 | } 8 | if (window.Framer) {window.Framer.Defaults.DeviceComponent = {"deviceScale":"fit","selectedHand":"","deviceType":"apple-iphone-6s-silver","contentScale":1,"orientation":0}; 9 | } 10 | window.FramerStudioInfo = {"deviceImagesUrl":"\/_server\/resources\/DeviceImages","documentTitle":"hook-example-spring.framer"}; 11 | 12 | Framer.Device = new Framer.DeviceView(); 13 | Framer.Device.setupContext(); -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/framer.init.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | 3 | function isFileLoadingAllowed() { 4 | return (window.location.protocol.indexOf("file") == -1) 5 | } 6 | 7 | function isHomeScreened() { 8 | return ("standalone" in window.navigator) && window.navigator.standalone == true 9 | } 10 | 11 | function isCompatibleBrowser() { 12 | return Utils.isWebKit() 13 | } 14 | 15 | var alertNode; 16 | 17 | function dismissAlert() { 18 | alertNode.parentElement.removeChild(alertNode) 19 | loadProject() 20 | } 21 | 22 | function showAlert(html) { 23 | 24 | alertNode = document.createElement("div") 25 | 26 | alertNode.classList.add("framerAlertBackground") 27 | alertNode.innerHTML = html 28 | 29 | document.addEventListener("DOMContentLoaded", function(event) { 30 | document.body.appendChild(alertNode) 31 | }) 32 | 33 | window.dismissAlert = dismissAlert; 34 | } 35 | 36 | function showBrowserAlert() { 37 | var html = "" 38 | html += "
" 39 | html += "Error: Not A WebKit Browser" 40 | html += "Your browser is not supported.
Please use Safari or Chrome.
" 41 | html += "Try anyway" 42 | html += "
" 43 | 44 | showAlert(html) 45 | } 46 | 47 | function showFileLoadingAlert() { 48 | var html = "" 49 | html += "
" 50 | html += "Error: Local File Restrictions" 51 | html += "Preview this prototype with Framer Mirror or learn more about " 52 | html += "file restrictions.
" 53 | html += "Try anyway" 54 | html += "
" 55 | 56 | showAlert(html) 57 | } 58 | 59 | function loadProject() { 60 | CoffeeScript.load("app.coffee") 61 | } 62 | 63 | function setDefaultPageTitle() { 64 | // If no title was set we set it to the project folder name so 65 | // you get a nice name on iOS if you bookmark to desktop. 66 | document.addEventListener("DOMContentLoaded", function() { 67 | if (document.title == "") { 68 | if (window.FramerStudioInfo && window.FramerStudioInfo.documentTitle) { 69 | document.title = window.FramerStudioInfo.documentTitle 70 | } else { 71 | document.title = window.location.pathname.replace(/\//g, "") 72 | } 73 | } 74 | }) 75 | } 76 | 77 | function init() { 78 | 79 | if (Utils.isFramerStudio()) { 80 | return 81 | } 82 | 83 | setDefaultPageTitle() 84 | 85 | if (!isCompatibleBrowser()) { 86 | return showBrowserAlert() 87 | } 88 | 89 | if (!isFileLoadingAllowed()) { 90 | return showFileLoadingAlert() 91 | } 92 | 93 | loadProject() 94 | 95 | } 96 | 97 | init() 98 | 99 | })() 100 | -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/framer.modules.js: -------------------------------------------------------------------------------- 1 | require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o this.length) { 146 | return false; 147 | } else { 148 | return this.indexOf(search, start) !== -1; 149 | } 150 | }; 151 | } 152 | 153 | Layer.prototype.hook = function(config) { 154 | var base, base1, f, name; 155 | if (!(config.property && config.type && (config.to || config.targetProperty))) { 156 | throw new Error('layer.hook() needs a property, a hook type and either a target object or target property to work'); 157 | } 158 | if (this.hooks == null) { 159 | this.hooks = { 160 | hooks: [], 161 | velocities: {}, 162 | defs: { 163 | zoom: 100, 164 | getDrag: (function(_this) { 165 | return function(velocity, drag, zoom) { 166 | velocity /= zoom; 167 | drag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity); 168 | if (_.isNaN(drag)) { 169 | return 0; 170 | } else { 171 | return drag; 172 | } 173 | }; 174 | })(this), 175 | getGravity: (function(_this) { 176 | return function(strength, distance, zoom) { 177 | var dist; 178 | dist = Math.max(1, distance / zoom); 179 | return strength * zoom / (dist * dist); 180 | }; 181 | })(this) 182 | } 183 | }; 184 | } 185 | if (config.zoom) { 186 | this.hooks.zoom = config.zoom; 187 | } 188 | f = Utils.parseFunction(config.type); 189 | config.type = f.name; 190 | config.strength = f.args[0]; 191 | config.friction = f.args[1] || 0; 192 | if (config.targetProperty == null) { 193 | config.targetProperty = config.property; 194 | } 195 | if (config.to == null) { 196 | config.to = this; 197 | } 198 | if (config.property.toLowerCase().includes('pos')) { 199 | config.prop = 'pos'; 200 | if (config.property.toLowerCase().includes('mid')) { 201 | config.thisX = 'midX'; 202 | config.thisY = 'midY'; 203 | } else if (config.property.toLowerCase().includes('max')) { 204 | config.thisX = 'maxX'; 205 | config.thisY = 'maxY'; 206 | } else { 207 | config.thisX = 'x'; 208 | config.thisY = 'y'; 209 | } 210 | if (config.targetProperty.toLowerCase().includes('mid')) { 211 | config.toX = 'midX'; 212 | config.toY = 'midY'; 213 | } else if (config.targetProperty.toLowerCase().includes('max')) { 214 | config.toX = 'maxX'; 215 | config.toY = 'maxY'; 216 | } else { 217 | config.toX = 'x'; 218 | config.toY = 'y'; 219 | } 220 | } else { 221 | config.prop = config.property; 222 | } 223 | this.hooks.hooks.push(config); 224 | if ((base = this.hooks.velocities)[name = config.prop] == null) { 225 | base[name] = config.prop === 'pos' ? { 226 | x: 0, 227 | y: 0 228 | } : 0; 229 | } 230 | return (base1 = this.hooks).emitter != null ? base1.emitter : base1.emitter = Framer.Loop.on('render', this.hookLoop, this); 231 | }; 232 | 233 | Layer.prototype.unHook = function(property, object) { 234 | var prop, remaining; 235 | if (!this.hooks) { 236 | return; 237 | } 238 | prop = property.toLowerCase().includes('pos') ? 'pos' : property; 239 | this.hooks.hooks = this.hooks.hooks.filter(function(hook) { 240 | return hook.to !== object || hook.property !== property; 241 | }); 242 | if (this.hooks.hooks.length === 0) { 243 | delete this.hooks; 244 | Framer.Loop.removeListener('render', this.hookLoop); 245 | return; 246 | } 247 | remaining = this.hooks.hooks.filter(function(hook) { 248 | return prop === hook.prop; 249 | }); 250 | if (remaining.length === 0) { 251 | return delete this.hooks.velocities[prop]; 252 | } 253 | }; 254 | 255 | Layer.prototype.hookLoop = function(delta) { 256 | var acceleration, damper, drag, force, gravity, hook, i, len, name, prop, ref, ref1, target, vLength, vector, velocity; 257 | if (this.hooks) { 258 | acceleration = {}; 259 | drag = {}; 260 | ref = this.hooks.hooks; 261 | for (i = 0, len = ref.length; i < len; i++) { 262 | hook = ref[i]; 263 | if (hook.prop === 'pos') { 264 | if (acceleration.pos == null) { 265 | acceleration.pos = { 266 | x: 0, 267 | y: 0 268 | }; 269 | } 270 | target = { 271 | x: hook.to[hook.toX], 272 | y: hook.to[hook.toY] 273 | }; 274 | if (hook.modulator) { 275 | target = hook.modulator(target); 276 | } 277 | vector = { 278 | x: target.x - this[hook.thisX], 279 | y: target.y - this[hook.thisY] 280 | }; 281 | vLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y)); 282 | if (hook.type === 'spring') { 283 | damper = { 284 | x: -hook.friction * this.hooks.velocities.pos.x, 285 | y: -hook.friction * this.hooks.velocities.pos.y 286 | }; 287 | vector.x *= hook.strength; 288 | vector.y *= hook.strength; 289 | acceleration.pos.x += (vector.x + damper.x) * delta; 290 | acceleration.pos.y += (vector.y + damper.y) * delta; 291 | } else if (hook.type === 'gravity') { 292 | drag.pos = hook.friction; 293 | gravity = this.hooks.defs.getGravity(hook.strength, vLength, this.hooks.defs.zoom); 294 | vector.x *= gravity / vLength; 295 | vector.y *= gravity / vLength; 296 | acceleration.pos.x += vector.x * delta; 297 | acceleration.pos.y += vector.y * delta; 298 | } 299 | } else { 300 | if (acceleration[name = hook.prop] == null) { 301 | acceleration[name] = 0; 302 | } 303 | target = hook.to[hook.targetProperty]; 304 | if (hook.modulator) { 305 | target = hook.modulator(target); 306 | } 307 | vector = target - this[hook.prop]; 308 | if (hook.type === 'spring') { 309 | force = vector * hook.strength; 310 | damper = -hook.friction * this.hooks.velocities[hook.prop]; 311 | acceleration[hook.prop] += (force + damper) * delta; 312 | } else if (hook.type === 'gravity') { 313 | drag[hook.prop] = hook.friction; 314 | force = this.hooks.defs.getGravity(hook.strength, vector, this.hooks.defs.zoom); 315 | acceleration[hook.prop] += force * delta; 316 | } 317 | } 318 | } 319 | ref1 = this.hooks.velocities; 320 | for (prop in ref1) { 321 | velocity = ref1[prop]; 322 | if (prop === 'pos') { 323 | if (drag.pos) { 324 | velocity.x += this.hooks.defs.getDrag(velocity.x, drag.pos, this.hooks.defs.zoom); 325 | velocity.y += this.hooks.defs.getDrag(velocity.y, drag.pos, this.hooks.defs.zoom); 326 | } 327 | velocity.x += acceleration.pos.x; 328 | velocity.y += acceleration.pos.y; 329 | this.x += velocity.x * delta; 330 | this.y += velocity.y * delta; 331 | } else { 332 | if (drag[prop]) { 333 | this.hooks.velocities[prop] += this.hooks.defs.getDrag(this.hooks.velocities[prop], drag[prop], this.hooks.defs.zoom); 334 | } 335 | this.hooks.velocities[prop] += acceleration[prop]; 336 | this[prop] += this.hooks.velocities[prop] * delta; 337 | } 338 | } 339 | return typeof this.onHookUpdate === "function" ? this.onHookUpdate(delta) : void 0; 340 | } 341 | }; 342 | 343 | 344 | },{}]},{},[]) 345 | //# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/sigurd/Repos/framer-hook/hook-example-spring.framer/modules/Hook.coffee"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","###\n--------------------------------------------------------------------------------\nHook module for Framer\n--------------------------------------------------------------------------------\n\nby:      Sigurd Mannsåker\ngithub:  https://github.com/sigtm/framer-hook\n\n················································································\n\n\nThe Hook module simply expands the Layer prototype, and lets you make any\nnumeric Layer property follow another property - either its own or another\nobject's - via a spring or gravity attraction.\n\n\n--------------------------------------------------------------------------------\nExample: Layered animation (eased + spring)\n--------------------------------------------------------------------------------\n\nmyLayer = new Layer\n\n# Make our own custom property for the x property to follow\nmyLayer.easedX = 0\n\n# Hook x to easedX via a spring\nmyLayer.hook\n\tproperty: \"x\"\n\ttargetProperty: \"easedX\"\n\ttype: \"spring(150, 15)\"\n\n# Animate easedX\nmyLayer.animate\n\tproperties:\n\t\teasedX: 200\n\ttime: 0.15\n\tcurve: \"cubic-bezier(0.2, 0, 0.4, 1)\"\n\nNOTE: \nTo attach both the x and y position, use \"pos\", \"midPos\" or \"maxPos\" as the\nproperty/targetProperty.\n\n\n--------------------------------------------------------------------------------\nExample: Hooking property to another layer\n--------------------------------------------------------------------------------\n\ntarget = new Layer\nhooked = new Layer\n\nhooked.hook\n\tproperty: \"scale\"\n\tto: target\n\ttype: \"spring(150, 15)\"\n\nThe \"hooked\" layer's scale will now continuously follow the target layer's scale\nwith a spring animation.\n\n\n--------------------------------------------------------------------------------\nlayer.hook(options)\n--------------------------------------------------------------------------------\n\nOptions are passed as a single object, like you would for a new Layer.\nThe options object takes the following properties:\n\n\nproperty [String]\n-----------------\nThe property you'd like to hook onto another object's property\n\n\ntype [String]\n-------------\nEither \"spring(strength, friction)\" or \"gravity(strength, drag)\". Only the last\nspecified drag value is used for each property, since it is only applied to\neach property once (and only if it has a gravity hook applied to it.)\n\n\nto [Object] (Optional)\n----------------------\nThe object to attach it to. Defaults to itself.\n\n\ntargetProperty [String] (Optional)\n----------------------------------\nSpecify the target object's property to follow, if you don't want to follow\nthe same property that the hook is applied to.\n\n\nmodulator [Function] (Optional)\n-------------------------------\nThe modulator function receives the target property's value, and lets you\nmodify it before it is fed into the physics calculations. Useful for anything\nfrom standard Utils.modulate() type stuff to snapping and conditional values.\n\n\nzoom [Number] (Optional)\n------------------------\nThis factor defines the distance that 1px represents in regards to gravity and\ndrag calculations. Only one value is stored per layer, so specifying it\noverwrites its existing value. Default is 100.\n\n\n--------------------------------------------------------------------------------\nlayer.unHook(property, object)\n--------------------------------------------------------------------------------\n\nThis removes all hooks for a given property and target object. Example:\n\n# Hook it\nlayer.hook\n\tproperty: \"x\"\n\tto: \"otherlayer\"\n\ttargetProperty: \"y\"\n\ttype: \"spring(200,20)\"\n\n# Unhook it\nlayer.unHook \"x\", otherlayer\n\n\n--------------------------------------------------------------------------------\nlayer.onHookUpdate(delta)\n--------------------------------------------------------------------------------\n\nAfter a layer is done applying accelerations to its hooked properties, it calls\nonHookUpdate() at the end of each frame, if it is defined. This is an easy way\nto animate or trigger other stuff, perhaps based on your layer's updated\nproperties or velocities.\n\nThe delta value from the Framer loop is passed on to onHookUpdate() as well,\nwhich is the time in seconds since the last animation frame.\n\nNote that if you unhook all your hooks, onHookUpdate() will of course no longer\nbe called for that layer.\n\n###\n\n\n# Since older versions of Safari seem to be missing String.prototype.includes()\n\nunless String.prototype.includes\n\tString::includes = (search, start) ->\n\t\t'use strict'\n\t\tstart = 0 if typeof start is 'number'\n\n\t\tif start + search.length > this.length\n\t\t\treturn false;\n\t\telse\n\t\t\treturn @indexOf(search, start) isnt -1\n\n# Expand layer\n\nLayer::hook = (config) ->\n\n\tthrow new Error 'layer.hook() needs a property, a hook type and either a target object or target property to work' unless config.property and config.type and (config.to or config.targetProperty)\n\n\t# Single array for all hooks, as opposed to nested arrays per property, because performance\n\t@hooks ?=\n\t\thooks: []\n\t\tvelocities: {}\n\t\tdefs:\n\t\t\tzoom: 100\n\t\t\tgetDrag: (velocity, drag, zoom) =>\n\t\t\t\tvelocity /= zoom\n\t\t\t\t# Dividing by 10 is unscientific, but it means a value of 2 equals roughly a 100g ball with 15cm radius in air\n\t\t\t\tdrag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity)\n\t\t\t\tif _.isNaN(drag) then return 0 else return drag\n\t\t\tgetGravity: (strength, distance, zoom) =>\n\t\t\t\tdist = Math.max(1, distance / zoom)\n\t\t\t\treturn strength * zoom / (dist * dist)\n\n\t# Update the zoom value if given\n\t@hooks.zoom = config.zoom if config.zoom\n\n\t# Parse physics config string\n\tf = Utils.parseFunction config.type\n\tconfig.type = f.name\n\tconfig.strength = f.args[0]\n\tconfig.friction = f.args[1] or 0\n\n\t# Default to same targetProperty on same object (hopefully you've set at least one of these to something else)\n\tconfig.targetProperty ?= config.property\n\tconfig.to ?= @\n\n\t# All position accelerations are added to a single 'pos' velocity. Store actual properties so we don't have to do it again every frame\n\n\tif config.property.toLowerCase().includes 'pos'\n\t\tconfig.prop = 'pos'\n\t\t\n\t\tif config.property.toLowerCase().includes 'mid'\n\t\t\tconfig.thisX = 'midX'\n\t\t\tconfig.thisY = 'midY'\n\t\t\n\t\telse if config.property.toLowerCase().includes 'max'\n\t\t\tconfig.thisX = 'maxX'\n\t\t\tconfig.thisY = 'maxY'\n\t\t\n\t\telse\n\t\t\tconfig.thisX = 'x'\n\t\t\tconfig.thisY = 'y'\n\t\t\n\t\tif config.targetProperty.toLowerCase().includes 'mid'\n\t\t\tconfig.toX = 'midX'\n\t\t\tconfig.toY = 'midY'\n\t\t\n\t\telse if config.targetProperty.toLowerCase().includes 'max'\n\t\t\tconfig.toX = 'maxX'\n\t\t\tconfig.toY = 'maxY'\t\t\n\t\telse\n\t\t\tconfig.toX = 'x'\n\t\t\tconfig.toY = 'y'\n\t\t\n\telse\n\t\tconfig.prop = config.property\n\n\t# Save hook to @hooks array\t\n\t@hooks.hooks.push(config)\n\n\t# Create velocity property if necessary\n\t@hooks.velocities[config.prop] ?= if config.prop is 'pos' then { x: 0, y: 0 } else 0\n\n\t# Use Framer's animation loop, slightly more robust than requestAnimationFrame directly\n\t# Save the returned AnimationLoop reference to make sure @hookLoop isn't added multiple times per layer\n\t@hooks.emitter ?= Framer.Loop.on('render', @hookLoop, this)\n\nLayer::unHook = (property, object) ->\n\t\n\treturn unless @hooks\n\n\tprop = if property.toLowerCase().includes 'pos' then 'pos' else property\n\n\t# Remove all matches\n\t@hooks.hooks = @hooks.hooks.filter (hook) ->\n\t\thook.to isnt object or hook.property isnt property\n\n\t# If there are no hooks left, shut it down\n\tif @hooks.hooks.length is 0\n\t\tdelete @hooks\n\t\tFramer.Loop.removeListener 'render', @hookLoop\n\t\treturn\n\n\t# Still here? Check if there are any remaining hooks affecting same velocity\n\tremaining = @hooks.hooks.filter (hook) ->\n\t\tprop is hook.prop\n\t\t\n\t# If not, delete velocity (otherwise it won't be reset if you make new hook for same property)\n\tdelete @hooks.velocities[prop] if remaining.length is 0\n\nLayer::hookLoop = (delta) ->\n\n\tif @hooks\n\n\t\t# Multiple hooks can affect the same property. Add accelerations to temporary object so the property's velocity is the same for all calculations within the same animation frame\n\t\tacceleration = {}\n\t\t\n\t\t# Save drag for each property to this object, since only most recently specified value is used for each property\n\t\tdrag = {}\n\t\t\n\t\t# Add accelerations\n\t\tfor hook in @hooks.hooks\n\t\t\n\t\t\tif hook.prop is 'pos'\n\n\t\t\t\tacceleration.pos ?= { x: 0, y: 0 }\n\n\t\t\t\ttarget = { x: hook.to[hook.toX], y: hook.to[hook.toY] }\n\n\t\t\t\ttarget = hook.modulator(target) if hook.modulator\n\n\t\t\t\tvector =\n\t\t\t\t\tx: target.x - @[hook.thisX]\n\t\t\t\t\ty: target.y - @[hook.thisY]\n\t\t\t\t\n\t\t\t\tvLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y))\n\n\t\t\t\tif hook.type is 'spring'\n\n\t\t\t\t\tdamper =\n\t\t\t\t\t\tx: -hook.friction * @hooks.velocities.pos.x\n\t\t\t\t\t\ty: -hook.friction * @hooks.velocities.pos.y\n\n\t\t\t\t\tvector.x *= hook.strength\n\t\t\t\t\tvector.y *= hook.strength\n\t\t\t\t\t\n\t\t\t\t\tacceleration.pos.x += (vector.x + damper.x) * delta\n\t\t\t\t\tacceleration.pos.y += (vector.y + damper.y) * delta\n\t\t\t\t\n\t\t\t\telse if hook.type is 'gravity'\n\t\t\t\t\n\t\t\t\t\tdrag.pos = hook.friction\n\t\t\t\t\t\n\t\t\t\t\tgravity = @hooks.defs.getGravity(hook.strength, vLength, @hooks.defs.zoom)\n\n\t\t\t\t\tvector.x *= gravity / vLength\n\t\t\t\t\tvector.y *= gravity / vLength\n\t\t\t\t\t\n\t\t\t\t\tacceleration.pos.x += vector.x * delta\n\t\t\t\t\tacceleration.pos.y += vector.y * delta\n\t\t\t\t\t\t\t\t\t\n\t\t\telse\n\t\t\t\t\n\t\t\t\tacceleration[hook.prop] ?= 0\n\n\t\t\t\ttarget = hook.to[hook.targetProperty]\n\n\t\t\t\ttarget = hook.modulator(target) if hook.modulator\n\n\t\t\t\tvector = target - @[hook.prop]\n\t\t\t\t\n\t\t\t\tif hook.type is 'spring'\n\n\t\t\t\t\tforce = vector * hook.strength\n\t\t\t\t\tdamper = -hook.friction * @hooks.velocities[hook.prop]\n\n\t\t\t\t\tacceleration[hook.prop] += (force + damper) * delta\n\n\t\t\t\t\n\t\t\t\telse if hook.type is 'gravity'\n\t\n\t\t\t\t\tdrag[hook.prop] = hook.friction\n\t\t\t\t\t\n\t\t\t\t\tforce = @hooks.defs.getGravity(hook.strength, vector, @hooks.defs.zoom)\n\t\t\t\t\t\n\t\t\t\t\tacceleration[hook.prop] += force * delta\n\t\t\n\t\t\n\t\t# Add velocities to properties. Doing this at the end in case there are multiple hooks affecting the same velocity\n\t\tfor prop, velocity of @hooks.velocities\n\t\t\n\t\t\tif prop is 'pos'\n\n\t\t\t\t# Add drag, if it exists\n\t\t\t\tif drag.pos\n\t\t\t\t\tvelocity.x += @hooks.defs.getDrag(velocity.x, drag.pos, @hooks.defs.zoom)\n\t\t\t\t\tvelocity.y += @hooks.defs.getDrag(velocity.y, drag.pos, @hooks.defs.zoom)\n\t\t\t\t\t\n\t\t\t\t# Add acceleration to velocity\n\t\t\t\tvelocity.x += acceleration.pos.x\n\t\t\t\tvelocity.y += acceleration.pos.y\n\t\t\t\t\n\t\t\t\t# Add velocity to position\n\t\t\t\t@x += velocity.x * delta\n\t\t\t\t@y += velocity.y * delta\n\t\t\t\n\t\t\telse\n\t\t\t\n\t\t\t\t# Add drag, if it exists\n\t\t\t\tif drag[prop]\n\t\t\t\t\t@hooks.velocities[prop] += @hooks.defs.getDrag(@hooks.velocities[prop], drag[prop], @hooks.defs.zoom)\n\t\t\t\t\n\t\t\t\t# Add acceleration to velocity\n\t\t\t\t@hooks.velocities[prop] += acceleration[prop]\n\t\t\t\t\n\t\t\t\t# Add velocity to property\n\t\t\t\t@[prop] += @hooks.velocities[prop] * delta\n\n\t\t@onHookUpdate?(delta)"]} 346 | -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/cursor-active.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/cursor-active.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/cursor-active@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/cursor-active@2x.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/cursor.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/cursor.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/cursor@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/cursor@2x.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/icon-120.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/icon-120.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/icon-152.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/icon-152.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/icon-180.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/icon-180.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/icon-192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/icon-192.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/images/icon-76.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/images/icon-76.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/manifest.txt: -------------------------------------------------------------------------------- 1 | app.coffee 2 | framer/backups/backup-2016-09-09 14.23.29.coffee 3 | framer/backups/backup-2016-09-09 14.27.29.coffee 4 | framer/coffee-script.js 5 | framer/config.json 6 | framer/framer.generated.js 7 | framer/framer.init.js 8 | framer/framer.js 9 | framer/framer.js.map 10 | framer/framer.modules.js 11 | framer/images/cursor-active.png 12 | framer/images/cursor-active@2x.png 13 | framer/images/cursor.png 14 | framer/images/cursor@2x.png 15 | framer/images/icon-120.png 16 | framer/images/icon-152.png 17 | framer/images/icon-180.png 18 | framer/images/icon-192.png 19 | framer/images/icon-76.png 20 | framer/manifest.txt 21 | framer/metadata.json 22 | framer/preview.png 23 | framer/style.css 24 | framer/version 25 | index.html 26 | modules/Hook.coffee 27 | spring-example-720.gif 28 | spring-example.gif -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/metadata.json: -------------------------------------------------------------------------------- 1 | {"title":"hook-example-spring","date":"2016-09-09T12:30:12Z"} -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/preview.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/framer/preview.png -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/style.css: -------------------------------------------------------------------------------- 1 | * { 2 | margin: 0; 3 | padding: 0; 4 | border: none; 5 | -webkit-user-select: none; 6 | -webkit-tap-highlight-color: rgba(0,0,0,0); 7 | } 8 | 9 | body { 10 | background-color: #fff; 11 | font: 28px/1em "Helvetica"; 12 | color: gray; 13 | overflow: hidden; 14 | } 15 | 16 | a { 17 | color: gray; 18 | } 19 | 20 | body { 21 | cursor: url('images/cursor.png') 32 32, auto; 22 | cursor: -webkit-image-set( 23 | url('images/cursor.png') 1x, 24 | url('images/cursor@2x.png') 2x 25 | ) 32 32, auto; 26 | } 27 | 28 | body:active { 29 | cursor: url('images/cursor-active.png') 32 32, auto; 30 | cursor: -webkit-image-set( 31 | url('images/cursor-active.png') 1x, 32 | url('images/cursor-active@2x.png') 2x 33 | ) 32 32, auto; 34 | } 35 | 36 | .framerAlertBackground { 37 | position: absolute; top:0px; left:0px; right:0px; bottom:0px; 38 | z-index: 1000; 39 | background-color: #fff; 40 | } 41 | 42 | .framerAlert { 43 | font:400 14px/1.4 "Helvetica Neue", Helvetica, Arial, sans-serif; 44 | -webkit-font-smoothing:antialiased; 45 | color:#616367; text-align:center; 46 | position: absolute; top:40%; left:50%; width:260px; margin-left:-130px; 47 | } 48 | .framerAlert strong { font-weight:500; color:#000; margin-bottom:8px; display:block; } 49 | .framerAlert a { color:#28AFFA; } 50 | .framerAlert .btn { 51 | font-weight:500; text-decoration:none; line-height:1; 52 | display:inline-block; padding:6px 12px 7px 12px; 53 | border-radius:3px; margin-top:12px; 54 | background:#28AFFA; color:#fff; 55 | } 56 | 57 | ::-webkit-scrollbar { 58 | display: none; 59 | } -------------------------------------------------------------------------------- /hook-example-spring.framer/framer/version: -------------------------------------------------------------------------------- 1 | 4 -------------------------------------------------------------------------------- /hook-example-spring.framer/images/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/images/.gitkeep -------------------------------------------------------------------------------- /hook-example-spring.framer/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | -------------------------------------------------------------------------------- /hook-example-spring.framer/modules/Hook.coffee: -------------------------------------------------------------------------------- 1 | ### 2 | -------------------------------------------------------------------------------- 3 | Hook module for Framer 4 | -------------------------------------------------------------------------------- 5 | 6 | by: Sigurd Mannsåker 7 | github: https://github.com/sigtm/framer-hook 8 | 9 | ················································································ 10 | 11 | 12 | The Hook module simply expands the Layer prototype, and lets you make any 13 | numeric Layer property follow another property - either its own or another 14 | object's - via a spring or gravity attraction. 15 | 16 | 17 | -------------------------------------------------------------------------------- 18 | Example: Layered animation (eased + spring) 19 | -------------------------------------------------------------------------------- 20 | 21 | myLayer = new Layer 22 | 23 | # Make our own custom property for the x property to follow 24 | myLayer.easedX = 0 25 | 26 | # Hook x to easedX via a spring 27 | myLayer.hook 28 | property: "x" 29 | targetProperty: "easedX" 30 | type: "spring(150, 15)" 31 | 32 | # Animate easedX 33 | myLayer.animate 34 | properties: 35 | easedX: 200 36 | time: 0.15 37 | curve: "cubic-bezier(0.2, 0, 0.4, 1)" 38 | 39 | NOTE: 40 | To attach both the x and y position, use "pos", "midPos" or "maxPos" as the 41 | property/targetProperty. 42 | 43 | 44 | -------------------------------------------------------------------------------- 45 | Example: Hooking property to another layer 46 | -------------------------------------------------------------------------------- 47 | 48 | target = new Layer 49 | hooked = new Layer 50 | 51 | hooked.hook 52 | property: "scale" 53 | to: target 54 | type: "spring(150, 15)" 55 | 56 | The "hooked" layer's scale will now continuously follow the target layer's scale 57 | with a spring animation. 58 | 59 | 60 | -------------------------------------------------------------------------------- 61 | layer.hook(options) 62 | -------------------------------------------------------------------------------- 63 | 64 | Options are passed as a single object, like you would for a new Layer. 65 | The options object takes the following properties: 66 | 67 | 68 | property [String] 69 | ----------------- 70 | The property you'd like to hook onto another object's property 71 | 72 | 73 | type [String] 74 | ------------- 75 | Either "spring(strength, friction)" or "gravity(strength, drag)". Only the last 76 | specified drag value is used for each property, since it is only applied to 77 | each property once (and only if it has a gravity hook applied to it.) 78 | 79 | 80 | to [Object] (Optional) 81 | ---------------------- 82 | The object to attach it to. Defaults to itself. 83 | 84 | 85 | targetProperty [String] (Optional) 86 | ---------------------------------- 87 | Specify the target object's property to follow, if you don't want to follow 88 | the same property that the hook is applied to. 89 | 90 | 91 | modulator [Function] (Optional) 92 | ------------------------------- 93 | The modulator function receives the target property's value, and lets you 94 | modify it before it is fed into the physics calculations. Useful for anything 95 | from standard Utils.modulate() type stuff to snapping and conditional values. 96 | 97 | 98 | zoom [Number] (Optional) 99 | ------------------------ 100 | This factor defines the distance that 1px represents in regards to gravity and 101 | drag calculations. Only one value is stored per layer, so specifying it 102 | overwrites its existing value. Default is 100. 103 | 104 | 105 | -------------------------------------------------------------------------------- 106 | layer.unHook(property, object) 107 | -------------------------------------------------------------------------------- 108 | 109 | This removes all hooks for a given property and target object. Example: 110 | 111 | # Hook it 112 | layer.hook 113 | property: "x" 114 | to: "otherlayer" 115 | targetProperty: "y" 116 | type: "spring(200,20)" 117 | 118 | # Unhook it 119 | layer.unHook "x", otherlayer 120 | 121 | 122 | -------------------------------------------------------------------------------- 123 | layer.onHookUpdate(delta) 124 | -------------------------------------------------------------------------------- 125 | 126 | After a layer is done applying accelerations to its hooked properties, it calls 127 | onHookUpdate() at the end of each frame, if it is defined. This is an easy way 128 | to animate or trigger other stuff, perhaps based on your layer's updated 129 | properties or velocities. 130 | 131 | The delta value from the Framer loop is passed on to onHookUpdate() as well, 132 | which is the time in seconds since the last animation frame. 133 | 134 | Note that if you unhook all your hooks, onHookUpdate() will of course no longer 135 | be called for that layer. 136 | 137 | ### 138 | 139 | 140 | # Since older versions of Safari seem to be missing String.prototype.includes() 141 | 142 | unless String.prototype.includes 143 | String::includes = (search, start) -> 144 | 'use strict' 145 | start = 0 if typeof start is 'number' 146 | 147 | if start + search.length > this.length 148 | return false; 149 | else 150 | return @indexOf(search, start) isnt -1 151 | 152 | # Expand layer 153 | 154 | Layer::hook = (config) -> 155 | 156 | throw new Error 'layer.hook() needs a property, a hook type and either a target object or target property to work' unless config.property and config.type and (config.to or config.targetProperty) 157 | 158 | # Single array for all hooks, as opposed to nested arrays per property, because performance 159 | @hooks ?= 160 | hooks: [] 161 | velocities: {} 162 | defs: 163 | zoom: 100 164 | getDrag: (velocity, drag, zoom) => 165 | velocity /= zoom 166 | # Dividing by 10 is unscientific, but it means a value of 2 equals roughly a 100g ball with 15cm radius in air 167 | drag = -(drag / 10) * velocity * velocity * velocity / Math.abs(velocity) 168 | if _.isNaN(drag) then return 0 else return drag 169 | getGravity: (strength, distance, zoom) => 170 | dist = Math.max(1, distance / zoom) 171 | return strength * zoom / (dist * dist) 172 | 173 | # Update the zoom value if given 174 | @hooks.zoom = config.zoom if config.zoom 175 | 176 | # Parse physics config string 177 | f = Utils.parseFunction config.type 178 | config.type = f.name 179 | config.strength = f.args[0] 180 | config.friction = f.args[1] or 0 181 | 182 | # Default to same targetProperty on same object (hopefully you've set at least one of these to something else) 183 | config.targetProperty ?= config.property 184 | config.to ?= @ 185 | 186 | # All position accelerations are added to a single 'pos' velocity. Store actual properties so we don't have to do it again every frame 187 | 188 | if config.property.toLowerCase().includes 'pos' 189 | config.prop = 'pos' 190 | 191 | if config.property.toLowerCase().includes 'mid' 192 | config.thisX = 'midX' 193 | config.thisY = 'midY' 194 | 195 | else if config.property.toLowerCase().includes 'max' 196 | config.thisX = 'maxX' 197 | config.thisY = 'maxY' 198 | 199 | else 200 | config.thisX = 'x' 201 | config.thisY = 'y' 202 | 203 | if config.targetProperty.toLowerCase().includes 'mid' 204 | config.toX = 'midX' 205 | config.toY = 'midY' 206 | 207 | else if config.targetProperty.toLowerCase().includes 'max' 208 | config.toX = 'maxX' 209 | config.toY = 'maxY' 210 | else 211 | config.toX = 'x' 212 | config.toY = 'y' 213 | 214 | else 215 | config.prop = config.property 216 | 217 | # Save hook to @hooks array 218 | @hooks.hooks.push(config) 219 | 220 | # Create velocity property if necessary 221 | @hooks.velocities[config.prop] ?= if config.prop is 'pos' then { x: 0, y: 0 } else 0 222 | 223 | # Use Framer's animation loop, slightly more robust than requestAnimationFrame directly 224 | # Save the returned AnimationLoop reference to make sure @hookLoop isn't added multiple times per layer 225 | @hooks.emitter ?= Framer.Loop.on('render', @hookLoop, this) 226 | 227 | Layer::unHook = (property, object) -> 228 | 229 | return unless @hooks 230 | 231 | prop = if property.toLowerCase().includes 'pos' then 'pos' else property 232 | 233 | # Remove all matches 234 | @hooks.hooks = @hooks.hooks.filter (hook) -> 235 | hook.to isnt object or hook.property isnt property 236 | 237 | # If there are no hooks left, shut it down 238 | if @hooks.hooks.length is 0 239 | delete @hooks 240 | Framer.Loop.removeListener 'render', @hookLoop 241 | return 242 | 243 | # Still here? Check if there are any remaining hooks affecting same velocity 244 | remaining = @hooks.hooks.filter (hook) -> 245 | prop is hook.prop 246 | 247 | # If not, delete velocity (otherwise it won't be reset if you make new hook for same property) 248 | delete @hooks.velocities[prop] if remaining.length is 0 249 | 250 | Layer::hookLoop = (delta) -> 251 | 252 | if @hooks 253 | 254 | # Multiple hooks can affect the same property. Add accelerations to temporary object so the property's velocity is the same for all calculations within the same animation frame 255 | acceleration = {} 256 | 257 | # Save drag for each property to this object, since only most recently specified value is used for each property 258 | drag = {} 259 | 260 | # Add accelerations 261 | for hook in @hooks.hooks 262 | 263 | if hook.prop is 'pos' 264 | 265 | acceleration.pos ?= { x: 0, y: 0 } 266 | 267 | target = { x: hook.to[hook.toX], y: hook.to[hook.toY] } 268 | 269 | target = hook.modulator(target) if hook.modulator 270 | 271 | vector = 272 | x: target.x - @[hook.thisX] 273 | y: target.y - @[hook.thisY] 274 | 275 | vLength = Math.sqrt((vector.x * vector.x) + (vector.y * vector.y)) 276 | 277 | if hook.type is 'spring' 278 | 279 | damper = 280 | x: -hook.friction * @hooks.velocities.pos.x 281 | y: -hook.friction * @hooks.velocities.pos.y 282 | 283 | vector.x *= hook.strength 284 | vector.y *= hook.strength 285 | 286 | acceleration.pos.x += (vector.x + damper.x) * delta 287 | acceleration.pos.y += (vector.y + damper.y) * delta 288 | 289 | else if hook.type is 'gravity' 290 | 291 | drag.pos = hook.friction 292 | 293 | gravity = @hooks.defs.getGravity(hook.strength, vLength, @hooks.defs.zoom) 294 | 295 | vector.x *= gravity / vLength 296 | vector.y *= gravity / vLength 297 | 298 | acceleration.pos.x += vector.x * delta 299 | acceleration.pos.y += vector.y * delta 300 | 301 | else 302 | 303 | acceleration[hook.prop] ?= 0 304 | 305 | target = hook.to[hook.targetProperty] 306 | 307 | target = hook.modulator(target) if hook.modulator 308 | 309 | vector = target - @[hook.prop] 310 | 311 | if hook.type is 'spring' 312 | 313 | force = vector * hook.strength 314 | damper = -hook.friction * @hooks.velocities[hook.prop] 315 | 316 | acceleration[hook.prop] += (force + damper) * delta 317 | 318 | 319 | else if hook.type is 'gravity' 320 | 321 | drag[hook.prop] = hook.friction 322 | 323 | force = @hooks.defs.getGravity(hook.strength, vector, @hooks.defs.zoom) 324 | 325 | acceleration[hook.prop] += force * delta 326 | 327 | 328 | # Add velocities to properties. Doing this at the end in case there are multiple hooks affecting the same velocity 329 | for prop, velocity of @hooks.velocities 330 | 331 | if prop is 'pos' 332 | 333 | # Add drag, if it exists 334 | if drag.pos 335 | velocity.x += @hooks.defs.getDrag(velocity.x, drag.pos, @hooks.defs.zoom) 336 | velocity.y += @hooks.defs.getDrag(velocity.y, drag.pos, @hooks.defs.zoom) 337 | 338 | # Add acceleration to velocity 339 | velocity.x += acceleration.pos.x 340 | velocity.y += acceleration.pos.y 341 | 342 | # Add velocity to position 343 | @x += velocity.x * delta 344 | @y += velocity.y * delta 345 | 346 | else 347 | 348 | # Add drag, if it exists 349 | if drag[prop] 350 | @hooks.velocities[prop] += @hooks.defs.getDrag(@hooks.velocities[prop], drag[prop], @hooks.defs.zoom) 351 | 352 | # Add acceleration to velocity 353 | @hooks.velocities[prop] += acceleration[prop] 354 | 355 | # Add velocity to property 356 | @[prop] += @hooks.velocities[prop] * delta 357 | 358 | @onHookUpdate?(delta) -------------------------------------------------------------------------------- /hook-example-spring.framer/spring-example-720.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/spring-example-720.gif -------------------------------------------------------------------------------- /hook-example-spring.framer/spring-example.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sigtm/framer-hook/ca54ec1268092b5af195518aeb0b3dbd90bd533f/hook-example-spring.framer/spring-example.gif --------------------------------------------------------------------------------