summaryrefslogtreecommitdiffhomepage
path: root/spline/spline.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'spline/spline.cpp')
-rw-r--r--spline/spline.cpp183
1 files changed, 86 insertions, 97 deletions
diff --git a/spline/spline.cpp b/spline/spline.cpp
index f28cc98d..ef1e9c58 100644
--- a/spline/spline.cpp
+++ b/spline/spline.cpp
@@ -13,6 +13,7 @@
#include <cmath>
#include <memory>
#include <cinttypes>
+#include <utility>
#include <QObject>
#include <QMutexLocker>
@@ -25,16 +26,16 @@
constexpr int spline::value_count;
-spline::spline(qreal maxx, qreal maxy, const QString& name) :
+spline::spline(const QString& name, const QString& axis_name, Axis axis) :
s(nullptr),
data(value_count, -16),
_mutex(QMutex::Recursive),
- max_x(maxx),
- max_y(maxy),
+ ctx(std::make_shared<QObject>()),
+ axis(axis),
activep(false),
validp(false)
{
- set_bundle(options::make_bundle(name));
+ set_bundle(options::make_bundle(name), axis_name, axis);
}
spline::~spline()
@@ -44,11 +45,15 @@ spline::~spline()
if (connection)
{
QObject::disconnect(connection);
+ QObject::disconnect(conn_maxx);
+ QObject::disconnect(conn_maxy);
connection = QMetaObject::Connection();
+ conn_maxx = QMetaObject::Connection();
+ conn_maxy = QMetaObject::Connection();
}
}
-spline::spline() : spline(0, 0, "") {}
+spline::spline() : spline("", "", Axis(-1)) {}
void spline::set_tracking_active(bool value)
{
@@ -58,6 +63,7 @@ void spline::set_tracking_active(bool value)
bundle spline::get_bundle()
{
+ QMutexLocker l(&_mutex); // avoid logic errors due to changes in value<t> data
return s->b;
}
@@ -68,32 +74,6 @@ void spline::clear()
validp = false;
}
-void spline::set_max_input(qreal max_input)
-{
- QMutexLocker l(&_mutex);
- max_x = max_input;
- validp = false;
-}
-
-void spline::set_max_output(qreal max_output)
-{
- QMutexLocker l(&_mutex);
- max_y = max_output;
- validp = false;
-}
-
-qreal spline::max_input() const
-{
- QMutexLocker l(&_mutex);
- return max_x;
-}
-
-qreal spline::max_output() const
-{
- QMutexLocker l(&_mutex);
- return max_y;
-}
-
float spline::get_value(double x)
{
QMutexLocker foo(&_mutex);
@@ -113,9 +93,6 @@ float spline::get_value_no_save_internal(double x)
{
QMutexLocker foo(&_mutex);
- if (max_x > 0)
- x = std::fmin(max_x, x);
-
float q = float(x * bucket_size_coefficient(s->points));
int xi = (int)q;
float yi = get_value_internal(xi);
@@ -149,23 +126,20 @@ float spline::get_value_internal(int x)
const float sign = signum(x);
x = std::abs(x);
const float ret_ = data[std::min(unsigned(x), unsigned(value_count)-1u)];
- float ret = sign * std::fmax(0, ret_);
- if (max_y > 0)
- ret = fmin(max_y, ret);
- return ret;
+ return sign * clamp(ret_, 0, 1000);
}
void spline::add_lone_point()
{
points_t points;
- points.push_back(QPointF(max_x, max_y));
+ points.push_back(QPointF(s->opts.clamp_x_, s->opts.clamp_y_));
s->points = points;
}
-QPointF spline::ensure_in_bounds(const QList<QPointF>& points, double max_x, int i)
+QPointF spline::ensure_in_bounds(const QList<QPointF>& points, int i)
{
- const int sz = element_count(points, max_x);
+ const int sz = points.size();
if (i < 0 || sz == 0)
return QPointF(0, 0);
@@ -176,20 +150,16 @@ QPointF spline::ensure_in_bounds(const QList<QPointF>& points, double max_x, int
return points[sz - 1];
}
-int spline::element_count(const QList<QPointF>& points, double max_x)
+int spline::element_count(const QList<QPointF>& points, double max_input)
{
- if (!(max_x > 0))
- return points.size();
- else
+ const unsigned sz = points.size();
+ for (unsigned k = 0; k < sz; k++)
{
- const unsigned sz = points.size();
- for (unsigned i = 0; i < sz; i++)
- {
- if (!(points[i].x() <= max_x))
- return i;
- }
- return points.size();
+ const QPointF& pt = points[k];
+ if (max_input > 1e-4 && pt.x() - 1e-2 > max_input)
+ return k;
}
+ return sz;
}
bool spline::sort_fn(const QPointF& one, const QPointF& two)
@@ -200,13 +170,13 @@ bool spline::sort_fn(const QPointF& one, const QPointF& two)
void spline::update_interp_data()
{
points_t points = s->points;
-
ensure_valid(points);
+ const int sz = points.size();
- int sz = element_count(points, max_x);
+ const double maxx = max_input();
if (sz == 0)
- points.prepend(QPointF(max_x, max_y));
+ points.prepend(QPointF(maxx, max_output()));
std::stable_sort(points.begin(), points.begin() + sz, sort_fn);
@@ -218,7 +188,7 @@ void spline::update_interp_data()
if (sz < 2)
{
- if (points[0].x() - 1e-2 <= max_x)
+ if (points[0].x() - 1e-2 < maxx)
{
const double x = points[0].x();
const double y = points[0].y();
@@ -232,15 +202,15 @@ void spline::update_interp_data()
}
else
{
- if (points[0].x() > 1e-2 && points[0].x() <= max_x)
+ if (points[0].x() > 1e-2 && points[0].x() <= maxx)
points.push_front(QPointF(0, 0));
for (int i = 0; i < sz; i++)
{
- const QPointF p0 = ensure_in_bounds(points, max_x, i - 1);
- const QPointF p1 = ensure_in_bounds(points, max_x, i + 0);
- const QPointF p2 = ensure_in_bounds(points, max_x, i + 1);
- const QPointF p3 = ensure_in_bounds(points, max_x, i + 2);
+ const QPointF p0 = ensure_in_bounds(points, i - 1);
+ const QPointF p1 = ensure_in_bounds(points, i + 0);
+ const QPointF p2 = ensure_in_bounds(points, i + 1);
+ const QPointF p3 = ensure_in_bounds(points, i + 2);
const double p0_x = p0.x(), p1_x = p1.x(), p2_x = p2.x(), p3_x = p3.x();
const double p0_y = p0.y(), p1_y = p1.y(), p2_y = p2.y(), p3_y = p3.y();
@@ -293,7 +263,7 @@ void spline::remove_point(int i)
QMutexLocker foo(&_mutex);
points_t points = s->points;
- const int sz = element_count(points, max_x);
+ const int sz = element_count(points, max_input());
if (i >= 0 && i < sz)
{
@@ -325,7 +295,7 @@ void spline::move_point(int idx, QPointF pt)
points_t points = s->points;
- const int sz = element_count(points, max_x);
+ const int sz = element_count(points, max_input());
if (idx >= 0 && idx < sz)
{
@@ -346,7 +316,7 @@ QList<QPointF> spline::get_points() const
int spline::get_point_count() const
{
QMutexLocker foo(&_mutex);
- return element_count(s->points, max_x);
+ return element_count(s->points, s->opts.clamp_x_);
}
void spline::reload()
@@ -361,7 +331,18 @@ void spline::save()
s->b->save();
}
-void spline::set_bundle(bundle b)
+void spline::invalidate_settings()
+{
+ // we're holding the mutex to allow signal disconnection in spline dtor
+ // before this slot gets called for the next time
+
+ QMutexLocker l(&_mutex);
+ validp = false;
+
+ emit s->recomputed();
+}
+
+void spline::set_bundle(bundle b, const QString& axis_name, Axis axis)
{
QMutexLocker l(&_mutex);
@@ -369,34 +350,44 @@ void spline::set_bundle(bundle b)
// the sentinel settings/bundle objects don't need any further branching once created
if (!s || s->b != b)
{
- s = std::make_shared<settings>(b);
+ s = std::make_shared<settings>(b, axis_name, axis);
if (connection)
+ {
QObject::disconnect(connection);
+ QObject::disconnect(conn_maxx);
+ QObject::disconnect(conn_maxy);
+ }
if (b)
{
connection = QObject::connect(b.get(), &bundle_::changed,
- s.get(), [&]()
- {
- // we're holding the mutex to allow signal disconnection in spline dtor
- // before this slot gets called for the next time
-
- // spline isn't a QObject and the connection context is incorrect
-
- QMutexLocker l(&_mutex);
- validp = false;
+ s.get(), [&]() { invalidate_settings(); });
- emit s->recomputed();
- },
- Qt::QueuedConnection);
+ // this isn't strictly necessary for the spline but helps the widget
+ conn_maxx = QObject::connect(&s->opts.clamp_x_, base_value::signal_fun<int>(),
+ ctx.get(), [&](double) { invalidate_settings(); });
+ conn_maxy = QObject::connect(&s->opts.clamp_y_, base_value::signal_fun<int>(),
+ ctx.get(), [&](double) { invalidate_settings(); });
}
validp = false;
}
}
-void spline::ensure_valid(const QList<QPointF>& the_points)
+double spline::max_input() const
+{
+ QMutexLocker l(&_mutex);
+ return s ? s->opts.clamp_x_.to<double>() : 0;
+}
+
+double spline::max_output() const
+{
+ QMutexLocker l(&_mutex);
+ return s ? std::fabs(s->opts.clamp_y_.to<double>()) : 0;
+}
+
+void spline::ensure_valid(QList<QPointF>& the_points)
{
QMutexLocker foo(&_mutex);
@@ -411,6 +402,8 @@ void spline::ensure_valid(const QList<QPointF>& the_points)
QList<QPointF> ret_list;
ret_list.reserve(sz);
+ const double maxx = max_input(), maxy = max_output();
+
for (int i = 0; i < sz; i++)
{
QPointF& pt(list[i]);
@@ -421,29 +414,22 @@ void spline::ensure_valid(const QList<QPointF>& the_points)
const QPointF& pt2(list[j]);
const QPointF tmp(pt - pt2);
const double dist_sq = QPointF::dotProduct(tmp, tmp);
- const double overlap = max_x / 500.;
+ const double overlap = maxx / 500.;
if (dist_sq < overlap * overlap)
return true;
}
return false;
);
- const bool over_limit = progn(
- bool ret = false;
- if (pt.y() - 1e-2 > max_y)
- {
- pt.setY(max_y);
- ret = true;
- }
- return ret;
- );
-
- if (!overlap && !over_limit)
+ if (!overlap)
ret_list.push_back(pt);
}
if (ret_list != the_points)
+ {
s->points = ret_list;
+ the_points = std::move(ret_list);
+ }
last_input_value = QPointF(0, 0);
activep = false;
@@ -466,23 +452,26 @@ double spline::bucket_size_coefficient(const QList<QPointF>& points) const
{
static constexpr double eps = 1e-4;
- if (unlikely(max_x < eps))
+ const double maxx = max_input();
+
+ if (maxx < eps)
return 0;
// needed to fill the buckets up to the last control point.
// space between that point and max_x doesn't matter.
- const int sz = element_count(points, max_x);
- const double last_x = sz ? points[sz - 1].x() : max_x;
+ const int sz = element_count(points, maxx);
+ const double last_x = sz ? points[sz - 1].x() : maxx;
- return clamp((value_count-1) / clamp(last_x, eps, max_x), 0., (value_count-1));
+ return clamp((value_count-1) / clamp(last_x, eps, maxx), 0., (value_count-1));
}
namespace spline_detail {
-settings::settings(bundle b):
+settings::settings(bundle b, const QString& axis_name, Axis idx):
b(b ? b : make_bundle("")),
- points(b, "points", QList<QPointF>())
+ points(b, "points", {}),
+ opts(axis_name, idx)
{}
settings::~settings()