view Framework/Viewport/WebAssemblyViewport.cpp @ 942:685c9a2d115f

Added missing ORTHANC_OVERRIDE + preparation for lost GL context handling + stubs for GL context event handlers
author Benjamin Golinvaux <bgo@osimis.io>
date Mon, 05 Aug 2019 12:27:27 +0200
parents a6c12fe88bcb
children 1091b2adeb5a
line wrap: on
line source

/**
 * Stone of Orthanc
 * Copyright (C) 2012-2016 Sebastien Jodogne, Medical Physics
 * Department, University Hospital of Liege, Belgium
 * Copyright (C) 2017-2019 Osimis S.A., Belgium
 *
 * This program is free software: you can redistribute it and/or
 * modify it under the terms of the GNU Affero General Public License
 * as published by the Free Software Foundation, either version 3 of
 * the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Affero General Public License for more details.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 **/


#include "WebAssemblyViewport.h"
#include <emscripten/html5.h>

namespace OrthancStone
{
  WebAssemblyOpenGLViewport::WebAssemblyOpenGLViewport(const std::string& canvas) :
    WebAssemblyViewport(canvas),
    context_(canvas)
  {
    compositor_.reset(new OpenGLCompositor(context_, GetScene()));
    RegisterContextCallbacks();
  }

  WebAssemblyOpenGLViewport::WebAssemblyOpenGLViewport(const std::string& canvas,
    boost::shared_ptr<Scene2D>& scene) :
    WebAssemblyViewport(canvas, scene),
    context_(canvas)
  {
    compositor_.reset(new OpenGLCompositor(context_, GetScene()));
    RegisterContextCallbacks();
  }

  void WebAssemblyOpenGLViewport::UpdateSize()
  {
    context_.UpdateSize();  // First read the size of the canvas
    compositor_->Refresh();  // Then refresh the content of the canvas
  }

  /*
  typedef EM_BOOL (*em_webgl_context_callback)(int eventType, const void *reserved, void *userData);

  EMSCRIPTEN_EVENT_WEBGLCONTEXTLOST EMSCRIPTEN_EVENT_WEBGLCONTEXTRESTORED

  EMSCRIPTEN_RESULT emscripten_set_webglcontextlost_callback(
    const char *target, void *userData, EM_BOOL useCapture, em_webgl_context_callback callback)

  EMSCRIPTEN_RESULT emscripten_set_webglcontextrestored_callback(
    const char *target, void *userData, EM_BOOL useCapture, em_webgl_context_callback callback)

  */

  EM_BOOL WebAssemblyOpenGLViewport_OpenGLContextLost_callback(
    int eventType, const void* reserved, void* userData)
  {
    ORTHANC_ASSERT(eventType == EMSCRIPTEN_EVENT_WEBGLCONTEXTLOST);
    WebAssemblyOpenGLViewport* viewport = reinterpret_cast<WebAssemblyOpenGLViewport*>(userData);
    return viewport->OpenGLContextLost();
  }

  EM_BOOL WebAssemblyOpenGLViewport_OpenGLContextRestored_callback(
    int eventType, const void* reserved, void* userData)
  {
    ORTHANC_ASSERT(eventType == EMSCRIPTEN_EVENT_WEBGLCONTEXTRESTORED);
    WebAssemblyOpenGLViewport* viewport = reinterpret_cast<WebAssemblyOpenGLViewport*>(userData);
    return viewport->OpenGLContextRestored();
  }

  void WebAssemblyOpenGLViewport::RegisterContextCallbacks()
  {
    // TODO: what's the impact of userCapture=true ?
    const char* canvasId = GetCanvasIdentifier().c_str();
    void* that = reinterpret_cast<void*>(this);
    EMSCRIPTEN_RESULT status = EMSCRIPTEN_RESULT_SUCCESS;

    status = emscripten_set_webglcontextlost_callback(canvasId, that, true, WebAssemblyOpenGLViewport_OpenGLContextLost_callback);
    if (status != EMSCRIPTEN_RESULT_SUCCESS)
    {
      std::stringstream ss;
      ss << "Error while calling emscripten_set_webglcontextlost_callback for: \"" << GetCanvasIdentifier() << "\"";
      std::string msg = ss.str();
      LOG(ERROR) << msg;
      ORTHANC_ASSERT(false, msg.c_str());
    }
    status = emscripten_set_webglcontextrestored_callback(canvasId, that, true, WebAssemblyOpenGLViewport_OpenGLContextRestored_callback);
    if (status != EMSCRIPTEN_RESULT_SUCCESS)
    {
      std::stringstream ss;
      ss << "Error while calling emscripten_set_webglcontextrestored_callback for: \"" << GetCanvasIdentifier() << "\"";
      std::string msg = ss.str();
      LOG(ERROR) << msg;
      ORTHANC_ASSERT(false, msg.c_str());
    }
  }

  bool WebAssemblyOpenGLViewport::OpenGLContextLost()
  {
    LOG(ERROR) << "WebAssemblyOpenGLViewport::OpenGLContextLost() for canvas: " << GetCanvasIdentifier();
    return false;
  }

  bool WebAssemblyOpenGLViewport::OpenGLContextRestored()
  {
    LOG(ERROR) << "WebAssemblyOpenGLViewport::OpenGLContextRestored() for canvas: " << GetCanvasIdentifier();
    return false;
  }

  WebAssemblyCairoViewport::WebAssemblyCairoViewport(const std::string& canvas) :
    WebAssemblyViewport(canvas),
    canvas_(canvas),
    compositor_(GetScene(), 1024, 768)
  {
  }

  WebAssemblyCairoViewport::WebAssemblyCairoViewport(const std::string& canvas,
    boost::shared_ptr<Scene2D>& scene) :
    WebAssemblyViewport(canvas, scene),
    canvas_(canvas),
    compositor_(GetScene(), 1024, 768)
  {
  }

  void WebAssemblyCairoViewport::UpdateSize()
  {
    LOG(INFO) << "updating cairo viewport size";
    double w, h;
    emscripten_get_element_css_size(canvas_.c_str(), &w, &h);

    /**
     * Emscripten has the function emscripten_get_element_css_size()
     * to query the width and height of a named HTML element. I'm
     * calling this first to get the initial size of the canvas DOM
     * element, and then call emscripten_set_canvas_size() to
     * initialize the framebuffer size of the canvas to the same
     * size as its DOM element.
     * https://floooh.github.io/2017/02/22/emsc-html.html
     **/
    unsigned int canvasWidth = 0;
    unsigned int canvasHeight = 0;

    if (w > 0 ||
      h > 0)
    {
      canvasWidth = static_cast<unsigned int>(boost::math::iround(w));
      canvasHeight = static_cast<unsigned int>(boost::math::iround(h));
    }

    emscripten_set_canvas_element_size(canvas_.c_str(), canvasWidth, canvasHeight);
    compositor_.UpdateSize(canvasWidth, canvasHeight);
  }

  void WebAssemblyCairoViewport::Refresh()
  {
    LOG(INFO) << "refreshing cairo viewport, TODO: blit to the canvans.getContext('2d')";
    GetCompositor().Refresh();
  }

}