1539 views
The need to calculate averages (arithmetic mean) comes up here and there in scripts. When you want the average of a value only when a given condition has occurred, you will often not be able to use the standard sma function in Pine because it will average all values in the series. Even if you take care to use your condition to set non-conforming values to zero, this ends up skewing your results with zero values.

This script calculates:
1. The running average of all up volumes,
2. The average of last x occurrences of up volume ,
3. The average up volume occurrences in last y bars.
Release Notes:
• Made code for cases 2 and 3 into functions.
• Added examples of multiple-result functions.
Release Notes: Conversion to v4.
Release Notes: Simplified Case 1 code.
Release Notes: Updated calculations.
Tools and ideas for all Pine coders: http://www.pinecoders.com
Our Pine FAQ & Code: http://www.pinecoders.com/faq_and_code/
Get Pine news and tips on Telegram: https://t.me/PineCodersSquawkBox hendecagon
@hendecagon, Thx. We need to update case 3. Would be more efficient and elegant to use:
```avgWhenInLast(_src, _cond, _cnt) => sum(_cond ? _src : 0, _cnt) / sum(_cond ? 1 : 0, _cnt)
``` flush6789
@flush6789, ) Thx for your nice comment. Appreciate. Happy you find the code useful. Well done!
But maybe you know how to limit averaging period with a custom event (stop loss, take profit or opposite condition) instead of amount of bars?
For example I am trying to find a script to replace the builtin "strategy.position_avg_price" function. I need it for my study script dca alertconditions. MoneyButton
@MoneyButton, If the point where you will need to start averaging from can be detected as your script is executing on it, then you can use technique 1b and add a condition to reset the values upon encountering that start condition. Don't even need a loop. The loop is only required when the conditions required for your calcs can only be known in the future, and so you need to wait for that future to occur, then go back. PineCoders
@PineCoders, thank you for a quick reply.

It would be great if you could show an example script with a "reset_condition" set to reset the values. :)
I am new with coding and it is difficult for me to imagine or develop scripts, however when I see a working one, I can understand how it works and then implement it into my strategy. MoneyButton
@MoneyButton, This is a new function which returns the average of a series counted from the last moment a given condition was true. The function also returns the number of bars elapsed since the condition was true, so the length over which the average is calculated:
```//@version=4
study("Average Since", "", true)

// Returns the average of `_src` since `_cond` was last true, and the number of bars since.
f_avgSince(_src, _cond) =>
// _src  : series to average.
// _cond : condition upon which average is reset.
var _cumTotal = 0.
var _cumCount = 0
if _cond
// When condition is true, reset avg to zero.
_cumTotal := _src
_cumCount := 1
else
// Continue calculating running numbers.
_cumTotal := _cumTotal + _src
_cumCount := _cumCount + 1
_avg = _cumTotal / _cumCount
[_avg, _cumCount - 1]

hi = highest(input(50))
// Condition which will reset our avg: when a new high is detected.
newHi = hi != hi
// Call our function and get 2 results from it simultaneously. 