From fb434dc49272ce0aa6c0b1ca2d676fa82246c9c4 Mon Sep 17 00:00:00 2001 From: Lord-Grey Date: Sat, 30 Nov 2024 22:08:41 +0100 Subject: [PATCH] Provide a lowest permissible interval time to effects and limit the update rate of selected effects --- effects/candle.py | 3 +++ effects/gif.py | 4 ++++ effects/ledtest-seq.py | 3 +++ effects/ledtest.py | 3 +++ effects/plasma.py | 3 +++ effects/running_dots.py | 5 ++++- effects/shutdown.py | 3 +++ effects/sparks.py | 3 +++ effects/swirl.py | 5 ++++- effects/traces.py | 3 +++ effects/trails.py | 5 ++++- effects/x-mas.py | 5 ++++- include/effectengine/Effect.h | 2 ++ include/effectengine/EffectModule.h | 1 + libsrc/effectengine/Effect.cpp | 6 ++++++ libsrc/effectengine/EffectModule.cpp | 8 ++++++++ 16 files changed, 58 insertions(+), 4 deletions(-) diff --git a/effects/candle.py b/effects/candle.py index bbe77b35..edcc6b11 100644 --- a/effects/candle.py +++ b/effects/candle.py @@ -20,6 +20,9 @@ brightness = float(hyperion.args.get('brightness', 100))/100.0 sleepTime = float(hyperion.args.get('sleepTime', 0.14)) +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + candles = hyperion.args.get('candles', "all") ledlist = hyperion.args.get('ledlist', "1") diff --git a/effects/gif.py b/effects/gif.py index 936f19b9..81ffe72a 100644 --- a/effects/gif.py +++ b/effects/gif.py @@ -11,6 +11,10 @@ cropBottom = int(hyperion.args.get('cropBottom', 0)) grayscale = bool(hyperion.args.get('grayscale', False)) sleepTime = 1./framesPerSecond + +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + imageFrameList = [] if imageData: diff --git a/effects/ledtest-seq.py b/effects/ledtest-seq.py index 535c06bb..092ba1f1 100644 --- a/effects/ledtest-seq.py +++ b/effects/ledtest-seq.py @@ -4,6 +4,9 @@ import time # Get parameters sleepTime = float(hyperion.args.get('sleepTime', 0.5)) +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + def TestRgb( iteration ): switcher = { diff --git a/effects/ledtest.py b/effects/ledtest.py index bbdb7056..ad5aa556 100644 --- a/effects/ledtest.py +++ b/effects/ledtest.py @@ -11,6 +11,9 @@ sleepTime = float(hyperion.args.get('sleepTime', 0.5)) testleds = hyperion.args.get('testleds', "all") ledlist = hyperion.args.get('ledlist', "1") +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + testlist = () if (testleds == "list") and (type(ledlist) is str): for s in ledlist.split(','): diff --git a/effects/plasma.py b/effects/plasma.py index ffc5dba5..fd4e0be2 100644 --- a/effects/plasma.py +++ b/effects/plasma.py @@ -6,6 +6,9 @@ width = hyperion.imageWidth() height = hyperion.imageHeight() sleepTime = float(hyperion.args.get('sleepTime', 0.2)) +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + def mapto(x, in_min, in_max, out_min, out_max): return float((x - in_min) * (out_max - out_min) / (in_max - in_min) + out_min) diff --git a/effects/running_dots.py b/effects/running_dots.py index 2fb9569b..d5106593 100644 --- a/effects/running_dots.py +++ b/effects/running_dots.py @@ -4,8 +4,11 @@ import hyperion, time sleepTime = float(hyperion.args.get('speed', 1.5)) * 0.005 whiteLevel = int(hyperion.args.get('whiteLevel', 0)) lvl = int(hyperion.args.get('colorLevel', 220)) + +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) -# check value +# check value whiteLevel = min( whiteLevel, 254 ) lvl = min( lvl, 255 ) diff --git a/effects/shutdown.py b/effects/shutdown.py index 12209988..e71c6b0d 100644 --- a/effects/shutdown.py +++ b/effects/shutdown.py @@ -21,6 +21,9 @@ height = 10 imageData = bytearray(height * width * (0,0,0)) +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + # Start the write data loop if initialBlink: for i in range(6): diff --git a/effects/sparks.py b/effects/sparks.py index d7e35d77..c4344324 100644 --- a/effects/sparks.py +++ b/effects/sparks.py @@ -8,6 +8,9 @@ saturation = float(hyperion.args.get('saturation', 100))/100.0 color = list(hyperion.args.get('color', (255,255,255))) randomColor = bool(hyperion.args.get('random-color', False)) +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + # Check parameters rotationTime = max(0.1, rotationTime) diff --git a/effects/swirl.py b/effects/swirl.py index fa5d0ed4..66034e72 100644 --- a/effects/swirl.py +++ b/effects/swirl.py @@ -22,7 +22,10 @@ def getPoint(rand = True ,x = 0.5, y = 0.5): def getSTime(rt, steps = 360): rt = float(rt) sleepTime = max(0.1, rt) / steps - + + # Limit update rate + sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + # adapt sleeptime to hardware minStepTime= float(hyperion.latchTime)/1000.0 if minStepTime == 0: minStepTime = 0.001 diff --git a/effects/traces.py b/effects/traces.py index 37338b14..aaeda242 100644 --- a/effects/traces.py +++ b/effects/traces.py @@ -10,6 +10,9 @@ minStepTime = float(hyperion.latchTime)/1000.0 if minStepTime == 0: minStepTime = 0.001 factor = 1 if sleepTime > minStepTime else int(math.ceil(minStepTime/sleepTime)) +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) + runners = [ { "i":0, "pos":0, "c":0, "step":9, "lvl":255}, { "i":1, "pos":0, "c":0, "step":8, "lvl":255}, diff --git a/effects/trails.py b/effects/trails.py index 3f8ba2e0..e626468c 100644 --- a/effects/trails.py +++ b/effects/trails.py @@ -11,7 +11,10 @@ sleepTime = float(hyperion.args.get('speed', 1)) / 1000.0 color = list(hyperion.args.get('color', (255,255,255))) randomise = bool(hyperion.args.get('random', False)) iWidth = hyperion.imageWidth() -iHeight = hyperion.imageHeight() +iHeight = hyperion.imageHeight() + +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) class trail: def __init__(self): diff --git a/effects/x-mas.py b/effects/x-mas.py index b0bf706e..0221941f 100644 --- a/effects/x-mas.py +++ b/effects/x-mas.py @@ -4,7 +4,10 @@ import hyperion, time sleepTime = float(hyperion.args.get('sleepTime', 1000))/1000.0 length = hyperion.args.get('length', 1) color1 = hyperion.args.get('color1', (255,255,255)) -color2 = hyperion.args.get('color2', (255,0,0)) +color2 = hyperion.args.get('color2', (255,0,0)) + +# Limit update rate +sleepTime = max(hyperion.lowestUpdateInterval(), sleepTime) # Initialize the led data i = 0 diff --git a/include/effectengine/Effect.h b/include/effectengine/Effect.h index 4766f372..b6c4b45b 100644 --- a/include/effectengine/Effect.h +++ b/include/effectengine/Effect.h @@ -103,4 +103,6 @@ private: QImage _image; QPainter *_painter; QVector _imageStack; + + double _lowestUpdateIntervalInSeconds; }; diff --git a/include/effectengine/EffectModule.h b/include/effectengine/EffectModule.h index f76e7f62..ae1a6d08 100644 --- a/include/effectengine/EffectModule.h +++ b/include/effectengine/EffectModule.h @@ -51,4 +51,5 @@ public: static PyObject* wrapImageCOffset (PyObject *self, PyObject *args); static PyObject* wrapImageCShear (PyObject *self, PyObject *args); static PyObject* wrapImageResetT (PyObject *self, PyObject *args); + static PyObject* wrapLowestUpdateInterval (PyObject* self, PyObject* args); }; diff --git a/libsrc/effectengine/Effect.cpp b/libsrc/effectengine/Effect.cpp index a03ed570..a1a93afe 100644 --- a/libsrc/effectengine/Effect.cpp +++ b/libsrc/effectengine/Effect.cpp @@ -13,6 +13,11 @@ // python utils #include +// Constants +namespace { + int DEFAULT_MAX_UPDATE_RATE_HZ { 200 }; +} //End of constants + Effect::Effect(Hyperion *hyperion, int priority, int timeout, const QString &script, const QString &name, const QJsonObject &args, const QString &imageData) : QThread() , _hyperion(hyperion) @@ -27,6 +32,7 @@ Effect::Effect(Hyperion *hyperion, int priority, int timeout, const QString &scr , _interupt(false) , _imageSize(hyperion->getLedGridSize()) , _image(_imageSize,QImage::Format_ARGB32_Premultiplied) + , _lowestUpdateIntervalInSeconds(1/static_cast(DEFAULT_MAX_UPDATE_RATE_HZ)) { _colors.resize(_hyperion->getLedCount()); _colors.fill(ColorRgb::BLACK); diff --git a/libsrc/effectengine/EffectModule.cpp b/libsrc/effectengine/EffectModule.cpp index bfa4a4c4..15b933ce 100644 --- a/libsrc/effectengine/EffectModule.cpp +++ b/libsrc/effectengine/EffectModule.cpp @@ -123,6 +123,7 @@ PyMethodDef EffectModule::effectMethods[] = { {"imageCOffset" , EffectModule::wrapImageCOffset , METH_VARARGS, "Add offset to the coordinate system"}, {"imageCShear" , EffectModule::wrapImageCShear , METH_VARARGS, "Shear of coordinate system by the given horizontal/vertical axis"}, {"imageResetT" , EffectModule::wrapImageResetT , METH_NOARGS, "Resets all coords modifications (rotate,offset,shear)"}, + {"lowestUpdateInterval" , EffectModule::wrapLowestUpdateInterval , METH_NOARGS, "Gets the lowest permissible interval time in seconds"}, {NULL, NULL, 0, NULL} }; @@ -1045,3 +1046,10 @@ PyObject* EffectModule::wrapImageResetT(PyObject *self, PyObject *args) getEffect()->_painter->resetTransform(); Py_RETURN_NONE; } + +PyObject* EffectModule::wrapLowestUpdateInterval(PyObject* self, PyObject* args) +{ + qDebug() << "_lowestUpdateIntervalInSeconds: " << getEffect()->_lowestUpdateIntervalInSeconds; + + return Py_BuildValue("d", getEffect()->_lowestUpdateIntervalInSeconds); +}