diff --git a/.gitignore b/.gitignore index d89e602c..30e831fe 100644 --- a/.gitignore +++ b/.gitignore @@ -12,4 +12,5 @@ OSP-MAGNUM.cbp CMakeSettings.json out/ .vs/ +.vscode/ build/ diff --git a/src/osp/util/logging.h b/src/osp/util/logging.h index 38a7eb26..1afde911 100644 --- a/src/osp/util/logging.h +++ b/src/osp/util/logging.h @@ -24,7 +24,7 @@ */ #pragma once -#define SPDLOG_ACTIVE_LEVEL SPDLOG_LEVEL_INFO +#define SPDLOG_ACTIVE_LEVEL SPDLOG_LEVEL_DEBUG #include namespace osp diff --git a/src/testapp/identifiers.h b/src/testapp/identifiers.h index 88fc2d1e..16b511ed 100644 --- a/src/testapp/identifiers.h +++ b/src/testapp/identifiers.h @@ -303,7 +303,8 @@ struct PlNewton #define TESTAPP_DATA_NEWTON_ACCEL 1, \ idAcceleration - +#define TESTAPP_DATA_NEWTON_ORIGIN_TRANSLATE 1, \ + idNwtSync #define TESTAPP_DATA_ROCKETS_NWT 1, \ idRocketsNwt diff --git a/src/testapp/main.cpp b/src/testapp/main.cpp index 97b61b6e..b4a5f8e4 100644 --- a/src/testapp/main.cpp +++ b/src/testapp/main.cpp @@ -308,6 +308,7 @@ void start_magnum_async() g_testApp.close_session(g_testApp.m_windowApp); OSP_LOG_INFO("Closed Magnum Application"); + std::exit(0); }); g_magnumThread.swap(t); } diff --git a/src/testapp/scenarios.cpp b/src/testapp/scenarios.cpp index 14afef6b..74e02ffb 100644 --- a/src/testapp/scenarios.cpp +++ b/src/testapp/scenarios.cpp @@ -370,6 +370,104 @@ static ScenarioMap_t make_scenarios() return setup_renderer; }); + add_scenario("lander", "Lander simulation game", [] (TestApp& rTestApp) -> RendererSetupFunc_t { + #define SCENE_SESSIONS scene, commonScene, uniCore, uniScnFrame, uniPlanet, physics, \ + prefabs, parts, signalsFloat, vehicleSpawn, vehicleSpawnVB, vehicles, \ + newton, nwtGravSet, nwtGrav, vehicleSpawnNwt, nwtRocketSet, rocketsNwt, \ + machRocket, machRcsDriver, nwtFollow + #define SCENE_SESSIONS_COUNT 21 + #define RENDERER_SESSIONS sceneRenderer, magnumScene, planetDraw, \ + cameraCtrl, cameraFree, shVisual, shFlat, shPhong, \ + prefabDraw, vehicleDraw, vehicleCtrl, cameraVehicle + #define RENDERER_SESSIONS_COUNT 12 + + using namespace testapp::scenes; + + auto const defaultPkg = rTestApp.m_defaultPkg; + auto const application = rTestApp.m_application; + auto & rTopData = rTestApp.m_topData; + + TopTaskBuilder builder{rTestApp.m_tasks, rTestApp.m_scene.m_edges, rTestApp.m_taskData}; + + auto & [SCENE_SESSIONS] = resize_then_unpack(rTestApp.m_scene.m_sessions); + + scene = setup_scene(builder, rTopData, application); + commonScene = setup_common_scene (builder, rTopData, scene, application, defaultPkg); + + auto const tgApp = application.get_pipelines< PlApplication >(); + uniCore = setup_uni_core (builder, rTopData, tgApp.mainLoop); + uniScnFrame = setup_uni_sceneframe (builder, rTopData, uniCore); + uniPlanet = setup_uni_landerplanet (builder, rTopData, uniCore, uniScnFrame); + + physics = setup_physics (builder, rTopData, scene, commonScene); + prefabs = setup_prefabs (builder, rTopData, application, scene, commonScene, physics); + parts = setup_parts (builder, rTopData, application, scene); + signalsFloat = setup_signals_float (builder, rTopData, scene, parts); + vehicleSpawn = setup_vehicle_spawn (builder, rTopData, scene); + vehicleSpawnVB = setup_vehicle_spawn_vb (builder, rTopData, application, scene, commonScene, prefabs, parts, vehicleSpawn, signalsFloat); + vehicles = setup_prebuilt_vehicles (builder, rTopData, application, scene); + + machRocket = setup_mach_rocket (builder, rTopData, scene, parts, signalsFloat); + machRcsDriver = setup_mach_rcsdriver (builder, rTopData, scene, parts, signalsFloat); + + newton = setup_newton (builder, rTopData, scene, commonScene, physics); + nwtGravSet = setup_newton_factors (builder, rTopData); + nwtGrav = setup_newton_force_grav_nbody(builder, rTopData, newton, nwtGravSet, uniCore, uniPlanet, uniScnFrame); + nwtFollow = setup_newton_origin_translate(builder, rTopData, scene, uniScnFrame, newton, physics); + vehicleSpawnNwt = setup_vehicle_spawn_newton (builder, rTopData, application, commonScene, physics, prefabs, parts, vehicleSpawn, newton); + nwtRocketSet = setup_newton_factors (builder, rTopData); + rocketsNwt = setup_rocket_thrust_newton (builder, rTopData, scene, commonScene, physics, prefabs, parts, signalsFloat, newton, nwtRocketSet); + + + OSP_DECLARE_GET_DATA_IDS(vehicleSpawn, TESTAPP_DATA_VEHICLE_SPAWN); + OSP_DECLARE_GET_DATA_IDS(vehicleSpawnVB, TESTAPP_DATA_VEHICLE_SPAWN_VB); + OSP_DECLARE_GET_DATA_IDS(vehicles, TESTAPP_DATA_TEST_VEHICLES); + + auto &rVehicleSpawn = top_get (rTopData, idVehicleSpawn); + auto &rVehicleSpawnVB = top_get (rTopData, idVehicleSpawnVB); + auto &rPrebuiltVehicles = top_get (rTopData, idPrebuiltVehicles); + + rVehicleSpawn.spawnRequest.push_back( + { + .position = {400.0f, 400.0f, 10.0f}, + .velocity = {-5.0f, -5.0f, 0.0f}, + .rotation = {} + }); + rVehicleSpawnVB.dataVB.push_back(rPrebuiltVehicles[gc_pbvSimpleCommandServiceModule].get()); + + RendererSetupFunc_t const setup_renderer = [] (TestApp& rTestApp) -> void + { + auto const application = rTestApp.m_application; + auto const windowApp = rTestApp.m_windowApp; + auto const magnum = rTestApp.m_magnum; + auto & rTopData = rTestApp.m_topData; + + TopTaskBuilder builder{rTestApp.m_tasks, rTestApp.m_renderer.m_edges, rTestApp.m_taskData}; + + auto & [SCENE_SESSIONS] = unpack(rTestApp.m_scene.m_sessions); + auto & [RENDERER_SESSIONS] = resize_then_unpack(rTestApp.m_renderer.m_sessions); + + sceneRenderer = setup_scene_renderer (builder, rTopData, application, windowApp, commonScene); + create_materials(rTopData, sceneRenderer, sc_materialCount); + + magnumScene = setup_magnum_scene (builder, rTopData, application, rTestApp.m_windowApp, sceneRenderer, rTestApp.m_magnum, scene, commonScene); + cameraCtrl = setup_camera_ctrl (builder, rTopData, windowApp, sceneRenderer, magnumScene); + // cameraFree = setup_camera_free (builder, rTopData, windowApp, scene, cameraCtrl); + shVisual = setup_shader_visualizer (builder, rTopData, windowApp, sceneRenderer, magnum, magnumScene, sc_matVisualizer); + // shFlat = setup_shader_flat (builder, rTopData, windowApp, sceneRenderer, magnum, magnumScene, sc_matFlat); + shPhong = setup_shader_phong (builder, rTopData, windowApp, sceneRenderer, magnum, magnumScene, sc_matPhong); + planetDraw = setup_landerplanet_draw (builder, rTopData, windowApp, sceneRenderer, cameraCtrl, commonScene, uniCore, uniScnFrame, uniPlanet, sc_matVisualizer, sc_matFlat); + + prefabDraw = setup_prefab_draw (builder, rTopData, application, windowApp, sceneRenderer, commonScene, prefabs, sc_matPhong); + vehicleDraw = setup_vehicle_spawn_draw (builder, rTopData, sceneRenderer, vehicleSpawn); + vehicleCtrl = setup_vehicle_control (builder, rTopData, windowApp, scene, parts, signalsFloat); + cameraVehicle = setup_camera_vehicle (builder, rTopData, windowApp, scene, sceneRenderer, commonScene, physics, parts, cameraCtrl, vehicleCtrl); + + setup_magnum_draw(rTestApp, scene, sceneRenderer, magnumScene); + }; + return setup_renderer; + }); + return scenarioMap; } diff --git a/src/testapp/sessions/newton.cpp b/src/testapp/sessions/newton.cpp index 22d59317..4b759765 100644 --- a/src/testapp/sessions/newton.cpp +++ b/src/testapp/sessions/newton.cpp @@ -33,6 +33,9 @@ #include #include #include +#include +#include +#include #include @@ -710,6 +713,156 @@ Session setup_rocket_thrust_newton( } // setup_rocket_thrust_newton +static void nbody_grav_force(NewtonBody const* pBody, BodyId const body, ACtxNwtWorld const& rNwt, ACtxNwtWorld::ForceFactorFunc::UserData_t data, Vector3& rForce, Vector3& rTorque) noexcept +{ + using namespace osp::universe; + + auto &rUniverse = *reinterpret_cast(data[0]); + auto &rMainSpace = *reinterpret_cast(data[1]); + auto &rScnFrame = *reinterpret_cast(data[2]); + + CoSpaceCommon &rMainSpaceCommon = rUniverse.m_coordCommon[rMainSpace]; + + float const invCommonScale = osp::math::mul_2pow(1.0f, -rMainSpaceCommon.m_precision); + float const sceneScale = osp::math::mul_2pow(1.0f, rScnFrame.m_precision); + + // get the positions of all the planets + auto const [x, y, z] = sat_views(rMainSpaceCommon.m_satPositions, rMainSpaceCommon.m_data, rMainSpaceCommon.m_satCount); + + // get the position/mass of this physics body + Vector3 pos = Vector3(0.0f); + float mass = 0.0f; + float dummy = 0.0f; + NewtonBodyGetPosition(pBody, pos.data()); + NewtonBodyGetMass(pBody, &mass, &dummy, &dummy, &dummy); + + // transform our local scene position to the universe space + Vector3g const bodyPos{rScnFrame.m_rotation.transformVector(Vector3d(rScnFrame.m_scenePosition) + Vector3d(pos))}; + Vector3g const areaPos{rScnFrame.m_position + bodyPos * sceneScale}; + + uint32_t const planetCount = rMainSpaceCommon.m_satCount; + + // sum up the forces + for (uint32_t i = 0; i < planetCount; ++i) + { + Vector3 const diff = (Vector3( x[i], y[i], z[i]) - Vector3(areaPos)) * invCommonScale; + float const r = diff.length(); + float const factor = 1000000.f; + Vector3 const accel = diff * factor / (r * r * r); + + rForce += accel * mass; + } +} + +osp::Session setup_newton_force_grav_nbody( + TopTaskBuilder& rBuilder, + ArrayView topData, + Session const& newton, + Session const& nwtFactors, + Session const& uniCore, + Session const& uniPlanets, + Session const& uniScnFrame) +{ + using namespace osp::universe; + using UserData_t = ACtxNwtWorld::ForceFactorFunc::UserData_t; + using CoSpaceIdVec_t = std::vector; + using Corrade::Containers::Array; + + OSP_DECLARE_GET_DATA_IDS(newton, TESTAPP_DATA_NEWTON); + OSP_DECLARE_GET_DATA_IDS(nwtFactors, TESTAPP_DATA_NEWTON_FORCES); + + auto &rNwt = top_get(topData, idNwt); + + OSP_DECLARE_GET_DATA_IDS(uniCore, TESTAPP_DATA_UNI_CORE); + auto &rUniverse = top_get< Universe >(topData, idUniverse); + + OSP_DECLARE_GET_DATA_IDS(uniPlanets, TESTAPP_DATA_UNI_PLANETS); + auto& rPlanetMainSpace = top_get(topData, idPlanetMainSpace); + + OSP_DECLARE_GET_DATA_IDS(uniScnFrame, TESTAPP_DATA_UNI_SCENEFRAME); + auto& rScnFrame = top_get(topData, idScnFrame); + + Session out; + OSP_DECLARE_CREATE_DATA_IDS(out, topData, TESTAPP_DATA_NEWTON_ACCEL); + + ACtxNwtWorld::ForceFactorFunc const factor + { + .m_func = nbody_grav_force, + .m_userData = {&rUniverse, &rPlanetMainSpace, &rScnFrame} + }; + + // Register force + + std::size_t const index = rNwt.m_factors.size(); + rNwt.m_factors.emplace_back(factor); + + auto factorBits = lgrn::bit_view(top_get(topData, idNwtFactors)); + factorBits.set(index); + + return out; +} // setup_newton_force_accel + +Session setup_newton_origin_translate( + TopTaskBuilder& rBuilder, + ArrayView topData, + Session const& scene, + Session const& sceneFrame, + Session const& newton, + Session const& physics) +{ + using namespace osp::universe; + + OSP_DECLARE_GET_DATA_IDS(scene, TESTAPP_DATA_SCENE); + auto const tgScn = scene.get_pipelines(); + + OSP_DECLARE_GET_DATA_IDS(newton, TESTAPP_DATA_NEWTON); + auto const tgNwt = newton.get_pipelines(); + + OSP_DECLARE_GET_DATA_IDS(sceneFrame, TESTAPP_DATA_UNI_SCENEFRAME); + auto const tgScnFrame = sceneFrame.get_pipelines(); + + OSP_DECLARE_GET_DATA_IDS(physics, TESTAPP_DATA_PHYSICS); + auto const tgPhys = physics.get_pipelines(); + + Session out; + OSP_DECLARE_CREATE_DATA_IDS(out, topData, TESTAPP_DATA_NEWTON_ORIGIN_TRANSLATE); + + rBuilder.task() + .name ("Update scene origin if body strays too far") + .run_on ({tgScn.update(Run)}) + .sync_with ({tgNwt.nwtBody(Prev), tgScnFrame.sceneFrame(Modify)}) + .push_to (out.m_tasks) + .args ({idNwt, idScnFrame, idPhys}) + .func([] (ACtxNwtWorld& rNwt, SceneFrame& rScnFrame, ACtxPhysics& rPhysics) noexcept + { + for (auto& pBody : rNwt.m_bodyPtrs) + { + Vector3 pos = Vector3(0.0f); + NewtonBodyGetPosition(pBody.get(), pos.data()); + + float const maxDist = 512.0f; + Vector3 const translate = sign(pos) * floor(abs(pos) / maxDist) * maxDist; + + if ( ! translate.isZero()) + { + Vector3 const rotated = Quaternion(rScnFrame.m_rotation).transformVector(translate); + rScnFrame.m_position += Vector3g(math::mul_2pow(rotated, rScnFrame.m_precision)); + + rPhysics.m_originTranslate -= translate; + // printf("Recentering local scene\n, new origin: %ld, %ld, %ld\n", rScnFrame.m_position.x(), rScnFrame.m_position.y(), rScnFrame.m_position.z()); + + SysNewton::update_translate(rPhysics, rNwt); + } + + // TODO: do the same but for velocity + + break; // We're just taking the first newton body because i am very lazy + } + }); + + return out; +} // setup_newton_scene_follow + } // namespace testapp::scenes diff --git a/src/testapp/sessions/newton.h b/src/testapp/sessions/newton.h index 18a659ee..3ced256b 100644 --- a/src/testapp/sessions/newton.h +++ b/src/testapp/sessions/newton.h @@ -69,6 +69,29 @@ osp::Session setup_newton_force_accel( osp::Session const& nwtFactors, osp::Vector3 accel); +/** + * @brief Setup a gravity force for every satellite in the universe which acts on the newton scene + */ +osp::Session setup_newton_force_grav_nbody( + osp::TopTaskBuilder& rBuilder, + osp::ArrayView topData, + osp::Session const& newton, + osp::Session const& nwtFactors, + osp::Session const& uniCore, + osp::Session const& uniPlanets, + osp::Session const& uniScnFrame); + +/** + * @brief Set the scene origin to follow a Newton Dynamics object + */ +osp::Session setup_newton_origin_translate( + osp::TopTaskBuilder& rBuilder, + osp::ArrayView topData, + osp::Session const& scene, + osp::Session const& sceneFrame, + osp::Session const& newton, + osp::Session const& physics); + /** * @brief Support for Shape Spawner physics using Newton Dynamics */ diff --git a/src/testapp/sessions/universe.cpp b/src/testapp/sessions/universe.cpp index 2a60c26c..f9e7e0d4 100644 --- a/src/testapp/sessions/universe.cpp +++ b/src/testapp/sessions/universe.cpp @@ -88,6 +88,110 @@ Session setup_uni_sceneframe( + + +Session setup_uni_landerplanet( + TopTaskBuilder& rBuilder, + ArrayView topData, + Session const& uniCore, + Session const& uniScnFrame) +{ + using CoSpaceIdVec_t = std::vector; + using Corrade::Containers::Array; + + OSP_DECLARE_GET_DATA_IDS(uniCore, TESTAPP_DATA_UNI_CORE); + OSP_DECLARE_GET_DATA_IDS(uniScnFrame, TESTAPP_DATA_UNI_SCENEFRAME); + + auto const tgUCore = uniCore .get_pipelines(); + auto const tgUSFrm = uniScnFrame.get_pipelines(); + + auto &rUniverse = top_get< Universe >(topData, idUniverse); + + constexpr int precision = 10; + constexpr int planetCount = 2; + + // Create coordinate spaces + CoSpaceId const mainSpace = rUniverse.m_coordIds.create(); + std::vector satSurfaceSpaces(planetCount); + rUniverse.m_coordIds.create(satSurfaceSpaces.begin(), satSurfaceSpaces.end()); + rUniverse.m_coordCommon.resize(rUniverse.m_coordIds.capacity()); + + CoSpaceCommon &rMainSpaceCommon = rUniverse.m_coordCommon[mainSpace]; + rMainSpaceCommon.m_satCount = planetCount; + rMainSpaceCommon.m_satCapacity = planetCount; + + // Associate each planet satellite with their surface coordinate space + for (SatId satId = 0; satId < planetCount; ++satId) + { + CoSpaceId const surfaceSpaceId = satSurfaceSpaces[satId]; + CoSpaceCommon &rCommon = rUniverse.m_coordCommon[surfaceSpaceId]; + rCommon.m_parent = mainSpace; + rCommon.m_parentSat = satId; + } + + // Coordinate space data is a single allocation partitioned to hold positions, velocities, and + // rotations. + // TODO: Alignment is needed for SIMD (not yet implemented). see Corrade alignedAlloc + + std::size_t bytesUsed = 0; + + // Positions and velocities are arranged as XXXX... YYYY... ZZZZ... + partition(bytesUsed, planetCount, rMainSpaceCommon.m_satPositions[0]); + partition(bytesUsed, planetCount, rMainSpaceCommon.m_satPositions[1]); + partition(bytesUsed, planetCount, rMainSpaceCommon.m_satPositions[2]); + partition(bytesUsed, planetCount, rMainSpaceCommon.m_satVelocities[0]); + partition(bytesUsed, planetCount, rMainSpaceCommon.m_satVelocities[1]); + partition(bytesUsed, planetCount, rMainSpaceCommon.m_satVelocities[2]); + + // Rotations use XYZWXYZWXYZWXYZW... + partition(bytesUsed, planetCount, rMainSpaceCommon.m_satRotations[0], + rMainSpaceCommon.m_satRotations[1], + rMainSpaceCommon.m_satRotations[2], + rMainSpaceCommon.m_satRotations[3]); + + // Allocate data for all planets + rMainSpaceCommon.m_data = Array{Corrade::NoInit, bytesUsed}; + + // Create easily accessible array views for each component + auto const [x, y, z] = sat_views(rMainSpaceCommon.m_satPositions, rMainSpaceCommon.m_data, planetCount); + auto const [vx, vy, vz] = sat_views(rMainSpaceCommon.m_satVelocities, rMainSpaceCommon.m_data, planetCount); + auto const [qx, qy, qz, qw] = sat_views(rMainSpaceCommon.m_satRotations, rMainSpaceCommon.m_data, planetCount); + + for (std::size_t i = 0; i < planetCount; ++i) + { + // Place the planet at the origin + x[i] = 0; + y[i] = 0; + z[i] = 0; + vx[i] = 0; + vy[i] = 0; + vz[i] = 0; + + // No rotation + qx[i] = 0.0; + qy[i] = 0.0; + qz[i] = 0.0; + qw[i] = 1.0; + } + + x[1] = 1000000; + y[1] = 1000000; + z[1] = 0; + + auto &rScnFrame = top_get(topData, idScnFrame); + rScnFrame.m_parent = mainSpace; + rScnFrame.m_position = math::mul_2pow({400, 400, 400}, precision); + + Session out; + OSP_DECLARE_CREATE_DATA_IDS(out, topData, TESTAPP_DATA_UNI_PLANETS); + + top_emplace< CoSpaceId > (topData, idPlanetMainSpace, mainSpace); + top_emplace< float > (topData, tgUniDeltaTimeIn, 1.0f / 60.0f); + top_emplace< CoSpaceIdVec_t > (topData, idSatSurfaceSpaces, std::move(satSurfaceSpaces)); + + return out; +} // setup_uni_landerplanet + Session setup_uni_testplanets( TopTaskBuilder& rBuilder, ArrayView topData, @@ -307,8 +411,6 @@ Session setup_uni_testplanets( } // setup_uni_testplanets - - struct PlanetDraw { DrawEntVec_t drawEnts; @@ -318,6 +420,176 @@ struct PlanetDraw MaterialId matAxis; }; +Session setup_landerplanet_draw( + TopTaskBuilder& rBuilder, + ArrayView const topData, + Session const& windowApp, + Session const& sceneRenderer, + Session const& cameraCtrl, + Session const& commonScene, + Session const& uniCore, + Session const& uniScnFrame, + Session const& uniLanderPlanet, + MaterialId const matPlanets, + MaterialId const matAxis +) { + OSP_DECLARE_GET_DATA_IDS(commonScene, TESTAPP_DATA_COMMON_SCENE); + OSP_DECLARE_GET_DATA_IDS(sceneRenderer, TESTAPP_DATA_SCENE_RENDERER); + OSP_DECLARE_GET_DATA_IDS(cameraCtrl, TESTAPP_DATA_CAMERA_CTRL); + OSP_DECLARE_GET_DATA_IDS(uniCore, TESTAPP_DATA_UNI_CORE); + OSP_DECLARE_GET_DATA_IDS(uniScnFrame, TESTAPP_DATA_UNI_SCENEFRAME); + OSP_DECLARE_GET_DATA_IDS(uniLanderPlanet, TESTAPP_DATA_UNI_PLANETS); + + auto const tgWin = windowApp .get_pipelines(); + auto const tgScnRdr = sceneRenderer .get_pipelines(); + auto const tgCmCt = cameraCtrl .get_pipelines(); + auto const tgUSFrm = uniScnFrame .get_pipelines(); + + Session out; + + auto const [idPlanetDraw] = out.acquire_data<1>(topData); + auto &rPlanetDraw = top_emplace(topData, idPlanetDraw); + rPlanetDraw.matPlanets = matPlanets; + rPlanetDraw.matAxis = matAxis; + + rBuilder.task() + .name ("Resync test planets, create DrawEnts") + .run_on ({tgWin.resync(Run)}) + .sync_with ({tgScnRdr.drawEntResized(ModifyOrSignal)}) + .push_to (out.m_tasks) + .args ({ idScnRender, idPlanetDraw, idUniverse, idPlanetMainSpace}) + .func([] (ACtxSceneRender& rScnRender, PlanetDraw& rPlanetDraw, Universe& rUniverse, CoSpaceId const planetMainSpace) noexcept + { + CoSpaceCommon &rMainSpace = rUniverse.m_coordCommon[planetMainSpace]; + + rPlanetDraw.drawEnts.resize(rMainSpace.m_satCount, lgrn::id_null()); + + rScnRender.m_drawIds.create(rPlanetDraw.drawEnts .begin(), rPlanetDraw.drawEnts .end()); + rScnRender.m_drawIds.create(rPlanetDraw.axis .begin(), rPlanetDraw.axis .end()); + rPlanetDraw.attractor = rScnRender.m_drawIds.create(); + }); + + rBuilder.task() + .name ("Resync test planets, add mesh and material") + .run_on ({tgWin.resync(Run)}) + .sync_with ({tgScnRdr.drawEntResized(Done), tgScnRdr.materialDirty(Modify_), tgScnRdr.entMeshDirty(Modify_)}) + .push_to (out.m_tasks) + .args ({ idDrawing, idScnRender, idNMesh, idPlanetDraw, idUniverse, idPlanetMainSpace}) + .func([] (ACtxDrawing& rDrawing, ACtxSceneRender& rScnRender, NamedMeshes& rNMesh, PlanetDraw& rPlanetDraw, Universe& rUniverse, CoSpaceId const planetMainSpace) noexcept + { + CoSpaceCommon &rMainSpace = rUniverse.m_coordCommon[planetMainSpace]; + + Material &rMatPlanet = rScnRender.m_materials[rPlanetDraw.matPlanets]; + Material &rMatAxis = rScnRender.m_materials[rPlanetDraw.matAxis]; + + MeshId const sphereMeshId = rNMesh.m_shapeToMesh.at(EShape::Sphere); + MeshId const cubeMeshId = rNMesh.m_shapeToMesh.at(EShape::Box); + + for (std::size_t i = 0; i < rMainSpace.m_satCount; ++i) + { + DrawEnt const drawEnt = rPlanetDraw.drawEnts[i]; + + rScnRender.m_mesh[drawEnt] = rDrawing.m_meshRefCounts.ref_add(sphereMeshId); + rScnRender.m_meshDirty.push_back(drawEnt); + rScnRender.m_visible.set(std::size_t(drawEnt)); + rScnRender.m_opaque.set(std::size_t(drawEnt)); + rMatPlanet.m_ents.set(std::size_t(drawEnt)); + rMatPlanet.m_dirty.push_back(drawEnt); + } + + for (DrawEnt const drawEnt : rPlanetDraw.axis) + { + rScnRender.m_mesh[drawEnt] = rDrawing.m_meshRefCounts.ref_add(cubeMeshId); + rScnRender.m_meshDirty.push_back(drawEnt); + rScnRender.m_visible.set(std::size_t(drawEnt)); + rScnRender.m_opaque.set(std::size_t(drawEnt)); + rMatAxis.m_ents.set(std::size_t(drawEnt)); + rMatAxis.m_dirty.push_back(drawEnt); + } + + rScnRender.m_color[rPlanetDraw.axis[0]] = {1.0f, 0.0f, 0.0f, 1.0f}; + rScnRender.m_color[rPlanetDraw.axis[1]] = {0.0f, 1.0f, 0.0f, 1.0f}; + rScnRender.m_color[rPlanetDraw.axis[2]] = {0.0f, 0.0f, 1.0f, 1.0f}; + }); + + rBuilder.task() + .name ("Reposition test planet DrawEnts") + .run_on ({tgScnRdr.render(Run)}) + .sync_with ({tgScnRdr.drawTransforms(Modify_), tgScnRdr.drawEntResized(Done), tgCmCt.camCtrl(Ready), tgUSFrm.sceneFrame(Modify)}) + .push_to (out.m_tasks) + .args ({ idDrawing, idScnRender, idPlanetDraw, idUniverse, idScnFrame, idPlanetMainSpace}) + .func([] (ACtxDrawing& rDrawing, ACtxSceneRender& rScnRender, PlanetDraw& rPlanetDraw, Universe& rUniverse, SceneFrame const& rScnFrame, CoSpaceId const planetMainSpace) noexcept + { + + CoSpaceCommon &rMainSpace = rUniverse.m_coordCommon[planetMainSpace]; + auto const [x, y, z] = sat_views(rMainSpace.m_satPositions, rMainSpace.m_data, rMainSpace.m_satCount); + auto const [qx, qy, qz, qw] = sat_views(rMainSpace.m_satRotations, rMainSpace.m_data, rMainSpace.m_satCount); + + // Calculate transform from universe to area/local-space for rendering. + // This can be generalized by finding a common ancestor within the tree + // of coordinate spaces. Since there's only two possibilities, an if + // statement works. + CoordTransformer mainToArea; + if (rScnFrame.m_parent == planetMainSpace) + { + mainToArea = coord_parent_to_child(rMainSpace, rScnFrame); + } + else + { + CoSpaceId const landedId = rScnFrame.m_parent; + CoSpaceCommon &rLanded = rUniverse.m_coordCommon[landedId]; + + CoSpaceTransform const landedTf = coord_get_transform(rLanded, rLanded, x, y, z, qx, qy, qz, qw); + CoordTransformer const mainToLanded = coord_parent_to_child(rMainSpace, landedTf); + CoordTransformer const landedToArea = coord_parent_to_child(landedTf, rScnFrame); + + mainToArea = coord_composite(landedToArea, mainToLanded); + } + Quaternion const mainToAreaRot{mainToArea.rotation()}; + + float const scale = math::mul_2pow(1.0f, -rMainSpace.m_precision); + + Vector3 const attractorPos = Vector3(mainToArea.transform_position({0, 0, 0})) * scale; + + // Attractor + rScnRender.m_drawTransform[rPlanetDraw.attractor] + = Matrix4::translation(attractorPos) + * Matrix4{mainToAreaRot.toMatrix()} + * Matrix4::scaling({500, 500, 500}); + + rScnRender.m_drawTransform[rPlanetDraw.axis[0]] + = Matrix4::translation(attractorPos) + * Matrix4{mainToAreaRot.toMatrix()} + * Matrix4::scaling({500000, 10, 10}); + rScnRender.m_drawTransform[rPlanetDraw.axis[1]] + = Matrix4::translation(attractorPos) + * Matrix4{mainToAreaRot.toMatrix()} + * Matrix4::scaling({10, 500000, 10}); + rScnRender.m_drawTransform[rPlanetDraw.axis[2]] + = Matrix4::translation(attractorPos) + * Matrix4{mainToAreaRot.toMatrix()} + * Matrix4::scaling({10, 10, 500000}); + + for (std::size_t i = 0; i < rMainSpace.m_satCount; ++i) + { + Vector3g const relative = mainToArea.transform_position({x[i], y[i], z[i]}); + Vector3 const relativeMeters = Vector3(relative) * scale; + + Quaterniond const rot{{qx[i], qy[i], qz[i]}, qw[i]}; + + DrawEnt const drawEnt = rPlanetDraw.drawEnts[i]; + + rScnRender.m_drawTransform[drawEnt] + = Matrix4::translation(relativeMeters) + * Matrix4::scaling({200, 200, 200}) + * Matrix4{(mainToAreaRot * Quaternion{rot}).toMatrix()}; + } + + }); + + return out; +} // setup_landerplanet_draw + Session setup_testplanets_draw( TopTaskBuilder& rBuilder, ArrayView const topData, @@ -531,4 +803,4 @@ Session setup_testplanets_draw( } // setup_testplanets_draw -} // namespace testapp::scenes +} // namespace testapp::scenes \ No newline at end of file diff --git a/src/testapp/sessions/universe.h b/src/testapp/sessions/universe.h index 94039d16..953969a0 100644 --- a/src/testapp/sessions/universe.h +++ b/src/testapp/sessions/universe.h @@ -46,7 +46,6 @@ osp::Session setup_uni_sceneframe( osp::TopTaskBuilder& rBuilder, osp::ArrayView topData, osp::Session const& uniCore); - /** * @brief Unrealistic planets test, allows SceneFrame to move around and get captured into planets */ @@ -72,4 +71,30 @@ osp::Session setup_testplanets_draw( osp::Session const& uniTestPlanets, osp::draw::MaterialId const matPlanets, osp::draw::MaterialId const matAxis); -} + + +/** + * @brief A single planet setup for landing on + */ +osp::Session setup_uni_landerplanet( + osp::TopTaskBuilder& rBuilder, + osp::ArrayView topData, + osp::Session const& uniCore, + osp::Session const& uniScnFrame); + +/** + * @brief Draw universe, specifically designed for setup_uni_landerplanet + */ +osp::Session setup_landerplanet_draw( + osp::TopTaskBuilder& rBuilder, + osp::ArrayView topData, + osp::Session const& windowApp, + osp::Session const& sceneRenderer, + osp::Session const& cameraCtrl, + osp::Session const& commonScene, + osp::Session const& uniCore, + osp::Session const& uniScnFrame, + osp::Session const& uniTestPlanets, + osp::draw::MaterialId const matPlanets, + osp::draw::MaterialId const matAxis); +} \ No newline at end of file