Skip to content

Commit cb15846

Browse files
Merge pull request #125 from sulcgroup/plugin
addig plugin functionality with a demo in examples
2 parents 5cb044a + 28a7656 commit cb15846

12 files changed

+212
-58
lines changed

dist/UI/UI.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -994,7 +994,7 @@ class fluxGraph {
994994
// this.chart.toBase64Image();
995995
//}
996996
},
997-
responsiveAnimationDuration: 0,
997+
responsiveAnimationDuration: 0, // animation duration after a resize
998998
responsive: true,
999999
title: {
10001000
display: true,

dist/api/plugin_api.js

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
// let's define a storage structure for the plugin API
2+
// it will sit in local storage and load the code for the individual plugins named as key
3+
// and the code as value packed away in a JSON object hidden at the key "plugins" in local storage
4+
// this is the object that will be stored in local storage
5+
// it will contain the code for the individual plugins
6+
// the key will be the name of the plugin, the value will be the code
7+
let plugins = {};
8+
// this function will load the plugins from local storage
9+
// it will be called at the beginning of the program
10+
function loadPlugins() {
11+
let pluginsString = localStorage.getItem("plugins");
12+
if (pluginsString) {
13+
plugins = JSON.parse(pluginsString);
14+
}
15+
// if there are no plugins in local storage, we will create an empty object
16+
else {
17+
plugins = {}; //"test": plugin_code};
18+
}
19+
//iterate over all the plugins and load them
20+
for (let plugin in plugins) {
21+
eval(plugins[plugin]);
22+
}
23+
}
24+
// this function will save the plugins to local storage (it will be called any time we add a new plugin)
25+
function savePlugins() {
26+
localStorage.setItem("plugins", JSON.stringify(plugins));
27+
}
28+
function addPlugin(name, plugin_code) {
29+
plugins[name] = plugin_code;
30+
eval(plugin_code);
31+
savePlugins();
32+
}
33+
loadPlugins();

dist/file_handling/order_parameter_selector.js

+5-5
Original file line numberDiff line numberDiff line change
@@ -58,12 +58,12 @@ let loadHyperSelector = () => {
5858
// mode:"dataset", // it is possible to remove the line view from the plot
5959
// intersect:true // on hover, but i find it to distracting
6060
},
61-
responsiveAnimationDuration: 0,
61+
responsiveAnimationDuration: 0, // animation duration after a resize
6262
scales: {
6363
xAxes: [{ display: true, scaleLabel: { display: true, labelString: 'Time' }, gridLines: { drawOnChartArea: false } }],
6464
yAxes: [{ display: true, gridLines: { drawOnChartArea: false } }],
6565
},
66-
spanGaps: true,
66+
spanGaps: true, // handle null in the chart data
6767
annotation: {
6868
events: ["click"],
6969
annotations: [
@@ -143,7 +143,7 @@ let handleParameterDrop = (files) => {
143143
else {
144144
console.log("adding new axis");
145145
myChart.options.scales.yAxes.push({
146-
type: 'linear',
146+
type: 'linear', // only linear but allow scale type registration. This allows extensions to exist solely for log scale for instance
147147
display: true,
148148
position: 'left',
149149
id: `y-axis-id${axis_counter}`,
@@ -159,7 +159,7 @@ let handleParameterDrop = (files) => {
159159
labels: labels,
160160
datasets: [
161161
{
162-
label: parameter_name,
162+
label: parameter_name, //files[0].name.split(".")[0],
163163
data: data,
164164
fill: false,
165165
borderColor: chartColorMap.get(),
@@ -170,7 +170,7 @@ let handleParameterDrop = (files) => {
170170
}
171171
else {
172172
myChart.data.datasets.push({
173-
label: parameter_name,
173+
label: parameter_name, //files[i].name.split(".")[0],
174174
data: data,
175175
fill: false,
176176
borderColor: chartColorMap.get(),

dist/file_handling/output_file.js

+7-7
Original file line numberDiff line numberDiff line change
@@ -128,9 +128,9 @@ function getNewIds(useNew = false) {
128128
});
129129
// these subtypes are not implemented in the CG-oxDNA model, just used for a 'relative' subtype that oxDNA will take as input
130130
let gsSubtypes = {
131-
subtypelist: [],
132-
masses: [],
133-
radii: [],
131+
subtypelist: [], // per particle subtype assignment
132+
masses: [], //
133+
radii: [], //
134134
subtype: -1
135135
};
136136
// Next, generic sphere objects
@@ -477,9 +477,9 @@ function makeUNFOutput(name) {
477477
let naStrandsSchema = {
478478
"id": strand.id,
479479
"name": strand.label,
480-
"isScaffold": strand.getLength() > 1000 ? true : false,
481-
"naType": strand.end5.isDNA() ? "DNA" : "RNA",
482-
"color": strand.end5.color ? '#'.concat(strand.end5.color.getHexString()) : '',
480+
"isScaffold": strand.getLength() > 1000 ? true : false, //entirely arbitrary, but generally right.
481+
"naType": strand.end5.isDNA() ? "DNA" : "RNA", // Nucleotide type is actually pretty poorly defined in oxView, so this is the best I can do
482+
"color": strand.end5.color ? '#'.concat(strand.end5.color.getHexString()) : '', // OxView defines colors on a per-nucleotide level while UNF defines it at the strand level.
483483
"fivePrimeId": strand.end5.id,
484484
"threePrimeId": strand.end3.id,
485485
"pdbFileId": 0,
@@ -504,7 +504,7 @@ function makeUNFOutput(name) {
504504
let aaChainSchema = {
505505
"id": strand.id,
506506
"chainName": strand.label,
507-
"color": strand.end5.color ? '#'.concat(strand.end5.color.getHexString()) : '',
507+
"color": strand.end5.color ? '#'.concat(strand.end5.color.getHexString()) : '', // OxView defines colors on a per-nucleotide level while UNF defines it at the strand level.
508508
"pdbFileId": 0,
509509
"nTerm": strand.end5.id,
510510
"cTerm": strand.end3.id,

dist/file_handling/pdb_worker.js

+22-22
Original file line numberDiff line numberDiff line change
@@ -20,48 +20,48 @@ this.onmessage = function (e) {
2020
postMessage(ret, undefined);
2121
};
2222
var backboneColors = [
23-
new THREE.Color(0xfdd291),
24-
new THREE.Color(0xffb322),
25-
new THREE.Color(0x437092),
23+
new THREE.Color(0xfdd291), //light yellow
24+
new THREE.Color(0xffb322), //goldenrod
25+
new THREE.Color(0x437092), //dark blue
2626
new THREE.Color(0x6ea4cc), //light blue
2727
];
2828
var nucleosideColors = [
29-
new THREE.Color(0x4747B8),
30-
new THREE.Color(0xFFFF33),
29+
new THREE.Color(0x4747B8), //A or K; Royal Blue
30+
new THREE.Color(0xFFFF33), //G or C; Medium Yellow
3131
//C or A
32-
new THREE.Color(0x8CFF8C),
32+
new THREE.Color(0x8CFF8C), //Medium green
3333
//T/U or T
34-
new THREE.Color(0xFF3333),
34+
new THREE.Color(0xFF3333), //Red
3535
//E
36-
new THREE.Color(0x660000),
36+
new THREE.Color(0x660000), //Dark Brown
3737
//S
38-
new THREE.Color(0xFF7042),
38+
new THREE.Color(0xFF7042), //Medium Orange
3939
//D
40-
new THREE.Color(0xA00042),
40+
new THREE.Color(0xA00042), //Dark Rose
4141
//N
42-
new THREE.Color(0xFF7C70),
42+
new THREE.Color(0xFF7C70), //Light Salmon
4343
//Q
44-
new THREE.Color(0xFF4C4C),
44+
new THREE.Color(0xFF4C4C), //Dark Salmon
4545
//H
46-
new THREE.Color(0x7070FF),
46+
new THREE.Color(0x7070FF), //Medium Blue
4747
//G
48-
new THREE.Color(0xEBEBEB),
48+
new THREE.Color(0xEBEBEB), // light GREY
4949
//P
50-
new THREE.Color(0x525252),
50+
new THREE.Color(0x525252), //Dark Grey
5151
//R
52-
new THREE.Color(0x00007C),
52+
new THREE.Color(0x00007C), //Dark Blue
5353
//V
54-
new THREE.Color(0x5E005E),
54+
new THREE.Color(0x5E005E), //Dark Purple
5555
//I
56-
new THREE.Color(0x004C00),
56+
new THREE.Color(0x004C00), //Dark Green
5757
//L
58-
new THREE.Color(0x455E45),
58+
new THREE.Color(0x455E45), //Olive Green
5959
//M
60-
new THREE.Color(0xB8A042),
60+
new THREE.Color(0xB8A042), //Light Brown
6161
//F
62-
new THREE.Color(0x534C42),
62+
new THREE.Color(0x534C42), //Olive Grey
6363
//Y
64-
new THREE.Color(0x8C704C),
64+
new THREE.Color(0x8C704C), //Medium Brown
6565
//W
6666
new THREE.Color(0x4F4600), //Olive Brown
6767
];

dist/main.js

+2
Original file line numberDiff line numberDiff line change
@@ -122,6 +122,7 @@ var lut, devs;
122122
const editHistory = new EditHistory(); // Track do/undo
123123
var tmpSystems = []; // Track memory for newly created systems
124124
var topologyEdited = false; // to keep track of if the topology was edited at any point.
125+
125126
function resetScene(resetCamera = true) {
126127
elements.clear();
127128
while (systems.length > 0) {
@@ -177,6 +178,7 @@ function resetScene(resetCamera = true) {
177178
if (resetCamera) {
178179
controls.reset();
179180
}
181+
180182
render();
181183
}
182184
///////////////////////////////////////////////////////////////////////////////////////////////////

dist/scene/mesh_setup.js

+22-22
Original file line numberDiff line numberDiff line change
@@ -38,49 +38,49 @@ function switchMaterial(material) {
3838
}
3939
// Default colors for the backbones
4040
var backboneColors = [
41-
new THREE.Color(0xfdd291),
42-
new THREE.Color(0xffb322),
43-
new THREE.Color(0x437092),
41+
new THREE.Color(0xfdd291), //light yellow
42+
new THREE.Color(0xffb322), //goldenrod
43+
new THREE.Color(0x437092), //dark blue
4444
new THREE.Color(0x6ea4cc), //light blue
4545
];
4646
// define nucleoside colors
4747
var nucleosideColors = [
48-
new THREE.Color(0x4747B8),
49-
new THREE.Color(0xFFFF33),
48+
new THREE.Color(0x4747B8), //A or K; Royal Blue
49+
new THREE.Color(0xFFFF33), //G or C; Medium Yellow
5050
//C or A
51-
new THREE.Color(0x8CFF8C),
51+
new THREE.Color(0x8CFF8C), //Medium green
5252
//T/U or T
53-
new THREE.Color(0xFF3333),
53+
new THREE.Color(0xFF3333), //Red
5454
//E
55-
new THREE.Color(0x660000),
55+
new THREE.Color(0x660000), //Dark Brown
5656
//S
57-
new THREE.Color(0xFF7042),
57+
new THREE.Color(0xFF7042), //Medium Orange
5858
//D
59-
new THREE.Color(0xA00042),
59+
new THREE.Color(0xA00042), //Dark Rose
6060
//N
61-
new THREE.Color(0xFF7C70),
61+
new THREE.Color(0xFF7C70), //Light Salmon
6262
//Q
63-
new THREE.Color(0xFF4C4C),
63+
new THREE.Color(0xFF4C4C), //Dark Salmon
6464
//H
65-
new THREE.Color(0x7070FF),
65+
new THREE.Color(0x7070FF), //Medium Blue
6666
//G
67-
new THREE.Color(0xEBEBEB),
67+
new THREE.Color(0xEBEBEB), // light GREY
6868
//P
69-
new THREE.Color(0x525252),
69+
new THREE.Color(0x525252), //Dark Grey
7070
//R
71-
new THREE.Color(0x00007C),
71+
new THREE.Color(0x00007C), //Dark Blue
7272
//V
73-
new THREE.Color(0x5E005E),
73+
new THREE.Color(0x5E005E), //Dark Purple
7474
//I
75-
new THREE.Color(0x004C00),
75+
new THREE.Color(0x004C00), //Dark Green
7676
//L
77-
new THREE.Color(0x455E45),
77+
new THREE.Color(0x455E45), //Olive Green
7878
//M
79-
new THREE.Color(0xB8A042),
79+
new THREE.Color(0xB8A042), //Light Brown
8080
//F
81-
new THREE.Color(0x534C42),
81+
new THREE.Color(0x534C42), //Olive Grey
8282
//Y
83-
new THREE.Color(0x8C704C),
83+
new THREE.Color(0x8C704C), //Medium Brown
8484
//W
8585
new THREE.Color(0x4F4600), //Olive Brown
8686
];
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
//This is a demo plugin that highlights overstretched positions in red.
2+
3+
let plugin_name = "HOS";
4+
5+
// The actual plugin code has to be a string, so we will define it here
6+
let plugin_code = `
7+
//let's write a test plugin, adding a button to the UI
8+
// this function will add a button to the UI
9+
function HSO_Plugin() {
10+
let button = document.createElement("button");
11+
// we will set the inner text of the button
12+
button.innerText = "Highlight Overstretched DNA";
13+
// we will add an event listener to the button
14+
button.addEventListener("click", () => {
15+
// we go over all the elements, collecting the strands of the ones that are overstretched
16+
// you can change the threshold to a different value but say anything which is longer than .6 is overstretched
17+
const delta = 0.6;
18+
19+
// find everything that is more than 5 ox units away from its n3
20+
let p3_extended = new Set();
21+
elements.forEach((b,id) => {
22+
if(b.n3){
23+
let p1 = b.getPos();
24+
let p2 = b.n3.getPos();
25+
let dist = p1.distanceTo(p2);
26+
if(dist > delta){
27+
p3_extended.add(b);
28+
p3_extended.add(b.n3);
29+
}
30+
}});
31+
if(p3_extended.size == 0){
32+
notify("No overstretched DNA found");
33+
return;
34+
}
35+
colorElements(new THREE.Color(1,0,0),[... p3_extended])
36+
});
37+
// we will add the button to the UI
38+
document.querySelector("#plugin-pane").appendChild(button);
39+
}
40+
HSO_Plugin();
41+
`;
42+
43+
// register the plugin with view
44+
addPlugin(plugin_name, plugin_code);

index.html

+30
Original file line numberDiff line numberDiff line change
@@ -382,6 +382,8 @@ <h2>TacoxDNA importer</h2>
382382
<li ondblclick="view.sectionClicked()"><a href="#section-camera">Camera</a></li>
383383
<li ondblclick="view.sectionClicked()"><a href="#section-dynamics">Dynamics</a></li>
384384
<li ondblclick="view.sectionClicked()"><a href="#section-protein">Protein</a></li>
385+
<li ondblclick="view.sectionClicked()"><a href="#section-plugins">Plugins</a></li>
386+
385387
</ul>
386388

387389
<div class="content-holder">
@@ -1178,6 +1180,32 @@ <h2>TacoxDNA importer</h2>
11781180
</button>
11791181
</div>
11801182
</div>
1183+
1184+
1185+
<!-- Plugin pane-->
1186+
<div class="section" id="section-plugins">
1187+
<div class=" group" style="height: 100px">
1188+
<div class="row flex-align-center" >
1189+
<div class="row">
1190+
<div class="group">
1191+
<div >
1192+
<button class="ribbon-button" onclick="localStorage.setItem('plugins', {});"
1193+
title="Clean all plugins from memory.">
1194+
<!-- <span class="mif-2x">VR</span> -->
1195+
<img src="img/ico/delete.svg" height="18px">
1196+
<span class="caption">Clean Plugins</span>
1197+
</button>
1198+
1199+
</div>
1200+
<span class="title">Plugin control</span>
1201+
</div>
1202+
1203+
<div class="group" id="plugin-pane"></div>
1204+
</div>
1205+
</div>
1206+
</div>
1207+
</div>
1208+
11811209
</div>
11821210
</nav>
11831211

@@ -1343,6 +1371,8 @@ <h2>TacoxDNA importer</h2>
13431371
<script src="./dist/file_handling/plateDB.js"></script>
13441372
<script src="./dist/editing/distance.js"></script>
13451373
<script src="./dist/editing/selections.js"></script>
1374+
<script src="./dist/api/plugin_api.js"></script>
1375+
13461376
<script>
13471377
//https://stackoverflow.com/questions/326069/how-to-identify-if-a-webpage-is-being-loaded-inside-an-iframe-or-directly-into-t
13481378
function inIframe () {

0 commit comments

Comments
 (0)