@datagrok-libraries/statistics
Advanced tools
Comparing version 1.2.4 to 1.2.5
# statistics changelog | ||
## 1.2.5 (2023-08-25) | ||
### Features | ||
* [#2101](https://github.com/datagrok-ai/public/issues/2101): Added lineStyle to the IFitSeriesOptions interface | ||
* [#2106](https://github.com/datagrok-ai/public/issues/2106): Added code comments about line style | ||
### Bug Fixes | ||
* [#2101](https://github.com/datagrok-ai/public/issues/2101): Improved curves properties and rendering: | ||
* Fixed the curve fitting algorithm | ||
* Fixed axes scales | ||
## 1.2.4 (2023-08-23) | ||
@@ -4,0 +17,0 @@ |
@@ -7,3 +7,3 @@ { | ||
"friendlyName": "statistics", | ||
"version": "1.2.4", | ||
"version": "1.2.5", | ||
"description": "", | ||
@@ -10,0 +10,0 @@ "dependencies": { |
@@ -72,2 +72,3 @@ import { Property } from 'datagrok-api/src/entities'; | ||
export declare type FitMarkerType = 'asterisk' | 'circle' | 'cross border' | 'diamond' | 'square' | 'star' | 'triangle bottom' | 'triangle left' | 'triangle right' | 'triangle top'; | ||
export declare type FitLineStyle = 'solid' | 'dotted' | 'dashed' | 'dashdotted'; | ||
/** A point in the fit series. Only x and y are required. Can override some fields defined in IFitSeriesOptions. */ | ||
@@ -127,2 +128,3 @@ export interface IFitPoint { | ||
markerType?: FitMarkerType; | ||
lineStyle?: FitLineStyle; | ||
pointColor?: string; | ||
@@ -129,0 +131,0 @@ fitLineColor?: string; |
@@ -37,2 +37,53 @@ /* eslint-disable valid-jsdoc */ | ||
} | ||
/** Returns median points from within multiple points with the same x. */ | ||
function getMedianPoints(data) { | ||
const medianPoints = { x: [], y: [] }; | ||
const currentPoints = { x: [data.x[0]], y: [data.y[0]] }; | ||
for (let i = 1; i < data.x.length; i++) { | ||
if (data.x[i] === currentPoints.x[0]) { | ||
currentPoints.x[currentPoints.x.length] = data.x[i]; | ||
currentPoints.y[currentPoints.y.length] = data.y[i]; | ||
continue; | ||
} | ||
const mid = Math.floor(currentPoints.y.length / 2); | ||
const sortedPoints = currentPoints.y.sort((a, b) => a - b); | ||
const median = sortedPoints.length % 2 === 0 ? (sortedPoints[mid - 1] + sortedPoints[mid]) / 2 : sortedPoints[mid]; | ||
medianPoints.x[medianPoints.x.length] = currentPoints.x[0]; | ||
medianPoints.y[medianPoints.y.length] = median; | ||
currentPoints.x = [data.x[i]]; | ||
currentPoints.y = [data.y[i]]; | ||
} | ||
return medianPoints; | ||
} | ||
/** Returns logarithmic IC50 parameter bounds. */ | ||
function logIC50ParameterBounds(ic50Bounds) { | ||
if (ic50Bounds) { | ||
if (ic50Bounds.max !== undefined) | ||
ic50Bounds.max = Math.log10(ic50Bounds.max); | ||
if (ic50Bounds.min !== undefined) { | ||
ic50Bounds.min = ic50Bounds.min === 0 ? | ||
-Number.MAX_VALUE : Math.log10(ic50Bounds.min); | ||
} | ||
} | ||
return ic50Bounds; | ||
} | ||
function changeBounds(bounds, chartOptions) { | ||
let x = bounds.x; | ||
let y = bounds.y; | ||
let width = bounds.width; | ||
let height = bounds.height; | ||
if (chartOptions.minX !== undefined && chartOptions.minX > 0) { | ||
width += x - chartOptions.minX; | ||
x = chartOptions.minX; | ||
} | ||
if (chartOptions.maxX !== undefined && chartOptions.maxX > 0) | ||
width += chartOptions.maxX - (x + width); | ||
if (chartOptions.minY !== undefined) { | ||
height += y - chartOptions.minY; | ||
y = chartOptions.minY; | ||
} | ||
if (chartOptions.maxY !== undefined) | ||
height += chartOptions.maxY - (y + height); | ||
return new DG.Rect(x, y, width, height); | ||
} | ||
/** Returns the bounds of an {@link IFitChartData} object */ | ||
@@ -42,4 +93,2 @@ export function getChartBounds(chartData) { | ||
const o = chartData.chartOptions; | ||
if ((o === null || o === void 0 ? void 0 : o.minX) && o.minY && o.maxX && o.maxY) | ||
return new DG.Rect(o.minX, o.minY, o.maxX - o.minX, o.maxY - o.minY); | ||
if (!((_a = chartData.series) === null || _a === void 0 ? void 0 : _a.length) || chartData.series.length === 0) | ||
@@ -54,3 +103,3 @@ return new DG.Rect(0, 0, 1, 1); | ||
} | ||
return bounds; | ||
return o ? changeBounds(bounds, o) : bounds; | ||
} | ||
@@ -70,3 +119,5 @@ } | ||
y: series.points.filter((p) => !p.outlier).map((p) => (logOptions === null || logOptions === void 0 ? void 0 : logOptions.logY) ? Math.log10(p.y) : p.y) }; | ||
return fitData(data, fitFunc, FitErrorModel.Constant, series.parameterBounds); | ||
if (series.parameterBounds && (logOptions === null || logOptions === void 0 ? void 0 : logOptions.logX)) | ||
series.parameterBounds[2] = logIC50ParameterBounds(series.parameterBounds[2]); | ||
return fitData(getMedianPoints(data), fitFunc, FitErrorModel.Constant, series.parameterBounds); | ||
} | ||
@@ -90,2 +141,2 @@ /** Returns series confidence interval functions */ | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fit-data.js","sourceRoot":"","sources":["fit-data.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable valid-jsdoc */\nimport * as DG from 'datagrok-api/dg';\n\nimport {\n  FitErrorModel,\n  fitData,\n  getCurveConfidenceIntervals,\n  getStatistics,\n  FitFunction,\n  getFittedCurve,\n  FitStatistics,\n  FitConfidenceIntervals,\n  FitCurve,\n  getOrCreateFitFunction,\n  IFitPoint,\n  IFitChartData,\n  IFitSeries,\n  fitSeriesProperties,\n  fitChartDataProperties,\n  TAG_FIT,\n} from './fit-curve';\n\nexport type LogOptions = {\n  logX: boolean | undefined,\n  logY: boolean | undefined\n};\n\n\n/** Creates new object with the default values specified in {@link properties} */\nfunction createFromProperties(properties: DG.Property[]): any {\n  const o: any = {};\n  for (const p of properties) {\n    if (p.defaultValue !== null)\n      o[p.name] = p.defaultValue;\n  }\n  return o;\n}\n\n// TODO: set column with fit readonly value (in detectors) - try to only show chart - remove editable or prevent it??\n/** Creates default {@link IFitChartData} object */\nexport function createDefaultChartData(): IFitChartData {\n  return {\n    chartOptions: createFromProperties(fitChartDataProperties),\n    seriesOptions: createFromProperties(fitSeriesProperties),\n  };\n}\n\n/** Returns existing, or creates new column default chart options. */\nexport function getColumnChartOptions(gridColumn: DG.GridColumn): IFitChartData {\n  return gridColumn.temp[TAG_FIT] ??= createDefaultChartData();\n}\n\n/** Returns points arrays from {@link IFitPoint} array */\nexport function getPointsArrays(points: IFitPoint[]): {xs: number[], ys: number[]} {\n  const xs: number[] = [];\n  const ys: number[] = [];\n  for (let i = 0; i < points.length; i++) {\n    xs[i] = points[i].x;\n    ys[i] = points[i].y;\n  }\n  return {xs: xs, ys: ys};\n}\n\n/** Returns the bounds of an {@link IFitChartData} object */\nexport function getChartBounds(chartData: IFitChartData): DG.Rect {\n  const o = chartData.chartOptions;\n  if (o?.minX && o.minY && o.maxX && o.maxY)\n    return new DG.Rect(o.minX, o.minY, o.maxX - o.minX, o.maxY - o.minY);\n  if (!chartData.series?.length || chartData.series.length === 0)\n    return new DG.Rect(0, 0, 1, 1);\n  else {\n    const {xs, ys} = getPointsArrays(chartData.series[0].points);\n    let bounds = DG.Rect.fromXYArrays(xs, ys);\n    for (let i = 1; i < chartData.series!.length; i++) {\n      const {xs, ys} = getPointsArrays(chartData.series[i].points);\n      bounds = bounds.union(DG.Rect.fromXYArrays(xs, ys));\n    }\n    return bounds;\n  }\n}\n\n/** Returns series fit function */\nexport function getSeriesFitFunction(series: IFitSeries): FitFunction {\n  return getOrCreateFitFunction(series.fitFunction!);\n}\n\n/** Returns a curve function, either using the pre-computed parameters or by fitting on-the-fly */\nexport function getCurve(series: IFitSeries, fitFunc: FitFunction): (x: number) => number {\n  return getFittedCurve(fitFunc.y, series.parameters!);\n}\n\n/** Fits the series data according to the series fitting settings */\nexport function fitSeries(series: IFitSeries, fitFunc: FitFunction, logOptions?: LogOptions): FitCurve {\n  const data = {x: series.points.filter((p) => !p.outlier).map((p) => logOptions?.logX ? Math.log10(p.x) : p.x),\n    y: series.points.filter((p) => !p.outlier).map((p) => logOptions?.logY ? Math.log10(p.y) : p.y)};\n  return fitData(data, fitFunc, FitErrorModel.Constant, series.parameterBounds);\n}\n\n/** Returns series confidence interval functions */\nexport function getSeriesConfidenceInterval(series: IFitSeries, fitFunc: FitFunction,\n  userParamsFlag: boolean): FitConfidenceIntervals {\n  const data = userParamsFlag ? {x: series.points.map((p) => p.x), y: series.points.map((p) => p.y)} :\n    {x: series.points.filter((p) => !p.outlier).map((p) => p.x),\n      y: series.points.filter((p) => !p.outlier).map((p) => p.y)};\n  if (!series.parameters)\n    series.parameters = fitSeries(series, fitFunc).parameters;\n  return getCurveConfidenceIntervals(data, series.parameters, fitFunc.y, 0.05, FitErrorModel.Constant);\n}\n\n/** Returns series statistics */\nexport function getSeriesStatistics(series: IFitSeries, fitFunc: FitFunction): FitStatistics {\n  const data = {x: series.points.filter((p) => !p.outlier).map((p) => p.x),\n    y: series.points.filter((p) => !p.outlier).map((p) => p.y)};\n  if (!series.parameters)\n    series.parameters = fitSeries(series, fitFunc).parameters;\n  return getStatistics(data, series.parameters, fitFunc.y, true);\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fit-data.js","sourceRoot":"","sources":["fit-data.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable valid-jsdoc */\nimport * as DG from 'datagrok-api/dg';\n\nimport {\n  FitErrorModel,\n  fitData,\n  getCurveConfidenceIntervals,\n  getStatistics,\n  FitFunction,\n  getFittedCurve,\n  FitStatistics,\n  FitConfidenceIntervals,\n  FitCurve,\n  getOrCreateFitFunction,\n  IFitPoint,\n  IFitChartData,\n  IFitSeries,\n  fitSeriesProperties,\n  fitChartDataProperties,\n  TAG_FIT,\n  FitParamBounds,\n  IFitChartOptions,\n} from './fit-curve';\n\nexport type LogOptions = {\n  logX: boolean | undefined,\n  logY: boolean | undefined\n};\n\n\n/** Creates new object with the default values specified in {@link properties} */\nfunction createFromProperties(properties: DG.Property[]): any {\n  const o: any = {};\n  for (const p of properties) {\n    if (p.defaultValue !== null)\n      o[p.name] = p.defaultValue;\n  }\n  return o;\n}\n\n// TODO: set column with fit readonly value (in detectors) - try to only show chart - remove editable or prevent it??\n/** Creates default {@link IFitChartData} object */\nexport function createDefaultChartData(): IFitChartData {\n  return {\n    chartOptions: createFromProperties(fitChartDataProperties),\n    seriesOptions: createFromProperties(fitSeriesProperties),\n  };\n}\n\n/** Returns existing, or creates new column default chart options. */\nexport function getColumnChartOptions(gridColumn: DG.GridColumn): IFitChartData {\n  return gridColumn.temp[TAG_FIT] ??= createDefaultChartData();\n}\n\n/** Returns points arrays from {@link IFitPoint} array */\nexport function getPointsArrays(points: IFitPoint[]): {xs: number[], ys: number[]} {\n  const xs: number[] = [];\n  const ys: number[] = [];\n  for (let i = 0; i < points.length; i++) {\n    xs[i] = points[i].x;\n    ys[i] = points[i].y;\n  }\n  return {xs: xs, ys: ys};\n}\n\n/** Returns median points from within multiple points with the same x. */\nfunction getMedianPoints(data: {x: number[], y: number[]}): {x: number[], y: number[]} {\n  const medianPoints: {x: number[], y: number[]} = {x: [], y: []};\n  const currentPoints: {x: number[], y: number[]} = {x: [data.x[0]], y: [data.y[0]]};\n  for (let i = 1; i < data.x.length; i++) {\n    if (data.x[i] === currentPoints.x[0]) {\n      currentPoints.x[currentPoints.x.length] = data.x[i];\n      currentPoints.y[currentPoints.y.length] = data.y[i];\n      continue;\n    }\n    const mid = Math.floor(currentPoints.y.length / 2);\n    const sortedPoints = currentPoints.y.sort((a, b) => a - b);\n    const median = sortedPoints.length % 2 === 0 ? (sortedPoints[mid - 1] + sortedPoints[mid]) / 2 : sortedPoints[mid];\n\n    medianPoints.x[medianPoints.x.length] = currentPoints.x[0];\n    medianPoints.y[medianPoints.y.length] = median;\n    currentPoints.x = [data.x[i]];\n    currentPoints.y = [data.y[i]];\n  }\n\n  return medianPoints;\n}\n\n/** Returns logarithmic IC50 parameter bounds. */\nfunction logIC50ParameterBounds(ic50Bounds: FitParamBounds): FitParamBounds {\n  if (ic50Bounds) {\n    if (ic50Bounds.max !== undefined)\n      ic50Bounds.max = Math.log10(ic50Bounds.max);\n    if (ic50Bounds.min !== undefined) {\n      ic50Bounds.min = ic50Bounds.min === 0 ?\n        -Number.MAX_VALUE : Math.log10(ic50Bounds.min);\n    }\n  }\n  return ic50Bounds;\n}\n\nfunction changeBounds(bounds: DG.Rect, chartOptions: IFitChartOptions): DG.Rect {\n  let x = bounds.x;\n  let y = bounds.y;\n  let width = bounds.width;\n  let height = bounds.height;\n\n  if (chartOptions.minX !== undefined && chartOptions.minX > 0) {\n    width += x - chartOptions.minX;\n    x = chartOptions.minX;\n  }\n  if (chartOptions.maxX !== undefined && chartOptions.maxX > 0)\n    width += chartOptions.maxX - (x + width);\n  if (chartOptions.minY !== undefined) {\n    height += y - chartOptions.minY;\n    y = chartOptions.minY;\n  }\n  if (chartOptions.maxY !== undefined)\n    height += chartOptions.maxY - (y + height);\n\n  return new DG.Rect(x, y, width, height);\n}\n\n/** Returns the bounds of an {@link IFitChartData} object */\nexport function getChartBounds(chartData: IFitChartData): DG.Rect {\n  const o = chartData.chartOptions;\n  if (!chartData.series?.length || chartData.series.length === 0)\n    return new DG.Rect(0, 0, 1, 1);\n  else {\n    const {xs, ys} = getPointsArrays(chartData.series[0].points);\n    let bounds = DG.Rect.fromXYArrays(xs, ys);\n    for (let i = 1; i < chartData.series!.length; i++) {\n      const {xs, ys} = getPointsArrays(chartData.series[i].points);\n      bounds = bounds.union(DG.Rect.fromXYArrays(xs, ys));\n    }\n    return o ? changeBounds(bounds, o!): bounds;\n  }\n}\n\n/** Returns series fit function */\nexport function getSeriesFitFunction(series: IFitSeries): FitFunction {\n  return getOrCreateFitFunction(series.fitFunction!);\n}\n\n/** Returns a curve function, either using the pre-computed parameters or by fitting on-the-fly */\nexport function getCurve(series: IFitSeries, fitFunc: FitFunction): (x: number) => number {\n  return getFittedCurve(fitFunc.y, series.parameters!);\n}\n\n/** Fits the series data according to the series fitting settings */\nexport function fitSeries(series: IFitSeries, fitFunc: FitFunction, logOptions?: LogOptions): FitCurve {\n  const data = {x: series.points.filter((p) => !p.outlier).map((p) => logOptions?.logX ? Math.log10(p.x) : p.x),\n    y: series.points.filter((p) => !p.outlier).map((p) => logOptions?.logY ? Math.log10(p.y) : p.y)};\n  if (series.parameterBounds && logOptions?.logX)\n    series.parameterBounds[2] = logIC50ParameterBounds(series.parameterBounds[2]);\n  return fitData(getMedianPoints(data), fitFunc, FitErrorModel.Constant, series.parameterBounds);\n}\n\n/** Returns series confidence interval functions */\nexport function getSeriesConfidenceInterval(series: IFitSeries, fitFunc: FitFunction,\n  userParamsFlag: boolean): FitConfidenceIntervals {\n  const data = userParamsFlag ? {x: series.points.map((p) => p.x), y: series.points.map((p) => p.y)} :\n    {x: series.points.filter((p) => !p.outlier).map((p) => p.x),\n      y: series.points.filter((p) => !p.outlier).map((p) => p.y)};\n  if (!series.parameters)\n    series.parameters = fitSeries(series, fitFunc).parameters;\n  return getCurveConfidenceIntervals(data, series.parameters, fitFunc.y, 0.05, FitErrorModel.Constant);\n}\n\n/** Returns series statistics */\nexport function getSeriesStatistics(series: IFitSeries, fitFunc: FitFunction): FitStatistics {\n  const data = {x: series.points.filter((p) => !p.outlier).map((p) => p.x),\n    y: series.points.filter((p) => !p.outlier).map((p) => p.y)};\n  if (!series.parameters)\n    series.parameters = fitSeries(series, fitFunc).parameters;\n  return getStatistics(data, series.parameters, fitFunc.y, true);\n}\n"]} |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is not supported yet
301923
1952