From ca48642868bf18ba49c61e5ce0ff66e03efdf0c4 Mon Sep 17 00:00:00 2001 From: Levin Li Date: Tue, 8 Aug 2023 16:47:41 +0800 Subject: [PATCH] Use optionals in timeline --- src/celengine/body.cpp | 2 +- src/celengine/body.h | 3 +-- src/celengine/parseobject.cpp | 22 +++++++++------- src/celengine/parseobject.h | 2 ++ src/celengine/solarsys.cpp | 44 ++++++++++++++++++++----------- src/celengine/timeline.cpp | 16 +++++++---- src/celengine/timeline.h | 5 ++-- src/celengine/timelinephase.cpp | 10 +++---- src/celengine/timelinephase.h | 32 +++++++++++++++------- src/celscript/lua/celx_object.cpp | 7 ++--- src/celscript/lua/celx_phase.cpp | 12 +++------ 11 files changed, 95 insertions(+), 60 deletions(-) diff --git a/src/celengine/body.cpp b/src/celengine/body.cpp index 51192afff98..e5dfec8d360 100644 --- a/src/celengine/body.cpp +++ b/src/celengine/body.cpp @@ -813,7 +813,7 @@ bool Body::extant(double t) const } -void Body::getLifespan(double& begin, double& end) const +void Body::getLifespan(std::optional& begin, std::optional& end) const { begin = timeline->startTime(); end = timeline->endTime(); diff --git a/src/celengine/body.h b/src/celengine/body.h index 27686ff9a57..dd3925ba613 100644 --- a/src/celengine/body.h +++ b/src/celengine/body.h @@ -308,8 +308,7 @@ class Body Eigen::Vector3d eclipticToPlanetocentric(const Eigen::Vector3d& ecl, double tdb) const; bool extant(double) const; - void setLifespan(double, double); - void getLifespan(double&, double&) const; + void getLifespan(std::optional&, std::optional&) const; Surface* getAlternateSurface(const std::string&) const; void addAlternateSurface(const std::string&, Surface*); diff --git a/src/celengine/parseobject.cpp b/src/celengine/parseobject.cpp index f23f8453f10..ce5f22f4cdb 100644 --- a/src/celengine/parseobject.cpp +++ b/src/celengine/parseobject.cpp @@ -87,27 +87,31 @@ GetDefaultUnits(bool usePlanetUnits, double& distanceScale) bool ParseDate(const Hash* hash, const string& name, double& jd) { - // Check first for a number value representing a Julian date - if (auto jdVal = hash->getNumber(name); jdVal.has_value()) + if (auto value = ParseDate(hash, name); value.has_value()) { - jd = *jdVal; + jd = value.value(); return true; } + return false; +} + +std::optional +ParseDate(const Hash* hash, const string& name) +{ + // Check first for a number value representing a Julian date + if (auto jdVal = hash->getNumber(name); jdVal.has_value()) + return *jdVal; if (const std::string* dateString = hash->getString(name); dateString != nullptr) { astro::Date date(1, 1, 1); if (astro::parseDate(*dateString, date)) - { - jd = (double) date; - return true; - } + return static_cast(date); } - return false; + return std::nullopt; } - /*! * Create a new Keplerian orbit from an ssc property table: * diff --git a/src/celengine/parseobject.h b/src/celengine/parseobject.h index 56fa86f2636..05bb5ed9ad1 100644 --- a/src/celengine/parseobject.h +++ b/src/celengine/parseobject.h @@ -14,6 +14,7 @@ #include #include +#include #include #include #include @@ -34,6 +35,7 @@ enum class DataDisposition bool ParseDate(const Hash* hash, const std::string& name, double& jd); +std::optional ParseDate(const Hash* hash, const std::string& name); celestia::ephem::Orbit* CreateOrbit(const Selection& centralObject, const Hash* planetData, diff --git a/src/celengine/solarsys.cpp b/src/celengine/solarsys.cpp index 49283ae9761..6356f51d3d2 100644 --- a/src/celengine/solarsys.cpp +++ b/src/celengine/solarsys.cpp @@ -272,24 +272,32 @@ TimelinePhase::SharedConstPtr CreateTimelinePhase(Body* body, const ReferenceFrame::SharedConstPtr& defaultBodyFrame, bool isFirstPhase, bool isLastPhase, - double previousPhaseEnd) + std::optional previousPhaseEnd) { - double beginning = previousPhaseEnd; - double ending = std::numeric_limits::infinity(); + std::optional beginning = previousPhaseEnd; + std::optional ending = std::nullopt; // Beginning is optional for the first phase of a timeline, and not // allowed for the other phases, where beginning is always the ending // of the previous phase. - bool hasBeginning = ParseDate(phaseData, "Beginning", beginning); - if (!isFirstPhase && hasBeginning) + auto newBeginning = ParseDate(phaseData, "Beginning"); + if (newBeginning.has_value()) { - GetLogger()->error("Error: Beginning can only be specified for initial phase of timeline.\n"); - return nullptr; + beginning = newBeginning.value(); + if (!isFirstPhase) + { + GetLogger()->error("Error: Beginning can only be specified for initial phase of timeline.\n"); + return nullptr; + } } // Ending is required for all phases except for the final one. - bool hasEnding = ParseDate(phaseData, "Ending", ending); - if (!isLastPhase && !hasEnding) + auto newEnding = ParseDate(phaseData, "Ending"); + if (newEnding.has_value()) + { + ending = newEnding.value(); + } + else if (!isLastPhase) { GetLogger()->error("Error: Ending is required for all timeline phases other than the final one.\n"); return nullptr; @@ -375,7 +383,7 @@ Timeline* CreateTimelineFromArray(Body* body, const ReferenceFrame::SharedConstPtr& defaultBodyFrame) { auto timeline = std::make_unique(); - double previousEnding = -std::numeric_limits::infinity(); + std::optional previousEnding = std::nullopt; if (timelineArray->empty()) { @@ -487,8 +495,8 @@ bool CreateTimeline(Body* body, ReferenceFrame::SharedConstPtr bodyFrame; celestia::ephem::Orbit* orbit = nullptr; celestia::ephem::RotationModel* rotationModel = nullptr; - double beginning = -std::numeric_limits::infinity(); - double ending = std::numeric_limits::infinity(); + std::optional beginning = std::nullopt; + std::optional ending = std::nullopt; // If any new timeline values are specified, we need to overrideOldTimeline will // be set to true. @@ -601,10 +609,16 @@ bool CreateTimeline(Body* body, rotationModel = CreateDefaultRotationModel(syncRotationPeriod); } - if (ParseDate(planetData, "Beginning", beginning)) + if (auto value = ParseDate(planetData, "Beginning"); value.has_value()) + { + beginning = value.value(); overrideOldTimeline = true; - if (ParseDate(planetData, "Ending", ending)) + } + if (auto value = ParseDate(planetData, "Ending"); value.has_value()) + { + ending = value.value(); overrideOldTimeline = true; + } // Something went wrong if the disposition isn't modify and no timeline // is to be created. @@ -612,7 +626,7 @@ bool CreateTimeline(Body* body, if (overrideOldTimeline) { - if (beginning >= ending) + if (beginning.has_value() && ending.has_value() && beginning.value() >= ending.value()) { GetLogger()->error("Beginning time must be before Ending time.\n"); delete rotationModel; diff --git a/src/celengine/timeline.cpp b/src/celengine/timeline.cpp index 8dee834babc..aa422e63791 100644 --- a/src/celengine/timeline.cpp +++ b/src/celengine/timeline.cpp @@ -40,7 +40,8 @@ Timeline::appendPhase(TimelinePhase::SharedConstPtr &phase) // no gaps and no overlaps. if (!phases.empty()) { - if (phase->startTime() != phases.back()->endTime()) + auto startTime = phase->startTime(); + if (!startTime.has_value() || startTime.value() != phases.back()->endTime()) return false; } @@ -64,7 +65,8 @@ Timeline::findPhase(double t) const { for (const auto& phase : phases) { - if (t < phase->endTime()) + auto endTime = phase->endTime(); + if (!endTime.has_value() || t < endTime.value()) return phase; } @@ -92,14 +94,14 @@ Timeline::phaseCount() const } -double +std::optional Timeline::startTime() const { return phases.front()->startTime(); } -double +std::optional Timeline::endTime() const { return phases.back()->endTime(); @@ -114,7 +116,11 @@ Timeline::endTime() const bool Timeline::includes(double t) const { - return phases.front()->startTime() <= t && t <= phases.back()->endTime(); + if (auto startTime = phases.front()->startTime(); startTime.has_value() && t < startTime.value()) + return false; + if (auto endTime = phases.back()->endTime(); endTime.has_value() && t > endTime.value()) + return false; + return true; } diff --git a/src/celengine/timeline.h b/src/celengine/timeline.h index 6fb32dc339a..261aaf22a4c 100644 --- a/src/celengine/timeline.h +++ b/src/celengine/timeline.h @@ -13,6 +13,7 @@ #pragma once #include +#include #include #include "timelinephase.h" @@ -27,8 +28,8 @@ class Timeline const TimelinePhase::SharedConstPtr& getPhase(unsigned int n) const; unsigned int phaseCount() const; - double startTime() const; - double endTime() const; + std::optional startTime() const; + std::optional endTime() const; bool includes(double t) const; void markChanged(); diff --git a/src/celengine/timelinephase.cpp b/src/celengine/timelinephase.cpp index c6c3d84c560..d456e434e2f 100644 --- a/src/celengine/timelinephase.cpp +++ b/src/celengine/timelinephase.cpp @@ -23,8 +23,8 @@ using namespace std; TimelinePhase::TimelinePhase(Body* _body, - double _startTime, - double _endTime, + std::optional _startTime, + std::optional _endTime, const ReferenceFrame::SharedConstPtr& _orbitFrame, celestia::ephem::Orbit* _orbit, const ReferenceFrame::SharedConstPtr& _bodyFrame, @@ -47,15 +47,15 @@ TimelinePhase::TimelinePhase(Body* _body, TimelinePhase::SharedConstPtr TimelinePhase::CreateTimelinePhase(Universe& universe, Body* body, - double startTime, - double endTime, + std::optional startTime, + std::optional endTime, const ReferenceFrame::SharedConstPtr& orbitFrame, celestia::ephem::Orbit& orbit, const ReferenceFrame::SharedConstPtr& bodyFrame, celestia::ephem::RotationModel& rotationModel) { // Validate the time range. - if (endTime <= startTime) + if (endTime.has_value() && startTime.has_value() && endTime.value() <= startTime.value()) return nullptr; // Get the frame tree to add the new phase to. Verify that the reference frame diff --git a/src/celengine/timelinephase.h b/src/celengine/timelinephase.h index 0b375d1690f..562f7c6509f 100644 --- a/src/celengine/timelinephase.h +++ b/src/celengine/timelinephase.h @@ -13,6 +13,7 @@ #pragma once #include +#include #include "frame.h" class FrameTree; @@ -36,12 +37,12 @@ class TimelinePhase return m_body; } - double startTime() const + std::optional startTime() const { return m_startTime; } - double endTime() const + std::optional endTime() const { return m_endTime; } @@ -79,13 +80,26 @@ class TimelinePhase */ bool includes(double t) const { - return m_startTime <= t && t < m_endTime; + if (m_startTime.has_value() && t < m_startTime.value()) + return false; + if (m_endTime.has_value() && t >= m_endTime.value()) + return false; + return true; + } + + double clamp(double t) const + { + if (m_startTime.has_value() && t < m_startTime.value()) + return m_startTime.value(); + if (m_endTime.has_value() && t > m_endTime.value()) + return m_endTime.value(); + return t; } static TimelinePhase::SharedConstPtr CreateTimelinePhase(Universe& universe, Body* body, - double startTime, - double endTime, + std::optional startTime, + std::optional endTime, const ReferenceFrame::SharedConstPtr& orbitFrame, celestia::ephem::Orbit& orbit, const ReferenceFrame::SharedConstPtr& bodyFrame, @@ -94,8 +108,8 @@ class TimelinePhase ~TimelinePhase() = default; TimelinePhase(Body* _body, - double _startTime, - double _endTime, + std::optional _startTime, + std::optional _endTime, const ReferenceFrame::SharedConstPtr& _orbitFrame, celestia::ephem::Orbit* _orbit, const ReferenceFrame::SharedConstPtr& _bodyFrame, @@ -108,8 +122,8 @@ class TimelinePhase private: Body* m_body; - double m_startTime; - double m_endTime; + std::optional m_startTime; + std::optional m_endTime; ReferenceFrame::SharedConstPtr m_orbitFrame; celestia::ephem::Orbit* m_orbit; diff --git a/src/celscript/lua/celx_object.cpp b/src/celscript/lua/celx_object.cpp index 40f057bdf89..f7ece01b522 100644 --- a/src/celscript/lua/celx_object.cpp +++ b/src/celscript/lua/celx_object.cpp @@ -670,10 +670,11 @@ static int object_getinfo(lua_State* l) float eqRadius = max(semiAxes.x(), semiAxes.z()); celx.setTable("oblateness", (eqRadius - polarRadius) / eqRadius); - double lifespanStart, lifespanEnd; + std::optional lifespanStart; + std::optional lifespanEnd; body->getLifespan(lifespanStart, lifespanEnd); - celx.setTable("lifespanStart", (lua_Number)lifespanStart); - celx.setTable("lifespanEnd", (lua_Number)lifespanEnd); + celx.setTable("lifespanStart", static_cast(lifespanStart.value_or(-std::numeric_limits::infinity()))); + celx.setTable("lifespanEnd", static_cast(lifespanEnd.value_or(std::numeric_limits::infinity()))); // TODO: atmosphere, surfaces ? PlanetarySystem* system = body->getSystem(); diff --git a/src/celscript/lua/celx_phase.cpp b/src/celscript/lua/celx_phase.cpp index 46254503cdf..7feb861ec54 100644 --- a/src/celscript/lua/celx_phase.cpp +++ b/src/celscript/lua/celx_phase.cpp @@ -76,7 +76,7 @@ static int phase_timespan(lua_State* l) celx.checkArgs(1, 1, "No arguments allowed for to phase:timespan"); auto phase = this_phase(l); - celx.push(phase->startTime(), phase->endTime()); + celx.push(phase->startTime().value_or(-std::numeric_limits::infinity()), phase->endTime().value_or(std::numeric_limits::infinity())); //lua_pushnumber(l, phase->startTime()); //lua_pushnumber(l, phase->endTime()); @@ -135,10 +135,7 @@ static int phase_getposition(lua_State* l) auto phase = this_phase(l); double tdb = celx.safeGetNumber(2, WrongType, "Argument to phase:getposition() must be number", 0.0); - if (tdb < phase->startTime()) - tdb = phase->startTime(); - else if (tdb > phase->endTime()) - tdb = phase->endTime(); + tdb = phase->clamp(tdb); celx.newPosition(UniversalCoord(phase->orbit()->positionAtTime(tdb) * astro::kilometersToMicroLightYears(1.0))); return 1; @@ -160,10 +157,7 @@ static int phase_getorientation(lua_State* l) auto phase = this_phase(l); double tdb = celx.safeGetNumber(2, WrongType, "Argument to phase:getorientation() must be number", 0.0); - if (tdb < phase->startTime()) - tdb = phase->startTime(); - else if (tdb > phase->endTime()) - tdb = phase->endTime(); + tdb = phase->clamp(tdb); celx.newRotation(phase->rotationModel()->orientationAtTime(tdb)); return 1;