diff --git a/components/content/MolstarPdbePlugin.vue b/components/content/MolstarPdbePlugin.vue index 82705a4adedb7c25aa2ccd81918158d4fe0b6820..a23a23bc993687733cfc327265941677d0764d16 100644 --- a/components/content/MolstarPdbePlugin.vue +++ b/components/content/MolstarPdbePlugin.vue @@ -1,6 +1,13 @@ <script setup lang="ts"> + import { withTrailingSlash, withLeadingSlash, joinURL } from 'ufo' import { useRuntimeConfig, computed } from '#imports' +import * as d3 from "d3"; +import * as Plot from "@observablehq/plot"; + + +import PlotFigure from "~/components/PlotFigure"; + export interface Props { height?: number dataUrls?: string[] @@ -47,8 +54,8 @@ const dialog = ref(false) // const show = ref(false) const computedWidth = computed(() => { - if (width > maxWidth) return maxWidth - return width + // if (toValue(width) > toValue(maxWidth)) return toValue(maxWidth) / 1.5 + return toValue(width) / 1.5 }) const computedHeight = computed(() => { @@ -90,6 +97,45 @@ const pdbeMolstarComponent = ref(null) // const selectedPdb = ref("/wiki/avs/AVAST_I,AVAST_I__Avs1A,0,V-plddts_85.07081.pdb") const selectedPdb = ref(null) +const selectedPaePath = computed(() => { + return selectedPdb.value ? `${selectedPdb.value.split(".").slice(0, -1).join('.')}.tsv` : null +}) + +const paeData = ref([]) +watch(selectedPaePath, async (newPaePath) => { + if (newPaePath !== null) { + const data = await d3.tsv(newPaePath); + // console.log(data) + paeData.value = data + } else { + paeData.value = [] + } + +}) + +const sanitizedPaeData = computed(() => { + return paeData.value.reduce((acc, curr, index) => { + const scoredResidu = index + for (const [alignedResidu, value] of Object.entries(curr)) { + // console.log(value) + acc.push({ alignedResidu: parseInt(alignedResidu), scoredResidu: parseInt(scoredResidu), value: parseFloat(value) }) + } + + return acc + }, []) +}) + + +const plotPaeOptions = computed(() => { + return { + color: { scheme: "Greens", legend: true, reverse: true, label: "Expected position error (Ångströms)" }, + y: { reverse: true }, + marks: [ + Plot.dot(sanitizedPaeData.value, { x: "scoredResidu", y: "alignedResidu", stroke: "value" }) + ] + } +}) + watch(selectedPdb, (selectedPdb, prevSelectPdb) => { if (selectedPdb !== null) { dialog.value = true @@ -130,13 +176,18 @@ watch(selectedPdb, (selectedPdb, prevSelectPdb) => { </v-toolbar> <v-card-text> - - <v-sheet v-if="selectedPdb" - class="d-flex align-center justify-center flex-wrap text-center mx-auto px-4 my-3" - :height="computedHeight" :max-width="maxWidth" :width="computedWidth" position="relative"> - <pdbe-molstar ref="pdbeMolstarComponent" hide-controls="true" landscape="true" :custom-data-url="selectedPdb" alphafold-view="true" - custom-data-format="pdb"></pdbe-molstar> - </v-sheet> + <v-row> + <v-sheet v-if="selectedPdb" + class="d-flex align-center justify-center flex-wrap text-center mx-auto px-4 my-3" + :height="computedHeight" :width="computedWidth" position="relative"> + <pdbe-molstar ref="pdbeMolstarComponent" hide-controls="true" landscape="true" + :custom-data-url="selectedPdb" alphafold-view="true" + custom-data-format="cif"></pdbe-molstar> + </v-sheet> + <v-col v-if="sanitizedPaeData?.length > 0"> + <PlotFigure defer :options="plotPaeOptions"></PlotFigure> + </v-col> + </v-row> </v-card-text> </v-card> diff --git a/components/content/StructureDb.vue b/components/content/StructureDb.vue index a78517a0ec3ce9a3134f5edf04860f4f88a88d9d..93ecadf11a8b1daf17165c4544eeb7da33d2d04f 100644 --- a/components/content/StructureDb.vue +++ b/components/content/StructureDb.vue @@ -33,8 +33,14 @@ const dataTableServerProps = computed(() => { }) function namesToCollapsibleChips(names: string[]) { - return names.filter((it) => it !== "").map(it => ({ title: it })) + return names.filter((it) => it !== "").map(it => ({ title: it.split("__").pop() })) } + +function pdbNameToCif(pdbPath: string) { + const cifPath = pdbPath.split(".").slice(0, -1).join(".") + return `${cifPath}.cif` +} + </script> <template> <ServerDbTable title="Predicted Structures" db="structure" :sortBy="sortBy" :facets="facets" @@ -50,7 +56,7 @@ function namesToCollapsibleChips(names: string[]) { <td :colspan="columns.length"> <v-card flat color="transparent"> <v-card-text> - <MolstarPdbePlugin :data-urls="[`/${item.system.toLowerCase()}/${item.pdb}`]"> + <MolstarPdbePlugin :data-urls="[`/${item.system.toLowerCase()}/${pdbNameToCif(item.pdb)}`]"> </MolstarPdbePlugin> </v-card-text> </v-card>