#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "datatypes/cframe.h" #include "editorcommon.h" #include "mainwindow.h" #include "objects/handles.h" #include "physics/util.h" #include "qcursor.h" #include "qevent.h" #include "qnamespace.h" #include "qwindowdefs.h" #include "rendering/renderer.h" #include "physics/simulation.h" #include "camera.h" #include "common.h" #include "rendering/shader.h" #include "mainglwidget.h" #include "../core/src/rendering/defaultmeshes.h" #include "math_helper.h" MainGLWidget::MainGLWidget(QWidget* parent): QOpenGLWidget(parent) { setFocusPolicy(Qt::FocusPolicy::ClickFocus); setMouseTracking(true); } void MainGLWidget::initializeGL() { glewInit(); renderInit(NULL, width(), height()); } extern int vpx, vpy; void MainGLWidget::resizeGL(int w, int h) { // Update projection matrix and other size related settings: // m_projection.setToIdentity(); // m_projection.perspective(45.0f, w / float(h), 0.01f, 100.0f); // ... // glViewport(0, 0, w, h); setViewport(w, h); } glm::vec2 firstPoint; glm::vec2 secondPoint; extern std::optional> draggingObject; extern std::optional draggingHandle; extern Shader* shader; void MainGLWidget::paintGL() { ::render(NULL); } bool isMouseRightDragging = false; QPoint lastMousePos; void MainGLWidget::handleCameraRotate(QMouseEvent* evt) { if (!isMouseRightDragging) return; camera.processRotation(evt->pos().x() - lastMousePos.x(), evt->pos().y() - lastMousePos.y()); lastMousePos = evt->pos(); // QCursor::setPos(lastMousePos); } bool isMouseDragging = false; std::optional> draggingObject; std::optional draggingHandle; void MainGLWidget::handleObjectDrag(QMouseEvent* evt) { if (!isMouseDragging || !draggingObject) return; QPoint position = evt->pos(); glm::vec3 pointDir = camera.getScreenDirection(glm::vec2(position.x(), position.y()), glm::vec2(width(), height())); std::optional rayHit = castRayNearest(camera.cameraPos, pointDir, 50000, [](std::shared_ptr part) { return (part == draggingObject->lock()) ? FilterResult::PASS : FilterResult::TARGET; }); if (!rayHit) return; Data::Vector3 vec = rayHit->worldPoint; vec = vec + Data::Vector3(rpToGlm(rayHit->worldNormal) * draggingObject->lock()->size / 2.f); draggingObject->lock()->cframe = draggingObject->lock()->cframe.Rotation() + vec; syncPartPhysics(draggingObject->lock()); } inline glm::vec3 vec3fy(glm::vec4 vec) { return vec / vec.w; } QPoint lastPoint; void MainGLWidget::handleHandleDrag(QMouseEvent* evt) { QPoint cLastPoint = lastPoint; lastPoint = evt->pos(); if (!isMouseDragging || !draggingHandle || !editorToolHandles->adornee || !editorToolHandles->active) return; QPoint position = evt->pos(); auto part = editorToolHandles->adornee->lock(); // This was actually quite a difficult problem to solve, managing to get the handle to go underneath the cursor glm::vec3 pointDir = camera.getScreenDirection(glm::vec2(position.x(), position.y()), glm::vec2(width(), height())); pointDir = glm::normalize(pointDir); Data::CFrame handleCFrame = editorToolHandles->GetCFrameOfHandle(draggingHandle.value()); // Current frame. Identity frame if worldMode == true, selected object's frame if worldMode == false Data::CFrame frame = editorToolHandles->worldMode ? Data::CFrame::IDENTITY + part->position() : part->cframe.Rotation(); // Segment from axis stretching -4096 to +4096 rel to handle's position glm::vec3 axisSegment0 = handleCFrame.Position() + (-handleCFrame.LookVector() * 4096.0f); glm::vec3 axisSegment1 = handleCFrame.Position() + (-handleCFrame.LookVector() * -4096.0f); // Segment from camera stretching 4096 forward glm::vec3 mouseSegment0 = camera.cameraPos; glm::vec3 mouseSegment1 = camera.cameraPos + pointDir * 4096.0f; // Closest point on the axis segment between the two segments glm::vec3 handlePoint, rb; get_closest_points_between_segments(axisSegment0, axisSegment1, mouseSegment0, mouseSegment1, handlePoint, rb); // Find new part position glm::vec3 centerPoint = editorToolHandles->PartCFrameFromHandlePos(draggingHandle.value(), handlePoint).Position(); // Apply snapping in the current frame glm::vec3 diff = centerPoint - (glm::vec3)editorToolHandles->adornee->lock()->position(); if (snappingFactor()) diff = frame * (glm::round(glm::vec3(frame.Inverse() * diff) / snappingFactor()) * snappingFactor()); switch (mainWindow()->selectedTool) { case SelectedTool::SELECT: break; case SelectedTool::MOVE: { // Add difference editorToolHandles->adornee->lock()->cframe = editorToolHandles->adornee->lock()->cframe + diff; } break; case SelectedTool::SCALE: { // Find local difference glm::vec3 localDiff = frame.Inverse() * diff; // Find outwarwd difference localDiff = localDiff * glm::sign(draggingHandle->normal); // Add local difference to size part->size += localDiff; // If ctrl is not pressed, offset the part by half the size difference to keep the other bound where it was originally if (!(evt->modifiers() & Qt::ControlModifier)) part->cframe = part->cframe + diff * 0.5f; } break; case SelectedTool::ROTATE: { // TODO: Implement rotation } break; } syncPartPhysics(std::dynamic_pointer_cast(editorToolHandles->adornee->lock())); } std::optional MainGLWidget::raycastHandle(glm::vec3 pointDir) { if (!editorToolHandles->adornee.has_value() || !editorToolHandles->active) return std::nullopt; return editorToolHandles->RaycastHandle(rp3d::Ray(glmToRp(camera.cameraPos), glmToRp(glm::normalize(pointDir)) * 50000)); } void MainGLWidget::handleCursorChange(QMouseEvent* evt) { QPoint position = evt->pos(); glm::vec3 pointDir = camera.getScreenDirection(glm::vec2(position.x(), position.y()), glm::vec2(width(), height())); if (raycastHandle(pointDir)) { setCursor(Qt::OpenHandCursor); return; }; std::optional rayHit = castRayNearest(camera.cameraPos, pointDir, 50000); if (rayHit && partFromBody(rayHit->body)->name != "Baseplate") { setCursor(Qt::OpenHandCursor); return; } setCursor(Qt::ArrowCursor); } void MainGLWidget::mouseMoveEvent(QMouseEvent* evt) { handleCameraRotate(evt); handleObjectDrag(evt); handleHandleDrag(evt); handleCursorChange(evt); } void MainGLWidget::mousePressEvent(QMouseEvent* evt) { switch(evt->button()) { // Camera drag case Qt::RightButton: { lastMousePos = evt->pos(); isMouseRightDragging = true; return; // Clicking on objects } case Qt::LeftButton: { QPoint position = evt->pos(); glm::vec3 pointDir = camera.getScreenDirection(glm::vec2(position.x(), position.y()), glm::vec2(width(), height())); // raycast handles auto handle = raycastHandle(pointDir); if (handle.has_value()) { isMouseDragging = true; draggingHandle = handle; return; } // raycast part std::optional rayHit = castRayNearest(camera.cameraPos, pointDir, 50000); if (!rayHit || !partFromBody(rayHit->body)) return; std::shared_ptr part = partFromBody(rayHit->body); if (part->name == "Baseplate") return; //part.selected = true; isMouseDragging = true; draggingObject = part; setSelection(std::vector { part }); // Disable bit so that we can ignore the part while raycasting // part->rigidBody->getCollider(0)->setCollisionCategoryBits(0b10); return; } default: return; } } void MainGLWidget::mouseReleaseEvent(QMouseEvent* evt) { // if (isMouseDragging) draggingObject->lock()->rigidBody->getCollider(0)->setCollisionCategoryBits(0b11); isMouseRightDragging = false; isMouseDragging = false; draggingObject = std::nullopt; draggingHandle = std::nullopt; } static int moveZ = 0; static int moveX = 0; static std::chrono::time_point lastTime = std::chrono::steady_clock::now(); void MainGLWidget::updateCycle() { float deltaTime = std::chrono::duration_cast>(std::chrono::steady_clock::now() - lastTime).count(); lastTime = std::chrono::steady_clock::now(); if (moveZ) camera.processMovement(moveZ == 1 ? DIRECTION_FORWARD : DIRECTION_BACKWARDS, deltaTime); if (moveX) camera.processMovement(moveX == 1 ? DIRECTION_LEFT : DIRECTION_RIGHT, deltaTime); } void MainGLWidget::keyPressEvent(QKeyEvent* evt) { if (evt->key() == Qt::Key_W) moveZ = 1; else if (evt->key() == Qt::Key_S) moveZ = -1; if (evt->key() == Qt::Key_A) moveX = 1; else if (evt->key() == Qt::Key_D) moveX = -1; if (evt->key() == Qt::Key_F) { workspace()->AddChild(lastPart = Part::New({ .position = camera.cameraPos + camera.cameraFront * glm::vec3(3), .rotation = glm::vec3(0), .size = glm::vec3(1, 1, 1), .color = glm::vec3(1.0f, 0.5f, 0.31f), })); syncPartPhysics(lastPart); } } void MainGLWidget::keyReleaseEvent(QKeyEvent* evt) { if (evt->key() == Qt::Key_W || evt->key() == Qt::Key_S) moveZ = 0; else if (evt->key() == Qt::Key_A || evt->key() == Qt::Key_D) moveX = 0; } MainWindow* MainGLWidget::mainWindow() { return dynamic_cast(window()); } float MainGLWidget::snappingFactor() { switch (mainWindow()->snappingMode) { case GridSnappingMode::SNAP_1_STUD: return 1; case GridSnappingMode::SNAP_05_STUDS: return 0.5; case GridSnappingMode::SNAP_OFF: return 0; } return 0; }