@datagrok-libraries/statistics 1.10.0 → 1.12.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/src/mpo/mpo.d.ts CHANGED
@@ -1,19 +1,69 @@
1
1
  import * as DG from 'datagrok-api/dg';
2
2
  export type DesirabilityLine = number[][];
3
- export type PropertyDesirability = {
3
+ export type DesirabilityMode = 'freeform' | 'gaussian' | 'sigmoid';
4
+ export type MissingValueConfig = {
5
+ strategy: 'exclude';
6
+ } | {
7
+ strategy: 'default';
8
+ score: number;
9
+ } | {
10
+ strategy: 'skip';
11
+ };
12
+ type BasePropertyDesirability = {
13
+ weight: number;
14
+ weightColumn?: string;
15
+ missingValues?: MissingValueConfig;
16
+ };
17
+ export type NumericalDesirability = BasePropertyDesirability & {
18
+ functionType: 'numerical';
4
19
  line: DesirabilityLine;
5
20
  min?: number;
6
21
  max?: number;
7
- weight: number;
22
+ mode?: DesirabilityMode;
23
+ mean?: number;
24
+ sigma?: number;
25
+ x0?: number;
26
+ k?: number;
27
+ freeformLine?: DesirabilityLine;
8
28
  };
29
+ export type CategoricalDesirability = BasePropertyDesirability & {
30
+ functionType: 'categorical';
31
+ categories: {
32
+ name: string;
33
+ desirability: number;
34
+ }[];
35
+ };
36
+ export type PropertyDesirability = NumericalDesirability | CategoricalDesirability;
37
+ export declare function isNumerical(p: PropertyDesirability): p is NumericalDesirability;
38
+ export declare function createDefaultNumerical(weight?: number, min?: number, max?: number): NumericalDesirability;
39
+ export declare function createDefaultCategorical(weight?: number, categories?: {
40
+ name: string;
41
+ desirability: number;
42
+ }[]): CategoricalDesirability;
43
+ export declare function migrateDesirability(raw: any): PropertyDesirability;
44
+ export declare const DESIRABILITY_PROFILE_TYPE = "MPO Desirability Profile";
9
45
  export type DesirabilityProfile = {
46
+ type: typeof DESIRABILITY_PROFILE_TYPE;
10
47
  name: string;
11
48
  description: string;
49
+ aggregation?: WeightedAggregation;
12
50
  properties: {
13
51
  [key: string]: PropertyDesirability;
14
52
  };
15
53
  };
54
+ export declare function isDesirabilityProfile(x: any): x is DesirabilityProfile;
55
+ export declare const WEIGHTED_AGGREGATIONS: readonly ["Average", "Sum", "Product", "Geomean", "Min", "Max"];
56
+ export declare const WEIGHTED_AGGREGATIONS_LIST: WeightedAggregation[];
57
+ export type WeightedAggregation = typeof WEIGHTED_AGGREGATIONS[number];
58
+ export declare const DEFAULT_AGGREGATION: WeightedAggregation;
16
59
  export declare function desirabilityScore(x: number, desirabilityLine: DesirabilityLine): number;
60
+ export declare function categoricalDesirabilityScore(value: string, prop: CategoricalDesirability): number | null;
61
+ export type MpoResult = {
62
+ scoreColumn: DG.Column;
63
+ desirabilityColumns?: DG.Column[];
64
+ };
17
65
  /** Calculates the multi parameter optimization score, 0-100, 100 is the maximum */
18
- export declare function mpo(dataFrame: DG.DataFrame, columns: DG.Column[]): DG.Column;
66
+ export declare function mpo(dataFrame: DG.DataFrame, columns: DG.Column[], profileName: string, aggregation: WeightedAggregation, isDifferent?: boolean, createDesirabilityColumns?: boolean): MpoResult;
67
+ export declare function aggregate(scores: number[], weights: number[], aggregation: WeightedAggregation): number;
68
+ export {};
19
69
  //# sourceMappingURL=mpo.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"mpo.d.ts","sourceRoot":"","sources":["mpo.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAItC,MAAM,MAAM,gBAAgB,GAAG,MAAM,EAAE,EAAE,CAAC;AAG1C,MAAM,MAAM,oBAAoB,GAAG;IACjC,IAAI,EAAE,gBAAgB,CAAC;IACvB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;CAChB,CAAA;AAGD,MAAM,MAAM,mBAAmB,GAAG;IAChC,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,UAAU,EAAE;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,oBAAoB,CAAA;KAAE,CAAC;CACrD,CAAA;AAKD,wBAAgB,iBAAiB,CAAC,CAAC,EAAE,MAAM,EAAE,gBAAgB,EAAE,gBAAgB,GAAG,MAAM,CAoBvF;AAED,mFAAmF;AACnF,wBAAgB,GAAG,CAAC,SAAS,EAAE,EAAE,CAAC,SAAS,EAAE,OAAO,EAAE,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE,CAAC,MAAM,CA8B5E"}
1
+ {"version":3,"file":"mpo.d.ts","sourceRoot":"","sources":["mpo.ts"],"names":[],"mappings":"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"}
package/src/mpo/mpo.js CHANGED
@@ -1,6 +1,29 @@
1
1
  /* eslint-disable guard-for-in */
2
2
  /* eslint-disable max-len */
3
3
  import * as DG from 'datagrok-api/dg';
4
+ export function isNumerical(p) {
5
+ return p.functionType === 'numerical';
6
+ }
7
+ export function createDefaultNumerical(weight = 1, min = 0, max = 1) {
8
+ return { functionType: 'numerical', weight, mode: 'freeform', min, max, line: [] };
9
+ }
10
+ export function createDefaultCategorical(weight = 1, categories) {
11
+ return { functionType: 'categorical', weight, categories: categories !== null && categories !== void 0 ? categories : [{ name: 'Category 1', desirability: 1 }] };
12
+ }
13
+ export function migrateDesirability(raw) {
14
+ if (raw.functionType)
15
+ return raw;
16
+ if (raw.categories)
17
+ return { ...raw, functionType: 'categorical' };
18
+ return { ...raw, functionType: 'numerical' };
19
+ }
20
+ export const DESIRABILITY_PROFILE_TYPE = 'MPO Desirability Profile';
21
+ export function isDesirabilityProfile(x) {
22
+ return x != null && typeof x === 'object' && x.type === DESIRABILITY_PROFILE_TYPE;
23
+ }
24
+ export const WEIGHTED_AGGREGATIONS = ['Average', 'Sum', 'Product', 'Geomean', 'Min', 'Max'];
25
+ export const WEIGHTED_AGGREGATIONS_LIST = [...WEIGHTED_AGGREGATIONS];
26
+ export const DEFAULT_AGGREGATION = 'Average';
4
27
  /// Calculates the desirability score for a given x value
5
28
  /// Returns 0 if x is outside the range of the desirability line
6
29
  /// Otherwise, returns the y value of the desirability line at x
@@ -23,30 +46,83 @@ export function desirabilityScore(x, desirabilityLine) {
23
46
  // Should not happen if x is within bounds, but return 0 as fallback
24
47
  return 0;
25
48
  }
49
+ export function categoricalDesirabilityScore(value, prop) {
50
+ var _a;
51
+ const found = prop.categories.find((c) => c.name === value);
52
+ return (_a = found === null || found === void 0 ? void 0 : found.desirability) !== null && _a !== void 0 ? _a : null;
53
+ }
26
54
  /** Calculates the multi parameter optimization score, 0-100, 100 is the maximum */
27
- export function mpo(dataFrame, columns) {
55
+ export function mpo(dataFrame, columns, profileName, aggregation, isDifferent = false, createDesirabilityColumns = false) {
56
+ var _a, _b;
28
57
  if (columns.length === 0)
29
58
  throw new Error('No columns provided for MPO calculation.');
30
- const resultColumnName = dataFrame.columns.getUnusedName('MPO'); // Ensure unique name
31
- const resultColumn = DG.Column.float(resultColumnName, columns[0].length);
32
- const desirabilityTemplates = columns.map((column) => {
59
+ const rowCount = columns[0].length;
60
+ const resultColumn = isDifferent ?
61
+ DG.Column.float(profileName, rowCount) :
62
+ ((_a = dataFrame.col(profileName)) !== null && _a !== void 0 ? _a : DG.Column.float(profileName, rowCount));
63
+ const desirabilityTemplates = [];
64
+ const weightColumns = [];
65
+ for (const column of columns) {
33
66
  const tag = column.getTag('desirabilityTemplate');
34
- return JSON.parse(tag);
35
- });
67
+ const d = migrateDesirability(JSON.parse(tag));
68
+ desirabilityTemplates.push(d);
69
+ weightColumns.push(d.weightColumn ? (_b = dataFrame.col(d.weightColumn)) !== null && _b !== void 0 ? _b : null : null);
70
+ }
71
+ const desirabilityColumns = createDesirabilityColumns ?
72
+ columns.map((col) => DG.Column.float(`${col.name} (${profileName} desirability)`, rowCount)) : undefined;
36
73
  resultColumn.init((i) => {
37
- let totalScore = 0;
38
- let maxScore = 0;
74
+ const scores = [];
75
+ const weights = [];
39
76
  for (let j = 0; j < columns.length; j++) {
40
77
  const desirability = desirabilityTemplates[j];
41
78
  const value = columns[j].get(i);
42
- const score = desirabilityScore(value, desirability.line);
43
- totalScore += desirability.weight * score;
44
- maxScore += desirability.weight;
79
+ let score;
80
+ if (columns[j].isNone(i)) {
81
+ const mv = desirability.missingValues;
82
+ if (!mv || mv.strategy === 'exclude')
83
+ return NaN;
84
+ if (mv.strategy === 'skip')
85
+ continue;
86
+ score = mv.score;
87
+ }
88
+ else {
89
+ score = isNumerical(desirability) ?
90
+ desirabilityScore(value, desirability.line) :
91
+ categoricalDesirabilityScore(String(value), desirability);
92
+ }
93
+ if (score === null)
94
+ return NaN;
95
+ if (createDesirabilityColumns)
96
+ desirabilityColumns[j].set(i, score, false);
97
+ scores.push(score);
98
+ const wCol = weightColumns[j];
99
+ const w = wCol && !wCol.isNone(i) ? Math.max(0, Math.min(1, wCol.get(i))) : desirability.weight;
100
+ weights.push(w);
45
101
  }
46
- return 100 * (totalScore / maxScore);
102
+ if (scores.length === 0)
103
+ return NaN;
104
+ return aggregate(scores, weights, aggregation);
47
105
  });
48
- // Add the column to the table
49
- dataFrame.columns.add(resultColumn);
50
- return resultColumn;
106
+ return { scoreColumn: resultColumn, desirabilityColumns };
107
+ }
108
+ export function aggregate(scores, weights, aggregation) {
109
+ switch (aggregation) {
110
+ case 'Sum':
111
+ return scores.reduce((sum, s, idx) => sum + s * weights[idx], 0);
112
+ case 'Average':
113
+ const totalWeight = weights.reduce((sum, w) => sum + w, 0);
114
+ return scores.reduce((sum, s, idx) => sum + s * weights[idx], 0) / totalWeight;
115
+ case 'Product':
116
+ return scores.reduce((prod, s, idx) => prod * Math.pow(s, weights[idx]), 1);
117
+ case 'Geomean':
118
+ const totalW = weights.reduce((sum, w) => sum + w, 0);
119
+ return scores.reduce((prod, s, idx) => prod * Math.pow(s, weights[idx] / totalW), 1);
120
+ case 'Min':
121
+ return Math.min(...scores.map((s, idx) => Math.pow(s, weights[idx])));
122
+ case 'Max':
123
+ return Math.max(...scores.map((s, idx) => Math.pow(s, weights[idx])));
124
+ default:
125
+ throw new Error(`Unknown aggregation type: ${aggregation}`);
126
+ }
51
127
  }
52
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibXBvLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsibXBvLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGlDQUFpQztBQUNqQyw0QkFBNEI7QUFDNUIsT0FBTyxLQUFLLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQXFCdEMseURBQXlEO0FBQ3pELGdFQUFnRTtBQUNoRSxnRUFBZ0U7QUFDaEUsTUFBTSxVQUFVLGlCQUFpQixDQUFDLENBQVMsRUFBRSxnQkFBa0M7SUFDN0UsMkRBQTJEO0lBQzNELElBQUksZ0JBQWdCLENBQUMsTUFBTSxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLGdCQUFnQixDQUFDLGdCQUFnQixDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDckgsT0FBTyxDQUFDLENBQUM7SUFFWCwwQ0FBMEM7SUFDMUMsS0FBSyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLGdCQUFnQixDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUU7UUFDcEQsTUFBTSxDQUFDLEVBQUUsRUFBRSxFQUFFLENBQUMsR0FBRyxnQkFBZ0IsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNyQyxNQUFNLENBQUMsRUFBRSxFQUFFLEVBQUUsQ0FBQyxHQUFHLGdCQUFnQixDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztRQUV6QyxJQUFJLENBQUMsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsRUFBRTtZQUN0Qiw4Q0FBOEM7WUFDOUMsSUFBSSxFQUFFLEtBQUssRUFBRTtnQkFBRSxPQUFPLEVBQUUsQ0FBQztZQUN6QixNQUFNLEtBQUssR0FBRyxDQUFDLEVBQUUsR0FBRyxFQUFFLENBQUMsR0FBRyxDQUFDLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQztZQUNwQyxPQUFPLEVBQUUsR0FBRyxLQUFLLEdBQUcsQ0FBQyxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUM7U0FDOUI7S0FDRjtJQUVELG9FQUFvRTtJQUNwRSxPQUFPLENBQUMsQ0FBQztBQUNYLENBQUM7QUFFRCxtRkFBbUY7QUFDbkYsTUFBTSxVQUFVLEdBQUcsQ0FBQyxTQUF1QixFQUFFLE9BQW9CO0lBQy9ELElBQUksT0FBTyxDQUFDLE1BQU0sS0FBSyxDQUFDO1FBQ3RCLE1BQU0sSUFBSSxLQUFLLENBQUMsMENBQTBDLENBQUMsQ0FBQztJQUU5RCxNQUFNLGdCQUFnQixHQUFHLFNBQVMsQ0FBQyxPQUFPLENBQUMsYUFBYSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMscUJBQXFCO0lBQ3RGLE1BQU0sWUFBWSxHQUFHLEVBQUUsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGdCQUFnQixFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUUxRSxNQUFNLHFCQUFxQixHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRTtRQUNuRCxNQUFNLEdBQUcsR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDLHNCQUFzQixDQUFDLENBQUM7UUFDbEQsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBeUIsQ0FBQztJQUNqRCxDQUFDLENBQUMsQ0FBQztJQUVILFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRTtRQUN0QixJQUFJLFVBQVUsR0FBRyxDQUFDLENBQUM7UUFDbkIsSUFBSSxRQUFRLEdBQUcsQ0FBQyxDQUFDO1FBRWpCLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxPQUFPLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFO1lBQ3ZDLE1BQU0sWUFBWSxHQUFHLHFCQUFxQixDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQzlDLE1BQU0sS0FBSyxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDaEMsTUFBTSxLQUFLLEdBQUcsaUJBQWlCLENBQUMsS0FBSyxFQUFFLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUMxRCxVQUFVLElBQUksWUFBWSxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUM7WUFDMUMsUUFBUSxJQUFJLFlBQVksQ0FBQyxNQUFNLENBQUM7U0FDakM7UUFFRCxPQUFPLEdBQUcsR0FBRyxDQUFDLFVBQVUsR0FBRyxRQUFRLENBQUMsQ0FBQztJQUN2QyxDQUFDLENBQUMsQ0FBQztJQUVILDhCQUE4QjtJQUM5QixTQUFTLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUNwQyxPQUFPLFlBQVksQ0FBQztBQUN0QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUgZ3VhcmQtZm9yLWluICovXG4vKiBlc2xpbnQtZGlzYWJsZSBtYXgtbGVuICovXG5pbXBvcnQgKiBhcyBERyBmcm9tICdkYXRhZ3Jvay1hcGkvZGcnO1xuXG4vLy8gQW4gYXJyYXkgb2YgW3gsIHldIHBvaW50cyByZXByZXNlbnRpbmcgdGhlIGRlc2lyYWJpbGl0eSBsaW5lXG4vLy8gW3gsIHldIHBhaXJzIGFyZSBzb3J0ZWQgYnkgeCBpbiBhc2NlbmRpbmcgb3JkZXJcbmV4cG9ydCB0eXBlIERlc2lyYWJpbGl0eUxpbmUgPSBudW1iZXJbXVtdO1xuXG4vLy8gQSBkZXNpcmFiaWxpdHkgbGluZSB3aXRoIGl0cyB3ZWlnaHRcbmV4cG9ydCB0eXBlIFByb3BlcnR5RGVzaXJhYmlsaXR5ID0ge1xuICBsaW5lOiBEZXNpcmFiaWxpdHlMaW5lO1xuICBtaW4/OiBudW1iZXI7IC8vLyBtaW4gdmFsdWUgb2YgdGhlIHByb3BlcnR5IChvcHRpb25hbDsgdXNlZCBmb3IgZWRpdGluZyB0aGUgbGluZSlcbiAgbWF4PzogbnVtYmVyOyAvLy8gbWF4IHZhbHVlIG9mIHRoZSBwcm9wZXJ0eSAob3B0aW9uYWw7IHVzZWQgZm9yIGVkaXRpbmcgdGhlIGxpbmUpXG4gIHdlaWdodDogbnVtYmVyOyAvLy8gMC0xXG59XG5cbi8vLyBBIG1hcCBvZiBkZXNpcmFiaWxpdHkgbGluZXMgd2l0aCB0aGVpciB3ZWlnaHRzXG5leHBvcnQgdHlwZSBEZXNpcmFiaWxpdHlQcm9maWxlID0ge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIHByb3BlcnRpZXM6IHsgW2tleTogc3RyaW5nXTogUHJvcGVydHlEZXNpcmFiaWxpdHkgfTtcbn1cblxuLy8vIENhbGN1bGF0ZXMgdGhlIGRlc2lyYWJpbGl0eSBzY29yZSBmb3IgYSBnaXZlbiB4IHZhbHVlXG4vLy8gUmV0dXJucyAwIGlmIHggaXMgb3V0c2lkZSB0aGUgcmFuZ2Ugb2YgdGhlIGRlc2lyYWJpbGl0eSBsaW5lXG4vLy8gT3RoZXJ3aXNlLCByZXR1cm5zIHRoZSB5IHZhbHVlIG9mIHRoZSBkZXNpcmFiaWxpdHkgbGluZSBhdCB4XG5leHBvcnQgZnVuY3Rpb24gZGVzaXJhYmlsaXR5U2NvcmUoeDogbnVtYmVyLCBkZXNpcmFiaWxpdHlMaW5lOiBEZXNpcmFiaWxpdHlMaW5lKTogbnVtYmVyIHtcbiAgLy8gSWYgdGhlIGxpbmUgaXMgZW1wdHkgb3IgeCBpcyBvdXRzaWRlIHRoZSByYW5nZSwgcmV0dXJuIDBcbiAgaWYgKGRlc2lyYWJpbGl0eUxpbmUubGVuZ3RoID09PSAwIHx8IHggPCBkZXNpcmFiaWxpdHlMaW5lWzBdWzBdIHx8IHggPiBkZXNpcmFiaWxpdHlMaW5lW2Rlc2lyYWJpbGl0eUxpbmUubGVuZ3RoIC0gMV1bMF0pXG4gICAgcmV0dXJuIDA7XG5cbiAgLy8gRmluZCB0aGUgdHdvIHBvaW50cyB0aGF0IHggbGllcyBiZXR3ZWVuXG4gIGZvciAobGV0IGkgPSAwOyBpIDwgZGVzaXJhYmlsaXR5TGluZS5sZW5ndGggLSAxOyBpKyspIHtcbiAgICBjb25zdCBbeDEsIHkxXSA9IGRlc2lyYWJpbGl0eUxpbmVbaV07XG4gICAgY29uc3QgW3gyLCB5Ml0gPSBkZXNpcmFiaWxpdHlMaW5lW2kgKyAxXTtcblxuICAgIGlmICh4ID49IHgxICYmIHggPD0geDIpIHtcbiAgICAgIC8vIExpbmVhciBpbnRlcnBvbGF0aW9uIGJldHdlZW4gdGhlIHR3byBwb2ludHNcbiAgICAgIGlmICh4MSA9PT0geDIpIHJldHVybiB5MTtcbiAgICAgIGNvbnN0IHNsb3BlID0gKHkyIC0geTEpIC8gKHgyIC0geDEpO1xuICAgICAgcmV0dXJuIHkxICsgc2xvcGUgKiAoeCAtIHgxKTtcbiAgICB9XG4gIH1cblxuICAvLyBTaG91bGQgbm90IGhhcHBlbiBpZiB4IGlzIHdpdGhpbiBib3VuZHMsIGJ1dCByZXR1cm4gMCBhcyBmYWxsYmFja1xuICByZXR1cm4gMDtcbn1cblxuLyoqIENhbGN1bGF0ZXMgdGhlIG11bHRpIHBhcmFtZXRlciBvcHRpbWl6YXRpb24gc2NvcmUsIDAtMTAwLCAxMDAgaXMgdGhlIG1heGltdW0gKi9cbmV4cG9ydCBmdW5jdGlvbiBtcG8oZGF0YUZyYW1lOiBERy5EYXRhRnJhbWUsIGNvbHVtbnM6IERHLkNvbHVtbltdKTogREcuQ29sdW1uIHtcbiAgaWYgKGNvbHVtbnMubGVuZ3RoID09PSAwKVxuICAgIHRocm93IG5ldyBFcnJvcignTm8gY29sdW1ucyBwcm92aWRlZCBmb3IgTVBPIGNhbGN1bGF0aW9uLicpO1xuXG4gIGNvbnN0IHJlc3VsdENvbHVtbk5hbWUgPSBkYXRhRnJhbWUuY29sdW1ucy5nZXRVbnVzZWROYW1lKCdNUE8nKTsgLy8gRW5zdXJlIHVuaXF1ZSBuYW1lXG4gIGNvbnN0IHJlc3VsdENvbHVtbiA9IERHLkNvbHVtbi5mbG9hdChyZXN1bHRDb2x1bW5OYW1lLCBjb2x1bW5zWzBdLmxlbmd0aCk7XG5cbiAgY29uc3QgZGVzaXJhYmlsaXR5VGVtcGxhdGVzID0gY29sdW1ucy5tYXAoKGNvbHVtbikgPT4ge1xuICAgIGNvbnN0IHRhZyA9IGNvbHVtbi5nZXRUYWcoJ2Rlc2lyYWJpbGl0eVRlbXBsYXRlJyk7XG4gICAgcmV0dXJuIEpTT04ucGFyc2UodGFnKSBhcyBQcm9wZXJ0eURlc2lyYWJpbGl0eTtcbiAgfSk7XG5cbiAgcmVzdWx0Q29sdW1uLmluaXQoKGkpID0+IHtcbiAgICBsZXQgdG90YWxTY29yZSA9IDA7XG4gICAgbGV0IG1heFNjb3JlID0gMDtcblxuICAgIGZvciAobGV0IGogPSAwOyBqIDwgY29sdW1ucy5sZW5ndGg7IGorKykge1xuICAgICAgY29uc3QgZGVzaXJhYmlsaXR5ID0gZGVzaXJhYmlsaXR5VGVtcGxhdGVzW2pdO1xuICAgICAgY29uc3QgdmFsdWUgPSBjb2x1bW5zW2pdLmdldChpKTtcbiAgICAgIGNvbnN0IHNjb3JlID0gZGVzaXJhYmlsaXR5U2NvcmUodmFsdWUsIGRlc2lyYWJpbGl0eS5saW5lKTtcbiAgICAgIHRvdGFsU2NvcmUgKz0gZGVzaXJhYmlsaXR5LndlaWdodCAqIHNjb3JlO1xuICAgICAgbWF4U2NvcmUgKz0gZGVzaXJhYmlsaXR5LndlaWdodDtcbiAgICB9XG5cbiAgICByZXR1cm4gMTAwICogKHRvdGFsU2NvcmUgLyBtYXhTY29yZSk7XG4gIH0pO1xuXG4gIC8vIEFkZCB0aGUgY29sdW1uIHRvIHRoZSB0YWJsZVxuICBkYXRhRnJhbWUuY29sdW1ucy5hZGQocmVzdWx0Q29sdW1uKTtcbiAgcmV0dXJuIHJlc3VsdENvbHVtbjtcbn1cbiJdfQ==
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mpo.js","sourceRoot":"","sources":["mpo.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable guard-for-in */\n/* eslint-disable max-len */\nimport * as DG from 'datagrok-api/dg';\n\n/// An array of [x, y] points representing the desirability line\n/// [x, y] pairs are sorted by x in ascending order\nexport type DesirabilityLine = number[][];\n\nexport type DesirabilityMode = 'freeform' | 'gaussian' | 'sigmoid';\n\nexport type MissingValueConfig =\n  | { strategy: 'exclude' }\n  | { strategy: 'default'; score: number }\n  | { strategy: 'skip' };\n\ntype BasePropertyDesirability = {\n  weight: number; /// 0-1\n  weightColumn?: string; /// optional column name to read per-row weights from (0-1)\n  missingValues?: MissingValueConfig;\n}\n\nexport type NumericalDesirability = BasePropertyDesirability & {\n  functionType: 'numerical';\n  line: DesirabilityLine;\n  min?: number; /// min value of the property (optional; used for editing the line)\n  max?: number; /// max value of the property (optional; used for editing the line)\n\n  mode?: DesirabilityMode;\n\n  /// Gaussian mode parameters\n  mean?: number;\n  sigma?: number;\n\n  /// Sigmoid mode parameters\n  x0?: number;\n  k?: number;\n\n  freeformLine?: DesirabilityLine;\n}\n\nexport type CategoricalDesirability = BasePropertyDesirability & {\n  functionType: 'categorical';\n  categories: { name: string; desirability: number }[];\n}\n\nexport type PropertyDesirability = NumericalDesirability | CategoricalDesirability;\n\nexport function isNumerical(p: PropertyDesirability): p is NumericalDesirability {\n  return p.functionType === 'numerical';\n}\n\nexport function createDefaultNumerical(weight = 1, min = 0, max = 1): NumericalDesirability {\n  return {functionType: 'numerical', weight, mode: 'freeform', min, max, line: []};\n}\n\nexport function createDefaultCategorical(\n  weight = 1,\n  categories?: {name: string; desirability: number}[],\n): CategoricalDesirability {\n  return {functionType: 'categorical', weight, categories: categories ?? [{name: 'Category 1', desirability: 1}]};\n}\n\nexport function migrateDesirability(raw: any): PropertyDesirability {\n  if (raw.functionType)\n    return raw;\n  if (raw.categories)\n    return {...raw, functionType: 'categorical'};\n  return {...raw, functionType: 'numerical'};\n}\n\nexport const DESIRABILITY_PROFILE_TYPE = 'MPO Desirability Profile';\n\n/// A map of desirability lines with their weights\nexport type DesirabilityProfile = {\n  type: typeof DESIRABILITY_PROFILE_TYPE;\n  name: string;\n  description: string;\n  aggregation?: WeightedAggregation;\n  properties: { [key: string]: PropertyDesirability };\n}\n\nexport function isDesirabilityProfile(x: any): x is DesirabilityProfile {\n  return x != null && typeof x === 'object' && x.type === DESIRABILITY_PROFILE_TYPE;\n}\n\nexport const WEIGHTED_AGGREGATIONS = ['Average', 'Sum', 'Product', 'Geomean', 'Min', 'Max'] as const;\nexport const WEIGHTED_AGGREGATIONS_LIST: WeightedAggregation[] = [...WEIGHTED_AGGREGATIONS];\nexport type WeightedAggregation = typeof WEIGHTED_AGGREGATIONS[number];\nexport const DEFAULT_AGGREGATION: WeightedAggregation = 'Average';\n\n/// Calculates the desirability score for a given x value\n/// Returns 0 if x is outside the range of the desirability line\n/// Otherwise, returns the y value of the desirability line at x\nexport function desirabilityScore(x: number, desirabilityLine: DesirabilityLine): number {\n  // If the line is empty or x is outside the range, return 0\n  if (desirabilityLine.length === 0 || x < desirabilityLine[0][0] || x > desirabilityLine[desirabilityLine.length - 1][0])\n    return 0;\n\n  // Find the two points that x lies between\n  for (let i = 0; i < desirabilityLine.length - 1; i++) {\n    const [x1, y1] = desirabilityLine[i];\n    const [x2, y2] = desirabilityLine[i + 1];\n\n    if (x >= x1 && x <= x2) {\n      // Linear interpolation between the two points\n      if (x1 === x2) return y1;\n      const slope = (y2 - y1) / (x2 - x1);\n      return y1 + slope * (x - x1);\n    }\n  }\n\n  // Should not happen if x is within bounds, but return 0 as fallback\n  return 0;\n}\n\nexport function categoricalDesirabilityScore(\n  value: string,\n  prop: CategoricalDesirability,\n): number | null {\n  const found = prop.categories.find((c) => c.name === value);\n  return found?.desirability ?? null;\n}\n\nexport type MpoResult = {\n  scoreColumn: DG.Column;\n  desirabilityColumns?: DG.Column[];\n}\n\n/** Calculates the multi parameter optimization score, 0-100, 100 is the maximum */\nexport function mpo(\n  dataFrame: DG.DataFrame,\n  columns: DG.Column[],\n  profileName: string,\n  aggregation: WeightedAggregation,\n  isDifferent: boolean = false,\n  createDesirabilityColumns: boolean = false,\n): MpoResult {\n  if (columns.length === 0)\n    throw new Error('No columns provided for MPO calculation.');\n\n  const rowCount = columns[0].length;\n  const resultColumn = isDifferent ?\n    DG.Column.float(profileName, rowCount) :\n    (dataFrame.col(profileName) ?? DG.Column.float(profileName, rowCount));\n\n  const desirabilityTemplates: PropertyDesirability[] = [];\n  const weightColumns: (DG.Column | null)[] = [];\n  for (const column of columns) {\n    const tag = column.getTag('desirabilityTemplate');\n    const d = migrateDesirability(JSON.parse(tag));\n    desirabilityTemplates.push(d);\n    weightColumns.push(d.weightColumn ? dataFrame.col(d.weightColumn) ?? null : null);\n  }\n\n  const desirabilityColumns = createDesirabilityColumns ?\n    columns.map((col) => DG.Column.float(`${col.name} (${profileName} desirability)`, rowCount)) : undefined;\n\n  resultColumn.init((i) => {\n    const scores: number[] = [];\n    const weights: number[] = [];\n\n    for (let j = 0; j < columns.length; j++) {\n      const desirability = desirabilityTemplates[j];\n      const value = columns[j].get(i);\n\n      let score: number | null;\n\n      if (columns[j].isNone(i)) {\n        const mv = desirability.missingValues;\n        if (!mv || mv.strategy === 'exclude')\n          return NaN;\n        if (mv.strategy === 'skip')\n          continue;\n        score = mv.score;\n      }\n      else {\n        score = isNumerical(desirability) ?\n          desirabilityScore(value, desirability.line) :\n          categoricalDesirabilityScore(String(value), desirability);\n      }\n\n      if (score === null)\n        return NaN;\n\n      if (createDesirabilityColumns)\n        desirabilityColumns![j].set(i, score, false);\n\n      scores.push(score);\n      const wCol = weightColumns[j];\n      const w = wCol && !wCol.isNone(i) ? Math.max(0, Math.min(1, wCol.get(i))) : desirability.weight;\n      weights.push(w);\n    }\n\n    if (scores.length === 0)\n      return NaN;\n\n    return aggregate(scores, weights, aggregation);\n  });\n\n  return {scoreColumn: resultColumn, desirabilityColumns};\n}\n\nexport function aggregate(scores: number[], weights: number[], aggregation: WeightedAggregation): number {\n  switch (aggregation) {\n  case 'Sum':\n    return scores.reduce((sum, s, idx) => sum + s * weights[idx], 0);\n\n  case 'Average':\n    const totalWeight = weights.reduce((sum, w) => sum + w, 0);\n    return scores.reduce((sum, s, idx) => sum + s * weights[idx], 0) / totalWeight;\n\n  case 'Product':\n    return scores.reduce((prod, s, idx) => prod * Math.pow(s, weights[idx]), 1);\n\n  case 'Geomean':\n    const totalW = weights.reduce((sum, w) => sum + w, 0);\n    return scores.reduce((prod, s, idx) => prod * Math.pow(s, weights[idx] / totalW), 1);\n\n  case 'Min':\n    return Math.min(...scores.map((s, idx) => Math.pow(s, weights[idx])));\n\n  case 'Max':\n    return Math.max(...scores.map((s, idx) => Math.pow(s, weights[idx])));\n\n  default:\n    throw new Error(`Unknown aggregation type: ${aggregation}`);\n  }\n}\n"]}
@@ -0,0 +1,6 @@
1
+ export declare const MPO_SCORE_CHANGED_EVENT = "grok-mpo-score-changed";
2
+ export declare const MPO_PROFILE_CHANGED_EVENT = "chem-mpo-profile-changed";
3
+ export declare const MPO_PROFILE_DELETED_EVENT = "chem-mpo-profile-deleted";
4
+ export declare function getNextAvailable(base: string, existing: Set<string>, format: (base: string, num?: number) => string): string;
5
+ export declare function generateMpoFileName(profileName: string, existingFileNames: Set<string>): string;
6
+ //# sourceMappingURL=utils.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["utils.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,uBAAuB,2BAA2B,CAAC;AAChE,eAAO,MAAM,yBAAyB,6BAA6B,CAAC;AACpE,eAAO,MAAM,yBAAyB,6BAA6B,CAAC;AAEpE,wBAAgB,gBAAgB,CAC9B,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,GAAG,CAAC,MAAM,CAAC,EACrB,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,MAAM,KAAK,MAAM,GAC7C,MAAM,CAUR;AAED,wBAAgB,mBAAmB,CAAC,WAAW,EAAE,MAAM,EAAE,iBAAiB,EAAE,GAAG,CAAC,MAAM,CAAC,GAAG,MAAM,CAG/F"}
@@ -0,0 +1,17 @@
1
+ export const MPO_SCORE_CHANGED_EVENT = 'grok-mpo-score-changed';
2
+ export const MPO_PROFILE_CHANGED_EVENT = 'chem-mpo-profile-changed';
3
+ export const MPO_PROFILE_DELETED_EVENT = 'chem-mpo-profile-deleted';
4
+ export function getNextAvailable(base, existing, format) {
5
+ const first = format(base);
6
+ if (!existing.has(first))
7
+ return first;
8
+ let num = 2;
9
+ while (existing.has(format(base, num)))
10
+ num++;
11
+ return format(base, num);
12
+ }
13
+ export function generateMpoFileName(profileName, existingFileNames) {
14
+ const base = profileName.trim().replace(/[\s/\\]+/g, '-').replace(/^-|-$/g, '') || 'profile';
15
+ return getNextAvailable(base, existingFileNames, (b, n) => n ? `${b}-${n}.json` : `${b}.json`);
16
+ }
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ1dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQUMsTUFBTSx1QkFBdUIsR0FBRyx3QkFBd0IsQ0FBQztBQUNoRSxNQUFNLENBQUMsTUFBTSx5QkFBeUIsR0FBRywwQkFBMEIsQ0FBQztBQUNwRSxNQUFNLENBQUMsTUFBTSx5QkFBeUIsR0FBRywwQkFBMEIsQ0FBQztBQUVwRSxNQUFNLFVBQVUsZ0JBQWdCLENBQzlCLElBQVksRUFDWixRQUFxQixFQUNyQixNQUE4QztJQUU5QyxNQUFNLEtBQUssR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDM0IsSUFBSSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDO1FBQ3RCLE9BQU8sS0FBSyxDQUFDO0lBRWYsSUFBSSxHQUFHLEdBQUcsQ0FBQyxDQUFDO0lBQ1osT0FBTyxRQUFRLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLENBQUM7UUFDcEMsR0FBRyxFQUFFLENBQUM7SUFFUixPQUFPLE1BQU0sQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLENBQUM7QUFDM0IsQ0FBQztBQUVELE1BQU0sVUFBVSxtQkFBbUIsQ0FBQyxXQUFtQixFQUFFLGlCQUE4QjtJQUNyRixNQUFNLElBQUksR0FBRyxXQUFXLENBQUMsSUFBSSxFQUFFLENBQUMsT0FBTyxDQUFDLFdBQVcsRUFBRSxHQUFHLENBQUMsQ0FBQyxPQUFPLENBQUMsUUFBUSxFQUFFLEVBQUUsQ0FBQyxJQUFJLFNBQVMsQ0FBQztJQUM3RixPQUFPLGdCQUFnQixDQUFDLElBQUksRUFBRSxpQkFBaUIsRUFBRSxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQztBQUNqRyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNvbnN0IE1QT19TQ09SRV9DSEFOR0VEX0VWRU5UID0gJ2dyb2stbXBvLXNjb3JlLWNoYW5nZWQnO1xuZXhwb3J0IGNvbnN0IE1QT19QUk9GSUxFX0NIQU5HRURfRVZFTlQgPSAnY2hlbS1tcG8tcHJvZmlsZS1jaGFuZ2VkJztcbmV4cG9ydCBjb25zdCBNUE9fUFJPRklMRV9ERUxFVEVEX0VWRU5UID0gJ2NoZW0tbXBvLXByb2ZpbGUtZGVsZXRlZCc7XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXROZXh0QXZhaWxhYmxlKFxuICBiYXNlOiBzdHJpbmcsXG4gIGV4aXN0aW5nOiBTZXQ8c3RyaW5nPixcbiAgZm9ybWF0OiAoYmFzZTogc3RyaW5nLCBudW0/OiBudW1iZXIpID0+IHN0cmluZyxcbik6IHN0cmluZyB7XG4gIGNvbnN0IGZpcnN0ID0gZm9ybWF0KGJhc2UpO1xuICBpZiAoIWV4aXN0aW5nLmhhcyhmaXJzdCkpXG4gICAgcmV0dXJuIGZpcnN0O1xuXG4gIGxldCBudW0gPSAyO1xuICB3aGlsZSAoZXhpc3RpbmcuaGFzKGZvcm1hdChiYXNlLCBudW0pKSlcbiAgICBudW0rKztcblxuICByZXR1cm4gZm9ybWF0KGJhc2UsIG51bSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZW5lcmF0ZU1wb0ZpbGVOYW1lKHByb2ZpbGVOYW1lOiBzdHJpbmcsIGV4aXN0aW5nRmlsZU5hbWVzOiBTZXQ8c3RyaW5nPik6IHN0cmluZyB7XG4gIGNvbnN0IGJhc2UgPSBwcm9maWxlTmFtZS50cmltKCkucmVwbGFjZSgvW1xccy9cXFxcXSsvZywgJy0nKS5yZXBsYWNlKC9eLXwtJC9nLCAnJykgfHwgJ3Byb2ZpbGUnO1xuICByZXR1cm4gZ2V0TmV4dEF2YWlsYWJsZShiYXNlLCBleGlzdGluZ0ZpbGVOYW1lcywgKGIsIG4pID0+IG4gPyBgJHtifS0ke259Lmpzb25gIDogYCR7Yn0uanNvbmApO1xufVxuIl19
@@ -1,10 +0,0 @@
1
- import { DesirabilityLine, PropertyDesirability } from './mpo';
2
- import { Subject } from 'rxjs';
3
- export declare class MpoDesirabilityLineEditor {
4
- root: HTMLDivElement;
5
- onChanged: Subject<unknown>;
6
- private _prop;
7
- constructor(prop: PropertyDesirability, width: number, height: number);
8
- get line(): DesirabilityLine;
9
- }
10
- //# sourceMappingURL=mpo-line-editor.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"mpo-line-editor.d.ts","sourceRoot":"","sources":["mpo-line-editor.ts"],"names":[],"mappings":"AAKA,OAAO,EAAC,gBAAgB,EAAE,oBAAoB,EAAC,MAAM,OAAO,CAAC;AAC7D,OAAO,EAAC,OAAO,EAAC,MAAM,MAAM,CAAC;AAO7B,qBAAa,yBAAyB;IACpC,IAAI,iBAAY;IAChB,SAAS,mBAAiB;IAC1B,OAAO,CAAC,KAAK,CAAuB;gBAExB,IAAI,EAAE,oBAAoB,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM;IA6PrE,IAAI,IAAI,IAAI,gBAAgB,CAE3B;CACF"}