-
Notifications
You must be signed in to change notification settings - Fork 2
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fixing Stage Transformation #5
Open
danzen
wants to merge
3
commits into
ReCreateJS:master
Choose a base branch
from
danzen:master
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Changes from 2 commits
Commits
Show all changes
3 commits
Select commit
Hold shift + click to select a range
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,122 @@ | ||
- Start Date: 2020-03-13 | ||
- RFC PR: https://github.com/ReCreateJS/rfcs/pull/5 | ||
- Relevant Project: EaselJS | ||
|
||
# Stage Transforms | ||
|
||
## Summary | ||
|
||
Transforming the CreateJS stage, for example scaling the stage, causes globalToLocal, localToGlobal, and mouse position to not behave as expected. This has led to confusion in the community and extra code to remedy. | ||
|
||
## Motivation | ||
|
||
Traditional practice has been to not scale the stage, but when CreateJS is exported from Adobe Animate, the stage is being scaled to accommodate the devicePixelRatio. Scaling in this manner definitely gives better visual result when the canvas is further scaled with CSS style. | ||
|
||
## Detailed design | ||
|
||
Here is a Web page that shows what happens when the CreateJS stage is scaled. | ||
|
||
https://zimjs.com/createjs/retina.html | ||
|
||
STAGE SCALING:<br> | ||
The canvas is created at a desired width and height. The stage is made at that width and height and then scaled by the window.devicePixelRatio. The canvas width and height properties are then set to match the new stage size. The canvas is finally scaled with style back to the original width and height. | ||
|
||
This technique is used by Adobe when publishing from Animate to CreateJS. I am not sure if it was GSkinner who recommended it but I suspect it was added by Adobe without realizing what it would do to CreateJS. | ||
|
||
Here is an example of the visual difference in ZIM which is built on CreateJS. ZIM introduced this type of scaling in 10.3.0 and termed it Retina! You can see that there is quite a remarkable and magical difference. It is well worth doing. | ||
|
||
https://zimjs.com/retina.html<br> | ||
https://zimjs.com/retina_off.html | ||
|
||
WHAT BREAKS:<br> | ||
In the https://zimjs.com/createjs/retina.html example, we show two circles following the mouse using the event object stageX and stageY properties. The blue circle uses the stageX and stageY properties directly and does not follow the mouse properly. The red circle has these properties divided by the stage scaleX and scaleY and does follow the mouse properly. | ||
|
||
There is also a Container called holder placed at 200, 200 with a Shape rectangle(0,0,100,50) inside. We ask for the holder.localToGlobal(100, 50) and place a black circle on the stage at that location. The circle is not in the right place. We would expect it to be visually placed at the right bottom corner of the rectangle. We fixed this with a violet circle by dividing the resulting localToGlobal point by the stage scaleX and scaleY. | ||
|
||
Rotating the stage introduces similar problems. Skewing the stage probably would as well. | ||
|
||
EXTERNAL SOLUTION:<br> | ||
Aside from dividing by stage scaleX and scaleY which works for scale, the resulting points can be transformed to accommodate the stage transformations. Lanny posted this solution for the mouse position: | ||
|
||
``` | ||
function mousedown(e){ | ||
var global = stage.localToGlobal(e.target.x, e.target.y); | ||
e.target.offset = {x:global.x-e.stageX, y:global.y-e.stageY}; | ||
} | ||
function mousemove(e){ | ||
var local = stage.globalToLocal(e.stageX+e.target.offset.x, e.stageY+e.target.offset.y); | ||
e.target.x = local.x; | ||
e.target.y = local.y; | ||
} | ||
``` | ||
|
||
INTERNAL SOLUTION:<br> | ||
It would be better to bring this type of solution into CreateJS so developers get what they expect. Here is a test page using modified createjs code. The fixes will be added to a pull request after discussed here. | ||
|
||
https://zimjs.com/createjs/retina2.html | ||
|
||
The blue circle shows a correct cursor rollover and can be dragged to follow the mouse using an unmodified event object stageX and stageY. The red circle is following rawX and rawY. The circles also work with stagemousemove and when the stage is rotated (commented out in the code). | ||
|
||
The rectangles show localToGlobal, globalToLocal and localToLocal all working as expected in a container. The container was scaled and rotated and cursor checked as well - all working. Skew and multiple containers were not tested. | ||
|
||
CREATEJS CHANGES:<br> | ||
Example changes can be found in here: https://zimjs.com/cdn/1.3.0/createjs_doc.js | ||
|
||
The general technique to fix these internally was provided in an existing comment in CreateJS on the Stage class on line 8707. This is an invert() on a transform: | ||
|
||
``` | ||
// TODO: account for stage transformations? | ||
// this._mtx = this.getConcatenatedMatrix(this._mtx).invert(); | ||
// var pt = this._mtx.transformPoint(o.x, o.y); | ||
var evt = new createjs.MouseEvent(type, bubbles, false, pt.x, pt.y, etc. | ||
target.dispatchEvent(evt); | ||
``` | ||
|
||
Instead of the change being made here, this technique was used in the following places in the modified CreateJS file with some being an invert() and some being a regular transform() adjustment: | ||
|
||
``` | ||
// line 6697 - localToGlobal (invert transform) | ||
// line 6726 - globalToLocal (transform) | ||
// line 6866 - hitTest (transform) - not being used see _getObjectsUnderPoint | ||
// line 7761 - _getObjectsUnderPoint (transform) | ||
// line 7784 - _getObjectsUnderPoint - small adjust | ||
// line 8518 - _updatePointerPosition (invert transform) | ||
``` | ||
|
||
A property on the createjs namespace was made on line 40: | ||
|
||
``` | ||
createjs.stageTransformable = true; | ||
``` | ||
|
||
Setting this to false will revert back to earlier CreateJS functionality. This property can also be read by code using createjs to adjust accordingly. For example, in ZIM, we have already adjusted for stage scale. We will now read this property and if it is true, we will not adjust for stage scale otherwise there would be a double adjustment. | ||
|
||
## How we teach this | ||
|
||
The stageTransformable property would go into the documentation along with an explanation. Various forum questions or reports could be answered accordingly. Otherwise, people would expect CreateJS to work in this manner so there is no need for teaching in general. | ||
|
||
## Drawbacks | ||
|
||
This will break existing code but can be turned off to not break existing code. To update to a new version of CreateJS but keep the existing code working, the stageTransformable could be set to false: | ||
|
||
``` | ||
createjs.stageTransformable = false; | ||
``` | ||
|
||
There could be issues with Adobe Animate export to CreateJS code for internal MovieClip creation, etc. | ||
|
||
|
||
## Alternatives | ||
|
||
The proposed solution may not be the best solution. It may be that someone who really knows what is going on with matrix transformations could reprogram the Matrix to automatically start at the stage transformation. If this is possible, perhaps only a change to the Matrix or maybe getConcatonatedMatrix would solve the issues. | ||
|
||
## Unresolved questions | ||
|
||
1. A Matrix expert should be consulted to make sure there is not a better solution - see Alternatives. | ||
|
||
2. Testing on Adobe Animate output should be conducted and especially with MovieClips. | ||
|
||
3. Skew and multiple nested and transformed containers should be tested. | ||
|
||
4. The name of the createjs.stageTransformable property may be slightly misleading as the stage can still be transformed even if it is set to false. What it means is that stage transformation effects will be accounted for. Possibly we could prevent the stage from being transformed if stageTransformable is not set to true. | ||
|
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
To be clear about this, would it be correct to say that it will only break existing code that fixes the issue in a different way? If people are unaware of the issue or don't scale the stage, then they won't be affected at all.