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

Support for 3D coordinate on Image #7172

Draft
wants to merge 26 commits into
base: main
Choose a base branch
from

Conversation

Forchapeatl
Copy link
Contributor

@Forchapeatl Forchapeatl commented Aug 14, 2024

Resolves #5861

Changes:

Screenshots of the change:

image

image

PR Checklist

Example code

let img;

function preload() {
  img = loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/69/June_odd-eyed-cat_cropped.jpg/712px-June_odd-eyed-cat_cropped.jpg'); // Load the image
}

function setup() {
  createCanvas(800, 600, WEBGL); // Create a WebGL canvas
}

function draw() {
  background(200);

  // Set up the camera
  let camX = map(mouseX, 0, width, -400, 400);
  let camY = map(mouseY, 0, height, -400, 400);
  camera(camX, camY, 500, 0, 0, 0, 0, 1, 0); // Adjust camera position based on mouse

  // Use the image function to place images in 3D space
  image(img, -500, -800,0, 675,     400, -100, -100,  200, 200);
  image(img, -800, -10,50, 675,     400, 0, 0,  200, 200);
  image(img, -800, -10,100, 675,     400, -50, 0,  200, 200);

}

@Forchapeatl
Copy link
Contributor Author

@davepagurek , please have a look at my PR. Is it in the right direction ?

Copy link
Contributor

@davepagurek davepagurek left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for taking this on! I think we'll have to do some updates in order to make sure 2D image() still works. Let me know what you think!

@@ -1097,6 +1097,7 @@ p5.prototype.image = function(
img,
dx,
dy,
dz,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think placing this argument here will break calls to image() that don't use z but do use the arguments that follow it.

I think we might need to follow the approach of some other 2D/3D functions like dist here, where we reinterpret the arguments array based on some criteria:

if (args.length === 4) {
//2D
return Math.hypot(args[2] - args[0], args[3] - args[1]);
} else if (args.length === 6) {
//3D
return Math.hypot(args[3] - args[0], args[4] - args[1], args[5] - args[2]);
}

In this case, the criteria will probably depend on the length of the arguments.

Since this is somewhat complex, I think it would also be a good idea to add some unit tests to make sure that this continues to work for the 2D case as well as the new 3D case.

Copy link
Contributor Author

@Forchapeatl Forchapeatl Aug 16, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for the review. I went with the length approach initially .

Odd number of parameters → Use 2D mode.
Even number of parameters → Check if the 10th parameter is a number.
If it's not a number, defer to 2D mode.
If it's a number, use 3D mode.

but starting from the third parameter single arguments become optional image(img, dx, dy, dWidth, dHeight, sx, sy, [sWidth], [sHeight], [fit], [xAlign], [yAlign]) hence my even/odd length logic fails as soon as the function starts receiving optional arguments.

Screenshot from 2024-08-16 19-26-04

The best solution I got at the moment was, to shift the values of the input argument to the right in case of 2D_rendering.

  if(this._renderer instanceof p5.RendererGL == false){
    // Store the value of 'yAlign' temporarily
    let   temp = yAlign;
    // From the 3rd arguement shift the assingment one position to the right   
    yAlign = xAlign;
    xAlign = fit;
    fit = sHeight;
    sHeight = sWidth;
    sWidth = sy;
    sy = sx;
    sx = dHeight;
    dHeight = dWidth;
    dWidth = dz;    
  }

The results have been successful in every test .

Screenshot from 2024-08-16 19-31-01

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this also work for 3D mode if the z coordinate is omitted?

Also a possible other approach for the even/odd check: you might need to check if the number of arguments up to the fit parameter (which should be a string instead of a number) is even or odd. That might mean something like:

let endIndex = args.findIndex(arg => arg === constants.COVER || arg === constants.CONTAIN)
if (endIndex === -1) {
  endIndex = args.length
}
const argsUpToFit = args.slice(0, endIndex)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this also work for 3D mode if the z coordinate is omitted?

yes.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you are in WebGL mode and call image without z but with some of the options after z, does that still work? e.g. this should draw the image fullscreen on the canvas:

imageMode(CENTER)
image(yourImg, 0, 0, width, height, 0, 0, yourImg.width, yourImg.height, COVER)

Just based on the code so far, it looks like in WebGL mode, it never shifts all the arguments over the way it does when this._renderer instanceof p5.RendererGL === false.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

image

It seems like the ImgeMode(CENTER) is a trap on WebGL mode. Is there any way to workaround this ? we could add a new imageModeGL that supports WebGL coordinates ?

@Forchapeatl Forchapeatl marked this pull request as ready for review August 20, 2024 14:38
@Forchapeatl
Copy link
Contributor Author

Forchapeatl commented Aug 20, 2024

The 2D image() nolonger breaks, but my 3D positioning does not seem to accurate. When compared to the 2D preview, the image on the left ( 3d mode) x=100 , y = 100 , z = 0 shows an off set in the positioning.

sample code

let img;

function preload() {
  // Load the image
  img = loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/69/June_odd-eyed-cat_cropped.jpg/712px-June_odd-eyed-cat_cropped.jpg');
}

function setup() {
  createCanvas(800, 800 , WEBGL); // Create a 3D canvas
  background(200);
}

function draw() {
  background(200);
  // Second image at another position
  image(img, 100, 100,0); // x, y, z coordinates
}

image

Every thing seems fine by my knowledge. Please for a hint as to why this problem occurs.
https://github.com/Forchapeatl/p5.js/blob/7942b7328d65b0094f9bb372d613375c21123a7d/src/webgl/p5.RendererGL.js#L2395#L2411

@davepagurek
Copy link
Contributor

Just to confirm, do you normalize the starting coordinate between WebGL and 2D modes, e.g. by doing translate(-width/2, -height/2) in WebGL mode so that they start at the same coordinate?

@Forchapeatl Forchapeatl marked this pull request as draft August 22, 2024 05:53
@Forchapeatl
Copy link
Contributor Author

image

Thank you Dave . For 3D mode , It is better to draw texture with vertex rather than plane because when we try to translate the plane. the size of the webgl canvas and plane are considered hence an accurate view of the positions will depend on the perspective.

let img;

function preload() {
  img = loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/69/June_odd-eyed-cat_cropped.jpg/712px-June_odd-eyed-cat_cropped.jpg');
}

function setup() {
  createCanvas(800, 800, WEBGL);  // Create a 3D canvas
  background(200);
}

function draw() {
  background(200);
  image(img, 100, 100, 0);
}

@Forchapeatl
Copy link
Contributor Author

Forchapeatl commented Aug 22, 2024

But when I introduce dz parameter I seem to be failing some test. How can I introuce the dz param here

image

https://github.com/processing/p5.js/blob/ca19e53b4b7dfddb8a6976fc3e489bad2fae5c6a/src/image/loading_displaying.js#L1183#L1184

@Forchapeatl
Copy link
Contributor Author

Forchapeatl commented Aug 23, 2024

This is the cause of the error 'JSDOC'
image

Please for a hint.

@davepagurek
Copy link
Contributor

The parameter checking is done based on the documentated parameter types, and by the looks of it, currently we've added dz in a way that is non optional. I think the way to fix that would be to provide second versions of the parameter overloads: the original without dz, and a new copy with dz.

@@ -2391,6 +2391,44 @@ p5.RendererGL = class RendererGL extends p5.Renderer {

return triangleVerts;
}

image3D(img,sx,sy,sz,sWidth,sHeight,dx,dy,dWidth,dHeight) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This shares a lot of code with p5.RendererGL.image, maybe we could make that function support a z coordinate rather than duplicating the code for this case?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The code uses 3D primitives and changes to p5.RendererGL has been removed

);
//if it is not graphics nor framebuffer but WEGL instance
//default to use 3D rendering
if (this._renderer instanceof p5.RendererGL && !isP5G && !isP5Fbo) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I feel like maybe this check should be a check for whether the dz parameter exists rather than the type of the image? since you could plausibly be drawing a graphic or a framebuffer both in 3D or without specifying a z

@Forchapeatl
Copy link
Contributor Author

@sproutleaf , sorry to bother you . Could you give me some hints or direction, I realize that when I add one more parameter (dz) to a function , I am caught with a friendly error message and failed tests. Please could you help me with a few hints on how to address this.

image

@@ -1110,7 +1213,20 @@ p5.prototype.image = function(
// set defaults per spec: https://goo.gl/3ykfOq

p5._validateParameters('image', arguments);

// If dz is not specified.
if (arguments.length % 2 === 1 || typeof arguments[9] === 'string'){
Copy link
Contributor

@perminder-17 perminder-17 Sep 18, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not completely sure if this is causing the test to fail, but I believe the code enters the if block in both cases—whether dz is specified or not.

For example, when you check arguments.length, if z is provided it comes out to be an odd number, and arguments[9] will be a number.

However, if the dz variable isn't specified, arguments.length will be even, and in this case, arguments[9] will be a string.

In both cases I am talking about this example:
image(yourImg, 0, 0, width, height, 0, 0, yourImg.width, yourImg.height, COVER) when z is not specified.

If you do something like:

if(arg.length%2 !== 1 || typeof arg[9] ==='string){...}

is that works for you?

Copy link
Contributor

@perminder-17 perminder-17 Sep 18, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also, I’m concerned that we might need a different approach for other cases. For example, if we have a function call like image(img, 100, 100, 0);, the number of arguments is four, we have arg.length % 2=== 1 for which z is not specified.

We could probably have a general case for all the cases which could work for image(no. of arguments).

let me know what you think?

@perminder-17
Copy link
Contributor

Hi @Forchapeatl ,

I apologize for the delayed response. The maintainers have been very busy with the version update to 2.0. I can help you figure out what's going on.

It's looking great so far! Do you mind sharing the sketch where you get friendly errors?

thanks for your work on this:)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet