2018-05-09 08:47:04 +00:00
|
|
|
#include "GLCanvas3DManager.hpp"
|
|
|
|
#include "../../slic3r/GUI/GUI.hpp"
|
|
|
|
#include "../../slic3r/GUI/AppConfig.hpp"
|
2018-06-11 13:13:13 +00:00
|
|
|
#include "../../slic3r/GUI/GLCanvas3D.hpp"
|
2018-05-09 08:47:04 +00:00
|
|
|
|
|
|
|
#include <GL/glew.h>
|
|
|
|
|
|
|
|
#include <boost/algorithm/string/split.hpp>
|
|
|
|
#include <boost/algorithm/string/classification.hpp>
|
|
|
|
|
|
|
|
#include <wx/glcanvas.h>
|
2018-05-30 13:18:45 +00:00
|
|
|
#include <wx/timer.h>
|
2018-05-09 08:47:04 +00:00
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
|
2018-06-04 08:14:09 +00:00
|
|
|
GLCanvas3DManager::GLInfo::GLInfo()
|
|
|
|
: version("")
|
|
|
|
, glsl_version("")
|
|
|
|
, vendor("")
|
|
|
|
, renderer("")
|
2018-05-09 08:47:04 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-06-20 12:34:20 +00:00
|
|
|
void GLCanvas3DManager::GLInfo::detect()
|
2018-05-09 08:47:04 +00:00
|
|
|
{
|
2018-06-04 08:14:09 +00:00
|
|
|
const char* data = (const char*)::glGetString(GL_VERSION);
|
2018-06-20 12:34:20 +00:00
|
|
|
if (data != nullptr)
|
|
|
|
version = data;
|
2018-06-04 08:14:09 +00:00
|
|
|
|
|
|
|
data = (const char*)::glGetString(GL_SHADING_LANGUAGE_VERSION);
|
2018-06-20 12:34:20 +00:00
|
|
|
if (data != nullptr)
|
|
|
|
glsl_version = data;
|
2018-06-04 08:14:09 +00:00
|
|
|
|
|
|
|
data = (const char*)::glGetString(GL_VENDOR);
|
2018-06-20 12:34:20 +00:00
|
|
|
if (data != nullptr)
|
|
|
|
vendor = data;
|
2018-06-04 08:14:09 +00:00
|
|
|
|
|
|
|
data = (const char*)::glGetString(GL_RENDERER);
|
2018-06-20 12:34:20 +00:00
|
|
|
if (data != nullptr)
|
|
|
|
renderer = data;
|
2018-06-04 08:14:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GLCanvas3DManager::GLInfo::is_version_greater_or_equal_to(unsigned int major, unsigned int minor) const
|
|
|
|
{
|
2018-05-09 08:47:04 +00:00
|
|
|
std::vector<std::string> tokens;
|
2018-06-04 08:14:09 +00:00
|
|
|
boost::split(tokens, version, boost::is_any_of(" "), boost::token_compress_on);
|
2018-05-09 08:47:04 +00:00
|
|
|
|
|
|
|
if (tokens.empty())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
std::vector<std::string> numbers;
|
|
|
|
boost::split(numbers, tokens[0], boost::is_any_of("."), boost::token_compress_on);
|
|
|
|
|
2018-06-04 08:14:09 +00:00
|
|
|
unsigned int gl_major = 0;
|
|
|
|
unsigned int gl_minor = 0;
|
|
|
|
|
2018-05-09 08:47:04 +00:00
|
|
|
if (numbers.size() > 0)
|
2018-06-04 08:14:09 +00:00
|
|
|
gl_major = ::atoi(numbers[0].c_str());
|
2018-05-09 08:47:04 +00:00
|
|
|
|
|
|
|
if (numbers.size() > 1)
|
2018-06-04 08:14:09 +00:00
|
|
|
gl_minor = ::atoi(numbers[1].c_str());
|
2018-05-09 08:47:04 +00:00
|
|
|
|
2018-06-04 08:14:09 +00:00
|
|
|
if (gl_major < major)
|
2018-05-09 08:47:04 +00:00
|
|
|
return false;
|
2018-06-04 08:14:09 +00:00
|
|
|
else if (gl_major > major)
|
2018-05-09 08:47:04 +00:00
|
|
|
return true;
|
|
|
|
else
|
2018-06-04 08:14:09 +00:00
|
|
|
return gl_minor >= minor;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string GLCanvas3DManager::GLInfo::to_string(bool format_as_html, bool extensions) const
|
|
|
|
{
|
|
|
|
std::stringstream out;
|
|
|
|
|
|
|
|
std::string h2_start = format_as_html ? "<b>" : "";
|
2018-07-12 11:34:39 +00:00
|
|
|
std::string h2_end = format_as_html ? "</b>" : "";
|
|
|
|
std::string b_start = format_as_html ? "<b>" : "";
|
|
|
|
std::string b_end = format_as_html ? "</b>" : "";
|
2018-06-04 08:14:09 +00:00
|
|
|
std::string line_end = format_as_html ? "<br>" : "\n";
|
|
|
|
|
|
|
|
out << h2_start << "OpenGL installation" << h2_end << line_end;
|
2018-07-12 11:34:39 +00:00
|
|
|
out << b_start << "GL version: " << b_end << (version.empty() ? "N/A" : version) << line_end;
|
|
|
|
out << b_start << "Vendor: " << b_end << (vendor.empty() ? "N/A" : vendor) << line_end;
|
|
|
|
out << b_start << "Renderer: " << b_end << (renderer.empty() ? "N/A" : renderer) << line_end;
|
|
|
|
out << b_start << "GLSL version: " << b_end << (glsl_version.empty() ? "N/A" : glsl_version) << line_end;
|
2018-07-12 10:15:30 +00:00
|
|
|
|
2018-06-04 08:14:09 +00:00
|
|
|
if (extensions)
|
|
|
|
{
|
|
|
|
std::vector<std::string> extensions_list;
|
2018-07-12 11:10:18 +00:00
|
|
|
std::string extensions_str = (const char*)::glGetString(GL_EXTENSIONS);
|
|
|
|
boost::split(extensions_list, extensions_str, boost::is_any_of(" "), boost::token_compress_off);
|
|
|
|
|
2018-07-12 10:15:30 +00:00
|
|
|
if (!extensions_list.empty())
|
2018-06-04 08:14:09 +00:00
|
|
|
{
|
2018-07-12 10:15:30 +00:00
|
|
|
out << h2_start << "Installed extensions:" << h2_end << line_end;
|
|
|
|
|
|
|
|
std::sort(extensions_list.begin(), extensions_list.end());
|
|
|
|
for (const std::string& ext : extensions_list)
|
|
|
|
{
|
|
|
|
out << ext << line_end;
|
|
|
|
}
|
2018-06-04 08:14:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return out.str();
|
2018-05-09 08:47:04 +00:00
|
|
|
}
|
|
|
|
|
2018-10-02 12:01:22 +00:00
|
|
|
GLCanvas3DManager::EMultisampleState GLCanvas3DManager::s_multisample = GLCanvas3DManager::MS_Unknown;
|
|
|
|
|
2018-05-09 08:47:04 +00:00
|
|
|
GLCanvas3DManager::GLCanvas3DManager()
|
2018-10-04 08:41:11 +00:00
|
|
|
: m_context(nullptr)
|
2018-06-11 13:49:04 +00:00
|
|
|
, m_gl_initialized(false)
|
2018-05-09 08:47:04 +00:00
|
|
|
, m_use_legacy_opengl(false)
|
|
|
|
, m_use_VBOs(false)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-10-04 08:41:11 +00:00
|
|
|
GLCanvas3DManager::~GLCanvas3DManager()
|
|
|
|
{
|
|
|
|
if (m_context != nullptr)
|
|
|
|
{
|
|
|
|
delete m_context;
|
|
|
|
m_context = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-11 13:49:04 +00:00
|
|
|
bool GLCanvas3DManager::add(wxGLCanvas* canvas)
|
|
|
|
{
|
|
|
|
if (canvas == nullptr)
|
|
|
|
return false;
|
|
|
|
|
2018-05-09 08:47:04 +00:00
|
|
|
if (_get_canvas(canvas) != m_canvases.end())
|
|
|
|
return false;
|
|
|
|
|
2018-06-27 09:31:11 +00:00
|
|
|
GLCanvas3D* canvas3D = new GLCanvas3D(canvas);
|
2018-05-09 08:47:04 +00:00
|
|
|
if (canvas3D == nullptr)
|
|
|
|
return false;
|
|
|
|
|
2018-06-06 12:19:28 +00:00
|
|
|
canvas3D->bind_event_handlers();
|
2018-10-04 08:41:11 +00:00
|
|
|
|
|
|
|
if (m_context == nullptr)
|
|
|
|
{
|
|
|
|
m_context = new wxGLContext(canvas);
|
|
|
|
if (m_context == nullptr)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
canvas3D->set_context(m_context);
|
|
|
|
|
2018-05-09 08:47:04 +00:00
|
|
|
m_canvases.insert(CanvasesMap::value_type(canvas, canvas3D));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GLCanvas3DManager::remove(wxGLCanvas* canvas)
|
|
|
|
{
|
|
|
|
CanvasesMap::iterator it = _get_canvas(canvas);
|
|
|
|
if (it == m_canvases.end())
|
|
|
|
return false;
|
|
|
|
|
2018-06-06 12:19:28 +00:00
|
|
|
it->second->unbind_event_handlers();
|
2018-05-09 08:47:04 +00:00
|
|
|
delete it->second;
|
|
|
|
m_canvases.erase(it);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GLCanvas3DManager::remove_all()
|
|
|
|
{
|
|
|
|
for (CanvasesMap::value_type& item : m_canvases)
|
|
|
|
{
|
2018-06-06 12:19:28 +00:00
|
|
|
item.second->unbind_event_handlers();
|
2018-05-09 08:47:04 +00:00
|
|
|
delete item.second;
|
|
|
|
}
|
|
|
|
m_canvases.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int GLCanvas3DManager::count() const
|
|
|
|
{
|
|
|
|
return (unsigned int)m_canvases.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GLCanvas3DManager::init_gl()
|
|
|
|
{
|
|
|
|
if (!m_gl_initialized)
|
|
|
|
{
|
|
|
|
glewInit();
|
2018-06-20 12:34:20 +00:00
|
|
|
m_gl_info.detect();
|
|
|
|
const AppConfig* config = GUI::get_app_config();
|
|
|
|
m_use_legacy_opengl = (config == nullptr) || (config->get("use_legacy_opengl") == "1");
|
|
|
|
m_use_VBOs = !m_use_legacy_opengl && m_gl_info.is_version_greater_or_equal_to(2, 0);
|
|
|
|
m_gl_initialized = true;
|
2018-05-09 08:47:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-04 08:14:09 +00:00
|
|
|
std::string GLCanvas3DManager::get_gl_info(bool format_as_html, bool extensions) const
|
|
|
|
{
|
|
|
|
return m_gl_info.to_string(format_as_html, extensions);
|
|
|
|
}
|
|
|
|
|
2018-06-04 11:15:28 +00:00
|
|
|
bool GLCanvas3DManager::init(wxGLCanvas* canvas)
|
2018-05-23 07:57:44 +00:00
|
|
|
{
|
|
|
|
CanvasesMap::const_iterator it = _get_canvas(canvas);
|
2018-06-04 11:15:28 +00:00
|
|
|
if (it != m_canvases.end())
|
|
|
|
return (it->second != nullptr) ? _init(*it->second) : false;
|
|
|
|
else
|
|
|
|
return false;
|
2018-05-23 07:57:44 +00:00
|
|
|
}
|
|
|
|
|
2018-10-11 08:24:19 +00:00
|
|
|
GLCanvas3D* GLCanvas3DManager::get_canvas(wxGLCanvas* canvas)
|
|
|
|
{
|
|
|
|
CanvasesMap::const_iterator it = _get_canvas(canvas);
|
|
|
|
return (it != m_canvases.end()) ? it->second : nullptr;
|
|
|
|
}
|
2018-06-08 07:40:00 +00:00
|
|
|
|
2018-10-01 14:48:08 +00:00
|
|
|
wxGLCanvas* GLCanvas3DManager::create_wxglcanvas(wxWindow *parent)
|
|
|
|
{
|
|
|
|
int attribList[] = { WX_GL_RGBA, WX_GL_DOUBLEBUFFER, WX_GL_DEPTH_SIZE, 24, WX_GL_SAMPLE_BUFFERS, GL_TRUE, WX_GL_SAMPLES, 4, 0 };
|
|
|
|
|
2018-10-02 12:01:22 +00:00
|
|
|
if (s_multisample == MS_Unknown)
|
|
|
|
{
|
|
|
|
_detect_multisample(attribList);
|
|
|
|
// debug output
|
|
|
|
std::cout << "Multisample " << (can_multisample() ? "enabled" : "disabled") << std::endl;
|
|
|
|
}
|
|
|
|
|
2018-10-01 14:48:08 +00:00
|
|
|
if (! can_multisample()) {
|
|
|
|
attribList[4] = 0;
|
|
|
|
}
|
|
|
|
|
2019-02-15 14:35:32 +00:00
|
|
|
return new wxGLCanvas(parent, wxID_ANY, attribList, wxDefaultPosition, wxDefaultSize, wxWANTS_CHARS);
|
2018-10-01 14:48:08 +00:00
|
|
|
}
|
|
|
|
|
2018-05-09 08:47:04 +00:00
|
|
|
GLCanvas3DManager::CanvasesMap::iterator GLCanvas3DManager::_get_canvas(wxGLCanvas* canvas)
|
|
|
|
{
|
|
|
|
return (canvas == nullptr) ? m_canvases.end() : m_canvases.find(canvas);
|
|
|
|
}
|
|
|
|
|
|
|
|
GLCanvas3DManager::CanvasesMap::const_iterator GLCanvas3DManager::_get_canvas(wxGLCanvas* canvas) const
|
|
|
|
{
|
|
|
|
return (canvas == nullptr) ? m_canvases.end() : m_canvases.find(canvas);
|
|
|
|
}
|
|
|
|
|
2018-06-04 11:15:28 +00:00
|
|
|
bool GLCanvas3DManager::_init(GLCanvas3D& canvas)
|
|
|
|
{
|
|
|
|
if (!m_gl_initialized)
|
|
|
|
init_gl();
|
|
|
|
|
|
|
|
return canvas.init(m_use_VBOs, m_use_legacy_opengl);
|
|
|
|
}
|
|
|
|
|
2018-10-02 12:01:22 +00:00
|
|
|
void GLCanvas3DManager::_detect_multisample(int* attribList)
|
|
|
|
{
|
|
|
|
int wxVersion = wxMAJOR_VERSION * 10000 + wxMINOR_VERSION * 100 + wxRELEASE_NUMBER;
|
|
|
|
const AppConfig* app_config = GUI::get_app_config();
|
|
|
|
bool enable_multisample = app_config != nullptr
|
|
|
|
&& app_config->get("use_legacy_opengl") != "1"
|
|
|
|
&& wxVersion >= 30003;
|
|
|
|
|
|
|
|
s_multisample = (enable_multisample && wxGLCanvas::IsDisplaySupported(attribList)) ? MS_Enabled : MS_Disabled;
|
|
|
|
// Alternative method: it was working on previous version of wxWidgets but not with the latest, at least on Windows
|
|
|
|
// s_multisample = enable_multisample && wxGLCanvas::IsExtensionSupported("WGL_ARB_multisample");
|
|
|
|
}
|
|
|
|
|
2018-05-09 08:47:04 +00:00
|
|
|
} // namespace GUI
|
|
|
|
} // namespace Slic3r
|