@datagrok-libraries/bio 0.0.6 → 0.0.8

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/.eslintrc.json CHANGED
@@ -6,10 +6,14 @@
6
6
  "extends": [
7
7
  "google"
8
8
  ],
9
+ "parser": "@typescript-eslint/parser",
9
10
  "parserOptions": {
10
11
  "ecmaVersion": 12,
11
12
  "sourceType": "module"
12
13
  },
14
+ "plugins": [
15
+ "@typescript-eslint"
16
+ ],
13
17
  "rules": {
14
18
  "indent": [
15
19
  "error",
@@ -19,11 +23,29 @@
19
23
  "error",
20
24
  120
21
25
  ],
26
+ "no-unused-vars": "warn",
22
27
  "require-jsdoc": "off",
23
- "spaced-comment": "off"
24
- },
25
- "parser": "@typescript-eslint/parser",
26
- "plugins": [
27
- "@typescript-eslint"
28
- ]
29
- }
28
+ "spaced-comment": "off",
29
+ "linebreak-style": "off",
30
+ "curly": [
31
+ "error",
32
+ "multi-or-nest",
33
+ "consistent"
34
+ ],
35
+ "brace-style": [
36
+ "error",
37
+ "1tbs",
38
+ {
39
+ "allowSingleLine": true
40
+ }
41
+ ],
42
+ "block-spacing": [2, "always"],
43
+ "comma-dangle": ["error", {
44
+ "arrays": "only-multiline",
45
+ "functions": "never",
46
+ "objects": "only-multiline",
47
+ "imports": "only-multiline"
48
+ }],
49
+ "guard-for-in": "off"
50
+ }
51
+ }
package/detectors.js CHANGED
@@ -1,3 +1,9 @@
1
+ import * as grok from 'datagrok-api/grok';
2
+ import * as ui from 'datagrok-api/ui';
3
+ import * as DG from 'datagrok-api/dg';
4
+ import {Aminoacids} from './src/aminoacids';
5
+ import {Nucleotides} from './src/nucleotides';
6
+
1
7
  /**
2
8
  * The class contains semantic type detectors.
3
9
  * Detectors are functions tagged with `DG.FUNC_TYPES.SEM_TYPE_DETECTOR`.
@@ -6,4 +12,45 @@
6
12
  * Follow this naming convention to ensure that your detectors are properly loaded.
7
13
  */
8
14
  class BioPackageDetectors extends DG.Package {
15
+ //tags: semTypeDetector
16
+ //input: column col
17
+ //output: string semType
18
+ detectNucleotidesSeq(col) {
19
+ const alphabet = Nucleotides.Names + {'-': 'gap'};
20
+ return DG.Detector.sampleCategories(col, (s) => {
21
+ return s.split('').every((n) => n in alphabet);
22
+ }) ? Nucleotides.SemType : null;
23
+ }
24
+
25
+ //tags: semTypeDetector
26
+ //input: column col
27
+ //output: string semType
28
+ detectAminoacidsSeq(col) {
29
+ const alphabet = Aminoacids.Names + {'-': 'gap'};
30
+ return DG.Detector.sampleCategories(col, (s) => {
31
+ return s.split('').every((aa) => aa in alphabet);
32
+ }) ? Aminoacids.SemType : null;
33
+ }
34
+
35
+ //tags: semTypeDetector
36
+ //input: column col
37
+ //output: string semType
38
+ detectMultipleAlignmentNucleotidesSeq(col) {
39
+ const alphabet = Nucleotides.Names + {'-': 'gap'};
40
+ const len = col.get(0).length;
41
+ return DG.Detector.sampleCategories(col, (s) => {
42
+ return s.length == len && s.split('').every((n) => n in alphabet);
43
+ }) ? Nucleotides.SemTypeMultipleAlignment : null;
44
+ }
45
+
46
+ //tags: semTypeDetector
47
+ //input: column col
48
+ //output: string semType
49
+ detectMultipleAlignmentAminoacidsSeq(col) {
50
+ const alphabet = Aminoacids.Names + {'-': 'gap'};
51
+ const len = col.get(0).length;
52
+ return DG.Detector.sampleCategories(col, (s) => {
53
+ return s.length == len && s.split('').every((aa) => aa in alphabet);
54
+ }) ? Aminoacids.SemTypeMultipleAlignment : null;
55
+ }
9
56
  }
package/package.json CHANGED
@@ -5,23 +5,23 @@
5
5
  },
6
6
  "beta": true,
7
7
  "friendlyName": "Datagrok bio library",
8
- "version": "0.0.6",
8
+ "version": "0.0.8",
9
9
  "description": "",
10
+ "main": "src/index.ts",
10
11
  "dependencies": {
11
- "datagrok-api": ">=0.108.0",
12
12
  "@datagrok-libraries/utils": ">=0.0.22",
13
13
  "cash-dom": "latest",
14
- "dayjs": "latest"
14
+ "datagrok-api": ">=0.108.0",
15
+ "dayjs": "latest",
16
+ "rxjs": "^6.5.5"
15
17
  },
16
18
  "devDependencies": {
17
- "typescript": "^4.2.3",
18
- "ts-loader": "^9.2.6",
19
- "webpack": "latest",
20
- "webpack-cli": "latest",
21
19
  "@typescript-eslint/eslint-plugin": "^4.29.1",
22
20
  "@typescript-eslint/parser": "^4.29.1",
23
- "eslint": "^7.32.0",
24
- "eslint-config-google": "^0.14.0"
21
+ "eslint": "latest",
22
+ "eslint-config-google": "latest",
23
+ "ts-loader": "^9.2.6",
24
+ "typescript": "^4.2.3"
25
25
  },
26
26
  "scripts": {
27
27
  "link": "npm link",
@@ -36,7 +36,7 @@
36
36
  "release-bio-public": "grok publish public --rebuild --release",
37
37
  "debug-bio-local": "grok publish local --rebuild",
38
38
  "release-bio-local": "grok publish local --rebuild --release",
39
- "lint": "eslint src --ext .ts",
40
- "lint-fix": "eslint src --ext .ts --fix"
39
+ "lint": "eslint \"./src/**/*.ts\"",
40
+ "lint-fix": "eslint \"./src/**/*.ts\" --fix"
41
41
  }
42
42
  }
@@ -0,0 +1,31 @@
1
+ import { StringDictionary } from '@datagrok-libraries/utils/src/type-declarations';
2
+ import { SeqPalette, SeqPaletteBase } from './seq-palettes';
3
+ export declare class AminoacidsPalettes extends SeqPaletteBase {
4
+ private static lesk;
5
+ static get Lesk(): SeqPalette;
6
+ private static grokGroups;
7
+ static get GrokGroups(): SeqPalette;
8
+ private static rasMol;
9
+ static get RasMol(): SeqPalette;
10
+ }
11
+ export declare class Aminoacids {
12
+ static readonly SemType: string;
13
+ static readonly SemTypeMultipleAlignment: string;
14
+ static undefinedColor: string;
15
+ static Names: StringDictionary;
16
+ static AASmiles: StringDictionary;
17
+ static AASmilesTruncated: StringDictionary;
18
+ /** TODO: Full?
19
+ */
20
+ static AAFullNames: StringDictionary;
21
+ static getPalette(scheme?: string): SeqPalette;
22
+ /**
23
+ * Returns divided amino acid with its content in the bracket, if the content is number, then its omitted
24
+ *
25
+ * @param {string} c raw amino
26
+ * @return {[string, string]} outer and inner content
27
+ */
28
+ static getInnerOuter(c: string): [string, string];
29
+ static getColorAAPivot(monomer?: string, scheme?: 'grok'): [string, string, string, number];
30
+ }
31
+ //# sourceMappingURL=aminoacids.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"aminoacids.d.ts","sourceRoot":"","sources":["aminoacids.ts"],"names":[],"mappings":"AAIA,OAAO,EAAC,gBAAgB,EAAC,MAAM,iDAAiD,CAAC;AACjF,OAAO,EAAC,UAAU,EAAE,cAAc,EAAC,MAAM,gBAAgB,CAAC;AAE1D,qBAAa,kBAAmB,SAAQ,cAAc;IACpD,OAAO,CAAC,MAAM,CAAC,IAAI,CAAa;IAEhC,WAAkB,IAAI,IAAI,UAAU,CAWnC;IAED,OAAO,CAAC,MAAM,CAAC,UAAU,CAAa;IAEtC,WAAkB,UAAU,IAAI,UAAU,CAYzC;IAED,OAAO,CAAC,MAAM,CAAC,MAAM,CAAa;IAElC,WAAkB,MAAM,IAAI,UAAU,CA4BrC;CACF;AAED,qBAAa,UAAU;IACrB,MAAM,CAAC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAgB;IAE/C,MAAM,CAAC,QAAQ,CAAC,wBAAwB,EAAE,MAAM,CAAiC;IAEjF,MAAM,CAAC,cAAc,SAAsB;IAE3C,OAAc,KAAK,EAAE,gBAAgB,CAqBnC;IAEF,OAAc,QAAQ,EAAE,gBAAgB,CAqBtC;IAEF,OAAc,iBAAiB,EAAE,gBAAgB,CAqB/C;IAEF;OACG;IACH,OAAc,WAAW,EAAE,gBAAgB,CAqBzC;WAEY,UAAU,CAAC,MAAM,GAAE,MAAe,GAAG,UAAU;IAW7D;;;;;OAKG;WACW,aAAa,CAAC,CAAC,EAAE,MAAM,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC;WAmB1C,eAAe,CAAC,OAAO,GAAE,MAAW,EAAE,MAAM,GAAE,MAAe,GAAG,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC;CA6C/G"}
@@ -0,0 +1,226 @@
1
+ import { SeqPaletteBase } from './seq-palettes';
2
+ export class AminoacidsPalettes extends SeqPaletteBase {
3
+ static get Lesk() {
4
+ if (this.lesk === void 0) {
5
+ this.lesk = this.makePalette([
6
+ [['G', 'A', 'S', 'T'], 'orange'],
7
+ [['C', 'V', 'I', 'L', 'P', 'F', 'Y', 'M', 'W'], 'all_green'],
8
+ [['N', 'Q', 'H'], 'magenta'],
9
+ [['D', 'E'], 'red'],
10
+ [['K', 'R'], 'all_blue'],
11
+ ]);
12
+ }
13
+ return this.lesk;
14
+ }
15
+ static get GrokGroups() {
16
+ if (this.grokGroups === void 0) {
17
+ this.grokGroups = this.makePalette([
18
+ [['C', 'U'], 'yellow'],
19
+ [['G', 'P'], 'red'],
20
+ [['A', 'V', 'I', 'L', 'M', 'F', 'Y', 'W'], 'all_green'],
21
+ [['R', 'H', 'K'], 'light_blue'],
22
+ [['D', 'E'], 'dark_blue'],
23
+ [['S', 'T', 'N', 'Q'], 'orange'],
24
+ ]);
25
+ }
26
+ return this.grokGroups;
27
+ }
28
+ static get RasMol() {
29
+ if (this.rasMol === void 0) {
30
+ this.rasMol = {
31
+ // http://acces.ens-lyon.fr/biotic/rastop/help/colour.htm
32
+ 'D': '#E60A0A',
33
+ 'E': '#E60A0A',
34
+ 'C': '#E6E600',
35
+ 'M': '#E6E600',
36
+ 'K': '#145AFF',
37
+ 'R': '#145AFF',
38
+ 'S': '#FA9600',
39
+ 'T': '#FA9600',
40
+ 'F': '#3232AA',
41
+ 'Y': '#3232AA',
42
+ 'N': '#00DCDC',
43
+ 'Q': '#00DCDC',
44
+ 'G': '#EBEBEB',
45
+ 'L': '#0F820F',
46
+ 'V': '#0F820F',
47
+ 'I': '#0F820F',
48
+ 'A': '#C8C8C8',
49
+ 'W': '#B45AB4',
50
+ 'H': '#8282D2',
51
+ 'P': '#DC9682',
52
+ 'others': '#BEA06E',
53
+ };
54
+ }
55
+ return this.rasMol;
56
+ }
57
+ }
58
+ export class Aminoacids {
59
+ static getPalette(scheme = 'grok') {
60
+ switch (scheme) {
61
+ case 'grok':
62
+ return AminoacidsPalettes.GrokGroups;
63
+ case 'lesk':
64
+ return AminoacidsPalettes.Lesk;
65
+ default:
66
+ throw new Error(`ChemPalette: scheme \`${scheme}\` does not exist`);
67
+ }
68
+ }
69
+ /**
70
+ * Returns divided amino acid with its content in the bracket, if the content is number, then its omitted
71
+ *
72
+ * @param {string} c raw amino
73
+ * @return {[string, string]} outer and inner content
74
+ */
75
+ static getInnerOuter(c) {
76
+ let isInner = 0;
77
+ let inner = '';
78
+ let outer = '';
79
+ for (const char of c) {
80
+ if (char == '(')
81
+ isInner++;
82
+ else if (char == ')')
83
+ isInner--;
84
+ else if (isInner)
85
+ inner += char;
86
+ else
87
+ outer += char;
88
+ }
89
+ return !isNaN(parseInt(inner)) ? [outer, ''] : [outer, inner];
90
+ }
91
+ static getColorAAPivot(monomer = '', scheme = 'grok') {
92
+ var _a, _b, _c;
93
+ //const chemPaletteInstance = AAPalettes.GrokGroups();
94
+ const chemPaletteInstance = this.getPalette(scheme);
95
+ let [outerMonomer, innerMonomer] = this.getInnerOuter(monomer);
96
+ outerMonomer = (outerMonomer.length > 6 ? `${outerMonomer.slice(0, 3)}...` : outerMonomer);
97
+ innerMonomer = (innerMonomer.length > 6 ? `${innerMonomer.slice(0, 3)}...` : innerMonomer);
98
+ if (monomer.length == 1 || monomer[1] == '(') {
99
+ const amino = (_a = monomer[0]) === null || _a === void 0 ? void 0 : _a.toUpperCase();
100
+ return amino in chemPaletteInstance ?
101
+ [chemPaletteInstance[amino], amino, innerMonomer, 1] :
102
+ [this.undefinedColor, outerMonomer, innerMonomer, 1];
103
+ }
104
+ if (monomer[0] == 'd' && monomer[1] in chemPaletteInstance) {
105
+ if (monomer.length == 2 || monomer[2] == '(') {
106
+ const amino = (_b = monomer[1]) === null || _b === void 0 ? void 0 : _b.toUpperCase();
107
+ return amino in chemPaletteInstance ?
108
+ [chemPaletteInstance[amino], amino, innerMonomer, 2] :
109
+ [this.undefinedColor, outerMonomer, innerMonomer, 2];
110
+ }
111
+ }
112
+ if (monomer.substring(0, 3) in this.AAFullNames) {
113
+ if (monomer.length == 3 || monomer[3] == '(') {
114
+ const amino = this.AAFullNames[monomer.substring(0, 3)];
115
+ return amino in chemPaletteInstance ?
116
+ [chemPaletteInstance[amino], amino, innerMonomer, 3] :
117
+ [this.undefinedColor, outerMonomer, innerMonomer, 3];
118
+ }
119
+ }
120
+ if (((_c = monomer[0]) === null || _c === void 0 ? void 0 : _c.toLowerCase()) == monomer[0]) {
121
+ if (monomer.substring(1, 3) in this.AAFullNames) {
122
+ if (monomer.length == 4 || monomer[4] == '(') {
123
+ const amino = this.AAFullNames[monomer.substring(1, 3)];
124
+ return amino in chemPaletteInstance ?
125
+ [chemPaletteInstance[amino], amino, innerMonomer, 4] :
126
+ [this.undefinedColor, outerMonomer, innerMonomer, 4];
127
+ }
128
+ }
129
+ }
130
+ return [this.undefinedColor, outerMonomer, innerMonomer, 0];
131
+ }
132
+ }
133
+ Aminoacids.SemType = 'Aminoacids';
134
+ Aminoacids.SemTypeMultipleAlignment = 'AminoacidsMultipleAlignment';
135
+ Aminoacids.undefinedColor = 'rgb(100,100,100)';
136
+ Aminoacids.Names = {
137
+ 'G': 'Glycine',
138
+ 'L': 'Leucine',
139
+ 'Y': 'Tyrosine',
140
+ 'S': 'Serine',
141
+ 'E': 'Glutamic acid',
142
+ 'Q': 'Glutamine',
143
+ 'D': 'Aspartic acid',
144
+ 'N': 'Asparagine',
145
+ 'F': 'Phenylalanine',
146
+ 'A': 'Alanine',
147
+ 'K': 'Lysine',
148
+ 'R': 'Arginine',
149
+ 'H': 'Histidine',
150
+ 'C': 'Cysteine',
151
+ 'V': 'Valine',
152
+ 'P': 'Proline',
153
+ 'W': 'Tryptophan',
154
+ 'I': 'Isoleucine',
155
+ 'M': 'Methionine',
156
+ 'T': 'Threonine',
157
+ };
158
+ Aminoacids.AASmiles = {
159
+ 'G': 'NCC(=O)O',
160
+ 'L': 'N[C@H](CC(C)C)C(=O)O',
161
+ 'Y': 'NC(CC1=CC=C(O)C=C1)C(=O)O',
162
+ 'S': 'NC(CO)C(=O)O',
163
+ 'E': 'N[C@@H](CCC(O)=O)C(=O)O',
164
+ 'Q': 'N[C@@H](CCC(N)=O)C(=O)O',
165
+ 'D': 'N[C@@H](CC(O)=O)C(=O)O',
166
+ 'N': 'N[C@@H](CC(N)=O)C(=O)O',
167
+ 'F': 'NC(CC1=CC=CC=C1)C(=O)O',
168
+ 'A': 'N[C@H](C)C(=O)O',
169
+ 'K': 'NC(CCCCN)C(=O)O',
170
+ 'R': 'N[C@H](CCCNC(=N)C)C(=O)O',
171
+ 'H': 'NC(CC1=CN=C[N]1)C(=O)O',
172
+ 'C': 'N[C@@H](CS)C(=O)O',
173
+ 'V': 'NC(C(C)C)C(=O)O',
174
+ 'P': 'N(CCC1)C1C(=O)O',
175
+ 'W': 'N[C@@H](Cc1c2ccccc2n([H])c1)C(=O)O',
176
+ 'I': 'N[C@H]([C@H](C)CC)C(=O)O',
177
+ 'M': 'NC(CCSC)C(=O)O',
178
+ 'T': 'NC(C(O)C)C(=O)O',
179
+ };
180
+ Aminoacids.AASmilesTruncated = {
181
+ 'G': '*C*',
182
+ 'L': 'CC(C)C[C@H](*)*',
183
+ 'Y': 'C1=CC(=CC=C1CC(*)*)O',
184
+ 'S': 'OCC(*)C*',
185
+ 'E': '*[C@@H](CCC(O)=O)*',
186
+ 'Q': '*N[C@@H](CCC(N)=O)*',
187
+ 'D': '*[C@@H](CC(O)=O)*',
188
+ 'N': '*[C@@H](CC(N)=O)*',
189
+ 'F': 'C1=CC=C(C=C1)CC(*)*',
190
+ 'A': 'C[C@H](*)*',
191
+ 'K': 'C(CCN)CC(*)*',
192
+ 'R': '*[C@H](CCCNC(=N)C)*',
193
+ 'H': 'C1=C(NC=N1)CC(*)*',
194
+ 'C': 'C([C@@H](*)*)S',
195
+ 'V': 'CC(C)C(*)*',
196
+ 'P': 'C1CCN(*)C1*',
197
+ 'W': '*[C@@H](Cc1c2ccccc2n([H])c1)*',
198
+ 'I': 'CC[C@H](C)[C@H](*)*',
199
+ 'M': 'CSCCC(*)*',
200
+ 'T': 'CC(O)C(*)*',
201
+ };
202
+ /** TODO: Full?
203
+ */
204
+ Aminoacids.AAFullNames = {
205
+ 'Ala': 'A',
206
+ 'Arg': 'R',
207
+ 'Asn': 'N',
208
+ 'Asp': 'D',
209
+ 'Cys': 'C',
210
+ 'Gln': 'Q',
211
+ 'Glu': 'E',
212
+ 'Gly': 'G',
213
+ 'His': 'H',
214
+ 'Ile': 'I',
215
+ 'Leu': 'L',
216
+ 'Lys': 'K',
217
+ 'Met': 'M',
218
+ 'Phe': 'F',
219
+ 'Pro': 'P',
220
+ 'Ser': 'S',
221
+ 'Thr': 'T',
222
+ 'Trp': 'W',
223
+ 'Tyr': 'Y',
224
+ 'Val': 'V',
225
+ };
226
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aminoacids.js","sourceRoot":"","sources":["aminoacids.ts"],"names":[],"mappings":"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","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\n\nimport {StringDictionary} from '@datagrok-libraries/utils/src/type-declarations';\nimport {SeqPalette, SeqPaletteBase} from './seq-palettes';\n\nexport class AminoacidsPalettes extends SeqPaletteBase {\n  private static lesk: SeqPalette;\n\n  public static get Lesk(): SeqPalette {\n    if (this.lesk === void 0) {\n      this.lesk = this.makePalette([\n        [['G', 'A', 'S', 'T'], 'orange'],\n        [['C', 'V', 'I', 'L', 'P', 'F', 'Y', 'M', 'W'], 'all_green'],\n        [['N', 'Q', 'H'], 'magenta'],\n        [['D', 'E'], 'red'],\n        [['K', 'R'], 'all_blue'],\n      ]);\n    }\n    return this.lesk;\n  }\n\n  private static grokGroups: SeqPalette;\n\n  public static get GrokGroups(): SeqPalette {\n    if (this.grokGroups === void 0) {\n      this.grokGroups = this.makePalette([\n        [['C', 'U'], 'yellow'],\n        [['G', 'P'], 'red'],\n        [['A', 'V', 'I', 'L', 'M', 'F', 'Y', 'W'], 'all_green'],\n        [['R', 'H', 'K'], 'light_blue'],\n        [['D', 'E'], 'dark_blue'],\n        [['S', 'T', 'N', 'Q'], 'orange'],\n      ]);\n    }\n    return this.grokGroups;\n  }\n\n  private static rasMol: SeqPalette;\n\n  public static get RasMol(): SeqPalette {\n    if (this.rasMol === void 0) {\n      this.rasMol = {\n        // http://acces.ens-lyon.fr/biotic/rastop/help/colour.htm\n        'D': '#E60A0A', // asp, aspartic acid, asp\n        'E': '#E60A0A', // glu, glutamic acid\n        'C': '#E6E600', // cys, cysteine\n        'M': '#E6E600', // met, methionine\n        'K': '#145AFF', // lys, lysine\n        'R': '#145AFF', // arg, arginine\n        'S': '#FA9600', // ser, serine\n        'T': '#FA9600', // thr, threonine\n        'F': '#3232AA', // phe, phenylalanine\n        'Y': '#3232AA', // tyr, tyrosine\n        'N': '#00DCDC', // asn, asparagine\n        'Q': '#00DCDC', // gln, glutamine\n        'G': '#EBEBEB', // gly, glycine\n        'L': '#0F820F', // leu, leucine\n        'V': '#0F820F', // val, valine\n        'I': '#0F820F', // ile, isoleucine\n        'A': '#C8C8C8', // ala, alanine\n        'W': '#B45AB4', // trp, tryptophan\n        'H': '#8282D2', // his, histidine\n        'P': '#DC9682', // pro, proline\n        'others': '#BEA06E',\n      };\n    }\n    return this.rasMol;\n  }\n}\n\nexport class Aminoacids {\n  static readonly SemType: string = 'Aminoacids';\n\n  static readonly SemTypeMultipleAlignment: string = 'AminoacidsMultipleAlignment';\n\n  static undefinedColor = 'rgb(100,100,100)';\n\n  public static Names: StringDictionary = {\n    'G': 'Glycine',\n    'L': 'Leucine',\n    'Y': 'Tyrosine',\n    'S': 'Serine',\n    'E': 'Glutamic acid',\n    'Q': 'Glutamine',\n    'D': 'Aspartic acid',\n    'N': 'Asparagine',\n    'F': 'Phenylalanine',\n    'A': 'Alanine',\n    'K': 'Lysine',\n    'R': 'Arginine',\n    'H': 'Histidine',\n    'C': 'Cysteine',\n    'V': 'Valine',\n    'P': 'Proline',\n    'W': 'Tryptophan',\n    'I': 'Isoleucine',\n    'M': 'Methionine',\n    'T': 'Threonine',\n  };\n\n  public static AASmiles: StringDictionary = {\n    'G': 'NCC(=O)O',\n    'L': 'N[C@H](CC(C)C)C(=O)O',\n    'Y': 'NC(CC1=CC=C(O)C=C1)C(=O)O',\n    'S': 'NC(CO)C(=O)O',\n    'E': 'N[C@@H](CCC(O)=O)C(=O)O',\n    'Q': 'N[C@@H](CCC(N)=O)C(=O)O',\n    'D': 'N[C@@H](CC(O)=O)C(=O)O',\n    'N': 'N[C@@H](CC(N)=O)C(=O)O',\n    'F': 'NC(CC1=CC=CC=C1)C(=O)O',\n    'A': 'N[C@H](C)C(=O)O',\n    'K': 'NC(CCCCN)C(=O)O',\n    'R': 'N[C@H](CCCNC(=N)C)C(=O)O',\n    'H': 'NC(CC1=CN=C[N]1)C(=O)O',\n    'C': 'N[C@@H](CS)C(=O)O',\n    'V': 'NC(C(C)C)C(=O)O',\n    'P': 'N(CCC1)C1C(=O)O',\n    'W': 'N[C@@H](Cc1c2ccccc2n([H])c1)C(=O)O',\n    'I': 'N[C@H]([C@H](C)CC)C(=O)O',\n    'M': 'NC(CCSC)C(=O)O',\n    'T': 'NC(C(O)C)C(=O)O',\n  };\n\n  public static AASmilesTruncated: StringDictionary = {\n    'G': '*C*',\n    'L': 'CC(C)C[C@H](*)*',\n    'Y': 'C1=CC(=CC=C1CC(*)*)O',\n    'S': 'OCC(*)C*',\n    'E': '*[C@@H](CCC(O)=O)*',\n    'Q': '*N[C@@H](CCC(N)=O)*',\n    'D': '*[C@@H](CC(O)=O)*',\n    'N': '*[C@@H](CC(N)=O)*',\n    'F': 'C1=CC=C(C=C1)CC(*)*',\n    'A': 'C[C@H](*)*',\n    'K': 'C(CCN)CC(*)*',\n    'R': '*[C@H](CCCNC(=N)C)*',\n    'H': 'C1=C(NC=N1)CC(*)*',\n    'C': 'C([C@@H](*)*)S',\n    'V': 'CC(C)C(*)*',\n    'P': 'C1CCN(*)C1*',\n    'W': '*[C@@H](Cc1c2ccccc2n([H])c1)*',\n    'I': 'CC[C@H](C)[C@H](*)*',\n    'M': 'CSCCC(*)*',\n    'T': 'CC(O)C(*)*',\n  };\n\n  /** TODO: Full?\n   */\n  public static AAFullNames: StringDictionary = {\n    'Ala': 'A',\n    'Arg': 'R',\n    'Asn': 'N',\n    'Asp': 'D',\n    'Cys': 'C',\n    'Gln': 'Q',\n    'Glu': 'E',\n    'Gly': 'G',\n    'His': 'H',\n    'Ile': 'I',\n    'Leu': 'L',\n    'Lys': 'K',\n    'Met': 'M',\n    'Phe': 'F',\n    'Pro': 'P',\n    'Ser': 'S',\n    'Thr': 'T',\n    'Trp': 'W',\n    'Tyr': 'Y',\n    'Val': 'V',\n  };\n\n  public static getPalette(scheme: string = 'grok'): SeqPalette {\n    switch (scheme) {\n    case 'grok':\n      return AminoacidsPalettes.GrokGroups;\n    case 'lesk':\n      return AminoacidsPalettes.Lesk;\n    default:\n      throw new Error(`ChemPalette: scheme \\`${scheme}\\` does not exist`);\n    }\n  }\n\n  /**\n   * Returns divided amino acid with its content in the bracket, if the content is number, then its omitted\n   *\n   * @param {string} c raw amino\n   * @return {[string, string]} outer and inner content\n   */\n  public static getInnerOuter(c: string): [string, string] {\n    let isInner = 0;\n    let inner = '';\n    let outer = '';\n\n    for (const char of c) {\n      if (char == '(')\n        isInner++;\n      else if (char == ')')\n        isInner--;\n      else if (isInner)\n        inner += char;\n      else\n        outer += char;\n    }\n\n    return !isNaN(parseInt(inner)) ? [outer, ''] : [outer, inner];\n  }\n\n  public static getColorAAPivot(monomer: string = '', scheme: 'grok' = 'grok'): [string, string, string, number] {\n    //const chemPaletteInstance = AAPalettes.GrokGroups();\n    const chemPaletteInstance = this.getPalette(scheme);\n    let [outerMonomer, innerMonomer] = this.getInnerOuter(monomer);\n    outerMonomer = (outerMonomer.length > 6 ? `${outerMonomer.slice(0, 3)}...` : outerMonomer);\n    innerMonomer = (innerMonomer.length > 6 ? `${innerMonomer.slice(0, 3)}...` : innerMonomer);\n\n    if (monomer.length == 1 || monomer[1] == '(') {\n      const amino = monomer[0]?.toUpperCase()!;\n      return amino in chemPaletteInstance ?\n        [chemPaletteInstance[amino], amino, innerMonomer, 1] :\n        [this.undefinedColor, outerMonomer, innerMonomer, 1];\n    }\n\n    if (monomer[0] == 'd' && monomer[1]! in chemPaletteInstance) {\n      if (monomer.length == 2 || monomer[2] == '(') {\n        const amino = monomer[1]?.toUpperCase()!;\n        return amino in chemPaletteInstance ?\n          [chemPaletteInstance[amino], amino, innerMonomer, 2] :\n          [this.undefinedColor, outerMonomer, innerMonomer, 2];\n      }\n    }\n\n    if (monomer.substring(0, 3) in this.AAFullNames) {\n      if (monomer.length == 3 || monomer[3] == '(') {\n        const amino = this.AAFullNames[monomer.substring(0, 3)];\n        return amino in chemPaletteInstance ?\n          [chemPaletteInstance[amino], amino, innerMonomer, 3] :\n          [this.undefinedColor, outerMonomer, innerMonomer, 3];\n      }\n    }\n\n    if (monomer[0]?.toLowerCase() == monomer[0]) {\n      if (monomer.substring(1, 3) in this.AAFullNames) {\n        if (monomer.length == 4 || monomer[4] == '(') {\n          const amino = this.AAFullNames[monomer.substring(1, 3)];\n          return amino in chemPaletteInstance ?\n            [chemPaletteInstance[amino], amino, innerMonomer, 4] :\n            [this.undefinedColor, outerMonomer, innerMonomer, 4];\n        }\n      }\n    }\n\n    return [this.undefinedColor, outerMonomer, innerMonomer, 0];\n  }\n}\n"]}
package/src/index.d.ts ADDED
@@ -0,0 +1,7 @@
1
+ import { SeqPalette } from './seq-palettes';
2
+ import { Aminoacids, AminoacidsPalettes } from './aminoacids';
3
+ import { Nucleotides, NucleotidesPalettes } from './nucleotides';
4
+ import { WebLogo, PositionInfo, PositionMonomerInfo } from './viewers/web-logo';
5
+ export { SeqPalette, AminoacidsPalettes, Aminoacids, NucleotidesPalettes, Nucleotides };
6
+ export { WebLogo, PositionInfo, PositionMonomerInfo };
7
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,UAAU,EAAC,MAAM,gBAAgB,CAAC;AAC1C,OAAO,EAAC,UAAU,EAAE,kBAAkB,EAAC,MAAM,cAAc,CAAC;AAC5D,OAAO,EAAC,WAAW,EAAE,mBAAmB,EAAC,MAAM,eAAe,CAAC;AAC/D,OAAO,EAAC,OAAO,EAAE,YAAY,EAAE,mBAAmB,EAAC,MAAM,oBAAoB,CAAC;AAE9E,OAAO,EAAC,UAAU,EAAE,kBAAkB,EAAE,UAAU,EAAE,mBAAmB,EAAE,WAAW,EAAC,CAAC;AACtF,OAAO,EAAC,OAAO,EAAE,YAAY,EAAE,mBAAmB,EAAC,CAAC"}
package/src/index.js ADDED
@@ -0,0 +1,6 @@
1
+ import { Aminoacids, AminoacidsPalettes } from './aminoacids';
2
+ import { Nucleotides, NucleotidesPalettes } from './nucleotides';
3
+ import { WebLogo, PositionInfo, PositionMonomerInfo } from './viewers/web-logo';
4
+ export { AminoacidsPalettes, Aminoacids, NucleotidesPalettes, Nucleotides };
5
+ export { WebLogo, PositionInfo, PositionMonomerInfo };
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUMsVUFBVSxFQUFFLGtCQUFrQixFQUFDLE1BQU0sY0FBYyxDQUFDO0FBQzVELE9BQU8sRUFBQyxXQUFXLEVBQUUsbUJBQW1CLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFDL0QsT0FBTyxFQUFDLE9BQU8sRUFBRSxZQUFZLEVBQUUsbUJBQW1CLEVBQUMsTUFBTSxvQkFBb0IsQ0FBQztBQUU5RSxPQUFPLEVBQWEsa0JBQWtCLEVBQUUsVUFBVSxFQUFFLG1CQUFtQixFQUFFLFdBQVcsRUFBQyxDQUFDO0FBQ3RGLE9BQU8sRUFBQyxPQUFPLEVBQUUsWUFBWSxFQUFFLG1CQUFtQixFQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1NlcVBhbGV0dGV9IGZyb20gJy4vc2VxLXBhbGV0dGVzJztcbmltcG9ydCB7QW1pbm9hY2lkcywgQW1pbm9hY2lkc1BhbGV0dGVzfSBmcm9tICcuL2FtaW5vYWNpZHMnO1xuaW1wb3J0IHtOdWNsZW90aWRlcywgTnVjbGVvdGlkZXNQYWxldHRlc30gZnJvbSAnLi9udWNsZW90aWRlcyc7XG5pbXBvcnQge1dlYkxvZ28sIFBvc2l0aW9uSW5mbywgUG9zaXRpb25Nb25vbWVySW5mb30gZnJvbSAnLi92aWV3ZXJzL3dlYi1sb2dvJztcblxuZXhwb3J0IHtTZXFQYWxldHRlLCBBbWlub2FjaWRzUGFsZXR0ZXMsIEFtaW5vYWNpZHMsIE51Y2xlb3RpZGVzUGFsZXR0ZXMsIE51Y2xlb3RpZGVzfTtcbmV4cG9ydCB7V2ViTG9nbywgUG9zaXRpb25JbmZvLCBQb3NpdGlvbk1vbm9tZXJJbmZvfTtcblxuIl19
package/src/index.ts ADDED
@@ -0,0 +1,8 @@
1
+ import {SeqPalette} from './seq-palettes';
2
+ import {Aminoacids, AminoacidsPalettes} from './aminoacids';
3
+ import {Nucleotides, NucleotidesPalettes} from './nucleotides';
4
+ import {WebLogo, PositionInfo, PositionMonomerInfo} from './viewers/web-logo';
5
+
6
+ export {SeqPalette, AminoacidsPalettes, Aminoacids, NucleotidesPalettes, Nucleotides};
7
+ export {WebLogo, PositionInfo, PositionMonomerInfo};
8
+
@@ -0,0 +1,12 @@
1
+ import { StringDictionary } from '@datagrok-libraries/utils/src/type-declarations';
2
+ import { SeqPalette, SeqPaletteBase } from './seq-palettes';
3
+ export declare class NucleotidesPalettes extends SeqPaletteBase {
4
+ private static chromatogram;
5
+ static get Chromatogram(): SeqPalette;
6
+ }
7
+ export declare class Nucleotides {
8
+ static readonly SemType: string;
9
+ static readonly SemTypeMultipleAlignment: string;
10
+ static Names: StringDictionary;
11
+ }
12
+ //# sourceMappingURL=nucleotides.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"nucleotides.d.ts","sourceRoot":"","sources":["nucleotides.ts"],"names":[],"mappings":"AAIA,OAAO,EAAC,gBAAgB,EAAC,MAAM,iDAAiD,CAAC;AACjF,OAAO,EAAC,UAAU,EAAE,cAAc,EAAC,MAAM,gBAAgB,CAAC;AAE1D,qBAAa,mBAAoB,SAAQ,cAAc;IACrD,OAAO,CAAC,MAAM,CAAC,YAAY,CAAa;IAExC,WAAkB,YAAY,IAAI,UAAU,CAW3C;CACF;AAED,qBAAa,WAAW;IACtB,MAAM,CAAC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAuB;IAEtD,MAAM,CAAC,QAAQ,CAAC,wBAAwB,EAAE,MAAM,CAAkC;IAElF,OAAc,KAAK,EAAE,gBAAgB,CAMnC;CACH"}
@@ -0,0 +1,27 @@
1
+ import { SeqPaletteBase } from './seq-palettes';
2
+ export class NucleotidesPalettes extends SeqPaletteBase {
3
+ static get Chromatogram() {
4
+ if (this.chromatogram === void 0) {
5
+ this.chromatogram = {
6
+ 'A': 'green',
7
+ 'C': 'blue',
8
+ 'G': 'black',
9
+ 'T': 'red', 'U': 'red',
10
+ 'others': 'gray',
11
+ };
12
+ }
13
+ return this.chromatogram;
14
+ }
15
+ }
16
+ export class Nucleotides {
17
+ }
18
+ Nucleotides.SemType = 'MultipleAlignment';
19
+ Nucleotides.SemTypeMultipleAlignment = 'NucleotidesMultipleAlignment';
20
+ Nucleotides.Names = {
21
+ 'A': 'Adenine',
22
+ 'C': 'Cytosine',
23
+ 'G': 'Guanine',
24
+ 'T': 'Thymine',
25
+ 'U': 'Uracil',
26
+ };
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnVjbGVvdGlkZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJudWNsZW90aWRlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLQSxPQUFPLEVBQWEsY0FBYyxFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFFMUQsTUFBTSxPQUFPLG1CQUFvQixTQUFRLGNBQWM7SUFHOUMsTUFBTSxLQUFLLFlBQVk7UUFDNUIsSUFBSSxJQUFJLENBQUMsWUFBWSxLQUFLLEtBQUssQ0FBQyxFQUFFO1lBQ2hDLElBQUksQ0FBQyxZQUFZLEdBQUc7Z0JBQ2xCLEdBQUcsRUFBRSxPQUFPO2dCQUNaLEdBQUcsRUFBRSxNQUFNO2dCQUNYLEdBQUcsRUFBRSxPQUFPO2dCQUNaLEdBQUcsRUFBRSxLQUFLLEVBQUUsR0FBRyxFQUFFLEtBQUs7Z0JBQ3RCLFFBQVEsRUFBRSxNQUFNO2FBQ2pCLENBQUM7U0FDSDtRQUNELE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQztJQUMzQixDQUFDO0NBQ0Y7QUFFRCxNQUFNLE9BQU8sV0FBVzs7QUFDTixtQkFBTyxHQUFXLG1CQUFtQixDQUFDO0FBRXRDLG9DQUF3QixHQUFXLDhCQUE4QixDQUFDO0FBRXBFLGlCQUFLLEdBQXFCO0lBQ3RDLEdBQUcsRUFBRSxTQUFTO0lBQ2QsR0FBRyxFQUFFLFVBQVU7SUFDZixHQUFHLEVBQUUsU0FBUztJQUNkLEdBQUcsRUFBRSxTQUFTO0lBQ2QsR0FBRyxFQUFFLFFBQVE7Q0FDZCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgZ3JvayBmcm9tICdkYXRhZ3Jvay1hcGkvZ3Jvayc7XG5pbXBvcnQgKiBhcyB1aSBmcm9tICdkYXRhZ3Jvay1hcGkvdWknO1xuaW1wb3J0ICogYXMgREcgZnJvbSAnZGF0YWdyb2stYXBpL2RnJztcblxuaW1wb3J0IHtTdHJpbmdEaWN0aW9uYXJ5fSBmcm9tICdAZGF0YWdyb2stbGlicmFyaWVzL3V0aWxzL3NyYy90eXBlLWRlY2xhcmF0aW9ucyc7XG5pbXBvcnQge1NlcVBhbGV0dGUsIFNlcVBhbGV0dGVCYXNlfSBmcm9tICcuL3NlcS1wYWxldHRlcyc7XG5cbmV4cG9ydCBjbGFzcyBOdWNsZW90aWRlc1BhbGV0dGVzIGV4dGVuZHMgU2VxUGFsZXR0ZUJhc2Uge1xuICBwcml2YXRlIHN0YXRpYyBjaHJvbWF0b2dyYW06IFNlcVBhbGV0dGU7XG5cbiAgcHVibGljIHN0YXRpYyBnZXQgQ2hyb21hdG9ncmFtKCk6IFNlcVBhbGV0dGUge1xuICAgIGlmICh0aGlzLmNocm9tYXRvZ3JhbSA9PT0gdm9pZCAwKSB7XG4gICAgICB0aGlzLmNocm9tYXRvZ3JhbSA9IHtcbiAgICAgICAgJ0EnOiAnZ3JlZW4nLFxuICAgICAgICAnQyc6ICdibHVlJyxcbiAgICAgICAgJ0cnOiAnYmxhY2snLCAvLyBvcmFuZ2UgP1xuICAgICAgICAnVCc6ICdyZWQnLCAnVSc6ICdyZWQnLFxuICAgICAgICAnb3RoZXJzJzogJ2dyYXknLFxuICAgICAgfTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuY2hyb21hdG9ncmFtO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBOdWNsZW90aWRlcyB7XG4gIHN0YXRpYyByZWFkb25seSBTZW1UeXBlOiBzdHJpbmcgPSAnTXVsdGlwbGVBbGlnbm1lbnQnO1xuXG4gIHN0YXRpYyByZWFkb25seSBTZW1UeXBlTXVsdGlwbGVBbGlnbm1lbnQ6IHN0cmluZyA9ICdOdWNsZW90aWRlc011bHRpcGxlQWxpZ25tZW50JztcblxuICBwdWJsaWMgc3RhdGljIE5hbWVzOiBTdHJpbmdEaWN0aW9uYXJ5ID0ge1xuICAgICdBJzogJ0FkZW5pbmUnLFxuICAgICdDJzogJ0N5dG9zaW5lJyxcbiAgICAnRyc6ICdHdWFuaW5lJyxcbiAgICAnVCc6ICdUaHltaW5lJyxcbiAgICAnVSc6ICdVcmFjaWwnLFxuICB9O1xufVxuIl19
@@ -0,0 +1 @@
1
+ //# sourceMappingURL=package.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"package.d.ts","sourceRoot":"","sources":["package.ts"],"names":[],"mappings":""}
package/src/package.js ADDED
@@ -0,0 +1,2 @@
1
+ "use strict";
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFja2FnZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInBhY2thZ2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIiJdfQ==
@@ -0,0 +1,16 @@
1
+ export interface SeqPalette {
2
+ /**
3
+ * @param {string} m Monomer character
4
+ * @return {string} Color
5
+ */
6
+ [m: string]: string;
7
+ }
8
+ export declare class SeqPaletteBase {
9
+ /** Palette with shades of primary colors
10
+ */
11
+ private static colourPalette;
12
+ protected static makePalette(dt: [string[], string][], simplified?: boolean): {
13
+ [key: string]: string;
14
+ };
15
+ }
16
+ //# sourceMappingURL=seq-palettes.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"seq-palettes.d.ts","sourceRoot":"","sources":["seq-palettes.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,UAAU;IACzB;;;OAGG;IACH,CAAC,CAAC,EAAE,MAAM,GAAG,MAAM,CAAC;CACrB;AAED,qBAAa,cAAc;IACzB;OACG;IACH,OAAO,CAAC,MAAM,CAAC,aAAa,CAqB1B;IAEF,SAAS,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,EAAE,CAAC,MAAM,EAAE,EAAE,MAAM,CAAC,EAAE,EAAE,UAAU,UAAQ;;;CAW1E"}
@@ -0,0 +1,38 @@
1
+ export class SeqPaletteBase {
2
+ static makePalette(dt, simplified = false) {
3
+ const palette = {};
4
+ dt.forEach((cp) => {
5
+ const objList = cp[0];
6
+ const colour = cp[1];
7
+ objList.forEach((obj, ind) => {
8
+ palette[obj] = this.colourPalette[colour][simplified ? 0 : ind];
9
+ });
10
+ });
11
+ return palette;
12
+ }
13
+ }
14
+ /** Palette with shades of primary colors
15
+ */
16
+ SeqPaletteBase.colourPalette = {
17
+ 'orange': ['rgb(255,187,120)', 'rgb(245,167,100)', 'rgb(235,137,70)', 'rgb(205, 111, 71)'],
18
+ 'all_green': ['rgb(44,160,44)', 'rgb(74,160,74)', 'rgb(23,103,57)', 'rgb(30,110,96)', 'rgb(60,131,95)',
19
+ 'rgb(24,110,79)', 'rgb(152,223,138)', 'rgb(182, 223, 138)', 'rgb(152, 193, 138)'],
20
+ 'all_blue': ['rgb(31,119,180)', 'rgb(23,190,207)', 'rgb(122, 102, 189)', 'rgb(158,218,229)', 'rgb(141, 124, 217)',
21
+ 'rgb(31, 120, 150)'],
22
+ 'magenta': ['rgb(162,106,192)', 'rgb(197,165,224)', 'rgb(208,113,218)'],
23
+ 'red': ['rgb(214,39,40)', 'rgb(255,152,150)'],
24
+ 'st_blue': ['rgb(23,190,207)', 'rgb(158,218,229)', 'rgb(31,119,180)'],
25
+ 'dark_blue': ['rgb(31,119,180)', 'rgb(31, 120, 150)'],
26
+ 'light_blue': ['rgb(23,190,207)', 'rgb(158,218,229)', 'rgb(108, 218, 229)', 'rgb(23,190,227)'],
27
+ 'lilac_blue': ['rgb(124,102,211)', 'rgb(149,134,217)', 'rgb(97, 81, 150)'],
28
+ 'dark_green': ['rgb(23,103,57)', 'rgb(30,110,96)', 'rgb(60,131,95)', 'rgb(24,110,79)'],
29
+ 'green': ['rgb(44,160,44)', 'rgb(74,160,74)'],
30
+ 'light_green': ['rgb(152,223,138)', 'rgb(182, 223, 138)', 'rgb(152, 193, 138)'],
31
+ 'st_green': ['rgb(44,160,44)', 'rgb(152,223,138)', 'rgb(39, 174, 96)', 'rgb(74,160,74)'],
32
+ 'pink': ['rgb(247,182,210)'],
33
+ 'brown': ['rgb(140,86,75)', 'rgb(102, 62, 54)'],
34
+ 'gray': ['rgb(127,127,127)', 'rgb(199,199,199)', 'rgb(196,156,148)', 'rgb(222, 222, 180)'],
35
+ 'yellow': ['rgb(188,189,34)'],
36
+ 'white': ['rgb(230,230,230)'],
37
+ };
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VxLXBhbGV0dGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsic2VxLXBhbGV0dGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQVlBLE1BQU0sT0FBTyxjQUFjO0lBMEJmLE1BQU0sQ0FBQyxXQUFXLENBQUMsRUFBd0IsRUFBRSxVQUFVLEdBQUcsS0FBSztRQUN2RSxNQUFNLE9BQU8sR0FBOEIsRUFBRSxDQUFDO1FBQzlDLEVBQUUsQ0FBQyxPQUFPLENBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBRTtZQUNoQixNQUFNLE9BQU8sR0FBRyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDdEIsTUFBTSxNQUFNLEdBQUcsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ3JCLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQyxHQUFHLEVBQUUsR0FBRyxFQUFFLEVBQUU7Z0JBQzNCLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQyxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQztZQUNsRSxDQUFDLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO1FBQ0gsT0FBTyxPQUFPLENBQUM7SUFDakIsQ0FBQzs7QUFuQ0Q7R0FDRztBQUNZLDRCQUFhLEdBQWdDO0lBQzFELFFBQVEsRUFBRSxDQUFDLGtCQUFrQixFQUFFLGtCQUFrQixFQUFFLGlCQUFpQixFQUFFLG1CQUFtQixDQUFDO0lBQzFGLFdBQVcsRUFBRSxDQUFDLGdCQUFnQixFQUFFLGdCQUFnQixFQUFFLGdCQUFnQixFQUFFLGdCQUFnQixFQUFFLGdCQUFnQjtRQUNwRyxnQkFBZ0IsRUFBRSxrQkFBa0IsRUFBRSxvQkFBb0IsRUFBRSxvQkFBb0IsQ0FBQztJQUNuRixVQUFVLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxpQkFBaUIsRUFBRSxvQkFBb0IsRUFBRSxrQkFBa0IsRUFBRSxvQkFBb0I7UUFDL0csbUJBQW1CLENBQUM7SUFDdEIsU0FBUyxFQUFFLENBQUMsa0JBQWtCLEVBQUUsa0JBQWtCLEVBQUUsa0JBQWtCLENBQUM7SUFDdkUsS0FBSyxFQUFFLENBQUMsZ0JBQWdCLEVBQUUsa0JBQWtCLENBQUM7SUFDN0MsU0FBUyxFQUFFLENBQUMsaUJBQWlCLEVBQUUsa0JBQWtCLEVBQUUsaUJBQWlCLENBQUM7SUFDckUsV0FBVyxFQUFFLENBQUMsaUJBQWlCLEVBQUUsbUJBQW1CLENBQUM7SUFDckQsWUFBWSxFQUFFLENBQUMsaUJBQWlCLEVBQUUsa0JBQWtCLEVBQUUsb0JBQW9CLEVBQUUsaUJBQWlCLENBQUM7SUFDOUYsWUFBWSxFQUFFLENBQUMsa0JBQWtCLEVBQUUsa0JBQWtCLEVBQUUsa0JBQWtCLENBQUM7SUFDMUUsWUFBWSxFQUFFLENBQUMsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCLENBQUM7SUFDdEYsT0FBTyxFQUFFLENBQUMsZ0JBQWdCLEVBQUUsZ0JBQWdCLENBQUM7SUFDN0MsYUFBYSxFQUFFLENBQUMsa0JBQWtCLEVBQUUsb0JBQW9CLEVBQUUsb0JBQW9CLENBQUM7SUFDL0UsVUFBVSxFQUFFLENBQUMsZ0JBQWdCLEVBQUUsa0JBQWtCLEVBQUUsa0JBQWtCLEVBQUUsZ0JBQWdCLENBQUM7SUFDeEYsTUFBTSxFQUFFLENBQUMsa0JBQWtCLENBQUM7SUFDNUIsT0FBTyxFQUFFLENBQUMsZ0JBQWdCLEVBQUUsa0JBQWtCLENBQUM7SUFDL0MsTUFBTSxFQUFFLENBQUMsa0JBQWtCLEVBQUUsa0JBQWtCLEVBQUUsa0JBQWtCLEVBQUUsb0JBQW9CLENBQUM7SUFDMUYsUUFBUSxFQUFFLENBQUMsaUJBQWlCLENBQUM7SUFDN0IsT0FBTyxFQUFFLENBQUMsa0JBQWtCLENBQUM7Q0FDOUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGdyb2sgZnJvbSAnZGF0YWdyb2stYXBpL2dyb2snO1xuaW1wb3J0ICogYXMgdWkgZnJvbSAnZGF0YWdyb2stYXBpL3VpJztcbmltcG9ydCAqIGFzIERHIGZyb20gJ2RhdGFncm9rLWFwaS9kZyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VxUGFsZXR0ZSB7XG4gIC8qKlxuICAgKiBAcGFyYW0ge3N0cmluZ30gbSBNb25vbWVyIGNoYXJhY3RlclxuICAgKiBAcmV0dXJuIHtzdHJpbmd9IENvbG9yXG4gICAqL1xuICBbbTogc3RyaW5nXTogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgU2VxUGFsZXR0ZUJhc2Uge1xuICAvKiogUGFsZXR0ZSB3aXRoIHNoYWRlcyBvZiBwcmltYXJ5IGNvbG9yc1xuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgY29sb3VyUGFsZXR0ZTogeyBba2V5OiBzdHJpbmddOiBzdHJpbmdbXSB9ID0ge1xuICAgICdvcmFuZ2UnOiBbJ3JnYigyNTUsMTg3LDEyMCknLCAncmdiKDI0NSwxNjcsMTAwKScsICdyZ2IoMjM1LDEzNyw3MCknLCAncmdiKDIwNSwgMTExLCA3MSknXSxcbiAgICAnYWxsX2dyZWVuJzogWydyZ2IoNDQsMTYwLDQ0KScsICdyZ2IoNzQsMTYwLDc0KScsICdyZ2IoMjMsMTAzLDU3KScsICdyZ2IoMzAsMTEwLDk2KScsICdyZ2IoNjAsMTMxLDk1KScsXG4gICAgICAncmdiKDI0LDExMCw3OSknLCAncmdiKDE1MiwyMjMsMTM4KScsICdyZ2IoMTgyLCAyMjMsIDEzOCknLCAncmdiKDE1MiwgMTkzLCAxMzgpJ10sXG4gICAgJ2FsbF9ibHVlJzogWydyZ2IoMzEsMTE5LDE4MCknLCAncmdiKDIzLDE5MCwyMDcpJywgJ3JnYigxMjIsIDEwMiwgMTg5KScsICdyZ2IoMTU4LDIxOCwyMjkpJywgJ3JnYigxNDEsIDEyNCwgMjE3KScsXG4gICAgICAncmdiKDMxLCAxMjAsIDE1MCknXSxcbiAgICAnbWFnZW50YSc6IFsncmdiKDE2MiwxMDYsMTkyKScsICdyZ2IoMTk3LDE2NSwyMjQpJywgJ3JnYigyMDgsMTEzLDIxOCknXSxcbiAgICAncmVkJzogWydyZ2IoMjE0LDM5LDQwKScsICdyZ2IoMjU1LDE1MiwxNTApJ10sXG4gICAgJ3N0X2JsdWUnOiBbJ3JnYigyMywxOTAsMjA3KScsICdyZ2IoMTU4LDIxOCwyMjkpJywgJ3JnYigzMSwxMTksMTgwKSddLFxuICAgICdkYXJrX2JsdWUnOiBbJ3JnYigzMSwxMTksMTgwKScsICdyZ2IoMzEsIDEyMCwgMTUwKSddLFxuICAgICdsaWdodF9ibHVlJzogWydyZ2IoMjMsMTkwLDIwNyknLCAncmdiKDE1OCwyMTgsMjI5KScsICdyZ2IoMTA4LCAyMTgsIDIyOSknLCAncmdiKDIzLDE5MCwyMjcpJ10sXG4gICAgJ2xpbGFjX2JsdWUnOiBbJ3JnYigxMjQsMTAyLDIxMSknLCAncmdiKDE0OSwxMzQsMjE3KScsICdyZ2IoOTcsIDgxLCAxNTApJ10sXG4gICAgJ2RhcmtfZ3JlZW4nOiBbJ3JnYigyMywxMDMsNTcpJywgJ3JnYigzMCwxMTAsOTYpJywgJ3JnYig2MCwxMzEsOTUpJywgJ3JnYigyNCwxMTAsNzkpJ10sXG4gICAgJ2dyZWVuJzogWydyZ2IoNDQsMTYwLDQ0KScsICdyZ2IoNzQsMTYwLDc0KSddLFxuICAgICdsaWdodF9ncmVlbic6IFsncmdiKDE1MiwyMjMsMTM4KScsICdyZ2IoMTgyLCAyMjMsIDEzOCknLCAncmdiKDE1MiwgMTkzLCAxMzgpJ10sXG4gICAgJ3N0X2dyZWVuJzogWydyZ2IoNDQsMTYwLDQ0KScsICdyZ2IoMTUyLDIyMywxMzgpJywgJ3JnYigzOSwgMTc0LCA5NiknLCAncmdiKDc0LDE2MCw3NCknXSxcbiAgICAncGluayc6IFsncmdiKDI0NywxODIsMjEwKSddLFxuICAgICdicm93bic6IFsncmdiKDE0MCw4Niw3NSknLCAncmdiKDEwMiwgNjIsIDU0KSddLFxuICAgICdncmF5JzogWydyZ2IoMTI3LDEyNywxMjcpJywgJ3JnYigxOTksMTk5LDE5OSknLCAncmdiKDE5NiwxNTYsMTQ4KScsICdyZ2IoMjIyLCAyMjIsIDE4MCknXSxcbiAgICAneWVsbG93JzogWydyZ2IoMTg4LDE4OSwzNCknXSxcbiAgICAnd2hpdGUnOiBbJ3JnYigyMzAsMjMwLDIzMCknXSxcbiAgfTtcblxuICBwcm90ZWN0ZWQgc3RhdGljIG1ha2VQYWxldHRlKGR0OiBbc3RyaW5nW10sIHN0cmluZ11bXSwgc2ltcGxpZmllZCA9IGZhbHNlKSB7XG4gICAgY29uc3QgcGFsZXR0ZTogeyBba2V5OiBzdHJpbmddOiBzdHJpbmcgfSA9IHt9O1xuICAgIGR0LmZvckVhY2goKGNwKSA9PiB7XG4gICAgICBjb25zdCBvYmpMaXN0ID0gY3BbMF07XG4gICAgICBjb25zdCBjb2xvdXIgPSBjcFsxXTtcbiAgICAgIG9iakxpc3QuZm9yRWFjaCgob2JqLCBpbmQpID0+IHtcbiAgICAgICAgcGFsZXR0ZVtvYmpdID0gdGhpcy5jb2xvdXJQYWxldHRlW2NvbG91cl1bc2ltcGxpZmllZCA/IDAgOiBpbmRdO1xuICAgICAgfSk7XG4gICAgfSk7XG4gICAgcmV0dXJuIHBhbGV0dGU7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,67 @@
1
+ import * as DG from 'datagrok-api/dg';
2
+ import { StringDictionary } from '@datagrok-libraries/utils/src/type-declarations';
3
+ declare module 'datagrok-api/src/grid' {
4
+ interface Rect {
5
+ contains(x: number, y: number): boolean;
6
+ }
7
+ }
8
+ declare global {
9
+ interface HTMLCanvasElement {
10
+ getCursorPosition(event: MouseEvent): DG.Point;
11
+ }
12
+ }
13
+ export declare class PositionMonomerInfo {
14
+ /** Sequences count with monomer in position
15
+ */
16
+ count: number;
17
+ /** Remember screen coords rect
18
+ */
19
+ bounds: DG.Rect;
20
+ constructor();
21
+ }
22
+ export declare class PositionInfo {
23
+ readonly name: string;
24
+ freq: {
25
+ [m: string]: PositionMonomerInfo;
26
+ };
27
+ rowCount: number;
28
+ /** freq = {}, rowCount = 0
29
+ * @param {string} name Name of position ('111A', '111.1', etc)
30
+ */
31
+ constructor(name: string);
32
+ }
33
+ export declare class WebLogo extends DG.JsViewer {
34
+ static residuesSet: string;
35
+ protected cp: StringDictionary | null;
36
+ private canvas?;
37
+ private slider?;
38
+ private textBaseline;
39
+ private axisHeight;
40
+ private positionWidth;
41
+ private seqCol;
42
+ private positions;
43
+ private rowsMasked;
44
+ private rowsNull;
45
+ minHeight: number;
46
+ maxHeight: number;
47
+ considerNullSequences: boolean;
48
+ sequenceColumnName: string;
49
+ startPositionName: string;
50
+ endPositionName: string;
51
+ private positionNames;
52
+ private startPosition;
53
+ private endPosition;
54
+ private get Length();
55
+ constructor();
56
+ init(): Promise<void>;
57
+ rootOnSizeChanged(args: any): void;
58
+ /** Assigns {@link seqCol} and {@link cp} based on {@link sequenceColumnName} and calls {@link render}().
59
+ */
60
+ updateSeqCol(): void;
61
+ onPropertyChanged(property: DG.Property): void;
62
+ onTableAttached(): void;
63
+ protected _nullSequence(fillerResidue?: string): string;
64
+ protected _calculate(): void;
65
+ render(recalc?: boolean): void;
66
+ }
67
+ //# sourceMappingURL=web-logo.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"web-logo.d.ts","sourceRoot":"","sources":["web-logo.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAItC,OAAO,EAAC,gBAAgB,EAAC,MAAM,iDAAiD,CAAC;AAOjF,OAAO,QAAQ,uBAAuB,CAAC;IACrC,UAAU,IAAI;QACZ,QAAQ,CAAC,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC;KACzC;CACF;AAED,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,iBAAiB;QACzB,iBAAiB,CAAC,KAAK,EAAE,UAAU,GAAG,EAAE,CAAC,KAAK,CAAC;KAChD;CACF;AAWD,qBAAa,mBAAmB;IAC9B;OACG;IACH,KAAK,EAAE,MAAM,CAAC;IAEd;OACG;IACH,MAAM,EAAE,EAAE,CAAC,IAAI,CAAC;;CAMjB;AAED,qBAAa,YAAY;IACvB,SAAgB,IAAI,EAAE,MAAM,CAAC;IAC7B,IAAI,EAAE;QAAE,CAAC,CAAC,EAAE,MAAM,GAAG,mBAAmB,CAAA;KAAE,CAAC;IAC3C,QAAQ,EAAE,MAAM,CAAC;IAEjB;;OAEG;gBACS,IAAI,EAAE,MAAM;CAKzB;AAED,qBAAa,OAAQ,SAAQ,EAAE,CAAC,QAAQ;IACtC,OAAc,WAAW,SAAiB;IAG1C,SAAS,CAAC,EAAE,EAAE,gBAAgB,GAAG,IAAI,CAAQ;IAG7C,OAAO,CAAC,MAAM,CAAC,CAAoB;IACnC,OAAO,CAAC,MAAM,CAAC,CAAiB;IAChC,OAAO,CAAC,YAAY,CAAqB;IAEzC,OAAO,CAAC,UAAU,CAAc;IAChC,OAAO,CAAC,aAAa,CAAc;IAEnC,OAAO,CAAC,MAAM,CAA0B;IAExC,OAAO,CAAC,SAAS,CAAsB;IAEvC,OAAO,CAAC,UAAU,CAAa;IAC/B,OAAO,CAAC,QAAQ,CAAa;IAGtB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,qBAAqB,EAAE,OAAO,CAAC;IAC/B,kBAAkB,EAAE,MAAM,CAAC;IAC3B,iBAAiB,EAAE,MAAM,CAAC;IAC1B,eAAe,EAAE,MAAM,CAAC;IAE/B,OAAO,CAAC,aAAa,CAAgB;IAErC,OAAO,CAAC,aAAa,CAAc;IAEnC,OAAO,CAAC,WAAW,CAAc;IAEjC,OAAO,KAAK,MAAM,GAAgE;;IAiB5E,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAmF3B,iBAAiB,CAAC,IAAI,EAAE,GAAG;IAmB3B;OACG;IACH,YAAY,IAAI,IAAI;IAuCpB,iBAAiB,CAAC,QAAQ,EAAE,EAAE,CAAC,QAAQ,GAAG,IAAI;IAyB9C,eAAe;IAiBf,SAAS,CAAC,aAAa,CAAC,aAAa,SAAM,GAAG,MAAM;IAOpD,SAAS,CAAC,UAAU;IAoFpB,MAAM,CAAC,MAAM,UAAO;CA8DrB"}
@@ -0,0 +1,345 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ import * as ui from 'datagrok-api/ui';
11
+ import * as DG from 'datagrok-api/dg';
12
+ import * as rxjs from 'rxjs';
13
+ import { Aminoacids, AminoacidsPalettes } from '../aminoacids';
14
+ import { Nucleotides, NucleotidesPalettes } from '../nucleotides';
15
+ HTMLCanvasElement.prototype.getCursorPosition = function (event) {
16
+ const rect = this.getBoundingClientRect();
17
+ return new DG.Point(event.clientX - rect.left, event.clientY - rect.top);
18
+ };
19
+ DG.Rect.prototype.contains = function (x, y) {
20
+ return this.left <= x && x <= this.right && this.top <= y && y <= this.bottom;
21
+ };
22
+ export class PositionMonomerInfo {
23
+ constructor() {
24
+ this.count = 0;
25
+ this.bounds = new DG.Rect(0, 0, 0, 0);
26
+ }
27
+ }
28
+ export class PositionInfo {
29
+ /** freq = {}, rowCount = 0
30
+ * @param {string} name Name of position ('111A', '111.1', etc)
31
+ */
32
+ constructor(name) {
33
+ this.name = name;
34
+ this.freq = {};
35
+ this.rowCount = 0;
36
+ }
37
+ }
38
+ export class WebLogo extends DG.JsViewer {
39
+ constructor() {
40
+ super();
41
+ // private readonly colorScheme: ColorScheme = ColorSchemes[NucleotidesWebLogo.residuesSet];
42
+ this.cp = null;
43
+ this.axisHeight = 12;
44
+ this.positionWidth = 16;
45
+ this.seqCol = null;
46
+ // private maxLength: number = 100;
47
+ this.positions = [];
48
+ this.rowsMasked = 0;
49
+ this.rowsNull = 0;
50
+ this.positionNames = [];
51
+ this.startPosition = -1;
52
+ this.endPosition = -1;
53
+ this.textBaseline = 'top';
54
+ this.minHeight = this.float('minHeight', 50);
55
+ this.maxHeight = this.float('maxHeight', 100);
56
+ this.considerNullSequences = this.bool('considerNullSequences', false);
57
+ this.sequenceColumnName = this.string('sequenceColumnName', null);
58
+ this.startPositionName = this.string('startPositionName', null);
59
+ this.endPositionName = this.string('endPositionName', null);
60
+ }
61
+ get Length() { return this.endPosition - this.startPosition + 1; }
62
+ init() {
63
+ return __awaiter(this, void 0, void 0, function* () {
64
+ this.canvas = ui.canvas();
65
+ this.canvas.style.width = '100%';
66
+ this.slider = ui.rangeSlider(0, 20, 2, 5);
67
+ this.slider.root.style.width = '100%';
68
+ this.slider.root.style.height = '12px';
69
+ // this.host = ui.divV([/*this.slider,*/this.canvas]);
70
+ const getMonomer = (p) => {
71
+ const jPos = Math.floor(p.x / this.positionWidth);
72
+ const position = this.positions[jPos];
73
+ if (position === void 0)
74
+ return [jPos, null, null];
75
+ const monomer = Object.keys(position.freq)
76
+ .find((m) => position.freq[m].bounds.contains(p.x, p.y));
77
+ if (monomer === undefined)
78
+ return [jPos, null, null];
79
+ return [jPos, monomer, position.freq[monomer]];
80
+ };
81
+ this.canvas.onmouseover = (e) => {
82
+ };
83
+ rxjs.fromEvent(this.canvas, 'mousemove').subscribe((e) => {
84
+ if (!this.canvas)
85
+ return;
86
+ const args = e;
87
+ const [jPos, monomer] = getMonomer(this.canvas.getCursorPosition(args));
88
+ //if (this.dataFrame && this.seqCol && monomer) {
89
+ if (this.dataFrame && this.seqCol && monomer) {
90
+ ui.tooltip.showRowGroup(this.dataFrame, (iRow) => {
91
+ const seq = this.seqCol.get(iRow);
92
+ const mSeq = seq ? seq[this.startPosition + jPos] : null;
93
+ return mSeq === monomer;
94
+ }, args.x + 16, args.y + 16);
95
+ }
96
+ else {
97
+ ui.tooltip.hide();
98
+ }
99
+ });
100
+ rxjs.fromEvent(this.canvas, 'mousedown').subscribe((e) => {
101
+ if (!this.canvas)
102
+ return;
103
+ const args = e;
104
+ const [jPos, monomer] = getMonomer(this.canvas.getCursorPosition(args));
105
+ // prevents deselect all rows if we miss monomer bounds
106
+ //if (this.dataFrame && this.seqCol && monomer) {
107
+ if (this.dataFrame && this.seqCol && monomer) {
108
+ this.dataFrame.selection.init((iRow) => {
109
+ const seq = this.seqCol.get(iRow);
110
+ const mSeq = seq ? seq[this.startPosition + jPos] : null;
111
+ return mSeq === monomer;
112
+ });
113
+ }
114
+ });
115
+ this.root.append(this.canvas);
116
+ // this.root.appendChild(this.slider.root);
117
+ // this.root.append(this.host);
118
+ // ui.onSizeChanged(this.canvas).subscribe(this.canvasOnSizeChanged.bind(this));
119
+ ui.onSizeChanged(this.root).subscribe(this.rootOnSizeChanged.bind(this));
120
+ this.render(true);
121
+ });
122
+ }
123
+ // canvasOnSizeChanged(args: any) {
124
+ // this.canvas.width = this.canvas.clientWidth;
125
+ // this.canvas.height = this.canvas.clientHeight;
126
+ // console.debug(`WebLogo.onSizeChanged() width=${this.canvas.width}, height=${this.canvas.height}`);
127
+ // this.render(false);
128
+ // }
129
+ rootOnSizeChanged(args) {
130
+ if (!this.canvas)
131
+ return;
132
+ // this.canvas.width calculate in this._calculate() method
133
+ const height = Math.min(this.maxHeight, Math.max(this.minHeight, this.root.clientHeight));
134
+ // if (this.canvas.width > this.root.clientWidth) /* horizontal scroller is enabled */
135
+ // height -= 6; /* free some space for horizontal scroller */
136
+ this.canvas.height = height;
137
+ this.canvas.style.height = `${height}px`;
138
+ // console.debug(`WebLogo.onRootSizeChanged() ` +
139
+ // `root.width=${this.root.clientWidth}, root.height=${this.root.clientHeight}, ` +
140
+ // `canvas.width=${this.canvas.width}, canvas.height=${this.canvas.height} .`);
141
+ this.render(true);
142
+ }
143
+ /** Assigns {@link seqCol} and {@link cp} based on {@link sequenceColumnName} and calls {@link render}().
144
+ */
145
+ updateSeqCol() {
146
+ if (this.dataFrame) {
147
+ this.seqCol = this.dataFrame.col(this.sequenceColumnName);
148
+ if (this.seqCol) {
149
+ let maxLength = 0;
150
+ for (const category of this.seqCol.categories)
151
+ maxLength = Math.max(maxLength, category.length);
152
+ // Get position names from data column tag 'positionNames'
153
+ const positionNamesTxt = this.seqCol.getTag('positionNames');
154
+ // Fallback if 'positionNames' tag is not provided
155
+ this.positionNames = positionNamesTxt ? positionNamesTxt.split(', ').map((n) => n.trim()) :
156
+ [...Array(maxLength).keys()].map((jPos) => `${jPos + 1}`);
157
+ this.startPosition = this.startPositionName && this.positionNames ?
158
+ this.positionNames.indexOf(this.startPositionName) : 0;
159
+ this.endPosition = this.endPositionName && this.positionNames ?
160
+ this.positionNames.indexOf(this.endPositionName) : maxLength;
161
+ //#region -- palette --
162
+ switch (this.seqCol.semType) {
163
+ case Aminoacids.SemTypeMultipleAlignment:
164
+ this.cp = AminoacidsPalettes.GrokGroups;
165
+ break;
166
+ case Nucleotides.SemTypeMultipleAlignment:
167
+ this.cp = NucleotidesPalettes.Chromatogram;
168
+ break;
169
+ }
170
+ //#endregion -- palette --
171
+ }
172
+ else {
173
+ this.cp = null;
174
+ this.positionNames = [];
175
+ this.startPosition = -1;
176
+ this.endPosition = -1;
177
+ }
178
+ }
179
+ this.render();
180
+ }
181
+ onPropertyChanged(property) {
182
+ super.onPropertyChanged(property);
183
+ switch (property.name) {
184
+ case 'considerNullSequences':
185
+ this.render();
186
+ break;
187
+ case 'sequenceColumnName':
188
+ this.updateSeqCol();
189
+ break;
190
+ case 'startPositionName':
191
+ this.updateSeqCol();
192
+ break;
193
+ case 'endPositionName':
194
+ this.updateSeqCol();
195
+ break;
196
+ case 'minHeight':
197
+ this.render(true);
198
+ break;
199
+ case 'maxHeight':
200
+ this.render(true);
201
+ break;
202
+ }
203
+ }
204
+ onTableAttached() {
205
+ this.updateSeqCol();
206
+ if (this.dataFrame !== void 0) {
207
+ // There are two approaches:
208
+ // first - look in the dataFrame for the first matching column by semType of the
209
+ // corresponding viewer (but we want only one class of a more universal viewer),
210
+ // second - draw column data if the passed column is of suitable semType
211
+ // We decided that we will not search, but we will display asked data if we can
212
+ // const semType = (<typeof NucleotidesWebLogo>(this.constructor)).residuesSet;
213
+ // this.seqCol = (this.dataFrame.columns as DG.ColumnList).bySemType(semType);
214
+ this.dataFrame.selection.onChanged.subscribe((_) => this.render());
215
+ this.dataFrame.filter.onChanged.subscribe((_) => this.render());
216
+ }
217
+ }
218
+ _nullSequence(fillerResidue = 'X') {
219
+ if (this.considerNullSequences)
220
+ return new Array(this.Length).fill(fillerResidue).join('');
221
+ return '';
222
+ }
223
+ _calculate() {
224
+ if (!this.canvas || !this.seqCol || !this.dataFrame || this.startPosition === -1 || this.endPosition === -1)
225
+ return;
226
+ const width = this.Length * this.positionWidth;
227
+ this.canvas.width = width;
228
+ this.canvas.style.width = `${width}px`;
229
+ this.root.style.width = `${this.canvas.width}px`;
230
+ this.positions = new Array(this.Length);
231
+ for (let jPos = 0; jPos < this.Length; jPos++) {
232
+ const posName = this.positionNames[this.startPosition + jPos];
233
+ this.positions[jPos] = new PositionInfo(posName);
234
+ }
235
+ // 2022-05-05 askalkin instructed to show WebLogo based on filter (not selection)
236
+ const indices = this.dataFrame.filter.getSelectedIndexes();
237
+ // const indices = this.dataFrame.selection.trueCount > 0 ? this.dataFrame.selection.getSelectedIndexes() :
238
+ // this.dataFrame.filter.getSelectedIndexes();
239
+ this.rowsMasked = indices.length;
240
+ this.rowsNull = 0;
241
+ for (const i of indices) {
242
+ let s = (this.seqCol.get(i));
243
+ if (!s) {
244
+ s = this._nullSequence();
245
+ ++this.rowsNull;
246
+ }
247
+ else {
248
+ for (let jPos = 0; jPos < this.Length; jPos++) {
249
+ const pmInfo = this.positions[jPos].freq;
250
+ const m = s[this.startPosition + jPos];
251
+ if (!(m in pmInfo))
252
+ pmInfo[m] = new PositionMonomerInfo();
253
+ pmInfo[m].count++;
254
+ }
255
+ }
256
+ }
257
+ //#region Polish freq counts
258
+ for (let jPos = 0; jPos < this.Length; jPos++) {
259
+ // delete this.positions[jPos].freq['-'];
260
+ this.positions[jPos].rowCount = 0;
261
+ for (const m in this.positions[jPos].freq)
262
+ this.positions[jPos].rowCount += this.positions[jPos].freq[m].count;
263
+ }
264
+ //#endregion
265
+ const maxHeight = this.canvas.height - this.axisHeight;
266
+ // console.debug(`WebLogo._calculate() maxHeight=${maxHeight}.`);
267
+ //#region Calculate screen
268
+ for (let jPos = 0; jPos < this.Length; jPos++) {
269
+ const freq = this.positions[jPos].freq;
270
+ const rowCount = this.positions[jPos].rowCount;
271
+ let y = this.axisHeight;
272
+ const entries = Object.entries(freq).sort((a, b) => {
273
+ if (a[0] !== '-' && b[0] !== '-')
274
+ return b[1].count - a[1].count;
275
+ else if (a[0] === '-' && b[0] === '-')
276
+ return 0;
277
+ else if (a[0] === '-')
278
+ return -1;
279
+ else /* (b[0] === '-') */
280
+ return +1;
281
+ });
282
+ for (const entry of entries) {
283
+ const pmInfo = entry[1];
284
+ // const m: string = entry[0];
285
+ const h = maxHeight * pmInfo.count / rowCount;
286
+ pmInfo.bounds = new DG.Rect(jPos * this.positionWidth, y, this.positionWidth, h);
287
+ y += h;
288
+ }
289
+ }
290
+ //#endregion
291
+ }
292
+ // reflect changes made to filter/selection
293
+ render(recalc = true) {
294
+ var _a;
295
+ if (!this.canvas || !this.seqCol || !this.dataFrame || !this.cp ||
296
+ this.startPosition === -1 || this.endPosition === -1)
297
+ return;
298
+ const g = this.canvas.getContext('2d');
299
+ if (!g)
300
+ return;
301
+ if (recalc)
302
+ this._calculate();
303
+ // let rowCount = this.rowsMasked;
304
+ // if (!this.considerNullSequences)
305
+ // rowCount -= this.rowsNull;
306
+ g.resetTransform();
307
+ g.clearRect(0, 0, this.canvas.width, this.canvas.height);
308
+ g.textBaseline = this.textBaseline;
309
+ for (let jPos = 0; jPos < this.Length; jPos++) {
310
+ const pos = this.positions[jPos];
311
+ g.resetTransform();
312
+ g.fillStyle = 'black';
313
+ g.textAlign = 'center';
314
+ g.font = '10px Roboto, Roboto Local, sans-serif';
315
+ const posNameTm = g.measureText(pos.name);
316
+ const jPosWidth = posNameTm.width < (this.positionWidth - 2) ? posNameTm.width : (this.positionWidth - 2);
317
+ g.setTransform(jPosWidth / posNameTm.width, 0, 0, 1, jPos * this.positionWidth + this.positionWidth / 2, 0);
318
+ g.fillText(pos.name, 0, 0);
319
+ for (const [monomer, pmInfo] of Object.entries(this.positions[jPos].freq)) {
320
+ if (monomer !== '-') {
321
+ const b = pmInfo.bounds;
322
+ const fontStyle = '16px Roboto, Roboto Local, sans-serif';
323
+ // Hacks to scale uppercase characters to target rectangle
324
+ const uppercaseLetterAscent = 0.25;
325
+ const uppercaseLetterHeight = 12.2;
326
+ g.resetTransform();
327
+ g.strokeStyle = 'lightgray';
328
+ g.lineWidth = 1;
329
+ g.rect(b.left, b.top, b.width, b.height);
330
+ g.fillStyle = (_a = this.cp[monomer]) !== null && _a !== void 0 ? _a : this.cp['other'];
331
+ g.textAlign = 'left';
332
+ g.font = fontStyle;
333
+ //g.fillRect(b.left, b.top, b.width, b.height);
334
+ const mTm = g.measureText(monomer);
335
+ // if (mM.actualBoundingBoxAscent != 0)
336
+ // console.debug(`m: ${m}, mM.actualBoundingBoxAscent: ${mM.actualBoundingBoxAscent}`);
337
+ g.setTransform(b.width / mTm.width, 0, 0, b.height / uppercaseLetterHeight, b.left, b.top);
338
+ g.fillText(monomer, 0, -uppercaseLetterAscent);
339
+ }
340
+ }
341
+ }
342
+ }
343
+ }
344
+ WebLogo.residuesSet = 'nucleotides';
345
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"web-logo.js","sourceRoot":"","sources":["web-logo.ts"],"names":[],"mappings":";;;;;;;;;AACA,OAAO,KAAK,EAAE,MAAM,iBAAiB,CAAC;AACtC,OAAO,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAEtC,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAI7B,OAAO,EAAC,UAAU,EAAE,kBAAkB,EAAC,MAAM,eAAe,CAAC;AAC7D,OAAO,EAAC,WAAW,EAAE,mBAAmB,EAAC,MAAM,gBAAgB,CAAC;AAgBhE,iBAAiB,CAAC,SAAS,CAAC,iBAAiB,GAAG,UAAS,KAAiB;IACxE,MAAM,IAAI,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC1C,OAAO,IAAI,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC;AAC3E,CAAC,CAAC;AAEF,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,GAAG,UAAS,CAAS,EAAE,CAAS;IACxD,OAAO,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC;AAChF,CAAC,CAAC;AAEF,MAAM,OAAO,mBAAmB;IAS9B;QACE,IAAI,CAAC,KAAK,GAAG,CAAC,CAAC;QACf,IAAI,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;IACxC,CAAC;CACF;AAED,MAAM,OAAO,YAAY;IAKvB;;OAEG;IACH,YAAY,IAAY;QACtB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,IAAI,GAAG,EAAE,CAAC;QACf,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC;IACpB,CAAC;CACF;AAED,MAAM,OAAO,OAAQ,SAAQ,EAAE,CAAC,QAAQ;IAqCtC;QACE,KAAK,EAAE,CAAC;QAnCV,4FAA4F;QAClF,OAAE,GAA4B,IAAI,CAAC;QAOrC,eAAU,GAAW,EAAE,CAAC;QACxB,kBAAa,GAAW,EAAE,CAAC;QAE3B,WAAM,GAAqB,IAAI,CAAC;QACxC,mCAAmC;QAC3B,cAAS,GAAmB,EAAE,CAAC;QAE/B,eAAU,GAAW,CAAC,CAAC;QACvB,aAAQ,GAAW,CAAC,CAAC;QAUrB,kBAAa,GAAa,EAAE,CAAC;QAE7B,kBAAa,GAAW,CAAC,CAAC,CAAC;QAE3B,gBAAW,GAAW,CAAC,CAAC,CAAC;QAO/B,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAE1B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;QAC7C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,GAAG,CAAC,CAAC;QAE9C,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,IAAI,CAAC,uBAAuB,EAAE,KAAK,CAAC,CAAC;QACvE,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM,CAAC,oBAAoB,EAAE,IAAI,CAAC,CAAC;QAElE,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;QAChE,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE,IAAI,CAAC,CAAC;IAC9D,CAAC;IAfD,IAAY,MAAM,KAAa,OAAO,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC;IAiB5E,IAAI;;YACR,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC;YAC1B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC;YAEjC,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,WAAW,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;YAC1C,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC;YACtC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC;YAEvC,sDAAsD;YAEtD,MAAM,UAAU,GAAG,CAAC,CAAW,EAAuD,EAAE;gBACtF,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC;gBAClD,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBAEtC,IAAI,QAAQ,KAAK,KAAK,CAAC;oBACrB,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBAE5B,MAAM,OAAO,GAAuB,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;qBAC3D,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBAC3D,IAAI,OAAO,KAAK,SAAS;oBACvB,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBAE5B,OAAO,CAAC,IAAI,EAAE,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;YACjD,CAAC,CAAC;YAEF,IAAI,CAAC,MAAM,CAAC,WAAW,GAAG,CAAC,CAAa,EAAE,EAAE;YAE5C,CAAC,CAAC;YAEF,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,SAAS,CAAC,CAAC,CAAQ,EAAE,EAAE;gBAC9D,IAAI,CAAC,IAAI,CAAC,MAAM;oBACd,OAAO;gBAET,MAAM,IAAI,GAAG,CAAe,CAAC;gBAC7B,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC;gBAExE,iDAAiD;gBACjD,IAAI,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,MAAM,IAAI,OAAO,EAAE;oBAC5C,EAAE,CAAC,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,IAAI,EAAE,EAAE;wBAC/C,MAAM,GAAG,GAAG,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;wBACnC,MAAM,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;wBACzD,OAAO,IAAI,KAAK,OAAO,CAAC;oBAC1B,CAAC,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;iBAC9B;qBAAM;oBACL,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;iBACnB;YACH,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,SAAS,CAAC,CAAC,CAAQ,EAAE,EAAE;gBAC9D,IAAI,CAAC,IAAI,CAAC,MAAM;oBACd,OAAO;gBAET,MAAM,IAAI,GAAG,CAAe,CAAC;gBAC7B,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC;gBAExE,uDAAuD;gBACvD,iDAAiD;gBACjD,IAAI,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,MAAM,IAAI,OAAO,EAAE;oBAC5C,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE;wBACrC,MAAM,GAAG,GAAG,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;wBACnC,MAAM,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;wBACzD,OAAO,IAAI,KAAK,OAAO,CAAC;oBAC1B,CAAC,CAAC,CAAC;iBACJ;YACH,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC9B,2CAA2C;YAC3C,+BAA+B;YAE/B,gFAAgF;YAChF,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;YAEzE,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACpB,CAAC;KAAA;IAED,mCAAmC;IACnC,iDAAiD;IACjD,mDAAmD;IACnD,uGAAuG;IACvG,wBAAwB;IACxB,IAAI;IAEJ,iBAAiB,CAAC,IAAS;QACzB,IAAI,CAAC,IAAI,CAAC,MAAM;YACd,OAAO;QAET,0DAA0D;QAE1D,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;QAC1F,sFAAsF;QACtF,+DAA+D;QAC/D,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC;QAC5B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC;QAEzC,iDAAiD;QACjD,qFAAqF;QACrF,iFAAiF;QAEjF,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IACpB,CAAC;IAED;OACG;IACH,YAAY;QACV,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;YAC1D,IAAI,IAAI,CAAC,MAAM,EAAE;gBACf,IAAI,SAAS,GAAG,CAAC,CAAC;gBAClB,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,MAAM,CAAC,UAAU;oBAC3C,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,QAAQ,CAAC,MAAM,CAAC,CAAC;gBAEnD,0DAA0D;gBAC1D,MAAM,gBAAgB,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;gBAC7D,kDAAkD;gBAClD,IAAI,CAAC,aAAa,GAAG,gBAAgB,CAAC,CAAC,CAAC,gBAAgB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;oBACzF,CAAC,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,GAAG,IAAI,GAAG,CAAC,EAAE,CAAC,CAAC;gBAE5D,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,aAAa,CAAC,CAAC;oBACjE,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBACzD,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,aAAa,CAAC,CAAC;oBAC7D,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;gBAE/D,uBAAuB;gBACvB,QAAQ,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE;oBAC7B,KAAK,UAAU,CAAC,wBAAwB;wBACtC,IAAI,CAAC,EAAE,GAAG,kBAAkB,CAAC,UAAU,CAAC;wBACxC,MAAM;oBACR,KAAK,WAAW,CAAC,wBAAwB;wBACvC,IAAI,CAAC,EAAE,GAAG,mBAAmB,CAAC,YAAY,CAAC;wBAC3C,MAAM;iBACP;gBACD,0BAA0B;aAC3B;iBAAM;gBACL,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;gBACf,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;gBACxB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;gBACxB,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC;aACvB;SACF;QACD,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;IAED,iBAAiB,CAAC,QAAqB;QACrC,KAAK,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;QAElC,QAAQ,QAAQ,CAAC,IAAI,EAAE;YACvB,KAAK,uBAAuB;gBAC1B,IAAI,CAAC,MAAM,EAAE,CAAC;gBACd,MAAM;YACR,KAAK,oBAAoB;gBACvB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,mBAAmB;gBACtB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,iBAAiB;gBACpB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;SACP;IACH,CAAC;IAED,eAAe;QACb,IAAI,CAAC,YAAY,EAAE,CAAC;QAEpB,IAAI,IAAI,CAAC,SAAS,KAAK,KAAK,CAAC,EAAE;YAC7B,4BAA4B;YAC5B,iFAAiF;YACjF,yFAAyF;YACzF,wEAAwE;YACxE,+EAA+E;YAC/E,+EAA+E;YAC/E,8EAA8E;YAE9E,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;YACnE,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;SACjE;IACH,CAAC;IAES,aAAa,CAAC,aAAa,GAAG,GAAG;QACzC,IAAI,IAAI,CAAC,qBAAqB;YAC5B,OAAO,IAAI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAE7D,OAAO,EAAE,CAAC;IACZ,CAAC;IAES,UAAU;QAClB,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,aAAa,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,WAAW,KAAK,CAAC,CAAC;YACzG,OAAO;QAET,MAAM,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC;QAC/C,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,KAAK,CAAC;QAC1B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,KAAK,IAAI,CAAC;QACvC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,CAAC;QAEjD,IAAI,CAAC,SAAS,GAAG,IAAI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACxC,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,OAAO,GAAW,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC;YACtE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,IAAI,YAAY,CAAC,OAAO,CAAC,CAAC;SAClD;QAED,iFAAiF;QACjF,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC;QAC3D,2GAA2G;QAC3G,gDAAgD;QAEhD,IAAI,CAAC,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC;QACjC,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC;QAElB,KAAK,MAAM,CAAC,IAAI,OAAO,EAAE;YACvB,IAAI,CAAC,GAAmB,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;YAE7C,IAAI,CAAC,CAAC,EAAE;gBACN,CAAC,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;gBACzB,EAAE,IAAI,CAAC,QAAQ,CAAC;aACjB;iBAAM;gBACL,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;oBAC7C,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;oBACzC,MAAM,CAAC,GAAW,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC;oBAC/C,IAAI,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC;wBAChB,MAAM,CAAC,CAAC,CAAC,GAAG,IAAI,mBAAmB,EAAE,CAAC;oBAExC,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC;iBACnB;aACF;SACF;QAED,4BAA4B;QAC5B,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,yCAAyC;YAEzC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,GAAG,CAAC,CAAC;YAClC,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI;gBACvC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;SACvE;QACD,YAAY;QAEZ,MAAM,SAAS,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC;QACvD,iEAAiE;QAEjE,0BAA0B;QAC1B,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,IAAI,GAAyC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;YAC7E,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC;YAE/C,IAAI,CAAC,GAAW,IAAI,CAAC,UAAU,CAAC;YAEhC,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;gBACjD,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBAC9B,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;qBAC5B,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBACnC,OAAO,CAAC,CAAC;qBACN,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBACnB,OAAO,CAAC,CAAC,CAAC;qBACP,oBAAoB;oBACvB,OAAO,CAAC,CAAC,CAAC;YACd,CAAC,CAAC,CAAC;YACH,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE;gBAC3B,MAAM,MAAM,GAAwB,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC7C,8BAA8B;gBAC9B,MAAM,CAAC,GAAW,SAAS,GAAG,MAAM,CAAC,KAAK,GAAG,QAAQ,CAAC;gBAEtD,MAAM,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC,EAAE,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC;gBACjF,CAAC,IAAI,CAAC,CAAC;aACR;SACF;QACD,YAAY;IACd,CAAC;IAED,2CAA2C;IAC3C,MAAM,CAAC,MAAM,GAAG,IAAI;;QAClB,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,EAAE;YAC7D,IAAI,CAAC,aAAa,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,WAAW,KAAK,CAAC,CAAC;YACpD,OAAO;QAET,MAAM,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QACvC,IAAI,CAAC,CAAC;YAAE,OAAO;QAEf,IAAI,MAAM;YACR,IAAI,CAAC,UAAU,EAAE,CAAC;QAEpB,kCAAkC;QAClC,mCAAmC;QACnC,+BAA+B;QAE/B,CAAC,CAAC,cAAc,EAAE,CAAC;QACnB,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACzD,CAAC,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QAEnC,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,GAAG,GAAiB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YAC/C,CAAC,CAAC,cAAc,EAAE,CAAC;YACnB,CAAC,CAAC,SAAS,GAAG,OAAO,CAAC;YACtB,CAAC,CAAC,SAAS,GAAG,QAAQ,CAAC;YACvB,CAAC,CAAC,IAAI,GAAG,uCAAuC,CAAC;YACjD,MAAM,SAAS,GAAgB,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACvD,MAAM,SAAS,GAAG,SAAS,CAAC,KAAK,GAAG,CAAC,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;YAC1G,CAAC,CAAC,YAAY,CACZ,SAAS,GAAG,SAAS,CAAC,KAAK,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EACpC,IAAI,GAAG,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;YACzD,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;YAE3B,KAAK,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE;gBACzE,IAAI,OAAO,KAAK,GAAG,EAAE;oBACnB,MAAM,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC;oBAExB,MAAM,SAAS,GAAG,uCAAuC,CAAC;oBAC1D,0DAA0D;oBAC1D,MAAM,qBAAqB,GAAG,IAAI,CAAC;oBACnC,MAAM,qBAAqB,GAAG,IAAI,CAAC;oBAEnC,CAAC,CAAC,cAAc,EAAE,CAAC;oBACnB,CAAC,CAAC,WAAW,GAAG,WAAW,CAAC;oBAC5B,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC;oBAChB,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC;oBACzC,CAAC,CAAC,SAAS,GAAG,MAAA,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC,mCAAI,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC;oBACnD,CAAC,CAAC,SAAS,GAAG,MAAM,CAAC;oBACrB,CAAC,CAAC,IAAI,GAAG,SAAS,CAAC;oBACnB,+CAA+C;oBAC/C,MAAM,GAAG,GAAgB,CAAC,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;oBAEhD,uCAAuC;oBACvC,yFAAyF;oBAEzF,CAAC,CAAC,YAAY,CACZ,CAAC,CAAC,KAAK,GAAG,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,MAAM,GAAG,qBAAqB,EAC3D,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC;oBACjB,CAAC,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC,qBAAqB,CAAC,CAAC;iBAChD;aACF;SACF;IACH,CAAC;;AApYa,mBAAW,GAAG,aAAa,CAAC","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\n\nimport * as rxjs from 'rxjs';\n\nimport {StringDictionary} from '@datagrok-libraries/utils/src/type-declarations';\n\nimport {Aminoacids, AminoacidsPalettes} from '../aminoacids';\nimport {Nucleotides, NucleotidesPalettes} from '../nucleotides';\n\n// Using color schemes from chem-palette\n\ndeclare module 'datagrok-api/src/grid' {\n  interface Rect {\n    contains(x: number, y: number): boolean;\n  }\n}\n\ndeclare global {\n  interface HTMLCanvasElement {\n    getCursorPosition(event: MouseEvent): DG.Point;\n  }\n}\n\nHTMLCanvasElement.prototype.getCursorPosition = function(event: MouseEvent): DG.Point {\n  const rect = this.getBoundingClientRect();\n  return new DG.Point(event.clientX - rect.left, event.clientY - rect.top);\n};\n\nDG.Rect.prototype.contains = function(x: number, y: number): boolean {\n  return this.left <= x && x <= this.right && this.top <= y && y <= this.bottom;\n};\n\nexport class PositionMonomerInfo {\n  /** Sequences count with monomer in position\n   */\n  count: number;\n\n  /** Remember screen coords rect\n   */\n  bounds: DG.Rect;\n\n  constructor() {\n    this.count = 0;\n    this.bounds = new DG.Rect(0, 0, 0, 0);\n  }\n}\n\nexport class PositionInfo {\n  public readonly name: string;\n  freq: { [m: string]: PositionMonomerInfo };\n  rowCount: number;\n\n  /** freq = {}, rowCount = 0\n   * @param {string} name Name of position ('111A', '111.1', etc)\n   */\n  constructor(name: string) {\n    this.name = name;\n    this.freq = {};\n    this.rowCount = 0;\n  }\n}\n\nexport class WebLogo extends DG.JsViewer {\n  public static residuesSet = 'nucleotides';\n\n  // private readonly colorScheme: ColorScheme = ColorSchemes[NucleotidesWebLogo.residuesSet];\n  protected cp: StringDictionary | null = null;\n\n  // private readonly host: HTMLDivElement;\n  private canvas?: HTMLCanvasElement;\n  private slider?: DG.RangeSlider;\n  private textBaseline: CanvasTextBaseline;\n\n  private axisHeight: number = 12;\n  private positionWidth: number = 16;\n\n  private seqCol: DG.Column | null = null;\n  // private maxLength: number = 100;\n  private positions: PositionInfo[] = [];\n\n  private rowsMasked: number = 0;\n  private rowsNull: number = 0;\n\n  // Viewer's properties (likely they should be public so that they can be set outside)\n  public minHeight: number;\n  public maxHeight: number;\n  public considerNullSequences: boolean;\n  public sequenceColumnName: string;\n  public startPositionName: string;\n  public endPositionName: string;\n\n  private positionNames: string[] = [];\n\n  private startPosition: number = -1;\n\n  private endPosition: number = -1;\n\n  private get Length(): number { return this.endPosition - this.startPosition + 1; }\n\n  constructor() {\n    super();\n\n    this.textBaseline = 'top';\n\n    this.minHeight = this.float('minHeight', 50);\n    this.maxHeight = this.float('maxHeight', 100);\n\n    this.considerNullSequences = this.bool('considerNullSequences', false);\n    this.sequenceColumnName = this.string('sequenceColumnName', null);\n\n    this.startPositionName = this.string('startPositionName', null);\n    this.endPositionName = this.string('endPositionName', null);\n  }\n\n  async init(): Promise<void> {\n    this.canvas = ui.canvas();\n    this.canvas.style.width = '100%';\n\n    this.slider = ui.rangeSlider(0, 20, 2, 5);\n    this.slider.root.style.width = '100%';\n    this.slider.root.style.height = '12px';\n\n    // this.host = ui.divV([/*this.slider,*/this.canvas]);\n\n    const getMonomer = (p: DG.Point): [number, string | null, PositionMonomerInfo | null] => {\n      const jPos = Math.floor(p.x / this.positionWidth);\n      const position = this.positions[jPos];\n\n      if (position === void 0)\n        return [jPos, null, null];\n\n      const monomer: string | undefined = Object.keys(position.freq)\n        .find((m) => position.freq[m].bounds.contains(p.x, p.y));\n      if (monomer === undefined)\n        return [jPos, null, null];\n\n      return [jPos, monomer, position.freq[monomer]];\n    };\n\n    this.canvas.onmouseover = (e: MouseEvent) => {\n\n    };\n\n    rxjs.fromEvent(this.canvas, 'mousemove').subscribe((e: Event) => {\n      if (!this.canvas)\n        return;\n\n      const args = e as MouseEvent;\n      const [jPos, monomer] = getMonomer(this.canvas.getCursorPosition(args));\n\n      //if (this.dataFrame && this.seqCol && monomer) {\n      if (this.dataFrame && this.seqCol && monomer) {\n        ui.tooltip.showRowGroup(this.dataFrame, (iRow) => {\n          const seq = this.seqCol!.get(iRow);\n          const mSeq = seq ? seq[this.startPosition + jPos] : null;\n          return mSeq === monomer;\n        }, args.x + 16, args.y + 16);\n      } else {\n        ui.tooltip.hide();\n      }\n    });\n\n    rxjs.fromEvent(this.canvas, 'mousedown').subscribe((e: Event) => {\n      if (!this.canvas)\n        return;\n\n      const args = e as MouseEvent;\n      const [jPos, monomer] = getMonomer(this.canvas.getCursorPosition(args));\n\n      // prevents deselect all rows if we miss monomer bounds\n      //if (this.dataFrame && this.seqCol && monomer) {\n      if (this.dataFrame && this.seqCol && monomer) {\n        this.dataFrame.selection.init((iRow) => {\n          const seq = this.seqCol!.get(iRow);\n          const mSeq = seq ? seq[this.startPosition + jPos] : null;\n          return mSeq === monomer;\n        });\n      }\n    });\n\n    this.root.append(this.canvas);\n    // this.root.appendChild(this.slider.root);\n    // this.root.append(this.host);\n\n    // ui.onSizeChanged(this.canvas).subscribe(this.canvasOnSizeChanged.bind(this));\n    ui.onSizeChanged(this.root).subscribe(this.rootOnSizeChanged.bind(this));\n\n    this.render(true);\n  }\n\n  // canvasOnSizeChanged(args: any) {\n  //   this.canvas.width = this.canvas.clientWidth;\n  //   this.canvas.height = this.canvas.clientHeight;\n  //   console.debug(`WebLogo.onSizeChanged() width=${this.canvas.width}, height=${this.canvas.height}`);\n  //   this.render(false);\n  // }\n\n  rootOnSizeChanged(args: any) {\n    if (!this.canvas)\n      return;\n\n    // this.canvas.width calculate in this._calculate() method\n\n    const height = Math.min(this.maxHeight, Math.max(this.minHeight, this.root.clientHeight));\n    // if (this.canvas.width > this.root.clientWidth) /* horizontal scroller is enabled */\n    //   height -= 6; /* free some space for horizontal scroller */\n    this.canvas.height = height;\n    this.canvas.style.height = `${height}px`;\n\n    // console.debug(`WebLogo.onRootSizeChanged() ` +\n    //   `root.width=${this.root.clientWidth}, root.height=${this.root.clientHeight}, ` +\n    //   `canvas.width=${this.canvas.width}, canvas.height=${this.canvas.height} .`);\n\n    this.render(true);\n  }\n\n  /** Assigns {@link seqCol} and {@link cp} based on {@link sequenceColumnName} and calls {@link render}().\n   */\n  updateSeqCol(): void {\n    if (this.dataFrame) {\n      this.seqCol = this.dataFrame.col(this.sequenceColumnName);\n      if (this.seqCol) {\n        let maxLength = 0;\n        for (const category of this.seqCol.categories)\n          maxLength = Math.max(maxLength, category.length);\n\n        // Get position names from data column tag 'positionNames'\n        const positionNamesTxt = this.seqCol.getTag('positionNames');\n        // Fallback if 'positionNames' tag is not provided\n        this.positionNames = positionNamesTxt ? positionNamesTxt.split(', ').map((n) => n.trim()) :\n          [...Array(maxLength).keys()].map((jPos) => `${jPos + 1}`);\n\n        this.startPosition = this.startPositionName && this.positionNames ?\n          this.positionNames.indexOf(this.startPositionName) : 0;\n        this.endPosition = this.endPositionName && this.positionNames ?\n          this.positionNames.indexOf(this.endPositionName) : maxLength;\n\n        //#region -- palette --\n        switch (this.seqCol.semType) {\n        case Aminoacids.SemTypeMultipleAlignment:\n          this.cp = AminoacidsPalettes.GrokGroups;\n          break;\n        case Nucleotides.SemTypeMultipleAlignment:\n          this.cp = NucleotidesPalettes.Chromatogram;\n          break;\n        }\n        //#endregion -- palette --\n      } else {\n        this.cp = null;\n        this.positionNames = [];\n        this.startPosition = -1;\n        this.endPosition = -1;\n      }\n    }\n    this.render();\n  }\n\n  onPropertyChanged(property: DG.Property): void {\n    super.onPropertyChanged(property);\n\n    switch (property.name) {\n    case 'considerNullSequences':\n      this.render();\n      break;\n    case 'sequenceColumnName':\n      this.updateSeqCol();\n      break;\n    case 'startPositionName':\n      this.updateSeqCol();\n      break;\n    case 'endPositionName':\n      this.updateSeqCol();\n      break;\n    case 'minHeight':\n      this.render(true);\n      break;\n    case 'maxHeight':\n      this.render(true);\n      break;\n    }\n  }\n\n  onTableAttached() {\n    this.updateSeqCol();\n\n    if (this.dataFrame !== void 0) {\n      // There are two approaches:\n      // first  - look in the dataFrame for the first matching column by semType of the\n      //          corresponding viewer (but we want only one class of a more universal viewer),\n      // second - draw column data if the passed column is of suitable semType\n      // We decided that we will not search, but we will display asked data if we can\n      // const semType = (<typeof NucleotidesWebLogo>(this.constructor)).residuesSet;\n      // this.seqCol = (this.dataFrame.columns as DG.ColumnList).bySemType(semType);\n\n      this.dataFrame.selection.onChanged.subscribe((_) => this.render());\n      this.dataFrame.filter.onChanged.subscribe((_) => this.render());\n    }\n  }\n\n  protected _nullSequence(fillerResidue = 'X'): string {\n    if (this.considerNullSequences)\n      return new Array(this.Length).fill(fillerResidue).join('');\n\n    return '';\n  }\n\n  protected _calculate() {\n    if (!this.canvas || !this.seqCol || !this.dataFrame || this.startPosition === -1 || this.endPosition === -1)\n      return;\n\n    const width = this.Length * this.positionWidth;\n    this.canvas.width = width;\n    this.canvas.style.width = `${width}px`;\n    this.root.style.width = `${this.canvas.width}px`;\n\n    this.positions = new Array(this.Length);\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const posName: string = this.positionNames[this.startPosition + jPos];\n      this.positions[jPos] = new PositionInfo(posName);\n    }\n\n    // 2022-05-05 askalkin instructed to show WebLogo based on filter (not selection)\n    const indices = this.dataFrame.filter.getSelectedIndexes();\n    // const indices = this.dataFrame.selection.trueCount > 0 ? this.dataFrame.selection.getSelectedIndexes() :\n    //   this.dataFrame.filter.getSelectedIndexes();\n\n    this.rowsMasked = indices.length;\n    this.rowsNull = 0;\n\n    for (const i of indices) {\n      let s: string = <string>(this.seqCol.get(i));\n\n      if (!s) {\n        s = this._nullSequence();\n        ++this.rowsNull;\n      } else {\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n          const pmInfo = this.positions[jPos].freq;\n          const m: string = s[this.startPosition + jPos];\n          if (!(m in pmInfo))\n            pmInfo[m] = new PositionMonomerInfo();\n\n          pmInfo[m].count++;\n        }\n      }\n    }\n\n    //#region Polish freq counts\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      // delete this.positions[jPos].freq['-'];\n\n      this.positions[jPos].rowCount = 0;\n      for (const m in this.positions[jPos].freq)\n        this.positions[jPos].rowCount += this.positions[jPos].freq[m].count;\n    }\n    //#endregion\n\n    const maxHeight = this.canvas.height - this.axisHeight;\n    // console.debug(`WebLogo._calculate() maxHeight=${maxHeight}.`);\n\n    //#region Calculate screen\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const freq: { [c: string]: PositionMonomerInfo } = this.positions[jPos].freq;\n      const rowCount = this.positions[jPos].rowCount;\n\n      let y: number = this.axisHeight;\n\n      const entries = Object.entries(freq).sort((a, b) => {\n        if (a[0] !== '-' && b[0] !== '-')\n          return b[1].count - a[1].count;\n        else if (a[0] === '-' && b[0] === '-')\n          return 0;\n        else if (a[0] === '-')\n          return -1;\n        else /* (b[0] === '-') */\n          return +1;\n      });\n      for (const entry of entries) {\n        const pmInfo: PositionMonomerInfo = entry[1];\n        // const m: string = entry[0];\n        const h: number = maxHeight * pmInfo.count / rowCount;\n\n        pmInfo.bounds = new DG.Rect(jPos * this.positionWidth, y, this.positionWidth, h);\n        y += h;\n      }\n    }\n    //#endregion\n  }\n\n  // reflect changes made to filter/selection\n  render(recalc = true) {\n    if (!this.canvas || !this.seqCol || !this.dataFrame || !this.cp ||\n      this.startPosition === -1 || this.endPosition === -1)\n      return;\n\n    const g = this.canvas.getContext('2d');\n    if (!g) return;\n\n    if (recalc)\n      this._calculate();\n\n    // let rowCount = this.rowsMasked;\n    // if (!this.considerNullSequences)\n    //   rowCount -= this.rowsNull;\n\n    g.resetTransform();\n    g.clearRect(0, 0, this.canvas.width, this.canvas.height);\n    g.textBaseline = this.textBaseline;\n\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const pos: PositionInfo = this.positions[jPos];\n      g.resetTransform();\n      g.fillStyle = 'black';\n      g.textAlign = 'center';\n      g.font = '10px Roboto, Roboto Local, sans-serif';\n      const posNameTm: TextMetrics = g.measureText(pos.name);\n      const jPosWidth = posNameTm.width < (this.positionWidth - 2) ? posNameTm.width : (this.positionWidth - 2);\n      g.setTransform(\n        jPosWidth / posNameTm.width, 0, 0, 1,\n        jPos * this.positionWidth + this.positionWidth / 2, 0);\n      g.fillText(pos.name, 0, 0);\n\n      for (const [monomer, pmInfo] of Object.entries(this.positions[jPos].freq)) {\n        if (monomer !== '-') {\n          const b = pmInfo.bounds;\n\n          const fontStyle = '16px Roboto, Roboto Local, sans-serif';\n          // Hacks to scale uppercase characters to target rectangle\n          const uppercaseLetterAscent = 0.25;\n          const uppercaseLetterHeight = 12.2;\n\n          g.resetTransform();\n          g.strokeStyle = 'lightgray';\n          g.lineWidth = 1;\n          g.rect(b.left, b.top, b.width, b.height);\n          g.fillStyle = this.cp[monomer] ?? this.cp['other'];\n          g.textAlign = 'left';\n          g.font = fontStyle;\n          //g.fillRect(b.left, b.top, b.width, b.height);\n          const mTm: TextMetrics = g.measureText(monomer);\n\n          // if (mM.actualBoundingBoxAscent != 0)\n          //   console.debug(`m: ${m}, mM.actualBoundingBoxAscent: ${mM.actualBoundingBoxAscent}`);\n\n          g.setTransform(\n            b.width / mTm.width, 0, 0, b.height / uppercaseLetterHeight,\n            b.left, b.top);\n          g.fillText(monomer, 0, -uppercaseLetterAscent);\n        }\n      }\n    }\n  }\n}\n"]}