Skip to content

Commit

Permalink
[VisionGlass] Render Wolvic in glasses and control UI on phone screen
Browse files Browse the repository at this point in the history
This is a large change because it's difficult to split it in smaller
chunks as we're revamping the way Wolvic works in the VisionGlass.
In general this is not a good idea but provided that we're in early
stages of VisionGlass port this should be fine as there should not
be regressions as the previous funtionality was quite limited.

These are the main changes:
   * Wolvic is now rendered just in the external screen (the glasses)
     using Android's Presentation API.
   * The phone screen shows a different UI which consists on a quite
     minimalistic and simple UI that allows users to performs clicks
     on a touchpad and also a "back" button.

Apart from that there are some other changes:
   * Fix a crash with early setControllerOrientation call
   * Enable WebXR by properly setting EyeResolution
   * Enable clicks on WebXR experiences
   * Use the actual FOV of the device
   * Improved the "wait for USB permissions" code
  • Loading branch information
svillar committed Jan 8, 2024
1 parent b254d1f commit ffc8d20
Show file tree
Hide file tree
Showing 7 changed files with 335 additions and 175 deletions.
35 changes: 20 additions & 15 deletions app/src/main/res/layout/visionglass_layout.xml
Original file line number Diff line number Diff line change
@@ -1,26 +1,31 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">

<android.opengl.GLSurfaceView
android:id="@+id/gl_view"
<View
android:id="@+id/touchpad"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
android:background="#333333" />

<TextView
android:id="@+id/frame_rate_text"
<Button
android:id="@+id/back_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="24dp"
android:background="#000000"
app:layout_constraintStart_toStartOf="@+id/gl_view"
app:layout_constraintTop_toTopOf="@+id/gl_view" />
android:text="@string/back_button"
android:drawableStart="@drawable/ic_icon_back"
android:layout_alignParentBottom="true"
android:layout_alignParentStart="true" />

<Button
android:id="@+id/home_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/home_tooltip"
android:drawableStart="@drawable/ic_icon_home"
android:layout_alignParentBottom="true"
android:layout_alignParentEnd="true"
android:visibility="gone"/>

</androidx.constraintlayout.widget.ConstraintLayout>
</RelativeLayout>
16 changes: 16 additions & 0 deletions app/src/main/res/layout/visionglass_presentation_layout.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent">

<android.opengl.GLSurfaceView
android:id="@+id/gl_presentation_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>
3 changes: 1 addition & 2 deletions app/src/visionglass/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,7 @@

<application>
<activity
android:name=".VRBrowserActivity"
android:screenOrientation="landscape">
android:name=".VRBrowserActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
Expand Down
114 changes: 45 additions & 69 deletions app/src/visionglass/cpp/DeviceDelegateVisionGlass.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,8 @@ jmethodID sSetRenderMode;

namespace crow {

static const float kHorizontalFOV = 36.0f;
static const float kVerticalFOV = 21.0f;
static const int32_t kControllerIndex = 0;
static const vrb::Vector& GetHomePosition() {
static vrb::Vector homePosition(0.0f, 1.55f, 3.0f);
Expand All @@ -43,12 +45,13 @@ static const vrb::Vector& GetHomePosition() {
struct DeviceDelegateVisionGlass::State {
vrb::RenderContextWeak context;
device::RenderMode renderMode;
ImmersiveDisplayPtr display;
ImmersiveDisplayPtr immersiveDisplay;
ControllerDelegatePtr controller;
vrb::CameraEyePtr cameras[2];
vrb::Color clearColor;
vrb::Matrix headingMatrix;
vrb::Vector position;
vrb::Quaternion controllerOrientation;
bool clicked;
GLsizei glWidth, glHeight;
float near, far;
Expand Down Expand Up @@ -78,21 +81,22 @@ struct DeviceDelegateVisionGlass::State {
}

void UpdateDisplay() {
if (!display)
if (!immersiveDisplay)
return;

vrb::Matrix fov = vrb::Matrix::PerspectiveMatrixWithResolutionDegrees(glWidth, glHeight,
60.0f, -1.0f,
kHorizontalFOV, kVerticalFOV,
near, far);
float left(0.0f), right(0.0f), top(0.0f), bottom(0.0f), n2(0.0f), f2(0.0f);
fov.DecomposePerspectiveDegrees(left, right, top, bottom, n2, f2);

cameras[0]->SetPerspective(fov);
cameras[1]->SetPerspective(fov);
display->SetFieldOfView(device::Eye::Left, left, right, top, bottom);
display->SetFieldOfView(device::Eye::Right, left, right, top, bottom);
immersiveDisplay->SetEyeResolution((int32_t)(glWidth / 2), glHeight);
immersiveDisplay->SetFieldOfView(device::Eye::Left, left, right, top, bottom);
immersiveDisplay->SetFieldOfView(device::Eye::Right, left, right, top, bottom);

display->SetCapabilityFlags(device::Position | device::Orientation | device::Present | device::InlineSession | device::ImmersiveVRSession);
immersiveDisplay->SetCapabilityFlags(device::Orientation | device::Present | device::InlineSession | device::ImmersiveVRSession);
}
};

Expand Down Expand Up @@ -134,11 +138,11 @@ DeviceDelegateVisionGlass::GetRenderMode() {

void
DeviceDelegateVisionGlass::RegisterImmersiveDisplay(ImmersiveDisplayPtr aDisplay) {
m.display = aDisplay;
if (m.display) {
m.display->SetDeviceName("Vision Glass");
m.immersiveDisplay = aDisplay;
if (m.immersiveDisplay) {
m.immersiveDisplay->SetDeviceName("Vision Glass");
m.UpdateDisplay();
m.display->CompleteEnumeration();
m.immersiveDisplay->CompleteEnumeration();
}
}

Expand Down Expand Up @@ -187,13 +191,16 @@ DeviceDelegateVisionGlass::SetClipPlanes(const float aNear, const float aFar) {
void
DeviceDelegateVisionGlass::SetControllerDelegate(ControllerDelegatePtr& aController) {
m.controller = aController;
m.controller->CreateController(kControllerIndex, -1, "Oculus Touch (Right)"); // "Wolvic Virtual Controller");
m.controller->CreateController(kControllerIndex, kControllerIndex, "Vision Glass Controller");
m.controller->SetEnabled(kControllerIndex, true);
m.controller->SetCapabilityFlags(kControllerIndex, device::Orientation | device::Position);
m.controller->SetButtonCount(kControllerIndex, 5);
m.controller->SetCapabilityFlags(kControllerIndex, device::Orientation);
m.controller->SetTargetRayMode(kControllerIndex, device::TargetRayMode::TrackedPointer);
static const float data[2] = {0.0f, 0.0f};
m.controller->SetAxes(kControllerIndex, data, 2);
m.controller->SetControllerType(kControllerIndex, device::VisionGlass);
m.controller->SetMode(kControllerIndex, ControllerMode::Device);
m.controller->SetAimEnabled(kControllerIndex, true);
m.controller->SetCapabilityFlags(kControllerIndex, device::Orientation);

m.controller->SetButtonCount(kControllerIndex, 5);
}

void
Expand Down Expand Up @@ -223,22 +230,28 @@ DeviceDelegateVisionGlass::StartFrame(const FramePrediction aPrediction) {
VRB_GL_CHECK(glEnable(GL_BLEND));
VRB_GL_CHECK(glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT));
mShouldRender = true;
if (m.controller) {
vrb::RenderContextPtr context = m.context.lock();
if (context) {
float level = 100.0 - std::fmod(context->GetTimestamp(), 100.0);
m.controller->SetBatteryLevel(kControllerIndex, (int32_t)level);
}
}

const float IPD = 0;
m.cameras[0]->SetEyeTransform(vrb::Matrix::Translation(vrb::Vector(-IPD * 0.5f, 0.f, 0.f)));
m.cameras[1]->SetEyeTransform(vrb::Matrix::Translation(vrb::Vector(IPD * 0.5f, 0.f, 0.f)));
const float IPD = 0.064f;
auto headTransform = m.headingMatrix.Translate(m.position);
m.cameras[0]->SetHeadTransform(headTransform);
m.cameras[1]->SetHeadTransform(headTransform);
m.display->SetEyeOffset(device::Eye::Left, -IPD * 0.5f, 0.f, 0.f);
m.display->SetEyeOffset(device::Eye::Right, IPD * 0.5f, 0.f, 0.f);
m.cameras[0]->SetEyeTransform(vrb::Matrix::Translation(vrb::Vector(-IPD * 0.5f, 0.f, 0.f)));
m.cameras[1]->SetEyeTransform(vrb::Matrix::Translation(vrb::Vector(IPD * 0.5f, 0.f, 0.f)));
m.immersiveDisplay->SetEyeTransform(device::Eye::Left, m.cameras[0]->GetEyeTransform());
m.immersiveDisplay->SetEyeTransform(device::Eye::Right, m.cameras[1]->GetEyeTransform());

// Update controller
if (!m.controller)
return;

if (auto context = m.context.lock()) {
float level = 100.0 - std::fmod(context->GetTimestamp(), 100.0);
m.controller->SetBatteryLevel(kControllerIndex, (int32_t)level);
}
auto transformMatrix = vrb::Matrix::Rotation(m.controllerOrientation);
m.controller->SetTransform(kControllerIndex, transformMatrix);
m.controller->SetBeamTransform(kControllerIndex, vrb::Matrix::Identity());
m.controller->SetImmersiveBeamTransform(kControllerIndex, vrb::Matrix::Identity());
}

void
Expand Down Expand Up @@ -317,48 +330,6 @@ Clamp(const float aValue) {
return aValue;
}

void
DeviceDelegateVisionGlass::TouchEvent(const bool aDown, const float aX, const float aY) {
static const vrb::Vector sForward(0.0f, 0.0f, -1.0f);
if (!m.controller) {
return;
}
if (m.renderMode == device::RenderMode::Immersive) {
m.controller->SetButtonState(kControllerIndex, ControllerDelegate::BUTTON_TOUCHPAD, 0, false, false);
m.clicked = false;
} else if (aDown != m.clicked) {
m.controller->SetButtonState(kControllerIndex, ControllerDelegate::BUTTON_TOUCHPAD, 0, aDown, aDown);
m.clicked = aDown;
}

const float viewportWidth = m.glWidth;
const float viewportHeight = m.glHeight;
if ((viewportWidth <= 0.0f) || (viewportHeight <= 0.0f)) {
return;
}
const float xModifier = (m.renderMode == device::RenderMode::Immersive ? viewportWidth / 2.0f : 0.0f);
const float width = Clamp((((aX - xModifier) / viewportWidth) * 2.0f) - 1.0f);
const float height = (((viewportHeight - aY) / viewportHeight) * 2.0f) - 1.0f;

vrb::Vector start(width, height, -1.0f);
vrb::Vector end(width, height, 1.0f);
vrb::Matrix inversePerspective = m.cameras[0]->GetPerspective().Inverse();
start = inversePerspective.MultiplyPosition(start);
end = inversePerspective.MultiplyPosition(end);
vrb::Matrix view = m.cameras[0]->GetTransform();
start = view.MultiplyPosition(start);
end = view.MultiplyPosition(end);
const vrb::Vector direction = (end - start).Normalize();
const vrb::Vector up = sForward.Cross(direction);
const float angle = acosf(sForward.Dot(direction));
vrb::Matrix transform = vrb::Matrix::Rotation(up, angle);
if (m.renderMode == device::RenderMode::Immersive) {
start += direction * 0.3f;
}
transform.TranslateInPlace(start);
m.controller->SetTransform(kControllerIndex, transform);
}

void
DeviceDelegateVisionGlass::ControllerButtonPressed(const bool aDown) {
if (!m.controller) {
Expand All @@ -380,6 +351,11 @@ DeviceDelegateVisionGlass::setHead(const float aX, const float aY, const float a
m.headingMatrix = vrb::Matrix::Rotation({aX,aY,-aZ,-aW});
}

void
DeviceDelegateVisionGlass::setControllerOrientation(const float aX, const float aY, const float aZ, const float aW) {
m.controllerOrientation = vrb::Quaternion(aX, aY, aZ, aW);
}

DeviceDelegateVisionGlass::DeviceDelegateVisionGlass(State& aState) : m(aState) {}
DeviceDelegateVisionGlass::~DeviceDelegateVisionGlass() { m.Shutdown(); }

Expand Down
1 change: 1 addition & 0 deletions app/src/visionglass/cpp/DeviceDelegateVisionGlass.h
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@ class DeviceDelegateVisionGlass : public DeviceDelegate {
void TouchEvent(const bool aDown, const float aX, const float aY);
void ControllerButtonPressed(const bool aDown);
void setHead(const float aX, const float aY, const float aZ, const float aW);
void setControllerOrientation(const float aX, const float aY, const float aZ, const float aW);
protected:
struct State;
DeviceDelegateVisionGlass(State& aState);
Expand Down
13 changes: 12 additions & 1 deletion app/src/visionglass/cpp/native-lib.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -85,14 +85,25 @@ JNI_METHOD(void, drawGL)

JNI_METHOD(void, touchEvent)
(JNIEnv*, jobject, jboolean aDown, jfloat aX, jfloat aY) {
sAppContext->mDevice->TouchEvent(aDown, aX, aY);
sAppContext->mDevice->ControllerButtonPressed(aDown);
}

JNI_METHOD(void, setHead)
(JNIEnv*, jobject, jdouble aX, jdouble aY, jdouble aZ, jdouble aW) {
if (!sAppContext || !sAppContext->mDevice) {
return;
}
sAppContext->mDevice->setHead(aX, aY, aZ, aW);
}

JNI_METHOD(void, setControllerOrientation)
(JNIEnv*, jobject, jdouble aX, jdouble aY, jdouble aZ, jdouble aW) {
if (!sAppContext || !sAppContext->mDevice) {
return;
}
sAppContext->mDevice->setControllerOrientation(aX, aY, aZ, aW);
}

jint JNI_OnLoad(JavaVM* aVm, void*) {
if (sAppContext) {
return JNI_VERSION_1_6;
Expand Down
Loading

0 comments on commit ffc8d20

Please sign in to comment.