Augmented Reality
Requirements
- A mobile phone or tablet (augmented reality headsets are not supported yet)
- ARCore (if on Android)
- browser with WebXR API and immersive-ar session type support
Warning
At the moment of writing, only Chromium-based browsers on Android support immersive-ar session type.
Other information
- It is generally recommended to use this extension with fps set to 0, which in turbowarp means running project at the screen refresh rate.
- While exiting AR mode, there is a small chance TurboWarp will lose WebGL context. This problem has also been observed on other websites with AR and likely can't be fixed. If that happens, you can save the project, refresh the tab, load the project and continue.
- It isn't possible to be in AR mode and have video sensing enabled at the same time. Entering AR disables video sensing and makes it not toggleable until AR session ends.
Blocks
If AR is not supported and the error message haven't been shown yet, shows a popup with an error message.
If AR is supported and project is currently not in AR, attempts to enter AR mode. While doing so, it will pause the script it is in, in the same way ask block does. It will first try to enter AR mode directly. That may fail because entering AR can only be triggred by user interaction. If it fails, it will make it so after user clicks/taps the project will attempt to enter AR. If that fails as well or either of those 2 attempts succeed it will resume the script execution. After it resumes, project may or may not be in AR mode. Projects should handle both cases.
The origin of coordinate system is placed at or close to position of the device at the time this block was called.
If the project is in AR mode, exits it.
Tells if the project is currently in AR mode.
Tells if AR is supported on this device.
Tells if AR engine knows what the current camera position and orientation is.
After entering AR mode, is is not immediately available as the map of the environment needs to be built first. After enough information about environment has been gathered and processed, it becomes available. It can temporarily become unavailable due to lack of detailed features in the view of camera that are used for motion tracking, fast motion causing camera image to become too blurry or camera getting covered.
Tells if AR engine knows where the point of ray intersection is.
Can become unavailable for the same reasons as [is [pose] available?]
Tells stage width in scratch units. Default is 480.
This value may change when entering AR.
Tells stage height in scratch units. Default is 360.
This value will not change when entering AR.
view matrix - is a matrix that can be used to transform points from the view space (with origin is at the camera) to the world space (relatively to the world origin). It includes rotation and translation of the camera.
Also:
view[13] = position[x] view[14] = position[y] view[15] = position[z]
inverse view matrix = view matrix-1
Describes the opposite transformation to the view matrix: From the world space (relatively to the world origin) to the view space (with origin is at the camera).
For perspective projection in scratch you are likely used to doing something like this:
screenX = x * dist / z screenY = y * dist / z
With this extension it's more complicated: projection is done by first doing a 4x4 projection matrix multiplication by 4D vector x,y,z,1, with result being X,Y,Z,W, then performing division of X,Y,Z by W to get screen coordinates in range from -1 to 1, and then multiplying them by half of stage width and height to get scratch coordinates.
The matrix is a perspective projection matrix with the assumption camera faces negative z and that screen coordinates range from -1 to 1.
It's calculated as:
                                      /\
                                     /||\
                                      ||     OUT
   IN                                 ||  [X,Y,Z,W]
[x,y,z,1]                             ||
      \     [ 2 * dist / width,                 0,                             0,  0 ]  [  1, 2, 3, 4 ]
_______\    [                0, 2 * dist / height,                             0,  0 ]  [  5, 6, 7, 8 ]
^^^^^^^/    [                0,                 0,   (near + far) / (near - far), -1 ]  [  9,10,11,12 ]
      /     [                0,                 0, near * far / (near - far) * 2,  0 ]  [ 13,14,15,16 ]
near = depth of near plane
far = depth of far plane
Point is only visible if -W < X,Y,Z < W.
screenX = X / W screenY = Y / W screenZ = Z / W (for depth buffer)
If the first codition was true, then -1 < screenX,screenY,screenZ < 1 is also always true. Note that the opposite is not always true (think of what happens when W is negative).
To get a better understanding of this topic, you may read:
- https://webglfundamentals.org/webgl/lessons/webgl-3d-perspective.html
- https://stackoverflow.com/questions/41085117/why-does-gl-divide-gl-position-by-w-for-you-rather-than-letting-you-do-it-your
Projection matrix contains a lot of 0s, and can be simplified. As the result of simplifications it is possible to make it look closer to how it is usually done on scratch:
screenX = x * (2 * dist / width) / -z * (width / 2) screenY = y * (2 * dist / height) / -z * (height / 2) screenX = x * (2 * dist / width) * -1 * (width / 2) / z screenY = y * (2 * dist / height) * -1 * (height / 2) / z negative_dist = (2 * dist / width) * (width / -2) = (2 * dist / height) * (height / -2) (2 * dist / width) = projection[1] (2 * dist / height) = projection[6] negative_dist = projection[1] * (width / -2) = projection[6] * (height / -2) screenX = x * negative_dist / z screenY = y * negative_dist / z
combined matrix = projection matrix * inverse view matrix
Transforms positions from world space to view space, then to projected.
Camera position relatively to the world origin.
Camera orientation represented as quaternion.
A ray that originates from camera in the direction the camera is facing (center of the screen) intersects the first detected real world surface. This block returns the coordinates of that intersection point.
Moves the coordinate system by specified amount.
It can also be understood as switching to a new coordinate system with origin at a given location in the current coordinate system.
Usage example:
After starting the project, it may be a good idea to give user a way to pick location for AR content and then perform this before starting the main AR game/animation/etc.
Internally it also does quaternion normalization, so you don't have to worry about doing it yourself.
Usage example:
This script can be used to rotate XZ around Y-axis:
accepts values from 0.1 to 1. changes the resolution at which the project is rendered
- 1- is native screen resolution
- 0.5- half the screen resolution
- 0.1- one tenth of screen resolution
Reducing resolution can improve performance and reduce memory usage.