Skip to content
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

Confusion between .elt and .canvas #3156

Open
Zalastax opened this issue Aug 17, 2018 · 3 comments
Open

Confusion between .elt and .canvas #3156

Zalastax opened this issue Aug 17, 2018 · 3 comments

Comments

@Zalastax
Copy link
Member

@Spongman highlights in #3155 that there is a confusion between the properties .elt and .canvas causing us to write defensive code like so:

cnv = img.canvas || img.elt;

What is the purpose of having both .elt and .canvas? How can it be that the canvas should sometimes come from .elt and sometimes from .canvas?

Searching for .canvas = I get these results:

this.canvas = document.createElement('canvas');

this.canvas = document.createElement('canvas');

this.canvas = elt;

this.canvas = document.createElement('canvas');

this._pInst.canvas = c;

Searching for .elt = I get these results:
this.elt = elt;

elt = this.elt = arg.elt;

elt = this.elt = existing_radios.elt;

and a comment about checking if the canvas is associated with the p5 instance

p5.js/lib/addons/p5.dom.js

Lines 1938 to 1939 in 5a46133

// main canvas associated with p5 instance
if (this._pInst._curElement.elt === this.elt) {

Some of the places that assign canvas also assign elt afterwards, some do not.
In #3155 @Spongman solves the problem via if (!this.canvas) this.loadPixels();, is that a general approach we should apply elsewhere?
Do we have a place where we can add (or already have?) YUIDocs for .elt and .canvas?

@Spongman
Copy link
Contributor

Spongman commented Aug 17, 2018

the solution to the #3155 issue is unrelated to this. p5.MediaElement does actually require both .elt and .canvas properties - the .elt is the video element, and the .canvas is the bitmap cache that supports get & set.

the issue here is that p5.Element defines a .elt property, p5.Graphics, p5.Renderer derive from p5.Element (so inherit the .elt property), but they also define their own .canvas property and much of the code assumes the existence of .canvas, however sometimes those methods are used to act on p5.Image objects that (being p5.Elements) only define .elt.

IMO, (except for p5.MediaElement) those canvas properties should be removed and code referencing them should instead refer to .elt.

ok, it's a little more complicated than that. MediaElement is an Element, but MediaElement pretends to be a Renderer when doing its pixels stuff which means there's a conflict between canvas and elt in MediaElement - they're not the same thing. so doing the above would break MediaElement, again.

@Zalastax
Copy link
Member Author

Thanks for the clarification!

Is this correct?
We have the class p5.Element which is extended and an interface Renderable

interface Renderable {
  canvas: HTMLCanvasElement
}

Some functions accept parameters of type p5.Element | Renderable.
A parameter that is both p5.Element and Renderable will be seen only as a Renderable (cnv = img.canvas || img.elt;).

What is the type for .elt? Any HMLElement but sometimes assumed to be HTMLCanvasElement?
Are there any actions we can and should take?

@Spongman
Copy link
Contributor

Spongman commented Aug 17, 2018

it does look a bit like that, yes. although instead of those functions accepting parameters of different types, they're invoked with this being objects of different types. just a syntactic difference.

i'm not sure there's anything short of major architectural reworking that can be done to fix this right now.

but... my pie-in-the-sky idea: instead of p5.Graphics 'borrowing' methods from p5, and p5.Image 'borrowing' methods from p5.Renderer, maybe things could be turned around a little to end up with an actual inheritance chain that looks something like this:

  • p5 -> p5.Graphics -> p5.RenderableElement -> p5.Element
  • p5.Image -> p5.Graphics
  • p5.MediaElement -> p5.RenderableElement
  • perhaps p5.Renderer -> p5.RenderableElement ? not sure about this one

the API would mostly stay the same except for the following:

  • most 'graphics' methods (line(), stroke(), etc...) would move into p5.Graphics (and be inherited by p5) the reference docs generator would need to updated to support this.
  • p5.Graphics would 'lose' methods like min(), millis(), trim() which it really has no business having today.
  • p5.Image would 'gain' all the p5.Graphics functionality. createImage() would just essentially be an alias for createGraphics(P2D) (but with its own p5.Image prototype to maintain backwards compat)
  • p5.RenderableElement would implement the 2d 'pixels' functions that are the source of this issue, and abstract away fetching the drawingContext (if necessary).
  • the this._XXX = (function(){ ... }).bind(this); trickery in p5 would no longer be necessary for hiding stuff from Graphics.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
Status: Proposal
Development

No branches or pull requests

4 participants