├── 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 | 
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 |
"
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
--------------------------------------------------------------------------------