diff --git a/.github/workflows/tests.yaml b/.github/workflows/tests.yaml index 122fc11..b45f102 100644 --- a/.github/workflows/tests.yaml +++ b/.github/workflows/tests.yaml @@ -15,20 +15,20 @@ jobs: strategy: matrix: os: ['ubuntu-latest', 'macos-latest', 'windows-latest'] - python-version: [3.7, 3.8, 3.9, '3.10'] + python-version: [3.8, 3.9, '3.10', '3.11'] exclude: - - os: windows-latest - python-version: 3.8 - os: windows-latest python-version: 3.9 - os: windows-latest python-version: '3.10' - - os: macos-latest - python-version: 3.8 + - os: windows-latest + python-version: '3.11' - os: macos-latest python-version: 3.9 - os: macos-latest python-version: '3.10' + - os: macos-latest + python-version: '3.11' env: CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }} diff --git a/DEVELOPER_GUIDE.md b/DEVELOPER_GUIDE.md index c4104d6..2981169 100644 --- a/DEVELOPER_GUIDE.md +++ b/DEVELOPER_GUIDE.md @@ -18,7 +18,7 @@ Below we describe how to install and use this project for development. To install for development you will need to create and activate a virtual environment ```bash -conda create --name panel-chemistry python=3.9 nodejs +conda create --name panel-chemistry python=3.10 nodejs conda activate panel-chemistry ``` diff --git a/examples/reference/JSMEEditor.ipynb b/examples/reference/JSMEEditor.ipynb index 37e1691..c02b6a5 100644 --- a/examples/reference/JSMEEditor.ipynb +++ b/examples/reference/JSMEEditor.ipynb @@ -161,7 +161,7 @@ ], "metadata": { "kernelspec": { - "display_name": "Python 3", + "display_name": "Python 3 (ipykernel)", "language": "python", "name": "python3" }, @@ -175,7 +175,7 @@ "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", - "version": "3.8.4" + "version": "3.11.6" } }, "nbformat": 4, diff --git a/examples/reference/NGLViewer.ipynb b/examples/reference/NGLViewer.ipynb index a376afb..bde4304 100644 --- a/examples/reference/NGLViewer.ipynb +++ b/examples/reference/NGLViewer.ipynb @@ -83,14 +83,7 @@ "cell_type": "code", "execution_count": null, "id": "6e35c77b", - "metadata": { - "jupyter": { - "outputs_hidden": false - }, - "pycharm": { - "name": "#%%\n" - } - }, + "metadata": {}, "outputs": [], "source": [ "file_input = pn.widgets.FileInput(accept=','.join('.' + s for s in EXTENSIONS[1:]))\n", @@ -183,7 +176,7 @@ ], "metadata": { "kernelspec": { - "display_name": "Python 3", + "display_name": "Python 3 (ipykernel)", "language": "python", "name": "python3" }, @@ -197,7 +190,7 @@ "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", - "version": "3.8.4" + "version": "3.11.6" } }, "nbformat": 4, diff --git a/examples/reference/PDBe_MolStar.ipynb b/examples/reference/PDBe_MolStar.ipynb index 34fefc7..cae9558 100644 --- a/examples/reference/PDBe_MolStar.ipynb +++ b/examples/reference/PDBe_MolStar.ipynb @@ -26,424 +26,9 @@ }, { "cell_type": "code", - "execution_count": 3, + "execution_count": null, "metadata": {}, - "outputs": [ - { - "data": { - "application/javascript": [ - "\n", - "(function(root) {\n", - " function now() {\n", - " return new Date();\n", - " }\n", - "\n", - " var force = true;\n", - "\n", - " if (typeof root._bokeh_onload_callbacks === \"undefined\" || force === true) {\n", - " root._bokeh_onload_callbacks = [];\n", - " root._bokeh_is_loading = undefined;\n", - " }\n", - "\n", - " if (typeof (root._bokeh_timeout) === \"undefined\" || force === true) {\n", - " root._bokeh_timeout = Date.now() + 5000;\n", - " root._bokeh_failed_load = false;\n", - " }\n", - "\n", - " function run_callbacks() {\n", - " try {\n", - " root._bokeh_onload_callbacks.forEach(function(callback) {\n", - " if (callback != null)\n", - " callback();\n", - " });\n", - " } finally {\n", - " delete root._bokeh_onload_callbacks\n", - " }\n", - " console.debug(\"Bokeh: all callbacks have finished\");\n", - " }\n", - "\n", - " function load_libs(css_urls, js_urls, js_modules, callback) {\n", - " if (css_urls == null) css_urls = [];\n", - " if (js_urls == null) js_urls = [];\n", - " if (js_modules == null) js_modules = [];\n", - "\n", - " root._bokeh_onload_callbacks.push(callback);\n", - " if (root._bokeh_is_loading > 0) {\n", - " console.debug(\"Bokeh: BokehJS is being loaded, scheduling callback at\", now());\n", - " return null;\n", - " }\n", - " if (js_urls.length === 0 && js_modules.length === 0) {\n", - " run_callbacks();\n", - " return null;\n", - " }\n", - " console.debug(\"Bokeh: BokehJS not loaded, scheduling load and callback at\", now());\n", - "\n", - " function on_load() {\n", - " root._bokeh_is_loading--;\n", - " if (root._bokeh_is_loading === 0) {\n", - " console.debug(\"Bokeh: all BokehJS libraries/stylesheets loaded\");\n", - " run_callbacks()\n", - " }\n", - " }\n", - "\n", - " function on_error() {\n", - " console.error(\"failed to load \" + url);\n", - " }\n", - "\n", - " for (var i = 0; i < css_urls.length; i++) {\n", - " var url = css_urls[i];\n", - " const element = document.createElement(\"link\");\n", - " element.onload = on_load;\n", - " element.onerror = on_error;\n", - " element.rel = \"stylesheet\";\n", - " element.type = \"text/css\";\n", - " element.href = url;\n", - " console.debug(\"Bokeh: injecting link tag for BokehJS stylesheet: \", url);\n", - " document.body.appendChild(element);\n", - " }\n", - "\n", - " var skip = [];\n", - " if (window.requirejs) {\n", - " window.requirejs.config({'packages': {}, 'paths': {}, 'shim': {}});\n", - " \n", - " root._bokeh_is_loading = css_urls.length + 0;\n", - " } else {\n", - " root._bokeh_is_loading = css_urls.length + js_urls.length + js_modules.length;\n", - " }\n", - " for (var i = 0; i < js_urls.length; i++) {\n", - " var url = js_urls[i];\n", - " if (skip.indexOf(url) >= 0) {\n", - "\tif (!window.requirejs) {\n", - "\t on_load();\n", - "\t}\n", - "\tcontinue;\n", - " }\n", - " var element = document.createElement('script');\n", - " element.onload = on_load;\n", - " element.onerror = on_error;\n", - " element.async = false;\n", - " element.src = url;\n", - " console.debug(\"Bokeh: injecting script tag for BokehJS library: \", url);\n", - " document.head.appendChild(element);\n", - " }\n", - " for (var i = 0; i < js_modules.length; i++) {\n", - " var url = js_modules[i];\n", - " if (skip.indexOf(url) >= 0) {\n", - "\tif (!window.requirejs) {\n", - "\t on_load();\n", - "\t}\n", - "\tcontinue;\n", - " }\n", - " var element = document.createElement('script');\n", - " element.onload = on_load;\n", - " element.onerror = on_error;\n", - " element.async = false;\n", - " element.src = url;\n", - " element.type = \"module\";\n", - " console.debug(\"Bokeh: injecting script tag for BokehJS library: \", url);\n", - " document.head.appendChild(element);\n", - " }\n", - " if (!js_urls.length && !js_modules.length) {\n", - " on_load()\n", - " }\n", - " };\n", - "\n", - " function inject_raw_css(css) {\n", - " const element = document.createElement(\"style\");\n", - " element.appendChild(document.createTextNode(css));\n", - " document.body.appendChild(element);\n", - " }\n", - "\n", - " var js_urls = [\"https://cdn.bokeh.org/bokeh/release/bokeh-2.4.2.min.js\", \"https://cdn.bokeh.org/bokeh/release/bokeh-gl-2.4.2.min.js\", \"https://cdn.bokeh.org/bokeh/release/bokeh-widgets-2.4.2.min.js\", \"https://cdn.bokeh.org/bokeh/release/bokeh-tables-2.4.2.min.js\", \"https://cdn.bokeh.org/bokeh/release/bokeh-mathjax-2.4.2.min.js\", \"https://www.ebi.ac.uk/pdbe/pdb-component-library/js/pdbe-molstar-plugin-1.2.1.js\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/panel.min.js\"];\n", - " var js_modules = [];\n", - " var css_urls = [\"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/alerts.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/card.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/dataframe.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/json.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/loading.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/markdown.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/widgets.css\"];\n", - " var inline_js = [\n", - " function(Bokeh) {\n", - " inject_raw_css(\"\\n .bk.pn-loading.arcs:before {\\n background-image: url(\\\"\\\");\\n max-height: 400px;\\n }\\n \");\n", - " },\n", - " function(Bokeh) {\n", - " Bokeh.set_log_level(\"info\");\n", - " },\n", - " function(Bokeh) {} // ensure no trailing comma for IE\n", - " ];\n", - "\n", - " function run_inline_js() {\n", - " if ((root.Bokeh !== undefined) || (force === true)) {\n", - " for (var i = 0; i < inline_js.length; i++) {\n", - " inline_js[i].call(root, root.Bokeh);\n", - " }} else if (Date.now() < root._bokeh_timeout) {\n", - " setTimeout(run_inline_js, 100);\n", - " } else if (!root._bokeh_failed_load) {\n", - " console.log(\"Bokeh: BokehJS failed to load within specified timeout.\");\n", - " root._bokeh_failed_load = true;\n", - " }\n", - " }\n", - "\n", - " if (root._bokeh_is_loading === 0) {\n", - " console.debug(\"Bokeh: BokehJS loaded, going straight to plotting\");\n", - " run_inline_js();\n", - " } else {\n", - " load_libs(css_urls, js_urls, js_modules, function() {\n", - " console.debug(\"Bokeh: BokehJS plotting callback run at\", now());\n", - " run_inline_js();\n", - " });\n", - " }\n", - "}(window));" - ], - "application/vnd.holoviews_load.v0+json": "\n(function(root) {\n function now() {\n return new Date();\n }\n\n var force = true;\n\n if (typeof root._bokeh_onload_callbacks === \"undefined\" || force === true) {\n root._bokeh_onload_callbacks = [];\n root._bokeh_is_loading = undefined;\n }\n\n if (typeof (root._bokeh_timeout) === \"undefined\" || force === true) {\n root._bokeh_timeout = Date.now() + 5000;\n root._bokeh_failed_load = false;\n }\n\n function run_callbacks() {\n try {\n root._bokeh_onload_callbacks.forEach(function(callback) {\n if (callback != null)\n callback();\n });\n } finally {\n delete root._bokeh_onload_callbacks\n }\n console.debug(\"Bokeh: all callbacks have finished\");\n }\n\n function load_libs(css_urls, js_urls, js_modules, callback) {\n if (css_urls == null) css_urls = [];\n if (js_urls == null) js_urls = [];\n if (js_modules == null) js_modules = [];\n\n root._bokeh_onload_callbacks.push(callback);\n if (root._bokeh_is_loading > 0) {\n console.debug(\"Bokeh: BokehJS is being loaded, scheduling callback at\", now());\n return null;\n }\n if (js_urls.length === 0 && js_modules.length === 0) {\n run_callbacks();\n return null;\n }\n console.debug(\"Bokeh: BokehJS not loaded, scheduling load and callback at\", now());\n\n function on_load() {\n root._bokeh_is_loading--;\n if (root._bokeh_is_loading === 0) {\n console.debug(\"Bokeh: all BokehJS libraries/stylesheets loaded\");\n run_callbacks()\n }\n }\n\n function on_error() {\n console.error(\"failed to load \" + url);\n }\n\n for (var i = 0; i < css_urls.length; i++) {\n var url = css_urls[i];\n const element = document.createElement(\"link\");\n element.onload = on_load;\n element.onerror = on_error;\n element.rel = \"stylesheet\";\n element.type = \"text/css\";\n element.href = url;\n console.debug(\"Bokeh: injecting link tag for BokehJS stylesheet: \", url);\n document.body.appendChild(element);\n }\n\n var skip = [];\n if (window.requirejs) {\n window.requirejs.config({'packages': {}, 'paths': {}, 'shim': {}});\n \n root._bokeh_is_loading = css_urls.length + 0;\n } else {\n root._bokeh_is_loading = css_urls.length + js_urls.length + js_modules.length;\n }\n for (var i = 0; i < js_urls.length; i++) {\n var url = js_urls[i];\n if (skip.indexOf(url) >= 0) {\n\tif (!window.requirejs) {\n\t on_load();\n\t}\n\tcontinue;\n }\n var element = document.createElement('script');\n element.onload = on_load;\n element.onerror = on_error;\n element.async = false;\n element.src = url;\n console.debug(\"Bokeh: injecting script tag for BokehJS library: \", url);\n document.head.appendChild(element);\n }\n for (var i = 0; i < js_modules.length; i++) {\n var url = js_modules[i];\n if (skip.indexOf(url) >= 0) {\n\tif (!window.requirejs) {\n\t on_load();\n\t}\n\tcontinue;\n }\n var element = document.createElement('script');\n element.onload = on_load;\n element.onerror = on_error;\n element.async = false;\n element.src = url;\n element.type = \"module\";\n console.debug(\"Bokeh: injecting script tag for BokehJS library: \", url);\n document.head.appendChild(element);\n }\n if (!js_urls.length && !js_modules.length) {\n on_load()\n }\n };\n\n function inject_raw_css(css) {\n const element = document.createElement(\"style\");\n element.appendChild(document.createTextNode(css));\n document.body.appendChild(element);\n }\n\n var js_urls = [\"https://cdn.bokeh.org/bokeh/release/bokeh-2.4.2.min.js\", \"https://cdn.bokeh.org/bokeh/release/bokeh-gl-2.4.2.min.js\", \"https://cdn.bokeh.org/bokeh/release/bokeh-widgets-2.4.2.min.js\", \"https://cdn.bokeh.org/bokeh/release/bokeh-tables-2.4.2.min.js\", \"https://cdn.bokeh.org/bokeh/release/bokeh-mathjax-2.4.2.min.js\", \"https://www.ebi.ac.uk/pdbe/pdb-component-library/js/pdbe-molstar-plugin-1.2.1.js\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/panel.min.js\"];\n var js_modules = [];\n var css_urls = [\"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/alerts.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/card.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/dataframe.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/json.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/loading.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/markdown.css\", \"https://unpkg.com/@holoviz/panel@0.12.6/dist/css/widgets.css\"];\n var inline_js = [\n function(Bokeh) {\n inject_raw_css(\"\\n .bk.pn-loading.arcs:before {\\n background-image: url(\\\"\\\");\\n max-height: 400px;\\n }\\n \");\n },\n function(Bokeh) {\n Bokeh.set_log_level(\"info\");\n },\n function(Bokeh) {} // ensure no trailing comma for IE\n ];\n\n function run_inline_js() {\n if ((root.Bokeh !== undefined) || (force === true)) {\n for (var i = 0; i < inline_js.length; i++) {\n inline_js[i].call(root, root.Bokeh);\n }} else if (Date.now() < root._bokeh_timeout) {\n setTimeout(run_inline_js, 100);\n } else if (!root._bokeh_failed_load) {\n console.log(\"Bokeh: BokehJS failed to load within specified timeout.\");\n root._bokeh_failed_load = true;\n }\n }\n\n if (root._bokeh_is_loading === 0) {\n console.debug(\"Bokeh: BokehJS loaded, going straight to plotting\");\n run_inline_js();\n } else {\n load_libs(css_urls, js_urls, js_modules, function() {\n console.debug(\"Bokeh: BokehJS plotting callback run at\", now());\n run_inline_js();\n });\n }\n}(window));" - }, - "metadata": {}, - "output_type": "display_data" - }, - { - "data": { - "application/javascript": [ - "\n", - "if ((window.PyViz === undefined) || (window.PyViz instanceof HTMLElement)) {\n", - " window.PyViz = {comms: {}, comm_status:{}, kernels:{}, receivers: {}, plot_index: []}\n", - "}\n", - "\n", - "\n", - " function JupyterCommManager() {\n", - " }\n", - "\n", - " JupyterCommManager.prototype.register_target = function(plot_id, comm_id, msg_handler) {\n", - " if (window.comm_manager || ((window.Jupyter !== undefined) && (Jupyter.notebook.kernel != null))) {\n", - " var comm_manager = window.comm_manager || Jupyter.notebook.kernel.comm_manager;\n", - " comm_manager.register_target(comm_id, function(comm) {\n", - " comm.on_msg(msg_handler);\n", - " });\n", - " } else if ((plot_id in window.PyViz.kernels) && (window.PyViz.kernels[plot_id])) {\n", - " window.PyViz.kernels[plot_id].registerCommTarget(comm_id, function(comm) {\n", - " comm.onMsg = msg_handler;\n", - " });\n", - " } else if (typeof google != 'undefined' && google.colab.kernel != null) {\n", - " google.colab.kernel.comms.registerTarget(comm_id, (comm) => {\n", - " var messages = comm.messages[Symbol.asyncIterator]();\n", - " function processIteratorResult(result) {\n", - " var message = result.value;\n", - " console.log(message)\n", - " var content = {data: message.data, comm_id};\n", - " var buffers = []\n", - " for (var buffer of message.buffers || []) {\n", - " buffers.push(new DataView(buffer))\n", - " }\n", - " var metadata = message.metadata || {};\n", - " var msg = {content, buffers, metadata}\n", - " msg_handler(msg);\n", - " return messages.next().then(processIteratorResult);\n", - " }\n", - " return messages.next().then(processIteratorResult);\n", - " })\n", - " }\n", - " }\n", - "\n", - " JupyterCommManager.prototype.get_client_comm = function(plot_id, comm_id, msg_handler) {\n", - " if (comm_id in window.PyViz.comms) {\n", - " return window.PyViz.comms[comm_id];\n", - " } else if (window.comm_manager || ((window.Jupyter !== undefined) && (Jupyter.notebook.kernel != null))) {\n", - " var comm_manager = window.comm_manager || Jupyter.notebook.kernel.comm_manager;\n", - " var comm = comm_manager.new_comm(comm_id, {}, {}, {}, comm_id);\n", - " if (msg_handler) {\n", - " comm.on_msg(msg_handler);\n", - " }\n", - " } else if ((plot_id in window.PyViz.kernels) && (window.PyViz.kernels[plot_id])) {\n", - " var comm = window.PyViz.kernels[plot_id].connectToComm(comm_id);\n", - " comm.open();\n", - " if (msg_handler) {\n", - " comm.onMsg = msg_handler;\n", - " }\n", - " } else if (typeof google != 'undefined' && google.colab.kernel != null) {\n", - " var comm_promise = google.colab.kernel.comms.open(comm_id)\n", - " comm_promise.then((comm) => {\n", - " window.PyViz.comms[comm_id] = comm;\n", - " if (msg_handler) {\n", - " var messages = comm.messages[Symbol.asyncIterator]();\n", - " function processIteratorResult(result) {\n", - " var message = result.value;\n", - " var content = {data: message.data};\n", - " var metadata = message.metadata || {comm_id};\n", - " var msg = {content, metadata}\n", - " msg_handler(msg);\n", - " return messages.next().then(processIteratorResult);\n", - " }\n", - " return messages.next().then(processIteratorResult);\n", - " }\n", - " }) \n", - " var sendClosure = (data, metadata, buffers, disposeOnDone) => {\n", - " return comm_promise.then((comm) => {\n", - " comm.send(data, metadata, buffers, disposeOnDone);\n", - " });\n", - " };\n", - " var comm = {\n", - " send: sendClosure\n", - " };\n", - " }\n", - " window.PyViz.comms[comm_id] = comm;\n", - " return comm;\n", - " }\n", - " window.PyViz.comm_manager = new JupyterCommManager();\n", - " \n", - "\n", - "\n", - "var JS_MIME_TYPE = 'application/javascript';\n", - "var HTML_MIME_TYPE = 'text/html';\n", - "var EXEC_MIME_TYPE = 'application/vnd.holoviews_exec.v0+json';\n", - "var CLASS_NAME = 'output';\n", - "\n", - "/**\n", - " * Render data to the DOM node\n", - " */\n", - "function render(props, node) {\n", - " var div = document.createElement(\"div\");\n", - " var script = document.createElement(\"script\");\n", - " node.appendChild(div);\n", - " node.appendChild(script);\n", - "}\n", - "\n", - "/**\n", - " * Handle when a new output is added\n", - " */\n", - "function handle_add_output(event, handle) {\n", - " var output_area = handle.output_area;\n", - " var output = handle.output;\n", - " if ((output.data == undefined) || (!output.data.hasOwnProperty(EXEC_MIME_TYPE))) {\n", - " return\n", - " }\n", - " var id = output.metadata[EXEC_MIME_TYPE][\"id\"];\n", - " var toinsert = output_area.element.find(\".\" + CLASS_NAME.split(' ')[0]);\n", - " if (id !== undefined) {\n", - " var nchildren = toinsert.length;\n", - " var html_node = toinsert[nchildren-1].children[0];\n", - " html_node.innerHTML = output.data[HTML_MIME_TYPE];\n", - " var scripts = [];\n", - " var nodelist = html_node.querySelectorAll(\"script\");\n", - " for (var i in nodelist) {\n", - " if (nodelist.hasOwnProperty(i)) {\n", - " scripts.push(nodelist[i])\n", - " }\n", - " }\n", - "\n", - " scripts.forEach( function (oldScript) {\n", - " var newScript = document.createElement(\"script\");\n", - " var attrs = [];\n", - " var nodemap = oldScript.attributes;\n", - " for (var j in nodemap) {\n", - " if (nodemap.hasOwnProperty(j)) {\n", - " attrs.push(nodemap[j])\n", - " }\n", - " }\n", - " attrs.forEach(function(attr) { newScript.setAttribute(attr.name, attr.value) });\n", - " newScript.appendChild(document.createTextNode(oldScript.innerHTML));\n", - " oldScript.parentNode.replaceChild(newScript, oldScript);\n", - " });\n", - " if (JS_MIME_TYPE in output.data) {\n", - " toinsert[nchildren-1].children[1].textContent = output.data[JS_MIME_TYPE];\n", - " }\n", - " output_area._hv_plot_id = id;\n", - " if ((window.Bokeh !== undefined) && (id in Bokeh.index)) {\n", - " window.PyViz.plot_index[id] = Bokeh.index[id];\n", - " } else {\n", - " window.PyViz.plot_index[id] = null;\n", - " }\n", - " } else if (output.metadata[EXEC_MIME_TYPE][\"server_id\"] !== undefined) {\n", - " var bk_div = document.createElement(\"div\");\n", - " bk_div.innerHTML = output.data[HTML_MIME_TYPE];\n", - " var script_attrs = bk_div.children[0].attributes;\n", - " for (var i = 0; i < script_attrs.length; i++) {\n", - " toinsert[toinsert.length - 1].childNodes[1].setAttribute(script_attrs[i].name, script_attrs[i].value);\n", - " }\n", - " // store reference to server id on output_area\n", - " output_area._bokeh_server_id = output.metadata[EXEC_MIME_TYPE][\"server_id\"];\n", - " }\n", - "}\n", - "\n", - "/**\n", - " * Handle when an output is cleared or removed\n", - " */\n", - "function handle_clear_output(event, handle) {\n", - " var id = handle.cell.output_area._hv_plot_id;\n", - " var server_id = handle.cell.output_area._bokeh_server_id;\n", - " if (((id === undefined) || !(id in PyViz.plot_index)) && (server_id !== undefined)) { return; }\n", - " var comm = window.PyViz.comm_manager.get_client_comm(\"hv-extension-comm\", \"hv-extension-comm\", function () {});\n", - " if (server_id !== null) {\n", - " comm.send({event_type: 'server_delete', 'id': server_id});\n", - " return;\n", - " } else if (comm !== null) {\n", - " comm.send({event_type: 'delete', 'id': id});\n", - " }\n", - " delete PyViz.plot_index[id];\n", - " if ((window.Bokeh !== undefined) & (id in window.Bokeh.index)) {\n", - " var doc = window.Bokeh.index[id].model.document\n", - " doc.clear();\n", - " const i = window.Bokeh.documents.indexOf(doc);\n", - " if (i > -1) {\n", - " window.Bokeh.documents.splice(i, 1);\n", - " }\n", - " }\n", - "}\n", - "\n", - "/**\n", - " * Handle kernel restart event\n", - " */\n", - "function handle_kernel_cleanup(event, handle) {\n", - " delete PyViz.comms[\"hv-extension-comm\"];\n", - " window.PyViz.plot_index = {}\n", - "}\n", - "\n", - "/**\n", - " * Handle update_display_data messages\n", - " */\n", - "function handle_update_output(event, handle) {\n", - " handle_clear_output(event, {cell: {output_area: handle.output_area}})\n", - " handle_add_output(event, handle)\n", - "}\n", - "\n", - "function register_renderer(events, OutputArea) {\n", - " function append_mime(data, metadata, element) {\n", - " // create a DOM node to render to\n", - " var toinsert = this.create_output_subarea(\n", - " metadata,\n", - " CLASS_NAME,\n", - " EXEC_MIME_TYPE\n", - " );\n", - " this.keyboard_manager.register_events(toinsert);\n", - " // Render to node\n", - " var props = {data: data, metadata: metadata[EXEC_MIME_TYPE]};\n", - " render(props, toinsert[0]);\n", - " element.append(toinsert);\n", - " return toinsert\n", - " }\n", - "\n", - " events.on('output_added.OutputArea', handle_add_output);\n", - " events.on('output_updated.OutputArea', handle_update_output);\n", - " events.on('clear_output.CodeCell', handle_clear_output);\n", - " events.on('delete.Cell', handle_clear_output);\n", - " events.on('kernel_ready.Kernel', handle_kernel_cleanup);\n", - "\n", - " OutputArea.prototype.register_mime_type(EXEC_MIME_TYPE, append_mime, {\n", - " safe: true,\n", - " index: 0\n", - " });\n", - "}\n", - "\n", - "if (window.Jupyter !== undefined) {\n", - " try {\n", - " var events = require('base/js/events');\n", - " var OutputArea = require('notebook/js/outputarea').OutputArea;\n", - " if (OutputArea.prototype.mime_types().indexOf(EXEC_MIME_TYPE) == -1) {\n", - " register_renderer(events, OutputArea);\n", - " }\n", - " } catch(err) {\n", - " }\n", - "}\n" - ], - "application/vnd.holoviews_load.v0+json": "\nif ((window.PyViz === undefined) || (window.PyViz instanceof HTMLElement)) {\n window.PyViz = {comms: {}, comm_status:{}, kernels:{}, receivers: {}, plot_index: []}\n}\n\n\n function JupyterCommManager() {\n }\n\n JupyterCommManager.prototype.register_target = function(plot_id, comm_id, msg_handler) {\n if (window.comm_manager || ((window.Jupyter !== undefined) && (Jupyter.notebook.kernel != null))) {\n var comm_manager = window.comm_manager || Jupyter.notebook.kernel.comm_manager;\n comm_manager.register_target(comm_id, function(comm) {\n comm.on_msg(msg_handler);\n });\n } else if ((plot_id in window.PyViz.kernels) && (window.PyViz.kernels[plot_id])) {\n window.PyViz.kernels[plot_id].registerCommTarget(comm_id, function(comm) {\n comm.onMsg = msg_handler;\n });\n } else if (typeof google != 'undefined' && google.colab.kernel != null) {\n google.colab.kernel.comms.registerTarget(comm_id, (comm) => {\n var messages = comm.messages[Symbol.asyncIterator]();\n function processIteratorResult(result) {\n var message = result.value;\n console.log(message)\n var content = {data: message.data, comm_id};\n var buffers = []\n for (var buffer of message.buffers || []) {\n buffers.push(new DataView(buffer))\n }\n var metadata = message.metadata || {};\n var msg = {content, buffers, metadata}\n msg_handler(msg);\n return messages.next().then(processIteratorResult);\n }\n return messages.next().then(processIteratorResult);\n })\n }\n }\n\n JupyterCommManager.prototype.get_client_comm = function(plot_id, comm_id, msg_handler) {\n if (comm_id in window.PyViz.comms) {\n return window.PyViz.comms[comm_id];\n } else if (window.comm_manager || ((window.Jupyter !== undefined) && (Jupyter.notebook.kernel != null))) {\n var comm_manager = window.comm_manager || Jupyter.notebook.kernel.comm_manager;\n var comm = comm_manager.new_comm(comm_id, {}, {}, {}, comm_id);\n if (msg_handler) {\n comm.on_msg(msg_handler);\n }\n } else if ((plot_id in window.PyViz.kernels) && (window.PyViz.kernels[plot_id])) {\n var comm = window.PyViz.kernels[plot_id].connectToComm(comm_id);\n comm.open();\n if (msg_handler) {\n comm.onMsg = msg_handler;\n }\n } else if (typeof google != 'undefined' && google.colab.kernel != null) {\n var comm_promise = google.colab.kernel.comms.open(comm_id)\n comm_promise.then((comm) => {\n window.PyViz.comms[comm_id] = comm;\n if (msg_handler) {\n var messages = comm.messages[Symbol.asyncIterator]();\n function processIteratorResult(result) {\n var message = result.value;\n var content = {data: message.data};\n var metadata = message.metadata || {comm_id};\n var msg = {content, metadata}\n msg_handler(msg);\n return messages.next().then(processIteratorResult);\n }\n return messages.next().then(processIteratorResult);\n }\n }) \n var sendClosure = (data, metadata, buffers, disposeOnDone) => {\n return comm_promise.then((comm) => {\n comm.send(data, metadata, buffers, disposeOnDone);\n });\n };\n var comm = {\n send: sendClosure\n };\n }\n window.PyViz.comms[comm_id] = comm;\n return comm;\n }\n window.PyViz.comm_manager = new JupyterCommManager();\n \n\n\nvar JS_MIME_TYPE = 'application/javascript';\nvar HTML_MIME_TYPE = 'text/html';\nvar EXEC_MIME_TYPE = 'application/vnd.holoviews_exec.v0+json';\nvar CLASS_NAME = 'output';\n\n/**\n * Render data to the DOM node\n */\nfunction render(props, node) {\n var div = document.createElement(\"div\");\n var script = document.createElement(\"script\");\n node.appendChild(div);\n node.appendChild(script);\n}\n\n/**\n * Handle when a new output is added\n */\nfunction handle_add_output(event, handle) {\n var output_area = handle.output_area;\n var output = handle.output;\n if ((output.data == undefined) || (!output.data.hasOwnProperty(EXEC_MIME_TYPE))) {\n return\n }\n var id = output.metadata[EXEC_MIME_TYPE][\"id\"];\n var toinsert = output_area.element.find(\".\" + CLASS_NAME.split(' ')[0]);\n if (id !== undefined) {\n var nchildren = toinsert.length;\n var html_node = toinsert[nchildren-1].children[0];\n html_node.innerHTML = output.data[HTML_MIME_TYPE];\n var scripts = [];\n var nodelist = html_node.querySelectorAll(\"script\");\n for (var i in nodelist) {\n if (nodelist.hasOwnProperty(i)) {\n scripts.push(nodelist[i])\n }\n }\n\n scripts.forEach( function (oldScript) {\n var newScript = document.createElement(\"script\");\n var attrs = [];\n var nodemap = oldScript.attributes;\n for (var j in nodemap) {\n if (nodemap.hasOwnProperty(j)) {\n attrs.push(nodemap[j])\n }\n }\n attrs.forEach(function(attr) { newScript.setAttribute(attr.name, attr.value) });\n newScript.appendChild(document.createTextNode(oldScript.innerHTML));\n oldScript.parentNode.replaceChild(newScript, oldScript);\n });\n if (JS_MIME_TYPE in output.data) {\n toinsert[nchildren-1].children[1].textContent = output.data[JS_MIME_TYPE];\n }\n output_area._hv_plot_id = id;\n if ((window.Bokeh !== undefined) && (id in Bokeh.index)) {\n window.PyViz.plot_index[id] = Bokeh.index[id];\n } else {\n window.PyViz.plot_index[id] = null;\n }\n } else if (output.metadata[EXEC_MIME_TYPE][\"server_id\"] !== undefined) {\n var bk_div = document.createElement(\"div\");\n bk_div.innerHTML = output.data[HTML_MIME_TYPE];\n var script_attrs = bk_div.children[0].attributes;\n for (var i = 0; i < script_attrs.length; i++) {\n toinsert[toinsert.length - 1].childNodes[1].setAttribute(script_attrs[i].name, script_attrs[i].value);\n }\n // store reference to server id on output_area\n output_area._bokeh_server_id = output.metadata[EXEC_MIME_TYPE][\"server_id\"];\n }\n}\n\n/**\n * Handle when an output is cleared or removed\n */\nfunction handle_clear_output(event, handle) {\n var id = handle.cell.output_area._hv_plot_id;\n var server_id = handle.cell.output_area._bokeh_server_id;\n if (((id === undefined) || !(id in PyViz.plot_index)) && (server_id !== undefined)) { return; }\n var comm = window.PyViz.comm_manager.get_client_comm(\"hv-extension-comm\", \"hv-extension-comm\", function () {});\n if (server_id !== null) {\n comm.send({event_type: 'server_delete', 'id': server_id});\n return;\n } else if (comm !== null) {\n comm.send({event_type: 'delete', 'id': id});\n }\n delete PyViz.plot_index[id];\n if ((window.Bokeh !== undefined) & (id in window.Bokeh.index)) {\n var doc = window.Bokeh.index[id].model.document\n doc.clear();\n const i = window.Bokeh.documents.indexOf(doc);\n if (i > -1) {\n window.Bokeh.documents.splice(i, 1);\n }\n }\n}\n\n/**\n * Handle kernel restart event\n */\nfunction handle_kernel_cleanup(event, handle) {\n delete PyViz.comms[\"hv-extension-comm\"];\n window.PyViz.plot_index = {}\n}\n\n/**\n * Handle update_display_data messages\n */\nfunction handle_update_output(event, handle) {\n handle_clear_output(event, {cell: {output_area: handle.output_area}})\n handle_add_output(event, handle)\n}\n\nfunction register_renderer(events, OutputArea) {\n function append_mime(data, metadata, element) {\n // create a DOM node to render to\n var toinsert = this.create_output_subarea(\n metadata,\n CLASS_NAME,\n EXEC_MIME_TYPE\n );\n this.keyboard_manager.register_events(toinsert);\n // Render to node\n var props = {data: data, metadata: metadata[EXEC_MIME_TYPE]};\n render(props, toinsert[0]);\n element.append(toinsert);\n return toinsert\n }\n\n events.on('output_added.OutputArea', handle_add_output);\n events.on('output_updated.OutputArea', handle_update_output);\n events.on('clear_output.CodeCell', handle_clear_output);\n events.on('delete.Cell', handle_clear_output);\n events.on('kernel_ready.Kernel', handle_kernel_cleanup);\n\n OutputArea.prototype.register_mime_type(EXEC_MIME_TYPE, append_mime, {\n safe: true,\n index: 0\n });\n}\n\nif (window.Jupyter !== undefined) {\n try {\n var events = require('base/js/events');\n var OutputArea = require('notebook/js/outputarea').OutputArea;\n if (OutputArea.prototype.mime_types().indexOf(EXEC_MIME_TYPE) == -1) {\n register_renderer(events, OutputArea);\n }\n } catch(err) {\n }\n}\n" - }, - "metadata": {}, - "output_type": "display_data" - } - ], + "outputs": [], "source": [ "from panel_chemistry.pane import PDBeMolStar\n", "import param\n", @@ -462,64 +47,9 @@ }, { "cell_type": "code", - "execution_count": 5, + "execution_count": null, "metadata": {}, - "outputs": [ - { - "data": {}, - "metadata": {}, - "output_type": "display_data" - }, - { - "data": { - "application/vnd.holoviews_exec.v0+json": "", - "text/html": [ - "
\n", - "\n", - "\n", - "\n", - "\n", - "\n", - "
\n", - "
\n", - "" - ], - "text/plain": [ - "PDBeMolStar(height=500, molecule_id='1qyn', sizing_mode='stretch_width')" - ] - }, - "execution_count": 5, - "metadata": { - "application/vnd.holoviews_exec.v0+json": { - "id": "1007" - } - }, - "output_type": "execute_result" - } - ], + "outputs": [], "source": [ "PDBeMolStar(\n", " height=500,\n", @@ -536,64 +66,9 @@ }, { "cell_type": "code", - "execution_count": 6, + "execution_count": null, "metadata": {}, - "outputs": [ - { - "data": {}, - "metadata": {}, - "output_type": "display_data" - }, - { - "data": { - "application/vnd.holoviews_exec.v0+json": "", - "text/html": [ - "
\n", - "\n", - "\n", - "\n", - "\n", - "\n", - "
\n", - "
\n", - "" - ], - "text/plain": [ - "PDBeMolStar(custom_data={'url': 'http://localhost:...}, height=500, name='Local File', sizing_mode='stretch_width')" - ] - }, - "execution_count": 6, - "metadata": { - "application/vnd.holoviews_exec.v0+json": { - "id": "1010" - } - }, - "output_type": "execute_result" - } - ], + "outputs": [], "source": [ "PDBeMolStar(\n", " name='Local File',\n", @@ -894,8 +369,7 @@ "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", - "pygments_lexer": "ipython3", - "version": "3.9.7" + "pygments_lexer": "ipython3" } }, "nbformat": 4, diff --git a/examples/reference/Py3DMol.ipynb b/examples/reference/Py3DMol.ipynb index ccde4af..6f946eb 100644 --- a/examples/reference/Py3DMol.ipynb +++ b/examples/reference/Py3DMol.ipynb @@ -307,8 +307,7 @@ "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", - "pygments_lexer": "ipython3", - "version": "3.8.4" + "pygments_lexer": "ipython3" } }, "nbformat": 4, diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..160b51d --- /dev/null +++ b/package-lock.json @@ -0,0 +1,10 @@ +{ + "name": "panel-chemistry", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "panel-chemistry" + } + } +} diff --git a/pyproject.toml b/pyproject.toml index b295611..85792ae 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -34,7 +34,7 @@ classifiers = [ "Topic :: Software Development :: Libraries", ] dependencies = [ - "panel", + "panel>=1.0", ] dynamic = ["version"] @@ -65,7 +65,7 @@ profile = "black" src_paths = ["src", "tests"] [tool.pylint.main] -py-version=3.9 +py-version=3.10 output-format = "colorized" max-attributes=12 max-args=10 @@ -75,7 +75,7 @@ max-module-lines = 1000 max-line-length=100 [tool.mypy] -python_version = "3.9" +python_version = "3.10" namespace_packages = true explicit_package_bases = true mypy_path = "src" @@ -84,11 +84,13 @@ exclude = [] [[tool.mypy.overrides]] module = [ "bokeh.*", + "comm.*", "holoviews.*", "hvplot.*", "param", - "pyviz_comms", "py3Dmol", + "pydantic.*", + "pyviz_comms", ] ignore_missing_imports = true diff --git a/script.py b/script.py new file mode 100644 index 0000000..f95a393 --- /dev/null +++ b/script.py @@ -0,0 +1,49 @@ +import panel as pn +from panel_chemistry.pane import NGLViewer # panel_chemistry needs to be imported before you run pn.extension() +from panel_chemistry.pane.ngl_viewer import EXTENSIONS + +pn.extension("ngl_viewer", sizing_mode="stretch_width") + +viewer = NGLViewer(object="1CRN", background_color="#F7F7F7", styles={"border": "1px solid lightgray"}, min_height=700, sizing_mode="stretch_both") + +settings = pn.Param( + viewer, + parameters=["object","extension","representation","color_scheme","custom_color_scheme","effect",], + name="⚙️ Settings" +) + +file_input = pn.widgets.FileInput(accept=','.join('.' + s for s in EXTENSIONS[1:])) + +def filename_callback(target, event): + target.extension = event.new.split('.')[1] + +def value_callback(target, event): + target.object = event.new.decode('utf-8') + +file_input.link(viewer, callbacks={'value': value_callback, 'filename': filename_callback}) + +header = pn.widgets.StaticText(value='{0}'.format("💾 File Input")) +file_column = pn.layout.Column(header, file_input) + +layout = pn.Param( + viewer, + parameters=["sizing_mode", "width", "height", "background_color"], + name="📐 Layout" +) + +pn.Row( + viewer, + pn.WidgetBox(settings, layout, width=300, sizing_mode="fixed",), +) + + +accent="#D2386C" + +pn.template.FastListTemplate( + site="Panel Chemistry", site_url="./", + title="NGLViewer", + sidebar=[file_column, settings, layout], + main=[viewer], + accent_base_color=accent, header_background=accent, +).servable(); + diff --git a/src/panel_chemistry/__init__.py b/src/panel_chemistry/__init__.py index 5710cf0..b0828c8 100644 --- a/src/panel_chemistry/__init__.py +++ b/src/panel_chemistry/__init__.py @@ -2,4 +2,4 @@ develop high-quality [Panel](https://awesome-panel.org) **data apps** within the domain of **chemistry**. """ -VERSION = "0.2.2" +VERSION = "0.3.0" diff --git a/src/panel_chemistry/bokeh_extensions/jsme_editor.py b/src/panel_chemistry/bokeh_extensions/jsme_editor.py index 15f87a4..7cb2384 100644 --- a/src/panel_chemistry/bokeh_extensions/jsme_editor.py +++ b/src/panel_chemistry/bokeh_extensions/jsme_editor.py @@ -1,27 +1,35 @@ """A Bokeh model for the JSMEEditor""" from bokeh.core.properties import List, String -from bokeh.models import HTMLBox from panel import extension +from panel.models.layout import HTMLBox # pylint: disable=protected-access extension._imports["jsme"] = "panel_chemistry.bokeh_extensions.jsme_editor" # pylint: enable=protected-access -class JSMEEditor(HTMLBox): +class JSMEEditor(HTMLBox): # pylint: disable=too-few-public-methods,too-many-ancestors """JSMEEditor Bokeh Model""" - __javascript__ = ["https://unpkg.com/jsme-editor@0.8.2/jsme.nocache.js"] + __javascript__ = [ + "https://cdn.jsdelivr.net/npm/jsme-editor@2023.7.31/jsme.nocache.js", + ] - value = String() - format = String() - subscriptions = List(String()) - options = List(String()) + __css__ = [ + "https://cdn.jsdelivr.net/npm/jsme-editor@2023.7.31/gwt/chrome/mosaic.css", + "https://cdn.jsdelivr.net/npm/jsme-editor@2023.7.31/jsa.css", + "https://cdn.jsdelivr.net/npm/jsme-editor@2023.7.31/gwt/chrome/chrome.css", + ] - jme = String() - smiles = String() - mol = String() - mol3000 = String() - sdf = String() + value = String(default="") + format = String(default="smiles") + subscriptions = List(String, default=[]) # type: ignore + options = List(String, default=[]) # type: ignore - guicolor = String() + jme = String(default="") + smiles = String(default="") + mol = String(default="") + mol3000 = String(default="") + sdf = String(default="") + + guicolor = String(default="#c0c0c0") diff --git a/src/panel_chemistry/bokeh_extensions/jsme_editor.ts b/src/panel_chemistry/bokeh_extensions/jsme_editor.ts index 6f048c7..64783b1 100644 --- a/src/panel_chemistry/bokeh_extensions/jsme_editor.ts +++ b/src/panel_chemistry/bokeh_extensions/jsme_editor.ts @@ -1,17 +1,10 @@ // See https://docs.bokeh.org/en/latest/docs/reference/models/layouts.html -import { HTMLBox, HTMLBoxView } from "@bokehjs/models/layouts/html_box" +import { uuidv4, HTMLBox, HTMLBoxView } from "./layout" // See https://docs.bokeh.org/en/latest/docs/reference/core/properties.html import * as p from "@bokehjs/core/properties" import {div} from "@bokehjs/core/dom" -function uuidv4() { - return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) { - var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8); - return v.toString(16); - }); - } - const notSubscribed = "Not Subscribed" function readSDFValue(jsmeElement: any) { @@ -24,12 +17,9 @@ function readSDFValue(jsmeElement: any) { } function setModelValue(model: JSMEEditor, jsmeElement: any){ - console.log("setValue - start", model.value) var value = model.value if (model.format==="smiles"){ - console.log("getting smiles") value = jsmeElement.smiles() - console.log("got smiles") } else if (model.format==="mol"){ value = jsmeElement.molFile(false) } else if (model.format==="mol3000"){ @@ -40,17 +30,13 @@ function setModelValue(model: JSMEEditor, jsmeElement: any){ value = jsmeElement.jmeFile() } if (model.value!==value && value!==null){ - console.log("setting value", value) model.value = value } - console.log("setValue - end", model.value) } function setModelValues(model: JSMEEditor, jsmeElement: any){ - console.log("setValues - start") setModelValue(model, jsmeElement) - setOtherModelValues(model, jsmeElement) - console.log("setValues - end") + setOtherModelValues(model, jsmeElement) } function resetOtherModelValues(model: JSMEEditor, jsmeElement: any){ @@ -72,13 +58,11 @@ function cleanValue(value: any){ } function setOtherModelValues(model: JSMEEditor, jsmeElement: any){ - console.log("setOtherValues - start") if (model.subscriptions.includes("jme")){model.jme = cleanValue(jsmeElement.jmeFile())} if (model.subscriptions.includes("smiles")){model.smiles = cleanValue(jsmeElement.smiles())} if (model.subscriptions.includes("mol")){model.mol = cleanValue(jsmeElement.molFile(false))} if (model.subscriptions.includes("mol3000")){model.mol3000 = cleanValue(jsmeElement.molFile(true))} if (model.subscriptions.includes("sdf")){model.sdf = cleanValue(readSDFValue(jsmeElement))} - console.log("setOtherValues - end") } // The view of the Bokeh extension/ HTML element @@ -89,12 +73,20 @@ export class JSMEEditorView extends HTMLBoxView { JSME = (window as any).JSApplet.JSME valueFunc: any valueChanging: boolean = true + container: HTMLDivElement + _intialized: boolean = false + + initialize(): void { + super.initialize() + this.container = div({ + style: {display: "contents"} + }) + } connect_signals(): void { super.connect_signals() this.connect(this.model.properties.value.change, () => { - console.log("value change", this.model.value) if (!this.valueChanging){ if (this.model.value===""){ this.jsmeElement.reset() @@ -104,29 +96,40 @@ export class JSMEEditorView extends HTMLBoxView { } }) this.connect(this.model.properties.format.change, () => { - console.log("format change", this.model.format) setModelValue(this.model, this.jsmeElement); }) this.connect(this.model.properties.subscriptions.change, () => { - console.log("subscriptions change", this.model.subscriptions) resetOtherModelValues(this.model, this.jsmeElement); }) this.connect(this.model.properties.options.change, () => { - console.log("options change", this.model.options) this.setJSMEOptions() }) this.connect(this.model.properties.guicolor.change, () => { - console.log("options change", this.model.options) this.setGUIColor() }) } render(): void { - console.log("render - start") super.render() const id = "jsme-" + uuidv4() - const container = div({class: "jsme-editor", id: id}); - this.el.appendChild(container) + const el = div({ + class: "jsme-editor", + id: id, + style: {width: "100%", height: "100%"} + }) + this.container.appendChild(el) + this._intialized = false + } + + createJSMEElement() { + if (this._intialized) + return + + const id = this.container.children[0].id + + // Need to add it to document body for JSME to be able to find the id + document.body.appendChild(this.container) + this.jsmeElement = new this.JSME(id, this.getHeight(), this.getWidth(), { "options": this.model.options.join(","), "guicolor": this.model.guicolor @@ -136,25 +139,26 @@ export class JSMEEditorView extends HTMLBoxView { setModelValues(this.model, this.jsmeElement) const this_ = this; - function showEvent(event: any){ - console.log("event", event) + function showEvent(_: any){ this_.valueChanging = true setModelValues(this_.model, this_.jsmeElement) this_.valueChanging = false } this.jsmeElement.setAfterStructureModifiedCallback(showEvent); - console.log("render - end") + // Remove from document body and add to shadow DOM + document.body.removeChild(this.container) + this.shadow_el.appendChild(this.container) + + this._intialized = true } setGUIColor(){ - console.log("setGUIColor", this.model.guicolor) this.jsmeElement.setUserInterfaceBackgroundColor(this.model.guicolor) } setJSMEOptions(){ const options = this.model.options.join(",") - console.log("setJSMEOptions", options) this.jsmeElement.options(options) } @@ -185,6 +189,7 @@ export class JSMEEditorView extends HTMLBoxView { after_layout(): void{ super.after_layout() + this.createJSMEElement() this.resizeJSMEElement() } } @@ -213,6 +218,7 @@ export interface JSMEEditor extends JSMEEditor.Attrs { } // The Bokeh .ts model corresponding to the Bokeh .py model export class JSMEEditor extends HTMLBox { properties: JSMEEditor.Props + __view_type__: JSMEEditorView constructor(attrs?: Partial) { super(attrs) @@ -220,7 +226,7 @@ export class JSMEEditor extends HTMLBox { static __module__ = "panel_chemistry.bokeh_extensions.jsme_editor" - static init_JSMEEditor(): void { + static { this.prototype.default_view = JSMEEditorView; this.define(({String, Array}) => ({ @@ -236,4 +242,4 @@ export class JSMEEditor extends HTMLBox { guicolor: [String, "#c0c0c0"], })) } -} \ No newline at end of file +} diff --git a/src/panel_chemistry/bokeh_extensions/layout.ts b/src/panel_chemistry/bokeh_extensions/layout.ts new file mode 100644 index 0000000..cda8b97 --- /dev/null +++ b/src/panel_chemistry/bokeh_extensions/layout.ts @@ -0,0 +1,104 @@ +import {isArray} from "@bokehjs/core/util/types" +import {LayoutDOM, LayoutDOMView} from "@bokehjs/models/layouts/layout_dom" +import * as p from "@bokehjs/core/properties" + +export function uuidv4() { + return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) { + var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8); + return v.toString(16); + }); +} + +export function set_size(el: HTMLElement, model: HTMLBox, adjustMargin: boolean = true): void { + let width_policy = model.width != null ? "fixed" : "fit" + let height_policy = model.height != null ? "fixed" : "fit" + const {sizing_mode, margin} = model + if (sizing_mode != null) { + if (sizing_mode == "fixed") + width_policy = height_policy = "fixed" + else if (sizing_mode == "stretch_both") + width_policy = height_policy = "max" + else if (sizing_mode == "stretch_width") + width_policy = "max" + else if (sizing_mode == "stretch_height") + height_policy = "max" + else { + switch (sizing_mode) { + case "scale_width": + width_policy = "max" + height_policy = "min" + break + case "scale_height": + width_policy = "min" + height_policy = "max" + break + case "scale_both": + width_policy = "max" + height_policy = "max" + break + default: + throw new Error("unreachable") + } + } + } + let wm: number, hm: number + if (!adjustMargin) { + hm = wm = 0 + } else if (isArray(margin)) { + if (margin.length === 4) { + hm = margin[0] + margin[2] + wm = margin[1] + margin[3] + } else { + hm = margin[0] * 2 + wm = margin[1] * 2 + } + } else if (margin == null) { + hm = wm = 0 + } else { + wm = hm = margin * 2 + } + if (width_policy == "fixed" && model.width) + el.style.width = model.width + "px"; + else if (width_policy == "max") + el.style.width = wm ? `calc(100% - ${wm}px)`: "100%"; + if (model.min_width != null) + el.style.minWidth = model.min_width + "px"; + if (model.max_width != null) + el.style.maxWidth = model.max_width + "px"; + if (height_policy == "fixed" && model.height) + el.style.height = model.height + "px"; + else if (height_policy == "max") + el.style.height = hm ? `calc(100% - ${hm}px)`: "100%"; + if (model.min_height != null) + el.style.minHeight = model.min_height + "px"; + if (model.max_width != null) + el.style.maxHeight = model.max_height + "px"; + } + + export abstract class HTMLBoxView extends LayoutDOMView { + override model: HTMLBox + + render(): void { + super.render() + set_size(this.el, this.model) + } + + get child_models(): LayoutDOM[] { + return [] + } + } + + export namespace HTMLBox { + export type Attrs = p.AttrsOf + export type Props = LayoutDOM.Props + } + + export interface HTMLBox extends HTMLBox.Attrs {} + + export abstract class HTMLBox extends LayoutDOM { + override properties: HTMLBox.Props + + constructor(attrs?: Partial) { + super(attrs) + } + } \ No newline at end of file diff --git a/src/panel_chemistry/bokeh_extensions/ngl_viewer.py b/src/panel_chemistry/bokeh_extensions/ngl_viewer.py index a40a2bd..940b228 100644 --- a/src/panel_chemistry/bokeh_extensions/ngl_viewer.py +++ b/src/panel_chemistry/bokeh_extensions/ngl_viewer.py @@ -3,19 +3,20 @@ from bokeh.models import LayoutDOM -class NGLViewer(LayoutDOM): +class NGLViewer(LayoutDOM): # pylint: disable=too-many-ancestors """The [NGL Viewer](https://github.com/nglviewer/ngl) can be used to show and analyse pdb molecule structures""" - object = String() - extension = String() - representation = String() - color_scheme = String() - effect = String() + object = String("") + extension = String("") + background_color = String("") + representation = String("ball+stick") + color_scheme = String("element") custom_color_scheme = List(List(String)) + effect = String("") __javascript__ = [ - "https://unpkg.com/ngl@2.0.0-dev.37/dist/ngl.js", + "https://unpkg.com/ngl@2.2.1/dist/ngl.js", ] __js_skip__ = { @@ -24,7 +25,7 @@ class NGLViewer(LayoutDOM): __js_require__ = { "paths": { - "NGL": "https://unpkg.com/ngl@2.0.0-dev.37/dist/ngl", + "NGL": "https://unpkg.com/ngl@2.2.1/dist/ngl", }, "exports": {"NGL": "NGL"}, } diff --git a/src/panel_chemistry/bokeh_extensions/ngl_viewer.ts b/src/panel_chemistry/bokeh_extensions/ngl_viewer.ts index e69f961..826b8c3 100644 --- a/src/panel_chemistry/bokeh_extensions/ngl_viewer.ts +++ b/src/panel_chemistry/bokeh_extensions/ngl_viewer.ts @@ -1,5 +1,6 @@ import * as p from "@bokehjs/core/properties" -import {HTMLBox, HTMLBoxView} from "@bokehjs/models/layouts/html_box" +import {uuidv4, HTMLBox, HTMLBoxView} from "./layout" +import {div} from "@bokehjs/core/dom" declare namespace NGL { class AtomProxy{ @@ -86,8 +87,19 @@ declare namespace NGL { } export class NGLViewerView extends HTMLBoxView { model: NGLViewer + container: HTMLDivElement; + _intialized: boolean = false _stage: any + initialize(): void { + super.initialize() + this.container = div({ + class: "ngl-viewer", + id: "ngl-viewer-" + uuidv4(), + style: {width: "100%", height: "100%"} + }) + } + connect_signals(): void { super.connect_signals() this.connect(this.model.properties.object.change, this.updateStage) @@ -96,27 +108,39 @@ export class NGLViewerView extends HTMLBoxView { this.connect(this.model.properties.color_scheme.change, this.updateParameters) this.connect(this.model.properties.custom_color_scheme.change, this.updateParameters) this.connect(this.model.properties.effect.change, this.updateEffect) - this.connect(this.model.properties.background.change, this.setBackgroundcolor) + this.connect(this.model.properties.background_color.change, this.setBackgroundcolor) } render(): void { super.render() - this.el.id = "viewport" + this._intialized = false + this.createNGLViewer() + } + setBackgroundcolor(): void { + this._stage.setParameters( { backgroundColor: this.model.background_color} ); + } + createNGLViewer(){ + if (this._intialized) + return + document.body.appendChild(this.container) + const wn = (window as any) const ngl = wn.NGL - - this._stage = new ngl.Stage(this.el); + + this._stage = new ngl.Stage(this.container.id); this.setBackgroundcolor() const stage = this._stage this.updateStage(); window.addEventListener( "resize", function(){ stage.handleResize(); }, false ); - } - setBackgroundcolor(): void { - console.log(this.model.background) - this._stage.setParameters( { backgroundColor: this.model.background} ); + + // Remove from document body and add to shadow DOM + document.body.removeChild(this.container) + this.shadow_el.appendChild(this.container) + + this._intialized = true } after_layout(): void { super.after_layout() @@ -177,6 +201,7 @@ export namespace NGLViewer { export type Attrs = p.AttrsOf export type Props = HTMLBox.Props & { object: p.Property, + background_color: p.Property, extension: p.Property, representation: p.Property, color_scheme: p.Property, @@ -188,6 +213,7 @@ export namespace NGLViewer { export interface NGLViewer extends NGLViewer.Attrs { } export class NGLViewer extends HTMLBox { properties: NGLViewer.Props + __view_type__: NGLViewerView constructor(attrs?: Partial) { super(attrs) @@ -195,15 +221,16 @@ export class NGLViewer extends HTMLBox { static __module__ = "panel_chemistry.bokeh_extensions.ngl_viewer" - static init_NGLViewer(): void { + static { this.prototype.default_view = NGLViewerView; this.define(({ String, Any }) => ({ object: [ String, ""], extension: [ String, ""], - representation: [ String, "ribbon"], - color_scheme: [ String, "chainid"], - custom_color_scheme: [ Any, "chainid"], + background_color: [ String, ""], + representation: [ String, "ball+stick"], + color_scheme: [ String, "element"], + custom_color_scheme: [ Any, ["white", "*"]], effect: [ String, ""], })) diff --git a/src/panel_chemistry/package-lock.json b/src/panel_chemistry/package-lock.json index bdf256f..84e63b4 100644 --- a/src/panel_chemistry/package-lock.json +++ b/src/panel_chemistry/package-lock.json @@ -1,784 +1,25 @@ { "name": "panel_chemistry", - "version": "0.2.2", - "lockfileVersion": 2, + "version": "0.3.0", + "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "panel_chemistry", - "version": "0.2.2", + "version": "0.3.0", "license": "Apache 2.0", "dependencies": { - "@bokeh/bokehjs": "^2.4.3" - } - }, - "node_modules/@babel/runtime": { - "version": "7.14.0", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.14.0.tgz", - "integrity": "sha512-JELkvo/DlpNdJ7dlyw/eY7E0suy5i5GQH+Vlxaq1nsNJ+H7f4Vtv3jMeCEgRhZZQFXTjldYfQgv2qmM6M1v5wA==", - "dependencies": { - "regenerator-runtime": "^0.13.4" + "@bokeh/bokehjs": "~3.3.1" } }, "node_modules/@bokeh/bokehjs": { - "version": "2.4.3", - "resolved": "https://registry.npmjs.org/@bokeh/bokehjs/-/bokehjs-2.4.3.tgz", - "integrity": "sha512-zORGCM+3A2if+owfH03QFCL4GdnkhZizMlGjjGekAOp7Hc4TRqH9rQolEKKaNov7pmIay/VAMjZX4pyMXYW+Ag==", - "dependencies": { - "@bokeh/numbro": "^1.6.2", - "@bokeh/slickgrid": "~2.4.2702", - "choices.js": "^9.0.1", - "es5-ext": "^0.10.53", - "es6-map": "^0.1.5", - "es6-promise": "4.2.8", - "es6-set": "^0.1.5", - "es6-symbol": "^3.1.3", - "es6-weak-map": "^2.0.2", - "flatbush": "^3.2.1", - "flatpickr": "4.6.6", - "hammerjs": "^2.0.4", - "mathjax-full": "^3.2.0", - "nouislider": "^15.4.0", - "proj4": "^2.7.5", - "regl": "^2.1.0", - "sprintf-js": "^1.1.2", - "timezone": "^1.0.23", - "tslib": "^2.3.1", - "underscore.template": "^0.1.7" - }, - "engines": { - "node": ">=14", - "npm": ">=7.4" - } - }, - "node_modules/@bokeh/numbro": { - "version": "1.6.2", - "resolved": "https://registry.npmjs.org/@bokeh/numbro/-/numbro-1.6.2.tgz", - "integrity": "sha512-owIECPc3T3QXHCb2v5Ez+/uE9SIxI7N4nd9iFlWnfBrOelr0/omvFn09VisRn37AAFAY39sJiCVgECwryHWUPA==", + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/@bokeh/bokehjs/-/bokehjs-3.3.1.tgz", + "integrity": "sha512-orQM4d6MP1g7sXpG6B76HcwQLZIwVAMfopU6n7pZEB5430mINNH5VXlAWloBDveavcEpOwy+c0vBZLTYqOsgBg==", "engines": { - "node": "*" - } - }, - "node_modules/@bokeh/slickgrid": { - "version": "2.4.2702", - "resolved": "https://registry.npmjs.org/@bokeh/slickgrid/-/slickgrid-2.4.2702.tgz", - "integrity": "sha512-W9tm8Qdw5BrylbZbaVWaQMgLfW/klesnj6J3FnyWpo18hCCOFApccUD8iOnRv7bF6PHlgWk84mW3JT5RSzYKjA==", - "dependencies": { - "@types/slickgrid": "^2.1.30", - "jquery": ">=3.4.0", - "jquery-ui": ">=1.8.0", - "tslib": "^1.10.0" + "node": ">=16.0", + "npm": ">=8.0" } - }, - "node_modules/@bokeh/slickgrid/node_modules/tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" - }, - "node_modules/@types/jquery": { - "version": "3.5.5", - "resolved": "https://registry.npmjs.org/@types/jquery/-/jquery-3.5.5.tgz", - "integrity": "sha512-6RXU9Xzpc6vxNrS6FPPapN1SxSHgQ336WC6Jj/N8q30OiaBZ00l1GBgeP7usjVZPivSkGUfL1z/WW6TX989M+w==", - "dependencies": { - "@types/sizzle": "*" - } - }, - "node_modules/@types/sizzle": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/@types/sizzle/-/sizzle-2.3.3.tgz", - "integrity": "sha512-JYM8x9EGF163bEyhdJBpR2QX1R5naCJHC8ucJylJ3w9/CVBaskdQ8WqBf8MmQrd1kRvp/a4TS8HJ+bxzR7ZJYQ==" - }, - "node_modules/@types/slickgrid": { - "version": "2.1.30", - "resolved": "https://registry.npmjs.org/@types/slickgrid/-/slickgrid-2.1.30.tgz", - "integrity": "sha512-9nTqNWD3BtEVK0CP+G+mBtvSrKTfQy3Dg5/al+GdTSVMHFm37UxsHJ1eURwPg7rYu6vc7xU95fGTCKMZbxsD5w==", - "dependencies": { - "@types/jquery": "*" - } - }, - "node_modules/choices.js": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/choices.js/-/choices.js-9.0.1.tgz", - "integrity": "sha512-JgpeDY0Tmg7tqY6jaW/druSklJSt7W68tXFJIw0GSGWmO37SDAL8o60eICNGbzIODjj02VNNtf5h6TgoHDtCsA==", - "dependencies": { - "deepmerge": "^4.2.0", - "fuse.js": "^3.4.5", - "redux": "^4.0.4" - } - }, - "node_modules/commander": { - "version": "8.3.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", - "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==", - "engines": { - "node": ">= 12" - } - }, - "node_modules/d": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz", - "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==", - "dependencies": { - "es5-ext": "^0.10.50", - "type": "^1.0.1" - } - }, - "node_modules/deepmerge": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz", - "integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/es5-ext": { - "version": "0.10.53", - "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.53.tgz", - "integrity": "sha512-Xs2Stw6NiNHWypzRTY1MtaG/uJlwCk8kH81920ma8mvN8Xq1gsfhZvpkImLQArw8AHnv8MT2I45J3c0R8slE+Q==", - "dependencies": { - "es6-iterator": "~2.0.3", - "es6-symbol": "~3.1.3", - "next-tick": "~1.0.0" - } - }, - "node_modules/es6-iterator": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz", - "integrity": "sha1-p96IkUGgWpSwhUQDstCg+/qY87c=", - "dependencies": { - "d": "1", - "es5-ext": "^0.10.35", - "es6-symbol": "^3.1.1" - } - }, - "node_modules/es6-map": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/es6-map/-/es6-map-0.1.5.tgz", - "integrity": "sha1-kTbgUD3MBqMBaQ8LsU/042TpSfA=", - "dependencies": { - "d": "1", - "es5-ext": "~0.10.14", - "es6-iterator": "~2.0.1", - "es6-set": "~0.1.5", - "es6-symbol": "~3.1.1", - "event-emitter": "~0.3.5" - } - }, - "node_modules/es6-promise": { - "version": "4.2.8", - "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", - "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==" - }, - "node_modules/es6-set": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/es6-set/-/es6-set-0.1.5.tgz", - "integrity": "sha1-0rPsXU2ADO2BjbU40ol02wpzzLE=", - "dependencies": { - "d": "1", - "es5-ext": "~0.10.14", - "es6-iterator": "~2.0.1", - "es6-symbol": "3.1.1", - "event-emitter": "~0.3.5" - } - }, - "node_modules/es6-set/node_modules/es6-symbol": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.1.tgz", - "integrity": "sha1-vwDvT9q2uhtG7Le2KbTH7VcVzHc=", - "dependencies": { - "d": "1", - "es5-ext": "~0.10.14" - } - }, - "node_modules/es6-symbol": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.3.tgz", - "integrity": "sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==", - "dependencies": { - "d": "^1.0.1", - "ext": "^1.1.2" - } - }, - "node_modules/es6-weak-map": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/es6-weak-map/-/es6-weak-map-2.0.3.tgz", - "integrity": "sha512-p5um32HOTO1kP+w7PRnB+5lQ43Z6muuMuIMffvDN8ZB4GcnjLBV6zGStpbASIMk4DCAvEaamhe2zhyCb/QXXsA==", - "dependencies": { - "d": "1", - "es5-ext": "^0.10.46", - "es6-iterator": "^2.0.3", - "es6-symbol": "^3.1.1" - } - }, - "node_modules/esm": { - "version": "3.2.25", - "resolved": "https://registry.npmjs.org/esm/-/esm-3.2.25.tgz", - "integrity": "sha512-U1suiZ2oDVWv4zPO56S0NcR5QriEahGtdN2OR6FiOG4WJvcjBVFB0qI4+eKoWFH483PKGuLuu6V8Z4T5g63UVA==", - "engines": { - "node": ">=6" - } - }, - "node_modules/event-emitter": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/event-emitter/-/event-emitter-0.3.5.tgz", - "integrity": "sha1-34xp7vFkeSPHFXuc6DhAYQsCzDk=", - "dependencies": { - "d": "1", - "es5-ext": "~0.10.14" - } - }, - "node_modules/ext": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/ext/-/ext-1.4.0.tgz", - "integrity": "sha512-Key5NIsUxdqKg3vIsdw9dSuXpPCQ297y6wBjL30edxwPgt2E44WcWBZey/ZvUc6sERLTxKdyCu4gZFmUbk1Q7A==", - "dependencies": { - "type": "^2.0.0" - } - }, - "node_modules/ext/node_modules/type": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/type/-/type-2.5.0.tgz", - "integrity": "sha512-180WMDQaIMm3+7hGXWf12GtdniDEy7nYcyFMKJn/eZz/6tSLXrUN9V0wKSbMjej0I1WHWbpREDEKHtqPQa9NNw==" - }, - "node_modules/flatbush": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/flatbush/-/flatbush-3.3.0.tgz", - "integrity": "sha512-F3EzQvKpdmXUbFwWxLKBpytOFEGYQMCTBLuqZ4GEajFOEAvnOIBiyxW3OFSZXIOtpCS8teN6bFEpNZtnVXuDQA==", - "dependencies": { - "flatqueue": "^1.2.0" - } - }, - "node_modules/flatpickr": { - "version": "4.6.6", - "resolved": "https://registry.npmjs.org/flatpickr/-/flatpickr-4.6.6.tgz", - "integrity": "sha512-EZ48CJMttMg3maMhJoX+GvTuuEhX/RbA1YeuI19attP3pwBdbYy6+yqAEVm0o0hSBFYBiLbVxscLW6gJXq6H3A==" - }, - "node_modules/flatqueue": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/flatqueue/-/flatqueue-1.2.1.tgz", - "integrity": "sha512-X86TpWS1rGuY7m382HuA9vngLeDuWA9lJvhEG+GfgKMV5onSvx5a71cl7GMbXzhWtlN9dGfqOBrpfqeOtUfGYQ==" - }, - "node_modules/fuse.js": { - "version": "3.6.1", - "resolved": "https://registry.npmjs.org/fuse.js/-/fuse.js-3.6.1.tgz", - "integrity": "sha512-hT9yh/tiinkmirKrlv4KWOjztdoZo1mx9Qh4KvWqC7isoXwdUY3PNWUxceF4/qO9R6riA2C29jdTOeQOIROjgw==", - "engines": { - "node": ">=6" - } - }, - "node_modules/hammerjs": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/hammerjs/-/hammerjs-2.0.8.tgz", - "integrity": "sha1-BO93hiz/K7edMPdpIJWTAiK/YPE=", - "engines": { - "node": ">=0.8.0" - } - }, - "node_modules/jquery": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/jquery/-/jquery-3.6.0.tgz", - "integrity": "sha512-JVzAR/AjBvVt2BmYhxRCSYysDsPcssdmTFnzyLEts9qNwmjmu4JTAMYubEfwVOSwpQ1I1sKKFcxhZCI2buerfw==" - }, - "node_modules/jquery-ui": { - "version": "1.13.2", - "resolved": "https://registry.npmjs.org/jquery-ui/-/jquery-ui-1.13.2.tgz", - "integrity": "sha512-wBZPnqWs5GaYJmo1Jj0k/mrSkzdQzKDwhXNtHKcBdAcKVxMM3KNYFq+iJ2i1rwiG53Z8M4mTn3Qxrm17uH1D4Q==", - "dependencies": { - "jquery": ">=1.8.0 <4.0.0" - } - }, - "node_modules/mathjax-full": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/mathjax-full/-/mathjax-full-3.2.0.tgz", - "integrity": "sha512-D2EBNvUG+mJyhn+M1C858k0f2Fc4KxXvbEX2WCMXroV10212JwfYqaBJ336ECBSz5X9L5LRoamxb7AJtg3KaJA==", - "dependencies": { - "esm": "^3.2.25", - "mhchemparser": "^4.1.0", - "mj-context-menu": "^0.6.1", - "speech-rule-engine": "^3.3.3" - } - }, - "node_modules/mgrs": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/mgrs/-/mgrs-1.0.0.tgz", - "integrity": "sha1-+5FYjnjJACVnI5XLQLJffNatGCk=" - }, - "node_modules/mhchemparser": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/mhchemparser/-/mhchemparser-4.1.1.tgz", - "integrity": "sha512-R75CUN6O6e1t8bgailrF1qPq+HhVeFTM3XQ0uzI+mXTybmphy3b6h4NbLOYhemViQ3lUs+6CKRkC3Ws1TlYREA==" - }, - "node_modules/mj-context-menu": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/mj-context-menu/-/mj-context-menu-0.6.1.tgz", - "integrity": "sha512-7NO5s6n10TIV96d4g2uDpG7ZDpIhMh0QNfGdJw/W47JswFcosz457wqz/b5sAKvl12sxINGFCn80NZHKwxQEXA==" - }, - "node_modules/next-tick": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.0.0.tgz", - "integrity": "sha1-yobR/ogoFpsBICCOPchCS524NCw=" - }, - "node_modules/nouislider": { - "version": "15.5.0", - "resolved": "https://registry.npmjs.org/nouislider/-/nouislider-15.5.0.tgz", - "integrity": "sha512-p0Rn0a4XzrBJ+JZRhNDYpRYr6sDPkajsjbvEQoTp/AZlNI3NirO15s1t11D25Gk3zVyvNJAzc1DO48cq/KX5Sw==" - }, - "node_modules/proj4": { - "version": "2.7.5", - "resolved": "https://registry.npmjs.org/proj4/-/proj4-2.7.5.tgz", - "integrity": "sha512-5ecXUXbHAfvdhfBQpU7EhUfPCQGUCPmVup/4gnZA3bJY3JcK/xxzm4QQDz1xiXokN6ux65VDczlCtBtKrTSpAQ==", - "dependencies": { - "mgrs": "1.0.0", - "wkt-parser": "^1.3.1" - } - }, - "node_modules/redux": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/redux/-/redux-4.1.0.tgz", - "integrity": "sha512-uI2dQN43zqLWCt6B/BMGRMY6db7TTY4qeHHfGeKb3EOhmOKjU3KdWvNLJyqaHRksv/ErdNH7cFZWg9jXtewy4g==", - "dependencies": { - "@babel/runtime": "^7.9.2" - } - }, - "node_modules/regenerator-runtime": { - "version": "0.13.7", - "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.7.tgz", - "integrity": "sha512-a54FxoJDIr27pgf7IgeQGxmqUNYrcV338lf/6gH456HZ/PhX+5BcwHXG9ajESmwe6WRO0tAzRUrRmNONWgkrew==" - }, - "node_modules/regl": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/regl/-/regl-2.1.0.tgz", - "integrity": "sha512-oWUce/aVoEvW5l2V0LK7O5KJMzUSKeiOwFuJehzpSFd43dO5spP9r+sSUfhKtsky4u6MCqWJaRL+abzExynfTg==" - }, - "node_modules/speech-rule-engine": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/speech-rule-engine/-/speech-rule-engine-3.3.3.tgz", - "integrity": "sha512-0exWw+0XauLjat+f/aFeo5T8SiDsO1JtwpY3qgJE4cWt+yL/Stl0WP4VNDWdh7lzGkubUD9lWP4J1ASnORXfyQ==", - "dependencies": { - "commander": ">=7.0.0", - "wicked-good-xpath": "^1.3.0", - "xmldom-sre": "^0.1.31" - }, - "bin": { - "sre": "bin/sre" - } - }, - "node_modules/sprintf-js": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", - "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==" - }, - "node_modules/timezone": { - "version": "1.0.23", - "resolved": "https://registry.npmjs.org/timezone/-/timezone-1.0.23.tgz", - "integrity": "sha512-yhQgk6qmSLB+TF8HGmApZAVI5bfzR1CoKUGr+WMZWmx75ED1uDewAZA8QMGCQ70TEv4GmM8pDB9jrHuxdaQ1PA==" - }, - "node_modules/tslib": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.3.1.tgz", - "integrity": "sha512-77EbyPPpMz+FRFRuAFlWMtmgUWGe9UOG2Z25NqCwiIjRhOf5iKGuzSe5P2w1laq+FkRy4p+PCuVkJSGkzTEKVw==" - }, - "node_modules/type": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz", - "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==" - }, - "node_modules/underscore.template": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/underscore.template/-/underscore.template-0.1.7.tgz", - "integrity": "sha1-MBPg6hgXVjBvFgnpWcr7xyKts+k=" - }, - "node_modules/wicked-good-xpath": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/wicked-good-xpath/-/wicked-good-xpath-1.3.0.tgz", - "integrity": "sha1-gbDpXoZQ5JyUsiKY//hoa1VTz2w=" - }, - "node_modules/wkt-parser": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/wkt-parser/-/wkt-parser-1.3.1.tgz", - "integrity": "sha512-XK5qV+Y5gsygQfHx2/cS5a7Zxsgleaw8iX5UPC5eOXPc0TgJAu1JB9lr0iYYX3zAnN3p0aNiaN5c+1Bdblxwrg==" - }, - "node_modules/xmldom-sre": { - "version": "0.1.31", - "resolved": "https://registry.npmjs.org/xmldom-sre/-/xmldom-sre-0.1.31.tgz", - "integrity": "sha512-f9s+fUkX04BxQf+7mMWAp5zk61pciie+fFLC9hX9UVvCeJQfNHRHXpeo5MPcR0EUf57PYLdt+ZO4f3Ipk2oZUw==", - "engines": { - "node": ">=0.1" - } - } - }, - "dependencies": { - "@babel/runtime": { - "version": "7.14.0", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.14.0.tgz", - "integrity": "sha512-JELkvo/DlpNdJ7dlyw/eY7E0suy5i5GQH+Vlxaq1nsNJ+H7f4Vtv3jMeCEgRhZZQFXTjldYfQgv2qmM6M1v5wA==", - "requires": { - "regenerator-runtime": "^0.13.4" - } - }, - "@bokeh/bokehjs": { - "version": "2.4.3", - "resolved": "https://registry.npmjs.org/@bokeh/bokehjs/-/bokehjs-2.4.3.tgz", - "integrity": "sha512-zORGCM+3A2if+owfH03QFCL4GdnkhZizMlGjjGekAOp7Hc4TRqH9rQolEKKaNov7pmIay/VAMjZX4pyMXYW+Ag==", - "requires": { - "@bokeh/numbro": "^1.6.2", - "@bokeh/slickgrid": "~2.4.2702", - "choices.js": "^9.0.1", - "es5-ext": "^0.10.53", - "es6-map": "^0.1.5", - "es6-promise": "4.2.8", - "es6-set": "^0.1.5", - "es6-symbol": "^3.1.3", - "es6-weak-map": "^2.0.2", - "flatbush": "^3.2.1", - "flatpickr": "4.6.6", - "hammerjs": "^2.0.4", - "mathjax-full": "^3.2.0", - "nouislider": "^15.4.0", - "proj4": "^2.7.5", - "regl": "^2.1.0", - "sprintf-js": "^1.1.2", - "timezone": "^1.0.23", - "tslib": "^2.3.1", - "underscore.template": "^0.1.7" - } - }, - "@bokeh/numbro": { - "version": "1.6.2", - "resolved": "https://registry.npmjs.org/@bokeh/numbro/-/numbro-1.6.2.tgz", - "integrity": "sha512-owIECPc3T3QXHCb2v5Ez+/uE9SIxI7N4nd9iFlWnfBrOelr0/omvFn09VisRn37AAFAY39sJiCVgECwryHWUPA==" - }, - "@bokeh/slickgrid": { - "version": "2.4.2702", - "resolved": "https://registry.npmjs.org/@bokeh/slickgrid/-/slickgrid-2.4.2702.tgz", - "integrity": "sha512-W9tm8Qdw5BrylbZbaVWaQMgLfW/klesnj6J3FnyWpo18hCCOFApccUD8iOnRv7bF6PHlgWk84mW3JT5RSzYKjA==", - "requires": { - "@types/slickgrid": "^2.1.30", - "jquery": ">=3.4.0", - "jquery-ui": ">=1.8.0", - "tslib": "^1.10.0" - }, - "dependencies": { - "tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" - } - } - }, - "@types/jquery": { - "version": "3.5.5", - "resolved": "https://registry.npmjs.org/@types/jquery/-/jquery-3.5.5.tgz", - "integrity": "sha512-6RXU9Xzpc6vxNrS6FPPapN1SxSHgQ336WC6Jj/N8q30OiaBZ00l1GBgeP7usjVZPivSkGUfL1z/WW6TX989M+w==", - "requires": { - "@types/sizzle": "*" - } - }, - "@types/sizzle": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/@types/sizzle/-/sizzle-2.3.3.tgz", - "integrity": "sha512-JYM8x9EGF163bEyhdJBpR2QX1R5naCJHC8ucJylJ3w9/CVBaskdQ8WqBf8MmQrd1kRvp/a4TS8HJ+bxzR7ZJYQ==" - }, - "@types/slickgrid": { - "version": "2.1.30", - "resolved": "https://registry.npmjs.org/@types/slickgrid/-/slickgrid-2.1.30.tgz", - "integrity": "sha512-9nTqNWD3BtEVK0CP+G+mBtvSrKTfQy3Dg5/al+GdTSVMHFm37UxsHJ1eURwPg7rYu6vc7xU95fGTCKMZbxsD5w==", - "requires": { - "@types/jquery": "*" - } - }, - "choices.js": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/choices.js/-/choices.js-9.0.1.tgz", - "integrity": "sha512-JgpeDY0Tmg7tqY6jaW/druSklJSt7W68tXFJIw0GSGWmO37SDAL8o60eICNGbzIODjj02VNNtf5h6TgoHDtCsA==", - "requires": { - "deepmerge": "^4.2.0", - "fuse.js": "^3.4.5", - "redux": "^4.0.4" - } - }, - "commander": { - "version": "8.3.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", - "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==" - }, - "d": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz", - "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==", - "requires": { - "es5-ext": "^0.10.50", - "type": "^1.0.1" - } - }, - "deepmerge": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz", - "integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==" - }, - "es5-ext": { - "version": "0.10.53", - "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.53.tgz", - "integrity": "sha512-Xs2Stw6NiNHWypzRTY1MtaG/uJlwCk8kH81920ma8mvN8Xq1gsfhZvpkImLQArw8AHnv8MT2I45J3c0R8slE+Q==", - "requires": { - "es6-iterator": "~2.0.3", - "es6-symbol": "~3.1.3", - "next-tick": "~1.0.0" - } - }, - "es6-iterator": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz", - "integrity": "sha1-p96IkUGgWpSwhUQDstCg+/qY87c=", - "requires": { - "d": "1", - "es5-ext": "^0.10.35", - "es6-symbol": "^3.1.1" - } - }, - "es6-map": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/es6-map/-/es6-map-0.1.5.tgz", - "integrity": "sha1-kTbgUD3MBqMBaQ8LsU/042TpSfA=", - "requires": { - "d": "1", - "es5-ext": "~0.10.14", - "es6-iterator": "~2.0.1", - "es6-set": "~0.1.5", - "es6-symbol": "~3.1.1", - "event-emitter": "~0.3.5" - } - }, - "es6-promise": { - "version": "4.2.8", - "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", - "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==" - }, - "es6-set": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/es6-set/-/es6-set-0.1.5.tgz", - "integrity": "sha1-0rPsXU2ADO2BjbU40ol02wpzzLE=", - "requires": { - "d": "1", - "es5-ext": "~0.10.14", - "es6-iterator": "~2.0.1", - "es6-symbol": "3.1.1", - "event-emitter": "~0.3.5" - }, - "dependencies": { - "es6-symbol": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.1.tgz", - "integrity": "sha1-vwDvT9q2uhtG7Le2KbTH7VcVzHc=", - "requires": { - "d": "1", - "es5-ext": "~0.10.14" - } - } - } - }, - "es6-symbol": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.3.tgz", - "integrity": "sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==", - "requires": { - "d": "^1.0.1", - "ext": "^1.1.2" - } - }, - "es6-weak-map": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/es6-weak-map/-/es6-weak-map-2.0.3.tgz", - "integrity": "sha512-p5um32HOTO1kP+w7PRnB+5lQ43Z6muuMuIMffvDN8ZB4GcnjLBV6zGStpbASIMk4DCAvEaamhe2zhyCb/QXXsA==", - "requires": { - "d": "1", - "es5-ext": "^0.10.46", - "es6-iterator": "^2.0.3", - "es6-symbol": "^3.1.1" - } - }, - "esm": { - "version": "3.2.25", - "resolved": "https://registry.npmjs.org/esm/-/esm-3.2.25.tgz", - "integrity": "sha512-U1suiZ2oDVWv4zPO56S0NcR5QriEahGtdN2OR6FiOG4WJvcjBVFB0qI4+eKoWFH483PKGuLuu6V8Z4T5g63UVA==" - }, - "event-emitter": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/event-emitter/-/event-emitter-0.3.5.tgz", - "integrity": "sha1-34xp7vFkeSPHFXuc6DhAYQsCzDk=", - "requires": { - "d": "1", - "es5-ext": "~0.10.14" - } - }, - "ext": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/ext/-/ext-1.4.0.tgz", - "integrity": "sha512-Key5NIsUxdqKg3vIsdw9dSuXpPCQ297y6wBjL30edxwPgt2E44WcWBZey/ZvUc6sERLTxKdyCu4gZFmUbk1Q7A==", - "requires": { - "type": "^2.0.0" - }, - "dependencies": { - "type": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/type/-/type-2.5.0.tgz", - "integrity": "sha512-180WMDQaIMm3+7hGXWf12GtdniDEy7nYcyFMKJn/eZz/6tSLXrUN9V0wKSbMjej0I1WHWbpREDEKHtqPQa9NNw==" - } - } - }, - "flatbush": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/flatbush/-/flatbush-3.3.0.tgz", - "integrity": "sha512-F3EzQvKpdmXUbFwWxLKBpytOFEGYQMCTBLuqZ4GEajFOEAvnOIBiyxW3OFSZXIOtpCS8teN6bFEpNZtnVXuDQA==", - "requires": { - "flatqueue": "^1.2.0" - } - }, - "flatpickr": { - "version": "4.6.6", - "resolved": "https://registry.npmjs.org/flatpickr/-/flatpickr-4.6.6.tgz", - "integrity": "sha512-EZ48CJMttMg3maMhJoX+GvTuuEhX/RbA1YeuI19attP3pwBdbYy6+yqAEVm0o0hSBFYBiLbVxscLW6gJXq6H3A==" - }, - "flatqueue": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/flatqueue/-/flatqueue-1.2.1.tgz", - "integrity": "sha512-X86TpWS1rGuY7m382HuA9vngLeDuWA9lJvhEG+GfgKMV5onSvx5a71cl7GMbXzhWtlN9dGfqOBrpfqeOtUfGYQ==" - }, - "fuse.js": { - "version": "3.6.1", - "resolved": "https://registry.npmjs.org/fuse.js/-/fuse.js-3.6.1.tgz", - "integrity": "sha512-hT9yh/tiinkmirKrlv4KWOjztdoZo1mx9Qh4KvWqC7isoXwdUY3PNWUxceF4/qO9R6riA2C29jdTOeQOIROjgw==" - }, - "hammerjs": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/hammerjs/-/hammerjs-2.0.8.tgz", - "integrity": "sha1-BO93hiz/K7edMPdpIJWTAiK/YPE=" - }, - "jquery": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/jquery/-/jquery-3.6.0.tgz", - "integrity": "sha512-JVzAR/AjBvVt2BmYhxRCSYysDsPcssdmTFnzyLEts9qNwmjmu4JTAMYubEfwVOSwpQ1I1sKKFcxhZCI2buerfw==" - }, - "jquery-ui": { - "version": "1.13.2", - "resolved": "https://registry.npmjs.org/jquery-ui/-/jquery-ui-1.13.2.tgz", - "integrity": "sha512-wBZPnqWs5GaYJmo1Jj0k/mrSkzdQzKDwhXNtHKcBdAcKVxMM3KNYFq+iJ2i1rwiG53Z8M4mTn3Qxrm17uH1D4Q==", - "requires": { - "jquery": ">=1.8.0 <4.0.0" - } - }, - "mathjax-full": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/mathjax-full/-/mathjax-full-3.2.0.tgz", - "integrity": "sha512-D2EBNvUG+mJyhn+M1C858k0f2Fc4KxXvbEX2WCMXroV10212JwfYqaBJ336ECBSz5X9L5LRoamxb7AJtg3KaJA==", - "requires": { - "esm": "^3.2.25", - "mhchemparser": "^4.1.0", - "mj-context-menu": "^0.6.1", - "speech-rule-engine": "^3.3.3" - } - }, - "mgrs": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/mgrs/-/mgrs-1.0.0.tgz", - "integrity": "sha1-+5FYjnjJACVnI5XLQLJffNatGCk=" - }, - "mhchemparser": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/mhchemparser/-/mhchemparser-4.1.1.tgz", - "integrity": "sha512-R75CUN6O6e1t8bgailrF1qPq+HhVeFTM3XQ0uzI+mXTybmphy3b6h4NbLOYhemViQ3lUs+6CKRkC3Ws1TlYREA==" - }, - "mj-context-menu": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/mj-context-menu/-/mj-context-menu-0.6.1.tgz", - "integrity": "sha512-7NO5s6n10TIV96d4g2uDpG7ZDpIhMh0QNfGdJw/W47JswFcosz457wqz/b5sAKvl12sxINGFCn80NZHKwxQEXA==" - }, - "next-tick": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.0.0.tgz", - "integrity": "sha1-yobR/ogoFpsBICCOPchCS524NCw=" - }, - "nouislider": { - "version": "15.5.0", - "resolved": "https://registry.npmjs.org/nouislider/-/nouislider-15.5.0.tgz", - "integrity": "sha512-p0Rn0a4XzrBJ+JZRhNDYpRYr6sDPkajsjbvEQoTp/AZlNI3NirO15s1t11D25Gk3zVyvNJAzc1DO48cq/KX5Sw==" - }, - "proj4": { - "version": "2.7.5", - "resolved": "https://registry.npmjs.org/proj4/-/proj4-2.7.5.tgz", - "integrity": "sha512-5ecXUXbHAfvdhfBQpU7EhUfPCQGUCPmVup/4gnZA3bJY3JcK/xxzm4QQDz1xiXokN6ux65VDczlCtBtKrTSpAQ==", - "requires": { - "mgrs": "1.0.0", - "wkt-parser": "^1.3.1" - } - }, - "redux": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/redux/-/redux-4.1.0.tgz", - "integrity": "sha512-uI2dQN43zqLWCt6B/BMGRMY6db7TTY4qeHHfGeKb3EOhmOKjU3KdWvNLJyqaHRksv/ErdNH7cFZWg9jXtewy4g==", - "requires": { - "@babel/runtime": "^7.9.2" - } - }, - "regenerator-runtime": { - "version": "0.13.7", - "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.7.tgz", - "integrity": "sha512-a54FxoJDIr27pgf7IgeQGxmqUNYrcV338lf/6gH456HZ/PhX+5BcwHXG9ajESmwe6WRO0tAzRUrRmNONWgkrew==" - }, - "regl": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/regl/-/regl-2.1.0.tgz", - "integrity": "sha512-oWUce/aVoEvW5l2V0LK7O5KJMzUSKeiOwFuJehzpSFd43dO5spP9r+sSUfhKtsky4u6MCqWJaRL+abzExynfTg==" - }, - "speech-rule-engine": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/speech-rule-engine/-/speech-rule-engine-3.3.3.tgz", - "integrity": "sha512-0exWw+0XauLjat+f/aFeo5T8SiDsO1JtwpY3qgJE4cWt+yL/Stl0WP4VNDWdh7lzGkubUD9lWP4J1ASnORXfyQ==", - "requires": { - "commander": ">=7.0.0", - "wicked-good-xpath": "^1.3.0", - "xmldom-sre": "^0.1.31" - } - }, - "sprintf-js": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", - "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==" - }, - "timezone": { - "version": "1.0.23", - "resolved": "https://registry.npmjs.org/timezone/-/timezone-1.0.23.tgz", - "integrity": "sha512-yhQgk6qmSLB+TF8HGmApZAVI5bfzR1CoKUGr+WMZWmx75ED1uDewAZA8QMGCQ70TEv4GmM8pDB9jrHuxdaQ1PA==" - }, - "tslib": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.3.1.tgz", - "integrity": "sha512-77EbyPPpMz+FRFRuAFlWMtmgUWGe9UOG2Z25NqCwiIjRhOf5iKGuzSe5P2w1laq+FkRy4p+PCuVkJSGkzTEKVw==" - }, - "type": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz", - "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==" - }, - "underscore.template": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/underscore.template/-/underscore.template-0.1.7.tgz", - "integrity": "sha1-MBPg6hgXVjBvFgnpWcr7xyKts+k=" - }, - "wicked-good-xpath": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/wicked-good-xpath/-/wicked-good-xpath-1.3.0.tgz", - "integrity": "sha1-gbDpXoZQ5JyUsiKY//hoa1VTz2w=" - }, - "wkt-parser": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/wkt-parser/-/wkt-parser-1.3.1.tgz", - "integrity": "sha512-XK5qV+Y5gsygQfHx2/cS5a7Zxsgleaw8iX5UPC5eOXPc0TgJAu1JB9lr0iYYX3zAnN3p0aNiaN5c+1Bdblxwrg==" - }, - "xmldom-sre": { - "version": "0.1.31", - "resolved": "https://registry.npmjs.org/xmldom-sre/-/xmldom-sre-0.1.31.tgz", - "integrity": "sha512-f9s+fUkX04BxQf+7mMWAp5zk61pciie+fFLC9hX9UVvCeJQfNHRHXpeo5MPcR0EUf57PYLdt+ZO4f3Ipk2oZUw==" } } } diff --git a/src/panel_chemistry/package.json b/src/panel_chemistry/package.json index 7c42cae..e9d0a3e 100644 --- a/src/panel_chemistry/package.json +++ b/src/panel_chemistry/package.json @@ -1,10 +1,10 @@ { "name": "panel_chemistry", - "version": "0.2.2", + "version": "0.3.0", "description": "Panel extension for data analysis and data apps within the domain of Chemistry", "license": "Apache 2.0", "keywords": [], "dependencies": { - "@bokeh/bokehjs": "^2.4.3" + "@bokeh/bokehjs": "~3.3.1" } } diff --git a/src/panel_chemistry/pane/ngl_viewer.py b/src/panel_chemistry/pane/ngl_viewer.py index 383e0ec..7ab361c 100644 --- a/src/panel_chemistry/pane/ngl_viewer.py +++ b/src/panel_chemistry/pane/ngl_viewer.py @@ -8,8 +8,6 @@ """ -from typing import Dict - import param from panel import extension from panel.pane.base import PaneBase @@ -98,10 +96,15 @@ class NGLViewer(PaneBase): # pylint: disable=too-many-ancestors to show and analyse pdb molecule structures""" object = param.String( + default="", doc=""" The object to display. For example an url like 'rcsb://3dqb.pdb', 'rcsb://1NKT', '1NKT'. You can also specify a extension string if you define the extension - in the extension parameter""" + in the extension parameter""", + ) + background_color = param.Color( + doc=""" + A custom background color""" ) extension = param.ObjectSelector( default="", @@ -161,15 +164,6 @@ def _get_model(self, doc, root=None, parent=None, comm=None): props["object"] = self.object model = _NGLViewer(**props) root = root or model - # self._link_props( - # model, - # [ - # "event", - # ], - # doc, - # root, - # comm, - # ) self._models[root.ref["id"]] = (model, parent) return model diff --git a/src/panel_chemistry/pane/pdbe_molstar.py b/src/panel_chemistry/pane/pdbe_molstar.py index ce84a3d..c8d18c4 100644 --- a/src/panel_chemistry/pane/pdbe_molstar.py +++ b/src/panel_chemistry/pane/pdbe_molstar.py @@ -29,6 +29,7 @@ "spacefill", ] + # See https://embed.plnkr.co/plunk/m3GxFYx9cBjIanBp for an example JS implementation class PDBeMolStar(ReactiveHTML): # pylint: disable=too-many-ancestors """PDBe MolStar structure viewer. @@ -127,10 +128,10 @@ class PDBeMolStar(ReactiveHTML): # pylint: disable=too-many-ancestors ) hide_controls = param.Boolean(default=True, doc="Hide the control menu") - + sequence_panel = param.Boolean( - default=True, - doc="Show the sequence panel. Currently shown only when the controls are toggled" + default=True, + doc="Show the sequence panel. Currently shown only when the controls are toggled", ) expanded = param.Boolean(default=False, doc="""Display full-screen by default on load""") diff --git a/src/panel_chemistry/pane/py3dmol_viewer.py b/src/panel_chemistry/pane/py3dmol_viewer.py index 9ad7657..d01d3b7 100644 --- a/src/panel_chemistry/pane/py3dmol_viewer.py +++ b/src/panel_chemistry/pane/py3dmol_viewer.py @@ -12,7 +12,7 @@ try: import py3Dmol except ModuleNotFoundError: - # pylint: disable=invalid-name + # pylint: disable=invalid-name, too-few-public-methods class py3Dmol(param.Parameterized): # type: ignore """Dummy py3Dmol class""" diff --git a/src/panel_chemistry/tsconfig.json b/src/panel_chemistry/tsconfig.json index 41720fb..6b4e52d 100644 --- a/src/panel_chemistry/tsconfig.json +++ b/src/panel_chemistry/tsconfig.json @@ -33,4 +33,4 @@ } }, "include": ["./**/*.ts"] -} \ No newline at end of file +} diff --git a/src/panel_chemistry/widgets/jsme_editor.py b/src/panel_chemistry/widgets/jsme_editor.py index fdece4f..d862b03 100644 --- a/src/panel_chemistry/widgets/jsme_editor.py +++ b/src/panel_chemistry/widgets/jsme_editor.py @@ -3,6 +3,10 @@ It is a wrapper of the free javascript JSME Molecule Editor. See https://jsme-editor.github.io/ """ +from __future__ import annotations + +from typing import ClassVar, Mapping + import param from panel.widgets.base import Widget @@ -67,11 +71,7 @@ class JSMEEditor(Widget): # pylint: disable=too-many-ancestors # Set the Bokeh model to use _widget_type = _BkJSMEEditor - # Rename Panel Parameters -> Bokeh Model properties - # Parameters like title that does not exist on the Bokeh model should be renamed to None - _rename = { - "title": None, - } + _rename: ClassVar[Mapping[str, str | None]] = {"name": None} # Parameters to be mapped to Bokeh model properties value = param.String( diff --git a/tests/conftest.py b/tests/conftest.py new file mode 100644 index 0000000..d681a1c --- /dev/null +++ b/tests/conftest.py @@ -0,0 +1,16 @@ +"""Shared fixtures""" +import pytest +from bokeh.document import Document +from pyviz_comms import Comm + + +@pytest.fixture +def document(): + """A Bokeh Document""" + return Document() + + +@pytest.fixture +def comm(): + """The pyviz communications""" + return Comm() diff --git a/tests/pane/test_ngl_viewer.py b/tests/pane/test_ngl_viewer.py index 2c17c63..f26db6c 100644 --- a/tests/pane/test_ngl_viewer.py +++ b/tests/pane/test_ngl_viewer.py @@ -5,21 +5,23 @@ from panel_chemistry.pane import NGLViewer -def test_can_create(): +def test_can_create(document, comm): """Test of the NGLViewer constructor""" - NGLViewer(object="1CRN", background="yellow", height=500) + viewer = NGLViewer(object="1CRN", background_color="yellow", height=500) + pane = viewer.get_root(document, comm=comm) + assert isinstance(pane, _BkNGLViewer) def test_has_bokeh_model(): """Test that the NGL Viewer Exists""" - assert _BkNGLViewer + assert _BkNGLViewer.__name__ -def test_app(): +def _create_app(): """Returns an app for manually testing the NGL Molecule Viewer""" pn.extension(sizing_mode="stretch_width") # 1NKT, 2GQ5, 3UOG and 5TXH - viewer = NGLViewer(object="1CRN", background="yellow", height=500) + viewer = NGLViewer(object="1CRN", background_color="yellow", height=500) parameters = [ "object", "extension", @@ -30,7 +32,7 @@ def test_app(): "sizing_mode", "width", "height", - "background", + "background_color", ] settings = pn.Param( viewer, @@ -46,5 +48,10 @@ def test_app(): ) +def test_app(): + """Can create test app""" + assert _create_app() + + if __name__.startswith("bokeh"): - test_app().servable() + _create_app().servable() diff --git a/tests/pane/test_pdbe_molstar_viewer.py b/tests/pane/test_pdbe_molstar_viewer.py index 6766e75..998f520 100644 --- a/tests/pane/test_pdbe_molstar_viewer.py +++ b/tests/pane/test_pdbe_molstar_viewer.py @@ -4,9 +4,11 @@ from panel_chemistry.pane import PDBeMolStar -def test_can_create(): +def test_can_create(document, comm): """Test of the PDBeMolStar constructor""" - PDBeMolStar(molecule_id="1qyn", lighting="metallic", height=300, width=300) + molstar = PDBeMolStar(molecule_id="1qyn", lighting="metallic", height=300, width=300) + widget = molstar.get_root(document, comm=comm) + assert widget def test_functions(): @@ -49,7 +51,7 @@ def test_functions(): viewer.reset(data) -def test_app(): +def _create_app(): """Returns an app for manually testing the PDBe Mol* Viewer""" pn.extension(sizing_mode="stretch_width") # 1NKT, 2GQ5, 3UOG and 5TXH @@ -102,5 +104,10 @@ def test_app(): return pn.Row(pn.WidgetBox(settings, width=300, sizing_mode="fixed"), viewer) +def test_app(): + """Can create test app""" + assert _create_app() + + if __name__.startswith("bokeh"): - test_app().servable() + _create_app().servable() diff --git a/tests/tests/test_jsme_editor.py b/tests/tests/test_jsme_editor.py index 13b5e1c..2b203a8 100644 --- a/tests/tests/test_jsme_editor.py +++ b/tests/tests/test_jsme_editor.py @@ -5,11 +5,13 @@ from panel_chemistry.widgets import JSMEEditor -def test_can_construct(): - JSMEEditor() +def test_can_construct(document, comm): + editor = JSMEEditor() + widget = editor.get_root(document, comm=comm) + assert isinstance(widget, editor._widget_type) # pylint: disable=protected-access -def test_jsme_editor_app(): +def _create_app(): pn.extension("jsme", sizing_mode="stretch_width") smiles = "N[C@@H](CCC(=O)N[C@@H](CS)C(=O)NCC(=O)O)C(=O)O" editor = JSMEEditor(value=smiles, height=500) @@ -49,5 +51,9 @@ def test_jsme_editor_app(): ) +def test_app(): + assert _create_app() + + if __name__.startswith("bokeh"): - test_jsme_editor_app().servable() + _create_app().servable()