genoverse 3.2.0 → 4.0.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/.eslintrc.js +93 -162
- package/.github/workflows/test.yml +9 -10
- package/.github/workflows/update-gh-pages.yml +33 -0
- package/LICENSE.TXT +2 -2
- package/README.md +174 -3
- package/{i → assets}/sort_handle.png +0 -0
- package/babel.config.js +19 -0
- package/dist/129.css +334 -0
- package/dist/129.css.map +1 -0
- package/dist/129.genoverse.js +2 -0
- package/dist/129.genoverse.js.map +1 -0
- package/dist/15d98c18221c8bcb2334.ttf +0 -0
- package/dist/166.css +2 -0
- package/dist/166.genoverse.js +1 -0
- package/dist/216.css +20 -0
- package/dist/216.css.map +1 -0
- package/dist/232.css +114 -0
- package/dist/232.css.map +1 -0
- package/dist/232.genoverse.js +2 -0
- package/dist/232.genoverse.js.map +1 -0
- package/dist/2e659e443f3e98569e9f.png +0 -0
- package/dist/394.css +114 -0
- package/dist/394.css.map +1 -0
- package/dist/394.genoverse.js +2 -0
- package/dist/394.genoverse.js.map +1 -0
- package/dist/469.css +24 -0
- package/dist/469.css.map +1 -0
- package/dist/469.genoverse.js +2 -0
- package/dist/469.genoverse.js.map +1 -0
- package/dist/4896d4b04430cc3dfb06.woff2 +0 -0
- package/dist/530.css +39 -0
- package/dist/530.css.map +1 -0
- package/dist/530.genoverse.js +2 -0
- package/dist/530.genoverse.js.map +1 -0
- package/dist/547.css +469 -0
- package/dist/547.css.map +1 -0
- package/dist/547.genoverse.js +1 -0
- package/dist/729.css +315 -0
- package/dist/729.css.map +1 -0
- package/dist/79da213423ac0def2058.ttf +0 -0
- package/dist/804.genoverse.js +2 -0
- package/dist/804.genoverse.js.map +1 -0
- package/dist/842.genoverse.js +2 -0
- package/dist/842.genoverse.js.map +1 -0
- package/dist/893.genoverse.js +2 -0
- package/dist/893.genoverse.js.map +1 -0
- package/dist/949.css +315 -0
- package/dist/949.css.map +1 -0
- package/dist/949.genoverse.js +2 -0
- package/dist/949.genoverse.js.map +1 -0
- package/dist/952.css +315 -0
- package/dist/952.css.map +1 -0
- package/dist/952.genoverse.js +2 -0
- package/dist/952.genoverse.js.map +1 -0
- package/dist/d79c2ec96ab9ff1161a2.woff2 +0 -0
- package/dist/genoverse.js +2 -0
- package/dist/genoverse.js.map +1 -0
- package/index.html +13 -14
- package/jest.config.js +5 -0
- package/jest.setup.js +13 -0
- package/package.json +29 -12
- package/{css → src/css}/controlPanel.css +0 -0
- package/{css → src/css}/fileDrop.css +0 -0
- package/src/css/fontawesome.css +3 -0
- package/{css → src/css}/fullscreen.css +0 -0
- package/{css → src/css}/genoverse.css +1 -1
- package/{css → src/css}/karyotype.css +2 -0
- package/{css → src/css}/resizer.css +0 -0
- package/{css → src/css}/tooltips.css +0 -0
- package/{css → src/css}/trackControls.css +0 -0
- package/src/js/Genoverse.js +1747 -0
- package/{js → src/js}/Track/Controller/Sequence.js +6 -4
- package/src/js/Track/Controller/Stranded.js +83 -0
- package/{js → src/js}/Track/Controller.js +201 -160
- package/src/js/Track/Model/File/BAM.js +47 -0
- package/src/js/Track/Model/File/BED.js +122 -0
- package/src/js/Track/Model/File/GFF.js +42 -0
- package/src/js/Track/Model/File/VCF.js +109 -0
- package/src/js/Track/Model/File/WIG.js +82 -0
- package/src/js/Track/Model/File.js +36 -0
- package/src/js/Track/Model/Gene/Ensembl.js +24 -0
- package/{js → src/js}/Track/Model/Gene.js +3 -1
- package/src/js/Track/Model/Sequence/Ensembl.js +6 -0
- package/{js → src/js}/Track/Model/Sequence/Fasta.js +24 -17
- package/{js → src/js}/Track/Model/Sequence.js +10 -7
- package/{js → src/js}/Track/Model/SequenceVariation.js +17 -11
- package/{js → src/js}/Track/Model/Stranded.js +11 -8
- package/src/js/Track/Model/Transcript/Ensembl.js +73 -0
- package/{js → src/js}/Track/Model/Transcript.js +3 -1
- package/{js → src/js}/Track/Model.js +125 -93
- package/{js → src/js}/Track/View/Gene/Ensembl.js +6 -4
- package/src/js/Track/View/Gene.js +8 -0
- package/{js → src/js}/Track/View/Sequence.js +18 -22
- package/src/js/Track/View/SequenceVariation.js +117 -0
- package/src/js/Track/View/Transcript/Ensembl.js +17 -0
- package/src/js/Track/View/Transcript.js +32 -0
- package/{js → src/js}/Track/View.js +200 -159
- package/{js → src/js}/Track/library/Chromosome.js +18 -13
- package/src/js/Track/library/File/BAM.js +34 -0
- package/src/js/Track/library/File/BED.js +27 -0
- package/src/js/Track/library/File/BIGBED.js +51 -0
- package/src/js/Track/library/File/BIGWIG.js +54 -0
- package/src/js/Track/library/File/GFF.js +10 -0
- package/{js → src/js}/Track/library/File/VCF.js +29 -22
- package/src/js/Track/library/File/WIG.js +8 -0
- package/{js → src/js}/Track/library/File.js +4 -2
- package/src/js/Track/library/Gene.js +44 -0
- package/src/js/Track/library/Graph/Bar.js +263 -0
- package/src/js/Track/library/Graph/Line.js +335 -0
- package/{js → src/js}/Track/library/Graph.js +137 -114
- package/{js → src/js}/Track/library/HighlightRegion.js +118 -93
- package/src/js/Track/library/Legend.js +258 -0
- package/{js → src/js}/Track/library/Scalebar.js +69 -49
- package/{js → src/js}/Track/library/Scaleline.js +29 -27
- package/src/js/Track/library/Static.js +82 -0
- package/{js → src/js}/Track/library/dbSNP.js +47 -50
- package/src/js/Track.js +651 -0
- package/{js → src/js}/genomes/grch37.js +52 -52
- package/{js → src/js}/genomes/grch38.js +52 -52
- package/src/js/lib/BWReader.js +562 -0
- package/src/js/lib/VCFReader.js +296 -0
- package/src/js/lib/dalliance/bam.js +517 -0
- package/src/js/lib/dalliance/bin.js +317 -0
- package/src/js/lib/dalliance/jszlib-inflate.js +2159 -0
- package/src/js/lib/dalliance/lh3utils.js +105 -0
- package/src/js/lib/dalliance/sha1.js +334 -0
- package/src/js/lib/import-tracks.js +42 -0
- package/{js/lib → src/js/lib/jquery-plugins}/jquery.mousehold.js +0 -0
- package/{js/lib → src/js/lib/jquery-plugins}/jquery.mousewheel.js +0 -0
- package/{js/lib → src/js/lib/jquery-plugins}/jquery.tipsy.js +0 -0
- package/src/js/lib/jquery.js +26 -0
- package/src/js/lib/polyfills.js +11 -0
- package/src/js/lib/wrap-functions.js +88 -0
- package/src/js/plugins/controlPanel.js +388 -0
- package/src/js/plugins/fileDrop.js +81 -0
- package/src/js/plugins/focusRegion.js +13 -0
- package/{js → src/js}/plugins/fullscreen.js +18 -14
- package/{js → src/js}/plugins/karyotype.js +51 -45
- package/src/js/plugins/resizer.js +52 -0
- package/{js → src/js}/plugins/tooltips.js +31 -29
- package/src/js/plugins/trackControls.js +159 -0
- package/test/View/render-legends.test.js +1 -1
- package/test/change-width.test.js +71 -0
- package/test/create-and-destroy.test.js +2 -2
- package/test/track-ordering.test.js +3 -2
- package/test/track_config/config-settings.test.js +1 -1
- package/test/utils.js +4 -2
- package/webpack.config.js +103 -34
- package/css/font-awesome.css +0 -3
- package/expanded.html +0 -120
- package/fontawesome/css/fontawesome.min.css +0 -5
- package/fontawesome/css/regular.min.css +0 -5
- package/fontawesome/css/solid.min.css +0 -5
- package/fontawesome/webfonts/fa-brands-400.ttf +0 -0
- package/fontawesome/webfonts/fa-brands-400.woff +0 -0
- package/fontawesome/webfonts/fa-brands-400.woff2 +0 -0
- package/fontawesome/webfonts/fa-regular-400.ttf +0 -0
- package/fontawesome/webfonts/fa-regular-400.woff +0 -0
- package/fontawesome/webfonts/fa-regular-400.woff2 +0 -0
- package/fontawesome/webfonts/fa-solid-900.ttf +0 -0
- package/fontawesome/webfonts/fa-solid-900.woff +0 -0
- package/fontawesome/webfonts/fa-solid-900.woff2 +0 -0
- package/help.pdf +0 -0
- package/index.js +0 -83
- package/js/Genoverse.js +0 -1681
- package/js/Track/Controller/Stranded.js +0 -73
- package/js/Track/Model/File/BAM.js +0 -44
- package/js/Track/Model/File/BED.js +0 -116
- package/js/Track/Model/File/GFF.js +0 -40
- package/js/Track/Model/File/VCF.js +0 -101
- package/js/Track/Model/File/WIG.js +0 -67
- package/js/Track/Model/File.js +0 -36
- package/js/Track/Model/Gene/Ensembl.js +0 -22
- package/js/Track/Model/Sequence/Ensembl.js +0 -4
- package/js/Track/Model/Transcript/Ensembl.js +0 -67
- package/js/Track/View/Gene.js +0 -6
- package/js/Track/View/Sequence/Variation.js +0 -115
- package/js/Track/View/Transcript/Ensembl.js +0 -12
- package/js/Track/View/Transcript.js +0 -28
- package/js/Track/library/File/BAM.js +0 -30
- package/js/Track/library/File/BED.js +0 -24
- package/js/Track/library/File/BIGBED.js +0 -47
- package/js/Track/library/File/BIGWIG.js +0 -52
- package/js/Track/library/File/GFF.js +0 -9
- package/js/Track/library/File/WIG.js +0 -5
- package/js/Track/library/Gene.js +0 -37
- package/js/Track/library/Graph/Bar.js +0 -235
- package/js/Track/library/Graph/Line.js +0 -296
- package/js/Track/library/Legend.js +0 -224
- package/js/Track/library/Static.js +0 -78
- package/js/Track.js +0 -632
- package/js/genoverse.min.js +0 -2
- package/js/genoverse.min.js.map +0 -1
- package/js/lib/BWReader.js +0 -578
- package/js/lib/Base.js +0 -145
- package/js/lib/VCFReader.js +0 -286
- package/js/lib/dalliance/js/bam.js +0 -494
- package/js/lib/dalliance/js/bin.js +0 -185
- package/js/lib/dalliance/js/das.js +0 -749
- package/js/lib/dalliance/js/utils.js +0 -370
- package/js/lib/dalliance-lib.js +0 -3594
- package/js/lib/dalliance-lib.min.js +0 -68
- package/js/lib/jDataView.js +0 -2
- package/js/lib/jParser.js +0 -192
- package/js/lib/jquery-ui.js +0 -8
- package/js/lib/jquery.js +0 -2
- package/js/lib/rtree.js +0 -1
- package/js/plugins/controlPanel.js +0 -395
- package/js/plugins/fileDrop.js +0 -62
- package/js/plugins/focusRegion.js +0 -12
- package/js/plugins/resizer.js +0 -45
- package/js/plugins/trackControls.js +0 -143
- package/utils/expandedTemplate.html +0 -46
- package/utils/git-hooks/post-commit +0 -9
- package/utils/git-hooks/pre-commit +0 -7
- package/utils/git-hooks/setup +0 -6
- package/utils/makeExpanded.js +0 -19
|
@@ -1,73 +0,0 @@
|
|
|
1
|
-
Genoverse.Track.Controller.Stranded = Genoverse.Track.Controller.extend({
|
|
2
|
-
constructor: function (properties) {
|
|
3
|
-
this.base(properties);
|
|
4
|
-
|
|
5
|
-
if (typeof this._makeImage === 'function') {
|
|
6
|
-
return;
|
|
7
|
-
}
|
|
8
|
-
|
|
9
|
-
var strand = this.prop('strand');
|
|
10
|
-
var featureStrand = this.prop('featureStrand');
|
|
11
|
-
|
|
12
|
-
if (strand === -1) {
|
|
13
|
-
this._makeImage = this.track.makeReverseImage ? $.proxy(this.track.makeReverseImage, this) : this.makeImage;
|
|
14
|
-
this.makeImage = $.noop;
|
|
15
|
-
} else {
|
|
16
|
-
strand = this.prop('strand', 1);
|
|
17
|
-
|
|
18
|
-
this._makeImage = this.makeImage;
|
|
19
|
-
this.makeImage = this.makeForwardImage;
|
|
20
|
-
|
|
21
|
-
var track = this.track;
|
|
22
|
-
|
|
23
|
-
setTimeout(function () {
|
|
24
|
-
track.reverseTrack = track.browser.addTrack(track.constructor.extend({
|
|
25
|
-
id : track.id ? track.id + 'Reverse' : undefined,
|
|
26
|
-
strand : -1,
|
|
27
|
-
url : false,
|
|
28
|
-
order : typeof track.orderReverse === 'number' ? track.orderReverse : track.order,
|
|
29
|
-
forwardTrack : track
|
|
30
|
-
}));
|
|
31
|
-
|
|
32
|
-
$.each(track.controller._deferredReverseTrackImages, function (i, args) { track.controller._makeReverseTrackImage.apply(track.controller, args); });
|
|
33
|
-
delete track.controller._deferredReverseTrackImages;
|
|
34
|
-
}, 1);
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
if (!featureStrand) {
|
|
38
|
-
this.prop('featureStrand', strand);
|
|
39
|
-
}
|
|
40
|
-
},
|
|
41
|
-
|
|
42
|
-
makeForwardImage: function (params) {
|
|
43
|
-
this._makeReverseTrackImage(params, this._makeImage(params));
|
|
44
|
-
},
|
|
45
|
-
|
|
46
|
-
_makeReverseTrackImage: function (params, deferred) {
|
|
47
|
-
var reverseTrack = this.prop('reverseTrack');
|
|
48
|
-
|
|
49
|
-
if (!reverseTrack) {
|
|
50
|
-
this._deferredReverseTrackImages = (this._deferredReverseTrackImages || []).concat([[ params, deferred ]]);
|
|
51
|
-
return;
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
if (deferred && typeof deferred.done === 'function') {
|
|
55
|
-
deferred.done(function () {
|
|
56
|
-
reverseTrack.controller._makeImage(params, deferred);
|
|
57
|
-
});
|
|
58
|
-
} else {
|
|
59
|
-
reverseTrack.controller._makeImage(params, deferred);
|
|
60
|
-
}
|
|
61
|
-
},
|
|
62
|
-
|
|
63
|
-
destroy: function () {
|
|
64
|
-
if (this.removing) {
|
|
65
|
-
return;
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
this.removing = true;
|
|
69
|
-
|
|
70
|
-
this.browser.removeTrack(this.prop('forwardTrack') || this.prop('reverseTrack'));
|
|
71
|
-
this.base();
|
|
72
|
-
}
|
|
73
|
-
});
|
|
@@ -1,44 +0,0 @@
|
|
|
1
|
-
Genoverse.Track.Model.File.BAM = Genoverse.Track.Model.File.extend({
|
|
2
|
-
getData: function (chr, start, end) {
|
|
3
|
-
var model = this;
|
|
4
|
-
var deferred = $.Deferred();
|
|
5
|
-
|
|
6
|
-
if (!this.bamFile) {
|
|
7
|
-
if (this.url) {
|
|
8
|
-
this.bamFile = new dallianceLib.URLFetchable(this.url);
|
|
9
|
-
this.baiFile = new dallianceLib.URLFetchable(this.url + this.prop('indexExt'));
|
|
10
|
-
} else if (this.dataFile && this.indexFile) {
|
|
11
|
-
this.bamFile = new dallianceLib.BlobFetchable(this.dataFile);
|
|
12
|
-
this.baiFile = new dallianceLib.BlobFetchable(this.indexFile);
|
|
13
|
-
} else {
|
|
14
|
-
return deferred.rejectWith(model, [ 'BAM files must be accompanied by a .bai index file' ]);
|
|
15
|
-
}
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
dallianceLib.makeBam(this.bamFile, this.baiFile, null, function (bam, makeBamError) {
|
|
19
|
-
if (makeBamError) {
|
|
20
|
-
console.error(makeBamError); // eslint-disable-line no-console
|
|
21
|
-
} else {
|
|
22
|
-
bam.fetch(chr, start, end, function (features, fetchBamError) {
|
|
23
|
-
if (fetchBamError) {
|
|
24
|
-
console.error(fetchBamError); // eslint-disable-line no-console
|
|
25
|
-
} else {
|
|
26
|
-
model.receiveData(features, chr, start, end);
|
|
27
|
-
deferred.resolveWith(model);
|
|
28
|
-
}
|
|
29
|
-
});
|
|
30
|
-
}
|
|
31
|
-
});
|
|
32
|
-
|
|
33
|
-
return deferred;
|
|
34
|
-
},
|
|
35
|
-
|
|
36
|
-
insertFeature: function (feature) {
|
|
37
|
-
feature.id = feature.chr + ':' + feature.readName + ':' + feature.pos;
|
|
38
|
-
feature.start = feature.pos + 1;
|
|
39
|
-
feature.end = feature.start + feature.seq.length;
|
|
40
|
-
feature.sequence = feature.seq;
|
|
41
|
-
|
|
42
|
-
return this.base(feature);
|
|
43
|
-
}
|
|
44
|
-
});
|
|
@@ -1,116 +0,0 @@
|
|
|
1
|
-
Genoverse.Track.Model.File.BED = Genoverse.Track.Model.File.extend({
|
|
2
|
-
parseData: function (data, chr) {
|
|
3
|
-
var lines = typeof data === 'string' ? data.split('\n') : data;
|
|
4
|
-
var thinHeight = this.prop('thinHeight');
|
|
5
|
-
var thickHeight = this.prop('thickHeight');
|
|
6
|
-
var fields, len, feature, subfeatures, subfeature, blockSizes, blockStarts, j, thinFeature, thinFeature1, thinFeature2, thickFeature;
|
|
7
|
-
|
|
8
|
-
function filterNumber(n) {
|
|
9
|
-
return !isNaN(n);
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
for (var i = 0; i < lines.length; i++) {
|
|
13
|
-
fields = lines[i].split('\t').filter(function (f) { return f; });
|
|
14
|
-
|
|
15
|
-
if (fields.length < 3 || fields[0] === 'track' || fields[0] === 'browser') {
|
|
16
|
-
continue;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
len = fields.length;
|
|
20
|
-
|
|
21
|
-
if (fields[0] === String(chr) || fields[0].toLowerCase() === 'chr' + chr || fields[0].match('[^1-9]' + chr + '$')) {
|
|
22
|
-
feature = {
|
|
23
|
-
chr : chr,
|
|
24
|
-
start : parseInt(fields[1], 10) + 1,
|
|
25
|
-
end : parseInt(fields[2], 10),
|
|
26
|
-
name : fields[3],
|
|
27
|
-
color : '#000000',
|
|
28
|
-
originalFeature : fields
|
|
29
|
-
};
|
|
30
|
-
|
|
31
|
-
if (len > 3) { feature.score = parseFloat(fields[4], 10); }
|
|
32
|
-
if (len > 5) { feature.strand = fields[5]; }
|
|
33
|
-
|
|
34
|
-
if (len > 7) {
|
|
35
|
-
feature.thickStart = parseInt(fields[6], 10) + 1;
|
|
36
|
-
feature.thickEnd = parseInt(fields[7], 10);
|
|
37
|
-
feature.drawThick = fields[6] !== fields[7];
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
if (fields[8]) {
|
|
41
|
-
feature.color = 'rgb(' + fields[8] + ')';
|
|
42
|
-
} else {
|
|
43
|
-
feature.color = this.scoreColor(isNaN(feature.score) ? 1000 : feature.score);
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
if (len === 12) { // subfeatures present
|
|
47
|
-
feature.blockCount = parseInt(fields[9], 10);
|
|
48
|
-
|
|
49
|
-
subfeatures = [];
|
|
50
|
-
blockSizes = fields[10].split(',').filter(filterNumber);
|
|
51
|
-
blockStarts = fields[11].split(',').filter(filterNumber);
|
|
52
|
-
|
|
53
|
-
for (j = 0; j < blockSizes.length; j++) {
|
|
54
|
-
subfeature = {
|
|
55
|
-
start : feature.start + parseInt(blockStarts[j], 10),
|
|
56
|
-
height : thinHeight // if subfeature lies entirely left / right to [ thickStart, thickEnd ]
|
|
57
|
-
};
|
|
58
|
-
|
|
59
|
-
subfeature.end = subfeature.start + parseInt(blockSizes[j], 10) - 1;
|
|
60
|
-
|
|
61
|
-
if (feature.drawThick && subfeature.start <= feature.thickEnd && subfeature.end >= feature.thickStart) {
|
|
62
|
-
// some kind of an overlap for sure
|
|
63
|
-
if (subfeature.start >= feature.thickStart && subfeature.end <= feature.thickEnd) {
|
|
64
|
-
// subfeature within thickBlock, draw thick
|
|
65
|
-
subfeature.height = thickHeight;
|
|
66
|
-
subfeatures.push(subfeature);
|
|
67
|
-
} else if (subfeature.start < feature.thickStart && subfeature.end <= feature.thickEnd) {
|
|
68
|
-
// left overlap, split subfeature into 2 - thin | thick
|
|
69
|
-
thinFeature = $.extend({}, subfeature, { end: feature.thickStart });
|
|
70
|
-
thickFeature = $.extend({}, subfeature, { start: feature.thickStart, height: thickHeight });
|
|
71
|
-
|
|
72
|
-
subfeatures = subfeatures.concat([ thinFeature, thickFeature ]);
|
|
73
|
-
} else if (subfeature.start >= feature.thickStart && subfeature.end > feature.thickEnd) {
|
|
74
|
-
// right overlap, split subfeature into 2 - thick | thin
|
|
75
|
-
thinFeature = $.extend({}, subfeature, { start: feature.thickEnd });
|
|
76
|
-
thickFeature = $.extend({}, subfeature, { end: feature.thickEnd, height: thickHeight });
|
|
77
|
-
|
|
78
|
-
subfeatures = subfeatures.concat([ thickFeature, thinFeature ]);
|
|
79
|
-
} else {
|
|
80
|
-
// thickBlock lies within subfeature, split into 3 - thin | thick | thin
|
|
81
|
-
// the least possible case but lets be prepared for the outliers
|
|
82
|
-
thinFeature1 = $.extend({}, subfeature, { end: feature.thickStart });
|
|
83
|
-
thinFeature2 = $.extend({}, subfeature, { start: feature.thickEnd });
|
|
84
|
-
thickFeature = { start: feature.thickStart, end: feature.thickEnd, height: thickHeight };
|
|
85
|
-
|
|
86
|
-
subfeatures = subfeatures.concat([ thinFeature1, thickFeature, thinFeature2 ]);
|
|
87
|
-
}
|
|
88
|
-
} else {
|
|
89
|
-
// no thick block
|
|
90
|
-
subfeatures.push(subfeature);
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
if (subfeatures.length) {
|
|
95
|
-
feature.subFeatures = subfeatures;
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
this.insertFeature(feature);
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
},
|
|
103
|
-
|
|
104
|
-
// As per https://genome.ucsc.edu/FAQ/FAQformat.html#format1 specification
|
|
105
|
-
scoreColor: function (score) {
|
|
106
|
-
if (score <= 166) { return 'rgb(219,219,219)'; }
|
|
107
|
-
if (score <= 277) { return 'rgb(186,186,186)'; }
|
|
108
|
-
if (score <= 388) { return 'rgb(154,154,154)'; }
|
|
109
|
-
if (score <= 499) { return 'rgb(122,122,122)'; }
|
|
110
|
-
if (score <= 611) { return 'rgb(94,94,94)'; }
|
|
111
|
-
if (score <= 722) { return 'rgb(67,67,67)'; }
|
|
112
|
-
if (score <= 833) { return 'rgb(42,42,42)'; }
|
|
113
|
-
if (score <= 944) { return 'rgb(21,21,21)'; }
|
|
114
|
-
return '#000000';
|
|
115
|
-
}
|
|
116
|
-
});
|
|
@@ -1,40 +0,0 @@
|
|
|
1
|
-
Genoverse.Track.Model.File.GFF = Genoverse.Track.Model.File.extend({
|
|
2
|
-
parseData: function (text, chr) {
|
|
3
|
-
var lines = text.split('\n');
|
|
4
|
-
|
|
5
|
-
for (var i = 0; i < lines.length; i++) {
|
|
6
|
-
if (!lines[i].length || lines[i].indexOf('#') === 0) {
|
|
7
|
-
continue;
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
var fields = lines[i].split('\t');
|
|
11
|
-
|
|
12
|
-
if (fields.length < 5) {
|
|
13
|
-
continue;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
var seqId = fields[0].toLowerCase();
|
|
17
|
-
|
|
18
|
-
if (
|
|
19
|
-
seqId === String(chr) ||
|
|
20
|
-
seqId === 'chr' + chr ||
|
|
21
|
-
seqId.match('[^1-9]' + chr + '$') ||
|
|
22
|
-
seqId.match('^' + chr + '\\b')
|
|
23
|
-
) {
|
|
24
|
-
this.insertFeature({
|
|
25
|
-
id : fields.slice(0, 5).join('|'),
|
|
26
|
-
chr : chr,
|
|
27
|
-
start : parseInt(fields[3], 10),
|
|
28
|
-
end : parseInt(fields[4], 10),
|
|
29
|
-
source : fields[1],
|
|
30
|
-
type : fields[2],
|
|
31
|
-
score : fields[5],
|
|
32
|
-
strand : fields[6] === '-' ? -1 : 1,
|
|
33
|
-
label : fields[1] + ' ' + fields[2] + ' ' + fields[3] + '-' + fields[4]
|
|
34
|
-
});
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
Genoverse.Track.Model.File.GTF = Genoverse.Track.Model.File.GFF;
|
|
@@ -1,101 +0,0 @@
|
|
|
1
|
-
Genoverse.Track.Model.File.VCF = Genoverse.Track.Model.File.extend({
|
|
2
|
-
getData: function (chr, start, end) {
|
|
3
|
-
var deferred = $.Deferred();
|
|
4
|
-
var model = this;
|
|
5
|
-
|
|
6
|
-
if (!this.prop('gz')) {
|
|
7
|
-
return this.base.apply(this, arguments);
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
if (!this.vcfFile) {
|
|
11
|
-
if (this.url) {
|
|
12
|
-
this.vcfFile = new dallianceLib.URLFetchable(this.url);
|
|
13
|
-
this.tbiFile = new dallianceLib.URLFetchable(this.url + this.prop('indexExt'));
|
|
14
|
-
} else if (this.dataFile && this.indexFile) {
|
|
15
|
-
this.vcfFile = new dallianceLib.BlobFetchable(this.dataFile);
|
|
16
|
-
this.tbiFile = new dallianceLib.BlobFetchable(this.indexFile);
|
|
17
|
-
} else {
|
|
18
|
-
return deferred.rejectWith(model, [ 'GZipped VCF files must be accompanied by a .tbi index file' ]);
|
|
19
|
-
}
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
this.makeVCF(this.vcfFile, this.tbiFile).then(function (vcf) {
|
|
23
|
-
model.cachedVCF = vcf;
|
|
24
|
-
|
|
25
|
-
vcf.getRecords(chr, start, end, function (records) {
|
|
26
|
-
model.receiveData(records, chr, start, end);
|
|
27
|
-
deferred.resolveWith(model);
|
|
28
|
-
});
|
|
29
|
-
});
|
|
30
|
-
|
|
31
|
-
return deferred;
|
|
32
|
-
},
|
|
33
|
-
|
|
34
|
-
makeVCF: function (vcfFile, tbiFile) {
|
|
35
|
-
var deferred = $.Deferred();
|
|
36
|
-
|
|
37
|
-
if (this.cachedVCF) {
|
|
38
|
-
deferred.resolve(this.cachedVCF);
|
|
39
|
-
} else {
|
|
40
|
-
var vcf = new VCFReader(vcfFile, tbiFile);
|
|
41
|
-
|
|
42
|
-
vcf.readTabix(function (tabix) {
|
|
43
|
-
vcf.tabix = tabix;
|
|
44
|
-
deferred.resolve(vcf);
|
|
45
|
-
});
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
return deferred;
|
|
49
|
-
},
|
|
50
|
-
|
|
51
|
-
parseData: function (text, chr) {
|
|
52
|
-
var lines = text.split('\n');
|
|
53
|
-
var maxQual = this.allData ? this.prop('maxQual') || 0 : false;
|
|
54
|
-
|
|
55
|
-
for (var i = 0; i < lines.length; i++) {
|
|
56
|
-
if (!lines[i].length || lines[i].indexOf('#') === 0) {
|
|
57
|
-
continue;
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
var fields = lines[i].split('\t');
|
|
61
|
-
|
|
62
|
-
if (fields.length < 5) {
|
|
63
|
-
continue;
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
if (fields[0] === String(chr) || fields[0] === 'chr' + chr) {
|
|
67
|
-
var id = fields.slice(0, 3).join('|');
|
|
68
|
-
var start = parseInt(fields[1], 10);
|
|
69
|
-
var alleles = fields[4].split(',');
|
|
70
|
-
|
|
71
|
-
alleles.unshift(fields[3]);
|
|
72
|
-
|
|
73
|
-
for (var j = 0; j < alleles.length; j++) {
|
|
74
|
-
var end = start + alleles[j].length - 1;
|
|
75
|
-
|
|
76
|
-
this.insertFeature({
|
|
77
|
-
id : id + '|' + alleles[j],
|
|
78
|
-
sort : j,
|
|
79
|
-
chr : chr,
|
|
80
|
-
start : start,
|
|
81
|
-
end : end,
|
|
82
|
-
width : end - start,
|
|
83
|
-
allele : j === 0 ? 'REF' : 'ALT',
|
|
84
|
-
sequence : alleles[j],
|
|
85
|
-
label : alleles[j],
|
|
86
|
-
labelColor : '#FFFFFF',
|
|
87
|
-
originalFeature : fields
|
|
88
|
-
});
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
if (maxQual !== false) {
|
|
92
|
-
maxQual = Math.max(maxQual, fields[5]);
|
|
93
|
-
}
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
if (maxQual) {
|
|
98
|
-
this.prop('maxQual', maxQual);
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
});
|
|
@@ -1,67 +0,0 @@
|
|
|
1
|
-
Genoverse.Track.Model.File.WIG = Genoverse.Track.Model.Graph.Bar.extend({
|
|
2
|
-
dataType: 'text',
|
|
3
|
-
|
|
4
|
-
getData: function () {
|
|
5
|
-
if (!this.url) {
|
|
6
|
-
this.isLocal = true;
|
|
7
|
-
this.dataFile = this.track.dataFile;
|
|
8
|
-
|
|
9
|
-
return Genoverse.Track.Model.File.prototype.getData.apply(this, arguments);
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
return this.base.apply(this, arguments);
|
|
13
|
-
},
|
|
14
|
-
|
|
15
|
-
parseData: function (text, chr, s, e) {
|
|
16
|
-
var lines = text.split('\n');
|
|
17
|
-
var features = [];
|
|
18
|
-
var fields, chrom, start, step, span, line, feature, i;
|
|
19
|
-
|
|
20
|
-
while (lines.length && (line = lines.shift())) {
|
|
21
|
-
if (line.indexOf('#') !== -1 || line.indexOf('browser') !== -1 || line.indexOf('track') !== -1) {
|
|
22
|
-
continue;
|
|
23
|
-
} else {
|
|
24
|
-
break;
|
|
25
|
-
}
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
if (line) {
|
|
29
|
-
fields = line.split(/\s+/);
|
|
30
|
-
chrom = parseInt(fields[1].split('=')[1].replace('chr', ''), 10);
|
|
31
|
-
|
|
32
|
-
if (fields[0] === 'fixedStep') {
|
|
33
|
-
start = parseInt(fields[2].split('=')[1], 10);
|
|
34
|
-
step = parseInt(fields[3].split('=')[1], 10);
|
|
35
|
-
span = fields[4] ? parseInt(fields[4].split('=')[1], 10) : 1;
|
|
36
|
-
|
|
37
|
-
for (i = 0; i < lines.length; i++) {
|
|
38
|
-
features.push({
|
|
39
|
-
chr : chrom,
|
|
40
|
-
start : start,
|
|
41
|
-
end : start + span,
|
|
42
|
-
height : parseFloat(lines[i])
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
start += step;
|
|
46
|
-
}
|
|
47
|
-
} else if (fields[0] === 'variableStep') {
|
|
48
|
-
span = fields[2] ? parseInt(fields[2].split('=')[1], 10) : 1;
|
|
49
|
-
|
|
50
|
-
for (i = 0; i < lines.length; i++) {
|
|
51
|
-
fields = lines[i].split(/\s+/);
|
|
52
|
-
feature = {
|
|
53
|
-
chr : chrom,
|
|
54
|
-
start : parseInt(fields[0], 10),
|
|
55
|
-
height : parseFloat(fields[1])
|
|
56
|
-
};
|
|
57
|
-
|
|
58
|
-
feature.end = feature.start + span;
|
|
59
|
-
|
|
60
|
-
features.push(feature);
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
return this.base.call(this, features, chr, s, e);
|
|
66
|
-
}
|
|
67
|
-
});
|
package/js/Track/Model/File.js
DELETED
|
@@ -1,36 +0,0 @@
|
|
|
1
|
-
Genoverse.Track.Model.File = Genoverse.Track.Model.extend({
|
|
2
|
-
dataType: 'text',
|
|
3
|
-
|
|
4
|
-
init: function () {
|
|
5
|
-
if (this.isLocal) {
|
|
6
|
-
this.url = false;
|
|
7
|
-
}
|
|
8
|
-
|
|
9
|
-
if (!(this.largeFile || this.indexFile)) {
|
|
10
|
-
this.allData = true;
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
this.base.apply(this, arguments);
|
|
14
|
-
},
|
|
15
|
-
|
|
16
|
-
getData: function (chr) {
|
|
17
|
-
var model = this;
|
|
18
|
-
|
|
19
|
-
if (this.isLocal && this.dataFile) {
|
|
20
|
-
var reader = new FileReader();
|
|
21
|
-
var deferred = $.Deferred();
|
|
22
|
-
|
|
23
|
-
reader.onload = function (e) {
|
|
24
|
-
deferred.done(function () {
|
|
25
|
-
this.receiveData(e.target.result, chr, 1, this.browser.getChromosomeSize(chr));
|
|
26
|
-
}).resolveWith(model);
|
|
27
|
-
};
|
|
28
|
-
|
|
29
|
-
reader.readAsText(this.dataFile);
|
|
30
|
-
|
|
31
|
-
return deferred;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
return this.base.apply(this, arguments);
|
|
35
|
-
}
|
|
36
|
-
});
|
|
@@ -1,22 +0,0 @@
|
|
|
1
|
-
// Ensembl REST API Gene model
|
|
2
|
-
Genoverse.Track.Model.Gene.Ensembl = Genoverse.Track.Model.Gene.extend({
|
|
3
|
-
url : '//rest.ensembl.org/overlap/region/human/__CHR__:__START__-__END__?feature=gene;content-type=application/json',
|
|
4
|
-
dataRequestLimit : 5000000, // As per e! REST API restrictions
|
|
5
|
-
|
|
6
|
-
// The url above responds in json format, data is an array
|
|
7
|
-
// We assume that parents always preceed children in data array, gene -> transcript -> exon
|
|
8
|
-
// See rest.ensembl.org/documentation/info/feature_region for more details
|
|
9
|
-
parseData: function (data, chr) {
|
|
10
|
-
for (var i = 0; i < data.length; i++) {
|
|
11
|
-
var feature = data[i];
|
|
12
|
-
|
|
13
|
-
if (feature.feature_type === 'gene' && !this.featuresById[feature.id]) {
|
|
14
|
-
feature.chr = feature.chr || chr;
|
|
15
|
-
feature.label = parseInt(feature.strand, 10) === 1 ? (feature.external_name || feature.id) + ' >' : '< ' + (feature.external_name || feature.id);
|
|
16
|
-
feature.transcripts = [];
|
|
17
|
-
|
|
18
|
-
this.insertFeature(feature);
|
|
19
|
-
}
|
|
20
|
-
}
|
|
21
|
-
}
|
|
22
|
-
});
|
|
@@ -1,67 +0,0 @@
|
|
|
1
|
-
// Ensembl REST API Transcript model
|
|
2
|
-
Genoverse.Track.Model.Transcript.Ensembl = Genoverse.Track.Model.Transcript.extend({
|
|
3
|
-
url : '//rest.ensembl.org/overlap/region/human/__CHR__:__START__-__END__?feature=transcript;feature=exon;feature=cds;content-type=application/json',
|
|
4
|
-
dataRequestLimit : 5000000, // As per e! REST API restrictions
|
|
5
|
-
|
|
6
|
-
setDefaults: function () {
|
|
7
|
-
this.geneIds = {};
|
|
8
|
-
this.seenGenes = 0;
|
|
9
|
-
|
|
10
|
-
this.base.apply(this, arguments);
|
|
11
|
-
},
|
|
12
|
-
|
|
13
|
-
// The url above responds in json format, data is an array
|
|
14
|
-
// See rest.ensembl.org/documentation/info/overlap_region for more details
|
|
15
|
-
parseData: function (data, chr) {
|
|
16
|
-
var model = this;
|
|
17
|
-
var featuresById = this.featuresById;
|
|
18
|
-
var ids = [];
|
|
19
|
-
|
|
20
|
-
data.filter(function (d) { return d.feature_type === 'transcript'; }).forEach(function (feature, i) {
|
|
21
|
-
if (!featuresById[feature.id]) {
|
|
22
|
-
model.geneIds[feature.Parent] = model.geneIds[feature.Parent] || ++model.seenGenes;
|
|
23
|
-
|
|
24
|
-
feature.chr = feature.chr || chr;
|
|
25
|
-
feature.label = parseInt(feature.strand, 10) === 1 ? (feature.external_name || feature.id) + ' >' : '< ' + (feature.external_name || feature.id);
|
|
26
|
-
feature.sort = (model.geneIds[feature.Parent] * 1e10) + (feature.logic_name.indexOf('ensembl_havana') === 0 ? 0 : 2e9) + (feature.biotype === 'protein_coding' ? 0 : 1e9) + feature.start + i;
|
|
27
|
-
feature.cdsStart = Infinity;
|
|
28
|
-
feature.cdsEnd = -Infinity;
|
|
29
|
-
feature.exons = {};
|
|
30
|
-
feature.subFeatures = [];
|
|
31
|
-
|
|
32
|
-
model.insertFeature(feature);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
ids.push(feature.id);
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
data.filter(function (d) { return d.feature_type === 'cds' && featuresById[d.Parent]; }).forEach(function (cds) {
|
|
39
|
-
featuresById[cds.Parent].cdsStart = Math.min(featuresById[cds.Parent].cdsStart, cds.start);
|
|
40
|
-
featuresById[cds.Parent].cdsEnd = Math.max(featuresById[cds.Parent].cdsEnd, cds.end);
|
|
41
|
-
});
|
|
42
|
-
|
|
43
|
-
data.filter(function (d) { return d.feature_type === 'exon' && featuresById[d.Parent] && !featuresById[d.Parent].exons[d.id]; }).forEach(function (exon) {
|
|
44
|
-
if (exon.end < featuresById[exon.Parent].cdsStart || exon.start > featuresById[exon.Parent].cdsEnd) {
|
|
45
|
-
featuresById[exon.Parent].subFeatures.push($.extend({ utr: true }, exon));
|
|
46
|
-
} else {
|
|
47
|
-
if (exon.start < featuresById[exon.Parent].cdsStart) {
|
|
48
|
-
featuresById[exon.Parent].subFeatures.push($.extend({ utr: true }, exon, { end: featuresById[exon.Parent].cdsStart }));
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
featuresById[exon.Parent].subFeatures.push($.extend({}, exon, {
|
|
52
|
-
start : Math.max(exon.start, featuresById[exon.Parent].cdsStart),
|
|
53
|
-
end : Math.min(exon.end, featuresById[exon.Parent].cdsEnd),
|
|
54
|
-
strand : featuresById[exon.Parent].strand
|
|
55
|
-
}));
|
|
56
|
-
|
|
57
|
-
if (exon.end > featuresById[exon.Parent].cdsEnd) {
|
|
58
|
-
featuresById[exon.Parent].subFeatures.push($.extend({ utr: true }, exon, { start: featuresById[exon.Parent].cdsEnd }));
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
ids.forEach(function (id) {
|
|
64
|
-
featuresById[id].subFeatures.sort(function (a, b) { return a.start - b.start; });
|
|
65
|
-
});
|
|
66
|
-
}
|
|
67
|
-
});
|