-
-
Notifications
You must be signed in to change notification settings - Fork 57
/
Copy pathprojectionOscillator.ts
92 lines (82 loc) · 2.14 KB
/
projectionOscillator.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
// Copyright (c) 2022 Onur Cinar. All Rights Reserved.
// https://github.com/cinar/indicatorts
import {
add,
divide,
generateNumbers,
multiply,
multiplyBy,
subtract,
} from '../../helper/numArray';
import { movingLeastSquare } from '../../helper/regression';
import { ema } from '../trend/exponentialMovingAverage';
import { mmax } from '../trend/movingMax';
import { mmin } from '../trend/movingMin';
/**
* Projection oscillator result object.
*/
export interface POResult {
poResult: number[];
spoResult: number[];
}
/**
* Optional configuration of PO parameters.
*/
export interface POConfig {
period?: number;
smooth?: number;
}
/**
* The default configuration of PO.
*/
export const PODefaultConfig: Required<POConfig> = {
period: 14,
smooth: 3,
};
/**
* ProjectionOscillator calculates the Projection Oscillator (PO). The PO
* uses the linear regression slope, along with highs and lows.
*
* Period defines the moving window to calculates the PO, and the smooth
* period defines the moving windows to take EMA of PO.
*
* PL = Min(period, (high + MLS(period, x, high)))
* PU = Max(period, (low + MLS(period, x, low)))
* PO = 100 * (Closing - PL) / (PU - PL)
* SPO = EMA(smooth, PO)
*
* @param highs high values.
* @param lows low values.
* @param closings closing values.
* @param config configuration.
* @return projection oscillator.
*/
export function po(
highs: number[],
lows: number[],
closings: number[],
config: POConfig = {}
): POResult {
const { period, smooth } = {
...PODefaultConfig,
...config,
};
const x = generateNumbers(0, closings.length, 1);
const lsHighs = movingLeastSquare(period, x, highs);
const lsLows = movingLeastSquare(period, x, lows);
const vHighs = add(highs, multiply(lsHighs.m, x));
const vLows = add(lows, multiply(lsLows.m, x));
const pu = mmax(vHighs, { period });
const pl = mmin(vLows, { period });
const poResult = divide(
multiplyBy(100, subtract(closings, pl)),
subtract(pu, pl)
);
const spoResult = ema(poResult, { period: smooth });
return {
poResult,
spoResult,
};
}
// Export full name
export { po as projectionOscillator };