mirror of
https://github.com/bitcoinresearchkit/brk.git
synced 2026-04-24 22:59:58 -07:00
website: snapshot
This commit is contained in:
@@ -22,7 +22,7 @@ function toRgba(color) {
|
||||
*/
|
||||
function tameColor(color) {
|
||||
if (color === "transparent") return color;
|
||||
return `${color.slice(0, -1)} / 50%)`;
|
||||
return `${color.slice(0, -1)} / 25%)`;
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@@ -36,7 +36,11 @@ import { resources } from "../resources.js";
|
||||
* @property {string} id
|
||||
* @property {number} paneIndex
|
||||
* @property {Signal<boolean>} active
|
||||
* @property {Signal<boolean>} highlighted
|
||||
* @property {() => void} show
|
||||
* @property {() => void} hide
|
||||
* @property {(order: number) => void} setOrder
|
||||
* @property {() => void} highlight
|
||||
* @property {() => void} tame
|
||||
* @property {() => boolean} hasData
|
||||
* @property {Signal<string | null>} url
|
||||
* @property {() => readonly T[]} getData
|
||||
@@ -165,23 +169,19 @@ export function createChart({
|
||||
});
|
||||
};
|
||||
|
||||
const visibleBarsCount = signals.createSignal(
|
||||
initialVisibleBarsCount ?? Infinity,
|
||||
);
|
||||
/** @type {() => 0 | 1 | 2 | 3} 0: <=200, 1: <=500, 2: <=1000, 3: >1000 */
|
||||
const visibleBarsCountBucket = signals.createMemo(() => {
|
||||
const count = visibleBarsCount();
|
||||
return count > 1000 ? 3 : count > 500 ? 2 : count > 200 ? 1 : 0;
|
||||
});
|
||||
const shouldShowLine = signals.createMemo(
|
||||
() => visibleBarsCountBucket() >= 2,
|
||||
);
|
||||
/** @typedef {(visibleBarsCount: number) => void} ZoomChangeCallback */
|
||||
|
||||
let visibleBarsCount = initialVisibleBarsCount ?? Infinity;
|
||||
/** @type {Set<ZoomChangeCallback>} */
|
||||
const onZoomChange = new Set();
|
||||
|
||||
ichart.timeScale().subscribeVisibleLogicalRangeChange(
|
||||
throttle((range) => {
|
||||
if (range) {
|
||||
visibleBarsCount.set(range.to - range.from);
|
||||
}
|
||||
if (!range) return;
|
||||
const count = range.to - range.from;
|
||||
if (count === visibleBarsCount) return;
|
||||
visibleBarsCount = count;
|
||||
onZoomChange.forEach((cb) => cb(count));
|
||||
}, 100),
|
||||
);
|
||||
|
||||
@@ -360,7 +360,11 @@ export function createChart({
|
||||
* @param {Accessor<WhitespaceData[]>} [args.data]
|
||||
* @param {number} args.paneIndex
|
||||
* @param {boolean} [args.defaultActive]
|
||||
* @param {(ctx: { active: Signal<boolean>, highlighted: Signal<boolean>, zOrder: number }) => void} args.setup
|
||||
* @param {(order: number) => void} args.setOrder
|
||||
* @param {() => void} args.show
|
||||
* @param {() => void} args.hide
|
||||
* @param {() => void} args.highlight
|
||||
* @param {() => void} args.tame
|
||||
* @param {() => readonly any[]} args.getData
|
||||
* @param {(data: any[]) => void} args.setData
|
||||
* @param {(data: any) => void} args.update
|
||||
@@ -376,7 +380,11 @@ export function createChart({
|
||||
defaultActive,
|
||||
colors,
|
||||
data,
|
||||
setup,
|
||||
setOrder,
|
||||
show,
|
||||
hide,
|
||||
highlight,
|
||||
tame,
|
||||
getData,
|
||||
setData,
|
||||
update,
|
||||
@@ -398,9 +406,12 @@ export function createChart({
|
||||
sharedActiveSignals.set(urlId, active);
|
||||
}
|
||||
|
||||
const highlighted = signals.createSignal(true);
|
||||
setOrder(-order);
|
||||
|
||||
setup({ active, highlighted, zOrder: -order });
|
||||
// Bridge signal to series methods
|
||||
signals.createEffect(active, (isActive) => {
|
||||
isActive ? show() : hide();
|
||||
});
|
||||
|
||||
let hasData = false;
|
||||
let lastTime = -Infinity;
|
||||
@@ -411,7 +422,11 @@ export function createChart({
|
||||
/** @type {AnySeries} */
|
||||
const series = {
|
||||
active,
|
||||
highlighted,
|
||||
setOrder,
|
||||
show,
|
||||
hide,
|
||||
highlight,
|
||||
tame,
|
||||
hasData: () => hasData,
|
||||
id,
|
||||
paneIndex,
|
||||
@@ -693,13 +708,40 @@ export function createChart({
|
||||
color: colors.default(),
|
||||
lineWidth,
|
||||
visible: false,
|
||||
priceLineVisible: false,
|
||||
priceLineVisible: true,
|
||||
},
|
||||
paneIndex,
|
||||
)
|
||||
);
|
||||
|
||||
let showLine = false;
|
||||
let active = true;
|
||||
let highlighted = true;
|
||||
let showLine = visibleBarsCount > 500;
|
||||
|
||||
function update() {
|
||||
candlestickISeries.applyOptions({
|
||||
visible: active && !showLine,
|
||||
lastValueVisible: highlighted,
|
||||
upColor: upColor.highlight(highlighted),
|
||||
downColor: downColor.highlight(highlighted),
|
||||
wickUpColor: upColor.highlight(highlighted),
|
||||
wickDownColor: downColor.highlight(highlighted),
|
||||
});
|
||||
lineISeries.applyOptions({
|
||||
visible: active && showLine,
|
||||
lastValueVisible: highlighted,
|
||||
color: colors.default.highlight(highlighted),
|
||||
});
|
||||
}
|
||||
|
||||
/** @type {ZoomChangeCallback} */
|
||||
function handleZoom(count) {
|
||||
const newShowLine = count > 500;
|
||||
if (newShowLine === showLine) return;
|
||||
showLine = newShowLine;
|
||||
update();
|
||||
}
|
||||
onZoomChange.add(handleZoom);
|
||||
|
||||
const series = addSeries({
|
||||
colors: [upColor, downColor],
|
||||
@@ -711,30 +753,29 @@ export function createChart({
|
||||
data,
|
||||
defaultActive,
|
||||
metric,
|
||||
setup: ({ active, highlighted, zOrder }) => {
|
||||
candlestickISeries.setSeriesOrder(zOrder);
|
||||
lineISeries.setSeriesOrder(zOrder);
|
||||
signals.createEffect(
|
||||
() => ({
|
||||
shouldShow: shouldShowLine(),
|
||||
active: active(),
|
||||
highlighted: highlighted(),
|
||||
}),
|
||||
({ shouldShow, active, highlighted }) => {
|
||||
showLine = shouldShow;
|
||||
candlestickISeries.applyOptions({
|
||||
visible: active && !showLine,
|
||||
upColor: upColor.highlight(highlighted),
|
||||
downColor: downColor.highlight(highlighted),
|
||||
wickUpColor: upColor.highlight(highlighted),
|
||||
wickDownColor: downColor.highlight(highlighted),
|
||||
});
|
||||
lineISeries.applyOptions({
|
||||
visible: active && showLine,
|
||||
color: colors.default.highlight(highlighted),
|
||||
});
|
||||
},
|
||||
);
|
||||
setOrder(order) {
|
||||
candlestickISeries.setSeriesOrder(order);
|
||||
lineISeries.setSeriesOrder(order);
|
||||
},
|
||||
show() {
|
||||
if (active) return;
|
||||
active = true;
|
||||
update();
|
||||
},
|
||||
hide() {
|
||||
if (!active) return;
|
||||
active = false;
|
||||
update();
|
||||
},
|
||||
highlight() {
|
||||
if (highlighted) return;
|
||||
highlighted = true;
|
||||
update();
|
||||
},
|
||||
tame() {
|
||||
if (!highlighted) return;
|
||||
highlighted = false;
|
||||
update();
|
||||
},
|
||||
setData: (data) => {
|
||||
candlestickISeries.setData(data);
|
||||
@@ -747,6 +788,7 @@ export function createChart({
|
||||
},
|
||||
getData: () => candlestickISeries.data(),
|
||||
onRemove: () => {
|
||||
onZoomChange.delete(handleZoom);
|
||||
ichart.removeSeries(candlestickISeries);
|
||||
ichart.removeSeries(lineISeries);
|
||||
},
|
||||
@@ -794,6 +836,17 @@ export function createChart({
|
||||
)
|
||||
);
|
||||
|
||||
let active = true;
|
||||
let highlighted = true;
|
||||
|
||||
function update() {
|
||||
iseries.applyOptions({
|
||||
visible: active,
|
||||
lastValueVisible: highlighted,
|
||||
color: positiveColor.highlight(highlighted),
|
||||
});
|
||||
}
|
||||
|
||||
const series = addSeries({
|
||||
colors: isDualColor ? [positiveColor, negativeColor] : [positiveColor],
|
||||
name,
|
||||
@@ -804,17 +857,26 @@ export function createChart({
|
||||
data,
|
||||
defaultActive,
|
||||
metric,
|
||||
setup: ({ active, highlighted, zOrder }) => {
|
||||
iseries.setSeriesOrder(zOrder);
|
||||
signals.createEffect(
|
||||
() => ({ active: active(), highlighted: highlighted() }),
|
||||
({ active, highlighted }) => {
|
||||
iseries.applyOptions({
|
||||
visible: active,
|
||||
color: positiveColor.highlight(highlighted),
|
||||
});
|
||||
},
|
||||
);
|
||||
setOrder: (order) => iseries.setSeriesOrder(order),
|
||||
show() {
|
||||
if (active) return;
|
||||
active = true;
|
||||
update();
|
||||
},
|
||||
hide() {
|
||||
if (!active) return;
|
||||
active = false;
|
||||
update();
|
||||
},
|
||||
highlight() {
|
||||
if (highlighted) return;
|
||||
highlighted = true;
|
||||
update();
|
||||
},
|
||||
tame() {
|
||||
if (!highlighted) return;
|
||||
highlighted = false;
|
||||
update();
|
||||
},
|
||||
setData: (data) => {
|
||||
if (isDualColor) {
|
||||
@@ -854,13 +916,14 @@ export function createChart({
|
||||
name,
|
||||
unit,
|
||||
order,
|
||||
color,
|
||||
color: _color,
|
||||
paneIndex = 0,
|
||||
defaultActive,
|
||||
data,
|
||||
options,
|
||||
}) {
|
||||
color ||= unit.id === "usd" ? colors.green : colors.orange;
|
||||
const color =
|
||||
_color ?? (unit.id === "usd" ? colors.green : colors.orange);
|
||||
|
||||
/** @type {LineISeries} */
|
||||
const iseries = /** @type {any} */ (
|
||||
@@ -877,6 +940,17 @@ export function createChart({
|
||||
)
|
||||
);
|
||||
|
||||
let active = true;
|
||||
let highlighted = true;
|
||||
|
||||
function update() {
|
||||
iseries.applyOptions({
|
||||
visible: active,
|
||||
lastValueVisible: highlighted,
|
||||
color: color.highlight(highlighted),
|
||||
});
|
||||
}
|
||||
|
||||
const series = addSeries({
|
||||
colors: [color],
|
||||
name,
|
||||
@@ -887,17 +961,26 @@ export function createChart({
|
||||
data,
|
||||
defaultActive,
|
||||
metric,
|
||||
setup: ({ active, highlighted, zOrder }) => {
|
||||
iseries.setSeriesOrder(zOrder);
|
||||
signals.createEffect(
|
||||
() => ({ active: active(), highlighted: highlighted() }),
|
||||
({ active, highlighted }) => {
|
||||
iseries.applyOptions({
|
||||
visible: active,
|
||||
color: color.highlight(highlighted),
|
||||
});
|
||||
},
|
||||
);
|
||||
setOrder: (order) => iseries.setSeriesOrder(order),
|
||||
show() {
|
||||
if (active) return;
|
||||
active = true;
|
||||
update();
|
||||
},
|
||||
hide() {
|
||||
if (!active) return;
|
||||
active = false;
|
||||
update();
|
||||
},
|
||||
highlight() {
|
||||
if (highlighted) return;
|
||||
highlighted = true;
|
||||
update();
|
||||
},
|
||||
tame() {
|
||||
if (!highlighted) return;
|
||||
highlighted = false;
|
||||
update();
|
||||
},
|
||||
setData: (data) => iseries.setData(data),
|
||||
update: (data) => iseries.update(data),
|
||||
@@ -923,13 +1006,14 @@ export function createChart({
|
||||
name,
|
||||
unit,
|
||||
order,
|
||||
color,
|
||||
color: _color,
|
||||
paneIndex = 0,
|
||||
defaultActive,
|
||||
data,
|
||||
options,
|
||||
}) {
|
||||
color ||= unit.id === "usd" ? colors.green : colors.orange;
|
||||
const color =
|
||||
_color ?? (unit.id === "usd" ? colors.green : colors.orange);
|
||||
|
||||
/** @type {LineISeries} */
|
||||
const iseries = /** @type {any} */ (
|
||||
@@ -948,6 +1032,28 @@ export function createChart({
|
||||
)
|
||||
);
|
||||
|
||||
let active = true;
|
||||
let highlighted = true;
|
||||
let radius =
|
||||
visibleBarsCount > 1000 ? 1 : visibleBarsCount > 200 ? 1.5 : 2;
|
||||
|
||||
function update() {
|
||||
iseries.applyOptions({
|
||||
visible: active,
|
||||
lastValueVisible: highlighted,
|
||||
color: color.highlight(highlighted),
|
||||
});
|
||||
}
|
||||
|
||||
/** @type {ZoomChangeCallback} */
|
||||
function handleZoom(count) {
|
||||
const newRadius = count > 1000 ? 1 : count > 200 ? 1.5 : 2;
|
||||
if (newRadius === radius) return;
|
||||
radius = newRadius;
|
||||
iseries.applyOptions({ pointMarkersRadius: radius });
|
||||
}
|
||||
onZoomChange.add(handleZoom);
|
||||
|
||||
const series = addSeries({
|
||||
colors: [color],
|
||||
name,
|
||||
@@ -958,26 +1064,34 @@ export function createChart({
|
||||
data,
|
||||
defaultActive,
|
||||
metric,
|
||||
setup: ({ active, highlighted, zOrder }) => {
|
||||
iseries.setSeriesOrder(zOrder);
|
||||
signals.createEffect(
|
||||
() => ({ active: active(), highlighted: highlighted() }),
|
||||
({ active, highlighted }) => {
|
||||
iseries.applyOptions({
|
||||
visible: active,
|
||||
color: color.highlight(highlighted),
|
||||
});
|
||||
},
|
||||
);
|
||||
signals.createEffect(visibleBarsCountBucket, (bucket) => {
|
||||
const radius = bucket === 3 ? 1 : bucket >= 1 ? 1.5 : 2;
|
||||
iseries.applyOptions({ pointMarkersRadius: radius });
|
||||
});
|
||||
setOrder: (order) => iseries.setSeriesOrder(order),
|
||||
show() {
|
||||
if (active) return;
|
||||
active = true;
|
||||
update();
|
||||
},
|
||||
hide() {
|
||||
if (!active) return;
|
||||
active = false;
|
||||
update();
|
||||
},
|
||||
highlight() {
|
||||
if (highlighted) return;
|
||||
highlighted = true;
|
||||
update();
|
||||
},
|
||||
tame() {
|
||||
if (!highlighted) return;
|
||||
highlighted = false;
|
||||
update();
|
||||
},
|
||||
setData: (data) => iseries.setData(data),
|
||||
update: (data) => iseries.update(data),
|
||||
getData: () => iseries.data(),
|
||||
onRemove: () => ichart.removeSeries(iseries),
|
||||
onRemove: () => {
|
||||
onZoomChange.delete(handleZoom);
|
||||
ichart.removeSeries(iseries);
|
||||
},
|
||||
});
|
||||
return series;
|
||||
},
|
||||
@@ -1032,6 +1146,18 @@ export function createChart({
|
||||
)
|
||||
);
|
||||
|
||||
let active = true;
|
||||
let highlighted = true;
|
||||
|
||||
function update() {
|
||||
iseries.applyOptions({
|
||||
visible: active,
|
||||
lastValueVisible: highlighted,
|
||||
topLineColor: topColor.highlight(highlighted),
|
||||
bottomLineColor: bottomColor.highlight(highlighted),
|
||||
});
|
||||
}
|
||||
|
||||
const series = addSeries({
|
||||
colors: [topColor, bottomColor],
|
||||
name,
|
||||
@@ -1042,18 +1168,26 @@ export function createChart({
|
||||
data,
|
||||
defaultActive,
|
||||
metric,
|
||||
setup: ({ active, highlighted, zOrder }) => {
|
||||
iseries.setSeriesOrder(zOrder);
|
||||
signals.createEffect(
|
||||
() => ({ active: active(), highlighted: highlighted() }),
|
||||
({ active, highlighted }) => {
|
||||
iseries.applyOptions({
|
||||
visible: active,
|
||||
topLineColor: topColor.highlight(highlighted),
|
||||
bottomLineColor: bottomColor.highlight(highlighted),
|
||||
});
|
||||
},
|
||||
);
|
||||
setOrder: (order) => iseries.setSeriesOrder(order),
|
||||
show() {
|
||||
if (active) return;
|
||||
active = true;
|
||||
update();
|
||||
},
|
||||
hide() {
|
||||
if (!active) return;
|
||||
active = false;
|
||||
update();
|
||||
},
|
||||
highlight() {
|
||||
if (highlighted) return;
|
||||
highlighted = true;
|
||||
update();
|
||||
},
|
||||
tame() {
|
||||
if (!highlighted) return;
|
||||
highlighted = false;
|
||||
update();
|
||||
},
|
||||
setData: (data) => iseries.setData(data),
|
||||
update: (data) => iseries.update(data),
|
||||
|
||||
@@ -7,7 +7,23 @@ import { stringToId } from "../utils/format.js";
|
||||
export function createLegend(signals) {
|
||||
const element = window.document.createElement("legend");
|
||||
|
||||
const hovered = signals.createSignal(/** @type {AnySeries | null} */ (null));
|
||||
/** @type {AnySeries | null} */
|
||||
let hoveredSeries = null;
|
||||
/** @type {Map<AnySeries, { span: HTMLSpanElement, color: Color }[]>} */
|
||||
const seriesColorSpans = new Map();
|
||||
|
||||
/** @param {AnySeries | null} series */
|
||||
function setHovered(series) {
|
||||
if (hoveredSeries === series) return;
|
||||
hoveredSeries = series;
|
||||
for (const [entrySeries, colorSpans] of seriesColorSpans) {
|
||||
const shouldHighlight = !hoveredSeries || hoveredSeries === entrySeries;
|
||||
shouldHighlight ? entrySeries.highlight() : entrySeries.tame();
|
||||
for (const { span, color } of colorSpans) {
|
||||
span.style.backgroundColor = color.highlight(shouldHighlight);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/** @type {HTMLElement[]} */
|
||||
const legends = [];
|
||||
@@ -62,37 +78,21 @@ export function createLegend(signals) {
|
||||
spanMain.append(spanName);
|
||||
|
||||
div.append(label);
|
||||
label.addEventListener("mouseover", () => {
|
||||
const h = hovered();
|
||||
if (!h || h !== series) {
|
||||
hovered.set(series);
|
||||
}
|
||||
});
|
||||
label.addEventListener("mouseleave", () => {
|
||||
hovered.set(null);
|
||||
});
|
||||
|
||||
const shouldHighlight = () => !hovered() || hovered() === series;
|
||||
|
||||
// Update series highlighted state
|
||||
signals.createEffect(shouldHighlight, (shouldHighlight) => {
|
||||
series.highlighted.set(shouldHighlight);
|
||||
});
|
||||
label.addEventListener("mouseover", () => setHovered(series));
|
||||
label.addEventListener("mouseleave", () => setHovered(null));
|
||||
|
||||
const spanColors = window.document.createElement("span");
|
||||
spanColors.classList.add("colors");
|
||||
spanMain.prepend(spanColors);
|
||||
/** @type {{ span: HTMLSpanElement, color: Color }[]} */
|
||||
const colorSpans = [];
|
||||
colors.forEach((color) => {
|
||||
const spanColor = window.document.createElement("span");
|
||||
spanColor.style.backgroundColor = color.highlight(true);
|
||||
spanColors.append(spanColor);
|
||||
|
||||
signals.createEffect(
|
||||
() => color.highlight(shouldHighlight()),
|
||||
(c) => {
|
||||
spanColor.style.backgroundColor = c;
|
||||
},
|
||||
);
|
||||
colorSpans.push({ span: spanColor, color });
|
||||
});
|
||||
seriesColorSpans.set(series, colorSpans);
|
||||
|
||||
const anchor = window.document.createElement("a");
|
||||
|
||||
|
||||
Reference in New Issue
Block a user