UNPKG

react-native-wagmi-charts

Version:

A sweet candlestick chart for React Native

269 lines (258 loc) 8.01 kB
'use strict'; Object.defineProperty(exports, '__esModule', { value: true, }); exports.LineChartCursorLine = LineChartCursorLine; var _react = _interopRequireDefault(require('react')); var _reactNative = require('react-native'); var _reactNativeReanimated = _interopRequireWildcard( require('react-native-reanimated') ); var _reactNativeSvg = _interopRequireWildcard(require('react-native-svg')); var _AnimatedText = require('../../components/AnimatedText'); var _Chart = require('./Chart'); var _Cursor = require('./Cursor'); var _useDatetime = require('./useDatetime'); var _useLineChart = require('./useLineChart'); var _usePrice = require('./usePrice'); function _interopRequireWildcard(e, t) { if ('function' == typeof WeakMap) var r = new WeakMap(), n = new WeakMap(); return (_interopRequireWildcard = function (e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || ('object' != typeof e && 'function' != typeof e)) return f; if ((o = t ? n : r)) { if (o.has(e)) return o.get(e); o.set(e, f); } for (const t in e) 'default' !== t && {}.hasOwnProperty.call(e, t) && ((i = (o = Object.defineProperty) && Object.getOwnPropertyDescriptor(e, t)) && (i.get || i.set) ? o(f, t, i) : (f[t] = e[t])); return f; })(e, t); } function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; } function _extends() { return ( (_extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }), _extends.apply(null, arguments) ); } LineChartCursorLine.displayName = 'LineChartCursorLine'; const TEXT_CONSTANTS = { DEFAULT_COLOR: '#1A1E27', DEFAULT_FONT_SIZE: 12, CHAR_WIDTH_RATIO: 0.6, MIN_WIDTH: 25, MAX_WIDTH: 150, INPUT_PADDING: 4, }; const SPACING = { VERTICAL_TEXT_OFFSET: 40, HORIZONTAL_TEXT_MARGIN: 8, HORIZONTAL_RIGHT_MARGIN: 16, BASE_LINE_GAP: 8, X_AXIS_LABEL_RESERVED_HEIGHT: 40, // Reserved space at bottom for x-axis labels }; const AnimatedLine = _reactNativeReanimated.default.createAnimatedComponent( _reactNativeSvg.Line ); function LineChartCursorLine({ children, color = 'gray', lineProps, format, textStyle, ...cursorProps }) { const isHorizontal = cursorProps?.orientation === 'horizontal'; const { height, width } = _react.default.useContext( _Chart.LineChartDimensionsContext ); const { currentX, currentY, isActive } = (0, _useLineChart.useLineChart)(); const price = (0, _usePrice.useLineChartPrice)({ format: isHorizontal ? format : undefined, precision: 2, }); const datetime = (0, _useDatetime.useLineChartDatetime)({ format: !isHorizontal ? format : undefined, }); const displayText = isHorizontal ? price.formatted : datetime.formatted; const calculateTextWidth = (text, fontSize) => { 'worklet'; const charWidth = fontSize * TEXT_CONSTANTS.CHAR_WIDTH_RATIO; const calculatedWidth = text.length * charWidth; return Math.max( TEXT_CONSTANTS.MIN_WIDTH, Math.min(TEXT_CONSTANTS.MAX_WIDTH, calculatedWidth) ); }; const textWidth = (0, _reactNativeReanimated.useDerivedValue)(() => { const text = displayText.value; if (!text) return TEXT_CONSTANTS.MIN_WIDTH; const fontSize = textStyle?.fontSize || TEXT_CONSTANTS.DEFAULT_FONT_SIZE; return calculateTextWidth(text, fontSize); }, [displayText, textStyle?.fontSize]); const lineEndX = (0, _reactNativeReanimated.useDerivedValue)(() => { if (!isHorizontal) return 0; const fontSize = textStyle?.fontSize || TEXT_CONSTANTS.DEFAULT_FONT_SIZE; const gap = Math.max(SPACING.BASE_LINE_GAP, fontSize * 0.5); return ( width - textWidth.value - gap - TEXT_CONSTANTS.INPUT_PADDING - SPACING.HORIZONTAL_RIGHT_MARGIN ); }); const lineEndY = (0, _reactNativeReanimated.useDerivedValue)(() => { if (isHorizontal) return 0; // For vertical cursor, extend line to the chart area (excluding reserved label space) return height - SPACING.X_AXIS_LABEL_RESERVED_HEIGHT; }); const containerStyle = (0, _reactNativeReanimated.useAnimatedStyle)(() => ({ opacity: isActive.value ? 1 : 0, height: '100%', transform: isHorizontal ? [ { translateY: currentY.value, }, ] : [ { translateX: currentX.value, }, ], })); const calculateFontSizeAdjustment = (fontSize) => { 'worklet'; return Math.max(0.6, Math.min(0.8, 0.7 + (fontSize - 12) * 0.01)); }; const textPositionStyle = (0, _reactNativeReanimated.useAnimatedStyle)(() => { const fontSize = textStyle?.fontSize || TEXT_CONSTANTS.DEFAULT_FONT_SIZE; const lineHeight = textStyle?.lineHeight || fontSize * 1.2; const baseStyle = { position: 'absolute', width: textWidth.value, fontSize, lineHeight, color: textStyle?.color || TEXT_CONSTANTS.DEFAULT_COLOR, ...textStyle, }; if (isHorizontal) { const fontSizeAdjustment = calculateFontSizeAdjustment(fontSize); const textCenterOffset = -(lineHeight * fontSizeAdjustment); return { ...baseStyle, left: width - textWidth.value - SPACING.HORIZONTAL_RIGHT_MARGIN + TEXT_CONSTANTS.INPUT_PADDING, top: textCenterOffset, textAlign: 'right', paddingLeft: 0, paddingRight: 0, }; } // For vertical cursor (x-axis label) const halfTextWidth = textWidth.value / 2; // Since the container is already translated by currentX, we need to calculate // the label position relative to the container's position const containerX = currentX.value; // Calculate where the label would be if centered let labelLeft = -halfTextWidth; // Check if label would overflow on the left if (containerX + labelLeft < 0) { labelLeft = -containerX; } // Check if label would overflow on the right if (containerX + labelLeft + textWidth.value > width) { labelLeft = width - containerX - textWidth.value; } // Position label in the reserved space at the bottom const labelTop = height - SPACING.X_AXIS_LABEL_RESERVED_HEIGHT + SPACING.HORIZONTAL_TEXT_MARGIN; return { ...baseStyle, left: labelLeft, top: labelTop, textAlign: 'center', }; }); const lineAnimatedProps = (0, _reactNativeReanimated.useAnimatedProps)( () => ({ x1: 0, y1: 0, x2: lineEndX.value, y2: lineEndY.value, }), [lineEndX, lineEndY] ); return /*#__PURE__*/ _react.default.createElement( _Cursor.LineChartCursor, _extends({}, cursorProps, { type: 'line', }), /*#__PURE__*/ _react.default.createElement( _reactNativeReanimated.default.View, { style: containerStyle, }, /*#__PURE__*/ _react.default.createElement( _reactNativeSvg.default, { style: styles.svg, }, /*#__PURE__*/ _react.default.createElement( AnimatedLine, _extends( { animatedProps: lineAnimatedProps, strokeWidth: 2, stroke: color, strokeDasharray: '3 3', }, lineProps ) ) ), /*#__PURE__*/ _react.default.createElement(_AnimatedText.AnimatedText, { text: displayText, style: textPositionStyle, }) ), children ); } const styles = _reactNative.StyleSheet.create({ svg: { ..._reactNative.StyleSheet.absoluteFillObject, height: '100%', }, }); //# sourceMappingURL=CursorLine.js.map