diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 0000000000000000000000000000000000000000..98d4832a4762bd4cbe321f8cbb413a7003886c13 --- /dev/null +++ b/.prettierignore @@ -0,0 +1,2 @@ +src/gql/codegen +src/assets/theme.css \ No newline at end of file diff --git a/README.md b/README.md index aaa2e843d4c5dbcb69e8d2d55fb7fb473cc78aaf..41f7346a9ea86b53206f8b84365ddb6979fa8f71 100644 --- a/README.md +++ b/README.md @@ -86,10 +86,15 @@ import ... from 'componentLib/some3PComponent' import ... from '@/components/someLocalComponent' import ... from '@/layout/someLayoutComponent' import ... from '@/views/someViewComponent' +/** + * Composables imports + */ +import ... from '@/composables/useSomeComposable' +import { ... } from '@vueuse/core' /** * Other 3rd-party imports */ -import { ... } from '@vueuse/core' // For VueUse e.g. +import { ... } from 'lodash-es' /** * Stores imports */ @@ -97,7 +102,7 @@ import { ... } from '@/stores/someStore' /** * Types imports */ -import { type ... } from 'lib/some3PTypes' +import { type ... } from 'lib/someTypes' import { type ... } from '@/typings/someTypes' /** * Utils imports diff --git a/package-lock.json b/package-lock.json index 88dcb77748fafe58b29b7dfb73a5c04843770dcf..dcd740875af49390748081ee0c4cc92990e25eb4 100644 --- a/package-lock.json +++ b/package-lock.json @@ -5046,16 +5046,18 @@ "dev": true }, "node_modules/@types/lodash": { - "version": "4.14.196", - "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.196.tgz", - "integrity": "sha512-22y3o88f4a94mKljsZcanlNWPzO0uBsBdzLAngf2tp533LzZcQzb6+eZPJ+vCTt+bqF2XnvT9gejTLsAcJAJyQ==", - "dev": true + "version": "4.17.13", + "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.17.13.tgz", + "integrity": "sha512-lfx+dftrEZcdBPczf9d0Qv0x+j/rfNCMuC6OcfXmO8gkfeNAY88PgKUbvG56whcN23gc27yenwF6oJZXGFpYxg==", + "dev": true, + "license": "MIT" }, "node_modules/@types/lodash-es": { "version": "4.17.8", "resolved": "https://registry.npmjs.org/@types/lodash-es/-/lodash-es-4.17.8.tgz", "integrity": "sha512-euY3XQcZmIzSy7YH5+Unb3b2X12Wtk54YWINBvvGQ5SmMvwb11JQskGsfkH/5HXK77Kr8GF0wkVDIxzAisWtog==", "dev": true, + "license": "MIT", "dependencies": { "@types/lodash": "*" } diff --git a/src/assets/icons/alignment.svg b/src/assets/icons/alignment.svg new file mode 100644 index 0000000000000000000000000000000000000000..653e40e9ed5a0a74d99c446ae1c323676645f848 --- /dev/null +++ b/src/assets/icons/alignment.svg @@ -0,0 +1,5 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" viewBox="0 0 512 512"> + <path + d="M0 96c0-17.7 14.3-32 32-32h448c17.7 0 32 14.3 32 32s-14.3 32-32 32H32c-17.7 0-32-14.3-32-32Zm224 320c0-17.7 14.3-32 32-32h224c17.7 0 32 14.3 32 32s-14.3 32-32 32H256c-17.7 0-32-14.3-32-32zM0 416c0-17.7 14.3-32 32-32h98c17.7 0 32 14.3 32 32s-14.3 32-32 32H32c-17.7 0-32-14.3-32-32Zm288-160c0 17.7-14.3 32-32 32H32c-17.7 0-32-14.3-32-32s14.3-32 32-32h224c17.7 0 32 14.3 32 32zm224 0c0 17.7-14.3 32-32 32h-98c-17.7 0-32-14.3-32-32s14.3-32 32-32h98c17.7 0 32 14.3 32 32z" + fill="currentColor" /> +</svg> \ No newline at end of file diff --git a/src/assets/icons/modification.svg b/src/assets/icons/modification.svg index cb196bffed9d9cbdc722e0dc2f892a1cd555e7c6..fd6ff998f1c7b35cf55e14ed161762c07907fc01 100644 --- a/src/assets/icons/modification.svg +++ b/src/assets/icons/modification.svg @@ -1,5 +1,4 @@ <svg width="1em" height="1em" viewBox="0 0 512 512" xmlns="http://www.w3.org/2000/svg"> - <path style="fill:currentColor" d="M56.8 267.5q-17.2-.7-27-8.2-9.9-7.6-13-22.3l-5-24.2 5.9 2H.5l-1.3-6q8.4-6.2 17.5-6.2 6.6 0 10.6 3.7 4.1 3.5 5.4 10.2l2.8 15.4q1.6 9.3 4.4 15 2.8 5.4 7.2 8 4.3 2.5 10.6 3.1h11q4-.3 7-1 3-.8 5.7-2.5 1.4-1.4 3-4.8 1.5-3.4 3-8.2 1.3-4.8 2.3-9.6l3-15.4q1.2-6.7 5.1-10.2 4-3.7 10-3.7 9.2 0 17.5 6.2l-1.2 6h-17.2l5.9-2-5 24q-3.2 14.8-13.2 22.3-10 7.6-27.3 8.4zm-16.3 28.2-.9-4.4q2.8-1.7 6-3.2 3.3-1.7 7.9-3.4l.1 11zm25.7 0 .4-9.9 19.3 2.8-.1 7.1zm-14.5 0v-92h21.2v92zm8.5-82-19.3-2.8.1-7.1h19.6zm10.8 1.1-.2-11h12.7l.9 4.4-6.7 3.5q-3 1.6-6.7 3.1zm60 100.9-5.4-6.5q4.2-3.7 6.7-7.7 2.7-4 2.8-7.7l-7.4-6q.9-5.2 3.8-8.7 3-3.6 7.3-5.9 6.3 2.3 9.3 6.6 3.1 4.2 3.1 9.2 0 5-2.7 10.1-2.6 5.2-7.2 9.5t-10.4 7.1z" diff --git a/src/assets/style.css b/src/assets/style.css index b5c61c956711f981a41e95f7fcf0038436cfbb22..45d56fcc15fde275e16056f29d0eda606d18248e 100644 --- a/src/assets/style.css +++ b/src/assets/style.css @@ -1,3 +1,9 @@ @tailwind base; @tailwind components; @tailwind utilities; + +@layer utilities { + .small-caps { + font-variant: small-caps; + } +} diff --git a/src/components/AlignmentForm.vue b/src/components/AlignmentForm.vue new file mode 100644 index 0000000000000000000000000000000000000000..5aa7d1f32c754d229c22787c476732568aec5dcc --- /dev/null +++ b/src/components/AlignmentForm.vue @@ -0,0 +1,120 @@ +<script setup lang="ts"> +/** + * Vue imports + */ +import { computed, ref } from 'vue' +/** + * Components imports + */ +import AlignmentFormGuide from './AlignmentFormGuide.vue' +import AlignmentFormTarget from './AlignmentFormTarget.vue' +import TabView from 'primevue/tabview' +import TabPanel from 'primevue/tabpanel' +/** + * Types imports + */ +import type { AlignmentModesType } from '@/views/AlignmentView.vue' +import { ALIGNMENT_MODES } from '@/utils/constant' +/** + * Utils imports + */ +import { getColorWithOptionalShade } from '@/utils/colors' + +/** + * Available tabs for type of data to align. + */ +enum TabEnum { + Guide = 0, + Target +} + +/** + * Component props + */ +const props = defineProps<{ + /** The current mode. */ + currentModeModel: AlignmentModesType + /** The selected IDs. */ + selectedIdsModel?: string[] +}>() + +/** + * Component events. + */ +const emit = defineEmits<{ + /** Event used to update the `v-model:currentModeModel` value. */ + 'update:currentModeModel': [currentModeModel: AlignmentModesType] + /** Event used to update the `v-model:selectedIdsModel` value. */ + 'update:selectedIdsModel': [selectedIdsModel?: string[]] +}>() + +/** + * The index of the tab currently selected, reactively updated when changed. + */ +const activeTabIndex = computed<TabEnum>({ + get: () => + ALIGNMENT_MODES.findIndex((mode) => mode === props.currentModeModel), + set: (newTabIndex) => + emit('update:currentModeModel', ALIGNMENT_MODES[newTabIndex]) +}) + +/** + * The IDs of the sequences selected for alignment. + */ +const selectedIds = ref<{ + /** The IDs of the guides selected for alignment. */ + guide: string[] + /** The IDs of the targets selected for alignment. */ + target: string[] +}>({ + guide: [], + target: [] +}) +</script> + +<template> + <TabView + v-model:active-index="activeTabIndex" + :pt="{ + nav: { + style: { + justifyContent: 'center', + marginBottom: '2rem', + fontSize: '1.25em', + background: `linear-gradient(white 0 0) padding-box, + linear-gradient(90deg, + ${getColorWithOptionalShade('slate', '300')}00 10%, + ${getColorWithOptionalShade('slate', '300')} 15%, + ${getColorWithOptionalShade('slate', '300')} 85%, + ${getColorWithOptionalShade('slate', '300')}00 90%) + border-box`, + borderStyle: 'dashed', + borderColor: 'white' + } + } + }" + @tab-change="$emit('update:selectedIdsModel', [])" + > + <TabPanel header="Guide"> + <AlignmentFormGuide + v-model:selected-guide-ids-model="selectedIds.guide" + class="mx-auto rounded-xl border border-slate-300 bg-slate-50 p-8" + @update:selected-guide-ids-model=" + (newSelectedGuideIds) => + $emit('update:selectedIdsModel', newSelectedGuideIds) + " + /> + </TabPanel> + + <TabPanel header="Target"> + <AlignmentFormTarget + v-model:selected-target-ids-model="selectedIds.target" + class="mx-auto rounded-xl border border-slate-300 bg-slate-50 p-8" + @update:selected-target-ids-model=" + (newSelectedTargetIds) => + $emit('update:selectedIdsModel', newSelectedTargetIds) + " + /> + </TabPanel> + </TabView> +</template> diff --git a/src/components/AlignmentFormGuide.vue b/src/components/AlignmentFormGuide.vue new file mode 100644 index 0000000000000000000000000000000000000000..9e01148e1332057e879bf830b72b10ed068b40da --- /dev/null +++ b/src/components/AlignmentFormGuide.vue @@ -0,0 +1,364 @@ +<script setup lang="ts"> +/** + * Vue imports + */ +import { computed, ref, toRef } from 'vue' +/** + * Components imports + */ +import BaseRenderedMarkdown from '@/components/BaseRenderedMarkdown.vue' +import SelectButton from 'primevue/selectbutton' +import Chip from 'primevue/chip' +import MultiSelect from 'primevue/multiselect' +import Dropdown from 'primevue/dropdown' +/** + * Composables imports + */ +import { useQuery } from '@urql/vue' +/** + * Other 3rd-party imports + */ +import { + uniqWith as _uniqWith, + isEqual as _isEqual, + groupBy as _groupBy, + mapValues as _mapValues +} from 'lodash-es' +/** + * Types imports + */ +import type { GuideClass } from '@/gql/codegen/graphql' +/** + * Utils imports + */ +import { guideAlignmentQuery } from '@/gql/queries' + +interface GuideAlignmentSelectionModel { + /** Currently selected guide subclasses. */ + guideSubclasses: GuideClass[] + /** Currently selected guide name. */ + guideName: string | undefined + /** Currently selected organism IDs.*/ + organismIds: number[] + /** Currently selected guide IDs for alignment. */ + guideIds: string[] +} + +/** + * Component props + */ +defineProps<{ + /** The currently selected guide IDs for alignment. */ + selectedGuideIdsModel?: string[] +}>() + +/** + * Component events. + */ +const emit = defineEmits<{ + /** Event used to update the `v-model:selectedGuideIdsModel` value. */ + 'update:selectedGuideIdsModel': [selectedGuideIdsModel?: string[]] +}>() + +/** + * Current selection of parameters/filters for the guides to pick. + */ +const selection = ref<GuideAlignmentSelectionModel>({ + guideSubclasses: [], + guideName: undefined, + organismIds: [], + guideIds: [] +}) + +/** + * Reactive urql GraphQL query object, updated with query state & response. + */ +const gqlQuery = useQuery({ + query: guideAlignmentQuery, + variables: toRef(() => ({ + guideSubclasses: selection.value.guideSubclasses.length + ? selection.value.guideSubclasses + : undefined, + guideName: selection.value.guideName || '', + organismIds: selection.value.organismIds.length + ? selection.value.organismIds + : undefined + })) +}) + +/** + * Available options for guide subclass selection, independently of other fields' + * selection. + */ +const guideSubclassOptions = computed(() => + _uniqWith( + gqlQuery.data.value?.guideBase, + (guideA, guideB) => guideA.subclass === guideB.subclass + ).map((guide) => ({ + label: guide.subclass_label, + value: guide.subclass + })) +) + +/** + * Available options for guide name selection, independently of other fields' + * selection. + */ +const guideNameOptionsBase = computed(() => + _uniqWith( + gqlQuery.data.value?.guideBase, + (guideA, guideB) => guideA.name === guideB.name + ).map((guide) => ({ + label: guide.name, + value: guide.name + })) +) + +/** + * Available guide names, filtered based on other fields' selection (only guide + * names which exists with selected options are present). + */ +const guideNamesFiltered = computed(() => + _uniqWith( + gqlQuery.data.value?.guideNamesFilteredBySubclass, + (guideA, guideB) => guideA.name === guideB.name + ).map((guide) => guide.name) +) + +/** + * Available options for guide name selection, with a `isDisabled` field on + * absence in `guideNamesFiltered`. + */ +const guideNameOptionsWithDisabling = computed(() => + guideNameOptionsBase.value.map((guideNameOption) => ({ + ...guideNameOption, + isDisabled: !guideNamesFiltered.value.some( + (guideNameFiltered) => guideNameFiltered === guideNameOption.value + ) + })) +) + +/** + * Available options for organism ID selection, independently of other fields' + * selection. + */ +const organismIdOptionsBase = computed(() => + _uniqWith(gqlQuery.data.value?.organismsBase, _isEqual).map((organism) => ({ + label: organism.label, + value: organism.id + })) +) + +/** + * For each guide name, an array of the IDs of the organisms which have at + * least a guide of this name registered. + */ +const organismIdsByGuideNameBase = computed(() => + _mapValues( + _groupBy(_uniqWith(gqlQuery.data.value?.guideBase, _isEqual), 'name'), + (guides) => guides.map((guide) => guide.genome?.organism?.id) + ) +) + +/** + * Available organism IDs, filtered based on other fields' selection (only + * organism IDs which exists with selected options are present). + */ +const organismIdsFiltered = computed( + () => + (selection.value.guideName && + organismIdsByGuideNameBase.value[selection.value.guideName]) || + [] +) + +/** + * Available options for organism ID selection, with a `isDisabled` field on + * absence in `organismIdsFiltered`. + */ +const organismIdOptionsWithDisabling = computed(() => + organismIdOptionsBase.value.map((organismIdOption) => ({ + ...organismIdOption, + isDisabled: !organismIdsFiltered.value.some( + (organismIdFiltered) => organismIdFiltered === organismIdOption.value + ) + })) +) + +/** + * Available options for guide ID selection for alignment given the other + * fields selection. + */ +const guideIdOptions = computed(() => + gqlQuery.data.value?.selectableGuides.map((guide) => ({ + label: guide.id, + optionLabel: `\`${guide.id}\` • ${guide.genome?.organism?.shortlabel}`, + value: guide.id + })) +) + +/** + * Clear the selected guide IDs. + */ +const clearGuideIds = () => { + emit('update:selectedGuideIdsModel', (selection.value.guideIds = [])) +} + +/** + * Remove an organism ID from the selected ones & clear the selected guide IDs. + */ +const removeOrganismAndClearGuideIds = (organismId: number) => { + const organismIdIndex = selection.value.organismIds.findIndex( + (currOrganismId) => currOrganismId === organismId + ) + organismIdIndex !== -1 && + selection.value.organismIds.splice(organismIdIndex, 1) + clearGuideIds() +} + +/** + * Clear the selected organisms & guide IDs. + */ +const clearOrganismsAndGuideIds = () => { + selection.value.organismIds = selection.value.organismIds.filter( + (organismId) => + selection.value.guideName && + organismIdsByGuideNameBase.value[selection.value.guideName]?.includes( + organismId + ) + ) + clearGuideIds() +} + +/** + * Clear the selection except the selected guide subclass. + */ +const clearSelectionExceptSubclass = () => { + selection.value.guideName = undefined + clearOrganismsAndGuideIds() +} +</script> + +<template> + <div class="flex max-w-max flex-wrap gap-8"> + <div class="flex flex-col gap-2"> + <h3 class="text-lg font-bold text-slate-700">Guide subclasses</h3> + <SelectButton + v-model="selection.guideSubclasses" + :options="guideSubclassOptions" + option-label="label" + option-value="value" + option-disabled="isDisabled" + multiple + @change="clearSelectionExceptSubclass" + > + <template #option="{ option }"> + <BaseRenderedMarkdown + :stringified-markdown="option.label" + class="font-medium" + /> + </template> + </SelectButton> + </div> + + <div class="flex max-w-min flex-col gap-2"> + <h3 class="text-lg font-bold text-slate-700">Guides</h3> + <Dropdown + v-model="selection.guideName" + :options="guideNameOptionsWithDisabling" + option-label="label" + option-value="value" + option-disabled="isDisabled" + placeholder="Select a guide..." + filter + @change="clearOrganismsAndGuideIds" + /> + </div> + + <div + v-tooltip.bottom=" + !selection.guideName && { + value: 'First select a guide to list its organisms.', + autoHide: false, + pt: { text: { style: { textAlign: 'center' } } } + } + " + class="flex max-w-min flex-col gap-2" + > + <h3 class="text-lg font-bold text-slate-700">Organisms</h3> + <MultiSelect + v-model="selection.organismIds" + :options="organismIdOptionsWithDisabling" + option-label="label" + option-value="value" + option-disabled="isDisabled" + multiple + placeholder="Select organisms..." + :max-selected-labels="3" + display="chip" + filter + :disabled="!selection.guideName" + @change="clearGuideIds" + > + <template #value> + <Chip + v-for="organismId in selection.organismIds.slice(0, 3)" + :key="organismId" + class="mr-1" + removable + @remove.stop="removeOrganismAndClearGuideIds(organismId)" + > + <BaseRenderedMarkdown + :stringified-markdown=" + organismIdOptionsBase.find( + (organismIdOption) => organismIdOption.value === organismId + )?.label || '' + " + class="my-1.5" + /> + </Chip> + <template v-if="selection.organismIds.length > 3"> + +{{ selection.organismIds.length - 3 }} others... + </template> + </template> + <template #option="{ option }"> + <BaseRenderedMarkdown :stringified-markdown="option.label" /> + </template> + </MultiSelect> + </div> + + <div + v-tooltip.bottom=" + !selection.guideName && { + value: 'First select a guide to list its sequences.', + autoHide: false, + pt: { text: { style: { textAlign: 'center' } } } + } + " + class="flex max-w-min flex-col gap-2" + > + <h3 class="text-lg font-bold text-slate-700">Sequences</h3> + <MultiSelect + v-model="selection.guideIds" + :options="guideIdOptions" + option-label="label" + option-value="value" + multiple + placeholder="Select sequences..." + :max-selected-labels="3" + filter + :disabled="!selection.guideName" + empty-message="First select a guide to list its sequences" + @update:model-value="(selectedGuideIdsModel: string[]) => $emit('update:selectedGuideIdsModel', selectedGuideIdsModel)" + > + <template #option="{ option }"> + <BaseRenderedMarkdown :stringified-markdown="option.optionLabel" /> + </template> + <template #value="{ value }"> + <span v-if="value.length" class="font-mono"> + {{ value.join(', ') }} + </span> + </template> + </MultiSelect> + </div> + </div> +</template> diff --git a/src/components/AlignmentFormTarget.vue b/src/components/AlignmentFormTarget.vue new file mode 100644 index 0000000000000000000000000000000000000000..8f5b96271cfb6262aa2df21b842838114fd96f3d --- /dev/null +++ b/src/components/AlignmentFormTarget.vue @@ -0,0 +1,301 @@ +<script setup lang="ts"> +/** + * Vue imports + */ +import { computed, ref, toRef } from 'vue' +/** + * Components imports + */ +import BaseRenderedMarkdown from '@/components/BaseRenderedMarkdown.vue' +import Chip from 'primevue/chip' +import MultiSelect from 'primevue/multiselect' +import Dropdown from 'primevue/dropdown' +/** + * Composables imports + */ +import { useQuery } from '@urql/vue' +/** + * Other 3rd-party imports + */ +import { + uniqWith as _uniqWith, + isEqual as _isEqual, + groupBy as _groupBy, + mapValues as _mapValues +} from 'lodash-es' +/** + * Utils imports + */ +import { targetAlignmentQuery } from '@/gql/queries' + +interface TargetAlignmentSelectionModel { + /** Currently selected target name. */ + targetName: string | undefined + /** Currently selected organism IDs.*/ + organismIds: number[] + /** Currently selected target IDs for alignment. */ + targetIds: string[] +} + +/** + * Component props + */ +defineProps<{ + /** The currently selected target IDs for alignment. */ + selectedTargetIdsModel?: string[] +}>() + +/** + * Component events. + */ +const emit = defineEmits<{ + /** Event used to update the `v-model:selectedTargetIdsModel` value. */ + 'update:selectedTargetIdsModel': [selectedTargetIdsModel?: string[]] +}>() + +/** + * Current selection of parameters/filters for the target sequences to pick. + */ +const selection = ref<TargetAlignmentSelectionModel>({ + targetName: undefined, + organismIds: [], + targetIds: [] +}) + +/** + * Reactive urql GraphQL query object, updated with query state & response. + */ +const gqlQuery = useQuery({ + query: targetAlignmentQuery, + variables: toRef(() => ({ + targetName: selection.value.targetName || '', + organismIds: selection.value.organismIds.length + ? selection.value.organismIds + : undefined + })) +}) + +/** + * Available options for target name selection, independently of other fields' + * selection. + */ +const targetNameOptionsGroups = computed(() => + Object.entries( + _groupBy( + _uniqWith( + gqlQuery.data.value?.targetBase, + (targetA, targetB) => targetA.name === targetB.name + ).map((target) => ({ + label: target.name, + value: target.name, + groupLabel: target.unit || 'Other' + })), + 'groupLabel' + ) + ).map(([targetUnit, targets]) => ({ label: targetUnit, children: targets })) +) + +/** + * Available options for organism ID selection, independently of other fields' + * selection. + */ +const organismIdOptionsBase = computed(() => + _uniqWith(gqlQuery.data.value?.organismsBase, _isEqual).map((organism) => ({ + label: organism.label, + value: organism.id + })) +) + +/** + * For each target name, an array of the IDs of the organisms which have at + * least a target of this name registered. + */ +const organismIdsByTargetNameBase = computed(() => + _mapValues( + _groupBy(_uniqWith(gqlQuery.data.value?.targetBase, _isEqual), 'name'), + (targets) => targets.map((target) => target.genome?.organism?.id) + ) +) + +/** + * Available organism IDs, filtered based on other fields' selection (only + * organism IDs which exists with selected options are present). + */ +const organismIdsFiltered = computed( + () => + (selection.value.targetName && + organismIdsByTargetNameBase.value[selection.value.targetName]) || + [] +) + +/** + * Available options for organism ID selection, with a `isDisabled` field on + * absence in `organismIdsFiltered`. + */ +const organismIdOptionsWithDisabling = computed(() => + organismIdOptionsBase.value.map((organismIdOption) => ({ + ...organismIdOption, + isDisabled: !organismIdsFiltered.value.some( + (organismIdFiltered) => organismIdFiltered === organismIdOption.value + ) + })) +) + +/** + * Available options for target ID selection for alignment given the other + * fields selection. + */ +const targetIdOptions = computed(() => + gqlQuery.data.value?.selectableTargets.map((target) => ({ + label: target.id, + optionLabel: `\`${target.id}\` • ${target.genome?.organism?.shortlabel}`, + value: target.id + })) +) + +/** + * Clear the selected target IDs. + */ +const clearTargetIds = () => { + emit('update:selectedTargetIdsModel', (selection.value.targetIds = [])) +} + +/** + * Remove an organism ID from the selected ones & clear the selected target IDs. + */ +const removeOrganismAndClearTargetIds = (organismId: number) => { + const organismIdIndex = selection.value.organismIds.findIndex( + (currOrganismId) => currOrganismId === organismId + ) + organismIdIndex !== -1 && + selection.value.organismIds.splice(organismIdIndex, 1) + clearTargetIds() +} + +/** + * Clear the selected organisms & target IDs. + */ +const clearOrganismsAndTargetIds = () => { + selection.value.organismIds = selection.value.organismIds.filter( + (organismId) => + selection.value.targetName && + organismIdsByTargetNameBase.value[selection.value.targetName]?.includes( + organismId + ) + ) + clearTargetIds() +} +</script> + +<template> + <div class="flex max-w-max flex-wrap gap-8"> + <div class="flex max-w-min flex-col gap-2"> + <h3 class="text-lg font-bold text-slate-700">Targets</h3> + <Dropdown + v-model="selection.targetName" + :options="targetNameOptionsGroups" + option-label="label" + option-value="value" + option-group-label="label" + option-group-children="children" + placeholder="Select a target..." + filter + :pt="{ + item: { + style: { + marginLeft: '1rem' + } + } + }" + @change="clearOrganismsAndTargetIds" + /> + </div> + + <div + v-tooltip.bottom=" + !selection.targetName && { + value: 'First select a target to list its organisms.', + autoHide: false, + pt: { text: { style: { textAlign: 'center' } } } + } + " + class="flex max-w-min flex-col gap-2" + > + <h3 class="text-lg font-bold text-slate-700">Organisms</h3> + <MultiSelect + v-model="selection.organismIds" + :options="organismIdOptionsWithDisabling" + option-label="label" + option-value="value" + option-disabled="isDisabled" + multiple + placeholder="Select organisms..." + :max-selected-labels="3" + display="chip" + filter + :disabled="!selection.targetName" + @change="clearTargetIds" + > + <template #value> + <Chip + v-for="organismId in selection.organismIds.slice(0, 3)" + :key="organismId" + class="mr-1" + removable + @remove.stop="removeOrganismAndClearTargetIds(organismId)" + > + <BaseRenderedMarkdown + :stringified-markdown=" + organismIdOptionsBase.find( + (organismIdOption) => organismIdOption.value === organismId + )?.label || '' + " + class="my-1.5" + /> + </Chip> + <template v-if="selection.organismIds.length > 3"> + +{{ selection.organismIds.length - 3 }} others... + </template> + </template> + <template #option="{ option }"> + <BaseRenderedMarkdown :stringified-markdown="option.label" /> + </template> + </MultiSelect> + </div> + + <div + v-tooltip.bottom=" + !selection.targetName && { + value: 'First select a target to list its sequences.', + autoHide: false, + pt: { text: { style: { textAlign: 'center' } } } + } + " + class="flex max-w-min flex-col gap-2" + > + <h3 class="text-lg font-bold text-slate-700">Sequences</h3> + <MultiSelect + v-model="selection.targetIds" + :options="targetIdOptions" + option-label="label" + option-value="value" + multiple + placeholder="Select sequences..." + :max-selected-labels="3" + filter + :disabled="!selection.targetName" + empty-message="First select a target to list its sequences" + @update:model-value="(selectedTargetIds: string[]) => $emit('update:selectedTargetIdsModel', selectedTargetIds)" + > + <template #option="{ option }"> + <BaseRenderedMarkdown :stringified-markdown="option.optionLabel" /> + </template> + <template #value="{ value }"> + <span v-if="value.length" class="font-mono"> + {{ value.join(', ') }} + </span> + </template> + </MultiSelect> + </div> + </div> +</template> diff --git a/src/components/BaseLegendButtonOverlay.vue b/src/components/BaseLegendButtonOverlay.vue index 68f0f79a1bf4dd8120e1d958e81e2e92e7a8bb27..058299341d2aa98d4408c29cfebbd0bcbb2e6ead 100644 --- a/src/components/BaseLegendButtonOverlay.vue +++ b/src/components/BaseLegendButtonOverlay.vue @@ -4,7 +4,7 @@ */ import { ref } from 'vue' /** - * Component imports + * Components imports */ import Button from 'primevue/button' import OverlayPanel from 'primevue/overlaypanel' diff --git a/src/components/BaseLockableTooltip.vue b/src/components/BaseLockableTooltip.vue index 52fb6f14caa4c73905e6fe597988f8af31dfc19a..d7d5c0f88dba2883f89e81efcdd53b35be37be01 100644 --- a/src/components/BaseLockableTooltip.vue +++ b/src/components/BaseLockableTooltip.vue @@ -8,7 +8,7 @@ import { computed, ref } from 'vue' */ import BaseTooltip from '@/components/BaseTooltip.vue' /** - * Other 3rd-party imports + * Composables imports */ import { useMouse, onClickOutside } from '@vueuse/core' diff --git a/src/components/ChromosomeMagnify.vue b/src/components/ChromosomeMagnify.vue index a9538657e3e4c51d3cc51b007343c09362b96ea8..300add17373f65a9f317061af39b9ef823b49248 100644 --- a/src/components/ChromosomeMagnify.vue +++ b/src/components/ChromosomeMagnify.vue @@ -8,9 +8,12 @@ import { computed, onMounted, ref } from 'vue' */ import BaseLockableTooltip from '@/components/BaseLockableTooltip.vue' /** - * Other 3rd-party imports + * Composables imports */ import { useMouseInElement } from '@vueuse/core' +/** + * Other 3rd-party imports + */ import { G, Rect, SVG, type Svg } from '@svgdotjs/svg.js' /** * Types imports diff --git a/src/components/HelpDialog.vue b/src/components/HelpDialog.vue index 225a44c8416be166b50d6dc634cf6e0b090eb63d..1254b4f7bee309022f1392500f5f77d166578343 100644 --- a/src/components/HelpDialog.vue +++ b/src/components/HelpDialog.vue @@ -4,7 +4,7 @@ */ import { ref, watchEffect } from 'vue' /** - * Component imports + * Components imports */ import Dialog from 'primevue/dialog' import Carousel from 'primevue/carousel' diff --git a/src/components/InteractionCard.vue b/src/components/InteractionCard.vue index 17dd546847cd57db689774a1d3493862c769437c..757548bb4608cf8731d7cb2df0255ccc018d3f0f 100644 --- a/src/components/InteractionCard.vue +++ b/src/components/InteractionCard.vue @@ -142,9 +142,9 @@ const HACAFormattedInteraction = computed<InteractionHACAModel | undefined>( () => { if (props.interaction?.modification.type === ModifType.Psi) { const duplexes = { - before: props.interaction.duplexes.find((duplex) => duplex.index === 1), - middle: props.interaction.duplexes.find((duplex) => duplex.index === 2), - after: props.interaction.duplexes.find((duplex) => duplex.index === 3) + before: props.interaction.duplexes.find((duplex) => duplex.index === 0), + middle: props.interaction.duplexes.find((duplex) => duplex.index === 1), + after: props.interaction.duplexes.find((duplex) => duplex.index === 2) } if ( diff --git a/src/components/InteractionCardCD.vue b/src/components/InteractionCardCD.vue index 01f46763033bfa378ed1f787f227ebedaa6478fc..3d445222a561829828fbe9e2d591233ba25a62ad 100644 --- a/src/components/InteractionCardCD.vue +++ b/src/components/InteractionCardCD.vue @@ -223,7 +223,9 @@ const bondsText = computed(() => const basePair = nucleotide + paddedFragments.value.target[nucleotidePositionInDuplex] || '' - return /GC|CG|AU|UA/.test(basePair) + return nucleotidePositionInDuplex === modificationPositionInDuplex.value + ? ' ' + : /GC|CG|AU|UA/.test(basePair) ? '|' : /GU|UG/.test(basePair) && nucleotidePositionInDuplex > firstBondPositionInDuplex.value && diff --git a/src/components/InteractionCardHACA.vue b/src/components/InteractionCardHACA.vue index 9d5046ba9309b3fe8221ec21955eb04c73b05612..35b93e0fd30d5e32577ab53d3929812c5f1fe4ca 100644 --- a/src/components/InteractionCardHACA.vue +++ b/src/components/InteractionCardHACA.vue @@ -9,7 +9,8 @@ import { ref, computed } from 'vue' import { isEqual as _isEqual, reverse as _reverse, - findLastIndex as _findLastIndex + findLastIndex as _findLastIndex, + mapValues as _mapValues } from 'lodash-es' /** * Types imports @@ -349,45 +350,42 @@ const lastBondIndex = computed(() => * Texts representing the bonds w/ their nature */ const bondsTexts = computed(() => - Object.entries({ - before: [ - splitPaddedFragments.value.guide.first.before, - splitPaddedFragments.value.target.before - ], - middle: [ - splitPaddedFragments.value.guide.first.middle, - _reverse(splitPaddedFragments.value.guide.second.middle.split('')).join( - '' - ) - ], - after: [ - splitPaddedFragments.value.guide.second.after, - splitPaddedFragments.value.target.after - ] - }).reduce<{ [duplexId: string]: string }>( - (bondsTexts, [duplexId, fragmentPair]) => ({ - ...bondsTexts, - [duplexId]: - fragmentPair[0] - ?.split('') - .map((nucleotide, inDuplexNucleotideIndex) => { - const basePair = - nucleotide + fragmentPair[1]?.[inDuplexNucleotideIndex] || '' - return /GC|CG|AU|UA/.test(basePair) - ? '|' - : /GU|UG/.test(basePair) && - inDuplexNucleotideIndex > firstBondIndex.value && - inDuplexNucleotideIndex < lastBondIndex.value - ? '∙' - : /GA|AG/.test(basePair) && - inDuplexNucleotideIndex > firstBondIndex.value && - inDuplexNucleotideIndex < lastBondIndex.value - ? '○' - : ' ' - }) - .join('') || '' - }), - {} + _mapValues( + { + before: [ + splitPaddedFragments.value.guide.first.before, + splitPaddedFragments.value.target.before + ], + middle: [ + splitPaddedFragments.value.guide.first.middle, + _reverse(splitPaddedFragments.value.guide.second.middle.split('')).join( + '' + ) + ], + after: [ + splitPaddedFragments.value.guide.second.after, + splitPaddedFragments.value.target.after + ] + }, + (fragmentPair) => + fragmentPair[0] + ?.split('') + .map((nucleotide, inDuplexNucleotideIndex) => { + const basePair = + nucleotide + fragmentPair[1]?.[inDuplexNucleotideIndex] || '' + return /GC|CG|AU|UA/.test(basePair) + ? '|' + : /GU|UG/.test(basePair) && + inDuplexNucleotideIndex > firstBondIndex.value && + inDuplexNucleotideIndex < lastBondIndex.value + ? '∙' + : /GA|AG/.test(basePair) && + inDuplexNucleotideIndex > firstBondIndex.value && + inDuplexNucleotideIndex < lastBondIndex.value + ? '○' + : ' ' + }) + .join('') || '' ) ) diff --git a/src/components/KaryotypeBoard.vue b/src/components/KaryotypeBoard.vue index 3eecb1199eb97ba399de2406a2c23ef82cac0ded..e7aa613efa7ce8a31c48147adc3e3517ea2d2019 100644 --- a/src/components/KaryotypeBoard.vue +++ b/src/components/KaryotypeBoard.vue @@ -9,12 +9,15 @@ import { computed, onMounted, ref } from 'vue' import Button from 'primevue/button' import IconTablerCircleArrowLeft from '~icons/tabler/circle-arrow-left' import IconTablerCircleArrowLeftFilled from '~icons/tabler/circle-arrow-left-filled' +/** + * Composables imports + */ +import { useElementBounding, useWindowScroll } from '@vueuse/core' /** * Other 3rd-party imports */ import { SVG, Svg } from '@svgdotjs/svg.js' import { find as _find } from 'lodash-es' -import { useElementBounding, useWindowScroll } from '@vueuse/core' /** * Types imports */ @@ -117,7 +120,10 @@ const expandedChromosomeLength = computed(() => */ const longestChromosome = computed(() => props.chromosomes.reduce( - (longestChromosome, currChromosome) => (longestChromosome && currChromosome.length > longestChromosome.length ? currChromosome : longestChromosome), + (longestChromosome, currChromosome) => + longestChromosome && currChromosome.length > longestChromosome.length + ? currChromosome + : longestChromosome, props.chromosomes[0] ) ) @@ -129,7 +135,7 @@ const objectsByChromosome = computed(() => props.chromosomes.reduce<{ [k: string]: GenomeObjectModel[] }>( (objectsByChromosome, currChromosome) => ({ ...objectsByChromosome, - [currChromosome.id] : (props.objects || []).filter( + [currChromosome.id]: (props.objects || []).filter( (object) => object.chromosomeId === currChromosome.id ) }), diff --git a/src/components/SecondaryStructure.vue b/src/components/SecondaryStructure.vue index d6f1490415c5d671a391e34c6778def16e827055..e3076650e1a336ea033d7a8e8d4581df16813064 100644 --- a/src/components/SecondaryStructure.vue +++ b/src/components/SecondaryStructure.vue @@ -19,10 +19,13 @@ import IconFa6SolidCropSimple from '~icons/fa6-solid/crop-simple' import IconFa6SolidExpand from '~icons/fa6-solid/expand' import IconFa6SolidDownload from '~icons/fa6-solid/download' /** - * Other3rd-party imports + * Composables imports */ -import G6 from '@antv/g6' import { useElementSize, useResizeObserver } from '@vueuse/core' +/** + * Other 3rd-party imports + */ +import G6 from '@antv/g6' import { find as _find } from 'lodash-es' import mime from 'mime' import FileSaver from 'file-saver' diff --git a/src/components/SelectionForm.vue b/src/components/SelectionForm.vue new file mode 100644 index 0000000000000000000000000000000000000000..4738b92058be52565b532f8ec6716d5994a7f871 --- /dev/null +++ b/src/components/SelectionForm.vue @@ -0,0 +1,166 @@ +<script setup lang="ts"> +/** + * Vue imports + */ +import { computed, ref, watch } from 'vue' +/** + * Components imports + */ +import SelectionFormGuide from './SelectionFormGuide.vue' +import SelectionFormTarget from './SelectionFormTarget.vue' +import SelectionFormModification from './SelectionFormModification.vue' +import TabView from 'primevue/tabview' +import TabPanel from 'primevue/tabpanel' +/** + * Types imports + */ +import type { SelectionModesType } from '@/views/SelectionView.vue' +import type { GuideSelectionModel } from './SelectionFormGuide.vue' +import type { TargetSelectionModel } from './SelectionFormTarget.vue' +import type { ModificationSelectionModel } from './SelectionFormModification.vue' +import { SELECTION_MODES } from '@/utils/constant' +/** + * Utils imports + */ +import { getColorWithOptionalShade } from '@/utils/colors' + +/** + * A selection for the different modes. + */ +export interface SelectionModel { + /** The selection of the modification. */ + modification?: ModificationSelectionModel + /** The selection of the guide. */ + guide?: GuideSelectionModel + /** The selection of the target. */ + target?: TargetSelectionModel +} + +/** + * Available tabs for type of data to select. + */ +enum TabEnum { + Modification = 0, + Guide, + Target +} + +/** + * Component props + */ +const props = defineProps<{ + /** The current mode. */ + currentModeModel: SelectionModesType + /** The current selection in the different modes (v-model value). */ + selectionModel: SelectionModel + /** The ID of the target which matches the selection if it is unique (v-model + * value). */ + onlyTargetIdByModeModel: { modification?: string; target?: string } +}>() + +/** + * Component events. + */ +const emit = defineEmits<{ + /** Event used to update the `v-model:currentModeModel` value. */ + 'update:currentModeModel': [currentModeModel: SelectionModesType] + /** Event used to update the `v-model:selectionModel` value. */ + 'update:selectionModel': [selectionModel: SelectionModel] + /** Event used to update the `v-model:onlyTargetIdByModeModel` value. */ + 'update:onlyTargetIdByModeModel': [ + onlyTargetIdByModeModel: { modification?: string; target?: string } + ] +}>() + +/** + * The index of the tab currently selected, reactively updated when changed. + */ +const activeTabIndex = computed<TabEnum>({ + get: () => + SELECTION_MODES.findIndex((mode) => mode === props.currentModeModel), + set: (newTabIndex) => + emit('update:currentModeModel', SELECTION_MODES[newTabIndex]) +}) + +/** + * The current selection in the different modes (local value). + */ +const selection = ref<SelectionModel>({ + modification: undefined, + guide: undefined, + target: undefined +}) + +/** + * The ID of the target which matches the selection if it is unique (i.e. a + * single target name & a single species is selected), `undefined` otherwise. + * One target ID is used for each mode able to provide one (local value). + */ +const onlyTargetIdByMode = ref<{ modification?: string; target?: string }>({ + modification: undefined, + target: undefined +}) + +/** + * Watchers to update the `v-model` value when modified locally. + */ +watch( + selection, + () => { + emit('update:selectionModel', selection.value) + }, + { deep: true } +) +watch( + onlyTargetIdByMode, + () => { + emit('update:onlyTargetIdByModeModel', onlyTargetIdByMode.value) + }, + { deep: true } +) +</script> + +<template> + <TabView + v-model:active-index="activeTabIndex" + :pt="{ + nav: { + style: { + justifyContent: 'center', + marginBottom: '2rem', + fontSize: '1.25em', + background: `linear-gradient(white 0 0) padding-box, + linear-gradient(90deg, + ${getColorWithOptionalShade('slate', '300')}00 10%, + ${getColorWithOptionalShade('slate', '300')} 15%, + ${getColorWithOptionalShade('slate', '300')} 85%, + ${getColorWithOptionalShade('slate', '300')}00 90%) + border-box`, + borderStyle: 'dashed', + borderColor: 'white' + } + } + }" + > + <TabPanel header="Modification"> + <SelectionFormModification + v-model:selection-model="selection.modification" + v-model:only-target-id="onlyTargetIdByMode.modification" + class="mx-auto rounded-xl border border-slate-300 bg-slate-50 p-8" + /> + </TabPanel> + <TabPanel header="Guide"> + <SelectionFormGuide + v-model:selection-model="selection.guide" + class="mx-auto rounded-xl border border-slate-300 bg-slate-50 p-8" + /> + </TabPanel> + <TabPanel header="Target"> + <SelectionFormTarget + v-model:selection-model="selection.target" + v-model:only-target-id="onlyTargetIdByMode.target" + class="mx-auto rounded-xl border border-slate-300 bg-slate-50 p-8" + /> + </TabPanel> + </TabView> +</template> diff --git a/src/components/GuideSelectionForm.vue b/src/components/SelectionFormGuide.vue similarity index 89% rename from src/components/GuideSelectionForm.vue rename to src/components/SelectionFormGuide.vue index 061f1795a9ac70d694ccc01e3a1299c92eadfcbb..91b30aeb8410f37965de545885db9e1cdca524df 100644 --- a/src/components/GuideSelectionForm.vue +++ b/src/components/SelectionFormGuide.vue @@ -4,7 +4,7 @@ */ import { computed, ref, toRef } from 'vue' /** - * Component imports + * Components imports */ import BaseRenderedMarkdown from './BaseRenderedMarkdown.vue' import Chip from 'primevue/chip' @@ -12,14 +12,18 @@ import SelectButton from 'primevue/selectbutton' import MultiSelect from 'primevue/multiselect' import { GuideClass, ModifType } from '@/gql/codegen/graphql' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' +/** + * Other 3rd-party imports + */ import { uniqWith as _uniqWith, isEqual as _isEqual } from 'lodash-es' /** * Utils imports */ import { guideSelectionQuery } from '@/gql/queries' +import { isNonNullish } from '@/typings/typeUtils' /** * A guide selection. @@ -27,6 +31,8 @@ import { guideSelectionQuery } from '@/gql/queries' export interface GuideSelectionModel { /** Currently selected guide subclasses. */ guideSubclasses: GuideClass[] + /** Currently selected guide subclass labels. */ + guideSubclassLabels: string[] /** Currently selected target names. */ targetNames: string[] /** Currently selected modification types. */ @@ -56,6 +62,7 @@ defineEmits<{ */ const selection = ref<GuideSelectionModel>({ guideSubclasses: [], + guideSubclassLabels: [], targetNames: [], modificationTypes: [], organismIds: [] @@ -67,7 +74,7 @@ const selection = ref<GuideSelectionModel>({ const gqlQuery = useQuery({ query: guideSelectionQuery, variables: toRef(() => ({ - guideSubclasses: selection.value.guideSubclasses?.length + guideSubclass: selection.value.guideSubclasses?.length ? selection.value.guideSubclasses : undefined, targetNames: selection.value.targetNames?.length @@ -93,6 +100,21 @@ const guideSubclassOptionsBase = computed(() => })) ) +/** + * The guide subclasses, associated to their labels. + */ +const guideSubclassLabelBySubclass = computed<{ + [guideSubclass: string]: string +}>(() => + guideSubclassOptionsBase.value.reduce( + (guideSubclassLabelBySubclass, guideSubclassOption) => ({ + ...guideSubclassLabelBySubclass, + [guideSubclassOption.value]: guideSubclassOption.label + }), + {} + ) +) + /** * Available guide subclasses, filtered based on other fields' selection (only guide * subclasses which exists with selected options are present). @@ -269,7 +291,7 @@ const organismIdOptionsWithDisabling = computed(() => <template> <div - class="grid grid-cols-[max-content_1fr] grid-rows-4 items-center gap-x-16 gap-y-8" + class="grid max-w-max grid-cols-[max-content_1fr] grid-rows-4 items-center gap-x-16 gap-y-8" > <h3 class="text-lg font-bold">Guide subclasses</h3> <SelectButton @@ -280,7 +302,20 @@ const organismIdOptionsWithDisabling = computed(() => option-disabled="isDisabled" multiple class="mr-auto" - @update:model-value="(selectedGuideSubclasses: GuideClass[]) => $emit('update:selectionModel', {...selection, guideSubclasses:selectedGuideSubclasses })" + @update:model-value="(selectedGuideSubclasses: GuideClass[]) => + { + selection.guideSubclassLabels = selectedGuideSubclasses + .map((guideSubclass) => + guideSubclassLabelBySubclass[guideSubclass] + ) + .filter(isNonNullish) + $emit('update:selectionModel', { + ...selection, + guideSubclasses: selectedGuideSubclasses, + guideSubclassLabels: selection.guideSubclassLabels + }) + } + " > <template #option="{ option }"> <BaseRenderedMarkdown @@ -305,7 +340,12 @@ const organismIdOptionsWithDisabling = computed(() => :max-selected-labels="3" display="chip" filter - @update:model-value="(selectedTargetNames: string[]) => $emit('update:selectionModel', {...selection, targetNames:selectedTargetNames })" + @update:model-value="(selectedTargetNames: string[]) => + $emit('update:selectionModel', { + ...selection, + targetNames: selectedTargetNames + }) + " > <template #value> <Chip diff --git a/src/components/ModificationSelectionForm.vue b/src/components/SelectionFormModification.vue similarity index 98% rename from src/components/ModificationSelectionForm.vue rename to src/components/SelectionFormModification.vue index 786ea2137026736a93b67adbbf33e5e51a100566..c0a2344c95fee0c7565ec5148dd1a7947a401618 100644 --- a/src/components/ModificationSelectionForm.vue +++ b/src/components/SelectionFormModification.vue @@ -4,7 +4,7 @@ */ import { computed, ref, toRef, watchEffect } from 'vue' /** - * Component imports + * Components imports */ import BaseRenderedMarkdown from './BaseRenderedMarkdown.vue' import Chip from 'primevue/chip' @@ -12,9 +12,12 @@ import SelectButton from 'primevue/selectbutton' import MultiSelect from 'primevue/multiselect' import { ModifType } from '@/gql/codegen/graphql' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' +/** + * Other 3rd-party imports + */ import { uniqWith as _uniqWith, isEqual as _isEqual } from 'lodash-es' /** * Utils imports @@ -253,7 +256,7 @@ watchEffect(() => { <template> <div - class="grid grid-cols-[max-content_1fr] grid-rows-4 items-center gap-x-16 gap-y-8" + class="grid max-w-max grid-cols-[max-content_1fr] grid-rows-3 items-center gap-x-16 gap-y-8" > <h3 class="text-lg font-bold">Modification types</h3> <SelectButton diff --git a/src/components/TargetSelectionForm.vue b/src/components/SelectionFormTarget.vue similarity index 98% rename from src/components/TargetSelectionForm.vue rename to src/components/SelectionFormTarget.vue index 8a7a8cbbded74ba2f63261465bd5fea98482e4d6..d22e087d57762f3e5635cbb4ef9cabbb1f4174a3 100644 --- a/src/components/TargetSelectionForm.vue +++ b/src/components/SelectionFormTarget.vue @@ -4,7 +4,7 @@ */ import { computed, ref, toRef, watchEffect } from 'vue' /** - * Component imports + * Components imports */ import BaseRenderedMarkdown from './BaseRenderedMarkdown.vue' import Chip from 'primevue/chip' @@ -12,9 +12,12 @@ import SelectButton from 'primevue/selectbutton' import MultiSelect from 'primevue/multiselect' import { ModifType } from '@/gql/codegen/graphql' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' +/** + * Other 3rd-party imports + */ import { uniqWith as _uniqWith, isEqual as _isEqual, @@ -257,7 +260,7 @@ watchEffect(() => { <template> <div - class="grid grid-cols-[max-content_1fr] grid-rows-4 items-center gap-x-16 gap-y-8" + class="grid max-w-max grid-cols-[max-content_1fr] grid-rows-3 items-center gap-x-16 gap-y-8" > <h3 class="text-lg font-bold">Targets</h3> <MultiSelect diff --git a/src/components/SequenceAlignment.vue b/src/components/SequenceAlignment.vue new file mode 100644 index 0000000000000000000000000000000000000000..3ca91569f080b2ceab57e47c77bec44a36e15cb0 --- /dev/null +++ b/src/components/SequenceAlignment.vue @@ -0,0 +1,986 @@ +<script setup lang="ts"> +/** + * Vue imports + */ +import { computed, onMounted, ref, toRef, watch } from 'vue' +/** + * Components imports + */ +import SequenceAlignmentToolbar from './SequenceAlignmentToolbar.vue' +import SequenceAlignmentTrackNames from './SequenceAlignmentTrackNames.vue' +/** + * Composables imports + */ +import { useConservationAnalysis } from '@/composables/useConservationAnalysis' +import { useElementBounding, useResizeObserver } from '@vueuse/core' +/** + * Other 3rd-party imports + */ +import { + inRange as _inRange, + uniqBy as _uniqBy, + mapValues as _mapValues +} from 'lodash-es' +/** + * Types imports + */ +import type { + HexColorCodeModel, + TailwindDefaultColorNameModel +} from '@/typings/styleTypes' +import type { RouteLocationRaw } from 'vue-router' +import type { StyleValue } from 'vue' +import type { BasesConservationConfigModel } from './SequenceAlignmentToolbar.vue' +import type { ConservationAnalysisConfigModel } from '@/composables/useConservationAnalysis' +/** + * Utils imports + */ +import { composeConservationRate } from '@/utils/sequences' +import { isDefined } from '@/typings/typeUtils' +import { range } from '@/utils/numbers' + +/** + * A legend item. + */ +export interface LegendItemModel { + /** Item id. */ + id: string + /** Item title. */ + title: string + /** Item description. */ + description?: string + /** Tailwind color name of the color to legend. */ + color?: TailwindDefaultColorNameModel +} + +/** + * An object on a track in an alignment. + */ +export interface AlignmentObjectModel { + /** The name of the object. */ + name?: string + /** The type of the object. */ + type?: string + /** The start position of the object (in the alignment reference). */ + start: number + /** The end position of the object (in the alignment reference). */ + end: number + /** The color in which to represent the object. */ + color?: TailwindDefaultColorNameModel + /** Link to go to when clicking on the object. */ + link?: RouteLocationRaw +} + +/** + * An object on a track in an alignment, with the ID of the track it is on. + */ +export interface AlignmentObjectWithTrackIdModel extends AlignmentObjectModel { + /** The ID of track the object is present on. */ + trackId: string +} + +/** + * A track in an alignment. + */ +export interface AlignmentTrackModel { + /** The ID of the track. */ + id: string + /** The name of the track. */ + name?: string + /** The short name of the track. */ + shortname?: string + /** The sequence of the track. */ + sequence: string + /** Wether the track is a reference sequence or not. */ + isReference?: boolean + /** A list of object present on the track to display. */ + objects?: { [objectId: string]: AlignmentObjectModel } + /** Wether the track is an information track. */ + isInformation?: boolean +} + +/** + * An alignment. + */ +export interface AlignmentModel { + /** The tracks composing the alignment. */ + tracks: AlignmentTrackModel[] +} + +/** + * The colours for base conservation colouring. + */ +const BASE_CONSERVATION_COLOURS = { + low: '#F87171' as HexColorCodeModel, + middle: '#FCD34D' as HexColorCodeModel, + high: '#D9EDC1' as HexColorCodeModel +} + +/** + * Component props. + */ +const props = defineProps<{ + /** The representation of the alignment. */ + alignment: AlignmentModel + /** The items of the legend to display for the highlighted groups. */ + legendItems?: LegendItemModel[] +}>() + +/** + * Size (in term of nucl.) of the groups into which the sequences are divided. + */ +const nucleotideGroupsSize = ref(10) + +/** + * Number of groups in which the sequences are divided. + */ +const nucleotideGroupsCount = computed(() => + Math.ceil( + (props.alignment.tracks[0]?.sequence.length || 1) / + nucleotideGroupsSize.value + ) +) + +/** + * Number of nucleotides in the last group. + */ +const lastNucleotideGroupSize = computed( + () => + (props.alignment.tracks[0]?.sequence.length || 1) % + nucleotideGroupsSize.value +) + +/** + * The tracks, with an additional track for the conservation analysis added + * at the end. + */ +const tracksWithConservationTrack = computed(() => [ + ...props.alignment.tracks, + conservationAnalysis.track.value +]) + +/** + * The sequences of the alignment, split and rearranged in the following nested + * arrays structure: + * ```ts + * [ + * [ // Group 0 + * [ // Nucleotide 0 + * sequence1[0], + * sequence2[0], + * ..., + * sequence<#ofTracks>[0] + * ], + * [ // Nucleotide 1 + * sequence1[1], + * sequence2[1], + * ..., + * sequence<#ofTracks>[1] + * ], + * ..., + * [ // Last nucleotide of group 0 + * sequence1[nucleotideGroupsSize - 1], + * sequence2[nucleotideGroupsSize - 1], + * ..., + * sequence<#ofTracks>[nucleotideGroupsSize - 1] + * ], + * ], + * [ // Group 1 + * [ // First nucleotide of group 1 + * sequence1[nucleotideGroupsSize], + * sequence2[nucleotideGroupsSize], + * ..., + * sequence<#ofTracks>[nucleotideGroupsSize] + * ], + * [ + * sequence1[nucleotideGroupsSize + 1], + * sequence2[nucleotideGroupsSize + 1], + * ..., + * sequence<#ofTracks>[nucleotideGroupsSize + 1] + * ], + * ..., + * [ // Last nucleotide of group 1 + * sequence1[2 * nucleotideGroupsSize - 1], + * sequence2[2 * nucleotideGroupsSize - 1], + * ..., + * sequence<#ofTracks>[2 * nucleotideGroupsSize - 1] + * ], + * ], + * ... + * ] + * ``` + */ +const splitAlignmentSequences = computed(() => + // Top-level array, size = group count, one cell = one group + Array.from({ length: nucleotideGroupsCount.value }, (_, groupIndex) => + // Second-level array, size = group size (if last group, special size), one + // cell = one position in alignment + Array.from( + { + length: + groupIndex === nucleotideGroupsCount.value - 1 + ? lastNucleotideGroupSize.value + : nucleotideGroupsSize.value + }, + (_, nucleotideIndexInGroup) => { + // Compute the position of the nucleotide + const nucleotideIndex = + groupIndex * nucleotideGroupsSize.value + nucleotideIndexInGroup + // Third-level array, size = # of sequences, one cell = the nucleotide + // at current position in the corresponding sequence + return Array.from( + { length: tracksWithConservationTrack.value.length }, + (_, sequenceIndex) => { + const nucleotide = + tracksWithConservationTrack.value[sequenceIndex]?.sequence[ + nucleotideIndex + ] + return nucleotide ? nucleotide : '-' + } + ) + } + ) + ) +) + +/** + * Wether a nucleotide is in a given object of a given track or not. + * @param trackIndex The index of the track in the alignment. + * @param position The position of the nucleotide to check. + * @param objectId The id of the object to check for. + * @returns `true` if position if in the given object, `false` otherwise. + */ +const isInObject = ( + trackIndex: number, + position: number, + objectId: string +): boolean => { + const object = + tracksWithConservationTrack.value[trackIndex]?.objects?.[objectId] + return !!object && _inRange(position, object.start, object.end + 1) +} + +/** + * Gets the IDs of the objects of a given track containing a nucleotide. + * @param trackIndex The index of the track in the alignment. + * @param position The position of the nucleotide for which to get object IDs. + * @returns A list of the IDs of the objects containing the nucleotide. + */ +const objectIdsContaining = ( + trackIndex: number, + position: number +): string[] => { + const trackObjects = tracksWithConservationTrack.value[trackIndex]?.objects + return trackObjects + ? Object.keys(trackObjects).filter((objectId) => + isInObject(trackIndex, position, objectId) + ) + : [] +} + +/** + * Whether a nucleotide is in any object of a given track. + * @param trackIndex The index of the track in the alignment. + * @param position The position of the nucleotide to check. + * @returns `true` if the nucleotide is in any object, `false` otherwise. + */ +const isInAnyObject = (trackIndex: number, position: number): boolean => + !!objectIdsContaining(trackIndex, position).length + +/** + * For each track, a `Set` of the boundary positions (start & end) of all + * objects. It is in the form of an array, indexed in the same order as the + * tracks in the alignment. + */ +const objectsBoundaryPositions = computed(() => + tracksWithConservationTrack.value.map((track) => { + // Objects present on the current track + const trackObjects = + track.isInformation && track.objects + ? Object.values(track.objects) + : referenceTrackModifications.value.filter( + (modification) => modification.trackId === track.id + ) + return trackObjects.reduce( + (objectsBoundaryPositions, object) => + object.start && object.end + ? objectsBoundaryPositions.add(object.start).add(object.end) + : objectsBoundaryPositions, + new Set<number>() + ) + }) +) + +/** + * Whether a nucleotide is a boundary position (start or end) of an object. + * @param position The position of the nucleotide to check. + * @returns `true` if the nucleotide is the start of an object, `false` otherwise. + */ +const isObjectBoundary = (trackIndex: number, position: number): boolean => + !!objectsBoundaryPositions.value[trackIndex]?.has(position) + +/** + * Composes the Tailwind color name in which to paint a nucleotide based on its + * position. + * @param trackIndex The index of the track in the alignment. + * @param position The position for which to compose the color. + * @returns `slate` if there is no or several objects at the given position, the + * Tailwind color name of the object if there is only one. + */ +const composeInObjectPositionColor = ( + trackIndex: number, + position: number +): TailwindDefaultColorNameModel => { + const objectIds = objectIdsContaining(trackIndex, position) + // If only one group, use its color, otherwise, use 'slate' + return objectIds.length === 1 && objectIds[0] + ? tracksWithConservationTrack.value[trackIndex]?.objects?.[objectIds[0]] + ?.color || 'slate' + : 'slate' +} + +/** + * Sequence container DOM `HTMLElement`. + */ +const sequenceContainerElement = ref<HTMLElement>() +/** + * Split sequences groups DOM `HTMLElement` array. + */ +const sequenceGroupElements = ref<HTMLElement[]>([]) +/** + * First & last nucleotides of each box DOM `HTMLElement` array. + */ +const boxBoundaryElements = ref<HTMLElement[]>([]) + +/** + * Bounding box of the first split sequences group DOM `HTMLElement`. + */ +const firstSequenceGroupElementBounding = useElementBounding( + toRef(() => sequenceGroupElements.value[0]) +) + +/** + * Bounding box of the split sequences groups DOM `HTMLElements`. + */ +const sequenceContainerElementBounding = useElementBounding( + sequenceContainerElement +) + +/** + * Number of sequence lines. + */ +const lineCount = computed(() => { + const firstSequenceGroupElement = sequenceGroupElements.value[0] + return ( + (firstSequenceGroupElement && + Math.round( + sequenceContainerElementBounding.height.value / + (parseInt( + window.getComputedStyle(firstSequenceGroupElement).marginTop + ) + + firstSequenceGroupElementBounding.height.value) + )) || + 1 + ) +}) + +/** + * Height of a sequence line. + */ +const lineHeight = computed( + () => + firstSequenceGroupElementBounding.bottom.value - + sequenceContainerElementBounding.top.value +) + +/** + * Dictionary of the number of line on which each object spreads, + * by group ID. + */ +const tracksObjectsLineCount = ref< + ({ [groupId: string]: number | undefined } | undefined)[] +>([]) + +/** + * Updates the number of line on which each object spreads. + * @returns The updated number of line on which each object spreads. + * @description Retrieves the line count by getting the difference between the + * top of the last nucleotide of the box and the top of the first one, and + * dividing by the line height. + */ +const updateObjectsLineCount = () => + (tracksObjectsLineCount.value = tracksWithConservationTrack.value.map( + (track, trackIndex) => + _mapValues(track.objects, (trackObject) => { + const objectStartNucleotideElementParent = + boxBoundaryElements.value.find( + (element) => + element.dataset.trackAndPosition === + `${trackIndex}:${trackObject.start}` + )?.offsetParent + + const objectEndNucleotideElementParent = boxBoundaryElements.value.find( + (element) => + element.dataset.trackAndPosition === + `${trackIndex}:${trackObject.end}` + )?.offsetParent + + if ( + !( + objectStartNucleotideElementParent instanceof HTMLElement && + objectEndNucleotideElementParent instanceof HTMLElement && + objectStartNucleotideElementParent.offsetParent instanceof + HTMLElement && + objectEndNucleotideElementParent.offsetParent instanceof HTMLElement + ) + ) { + return undefined + } + + return ( + lineHeight.value && + Math.round( + (objectEndNucleotideElementParent.offsetParent.offsetTop - + objectStartNucleotideElementParent.offsetParent.offsetTop) / + lineHeight.value + ) + 1 + ) + }) + )) + +/** + * Width of the sequence container. + */ +const sequenceContainerContentWidth = ref<number>() + +/** + * Updates the width of the sequence container. + */ +const updateSequenceContainerContentWidth = + (): typeof sequenceContainerContentWidth.value => { + // First sequence group DOM element + const firstSequenceGroupElement = sequenceGroupElements.value[0] + if (!firstSequenceGroupElement) return + + // Size of the right margin of a sequence group DOM element + const sequenceGroupElementRightMargin = parseInt( + window.getComputedStyle(firstSequenceGroupElement).marginRight + ) + if (!sequenceGroupElementRightMargin) return + + // Width of a sequence group DOM element + const sequenceGroupElementWidth = + firstSequenceGroupElement.offsetWidth + sequenceGroupElementRightMargin + if (!sequenceGroupElementWidth) return + + // Width of the sequences container DOM element + const sequenceContainerElementWidth = + sequenceContainerElement.value?.offsetWidth + if (!sequenceContainerElementWidth) return + + // Sequence content width = container width - (space not actually occupied by + // sequence + 1 * right margin of a sequence group) + return (sequenceContainerContentWidth.value = + sequenceContainerElementWidth - + ((sequenceContainerElementWidth % sequenceGroupElementWidth) + + sequenceGroupElementRightMargin)) + } + +/** + * Updates all non-reactive values. + */ +const update = (): void => { + updateObjectsLineCount() + updateSequenceContainerContentWidth() +} + +/** + * Sets hook to update non-reactive values on component mount. + */ +onMounted(update) + +/** + * Sets an observer to update on container resize. + */ +useResizeObserver(sequenceContainerElement, update) + +/** + * Reactive array, for each track, the style to apply to each fragment of each + * of its objects. + */ +const boxStyles = computed(() => + // Map tracks + tracksWithConservationTrack.value.map((track, trackIndex) => { + // No object on current track + if (!track.objects) { + return {} + } + + // Map track objects to the style of each of their fragment + return _mapValues(track.objects, (trackObject, trackObjectId) => { + // DOM element of the first nucleotide of the box + const objectStartNucleotideElementParent = boxBoundaryElements.value.find( + (element) => + element.dataset.trackAndPosition === + `${trackIndex}:${trackObject.start}` + )?.offsetParent + + // DOM element of the last nucleotide of the box + const objectEndNucleotideElementParent = boxBoundaryElements.value.find( + (element) => + element.dataset.trackAndPosition === + `${trackIndex}:${trackObject.end}` + )?.offsetParent + + if ( + !( + objectStartNucleotideElementParent instanceof HTMLElement && + objectEndNucleotideElementParent instanceof HTMLElement && + objectStartNucleotideElementParent.offsetParent instanceof + HTMLElement && + objectEndNucleotideElementParent.offsetParent instanceof HTMLElement + ) + ) { + return + } + + // Absolute positions of the first and last nucleotide of the object DOM + // elements + const objectStartNucleotideElementTop = + objectStartNucleotideElementParent.offsetParent.offsetTop + + (objectStartNucleotideElementParent?.offsetTop || 0) + const objectStartNucleotideElementLeft = + objectStartNucleotideElementParent.offsetParent.offsetLeft + + (objectStartNucleotideElementParent?.offsetLeft || 0) + const objectEndNucleotideElementLeft = + objectEndNucleotideElementParent.offsetParent.offsetLeft + + (objectEndNucleotideElementParent?.offsetLeft || 0) + const objectEndNucleotideElementRight = + objectEndNucleotideElementLeft + + (objectEndNucleotideElementParent?.offsetWidth || 0) + + // Number of fragment in which the current object is divided + const boxFragmentCount = + tracksObjectsLineCount.value?.[trackIndex]?.[trackObjectId] || 0 + + // Array of length equals # of fragments, containing style for each of them + return Array.from( + { length: boxFragmentCount }, + (_, fragmentIndex): StyleValue => ({ + top: `${ + objectStartNucleotideElementTop + + fragmentIndex * (lineHeight.value || 0) + }px`, + left: + fragmentIndex === 0 + ? `calc(${objectStartNucleotideElementLeft}px - 0.125rem)` + : '0', + right: `calc(${ + (sequenceContainerElementBounding.width.value || 0) - + (fragmentIndex === boxFragmentCount - 1 + ? objectEndNucleotideElementRight + : sequenceContainerContentWidth.value || 0) + }px - 0.125rem)` + }) + ) + }) + }) +) + +/** + * Compute the position of a nucleotide in the reference of a sequence from the + * position on the same sequence gapped with `-` for alignment. + * @param alignedSequence The sequence, gapped with `-` for alignment with other + * sequences. + * @param inAlignmentPosition The position in the reference of the alignment to + * convert to the source sequence reference (starts at 1). + */ +const composePositionOnSource = ( + alignedSequence: string, + inAlignmentPosition: number +): number | undefined => + alignedSequence[inAlignmentPosition - 1] === '-' + ? undefined + : inAlignmentPosition - + (alignedSequence.slice(0, inAlignmentPosition).split('-').length - 1) + +/** + * Compute the position of a nucleotide on a sequence gapped with `-` for + * alignment from the position in the source reference of the same sequence. + * @param alignedSequence The sequence, gapped with `-` for alignment with other + * sequences. + * @param inSourcePosition The position in the source sequence reference to + * convert to the reference of the alignment (starts at 1). + * @returns The position of the nucleotide on the aligned sequence, -1 if the + * requested position exceeded the actual length of the source sequence. + */ +const composePositionOnAlignment = ( + alignedSequence: string, + inSourcePosition: number +): number => { + const firstDashPositionOnAlignment = alignedSequence.indexOf('-') + 1 + + // If the aligned sequence is shorter than the requested position, it means + // that the latter doesn't exists. + if (alignedSequence.length < inSourcePosition) { + return -1 + } + + // If the requested position is closer to the start than the first dash (or if + // there is no dash in it), then the position on the alignment is the same + // than on the source sequence. + if ( + inSourcePosition < firstDashPositionOnAlignment || + firstDashPositionOnAlignment === 0 + ) { + return inSourcePosition + } + + // Otherwise, remove the part to the first dash and compute the position on + // the remaining aligned sequence (& handle case were the position doesn't + // exists). + const positionOnRemainingAlignment = composePositionOnAlignment( + alignedSequence.slice(firstDashPositionOnAlignment), + inSourcePosition - firstDashPositionOnAlignment - 1 + ) + return positionOnRemainingAlignment === -1 + ? -1 + : positionOnRemainingAlignment + firstDashPositionOnAlignment +} + +/** + * Maximum width of the names of the tracks DOM `HTMLElements`. + */ +const trackNamesWidth = ref<number>(0) + +/** + * Width of the container element of the track names, express as a ratio to + * apply on the width of the sequence group elements (e.g., if it is 3, this + * means that the container element of the track names should be 3 times wider + * than a sequence group element). + */ +const trackNamesContainerWidthRatio = computed(() => + Math.ceil( + trackNamesWidth.value / (firstSequenceGroupElementBounding.width.value || 1) + ) +) + +/** + * Configuration for the bases conservation colouring. + */ +const basesConservationConfig = ref<BasesConservationConfigModel>({ + isEnabled: false, + thresholds: { low: 0.25, high: 0.75 } +}) + +/** + * Computes the conservation rate of the reference nucleotide in the provided + * nucl. list, and derivates from it the colour to use for the background. + * @param nucleotides The list of nucleotides on which to compute the + * conservation colour. + * @param referenceNucleotide The nucleotide for which to compute the + * conservation colour. + * @returns The colour in which to set the background according to the + * computed conservation rate. + */ +const composeConservationColour = ( + nucleotides: string[], + referenceNucleotide: string +): string => { + const conservationRate = composeConservationRate( + nucleotides, + referenceNucleotide + ) + if (conservationRate > basesConservationConfig.value.thresholds.high) { + return BASE_CONSERVATION_COLOURS.high + } + if (conservationRate > basesConservationConfig.value.thresholds.low) { + return BASE_CONSERVATION_COLOURS.middle + } + return BASE_CONSERVATION_COLOURS.low +} + +/** + * All the modifications present on the reference track sequences. + */ +const referenceTrackModifications = computed<AlignmentObjectWithTrackIdModel[]>( + () => + props.alignment.tracks + .filter((track) => track.isReference) + .map( + (track) => + track.objects && + Object.values(track.objects).map((object) => ({ + ...object, + trackId: track.id + })) + ) + .flat() + .filter(isDefined) +) + +/** + * Available options for track selection for conservation analysis. + */ +const conservationAnalysisTrackOptions = computed(() => + props.alignment.tracks + .filter((track) => track.isReference && !track.isInformation) + .map((track) => ({ + label: track.name || track.id, + id: track.id + })) +) + +/** + * A list of the types of the objects present on the sequences of the alignment. + */ +const conservationAnalysisObjectTypeOptions = computed(() => + _uniqBy( + Array.from( + props.alignment.tracks + .filter((track) => + conservationAnalysisConfig.value.trackIds.includes(track.id) + ) + .map( + (track) => + track.objects && + Object.values(track.objects).map((object) => + object.type === undefined + ? undefined + : { label: object.type, type: object.type } + ) + ) + .flat() + .filter(isDefined) + ), + 'type' + ) +) + +/** + * Configuration for the conservation analysis. + */ +const conservationAnalysisConfig = ref<ConservationAnalysisConfigModel>({ + trackIds: [], + objectTypes: [], + analysisType: null +}) + +/** + * Update non-reactive values when config changes. + */ +watch(conservationAnalysisConfig, () => { + setTimeout(update, 100) +}) + +/** + * Conservation analysis, reactively updated when analysis parameters change. + */ +const conservationAnalysis = useConservationAnalysis( + props.alignment.tracks, + conservationAnalysisConfig +) + +/** + * Wether a conservation analysis is displayed or not. + */ +const isConservationAnalysisDisplayed = computed( + () => conservationAnalysisConfig.value.analysisType !== null +) + +/** + * The value to apply to the conservation analysis track `display` CSS property. + */ +const conservationAnalysisTrackDisplay = computed(() => ({ + trackLabel: isConservationAnalysisDisplayed.value ? 'flex' : 'none', + trackSequence: isConservationAnalysisDisplayed.value ? undefined : 'none' +})) +</script> + +<template> + <SequenceAlignmentToolbar + v-model:nucleotide-groups-size-model="nucleotideGroupsSize" + v-model:bases-conservation-config-model="basesConservationConfig" + v-model:conservation-analysis-config-model="conservationAnalysisConfig" + :bases-conservation-colours="BASE_CONSERVATION_COLOURS" + :conservation-analysis-track-options="conservationAnalysisTrackOptions" + :conservation-analysis-object-type-options=" + conservationAnalysisObjectTypeOptions + " + /> + + <div + class="mt-8 grid gap-4 text-2xl leading-[normal]" + :style="{ + gridTemplateColumns: `${ + trackNamesContainerWidthRatio * nucleotideGroupsSize * 1.5 + }rem 1fr` + }" + > + <!-- Track names --> + <div> + <SequenceAlignmentTrackNames + v-model:track-names-width-model="trackNamesWidth" + :tracks="tracksWithConservationTrack" + /> + <SequenceAlignmentTrackNames + v-for="line in range(lineCount - 1)" + :key="line" + :tracks="tracksWithConservationTrack" + /> + </div> + + <div ref="sequenceContainerElement" class="relative"> + <!-- Track content --> + <span + v-for="(sequenceGroup, groupIndex) in splitAlignmentSequences" + :key="groupIndex" + ref="sequenceGroupElements" + :data-group-index="groupIndex" + class="relative z-10 mr-4 mt-8 inline-block select-none whitespace-nowrap font-mono" + :style="{ width: `${nucleotideGroupsSize * 1.5}rem` }" + > + <span class="absolute -top-4 left-0 select-none text-sm text-slate-400"> + {{ groupIndex * nucleotideGroupsSize + 1 }} + </span> + <span + v-for="(nucleotide, nucleotideIndexInGroup) in sequenceGroup" + :key="groupIndex * nucleotideGroupsSize + nucleotideIndexInGroup" + :data-nucleotide-index=" + groupIndex * nucleotideGroupsSize + nucleotideIndexInGroup + " + class="inline-flex flex-col" + > + <span + v-for="(trackNucleotide, trackIndex) in nucleotide" + :key="`${trackIndex}:${ + groupIndex * nucleotideGroupsSize + nucleotideIndexInGroup + }`" + v-tooltip.top="{ + value: + composePositionOnSource( + alignment.tracks[trackIndex]?.sequence || '', + groupIndex * nucleotideGroupsSize + nucleotideIndexInGroup + 1 + )?.toString() || '<em>N/A</em>', + pt: { + text: { + style: { textAlign: 'center', fontWeight: '600' } + } + }, + escape: true, + autoHide: false + }" + :class="[ + 'relative mx-[.0625rem] cursor-help border-2 border-transparent px-0.5 pt-0.5', + { + 'rounded-t-md': trackIndex === 0, + 'rounded-b-md': + trackIndex === props.alignment.tracks.length - 1, + 'conservation-track-sequence': + tracksWithConservationTrack[trackIndex]?.isInformation + }, + isInAnyObject( + trackIndex, + groupIndex * nucleotideGroupsSize + nucleotideIndexInGroup + 1 + ) && [ + `text-${composeInObjectPositionColor( + trackIndex, + groupIndex * nucleotideGroupsSize + nucleotideIndexInGroup + 1 + )}-600`, + 'modification font-semibold' + ] + ]" + :style=" + (basesConservationConfig.isEnabled && + trackNucleotide !== '-' && + !tracksWithConservationTrack[trackIndex]?.isInformation && { + backgroundColor: composeConservationColour( + nucleotide, + trackNucleotide + ) + }) || + undefined + " + > + <span + v-if=" + isObjectBoundary( + trackIndex, + groupIndex * nucleotideGroupsSize + nucleotideIndexInGroup + 1 + ) + " + ref="boxBoundaryElements" + :data-track-and-position="`${trackIndex}:${ + groupIndex * nucleotideGroupsSize + nucleotideIndexInGroup + 1 + }`" + > + {{ trackNucleotide }} + </span> + <span v-else> + {{ trackNucleotide }} + </span> + </span> + </span> + </span> + + <!-- Object boxes --> + <span + v-for="(track, trackIndex) in tracksWithConservationTrack" + :key="track.id" + > + <span + v-for="[objectId, object] in Object.entries(track.objects || {})" + :key="objectId" + class="object" + > + <span + v-for="fragmentIndex in range( + tracksObjectsLineCount?.[trackIndex]?.[objectId] || 0 + )" + :key="fragmentIndex" + :style="boxStyles?.[trackIndex]?.[objectId]?.[fragmentIndex]" + :class="[ + 'object-fragment absolute z-10 h-8 border-y-2 bg-opacity-50 px-0.5 text-2xl mix-blend-multiply', + [`!border-${object.color}-600`, `bg-${object.color}-100`], + { + 'rounded-l-xl border-l-2': fragmentIndex === 0, + 'rounded-r-xl border-r-2': + tracksObjectsLineCount && + fragmentIndex + 1 === + tracksObjectsLineCount[trackIndex]?.[objectId] + } + ]" + /> + </span> + </span> + </div> + </div> +</template> + +<style lang="scss"> +.conservation-track-label { + display: v-bind('conservationAnalysisTrackDisplay.trackLabel'); +} + +.conservation-track-sequence { + display: v-bind('conservationAnalysisTrackDisplay.trackSequence'); +} + +.track-name-arrow { + &::before { + // Custom dashed border + border-image: url('') + 2 0 0 / 1 1 0 repeat; + } + &::after { + background: rgb(255, 255, 255); + background: linear-gradient( + 45deg, + rgba(255, 255, 255, 0) 0%, + rgba(255, 255, 255, 0) 50%, + rgba(255, 255, 255, 1) 50%, + rgba(255, 255, 255, 1) 100% + ); + } +} +</style> diff --git a/src/components/SequenceAlignmentToolbar.vue b/src/components/SequenceAlignmentToolbar.vue new file mode 100644 index 0000000000000000000000000000000000000000..25228b1219f853b06cb879481caf75dad8520934 --- /dev/null +++ b/src/components/SequenceAlignmentToolbar.vue @@ -0,0 +1,586 @@ +<script setup lang="ts"> +/** + * Vue imports + */ +import { ref, watch } from 'vue' +/** + * Component imports + */ +import BaseRenderedMarkdown from './BaseRenderedMarkdown.vue' +import Toolbar from 'primevue/toolbar' +import Dropdown from 'primevue/dropdown' +import TabView from 'primevue/tabview' +import TabPanel from 'primevue/tabpanel' +import Slider, { type SliderSlideEndEvent } from 'primevue/slider' +import MultiSelect from 'primevue/multiselect' +import SelectButton from 'primevue/selectbutton' +import ToggleButton from 'primevue/togglebutton' +import IconFa6RegularCircleQuestion from '~icons/fa6-regular/circle-question' + +/** + * Types imports. + */ +import type { HexColorCodeModel } from '@/typings/styleTypes' +import { + ConservationAnalysisTypesEnum, + type ConservationAnalysisConfigModel +} from '@/composables/useConservationAnalysis' + +/** + * Configuration for the bases conservation colouring. + */ +export interface BasesConservationConfigModel { + /** Wether the bases conservation colouring is enabled. */ + isEnabled: boolean + /** Conservation rate thresholds for bases conservation colouring. */ + thresholds: { + low: number + high: number + } +} + +/** + * Available options for conservation analysis type. + */ +const CONSERVATION_ANALYSIS_TYPES_OPTIONS = [ + { + label: 'Specific', + analysisType: ConservationAnalysisTypesEnum.Specific + }, + { + label: 'Common', + analysisType: ConservationAnalysisTypesEnum.Common + } +] + +/** + * Component props. + */ +const props = withDefaults( + defineProps<{ + /** The selected nucleotide group size. */ + nucleotideGroupsSizeModel: number + /** The config for the bases conservation colouring. */ + basesConservationConfigModel: BasesConservationConfigModel + /** Colours to use for bases conservation colouring */ + basesConservationColours?: { + low: HexColorCodeModel + middle: HexColorCodeModel + high: HexColorCodeModel + } + /** The config for the conservation analysis. */ + conservationAnalysisConfigModel: ConservationAnalysisConfigModel + /** Available options for track selection for conservation analysis. */ + conservationAnalysisTrackOptions: { label: string; id: string }[] + /** Available options for selection of the types of the objects present on + * the sequences for conservation analysis. */ + conservationAnalysisObjectTypeOptions: { label?: string; type: string }[] + }>(), + { + basesConservationColours: () => ({ + low: '#F87171' as HexColorCodeModel, + middle: '#FCD34D' as HexColorCodeModel, + high: '#D9EDC1' as HexColorCodeModel + }) + } +) + +const emit = defineEmits<{ + /** Event used to update the `v-model:nucleotideGroupsSizeModel` value. */ + 'update:nucleotideGroupsSizeModel': [nucleotideGroupsSizeModel: number] + /** Event used to update the `v-model:basesConservationConfigModel` value. */ + 'update:basesConservationConfigModel': [ + basesConservationConfigModel: BasesConservationConfigModel + ] + /** Event used to update the `v-model:conservationAnalysisConfigModel` value. */ + 'update:conservationAnalysisConfigModel': [ + conservationAnalysisConfigModel: ConservationAnalysisConfigModel + ] +}>() + +/** + * The currently active tab. + */ +const activeTabIndex = ref(0) + +/** + * Clears the conservation analysis type if no track or modification type is + * selected anymore for conservation analysis. + */ +const clearAnalysisTypeIfNeeded = () => { + if ( + props.conservationAnalysisConfigModel.objectTypes.length === 0 || + props.conservationAnalysisConfigModel.trackIds.length === 0 + ) { + emit('update:conservationAnalysisConfigModel', { + ...props.conservationAnalysisConfigModel, + analysisType: null + }) + } +} + +/** + * Callbacks to use for events in template. + */ +const eventHandlers = { + /** Callbacks for values related to bases conservation colouring. */ + basesConservation: { + /** Callback for the toggling of bases conservation colouring. */ + isEnabled: (newValue: boolean) => + emit('update:basesConservationConfigModel', { + ...props.basesConservationConfigModel, + isEnabled: newValue + }), + /** Callback for bases conservation colouring low threshold update by slider. */ + lowThreshold: (slideEndEvent: SliderSlideEndEvent) => { + emit('update:basesConservationConfigModel', { + ...props.basesConservationConfigModel, + thresholds: { + ...props.basesConservationConfigModel.thresholds, + low: slideEndEvent.value + } + }) + }, + /** Callback for bases conservation colouring high threshold update by slider. */ + highThreshold: (slideEndEvent: SliderSlideEndEvent) => { + emit('update:basesConservationConfigModel', { + ...props.basesConservationConfigModel, + thresholds: { + ...props.basesConservationConfigModel.thresholds, + high: slideEndEvent.value + } + }) + } + }, + /** Callbacks for values related to conservation analysis. */ + conservationAnalysis: { + /** Callback for track IDs selection update in Multiselect. */ + trackIds: (newValue: string[]) => { + clearAnalysisTypeIfNeeded() + emit('update:conservationAnalysisConfigModel', { + ...props.conservationAnalysisConfigModel, + trackIds: newValue + }) + }, + /** Callback for object types selection update in Multiselect. */ + objectTypes: (newValue: string[]) => { + clearAnalysisTypeIfNeeded() + emit('update:conservationAnalysisConfigModel', { + ...props.conservationAnalysisConfigModel, + objectTypes: newValue + }) + }, + /** Callback for object types selection update in Multiselect. */ + analysisType: (newValue: ConservationAnalysisTypesEnum) => { + emit('update:conservationAnalysisConfigModel', { + ...props.conservationAnalysisConfigModel, + analysisType: newValue + }) + } + } +} + +/** + * Local values of the thresholds for bases conservation colouring, to use as + * `v-model` for the sliders. + * + * @description We use this instead of + * {@link props.basesConservationConfigModel.thresholds} directly because + * sliders update their value immediately, but we only wan't to update the + * actual thresholds when mouse is released (on `slideend` event). + * Thus, those intermediary values allow displaying the new threshold in real + * time, but only updating the actual value when the mouse is released. + */ +const localBasesConservationThresholds = ref({ + ...props.basesConservationConfigModel.thresholds +}) + +/** + * Watcher to update the local value of the thresholds for bases conservation + * colouring when actual thresholds are updated using `v-model`. + */ +watch( + () => props.basesConservationConfigModel.thresholds, + () => + (localBasesConservationThresholds.value = { + ...props.basesConservationConfigModel.thresholds + }) +) +</script> + +<template> + <Toolbar + :pt="{ + start: { + style: { + gap: '2rem', + alignItems: 'stretch' + } + } + }" + class="mx-auto w-5/6" + > + <template #start> + <label class="flex flex-col gap-2 text-sm"> + Group length + <div class="flex grow flex-col justify-center"> + <Dropdown + :model-value="nucleotideGroupsSizeModel" + :options="[5, 10, 20]" + class="w-52" + @update:model-value=" + (newValue: number) => $emit('update:nucleotideGroupsSizeModel', newValue) + " + > + <template #option="{ option }">{{ option }} nucleotides</template> + <template #value="{ value }">{{ value }} nucleotides</template> + </Dropdown> + </div> + </label> + + <!-- <div class="self-stretch rounded border border-slate-200" /> --> + </template> + + <template #center> + <TabView + v-model:active-index="activeTabIndex" + :pt="{ + nav: { + style: { + background: 'none', + justifyContent: 'center' + } + }, + panelContainer: { + style: { + background: 'none' + } + } + }" + > + <TabPanel + header="Bases conservation" + :pt="{ + headerAction: { + style: { + background: 'none' + } + } + }" + > + <div class="flex flex-wrap gap-8"> + <div class="flex flex-col gap-2"> + <label + for="sequence-conservation-colouring-switch" + class="text-sm" + > + Conservation colouring + </label> + <div class="flex grow flex-col justify-center"> + <ToggleButton + :model-value="basesConservationConfigModel.isEnabled" + on-label="On" + off-label="Off" + class="w-full" + input-id="sequence-conservation-colouring-switch" + @update:model-value=" + eventHandlers.basesConservation.isEnabled + " + /> + </div> + </div> + + <label class="flex flex-col gap-2 text-sm"> + Conservation thresholds + <div class="flex grow flex-col justify-center"> + <div + v-tooltip.bottom=" + !basesConservationConfigModel.isEnabled && { + value: + 'To set the thresholds, first activate the colouring.', + autoHide: false, + pt: { text: 'text-xs text-center' } + } + " + class="flex gap-4 rounded-md border border-zinc-300 bg-white p-2 shadow-sm" + > + <div + :class="{ + 'text-slate-400': !basesConservationConfigModel.isEnabled + }" + > + <label + id="low-conservation-threshold-label" + class="font-thin italic" + > + Little conserved + <span + v-tooltip.bottom=" + basesConservationConfigModel.isEnabled && { + value: + 'The conservation rate under which to color the nucleotides with the \'little conserved\' color.', + autoHide: false, + pt: { text: 'text-xs text-center' } + } + " + > + <icon-fa6-regular-circle-question + class="mb-0.5 ml-1 inline" + /> + </span> + </label> + <Slider + v-model="localBasesConservationThresholds.low" + :pt="{ + root: { + style: { + backgroundColor: basesConservationColours.middle + } + }, + range: { + style: { + backgroundColor: basesConservationColours.low + } + } + }" + :disabled="!basesConservationConfigModel.isEnabled" + class="my-2 w-full" + :step="0.05" + :min="0" + :max="0.5" + aria-labelledby="low-conservation-threshold-label" + @slideend="eventHandlers.basesConservation.lowThreshold" + /> + <div class="w-full text-center font-mono"> + {{ + Math.round(localBasesConservationThresholds.low * 100) + }}% + </div> + </div> + <div + :class="{ + 'text-slate-400': !basesConservationConfigModel.isEnabled + }" + > + <label + id="high-conservation-threshold-label" + class="font-thin italic" + > + Highly conserved + <span + v-tooltip.bottom=" + basesConservationConfigModel.isEnabled && { + value: + 'The conservation rate above which to color the nucleotides with the \'highly conserved\' color.', + autoHide: false, + pt: { text: 'text-xs text-center' } + } + " + > + <icon-fa6-regular-circle-question + class="mb-0.5 ml-1 inline" + /> + </span> + </label> + <Slider + v-model="localBasesConservationThresholds.high" + :pt="{ + root: { + style: { + backgroundColor: basesConservationColours.high + } + }, + range: { + style: { + backgroundColor: basesConservationColours.middle + } + } + }" + :disabled="!basesConservationConfigModel.isEnabled" + class="my-2 w-full" + :step="0.05" + :min="0.5" + :max="1" + aria-labelledby="high-conservation-threshold-label" + @slideend="eventHandlers.basesConservation.highThreshold" + /> + <div class="w-full text-center font-mono"> + {{ + Math.round(localBasesConservationThresholds.high * 100) + }}% + </div> + </div> + </div> + </div> + </label> + </div> + </TabPanel> + + <TabPanel + header="Modifications conservation" + :pt="{ + headerAction: { + style: { + background: 'none' + } + } + }" + > + <div class="flex flex-wrap gap-8"> + <div class="flex flex-col gap-2"> + <label for="track-selection" class="text-sm"> Tracks </label> + <div class="flex grow flex-col justify-center"> + <MultiSelect + :model-value="conservationAnalysisConfigModel.trackIds" + :options="conservationAnalysisTrackOptions" + option-label="label" + option-value="id" + multiple + placeholder="Select tracks..." + input-id="track-selection" + @update:model-value=" + eventHandlers.conservationAnalysis.trackIds + " + /> + </div> + </div> + <div class="flex flex-col gap-2"> + <label for="object-types-selection" class="text-sm"> + Modification type + </label> + <div class="flex grow flex-col justify-center"> + <MultiSelect + :model-value="conservationAnalysisConfigModel.objectTypes" + :options="conservationAnalysisObjectTypeOptions" + option-label="label" + option-value="type" + multiple + placeholder="Select object types..." + input-id="object-types-selection" + @update:model-value=" + eventHandlers.conservationAnalysis.objectTypes + " + > + <template #option="{ option }"> + <BaseRenderedMarkdown + :stringified-markdown="option.label" + inline-content + /> + </template> + <template #value="{ value }"> + <BaseRenderedMarkdown + v-if="value && value.length" + :stringified-markdown="value.join(', ')" + inline-content + /> + </template> + </MultiSelect> + </div> + </div> + + <div class="flex flex-col gap-2"> + <label for="analysis-type-selection" class="text-sm"> + Analysis type + <span + v-tooltip.bottom="{ + value: `• Specific: highlights positions modified on the selected sequences only (all of them). + + • Common: highlights positions modified on the selected sequences at least (all of them).`, + autoHide: false, + pt: { text: 'text-xs w-[40ch]' } + }" + > + <icon-fa6-regular-circle-question + class="mb-0.5 ml-1 inline" + /> + </span> + </label> + <div + v-tooltip.bottom=" + !( + conservationAnalysisConfigModel.objectTypes.length && + props.conservationAnalysisConfigModel.objectTypes.length + ) && { + value: + 'First select some sequences and modification types to chose a conservation analysis type.', + autoHide: false, + pt: { text: 'text-xs' } + } + " + class="flex grow flex-col justify-center" + > + <SelectButton + :model-value="conservationAnalysisConfigModel.analysisType" + :options="CONSERVATION_ANALYSIS_TYPES_OPTIONS" + option-label="label" + option-value="analysisType" + :disabled=" + !( + conservationAnalysisConfigModel.objectTypes.length && + props.conservationAnalysisConfigModel.objectTypes.length + ) + " + input-id="analysis-type-selection" + @update:model-value=" + eventHandlers.conservationAnalysis.analysisType + " + /> + </div> + </div> + </div> + </TabPanel> + </TabView> + </template> + + <!-- <template v-if="legendItems" #center> + <BaseLegendButtonOverlay button-text="Legend" :items="legendItems"> + <template #item="{ item }"> + <slot name="legend-item" :item="item" /> + </template> + <template #item-icon="{ item }"> + <slot name="legend-item-icon" :item="item" /> + </template> + <template #item-title="{ item }"> + <slot name="legend-item-title" :item="item" /> + </template> + <template #item-description="{ item }"> + <slot name="legend-item-description" :item="item" /> + </template> + </BaseLegendButtonOverlay> + </template> --> + + <!-- <template #end> + <Button + outlined + severity="secondary" + :class="[ + clipboardState === ClipboardStateModel.PostCopy && + '!bg-lime-100 !text-lime-700', + 'flex min-w-[18ch] justify-center' + ]" + :loading="clipboardState === ClipboardStateModel.Busy" + @click="sequenceToClipboard" + > + <span + v-if="clipboardState === ClipboardStateModel.Busy" + class="mr-2 h-5 overflow-hidden text-xl" + > + <icon-fa6-solid-dna class="animate-[dnaSpin_.5s_linear_infinite]" /> + <icon-fa6-solid-dna class="animate-[dnaSpin_.5s_linear_infinite]" /> + </span> + <icon-fa6-solid-circle-check + v-else-if="clipboardState === ClipboardStateModel.PostCopy" + class="mr-2 text-xl" + /> + <icon-fa6-regular-copy v-else class="mr-2 text-xl" /> + {{ + clipboardState === ClipboardStateModel.Busy + ? 'Copying...' + : clipboardState === ClipboardStateModel.PostCopy + ? 'Copied !' + : 'Copy sequence' + }} + </Button> + </template> --> + </Toolbar> +</template> diff --git a/src/components/SequenceAlignmentTrackNames.vue b/src/components/SequenceAlignmentTrackNames.vue new file mode 100644 index 0000000000000000000000000000000000000000..e093b10f650a7e6ac1c661d8f9c25fc2fc31ec8e --- /dev/null +++ b/src/components/SequenceAlignmentTrackNames.vue @@ -0,0 +1,123 @@ +<script setup lang="ts"> +/** + * Vue imports + */ +import { computed, ref, watchEffect } from 'vue' +/** + * Composables imports. + */ +import { useElementSize } from '@vueuse/core' +/** + * Types imports. + */ +import type { AlignmentTrackModel } from './SequenceAlignment.vue' + +const props = defineProps<{ + /** The tracks of which to display the names. */ + tracks: AlignmentTrackModel[] + /** The width of the track names, based on the longest name width. */ + trackNamesWidthModel?: number +}>() + +const emit = defineEmits<{ + /** Event used to update the `v-model:trackNamesWidthModel` value. */ + 'update:trackNamesWidthModel': [trackNamesWidthModel: number] +}>() + +/** + * Names of the tracks DOM `HTMLElement` array. + */ +const trackNameElements = ref<HTMLElement[]>([]) + +/** + * Sizes of the names of the tracks DOM `HTMLElements`. + */ +const trackNameElementSizes = computed(() => + trackNameElements.value.map((trackNameElement) => { + return useElementSize(trackNameElement) + }) +) + +/** + * Watcher to compute & rise the width of the track names to parent. + */ +watchEffect(() => { + emit( + 'update:trackNamesWidthModel', + trackNameElementSizes.value.reduce( + // Get the maximum amongst the track name element's widths. + (maxWidth, trackNameElementSize, trackIndex) => { + const trackNameElementWidth = + trackNameElementSize.width.value + + (props.tracks[trackIndex]?.isReference ? 50 : 0) + + 15 + return trackNameElementWidth > maxWidth + ? trackNameElementWidth + : maxWidth + }, + 0 + ) + ) +}) +</script> + +<template> + <div + class="mt-8 inline-flex w-full flex-col whitespace-nowrap transition-transform duration-500" + > + <div + v-for="track in tracks" + :key="track.id" + :class="[ + 'flex min-h-[2.125rem] border-2 border-transparent italic text-slate-400', + { + 'conservation-track-label justify-end': track.isInformation + } + ]" + > + <component + :is="track.isReference ? 'RouterLink' : 'span'" + ref="trackNameElements" + v-tooltip="{ + value: track.name, + pt: { + root: 'ml-2', + text: { + style: { textAlign: 'center', fontWeight: '600' } + } + } + }" + :to=" + track.isReference && { + name: 'targetDetails', + query: { id: track.id } + } + " + :class="{ + 'rounded-full border border-slate-400 px-2 text-center font-mono text-xl not-italic': + track.isInformation, + 'font-bold underline decoration-transparent decoration-dashed transition-all duration-200 hover:decoration-inherit': + track.isReference + }" + > + {{ track.shortname || track.id }} + </component> + + <span + v-if="track.isReference" + v-tooltip="{ + value: 'This sequence is a reference sequence.', + pt: { text: 'text-xs' } + }" + class="small-caps my-auto ml-3 rounded-lg border-2 border-solid border-slate-400 bg-slate-100 px-1.5 text-sm font-bold text-slate-400" + > + Ref + </span> + + <div + v-if="!track.isInformation" + class="track-name-arrow relative grow before:absolute before:left-2 before:right-0 before:top-1/2 before:-translate-y-[1px] before:border-t-2 after:absolute after:right-0 after:top-1/2 after:h-2 after:w-2 after:-translate-y-1/2 after:rotate-45 after:border-r-2 after:border-t-2 after:border-slate-400" + /> + </div> + </div> +</template> diff --git a/src/components/SequenceBoard.vue b/src/components/SequenceBoard.vue index 3c012016b7eebb36029ad42baf89822feb93a54a..df6849ba99e62b4483eefad43b90731968f006c8 100644 --- a/src/components/SequenceBoard.vue +++ b/src/components/SequenceBoard.vue @@ -16,7 +16,7 @@ import IconFa6SolidCircleCheck from '~icons/fa6-solid/circle-check' import IconFa6RegularCopy from '~icons/fa6-regular/copy' import IconFa6SolidCircleInfo from '~icons/fa6-solid/circle-info' /** - * Other 3rd-party imports + * Composables imports */ import { useElementBounding, @@ -24,13 +24,17 @@ import { useMutationObserver, useResizeObserver } from '@vueuse/core' -import { inRange as _inRange } from 'lodash-es' +/** + * Other 3rd-party imports + */ +import { inRange as _inRange, mapValues as _mapValues } from 'lodash-es' /** * Types imports */ import type { TailwindDefaultColorNameModel } from '@/typings/styleTypes' import type { LegendItemModel } from '@/components/BaseLegendButtonOverlay.vue' import type { RouteLocationRaw } from 'vue-router' +import type { StyleValue } from 'vue' /** * Utils imports */ @@ -38,22 +42,22 @@ import { promisedWait } from '@/utils/promise' import { range } from '@/utils/numbers' /** - * A group of nucleotides to highlight on the sequence. + * An object to highlight on the sequence. */ -export interface highlightGroupModel { - /** First position to highlight (starts at 1). */ +export interface objectModel { + /** Start position of the object (starts at 1). */ start: number - /** Last position to highlight (included). */ + /** End position of the object (included). */ end: number - /** Color(s) to use to highlight the group. */ + /** Color(s) to use to highlight the object. */ color: TailwindDefaultColorNameModel - /** Group name. */ + /** Object name. */ name?: string - /** Group type. */ + /** Object type. */ type?: string - /** Link to go to when clicking on the group. */ + /** Link to go to when clicking on the object. */ link?: RouteLocationRaw - /** Wether to show a tooltip on hover on this group. */ + /** Wether to show a tooltip on hover on this object. */ shouldTooltip?: boolean } @@ -77,9 +81,9 @@ const props = defineProps<{ sequenceId: string /** The sequence to represent. */ sequence: string - /** Groups to highlight on the sequence. */ - highlightedGroups?: { [groupId: string]: highlightGroupModel } - /** The items of the legend to display for the highlighted groups. */ + /** Objects to highlight on the sequence. */ + objects?: { [objectId: string]: objectModel } + /** The items of the legend to display for the objects. */ legendItems?: LegendItemModel[] }>() @@ -105,12 +109,12 @@ defineSlots<{ 'legend-item-description': (props: { /** The item of which to customise the description. */ item: LegendItemModel }) => any - /** Custom tooltip group items template */ + /** Custom tooltip object items template */ 'tooltip-item': (props: { - /** The group of the item to customise */ - group: highlightGroupModel - /** The ID of the group of the item to customise */ - groupId: string + /** The object of the item to customise */ + object: objectModel + /** The ID of the object of the item to customise */ + objectId: string }) => any }>() @@ -161,93 +165,92 @@ const splitSequence = computed(() => ) /** - * Wether a nucleotide is in a given highlighted group or not. + * Wether a nucleotide is in a given object or not. * @param position The position of the nucleotide to check. - * @param groupId The id of the group to check for. - * @returns `true` if position if in the given group, `false` otherwise. + * @param objectId The id of the object to check for. + * @returns `true` if position if in the given object, `false` otherwise. */ -const isInHighlightedGroup = (position: number, groupId: string): boolean => { - const highlightedGroup = props.highlightedGroups?.[groupId] +const isInObject = (position: number, objectId: string): boolean => { + const object = props.objects?.[objectId] return ( - !!highlightedGroup && - _inRange(position, highlightedGroup.start, highlightedGroup.end + 1) + !!object && + _inRange(position, object.start, object.end + 1) ) } /** - * Gets the IDs of the highlighted groups containing a nucleotide. - * @param position The position of the nucleotide for which to get group IDs. - * @returns A list of the IDs of the highlighted groups containing the nucleotide. + * Gets the IDs of the objects containing a nucleotide. + * @param position The position of the nucleotide for which to get object IDs. + * @returns A list of the IDs of the objects containing the nucleotide. */ -const highlightedGroupIdsContaining = (position: number): string[] => - props.highlightedGroups - ? Object.keys(props.highlightedGroups).filter((groupId) => - isInHighlightedGroup(position, groupId) +const objectIdsContaining = (position: number): string[] => + props.objects + ? Object.keys(props.objects).filter((objectId) => + isInObject(position, objectId) ) : [] /** - * Gets the highlighted groups containing a nucleotide. - * @param position The position of the nucleotide for which to get group. - * @returns A list of tuples, `[group ID, group]` for each of the highlighted - * groups containing the nucleotide. + * Gets the objects containing a nucleotide. + * @param position The position of the nucleotide for which to get objects. + * @returns A list of tuples, `[object ID, object]` for each of the objects + * containing the nucleotide. */ -const highlightedGroupsContaining = ( +const objectsContaining = ( position: number -): [string, highlightGroupModel][] => - props.highlightedGroups - ? Object.entries(props.highlightedGroups).filter(([groupId]) => - isInHighlightedGroup(position, groupId) +): [string, objectModel][] => + props.objects + ? Object.entries(props.objects).filter(([objectId]) => + isInObject(position, objectId) ) : [] /** - * Whether a nucleotide is in any highlighted group. + * Whether a nucleotide is in any object. * @param position The position of the nucleotide to check. - * @returns `true` if the nucleotide is in any highlighted group, `false` otherwise. + * @returns `true` if the nucleotide is in any object, `false` otherwise. */ -const isInAnyHighlightedGroup = (position: number): boolean => - !!highlightedGroupIdsContaining(position).length +const isInAnyObject = (position: number): boolean => + !!objectIdsContaining(position).length /** - * Set of all the boundary positions (start & end) of all highlighted groups on + * Set of all the boundary positions (start & end) of all objects on * the sequence. */ -const highlightedGroupsBoundaryPositions = computed( +const objectsBoundaryPositions = computed( () => - props.highlightedGroups && - Object.values(props.highlightedGroups).reduce( - (highlightedGroupsBoundaryPositions, highlightedGroup) => - highlightedGroupsBoundaryPositions - .add(highlightedGroup.start) - .add(highlightedGroup.end), + props.objects && + Object.values(props.objects).reduce( + (objectsBoundaryPositions, object) => + objectsBoundaryPositions + .add(object.start) + .add(object.end), new Set<number>() ) ) /** - * Whether a nucleotide is a boundary position (start or end) of an highlighted - * group. + * Whether a nucleotide is a boundary position (start or end) of an object. * @param position The position of the nucleotide to check. - * @returns `true` if the nucleotide is the start of an highlighted group, `false` otherwise. + * @returns `true` if the nucleotide is the start of an object, `false` otherwise. */ -const isHighlightedGroupBoundary = (position: number): boolean => - !!highlightedGroupsBoundaryPositions.value?.has(position) +const isObjectBoundary = (position: number): boolean => + !!objectsBoundaryPositions.value?.has(position) /** * Composes the Tailwind color name in which to paint a nucleotide based on its * position. * @param position The position for which to compose the color. - * @returns `slate` if there is no or several boxes at the given position, the - * Tailwind color name of the box if there is only one. + * @returns `slate` if there is no or several object at the given position, the + * Tailwind color name of the object if there is only one. */ -const composeHighlightedPositionColor = ( +const composeInObjectPositionColor = ( position: number ): TailwindDefaultColorNameModel => { - const highlightedGroupIds = highlightedGroupIdsContaining(position) - // If only one group, use its color, otherwise, use 'slate' - return highlightedGroupIds.length === 1 && highlightedGroupIds[0] - ? props.highlightedGroups?.[highlightedGroupIds[0]]?.color || 'slate' + const objectIds = objectIdsContaining(position) + // If only one object, use its color, otherwise, use 'slate' + return objectIds.length === 1 && objectIds[0] + ? props.objects?.[objectIds[0]]?.color || 'slate' : 'slate' } @@ -264,18 +267,9 @@ const sequenceContainerElement = ref<HTMLElement>() */ const sequenceGroupElements = ref<HTMLElement[]>([]) /** - * Labels of the positions of the first nucleotide of each group DOM `HTMLElement`. - * array - */ -const groupsPositionsElements = ref<HTMLElement[]>([]) -/** - * First & last nucleotides of each box DOM `HTMLElement` array. + * First & last nucleotides of each object DOM `HTMLElement` array. */ -const boxBoundaryElements = ref<HTMLElement[]>([]) -/** - * Boxes DOM `HTMLElement` array. - */ -const boxElements = ref<HTMLElement[]>([]) +const objectBoundaryElements = ref<HTMLElement[]>([]) /** * Last split sequence group DOM `HTMLElement`. @@ -342,44 +336,52 @@ const lineHeight = computed( ) /** - * Dictionary of the number of line on which each highlighted group spreads, - * by group ID. + * Dictionary of the number of line on which each object spreads, by object ID. */ -const highlightedGroupsLineCount = ref<{ [groupId: string]: number }>() +const objectsLineCount = ref<{ + [objectId: string]: number | undefined +}>() /** - * Updates the number of line on which each highlighted group spreads. - * @returns The updated number of line on which each highlighted group spreads. + * Updates the number of line on which each object spreads. + * @returns The updated number of line on which each object spreads. * @description Retrieves the line count by getting the difference between the - * top of the last nucleotide of the box and the top of the first one, and + * top of the last nucleotide of the object and the top of the first one, and * dividing by the line height. */ -const updateHighlightedGroupsLineCount = - (): typeof highlightedGroupsLineCount.value => - (highlightedGroupsLineCount.value = - props.highlightedGroups && - Object.entries(props.highlightedGroups).reduce( - (highlightedGroupsLineCount, [groupId, group]) => ({ - ...highlightedGroupsLineCount, - [groupId]: - lineHeight.value && - Math.round( - (( - boxBoundaryElements.value.find( - (element) => element.dataset.position === group.end.toString() - )?.offsetParent as HTMLElement - )?.offsetTop - - ( - boxBoundaryElements.value.find( - (element) => - element.dataset.position === group.start.toString() - )?.offsetParent as HTMLElement - )?.offsetTop) / - lineHeight.value - ) + 1 - }), - {} - )) +const updateObjectsLineCount = () => + (objectsLineCount.value = _mapValues( + props.objects, + (object) => { + // Parent DOM elements of the first and last nucleotide of the object + const objectStartNucleotideElementParent = objectBoundaryElements.value.find( + (element) => + element.dataset.position === object.start.toString() + )?.offsetParent + const objectEndNucleotideElementParent = objectBoundaryElements.value.find( + (element) => + element.dataset.position === object.end.toString() + )?.offsetParent + + if ( + !( + objectStartNucleotideElementParent instanceof HTMLElement && + objectEndNucleotideElementParent instanceof HTMLElement + ) + ) { + return undefined + } + + return ( + lineHeight.value && + Math.round( + (objectEndNucleotideElementParent.offsetTop - + objectStartNucleotideElementParent.offsetTop) / + lineHeight.value + ) + 1 + ) + } + )) /** * Width of the sequence container. @@ -426,120 +428,17 @@ const updateSequenceContainerContentWidth = } /** - * Updates all the DOM positions & sizes. + * Updates all non-reactive values. */ const update = (): void => { - updateHighlightedGroupsLineCount() + updateObjectsLineCount() updateSequenceContainerContentWidth() } /** - * Redraws the nucleotide positions of the start of each split sequence groups. - */ -const redrawGroupNumbers = (): void => { - groupsPositionsElements.value.forEach((groupNumberElement) => { - const matchingSequenceGroupElement = sequenceGroupElements.value.find( - (element) => { - return ( - element.dataset.groupIndex === groupNumberElement.dataset.groupIndex - ) - } - ) - - groupNumberElement.style.left = `${ - matchingSequenceGroupElement?.offsetLeft || 0 - }px` - groupNumberElement.style.top = `calc(${ - matchingSequenceGroupElement?.offsetTop || 0 - }px - 1.25rem)` - }) -} - -/** - * Redraws the boxes of highlighted groups. - */ -const redrawBoxes = (): void => { - boxElements.value.forEach((boxElement) => { - const matchingBoxFirstNucleotideElement = boxBoundaryElements.value.find( - (element) => element.dataset.position === boxElement.dataset.start - ) - const matchingBoxLastNucleotideElement = boxBoundaryElements.value.find( - (element) => element.dataset.position === boxElement.dataset.end - ) - - if ( - !matchingBoxFirstNucleotideElement?.offsetParent || - !matchingBoxLastNucleotideElement?.offsetParent - ) { - return - } - - const matchingBoxFirstNucleotideElementTop = - (matchingBoxFirstNucleotideElement?.offsetParent as HTMLElement) - .offsetTop + (matchingBoxFirstNucleotideElement?.offsetTop || 0) - const matchingBoxFirstNucleotideElementLeft = - (matchingBoxFirstNucleotideElement?.offsetParent as HTMLElement) - .offsetLeft + (matchingBoxFirstNucleotideElement?.offsetLeft || 0) - const matchingBoxLastNucleotideElementLeft = - (matchingBoxLastNucleotideElement?.offsetParent as HTMLElement) - .offsetLeft + (matchingBoxLastNucleotideElement?.offsetLeft || 0) - const matchingBoxLastNucleotideElementRight = - matchingBoxLastNucleotideElementLeft + - (matchingBoxLastNucleotideElement?.offsetWidth || 0) - - Array.from(boxElement.children as HTMLCollectionOf<HTMLElement>).forEach( - (boxFragmentElement, boxFragmentIndex) => { - boxFragmentElement.style.top = `${ - matchingBoxFirstNucleotideElementTop + - boxFragmentIndex * (lineHeight.value || 0) - }px` - - const boxFragmentElementLeft = - boxFragmentIndex === 0 ? matchingBoxFirstNucleotideElementLeft : 0 - - boxFragmentElement.style.left = - boxFragmentIndex === 0 - ? `calc(${boxFragmentElementLeft}px - 0.125rem)` - : '0' - - boxFragmentElement.style.width = `calc(${ - boxFragmentIndex === boxElement.children.length - 1 - ? boxFragmentIndex === 0 - ? `${ - matchingBoxLastNucleotideElementRight - - matchingBoxFirstNucleotideElementLeft - }px` - : `${matchingBoxLastNucleotideElementRight}px` - : `${ - (sequenceContainerContentWidth.value || 0) - - boxFragmentElementLeft - }px` - } + 0.25rem)` - } - ) - }) -} - -/** - * Redraws every absolutely positioned objects. - */ -const redraw = (): void => { - redrawGroupNumbers() - redrawBoxes() -} - -/** - * Updates values then redraws every absolutely positioned objects. - */ -const updateAndRedraw = (): void => { - update() - redraw() -} - -/** - * Exposes methods to manually trigger update & redraw of boxes. + * Exposes methods to manually trigger updates. */ -defineExpose({ update, redraw, updateAndRedraw }) +defineExpose({ update }) /** * Whether the sequence is visible or not. @@ -549,49 +448,156 @@ const isSequenceContainerElementVisible = useElementVisibility( ) /** - * Sets hook to update and (re)draw on component mount. + * Sets hook to update on component mount. */ -onMounted(updateAndRedraw) +onMounted(update) /** - * Sets an observer to update and redraw on container resize. + * Sets an observer to update on container resize. */ useResizeObserver(sequenceContainerElement, () => { - updateAndRedraw() + update() setTimeout(() => { - updateAndRedraw() + update() }, 100) }) /** - * Sets a watcher to update and redraw when resuming sequence visibility. + * Sets a watcher to update when resuming sequence visibility. */ watch( [isSequenceContainerElementVisible, () => props.sequenceId], ([isSequenceContainerElementVisible, newSequenceId], [, oldSequenceId]) => { if (isSequenceContainerElementVisible || newSequenceId !== oldSequenceId) { setTimeout(() => { - updateAndRedraw() + update() }, 100) } } ) /** - * The highlighted boxes currently being hovered. + * Bounding box of sequence container DOM `HTMLElement`. */ -const hoveredHighlightedGroups = ref(new Set<[string, highlightGroupModel]>()) +const sequenceContainerElementBounding = useElementBounding( + sequenceContainerElement +) /** - * The highlighted boxes currently being hovered. + * For each sequence group first position label, the bounding box of its + * reference sequence group DOM element. */ -const lockedTooltipHighlightedGroups = ref<Set<[string, highlightGroupModel]>>() +const groupNumberReferenceSequenceGroupElementsBounding = computed(() => + Array.from({ length: nucleotideGroupsCount.value }, (_, groupNumberIndex) => + useElementBounding( + sequenceGroupElements.value.find( + (sequenceGroupElement) => + sequenceGroupElement.dataset.groupIndex === + groupNumberIndex.toString() + ) + ) + ) +) + +/** + * The style to apply to the sequence group first position label DOM element. + */ +const groupNumberStyles = computed(() => + groupNumberReferenceSequenceGroupElementsBounding.value.map<StyleValue>( + (groupNumberReferenceSequenceGroupElementBounding) => ({ + left: `${ + groupNumberReferenceSequenceGroupElementBounding.left.value - + sequenceContainerElementBounding.left.value + }px`, + top: `calc(${ + groupNumberReferenceSequenceGroupElementBounding.top.value - + sequenceContainerElementBounding.top.value + }px - 1.25rem)` + }) + ) +) + +/** + * The style to apply to each fragment of each of the object boxes. + */ +const objectBoxStyles = computed(() => + // Map objects to the styles of each of their fragment + _mapValues(props.objects, (object, objectId) => { + // DOM element of the first and last nucleotides of the object + const objectStartNucleotideElement = objectBoundaryElements.value.find( + (element) => + element.dataset.position === object.start.toString() + ) + const objectEndNucleotideElement = objectBoundaryElements.value.find( + (element) => element.dataset.position === object.end.toString() + ) + + if ( + !( + objectStartNucleotideElement?.offsetParent instanceof HTMLElement && + objectEndNucleotideElement?.offsetParent instanceof HTMLElement + ) + ) { + return + } + + // Absolute positions of the DOM elements of the first and last nucleotide + // of the object + const objectStartNucleotideElementTop = + objectStartNucleotideElement.offsetParent.offsetTop + + (objectStartNucleotideElement?.offsetTop || 0) + const objectStartNucleotideElementLeft = + objectStartNucleotideElement.offsetParent.offsetLeft + + (objectStartNucleotideElement?.offsetLeft || 0) + const objectEndNucleotideElementLeft = + objectEndNucleotideElement.offsetParent.offsetLeft + + (objectEndNucleotideElement.offsetLeft || 0) + const objectEndNucleotideElementRight = + objectEndNucleotideElementLeft + + (objectEndNucleotideElement.offsetWidth || 0) + + // Number of fragment in which the current object box is divided + const objectBoxFragmentsCount = + objectsLineCount.value?.[objectId] || 0 + + // Array of length equals # of fragments, containing style for each of them + return Array.from( + { length: objectBoxFragmentsCount }, + (_, fragmentIndex): StyleValue => ({ + top: `${ + objectStartNucleotideElementTop + + fragmentIndex * (lineHeight.value || 0) + }px`, + left: + fragmentIndex === 0 + ? `calc(${objectStartNucleotideElementLeft}px - 0.125rem)` + : '0', + right: `calc(${ + (sequenceContainerElementBounding.width.value || 0) - + (fragmentIndex === objectBoxFragmentsCount - 1 + ? objectEndNucleotideElementRight + : sequenceContainerContentWidth.value || 0) + }px - 0.125rem)` + }) + ) + }) +) + +/** + * The objects currently being hovered. + */ +const hoveredObjects = ref(new Set<[string, objectModel]>()) + +/** + * The objects currently being hovered. + */ +const lockedTooltipObjects = ref<Set<[string, objectModel]>>() /** * DOM elements of objects to display in the tooltip. */ -const tooltipHighlightedGroups = computed( - () => lockedTooltipHighlightedGroups.value || hoveredHighlightedGroups.value +const tooltipObjects = computed( + () => lockedTooltipObjects.value || hoveredObjects.value ) /** @@ -603,7 +609,7 @@ const tooltipComponent = ref<InstanceType<typeof BaseLockableTooltip>>() * Locks the tooltip and the values to display in it. */ const lockTooltip = () => { - lockedTooltipHighlightedGroups.value = new Set(hoveredHighlightedGroups.value) + lockedTooltipObjects.value = new Set(hoveredObjects.value) tooltipComponent.value?.lockTooltipIfUnlocked() } </script> @@ -617,7 +623,7 @@ const lockTooltip = () => { v-model="nucleotideGroupsSize" :options="[5, 10, 20, 50, 100]" class="mt-2 w-52" - @hide="updateAndRedraw" + @hide="update" > <template #option="{ option }">{{ option }} nucleotides</template> <template #value="{ value }">{{ value }} nucleotides</template> @@ -711,59 +717,57 @@ const lockTooltip = () => { > <span v-if=" - isHighlightedGroupBoundary( + isObjectBoundary( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 ) " - ref="boxBoundaryElements" + ref="objectBoundaryElements" :data-position=" nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 " :class="[ 'relative mx-[.0625rem] border-2 border-transparent px-0.5 pt-0.5', - isInAnyHighlightedGroup( - nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 - ) && [ - `text-${composeHighlightedPositionColor( + [ + `text-${composeInObjectPositionColor( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 )}-600`, - 'modification font-semibold' + 'font-semibold' ] ]" > <span v-if=" - highlightedGroupsContaining( + objectsContaining( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 ).find( - ([_highlightedGroupId, highlightedGroup]) => - highlightedGroup.shouldTooltip + ([, object]) => + object.shouldTooltip ) " :class="[ 'absolute -bottom-0.5 -left-[0.1875rem] -right-[0.1875rem] -top-0.5', - highlightedGroupsContaining( + objectsContaining( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 ).find( - ([_highlightedGroupId, highlightedGroup]) => - highlightedGroup.link + ([, object]) => + object.link ) ? 'cursor-pointer' : 'cursor-help' ]" @mouseenter=" - highlightedGroupsContaining( + objectsContaining( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 ).forEach( - ([highlightedGroupId, highlightedGroup]) => - highlightedGroup.shouldTooltip && - hoveredHighlightedGroups.add([ - highlightedGroupId, - highlightedGroup + ([objectId, object]) => + object.shouldTooltip && + hoveredObjects.add([ + objectId, + object ]) ) " - @mouseleave="hoveredHighlightedGroups.clear()" + @mouseleave="hoveredObjects.clear()" @click="lockTooltip" />{{ nucleotide }} </span> @@ -771,49 +775,49 @@ const lockTooltip = () => { v-else :class="[ 'relative mx-[.0625rem] border-2 border-transparent px-0.5 pt-0.5', - isInAnyHighlightedGroup( + isInAnyObject( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 ) && [ - `text-${composeHighlightedPositionColor( + `text-${composeInObjectPositionColor( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 )}-600`, - 'modification font-semibold' + 'font-semibold' ] ]" > <span v-if=" - highlightedGroupsContaining( + objectsContaining( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 ).find( - ([_highlightedGroupId, highlightedGroup]) => - highlightedGroup.shouldTooltip + ([, object]) => + object.shouldTooltip ) " :class="[ 'absolute -bottom-0.5 -left-[0.1875rem] -right-[0.1875rem] -top-0.5', - highlightedGroupsContaining( + objectsContaining( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 ).find( - ([_highlightedGroupId, highlightedGroup]) => - highlightedGroup.link + ([, object]) => + object.link ) ? 'cursor-pointer' : 'cursor-help' ]" @mouseenter=" - highlightedGroupsContaining( + objectsContaining( nucleotideIndex + groupIndex * nucleotideGroupsSize + 1 ).forEach( - ([highlightedGroupId, highlightedGroup]) => - highlightedGroup.shouldTooltip && - hoveredHighlightedGroups.add([ - highlightedGroupId, - highlightedGroup + ([objectId, object]) => + object.shouldTooltip && + hoveredObjects.add([ + objectId, + object ]) ) " - @mouseleave="hoveredHighlightedGroups.clear()" + @mouseleave="hoveredObjects.clear()" @click="lockTooltip" />{{ nucleotide }} </span> @@ -823,86 +827,83 @@ const lockTooltip = () => { <span v-for="(_sequenceGroup, groupIndex) in splitSequence" :key="groupIndex" - ref="groupsPositionsElements" class="absolute select-none text-sm text-slate-400" - :data-group-index="groupIndex" + :style="groupNumberStyles[groupIndex]" > {{ groupIndex * nucleotideGroupsSize + 1 }} </span> <span - v-for="[highlightedGroupId, highlightedGroup] in Object.entries( - highlightedGroups || {} + v-for="[objectId, object] in Object.entries( + objects || {} )" - :key="highlightedGroupId" - ref="boxElements" - :data-start="highlightedGroup.start" - :data-end="highlightedGroup.end" - class="highlight-group" + :key="objectId" + class="object" > <span - v-for="highlightedGroupLine in range( - highlightedGroupsLineCount?.[highlightedGroupId] || 0 + v-for="fragmentIndex in range( + objectsLineCount?.[objectId] || 0 )" - :key="highlightedGroupLine" + :key="fragmentIndex" + :style="objectBoxStyles?.[objectId]?.[fragmentIndex]" :class="[ - 'highlight-group-fragment absolute h-8 border-y-2 bg-opacity-50 px-0.5 text-2xl mix-blend-multiply', + 'object-fragment absolute h-8 border-y-2 bg-opacity-50 px-0.5 text-2xl mix-blend-multiply', [ - `!border-${highlightedGroup.color}-600`, - `bg-${highlightedGroup.color}-100` + `!border-${object.color}-600`, + `bg-${object.color}-100` ], { - 'rounded-l-xl border-l-2': highlightedGroupLine === 0, + 'rounded-l-xl border-l-2': fragmentIndex === 0, 'rounded-r-xl border-r-2': - highlightedGroupsLineCount && - highlightedGroupLine + 1 === - highlightedGroupsLineCount[highlightedGroupId] + objectsLineCount && + fragmentIndex + 1 === + objectsLineCount[objectId] }, - highlightedGroup.shouldTooltip && - (highlightedGroup.link ? 'cursor-pointer' : 'cursor-help') + object.shouldTooltip && + (object.link ? 'cursor-pointer' : 'cursor-help') ]" /> </span> <BaseLockableTooltip ref="tooltipComponent" - :show="!!hoveredHighlightedGroups.size" + :show="!!hoveredObjects.size" class="z-20 max-w-min font-sans text-base shadow-xl" - @unlock="lockedTooltipHighlightedGroups = undefined" + @unlock="lockedTooltipObjects = undefined" > <ul class="flex flex-col gap-1"> <li - v-for="(highlightedGroup, index) in tooltipHighlightedGroups" + v-for="(object, index) in tooltipObjects" :key="index" > <RouterLink - v-if="highlightedGroup[1].link" - :to="highlightedGroup[1].link" + v-if="object[1].link" + :to="object[1].link" :class="[ - `text-${highlightedGroup[1].color}-600`, + `text-${object[1].color}-600`, 'whitespace-nowrap' ]" > <slot name="tooltip-item" - :group="highlightedGroup[1]" - :group-id="highlightedGroup[0]" + :object="object[1]" + :object-id="object[0]" > - {{ highlightedGroup[1].name || highlightedGroup[1].link }} + {{ object[1].name || object[1].link }} {{ - highlightedGroup[1].type && ` - ${highlightedGroup[1].type}` + object[1].type && ` - ${object[1].type}` }} </slot> </RouterLink> <span v-else> <slot name="tooltip-item" - :group="highlightedGroup[1]" - :group-id="highlightedGroup[0]" + :object="object[1]" + :object-id="object[0]" > - {{ highlightedGroup[1].name || highlightedGroup[1].link }} + {{ object[1].name || object[1].link }} {{ - highlightedGroup[1].type && ` - ${highlightedGroup[1].type}` + object[1].type && ` - ${object[1].type}` }} </slot> </span> @@ -917,10 +918,10 @@ const lockTooltip = () => { </template> <style lang="scss"> -// .sequence-parent:has(.highlight-group:hover) { +// .sequence-parent:has(.object:hover) { // // color: var(--gray-300); // -// .highlight-group-fragment { +// .object-fragment { // border-color: var(--gray-300); // background-color: var(--gray-100); // } diff --git a/src/composables/useConservationAnalysis.ts b/src/composables/useConservationAnalysis.ts new file mode 100644 index 0000000000000000000000000000000000000000..9d64a7f38dbfdecbf097f7a004bfd5dc2febc59a --- /dev/null +++ b/src/composables/useConservationAnalysis.ts @@ -0,0 +1,226 @@ +/** + * Vue imports + */ +import { computed, toRef, type MaybeRef, type ComputedRef } from 'vue' +/** + * Other 3rd-party imports + */ +import { groupBy as _groupBy, pick as _pick } from 'lodash-es' +/** + * Types imports + */ +import type { + AlignmentObjectWithTrackIdModel, + AlignmentTrackModel +} from '@/components/SequenceAlignment.vue' +/** + * Utils imports + */ +import { isDefined, isNotEmpty } from '@/typings/typeUtils' + +/** + * A conservation analysis type. + */ +export enum ConservationAnalysisTypesEnum { + /** + * Visualise modifications present **only** on **all** the selected + * sequences + */ + Specific = 0, + /** + * Visualise modifications present **at least** on **all** the selected + * sequences + */ + Common +} + +/** + * Configuration for the conservation analysis. + */ +export interface ConservationAnalysisConfigModel { + /** The IDs of the tracks selected for conservation analysis. */ + trackIds: string[] + /** The object types selected for conservation analysis. */ + objectTypes: string[] + /** The type of conservation analysis to perform + * See {@link ConservationAnalysisTypesEnum} for the description of each mode. */ + analysisType: ConservationAnalysisTypesEnum | null +} + +/** + * Reactive conservation analysis between sequences (mainly track) + * @param tracks The tracks of the alignment on which to perform the + * conservation analysis (contains selected AND unselected tracks, to be able to + * perform the *Specific* analysis) + * @param selectedAnalysisType + */ +export const useConservationAnalysis = ( + tracks: MaybeRef<AlignmentTrackModel[]>, + config: MaybeRef<ConservationAnalysisConfigModel | null> +): { + /** The track representing the conservation analysis in the alignment. */ + track: ComputedRef<AlignmentTrackModel> +} => { + /** + * All the modifications present on the reference track sequences. + */ + const referenceTrackModifications = computed< + AlignmentObjectWithTrackIdModel[] + >(() => + toRef(tracks) + .value.filter((track) => track.isReference) + .map( + (track) => + track.objects && + Object.values(track.objects).map((object) => ({ + ...object, + trackId: track.id + })) + ) + .flat() + .filter(isDefined) + ) + + /** + * Modifications present on selected track sequences. + */ + const selectedModifications = computed(() => + referenceTrackModifications.value.filter((modification) => + toRef(config).value?.trackIds.includes(modification.trackId) + ) + ) + + /** + * Modifications present on selected track sequences, filtered according to the + * currently selected mode (`selectedConservationAnalysisType`) & modification + * type(`selectedConservationAnalysisObjectTypes`). + * + * See {@link ConservationAnalysisTypesEnum} for the description of each mode. + */ + const filteredSelectedModifications = computed(() => { + if (toRef(config).value?.analysisType == null) { + return [] + } + return ( + // Group modifications by position & type + Object.values( + _groupBy( + selectedModifications.value, + (selectedModification) => + `${selectedModification.start}-${selectedModification.end}:${selectedModification.type}` + ) + ) + // Keep only modifications when present on every track selected for + // analysis (i.e. the number of modifications on a position is the same + // as the number of selected tracks) + .filter(isNotEmpty) + .filter( + (selectedModificationsByPosition) => + selectedModificationsByPosition.length === + toRef(config).value?.trackIds.length + ) + // Keep only the modification informations common to all selected tracks + // for each modification (reduce modification arrays to a single one for + // each) + .map((selectedModificationsByPosition) => { + return _pick< + (typeof selectedModificationsByPosition)[0], + 'start' | 'end' | 'color' | 'type' | 'trackId' + >(selectedModificationsByPosition[0], [ + 'start', + 'end', + 'color', + 'type', + 'trackId' + ]) + }) + // Keep only modifications of the selected types + .filter( + (selectedModification) => + selectedModification.type && + toRef(config).value?.objectTypes.includes(selectedModification.type) + ) + // If in "specific" analysis mode, keep only modification if absent on + // every non-selected track (i.e. there is no modification with same + // positions on a non-selected track) + .filter((selectedModification) => + toRef(config).value?.analysisType === + ConservationAnalysisTypesEnum.Specific + ? !toRef(referenceTrackModifications).value.find( + (referenceTrackModification) => + // Using a cross-organism ID here would avoid matching 2 different + // modifications taking place at the same position + referenceTrackModification.start === + selectedModification?.start && + referenceTrackModification.end === + selectedModification?.end && + !toRef(config).value?.trackIds.includes( + referenceTrackModification.trackId + ) + ) + : selectedModification + ) + ) + }) + + /** + * Selected tracks for conservation analysis. + */ + const selectedTracks = computed<AlignmentTrackModel[]>( + () => + toRef(config) + .value?.trackIds.map((selectedTrackId) => + toRef(tracks).value.find((track) => track.id === selectedTrackId) + ) + .filter(isDefined) || [] + ) + + /** + * If a conservation analysis is selected, the track to display to represent the + * analysis. + */ + const conservationAnalysisTrack = computed<AlignmentTrackModel>(() => { + const objects: { + [objectId: string]: (typeof filteredSelectedModifications.value)[number] + } = {} + + const sequence = Array.from( + { + length: selectedTracks.value?.[0]?.sequence.length || 0 + }, + (_, index) => { + const matchingModification = filteredSelectedModifications.value.find( + (modification) => modification?.start === index + 1 + ) + + // No conserved modification at current position + if (!matchingModification) { + return '-' + } + + // Add modification to object list of conservation analysis track + objects[`CONS_OBJ_${index}`] = { + ...matchingModification, + trackId: 'CONSERVATION_TRACK' + } + + const modificationNucleotide = toRef(tracks).value.find( + (track) => track.id === matchingModification.trackId + )?.sequence[index] + + return modificationNucleotide || '↯' + } + ).join('') + + return { + id: 'CONSERVATION_TRACK', + name: 'Conservation analysis', + shortname: 'Conservation', + sequence, + isInformation: true, + objects + } + }) + + return { track: conservationAnalysisTrack } +} diff --git a/src/gql/codegen/gql.ts b/src/gql/codegen/gql.ts index 1cdd8fd0b5c5297451891af1c63484e3dfa739b3..6e7d7be3ef22375a79e768db653a8cd7c8a8bc56 100644 --- a/src/gql/codegen/gql.ts +++ b/src/gql/codegen/gql.ts @@ -22,6 +22,8 @@ const documents = { "\n query guideByIdQuery($id: ID) {\n guides(where: { id: $id }) {\n id\n name\n altnames\n description\n length\n class\n subclass_label\n chromosomeConnection: parentConnection(\n where: { node: { graphql_type: Chromosome } }\n ) {\n edges {\n properties {\n start\n end\n strand\n }\n node {\n id\n name\n length\n graphql_type\n }\n }\n }\n host_genes\n seq\n genome {\n organism {\n id\n label\n }\n }\n boxConnections: featuresConnection(\n where: { NOT: { node: { class: DuplexFragment } } }\n ) {\n edges {\n properties {\n ... on HasFeature {\n start\n end\n }\n }\n node {\n id\n annotation\n }\n }\n }\n modifications(options: { sort: [{ position: ASC }] }) {\n id\n name\n type\n }\n modificationsAggregate {\n count\n }\n interactions {\n duplexes {\n primaryStrandsConnection: strandsConnection(\n where: { edge: { primary: true } }\n ) {\n edges {\n properties {\n start\n end\n primary\n }\n node {\n seq\n parentConnection {\n edges {\n properties {\n start\n end\n }\n }\n }\n }\n }\n }\n secondaryStrandsConnection: strandsConnection(\n where: { edge: { primary: false } }\n ) {\n edges {\n properties {\n start\n end\n primary\n }\n node {\n seq\n parentConnection {\n edges {\n properties {\n start\n end\n }\n }\n }\n }\n }\n }\n index\n }\n modification {\n id\n name\n position\n symbol\n symbol_label\n type\n type_short_label\n }\n target {\n id\n name\n class\n unit\n }\n }\n cluster {\n id\n }\n clusterAggregate {\n count\n }\n isoform {\n guides {\n id\n name\n }\n guidesAggregate {\n count\n }\n }\n isoformAggregate {\n count\n }\n chebi_id\n so_id\n }\n\n targets(\n where: { modifications_SOME: { guides_SOME: { id: $id } } }\n options: { sort: [{ name: ASC }] }\n ) {\n id\n name\n class\n unit\n }\n }\n": types.GuideByIdQueryDocument, "\n query targetByIdQuery($id: ID) {\n targets(where: { id: $id }) {\n id\n name\n altnames\n description\n length\n class\n unit\n chromosomeConnection: parentConnection(\n where: { node: { graphql_type: Chromosome } }\n ) {\n edges {\n properties {\n start\n end\n strand\n }\n node {\n name\n graphql_type\n }\n }\n }\n seq\n genome {\n organism {\n id\n label\n }\n }\n modifications(options: { sort: [{ position: ASC }] }) {\n id\n name\n position\n type\n symbol\n }\n modificationsAggregate {\n count\n }\n interactions {\n duplexes {\n primaryStrandsConnection: strandsConnection(\n where: { edge: { primary: true } }\n ) {\n edges {\n properties {\n start\n end\n primary\n }\n node {\n seq\n parentConnection {\n edges {\n properties {\n start\n end\n }\n }\n }\n }\n }\n }\n secondaryStrandsConnection: strandsConnection(\n where: { edge: { primary: false } }\n ) {\n edges {\n properties {\n start\n end\n primary\n }\n node {\n seq\n parentConnection {\n edges {\n properties {\n start\n end\n }\n }\n }\n }\n }\n }\n index\n }\n modification {\n id\n name\n position\n symbol\n symbol_label\n type\n type_short_label\n }\n guide {\n id\n name\n subclass_label\n class\n }\n }\n chebi_id\n so_id\n url\n secondary_struct_file\n }\n\n guides(\n where: { modifications_SOME: { target: { id: $id } } }\n options: { sort: [{ id: ASC }] }\n ) {\n id\n name\n class\n chromosome: parent(where: { graphql_type: Chromosome }) {\n name\n }\n }\n }\n": types.TargetByIdQueryDocument, "\n query clusterByIdQuery($id: ID) {\n clusters(where: { id: $id }) {\n id\n guides(options: { sort: [{ id: ASC }] }) {\n id\n name\n class\n subclass_label\n modificationsAggregate {\n count\n }\n chromosomeConnection: parentConnection(\n where: { node: { graphql_type: Chromosome } }\n ) {\n edges {\n properties {\n start\n end\n }\n }\n }\n }\n guidesAggregate {\n count\n }\n referenceGuide: guides(options: { limit: 1 }) {\n chromosome: parent(where: { graphql_type: Chromosome }) {\n id\n name\n }\n genome {\n organism {\n label\n id\n }\n }\n }\n }\n }\n": types.ClusterByIdQueryDocument, + "\n query targetAlignmentQuery($targetName: String, $organismIds: [Int!]) {\n targetBase: targets {\n name\n unit\n genome {\n organism {\n id\n label\n }\n }\n }\n\n organismsBase: organisms {\n label\n id\n }\n\n organisms(\n where: {\n tableEntries_SOME: { modification: { target: { name: $targetName } } }\n }\n ) {\n id\n }\n\n selectableTargets: targets(\n where: {\n name: $targetName\n genome: { organism: { id_IN: $organismIds } }\n }\n ) {\n id\n genome {\n organism {\n shortlabel\n }\n }\n }\n }\n": types.TargetAlignmentQueryDocument, + "\n query guideAlignmentQuery(\n $guideSubclasses: [GuideClass!]\n $guideName: String\n $organismIds: [Int!]\n ) {\n guideBase: guides {\n name\n subclass\n subclass_label\n genome {\n organism {\n id\n }\n }\n }\n\n guideNamesFilteredBySubclass: guides(\n where: { subclass_IN: $guideSubclasses }\n ) {\n name\n genome {\n organism {\n id\n }\n }\n }\n\n organismsBase: organisms {\n label\n id\n }\n\n organisms(where: { tableEntries_SOME: { guide: { name: $guideName } } }) {\n id\n }\n\n selectableGuides: guides(\n where: { name: $guideName, genome: { organism: { id_IN: $organismIds } } }\n ) {\n id\n genome {\n organism {\n shortlabel\n }\n }\n }\n }\n": types.GuideAlignmentQueryDocument, "\n query databaseStatisticsQuery {\n organismsAggregate {\n count\n }\n\n allModifications: modifications {\n type\n }\n\n allModificationsCount: modificationsAggregate {\n count\n }\n\n nonOrphanModificationsCount: modificationsAggregate(\n where: { guidesAggregate: { count_GT: 0 } }\n ) {\n count\n }\n\n allGuides: guides {\n subclass\n }\n\n allGuidesCount: guidesAggregate {\n count\n }\n\n nonOrphanGuidesCount: guidesAggregate(\n where: { modificationsAggregate: { count_GT: 0 } }\n ) {\n count\n }\n }\n": types.DatabaseStatisticsQueryDocument, "\n query legalDocumentListQuery {\n documents(where: { types_INCLUDES: \"legal\" }) {\n id\n menu_label\n }\n }\n": types.LegalDocumentListQueryDocument, "\n query documentByIdQuery($id: ID) {\n documents(where: { id: $id }) {\n id\n menu_label\n content\n }\n }\n": types.DocumentByIdQueryDocument, @@ -77,6 +79,14 @@ export function graphql(source: "\n query targetByIdQuery($id: ID) {\n targe * The graphql function is used to parse GraphQL queries into a document that can be used by GraphQL clients. */ export function graphql(source: "\n query clusterByIdQuery($id: ID) {\n clusters(where: { id: $id }) {\n id\n guides(options: { sort: [{ id: ASC }] }) {\n id\n name\n class\n subclass_label\n modificationsAggregate {\n count\n }\n chromosomeConnection: parentConnection(\n where: { node: { graphql_type: Chromosome } }\n ) {\n edges {\n properties {\n start\n end\n }\n }\n }\n }\n guidesAggregate {\n count\n }\n referenceGuide: guides(options: { limit: 1 }) {\n chromosome: parent(where: { graphql_type: Chromosome }) {\n id\n name\n }\n genome {\n organism {\n label\n id\n }\n }\n }\n }\n }\n"): (typeof documents)["\n query clusterByIdQuery($id: ID) {\n clusters(where: { id: $id }) {\n id\n guides(options: { sort: [{ id: ASC }] }) {\n id\n name\n class\n subclass_label\n modificationsAggregate {\n count\n }\n chromosomeConnection: parentConnection(\n where: { node: { graphql_type: Chromosome } }\n ) {\n edges {\n properties {\n start\n end\n }\n }\n }\n }\n guidesAggregate {\n count\n }\n referenceGuide: guides(options: { limit: 1 }) {\n chromosome: parent(where: { graphql_type: Chromosome }) {\n id\n name\n }\n genome {\n organism {\n label\n id\n }\n }\n }\n }\n }\n"]; +/** + * The graphql function is used to parse GraphQL queries into a document that can be used by GraphQL clients. + */ +export function graphql(source: "\n query targetAlignmentQuery($targetName: String, $organismIds: [Int!]) {\n targetBase: targets {\n name\n unit\n genome {\n organism {\n id\n label\n }\n }\n }\n\n organismsBase: organisms {\n label\n id\n }\n\n organisms(\n where: {\n tableEntries_SOME: { modification: { target: { name: $targetName } } }\n }\n ) {\n id\n }\n\n selectableTargets: targets(\n where: {\n name: $targetName\n genome: { organism: { id_IN: $organismIds } }\n }\n ) {\n id\n genome {\n organism {\n shortlabel\n }\n }\n }\n }\n"): (typeof documents)["\n query targetAlignmentQuery($targetName: String, $organismIds: [Int!]) {\n targetBase: targets {\n name\n unit\n genome {\n organism {\n id\n label\n }\n }\n }\n\n organismsBase: organisms {\n label\n id\n }\n\n organisms(\n where: {\n tableEntries_SOME: { modification: { target: { name: $targetName } } }\n }\n ) {\n id\n }\n\n selectableTargets: targets(\n where: {\n name: $targetName\n genome: { organism: { id_IN: $organismIds } }\n }\n ) {\n id\n genome {\n organism {\n shortlabel\n }\n }\n }\n }\n"]; +/** + * The graphql function is used to parse GraphQL queries into a document that can be used by GraphQL clients. + */ +export function graphql(source: "\n query guideAlignmentQuery(\n $guideSubclasses: [GuideClass!]\n $guideName: String\n $organismIds: [Int!]\n ) {\n guideBase: guides {\n name\n subclass\n subclass_label\n genome {\n organism {\n id\n }\n }\n }\n\n guideNamesFilteredBySubclass: guides(\n where: { subclass_IN: $guideSubclasses }\n ) {\n name\n genome {\n organism {\n id\n }\n }\n }\n\n organismsBase: organisms {\n label\n id\n }\n\n organisms(where: { tableEntries_SOME: { guide: { name: $guideName } } }) {\n id\n }\n\n selectableGuides: guides(\n where: { name: $guideName, genome: { organism: { id_IN: $organismIds } } }\n ) {\n id\n genome {\n organism {\n shortlabel\n }\n }\n }\n }\n"): (typeof documents)["\n query guideAlignmentQuery(\n $guideSubclasses: [GuideClass!]\n $guideName: String\n $organismIds: [Int!]\n ) {\n guideBase: guides {\n name\n subclass\n subclass_label\n genome {\n organism {\n id\n }\n }\n }\n\n guideNamesFilteredBySubclass: guides(\n where: { subclass_IN: $guideSubclasses }\n ) {\n name\n genome {\n organism {\n id\n }\n }\n }\n\n organismsBase: organisms {\n label\n id\n }\n\n organisms(where: { tableEntries_SOME: { guide: { name: $guideName } } }) {\n id\n }\n\n selectableGuides: guides(\n where: { name: $guideName, genome: { organism: { id_IN: $organismIds } } }\n ) {\n id\n genome {\n organism {\n shortlabel\n }\n }\n }\n }\n"]; /** * The graphql function is used to parse GraphQL queries into a document that can be used by GraphQL clients. */ diff --git a/src/gql/codegen/graphql.ts b/src/gql/codegen/graphql.ts index f6673f5d2b9fc6cc764985ac33b49d0af780dee3..29a5f235a76b7ff9f6cc47b14a4f9332b33c0a2f 100644 --- a/src/gql/codegen/graphql.ts +++ b/src/gql/codegen/graphql.ts @@ -9829,6 +9829,23 @@ export type ClusterByIdQueryQueryVariables = Exact<{ export type ClusterByIdQueryQuery = { __typename?: 'Query', clusters: Array<{ __typename?: 'Cluster', id: string, guides: Array<{ __typename?: 'Guide', id: string, name?: string | null, class: SequenceClass, subclass_label: string, modificationsAggregate?: { __typename?: 'GuideModificationModificationsAggregationSelection', count: number } | null, chromosomeConnection: { __typename?: 'GuideParentConnection', edges: Array<{ __typename?: 'GuideParentRelationship', properties: { __typename?: 'HasFeature', start: number, end: number } }> } }>, guidesAggregate?: { __typename?: 'ClusterGuideGuidesAggregationSelection', count: number } | null, referenceGuide: Array<{ __typename?: 'Guide', chromosome?: { __typename?: 'Chromosome', id: string, name?: string | null } | { __typename?: 'GenericSequence', id: string, name?: string | null } | { __typename?: 'Guide', id: string, name?: string | null } | { __typename?: 'Target', id: string, name?: string | null } | null, genome?: { __typename?: 'Genome', organism?: { __typename?: 'Organism', label: string, id: number } | null } | null }> }> }; +export type TargetAlignmentQueryQueryVariables = Exact<{ + targetName?: InputMaybe<Scalars['String']['input']>; + organismIds?: InputMaybe<Array<Scalars['Int']['input']> | Scalars['Int']['input']>; +}>; + + +export type TargetAlignmentQueryQuery = { __typename?: 'Query', targetBase: Array<{ __typename?: 'Target', name?: string | null, unit?: string | null, genome?: { __typename?: 'Genome', organism?: { __typename?: 'Organism', id: number, label: string } | null } | null }>, organismsBase: Array<{ __typename?: 'Organism', label: string, id: number }>, organisms: Array<{ __typename?: 'Organism', id: number }>, selectableTargets: Array<{ __typename?: 'Target', id: string, genome?: { __typename?: 'Genome', organism?: { __typename?: 'Organism', shortlabel: string } | null } | null }> }; + +export type GuideAlignmentQueryQueryVariables = Exact<{ + guideSubclasses?: InputMaybe<Array<GuideClass> | GuideClass>; + guideName?: InputMaybe<Scalars['String']['input']>; + organismIds?: InputMaybe<Array<Scalars['Int']['input']> | Scalars['Int']['input']>; +}>; + + +export type GuideAlignmentQueryQuery = { __typename?: 'Query', guideBase: Array<{ __typename?: 'Guide', name?: string | null, subclass: GuideClass, subclass_label: string, genome?: { __typename?: 'Genome', organism?: { __typename?: 'Organism', id: number } | null } | null }>, guideNamesFilteredBySubclass: Array<{ __typename?: 'Guide', name?: string | null, genome?: { __typename?: 'Genome', organism?: { __typename?: 'Organism', id: number } | null } | null }>, organismsBase: Array<{ __typename?: 'Organism', label: string, id: number }>, organisms: Array<{ __typename?: 'Organism', id: number }>, selectableGuides: Array<{ __typename?: 'Guide', id: string, genome?: { __typename?: 'Genome', organism?: { __typename?: 'Organism', shortlabel: string } | null } | null }> }; + export type DatabaseStatisticsQueryQueryVariables = Exact<{ [key: string]: never; }>; @@ -9856,6 +9873,8 @@ export const ModificationByIdQueryDocument = {"kind":"Document","definitions":[{ export const GuideByIdQueryDocument = {"kind":"Document","definitions":[{"kind":"OperationDefinition","operation":"query","name":{"kind":"Name","value":"guideByIdQuery"},"variableDefinitions":[{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"id"}},"type":{"kind":"NamedType","name":{"kind":"Name","value":"ID"}}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"guides"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id"},"value":{"kind":"Variable","name":{"kind":"Name","value":"id"}}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"altnames"}},{"kind":"Field","name":{"kind":"Name","value":"description"}},{"kind":"Field","name":{"kind":"Name","value":"length"}},{"kind":"Field","name":{"kind":"Name","value":"class"}},{"kind":"Field","name":{"kind":"Name","value":"subclass_label"}},{"kind":"Field","alias":{"kind":"Name","value":"chromosomeConnection"},"name":{"kind":"Name","value":"parentConnection"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"node"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"graphql_type"},"value":{"kind":"EnumValue","value":"Chromosome"}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}},{"kind":"Field","name":{"kind":"Name","value":"strand"}}]}},{"kind":"Field","name":{"kind":"Name","value":"node"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"length"}},{"kind":"Field","name":{"kind":"Name","value":"graphql_type"}}]}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"host_genes"}},{"kind":"Field","name":{"kind":"Name","value":"seq"}},{"kind":"Field","name":{"kind":"Name","value":"genome"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organism"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"label"}}]}}]}},{"kind":"Field","alias":{"kind":"Name","value":"boxConnections"},"name":{"kind":"Name","value":"featuresConnection"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"NOT"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"node"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"class"},"value":{"kind":"EnumValue","value":"DuplexFragment"}}]}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"InlineFragment","typeCondition":{"kind":"NamedType","name":{"kind":"Name","value":"HasFeature"}},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"node"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"annotation"}}]}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"modifications"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"options"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"sort"},"value":{"kind":"ListValue","values":[{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"position"},"value":{"kind":"EnumValue","value":"ASC"}}]}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"type"}}]}},{"kind":"Field","name":{"kind":"Name","value":"modificationsAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","name":{"kind":"Name","value":"interactions"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"duplexes"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","alias":{"kind":"Name","value":"primaryStrandsConnection"},"name":{"kind":"Name","value":"strandsConnection"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"edge"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"primary"},"value":{"kind":"BooleanValue","value":true}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}},{"kind":"Field","name":{"kind":"Name","value":"primary"}}]}},{"kind":"Field","name":{"kind":"Name","value":"node"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"seq"}},{"kind":"Field","name":{"kind":"Name","value":"parentConnection"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}}]}}]}}]}}]}}]}}]}},{"kind":"Field","alias":{"kind":"Name","value":"secondaryStrandsConnection"},"name":{"kind":"Name","value":"strandsConnection"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"edge"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"primary"},"value":{"kind":"BooleanValue","value":false}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}},{"kind":"Field","name":{"kind":"Name","value":"primary"}}]}},{"kind":"Field","name":{"kind":"Name","value":"node"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"seq"}},{"kind":"Field","name":{"kind":"Name","value":"parentConnection"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}}]}}]}}]}}]}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"index"}}]}},{"kind":"Field","name":{"kind":"Name","value":"modification"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"position"}},{"kind":"Field","name":{"kind":"Name","value":"symbol"}},{"kind":"Field","name":{"kind":"Name","value":"symbol_label"}},{"kind":"Field","name":{"kind":"Name","value":"type"}},{"kind":"Field","name":{"kind":"Name","value":"type_short_label"}}]}},{"kind":"Field","name":{"kind":"Name","value":"target"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"class"}},{"kind":"Field","name":{"kind":"Name","value":"unit"}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"cluster"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}}]}},{"kind":"Field","name":{"kind":"Name","value":"clusterAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","name":{"kind":"Name","value":"isoform"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"guides"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}}]}},{"kind":"Field","name":{"kind":"Name","value":"guidesAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"isoformAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","name":{"kind":"Name","value":"chebi_id"}},{"kind":"Field","name":{"kind":"Name","value":"so_id"}}]}},{"kind":"Field","name":{"kind":"Name","value":"targets"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"modifications_SOME"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"guides_SOME"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id"},"value":{"kind":"Variable","name":{"kind":"Name","value":"id"}}}]}}]}}]}},{"kind":"Argument","name":{"kind":"Name","value":"options"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"sort"},"value":{"kind":"ListValue","values":[{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"name"},"value":{"kind":"EnumValue","value":"ASC"}}]}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"class"}},{"kind":"Field","name":{"kind":"Name","value":"unit"}}]}}]}}]} as unknown as DocumentNode<GuideByIdQueryQuery, GuideByIdQueryQueryVariables>; export const TargetByIdQueryDocument = {"kind":"Document","definitions":[{"kind":"OperationDefinition","operation":"query","name":{"kind":"Name","value":"targetByIdQuery"},"variableDefinitions":[{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"id"}},"type":{"kind":"NamedType","name":{"kind":"Name","value":"ID"}}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"targets"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id"},"value":{"kind":"Variable","name":{"kind":"Name","value":"id"}}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"altnames"}},{"kind":"Field","name":{"kind":"Name","value":"description"}},{"kind":"Field","name":{"kind":"Name","value":"length"}},{"kind":"Field","name":{"kind":"Name","value":"class"}},{"kind":"Field","name":{"kind":"Name","value":"unit"}},{"kind":"Field","alias":{"kind":"Name","value":"chromosomeConnection"},"name":{"kind":"Name","value":"parentConnection"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"node"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"graphql_type"},"value":{"kind":"EnumValue","value":"Chromosome"}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}},{"kind":"Field","name":{"kind":"Name","value":"strand"}}]}},{"kind":"Field","name":{"kind":"Name","value":"node"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"graphql_type"}}]}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"seq"}},{"kind":"Field","name":{"kind":"Name","value":"genome"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organism"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"label"}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"modifications"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"options"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"sort"},"value":{"kind":"ListValue","values":[{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"position"},"value":{"kind":"EnumValue","value":"ASC"}}]}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"position"}},{"kind":"Field","name":{"kind":"Name","value":"type"}},{"kind":"Field","name":{"kind":"Name","value":"symbol"}}]}},{"kind":"Field","name":{"kind":"Name","value":"modificationsAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","name":{"kind":"Name","value":"interactions"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"duplexes"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","alias":{"kind":"Name","value":"primaryStrandsConnection"},"name":{"kind":"Name","value":"strandsConnection"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"edge"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"primary"},"value":{"kind":"BooleanValue","value":true}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}},{"kind":"Field","name":{"kind":"Name","value":"primary"}}]}},{"kind":"Field","name":{"kind":"Name","value":"node"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"seq"}},{"kind":"Field","name":{"kind":"Name","value":"parentConnection"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}}]}}]}}]}}]}}]}}]}},{"kind":"Field","alias":{"kind":"Name","value":"secondaryStrandsConnection"},"name":{"kind":"Name","value":"strandsConnection"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"edge"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"primary"},"value":{"kind":"BooleanValue","value":false}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}},{"kind":"Field","name":{"kind":"Name","value":"primary"}}]}},{"kind":"Field","name":{"kind":"Name","value":"node"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"seq"}},{"kind":"Field","name":{"kind":"Name","value":"parentConnection"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}}]}}]}}]}}]}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"index"}}]}},{"kind":"Field","name":{"kind":"Name","value":"modification"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"position"}},{"kind":"Field","name":{"kind":"Name","value":"symbol"}},{"kind":"Field","name":{"kind":"Name","value":"symbol_label"}},{"kind":"Field","name":{"kind":"Name","value":"type"}},{"kind":"Field","name":{"kind":"Name","value":"type_short_label"}}]}},{"kind":"Field","name":{"kind":"Name","value":"guide"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"subclass_label"}},{"kind":"Field","name":{"kind":"Name","value":"class"}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"chebi_id"}},{"kind":"Field","name":{"kind":"Name","value":"so_id"}},{"kind":"Field","name":{"kind":"Name","value":"url"}},{"kind":"Field","name":{"kind":"Name","value":"secondary_struct_file"}}]}},{"kind":"Field","name":{"kind":"Name","value":"guides"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"modifications_SOME"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"target"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id"},"value":{"kind":"Variable","name":{"kind":"Name","value":"id"}}}]}}]}}]}},{"kind":"Argument","name":{"kind":"Name","value":"options"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"sort"},"value":{"kind":"ListValue","values":[{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id"},"value":{"kind":"EnumValue","value":"ASC"}}]}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"class"}},{"kind":"Field","alias":{"kind":"Name","value":"chromosome"},"name":{"kind":"Name","value":"parent"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"graphql_type"},"value":{"kind":"EnumValue","value":"Chromosome"}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"name"}}]}}]}}]}}]} as unknown as DocumentNode<TargetByIdQueryQuery, TargetByIdQueryQueryVariables>; export const ClusterByIdQueryDocument = {"kind":"Document","definitions":[{"kind":"OperationDefinition","operation":"query","name":{"kind":"Name","value":"clusterByIdQuery"},"variableDefinitions":[{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"id"}},"type":{"kind":"NamedType","name":{"kind":"Name","value":"ID"}}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"clusters"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id"},"value":{"kind":"Variable","name":{"kind":"Name","value":"id"}}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"guides"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"options"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"sort"},"value":{"kind":"ListValue","values":[{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id"},"value":{"kind":"EnumValue","value":"ASC"}}]}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"class"}},{"kind":"Field","name":{"kind":"Name","value":"subclass_label"}},{"kind":"Field","name":{"kind":"Name","value":"modificationsAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"chromosomeConnection"},"name":{"kind":"Name","value":"parentConnection"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"node"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"graphql_type"},"value":{"kind":"EnumValue","value":"Chromosome"}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"edges"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"properties"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"start"}},{"kind":"Field","name":{"kind":"Name","value":"end"}}]}}]}}]}}]}},{"kind":"Field","name":{"kind":"Name","value":"guidesAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"referenceGuide"},"name":{"kind":"Name","value":"guides"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"options"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"limit"},"value":{"kind":"IntValue","value":"1"}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","alias":{"kind":"Name","value":"chromosome"},"name":{"kind":"Name","value":"parent"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"graphql_type"},"value":{"kind":"EnumValue","value":"Chromosome"}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"name"}}]}},{"kind":"Field","name":{"kind":"Name","value":"genome"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organism"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"label"}},{"kind":"Field","name":{"kind":"Name","value":"id"}}]}}]}}]}}]}}]}}]} as unknown as DocumentNode<ClusterByIdQueryQuery, ClusterByIdQueryQueryVariables>; +export const TargetAlignmentQueryDocument = {"kind":"Document","definitions":[{"kind":"OperationDefinition","operation":"query","name":{"kind":"Name","value":"targetAlignmentQuery"},"variableDefinitions":[{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"targetName"}},"type":{"kind":"NamedType","name":{"kind":"Name","value":"String"}}},{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"organismIds"}},"type":{"kind":"ListType","type":{"kind":"NonNullType","type":{"kind":"NamedType","name":{"kind":"Name","value":"Int"}}}}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","alias":{"kind":"Name","value":"targetBase"},"name":{"kind":"Name","value":"targets"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"unit"}},{"kind":"Field","name":{"kind":"Name","value":"genome"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organism"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"label"}}]}}]}}]}},{"kind":"Field","alias":{"kind":"Name","value":"organismsBase"},"name":{"kind":"Name","value":"organisms"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"label"}},{"kind":"Field","name":{"kind":"Name","value":"id"}}]}},{"kind":"Field","name":{"kind":"Name","value":"organisms"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"tableEntries_SOME"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"modification"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"target"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"name"},"value":{"kind":"Variable","name":{"kind":"Name","value":"targetName"}}}]}}]}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"selectableTargets"},"name":{"kind":"Name","value":"targets"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"name"},"value":{"kind":"Variable","name":{"kind":"Name","value":"targetName"}}},{"kind":"ObjectField","name":{"kind":"Name","value":"genome"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"organism"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id_IN"},"value":{"kind":"Variable","name":{"kind":"Name","value":"organismIds"}}}]}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"genome"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organism"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"shortlabel"}}]}}]}}]}}]}}]} as unknown as DocumentNode<TargetAlignmentQueryQuery, TargetAlignmentQueryQueryVariables>; +export const GuideAlignmentQueryDocument = {"kind":"Document","definitions":[{"kind":"OperationDefinition","operation":"query","name":{"kind":"Name","value":"guideAlignmentQuery"},"variableDefinitions":[{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"guideSubclasses"}},"type":{"kind":"ListType","type":{"kind":"NonNullType","type":{"kind":"NamedType","name":{"kind":"Name","value":"GuideClass"}}}}},{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"guideName"}},"type":{"kind":"NamedType","name":{"kind":"Name","value":"String"}}},{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"organismIds"}},"type":{"kind":"ListType","type":{"kind":"NonNullType","type":{"kind":"NamedType","name":{"kind":"Name","value":"Int"}}}}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","alias":{"kind":"Name","value":"guideBase"},"name":{"kind":"Name","value":"guides"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"subclass"}},{"kind":"Field","name":{"kind":"Name","value":"subclass_label"}},{"kind":"Field","name":{"kind":"Name","value":"genome"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organism"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}}]}}]}}]}},{"kind":"Field","alias":{"kind":"Name","value":"guideNamesFilteredBySubclass"},"name":{"kind":"Name","value":"guides"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"subclass_IN"},"value":{"kind":"Variable","name":{"kind":"Name","value":"guideSubclasses"}}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"name"}},{"kind":"Field","name":{"kind":"Name","value":"genome"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organism"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}}]}}]}}]}},{"kind":"Field","alias":{"kind":"Name","value":"organismsBase"},"name":{"kind":"Name","value":"organisms"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"label"}},{"kind":"Field","name":{"kind":"Name","value":"id"}}]}},{"kind":"Field","name":{"kind":"Name","value":"organisms"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"tableEntries_SOME"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"guide"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"name"},"value":{"kind":"Variable","name":{"kind":"Name","value":"guideName"}}}]}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"selectableGuides"},"name":{"kind":"Name","value":"guides"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"name"},"value":{"kind":"Variable","name":{"kind":"Name","value":"guideName"}}},{"kind":"ObjectField","name":{"kind":"Name","value":"genome"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"organism"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id_IN"},"value":{"kind":"Variable","name":{"kind":"Name","value":"organismIds"}}}]}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"genome"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organism"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"shortlabel"}}]}}]}}]}}]}}]} as unknown as DocumentNode<GuideAlignmentQueryQuery, GuideAlignmentQueryQueryVariables>; export const DatabaseStatisticsQueryDocument = {"kind":"Document","definitions":[{"kind":"OperationDefinition","operation":"query","name":{"kind":"Name","value":"databaseStatisticsQuery"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"organismsAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"allModifications"},"name":{"kind":"Name","value":"modifications"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"type"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"allModificationsCount"},"name":{"kind":"Name","value":"modificationsAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"nonOrphanModificationsCount"},"name":{"kind":"Name","value":"modificationsAggregate"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"guidesAggregate"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"count_GT"},"value":{"kind":"IntValue","value":"0"}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"allGuides"},"name":{"kind":"Name","value":"guides"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"subclass"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"allGuidesCount"},"name":{"kind":"Name","value":"guidesAggregate"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}},{"kind":"Field","alias":{"kind":"Name","value":"nonOrphanGuidesCount"},"name":{"kind":"Name","value":"guidesAggregate"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"modificationsAggregate"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"count_GT"},"value":{"kind":"IntValue","value":"0"}}]}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"count"}}]}}]}}]} as unknown as DocumentNode<DatabaseStatisticsQueryQuery, DatabaseStatisticsQueryQueryVariables>; export const LegalDocumentListQueryDocument = {"kind":"Document","definitions":[{"kind":"OperationDefinition","operation":"query","name":{"kind":"Name","value":"legalDocumentListQuery"},"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"documents"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"types_INCLUDES"},"value":{"kind":"StringValue","value":"legal","block":false}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"menu_label"}}]}}]}}]} as unknown as DocumentNode<LegalDocumentListQueryQuery, LegalDocumentListQueryQueryVariables>; export const DocumentByIdQueryDocument = {"kind":"Document","definitions":[{"kind":"OperationDefinition","operation":"query","name":{"kind":"Name","value":"documentByIdQuery"},"variableDefinitions":[{"kind":"VariableDefinition","variable":{"kind":"Variable","name":{"kind":"Name","value":"id"}},"type":{"kind":"NamedType","name":{"kind":"Name","value":"ID"}}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"documents"},"arguments":[{"kind":"Argument","name":{"kind":"Name","value":"where"},"value":{"kind":"ObjectValue","fields":[{"kind":"ObjectField","name":{"kind":"Name","value":"id"},"value":{"kind":"Variable","name":{"kind":"Name","value":"id"}}}]}}],"selectionSet":{"kind":"SelectionSet","selections":[{"kind":"Field","name":{"kind":"Name","value":"id"}},{"kind":"Field","name":{"kind":"Name","value":"menu_label"}},{"kind":"Field","name":{"kind":"Name","value":"content"}}]}}]}}]} as unknown as DocumentNode<DocumentByIdQueryQuery, DocumentByIdQueryQueryVariables>; \ No newline at end of file diff --git a/src/gql/queries.ts b/src/gql/queries.ts index 8c4c412ba4937932b616a9469b7de77ebeac4e49..ba8c44dad3a7f2cb480e63e087c7f808893ee456 100644 --- a/src/gql/queries.ts +++ b/src/gql/queries.ts @@ -820,6 +820,104 @@ export const clusterByIdQuery = graphql(/* GraphQL */ ` } `) +/** + * Get necessary data to select sequences for alignment. + */ +export const targetAlignmentQuery = graphql(/* GraphQL */ ` + query targetAlignmentQuery($targetName: String, $organismIds: [Int!]) { + targetBase: targets { + name + unit + genome { + organism { + id + label + } + } + } + + organismsBase: organisms { + label + id + } + + organisms( + where: { + tableEntries_SOME: { modification: { target: { name: $targetName } } } + } + ) { + id + } + + selectableTargets: targets( + where: { + name: $targetName + genome: { organism: { id_IN: $organismIds } } + } + ) { + id + genome { + organism { + shortlabel + } + } + } + } +`) + +/** + * Get necessary data to select sequences for alignment. + */ +export const guideAlignmentQuery = graphql(/* GraphQL */ ` + query guideAlignmentQuery( + $guideSubclasses: [GuideClass!] + $guideName: String + $organismIds: [Int!] + ) { + guideBase: guides { + name + subclass + subclass_label + genome { + organism { + id + } + } + } + + guideNamesFilteredBySubclass: guides( + where: { subclass_IN: $guideSubclasses } + ) { + name + genome { + organism { + id + } + } + } + + organismsBase: organisms { + label + id + } + + organisms(where: { tableEntries_SOME: { guide: { name: $guideName } } }) { + id + } + + selectableGuides: guides( + where: { name: $guideName, genome: { organism: { id_IN: $organismIds } } } + ) { + id + genome { + organism { + shortlabel + } + } + } + } +`) + /** * Get statistics about the database. */ diff --git a/src/layouts/FooterLayout.vue b/src/layouts/FooterLayout.vue index b8b9b65f243c0bbdb24db3f827b1ef4222b926e5..40c65ef5e1201d99a8ca122d0149b7d9fad79625 100644 --- a/src/layouts/FooterLayout.vue +++ b/src/layouts/FooterLayout.vue @@ -11,7 +11,7 @@ import Button from 'primevue/button' import IconFa6SolidAddressCard from '~icons/fa6-solid/address-card' import IconFa6SolidChevronUp from '~icons/fa6-solid/chevron-up' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' /** diff --git a/src/layouts/MainLayout.vue b/src/layouts/MainLayout.vue index b7e4fc416d89a519a23fbb24fb6a69bac533dcfc..1be2df3923fde964a1d8950fb9e7157883a2e3ae 100644 --- a/src/layouts/MainLayout.vue +++ b/src/layouts/MainLayout.vue @@ -3,7 +3,6 @@ * Vue imports */ import { ref } from 'vue' -import { useRouter } from 'vue-router' /** * Components imports */ @@ -15,6 +14,10 @@ import Menubar from 'primevue/menubar' import IconFa6SolidMagnifyingGlass from '~icons/fa6-solid/magnifying-glass' import IconFa6SolidCircleXmark from '~icons/fa6-solid/circle-xmark' import IconFa6SolidArrowRight from '~icons/fa6-solid/arrow-right' +/** + * Composables imports + */ +import { useRouter } from 'vue-router' /** * Utils imports */ diff --git a/src/router/index.ts b/src/router/index.ts index f29dd5bb5bb141619f0c57f083dbdc477b08ec3f..b12de9de6a7de5ad7905aa931aee98a2d5b3491e 100644 --- a/src/router/index.ts +++ b/src/router/index.ts @@ -7,18 +7,22 @@ import { createRouter, createWebHistory } from 'vue-router' */ import HomeView from '@/views/HomeView.vue' /** - * Other 3rd-party imports + * Composables imports */ import { useTitle } from '@vueuse/core' +/** + * Other 3rd-party imports + */ import { isNonNullish } from '@/typings/typeUtils' /** * Types imports */ -import type { SelectionFormModesType } from '@/views/SelectionView.vue' +import type { SelectionModesType } from '@/views/SelectionView.vue' +// import type { AlignmentModesType } from '@/views/AlignmentView.vue' /** * Utils imports */ -import { SELECTION_FORM_MODES } from '@/utils/constant' +import { /* ALIGNMENT_MODES, */ SELECTION_MODES } from '@/utils/constant' declare module 'vue-router' { interface RouteMeta { @@ -42,20 +46,18 @@ const router = createRouter({ name: 'selection', component: () => import('@/views/SelectionView.vue'), props: (to) => ({ - mode: to.query.mode + initialMode: to.query.mode }), beforeEnter: (to) => { if ( typeof to.query.mode !== 'string' || - !SELECTION_FORM_MODES.includes( - to.query.mode as SelectionFormModesType - ) + !SELECTION_MODES.includes(to.query.mode as SelectionModesType) ) { - router.replace({ name: 'lost' }) + router.replace({ name: 'selection', query: { mode: 'guide' } }) } }, meta: { - title: 'Advanced selection' + managedTitle: true } }, { @@ -109,6 +111,9 @@ const router = createRouter({ ) { router.replace({ name: 'notFound' }) } + }, + meta: { + managedTitle: true } }, { @@ -132,6 +137,9 @@ const router = createRouter({ if (typeof to.query.id !== 'string') { router.replace({ name: 'notFound' }) } + }, + meta: { + managedTitle: true } }, { @@ -146,6 +154,9 @@ const router = createRouter({ if (typeof to.query.id !== 'string') { router.replace({ name: 'notFound' }) } + }, + meta: { + managedTitle: true } }, { @@ -160,6 +171,9 @@ const router = createRouter({ if (typeof to.query.id !== 'string') { router.replace({ name: 'notFound' }) } + }, + meta: { + managedTitle: true } }, { @@ -171,10 +185,32 @@ const router = createRouter({ if (typeof to.query.id !== 'string') { router.replace({ name: 'notFound' }) } + }, + meta: { + managedTitle: true } } ] }, + // { + // path: '/alignment', + // name: 'alignment', + // component: () => import('@/views/AlignmentView.vue'), + // props: (to) => ({ + // initialMode: to.query.mode + // }), + // beforeEnter: (to) => { + // if ( + // typeof to.query.mode !== 'string' || + // !ALIGNMENT_MODES.includes(to.query.mode as AlignmentModesType) + // ) { + // router.replace({ name: 'alignment', query: { mode: 'guide' } }) + // } + // }, + // meta: { + // managedTitle: true + // } + // }, { path: '/statistics', name: 'statistics', @@ -263,8 +299,10 @@ const router = createRouter({ }) router.afterEach((to) => { - useTitle((to.meta.title ? `${to.meta.title} | ` : '') + 'SnoBoard') - console.info('Title changed.') + if (!to.meta.managedTitle) { + useTitle((to.meta.title ? `${to.meta.title} | ` : '') + 'SnoBoard') + console.info('Title changed.') + } }) export default router diff --git a/src/typings/typeUtils.ts b/src/typings/typeUtils.ts index d44a1907db913128da2440626b8e95b317986e1f..d23a9f3ca3a829b9da9dbd532febf9e152bc4c57 100644 --- a/src/typings/typeUtils.ts +++ b/src/typings/typeUtils.ts @@ -110,3 +110,11 @@ export type SubType<BaseType, ConditionType> = Pick< [Key in keyof BaseType]: BaseType[Key] extends ConditionType ? Key : never }[keyof BaseType] > + +/** + * Checks if an array contains at least one element, narrowing its type. + * @param array The array for which to check the length. + */ +export function isNotEmpty<T>(array: T[]): array is [T,...T[]] { + return array.length >= 1 +} \ No newline at end of file diff --git a/src/utils/constant.ts b/src/utils/constant.ts index 0b0b8efa33038c101af9aa6c12c025a1443015f9..6aaf040b1448c319ec0d35540a6bf70a091772f1 100644 --- a/src/utils/constant.ts +++ b/src/utils/constant.ts @@ -1,5 +1,5 @@ /** - * Component imports + * Components imports */ import IconFa6SolidTable from '~icons/fa6-solid/table' import IconFa6SolidDatabase from '~icons/fa6-solid/database' @@ -36,10 +36,14 @@ interface HelpTourItem { } /** - * The different modes available for advanced selection form on the selection - * page. + * The different modes available for advanced selection. */ -export const SELECTION_FORM_MODES = ['modification', 'guide', 'target'] as const +export const SELECTION_MODES = ['modification', 'guide', 'target'] as const + +/** + * The different modes available for alignment. + */ +export const ALIGNMENT_MODES = ['guide', 'target'] as const /** * The items to display in the main navigation menu. diff --git a/src/utils/graphFormat.ts b/src/utils/graphFormat.ts index 70e62750ecb3af9d425591e48452df58b99a7fe8..aedd4cfdfb1bb1706ca2ad142db1593d667dfe40 100644 --- a/src/utils/graphFormat.ts +++ b/src/utils/graphFormat.ts @@ -1,5 +1,9 @@ -import type { C4GGraphModel, C4GNodeModel } from '@/typings/Codev4GraphFormat' +/** + * Types imports + */ +import type { C4GGraphModel } from '@/typings/Codev4GraphFormat' import type { JGFGraphModel } from '@/typings/JSONGraphFormat' +import { mapValues as _mapValues } from 'lodash-es' /** * "Fix" a graph by transforming it to proper C4G format @@ -22,30 +26,22 @@ export const fixGraphFormat = (JGFGraph: JGFGraphModel): C4GGraphModel => { }) ) }, - nodes: Object.entries(JGFGraph.graph.nodes || {}).reduce<{ - [nodeId: string]: C4GNodeModel - }>( - (nodes, [nodeId, node]) => ({ - ...nodes, - [nodeId]: { - label: node.label, - metadata: { - position: { - x: node.metadata?.x, - y: node.metadata?.y - }, - data: [ - { - label: 'nucleotidePosition', - type: 'number', - value: node.metadata?.position - } - ] + nodes: _mapValues(JGFGraph.graph.nodes, (node) => ({ + label: node.label, + metadata: { + position: { + x: node.metadata?.x, + y: node.metadata?.y + }, + data: [ + { + label: 'nucleotidePosition', + type: 'number', + value: node.metadata?.position } - } - }), - {} - ), + ] + } + })), edges: JGFGraph.graph.edges?.map((edge) => ({ ...edge, metadata: { diff --git a/src/utils/normalise.ts b/src/utils/normalise.ts index b6f9a32aef922969c7bc17dab3f93db4ddd0b08c..18f945dd9bdee923f680aff2eec33968642ed30d 100644 --- a/src/utils/normalise.ts +++ b/src/utils/normalise.ts @@ -1,7 +1,7 @@ /** * Other 3rd-party imports */ -import { findIndex as _findIndex } from 'lodash-es' +import { findIndex as _findIndex, mapValues as _mapValues } from 'lodash-es' /** * Types imports */ @@ -96,9 +96,7 @@ export const composeNormalisedGraph = ( normalisedEdgeLength / (referenceEdgeLength || normalisedEdgeLength) // Apply normalisation to the nodes - const normalisedNodes = Object.entries(graph.graph.nodes).reduce<{ - [nodeId: string]: C4GNodeModel - }>((nodes, [nodeId, node]) => { + const normalisedNodes = _mapValues(graph.graph.nodes, (node) => { // Normalise position if present const normalisedNode = node.metadata?.position ? { @@ -116,10 +114,7 @@ export const composeNormalisedGraph = ( // Check if additional metadata is present, if not return & continue // normalisation on next node if (!normalisedNode.metadata?.data) { - return { - ...nodes, - [`${nodeId}`]: normalisedNode - } + return normalisedNode } // Get indexes of additional metadata to normalise (-1 if not present) @@ -166,11 +161,8 @@ export const composeNormalisedGraph = ( nucleotidePositionLineDataValue.y2 *= normalisationRatio } - return { - ...nodes, - [`${nodeId}`]: normalisedNode - } - }, {}) + return normalisedNode + }) return { graph: { diff --git a/src/utils/sequences.ts b/src/utils/sequences.ts new file mode 100644 index 0000000000000000000000000000000000000000..24412249a8fb4411ce855f717d3f6bb9ea440b48 --- /dev/null +++ b/src/utils/sequences.ts @@ -0,0 +1,15 @@ +/** + * In an array of sequences (e.g. for an alignment), computes the conservation + * rate of a reference seq., i.e. the proportion of seq. in the array which are + * identical to it. + * @param sequences The sequences among which to compute the conservation. + * @param referenceSequence The sequence for which to compute the conservation. + * @returns The proportion of the sequences in the array which are identical to + * the reference one. + */ +export const composeConservationRate = ( + sequences: string[], + referenceSequence: string +): number => + sequences.filter((sequence) => sequence && sequence === referenceSequence) + .length / (sequences.length || 1) diff --git a/src/views/APIView.vue b/src/views/APIView.vue index 73bdbca3897e35d921c08f8c947aed8adb05c0a8..b98200b0ce02a7297f4c3dd6a3879d23fc259844 100644 --- a/src/views/APIView.vue +++ b/src/views/APIView.vue @@ -12,11 +12,14 @@ import Dialog from 'primevue/dialog' import Checkbox from 'primevue/checkbox' import IconDeviconNeo4j from '~icons/devicon/neo4j' import IconLogosGraphql from '~icons/logos/graphql' +/** + * Composables imports + */ +import { useStorage } from '@vueuse/core' /** * Other 3rd-party imports */ import { ApolloSandbox } from '@apollo/sandbox' -import { useStorage } from '@vueuse/core' /** * Reactive local storage entry to show or not the info dialog at landing. diff --git a/src/views/AlignmentView.vue b/src/views/AlignmentView.vue new file mode 100644 index 0000000000000000000000000000000000000000..f9dad3e1bc59d45d4f3895b687ea04721f7721c4 --- /dev/null +++ b/src/views/AlignmentView.vue @@ -0,0 +1,179 @@ +<script setup lang="ts"> +/** + * Vue imports + */ +import { computed, ref, onMounted } from 'vue' +/** + * Components imports + */ +import MainLayout from '@/layouts/MainLayout.vue' +import SequenceAlignment from '@/components/SequenceAlignment.vue' +import AlignmentForm from '@/components/AlignmentForm.vue' +import IconSnoboardAlignment from '~icons/snoboard/alignment' +/** + * Composables imports + */ +import { useRouter } from 'vue-router' +import { useTitle } from '@vueuse/core' +/** + * Other 3rd-party import + */ +import { capitalize as _capitalize } from 'lodash-es' +/** + * Type imports + */ +import type { AlignmentModel } from '@/components/SequenceAlignment.vue' +/** + * Utils imports + */ +import { ALIGNMENT_MODES } from '@/utils/constant' + +/** + * Type of sequence to align. + */ +export type AlignmentModesType = (typeof ALIGNMENT_MODES)[number] + +const ALIGNMENT: AlignmentModel = { + tracks: [ + { + id: '18S_Arabidopsis_Iour', + name: 'A.thaliana - Iouri', + shortname: 'Ath-I', + sequence: + 'UACCUGGUUGAUCCUGCCAGUAGUCAUAUGCUUGUCUCAAAGAUUAAGCCAUGCAUGUGUAAGUAUGAACGAAUUCAGACUGUGAAACUGCGAAUGGCUCAUUAAAUCAGUUAUAGUUUGUUUGAUGGUAA---CUACUACUC---GGAUAACCGUAGUAAUUCUAGAGCUAAUACGUGCAAC---------AAACCCCGACU-UAUGGAAGGGACGCAUUUAUUAGAUAAAAG----------------------------GUCGACGCGGGCUCUGCCCGUUGCUCUGAUGAUUCAUGAUAACUC--GACGGAUCGCAUGGCCUCUGUGCUGGCGACGCAUCAUUCAAAUUUCUGCCCUAUCAACUUUCGAUGGUAGGAUAGUGGCCUACCAUGGUGGUAACGGGUGACGGAGAAUUAGGGUUCGAUUCCGGAGAGGGAGCCUGAGAAACGGCUACCACAUCCAAGGAAGGCAGCAGGCGCGCAAAUUACCCAAUCCUGACACGGGGAGGUAGUGACAAUAAAUAACAAUACCGGGCUCUUUCGAGUCU-GGUAAUUGGAAUGAGUACAAUCUAAAUCCCUUAACGAGGAUCCAUUGGAGGGCAAGUCUGGUGCCAGCAGCCGCGGUAAUUCCAGCUCCAAUAGCGUAUAUUUAAGUUGUUGCAGUUAAAAAGCUCGUAGUUGAACCUUGGGAUGGGUCGGCCGGUCCGCCUUUGGUGUGCAUUGGUC--------GGCUUGUCCCUUCGGUCGGCGAUACGCUCCUGGUCUUAAUUGGCCGGGUCGUGCCUCCGGCGCUGUUACUUUGAAGAAAUUAGAGUGCUCAAAGCAAGCCUACGCU--CUGGAUACAUUAGCAUGGGAUAACAUCAUAGGAU-UUCGAUCCUAUUGUGUUGGCCUUCGGGAUCGGAGUAAUGAUUAACAGGGACAGUCGGGGGCAUUCGUAUUUCAUAGUCAGAGGUGAAAUUCUUGGAUUUAUGAAAGACGAACAACUGCGAAAGCAUUUGCCAAGGAUGUUUUCAUUAAUCAAGAACGAAAGUUGGGGGCUCGAAGACGAUCAGAUACCGUCCUAGUCUCAACCAUAAACGAUGCCGACCAGGGAUCAGCGGAUGUUGCUUAUAGGACUCCGCUGGCACCUUAUGAGAAAUCAAAGUUUUUGGGUUCCGGGGGGAGUAUGGUCGCAAGGCUGAAACUUAAAGGAAUUGACGGAAGGGCACCACCAGGAGUGGAGCCUGCGGCUUAAUUUGACUCAACACGGGGAAACUUACCAGGUCCAGACAUAGUAAGGAUUGACAGACUGAGAGCUCUUUCUUGAUUCUAUGGGUGGUGGUGCAUGGCCGUUCUUAGUUGGUGGAGCGAUUUGUCUGGUUAAUUCCGUUAACGAACGAGACCUCAGCCUGCUAACUAGCUACGUG-----GAGGCAUCCCUUCACGGCCGGCUUCUUAGAGGGACUAUGGCCGUUUAGGCCAAGGAAGUUUGAGGCAAUAACAGGUCUGUGAUGCCCUUAGAUGUUCUGGGCCGCACGCGCGCUACACUGAUGUAUUCAACGAGUUCACACCUUG-GCCGACAGGCCCGGGUAAUCUU-UGAAAUUUCAUCGUGAUGGGGAUAGAUCAUUGCAAUUGUUGGUCUUCAACGAGGAAUUCCU--AGUAAGCGCGAGUCAUCAGCUCGCGUUGACUACGUCCCUGCCCUUUGUACACACCGCCCGUCGCUCCUACCGAUUGAAUGAUCCGGUGAAGUGUUCGGAUCGCGGCGACGUGGGUGGUUCGCCGCCC---GCGACGUCGCGAGAAGUCCACUAAACCUUAUCAUUUAGAGGAAGGAGAAGUCGUAACAAGGUUUCCGUAGGUGAACCUGCGGAAGGAUCAUUG', + isReference: true, + objects: { + MOD_00030: { + name: 'Am28_18S', + start: 28, + end: 28, + type: "2'-*O*-Me", + color: 'sky', + link: { name: 'modificationDetails', query: { id: 'MOD_00030' } } + }, + MOD_00187: { + name: 'Psi35_18S', + start: 35, + end: 35, + type: 'Pseudouridylation', + color: 'violet', + link: { name: 'modificationDetails', query: { id: 'MOD_00187' } } + }, + MOD_00012: { + name: 'Am162_18S', + start: 168, + end: 168, + type: "2'-*O*-Me", + color: 'sky', + link: { name: 'modificationDetails', query: { id: 'MOD_00012' } } + } + } + }, + { + id: '18S_Arabidopsis_thal', + name: 'A.thaliana - ref', + shortname: 'Ath-Ref', + sequence: + 'UACCUGGUUGAUCCUGCCAGUAGUCAUAUGCUUGUCUCAAAGAUUAAGCCAUGCAUGUGUAAGUAUGAACGAAUUCAGACUGUGAAACUGCGAAUGGCUCAUUAAAUCAGUUAUAGUUUGUUUGAUGGUAA---CUACUACUC---GGAUAACCGUAGUAAUUCUAGAGCUAAUACGUGCAAC---------AAACCCCGACU-UAUGGAAGGGACGCAUUUAUUAGAUAAAAG----------------------------GUCGACGCGGGCUCUGGC--UUGCUCUGAUGAUUCAUGAUAACUC--GACGGAUCGCAUGGCCUCUGUGCUGGCGACGCAUCAUUCAAAUUUCUGCCCUAUCAACUUUCGAUGGUAGGAUAGUGGCCUACCAUGGUGGUAACGGGUGACGGAGAAUUAGGGUUCGAUUCCGGAGAGGGAGCCUGAGAAACGGCUACCACAUCCAAGGAAGGCAGCAGGCGCGCAAAUUACCCAAUCCUGACACGGGGAGGUAGUGACAAUAAAUAACAAUACCGGGCUCUUUCGAGUCU-GGUAAUUGGAAUGAGUACAAUCUAAAUCCCUUAACGAGGAUCCAUUGGAGGGCAAGUCUGGUGCCAGCAGCCGCGGUAAUUCCAGCUCCAAUAGCGUAUAUUUAAGUUGUUGCAGUUAAAAAGCUCGUAGUUGAACCUUGGGAUGGGUCGGCCGGUCCGCCUUUGGUGUGCAUUGGUC--------GGCUUGUCCCUUCGGUCGGCGAUACGCUCCUGGUCUUAAUUGGCCGGGUCGUGCCUCCGGCGCUGUUACUUUGAAGAAAUUAGAGUGCUCAAAGCAAGCCUACGCU--CUGGAUACAUUAGCAUGGGAUAACAUCAUAGGAU-UUCGAUCCUAUUGUGUUGGC-UUCGGGAUCGGAGUAAUGAUUAACAGGGACAGUCGGGGGCAUUCGUAUUUCAUAGUCAGAGGUGAAAUUCUUGGAUUUAUGAAAGACGAACAACUGCGAAAGCAUUUGCCAAGGAUGUUUUCAUUAAUCAAGAACGAAAGUUGGGGGCUCGAAGACGAUCAGAUACCGUCCUAGUCUCAACCAUAAACGAUGCCGACCAGGGAUCAGCGGAUGUUGCUUAUAGGACUCCGCUGGCACCUUAUGAGAAAUCAAAGUUUUUGGGUUCCGGGGGGAGUAUGGUCGCAAGGCUGAAACUUAAAGGAAUUGACGGAAGGGCACCACCAGGAGUGGAGCCUGCGGCUUAAUUUGACUCAACACGGGGAAACUUACCAGGUCCAGACAUAGUAAGGAUUGACAGACUGAGAGCUCUUUCUUGAUUCUAUGGGUGGUGGUGCAUGGCCGUUCUUAGUUGGUGGAGCGAUUUGUCUGGUUAAUUCCGUUAACGAACGAGACCUCAGCCUGCUAACUAGCUACGUG-----GAGGCAUCCCUUCACGGCCGGCUUCUUAGAGGGACUAUGGCCGUUUAGGCCAAGGAAGUUUGAGGCAAUAACAGGUCUGUGAUGCCCUUAGAUGUUCUGGGCCGCACGCGCGCUACACUGAUGUAUUCAACGAGUUCACACCUUG--CCGACAGGCCCGGGUAAUCUU-UGAAAUUUCAUCGUGAUGGGGAUAGAUCAUUGCAAUUGUUGGUCUUCAACGAGGAAUUCCU--AGUAAGCGCGAGUCAUCAGCUCGCGUUGACUACGUCCCUGCCCUUUGUACACACCGCCCGUCGCUCCUACCGAUUGAAUGAUCCGGUGAAGUGUUCGGAUCGCGGCGACGUGGGUGGUUCGCCGCCC---GCGACGUCGCGAGAAGUCCACUAAACCUUAUCAUUUAGAGGAAGGAGAAGUCGUAACAAGGUUUCCGUAGGUGAACCUGCGGAAGGAUCAUUG' + }, + { + id: '18S_S.cerevisiae', + name: 'S.cerevisiae - ref', + shortname: 'Sc-Ref', + sequence: + 'UAUCUGGUUGAUCCUGCCAGUAGUCAUAUGCUUGUCUCAAAGAUUAAGCCAUGCAUGUCUAAGUAUAAGC-AAUUUAUACAGUGAAACUGCGAAUGGCUCAUUAAAUCAGUUAUCGUUUAUUUGAUAGUUC---CUUUACUACAUGGUAUAACUGUGGUAAUUCUAGAGCUAAUACAUGCUUA---------AAAUCUCGACCCUUUGGAAGAGAUGUAUUUAUUAGAUAAAAA----------------------------AUCAAUGUCUUCG------GACUCUUUGAUGAUUCAUAAUAACUU--UUCGAAUCGCAUGGCCU-UGUGCUGGCGAUGGUUCAUUCAAAUUUCUGCCCUAUCAACUUUCGAUGGUAGGAUAGUGGCCUACCAUGGUUUCAACGGGUAACGGGGAAUAAGGGUUCGAUUCCGGAGAGGGAGCCUGAGAAACGGCUACCACAUCCAAGGAAGGCAGCAGGCGCGCAAAUUACCCAAUCCUAAUUCAGGGAGGUAGUGACAAUAAAUAACGAUACAGGGCCCAUUCGGGUCU-UGUAAUUGGAAUGAGUACAAUGUAAAUACCUUAACGAGGAACAAUUGGAGGGCAAGUCUGGUGCCAGCAGCCGCGGUAAUUCCAGCUCCAAUAGCGUAUAUUAAAGUUGUUGCAGUUAAAAAGCUCGUAGUUGAACUUUGGGCCCGGUUGGCCGGUCCGAUUUUUUCGUGUACUGGAUUUCCAACGGGGCCUUUCCUUCUGGCUAACCUUGAGUCCUUGUGGCUCUUGGCGAA---------CCAGGACUUUUACUUUGAAAAAAUUAGAGUGUUCAAAGCAGGCGUAUUGC--UCGAAUAUAUUAGCAUGGAAUAAUAGAAUAGGACGUUUGGUUCUAUUUUGUUGGUUUCUAGGACCAUCGUAAUGAUUAAUAGGGACGGUCGGGGGCAUCAGUAUUCAAUUGUCAGAGGUGAAAUUCUUGGAUUUAUUGAAGACUAACUACUGCGAAAGCAUUUGCCAAGGACGUUUUCAUUAAUCAAGAACGAAAGUUAGGGGAUCGAAGAUGAUCAGAUACCGUCGUAGUCUUAACCAUAAACUAUGCCGACUAGGGAUCGGGUGGUGUUUUUUUAAUGACCCACUCGGCACCUUACGAGAAAUCAAAGUCUUUGGGUUCUGGGGGGAGUAUGGUCGCAAGGCUGAAACUUAAAGGAAUUGACGGAAGGGCACCACCAGGAGUGGAGCCUGCGGCUUAAUUUGACUCAACACGGGGAAACUCACCAGGUCCAGACACAAUAAGGAUUGACAGAUUGAGAGCUCUUUCUUGAUUUUGUGGGUGGUGGUGCAUGGCCGUUCUUAGUUGGUGGAGUGAUUUGUCUGCUUAAUUGCGAUAACGAACGAGACCUUAACCUACUAAAUAGUG--GUG-----CUAGCAUUUGCUGGUUAUCCACUUCUUAGAGGGACUAUCGGUUUCAAGCCGAUGGAAGUUUGAGGCAAUAACAGGUCUGUGAUGCCCUUAGACGUUCUGGGCCGCACGCGCGCUACACUGACGGAGCCAGCGAGUCUA-ACCUUG-GCCGAGAGGUCUUGGUAAUCUUGUGAAACUCCGUCGUGCUGGGGAUAGAGCAUUGUAAUUAUUGCUCUUCAACGAGGAAUUCCU--AGUAAGCGCAAGUCAUCAGCUUGCGUUGAUUACGUCCCUGCCCUUUGUACACACCGCCCGUCGCUAGUACCGAUUGAAUGGCUUAGUGAGGCCUCAGGAUCUGCUUAGAGAAGGGGGG-CAACUCCA---UCUCAGAGCGGAGAAUUUGGACAAACUUGGUCAUUUAGAGGAACUAAAAGUCGUAACAAGGUUUCCGUAGGUGAACCUGCGGAAGGAUCAUUA', + isReference: true, + objects: { + MOD_TEST: { + name: 'Am164_18S_Sc', + start: 168, + end: 168, + type: "2'-*O*-Me", + color: 'sky', + link: { name: 'modificationDetails', query: { id: 'MOD_TEST' } } + } + } + }, + { + id: '18S_hs_Iouri_NR_0462', + name: 'H.sapiens - Iouri - NR_0462', + shortname: 'Hs-NR_0462', + sequence: + 'UACCUGGUUGAUCCUGCCAGUAG-CAUAUGCUUGUCUCAAAGAUUAAGCCAUGCAUGUCUGAGUACGCACGGCCGG-UACAGUGAAACUGCGAAUGGCUCAUUAAAUCAGUUAUGGUUCCUUUGGUCGCUCGCUCCUCUCCUACUUGGAUAACUGUGGUAAUUCUAGAGCUAAUACAUGCCGACGGGCGCUGACCCCCUUCGCGGGGGGGAUGCGUGCAUUUAUCAGAUCAAAACCAACCCGGUCAGCCCCUCUCCGGCCCCGGCCGGGGGGCGGGCGCCGGCGGCUUUGGUGACUCUAGAUAACCUCGGGCCGAUCGCACGCCCCCCGUGGCGGCGACGACCCAUUCGAACGUCUGCCCUAUCAACUUUCGAUGGUAGUCGCCGUGCCUACCAUGGUGACCACGGGUGACGGGGAAUCAGGGUUCGAUUCCGGAGAGGGAGCCUGAGAAACGGCUACCACAUCCAAGGAAGGCAGCAGGCGCGCAAAUUACCCACUCCCGACCCGGGGAGGUAGUGACGAAAAAUAACAAUACAGGACUCUUUCGAGGCCCUGUAAUUGGAAUGAGUCCACUUUAAAUCCUUUAACGAGGAUCCAUUGGAGGGCAAGUCUGGUGCCAGCAGCCGCGGUAAUUCCAGCUCCAAUAGCGUAUAUUAAAGUUGCUGCAGUUAAAAAGCUCGUAGUUGGAUCUUGGGAGCGGGCGGGCGGUCCGCCGCGAGGCGAGCCACCGCCCGUCCCCGCCCCUUGCCUCUCGGCGCCCCCUCGAUGCUCUUAGCUGAGUGUCCCGCGGGGC--CCGAAGCGUUUACUUUGAAAAAAUUAGAGUGUUCAAAGCAGGCCCGAGCCGCCUGGAUACCGCAGCUAGGAAUAAUGGAAUAGGACCG-CGGUUCUAUUUUGUUGGUUUUCGGAACUGAGGCCAUGAUUAAGAGGGACGGCCGGGGGCAUUCGUAUUGCGCCGCUAGAGGUGAAAUUCUUGGACCGGCGCAAGACGGACCAGAGCGAAAGCAUUUGCCAAGAAUGUUUUCAUUAAUCAAGAACGAAAGUCGGAGGUUCGAAGACGAUCAGAUACCGUCGUAGUUCCGACCAUAAACGAUGCCGACCGGCGAUGCGGCGGCGUUAUUCCCAUGACCCGCCGGGCAGCUUCCGGGAAACCAAAGUCUUUGGGUUCCGGGGGGAGUAUGGUUGCAAAGCUGAAACUUAAAGGAAUUGACGGAAGGGCACCACCAGGAGUGGAGCCUGCGGCUUAAUUUGACUCAACACGGGAAACCUCACCCGGCCCGGACACGGACAGGAUUGACAGAUUGAUAGCUCUUUCUCGAUUCCGUGGGUGGUGGUGCAUGGCCGUUCUUAGUUGGUGGAGCGAUUUGUCUGGUUAAUUCCGAUAACGAACGAGACUCUGGCAUGCUAACUAGUUACGCGACCCCCGAGCGGUCGGCGUCCCCCAACUUCUUAGAGGGACAAGUGGCGUUCAGCCACCCGAGAUUGA--GCAAUAACAGGUCUGUGAUGCCCUUAGAUGUCCGGGGCUGCACGCGCGCUACACUGACUGGCUCAGCGUGUGCCUACCCUACGCCGGCAGGCGCGGGUAACCCGUUGAACCCCAUUCGUGAUGGGGAUCGGGGAUUGCAAUUAUUCCCCAUGAACGAGGAAUUCCC--AGUAAGUGCGGGUCAUAAGCUUGCGUUGAUUAAGUCCCUGCCCUUUGUACACACCGCCCGUCGCUACUACCGAUUGGAUGGUUUAGUGAGGCCCUCGGAUCGGCCCCGCCGGGGUCGGCCCACGGCCCUGGCGGAGCGCUGAGAAGACGGUCGAACUUGACUAUCUAGAGGAAGUAAAAGUCGUAACAAGGUUUCCGUAGGUGAACCUGCGGAAGGAUCAUUA' + }, + { + id: '18S_Human_reference', + name: 'H.sapiens - ref', + shortname: 'Hs-Ref', + sequence: + 'UACCUGGUUGAUCCUGCCAGUAG-CAUAUGCUUGUCUCAAAGAUUAAGCCAUGCAUGUCUAAGUACGCACGGCCGG-UACAGUGAAACUGCGAAUGGCUCAUUAAAUCAGUUAUGGUUCCUUUGGUCGCUCGCUCCUCUCCUACUUGGAUAACUGUGGUAAUUCUAGAGCUAAUACAUGCCGACGGGCGCUGACCCCCUUCGCGGGGGGGAUGCGUGCAUUUAUCAGAUCAAAACCAACCCGGUCAGCCCCUCUCCGGCCCCGGCCGGGGGGCGGGCGCCGGCGGCUUUGGUGACUCUAGAUAACCUCGGGCCGAUCGCACGCCCCCCGUGGCGGCGACGACCCAUUCGAACGUCUGCCCUAUCAACUUUCGAUGGUAGUCGCCGUGCCUACCAUGGUGACCACGGGUGACGGGGAAUCAGGGUUCGAUUCCGGAGAGGGAGCCUGAGAAACGGCUACCACAUCCAAGGAAGGCAGCAGGCGCGCAAAUUACCCACUCCCGACCCGGGGAGGUAGUGACGAAAAAUAACAAUACAGGACUCUUUCGAGGCCCUGUAAUUGGAAUGAGUCCACUUUAAAUCCUUUAACGAGGAUCCAUUGGAGGGCAAGUCUGGUGCCAGCAGCCGCGGUAAUUCCAGCUCCAAUAGCGUAUAUUAAAGUUGCUGCAGUUAAAAAGCUCGUAGUUGGAUCUUGGGAGCGGGCGGGCGGUCCGCCGCGAGGCGAGCCACCGCCCGUCCCCGCCCCUUGCCUCUCGGCGCCCCCUCGAUGCUCUUAGCUGAGUGUCCCGCGGGGC--CCGAAGCGUUUACUUUGAAAAAAUUAGAGUGUUCAAAGCAGGCCCGAGCCGCCUGGAUACCGCAGCUAGGAAUAAUGGAAUAGGACCG-CGGUUCUAUUUUGUUGGUUUUCGGAACUGAGGCCAUGAUUAAGAGGGACGGCCGGGGGCAUUCGUAUUGCGCCGCUAGAGGUGAAAUUCUUGGACCGGCGCAAGACGGACCAGAGCGAAAGCAUUUGCCAAGAAUGUUUUCAUUAAUCAAGAACGAAAGUCGGAGGUUCGAAGACGAUCAGAUACCGUCGUAGUUCCGACCAUAAACGAUGCCGACCGGCGAUGCGGCGGCGUUAUUCCCAUGACCCGCCGGGCAGCUUCCGGGAAACCAAAGUCUUUGGGUUCCGGGGGGAGUAUGGUUGCAAAGCUGAAACUUAAAGGAAUUGACGGAAGGGCACCACCAGGAGUGGAGCCUGCGGCUUAAUUUGACUCAACACGGGAAACCUCACCCGGCCCGGACACGGACAGGAUUGACAGAUUGAUAGCUCUUUCUCGAUUCCGUGGGUGGUGGUGCAUGGCCGUUCUUAGUUGGUGGAGCGAUUUGUCUGGUUAAUUCCGAUAACGAACGAGACUCUGGCAUGCUAACUAGUUACGCGACCCCCGAGCGGUCGGCGUCCCCCAACUUCUUAGAGGGACAAGUGGCGUUCAGCCACCCGAGAUUGA--GCAAUAACAGGUCUGUGAUGCCCUUAGAUGUCCGGGGCUGCACGCGCGCUACACUGACUGGCUCAGCGUGUGCCUACCCUACGCCGGCAGGCGCGGGUAACCCGUUGAACCCCAUUCGUGAUGGGGAUCGGGGAUUGCAAUUAUUCCCCAUGAACGAGGGAAUUCCCGAGUAAGUGCGGGUCAUAAGCUUGCGUUGAUUAAGUCCCUGCCCUUUGUACACACCGCCCGUCGCUACUACCGAUUGGAUGGUUUAGUGAGGCCCUCGGAUCGGCCCCGCCGGGGUCGGCCCACGGCCCUGGCGGAGCGCUGAGAAGACGGUCGAACUUGACUAUCUAGAGGAAGUAAAAGUCGUAACAAGGUUUCCGUAGGUGAACCUGCGGAAGGAUCAUUA' + } + ] +} + +/** + * Component props. + */ +const props = defineProps<{ + /** The mode in which the alignment form is at landing (usually passed + * from URL). */ + initialMode: AlignmentModesType +}>() + +/** + * Vue Router instance reactive object. + */ +const router = useRouter() + +/** + * The currently selected mode. + */ +const activeMode = ref(props.initialMode) + +/** + * Title of the page, reactively updated when data is fetched. + */ +const pageTitle = computed( + () => `${_capitalize(activeMode.value)} • Alignment | SnoBoard` +) +// Bind actual page title to computed one +onMounted(() => useTitle(pageTitle)) + +/** + * Callback to change the URL when switching tab. + * @param e The event emitted when changing tab. + */ +const updateUrlQuery = (newMode: AlignmentModesType) => { + if (newMode) { + router.replace({ + name: 'alignment', + query: { mode: newMode } + }) + } +} + +/** + * The IDs of currently selected sequences. + */ +const selectedSequenceIds = ref<string[]>([]) +</script> + +<template> + <MainLayout padded> + <h1 class="mb-8 text-center text-3xl font-semibold text-slate-700"> + <icon-snoboard-alignment class="mb-1 mr-2 inline" /> + Sequences alignment + </h1> + + <AlignmentForm + v-model:current-mode-model="activeMode" + v-model:selected-ids-model="selectedSequenceIds" + class="mb-8" + @update:current-mode-model="updateUrlQuery" + /> + + <div class="my-8"> + <SequenceAlignment :alignment="ALIGNMENT" /> + </div> + </MainLayout> +</template> diff --git a/src/views/ClusterView.vue b/src/views/ClusterView.vue index 769549f121408f4d998e8c48c0bcd030c1d7dce8..a0bee4e2b1f2df59ad2b7b7e823a0d8a8518a097 100644 --- a/src/views/ClusterView.vue +++ b/src/views/ClusterView.vue @@ -16,11 +16,14 @@ import IconFa6SolidArrowUpRightFromSquare from '~icons/fa6-solid/arrow-up-right- import IconFa6SolidList from '~icons/fa6-solid/list' import IconEmojioneThinkingFace from '~icons/emojione/thinking-face' /** - * Other 3rd-party imports + * Composables imports */ -import { omit as _omit, minBy as _minBy, maxBy as _maxBy } from 'lodash-es' import { useQuery } from '@urql/vue' import { useTitle } from '@vueuse/core' +/** + * Other 3rd-party imports + */ +import { omit as _omit, minBy as _minBy, maxBy as _maxBy } from 'lodash-es' /** * Types imports */ diff --git a/src/views/DataTableView.vue b/src/views/DataTableView.vue index 235f9d03773e359d834159ddb0e4a8ff858aa7f7..acbaaac7cde290d3459849b59e20217a422ad269 100644 --- a/src/views/DataTableView.vue +++ b/src/views/DataTableView.vue @@ -25,6 +25,10 @@ import IconFa6SolidCircleQuestion from '~icons/fa6-solid/circle-question' import IconFa6SolidCircleXmark from '~icons/fa6-solid/circle-xmark' import IconFa6SolidFilter from '~icons/fa6-solid/filter' import IconSnoboardFilterWarning from '~icons/snoboard/filter-warning' +/** + * Composables imports + */ +import { useQuery } from '@urql/vue' /** * Other 3rd-party imports */ @@ -36,7 +40,6 @@ import { findLast as _findLast } from 'lodash-es' import { FilterMatchMode as primeVueFilterMatchMode } from 'primevue/api' -import { useQuery } from '@urql/vue' /** * Types imports */ @@ -265,46 +268,36 @@ const targetNameOptionsBase = computed(() => /** * Available options for target name selection, grouped. */ -const targetNameOptionsWithDisablingGroups = computed(() => +const targetNameOptionGroups = computed(() => targetNameOptionsBase.value.reduce< { label: string; children: { label: string; value: any }[] }[] - >((targetNameOptionsWithDisablingGroups, targetNameOption) => { + >((targetNameOptionGroups, targetNameOption) => { // Get index of the group of the current option - const targetNameOptionGroupIndex = - targetNameOptionsWithDisablingGroups.findIndex( - (group) => group.label === targetNameOption.groupLabel - ) + const targetNameOptionGroupIndex = targetNameOptionGroups.findIndex( + (group) => group.label === targetNameOption.groupLabel + ) // Insert option in existing group if present // Using non-null type assertion for - // `targetNameOptionsWithDisablingGroups[targetNameOptionGroupIndex]` + // `targetNameOptionGroups[targetNameOptionGroupIndex]` // because it is checked before return targetNameOptionGroupIndex !== -1 ? [ - ...targetNameOptionsWithDisablingGroups.slice( - 0, - targetNameOptionGroupIndex - ), + ...targetNameOptionGroups.slice(0, targetNameOptionGroupIndex), { - label: - targetNameOptionsWithDisablingGroups[targetNameOptionGroupIndex]! - .label, + label: targetNameOptionGroups[targetNameOptionGroupIndex]!.label, children: [ - ...targetNameOptionsWithDisablingGroups[ - targetNameOptionGroupIndex - ]!.children, + ...targetNameOptionGroups[targetNameOptionGroupIndex]!.children, { label: targetNameOption.label || '', value: targetNameOption.value } ] }, - ...targetNameOptionsWithDisablingGroups.slice( - targetNameOptionGroupIndex + 1 - ) + ...targetNameOptionGroups.slice(targetNameOptionGroupIndex + 1) ] : // Otherwise add the group with the option [ - ...targetNameOptionsWithDisablingGroups, + ...targetNameOptionGroups, { label: targetNameOption.groupLabel, children: [ @@ -733,7 +726,7 @@ const specificFilterConfigs = computed<{ columns.find((column) => column.id == 'targetName')?.field || '', filter: { matchMode: primeVueFilterMatchMode.IN, - options: targetNameOptionsWithDisablingGroups.value + options: targetNameOptionGroups.value }, grouped: true }, diff --git a/src/views/GuideView.vue b/src/views/GuideView.vue index 004dc3bc91150dd534becb3b66af2fceef5536f5..d518217bdfabf68a820f773b15c85e08684e8556 100644 --- a/src/views/GuideView.vue +++ b/src/views/GuideView.vue @@ -26,17 +26,20 @@ import IconFa6SolidCirclePlus from '~icons/fa6-solid/circle-plus' import IconFa6SolidCircleQuestion from '~icons/fa6-solid/circle-question' import IconFa6SolidCircleXmark from '~icons/fa6-solid/circle-xmark' /** - * Other 3rd-party imports + * Composables imports */ -import { uniq as _uniq } from 'lodash-es' import { useQuery } from '@urql/vue' import { useTitle } from '@vueuse/core' +/** + * Other 3rd-party imports + */ +import { uniq as _uniq } from 'lodash-es' /** * Types import */ import type { TailwindDefaultColorNameModel } from '@/typings/styleTypes' import type { InteractionCardModel } from '@/components/InteractionCard.vue' -import type { highlightGroupModel } from '@/components/SequenceBoard.vue' +import type { objectModel } from '@/components/SequenceBoard.vue' import { GraphQlType, ModifType, @@ -269,7 +272,7 @@ const filteredFacingModifications = computed(() => */ const sequenceChunks = computed(() => ({ ...guide.value?.boxConnections.edges.reduce<{ - [groupId: string]: highlightGroupModel + [groupId: string]: objectModel }>((boxes, boxConnection) => { return boxConnection.node.annotation ? { @@ -288,7 +291,7 @@ const sequenceChunks = computed(() => ({ : boxes }, {}), ...filteredFacingModifications.value?.reduce<{ - [groupId: string]: highlightGroupModel + [groupId: string]: objectModel }>( (sequenceChunks, facingModification) => ({ ...sequenceChunks, @@ -789,7 +792,7 @@ const selectedGraphicsTab = ref(GraphicsTabEnum[props.initialGraphicsPanelTab]) v-if="guide?.seq" :sequence="guide.seq.replace(/T/g, 'U')" :sequence-id="guide.id" - :highlighted-groups="sequenceChunks" + :objects="sequenceChunks" :legend-items="GUIDE_LEGEND_ITEMS" > <template #legend-item-title="{ item }"> @@ -802,7 +805,7 @@ const selectedGraphicsTab = ref(GraphicsTabEnum[props.initialGraphicsPanelTab]) </span> </template> - <template #tooltip-item="{ group }"> + <template #tooltip-item="{ object: group }"> <Chip v-if="group.type && isInEnum(group.type, ModifType)" :class="[ diff --git a/src/views/HomeView.vue b/src/views/HomeView.vue index 4097e8afbf6145dfaa55ed1a0350fe6933ea8b45..65e5b081470b069e103e86ee35f6f9042a54b507 100644 --- a/src/views/HomeView.vue +++ b/src/views/HomeView.vue @@ -3,7 +3,6 @@ * Vue imports */ import { computed, ref } from 'vue' -import { useRouter } from 'vue-router' /** * Components imports */ @@ -28,9 +27,13 @@ import IconTablerCircleArrowDownFilled from '~icons/tabler/circle-arrow-down-fil import IconSnoboardGuide from '~icons/snoboard/guide' import IconSnoboardModification from '~icons/snoboard/modification' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' +import { useRouter } from 'vue-router' +/** + * Other 3rd-party imports + */ import { uniqWith as _uniqWith, isEqual as _isEqual } from 'lodash-es' /** * Types imports. @@ -302,39 +305,39 @@ const menuItems = computed<MenuItem[]>(() => [ key: 'conservation', label: 'Conservation', items: [ - { - key: 'conservationModifications', - label: 'Modifications', - route: { - // name: 'conservation', - // query: { - // type: 'modification' - // } - }, - iconComponent: IconSnoboardModification, - disabled: true - }, + // { + // key: 'conservationModifications', + // label: 'Modifications', + // route: { + // name: 'alignment' + // // query: { + // // type: 'modification' + // // } + // }, + // iconComponent: IconSnoboardModification, + // disabled: true + // }, { key: 'conservationGuides', label: 'Guides', - route: { - // name: 'conservation', - // query: { - // type: 'modification' - // } - }, + // route: { + // name: 'alignment', + // query: { + // mode: 'guide' + // } + // }, iconComponent: IconSnoboardGuide, disabled: true }, { key: 'conservationTargets', label: 'Targets', - route: { - // name: 'conservation', - // query: { - // type: 'modification' - // } - }, + // route: { + // name: 'alignment', + // query: { + // mode: 'target' + // } + // }, iconComponent: IconFa6SolidBullseye, disabled: true } diff --git a/src/views/LegalView.vue b/src/views/LegalView.vue index 03dc9222ea51160ba7ae273db20b5f656b0a6def..b0b28e258d3759391f5c12c4dd29c81027473359 100644 --- a/src/views/LegalView.vue +++ b/src/views/LegalView.vue @@ -9,7 +9,7 @@ import { computed, toRef } from 'vue' import MainLayout from '@/layouts/MainLayout.vue' import BaseRenderedMarkdown from '@/components/BaseRenderedMarkdown.vue' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' /** diff --git a/src/views/ModificationView.vue b/src/views/ModificationView.vue index db242e8a601882df0f82378480acb12048df1862..ff226c126b3c5ad16787c5af3f68c7eb58d93237 100644 --- a/src/views/ModificationView.vue +++ b/src/views/ModificationView.vue @@ -18,10 +18,13 @@ import Tag from 'primevue/tag' import IconFa6SolidCircleInfo from '~icons/fa6-solid/circle-info' import IconFa6SolidDrawPolygon from '~icons/fa6-solid/draw-polygon' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' import { useTitle } from '@vueuse/core' +/** + * Other 3rd-party imports + */ import { sortBy as _sortBy } from 'lodash-es' /** * Types imports diff --git a/src/views/OrganismView.vue b/src/views/OrganismView.vue index 19b000064e708a9068ecdd61b5bb30adb513137f..3379ae7e0bb87928172583f08f2ad8bf12a74306 100644 --- a/src/views/OrganismView.vue +++ b/src/views/OrganismView.vue @@ -25,10 +25,13 @@ import IconFa6SolidFileLines from '~icons/fa6-solid/file-lines' import IconCarbonAccessibilityColor from '~icons/carbon/accessibility-color' import IconCarbonAccessibilityColorFilled from '~icons/carbon/accessibility-color-filled' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' import { useTitle } from '@vueuse/core' +/** + * Other 3rd-party imports + */ import { sortBy as _sortBy, capitalize as _capitalize, diff --git a/src/views/SelectionView.vue b/src/views/SelectionView.vue index b6b100ac642f78a99f519fbfcd93d8f3485add9e..cf4fd8e6607ebbc3c2cf00cd2fccb18370d86d7f 100644 --- a/src/views/SelectionView.vue +++ b/src/views/SelectionView.vue @@ -3,48 +3,39 @@ * Vue imports */ import { computed, onMounted, ref } from 'vue' -import { useRouter } from 'vue-router' /** * Components imports */ import MainLayout from '@/layouts/MainLayout.vue' -import ModificationSelectionForm, { - type ModificationSelectionModel -} from '@/components/ModificationSelectionForm.vue' -import GuideSelectionForm, { - type GuideSelectionModel -} from '@/components/GuideSelectionForm.vue' -import TargetSelectionForm, { - type TargetSelectionModel -} from '@/components/TargetSelectionForm.vue' -import TabView, { type TabViewChangeEvent } from 'primevue/tabview' -import TabPanel from 'primevue/tabpanel' -import Card from 'primevue/card' +import SelectionForm from '@/components/SelectionForm.vue' import Button from 'primevue/button' +import IconFa6SolidSliders from '~icons/fa6-solid/sliders' import IconFa6SolidTable from '~icons/fa6-solid/table' import IconFa6SolidBullseye from '~icons/fa6-solid/bullseye' import IconFa6SolidCircleNodes from '~icons/fa6-solid/circle-nodes' import IconSnoboardSequence from '~icons/snoboard/sequence' /** - * Other 3rd-party imports + * Composables imports */ +import { useRouter } from 'vue-router' import { useTitle } from '@vueuse/core' +/** + * Other 3rd-party imports + */ import { capitalize as _capitalize } from 'lodash-es' +/** + * Types imports + */ +import type { SelectionModel } from '@/components/SelectionForm.vue' /** * Utils imports */ -import { SELECTION_FORM_MODES } from '@/utils/constant' -import { getColorWithOptionalShade } from '@/utils/colors' +import { SELECTION_MODES } from '@/utils/constant' /** * Type of data to select. */ -export type SelectionFormModesType = (typeof SELECTION_FORM_MODES)[number] -enum TabEnum { - Modification = 0, - Guide, - Target -} +export type SelectionModesType = (typeof SELECTION_MODES)[number] /** * A selection option. @@ -99,7 +90,9 @@ const TABLE_COLUMNS_BY_MODE = { * Component props. */ const props = defineProps<{ - mode: SelectionFormModesType + /** The mode in which the selection form is at landing (usually passed + * from URL). */ + initialMode: SelectionModesType }>() /** @@ -108,16 +101,9 @@ const props = defineProps<{ const router = useRouter() /** - * The index of the tab currently selected, reactively updated when changed. - */ -const activeTabIndex = ref<TabEnum>( - SELECTION_FORM_MODES.findIndex((mode) => mode === props.mode) -) - -/** - * The mode corresponding to the tab currently selected. + * The currently selected mode. */ -const activeMode = computed(() => SELECTION_FORM_MODES[activeTabIndex.value]) +const activeMode = ref(props.initialMode) /** * Title of the page, reactively updated when data is fetched. @@ -132,21 +118,19 @@ onMounted(() => useTitle(pageTitle)) * Callback to change the URL when switching tab. * @param e The event emitted when changing tab. */ -const updateUrlQuery = (e: TabViewChangeEvent) => { - const newMode = SELECTION_FORM_MODES[e.index] +const updateUrlQuery = (newMode: SelectionModesType) => { if (newMode) { - router.replace({ name: 'selection', query: { mode: newMode } }) + router.replace({ + name: 'selection', + query: { mode: newMode } + }) } } /** * The current selection. */ -const selection = ref<{ - modification?: ModificationSelectionModel - guide?: GuideSelectionModel - target?: TargetSelectionModel -}>({ +const selection = ref<SelectionModel>({ modification: undefined, guide: undefined, target: undefined @@ -159,7 +143,7 @@ const selection = ref<{ const tableFilters = computed(() => ({ guideSubclass: activeMode.value === 'guide' - ? selection.value[activeMode.value]?.guideSubclasses + ? selection.value.guide?.guideSubclassLabels : undefined, targetName: selection.value[activeMode.value]?.targetNames, modificationType: selection.value[activeMode.value]?.modificationTypes, @@ -191,152 +175,105 @@ const onlyTargetIdActiveMode = computed(() => <template> <MainLayout padded> - <h1 class="text-center text-3xl font-semibold text-slate-700"> + <h1 class="mb-8 text-center text-3xl font-semibold text-slate-700"> + <icon-fa6-solid-sliders class="mb-1 mr-2 inline" /> Advanced selection </h1> - <Card - class="mx-16 !rounded-2xl border-none !shadow-none" - :pt="{ - footer: { - style: { - display: 'flex', - justifyContent: 'center' + <SelectionForm + v-model:current-mode-model="activeMode" + v-model:selection-model="selection" + v-model:only-target-id-by-mode-model="onlyTargetIdByMode" + @update:current-mode-model="updateUrlQuery" + /> + + <div class="mt-8 flex justify-center gap-4"> + <RouterLink + :to="{ + name: 'table', + query: { + columns: TABLE_COLUMNS_BY_MODE[activeMode], + ...tableFilters } - } - }" - > - <template #content> - <TabView - v-model:active-index="activeTabIndex" - :pt="{ - nav: { - style: { - justifyContent: 'center', - marginBottom: '2rem', - fontSize: '1.25em', - background: `linear-gradient(white 0 0) padding-box, - linear-gradient(90deg, - ${getColorWithOptionalShade('slate', '300')}00 10%, - ${getColorWithOptionalShade('slate', '300')} 15%, - ${getColorWithOptionalShade('slate', '300')} 85%, - ${getColorWithOptionalShade('slate', '300')}00 90%) - border-box`, - borderStyle: 'dashed', - borderColor: 'white' + }" + > + <Button> + <icon-fa6-solid-table /> + <span class="ml-2">View in table</span> + </Button> + </RouterLink> + + <div + v-if="activeMode === 'target'" + v-tooltip.top=" + !onlyTargetIdByMode.target && { + value: 'Select a unique target/species pair', + pt: { + text: { + style: { + textAlign: 'center', + fontStyle: 'italic' + } } } + } + " + > + <RouterLink + :to="{ + name: 'targetDetails', + query: { + id: onlyTargetIdByMode.target + }, + hash: '#sequence-panel' }" - @tab-change="updateUrlQuery" + :class="{ 'pointer-events-none': !onlyTargetIdByMode.target }" > - <TabPanel header="Modification"> - <ModificationSelectionForm - v-model:selection-model="selection.modification" - v-model:only-target-id="onlyTargetIdByMode.modification" - /> - </TabPanel> - <TabPanel header="Guide"> - <GuideSelectionForm v-model:selection-model="selection.guide" /> - </TabPanel> - <TabPanel header="Target"> - <TargetSelectionForm - v-model:selection-model="selection.target" - v-model:only-target-id="onlyTargetIdByMode.target" - /> - </TabPanel> - </TabView> - </template> + <Button :disabled="!onlyTargetIdByMode.target"> + <icon-snoboard-sequence /> + <span class="ml-2">View sequence</span> + </Button> + </RouterLink> + </div> - <template #footer> - <div class="flex gap-4"> - <RouterLink - :to="{ - name: 'table', - query: { - columns: TABLE_COLUMNS_BY_MODE[activeMode], - ...tableFilters - } - }" - > - <Button> - <icon-fa6-solid-table /> - <span class="ml-2">View in table</span> - </Button> - </RouterLink> - - <div - v-if="activeMode === 'target'" - v-tooltip.top=" - !onlyTargetIdByMode.target && { - value: 'Select a unique target/species pair', - pt: { - text: { - style: { - textAlign: 'center', - fontStyle: 'italic' - } - } + <div + v-if="activeMode === 'target' || activeMode === 'modification'" + v-tooltip.top=" + !onlyTargetIdActiveMode && { + value: 'Select a unique target/species pair', + pt: { + text: { + style: { + textAlign: 'center', + fontStyle: 'italic' } } - " - > - <RouterLink - :to="{ - name: 'targetDetails', - query: { - id: onlyTargetIdByMode.target - }, - hash: '#sequence-panel' - }" - :class="{ 'pointer-events-none': !onlyTargetIdByMode.target }" - > - <Button :disabled="!onlyTargetIdByMode.target"> - <icon-snoboard-sequence /> - <span class="ml-2">View sequence</span> - </Button> - </RouterLink> - </div> - - <div - v-if="activeMode === 'target' || activeMode === 'modification'" - v-tooltip.top=" - !onlyTargetIdActiveMode && { - value: 'Select a unique target/species pair', - pt: { - text: { - style: { - textAlign: 'center', - fontStyle: 'italic' - } - } - } - } - " - > - <RouterLink - :to="{ - name: 'targetDetails', - query: { - id: onlyTargetIdActiveMode, - graphicsTab: '2d-struct' - }, - hash: '#graphics-panel' - }" - :class="{ 'pointer-events-none': !onlyTargetIdActiveMode }" - > - <Button :disabled="!onlyTargetIdActiveMode"> - <icon-fa6-solid-bullseye v-if="activeMode === 'modification'" /> - <icon-fa6-solid-circle-nodes v-else /> - <span class="ml-2"> - View - {{ activeMode === 'modification' ? 'on target' : '' }} - structure - </span> - </Button> - </RouterLink> - </div> - </div> - </template> - </Card> + } + } + " + > + <RouterLink + :to="{ + name: 'targetDetails', + query: { + id: onlyTargetIdActiveMode, + graphicsTab: '2d-struct' + }, + hash: '#graphics-panel' + }" + :class="{ 'pointer-events-none': !onlyTargetIdActiveMode }" + > + <Button :disabled="!onlyTargetIdActiveMode"> + <icon-fa6-solid-bullseye v-if="activeMode === 'modification'" /> + <icon-fa6-solid-circle-nodes v-else /> + <span class="ml-2"> + View + {{ activeMode === 'modification' ? 'on target' : '' }} + structure + </span> + </Button> + </RouterLink> + </div> + </div> </MainLayout> </template> diff --git a/src/views/StatisticsView.vue b/src/views/StatisticsView.vue index 9729c7818eccb53e49cf6d35c2c67d0ad6b89293..e2459d6fcece7382dc3341ccc00eb976c8b938b9 100644 --- a/src/views/StatisticsView.vue +++ b/src/views/StatisticsView.vue @@ -17,9 +17,12 @@ import IconCarbonAccessibilityColorFilled from '~icons/carbon/accessibility-colo import IconSnoboardModification from '~icons/snoboard/modification' import IconSnoboardGuide from '~icons/snoboard/guide' /** - * Other 3rd-party imports + * Composables imports */ import { useQuery } from '@urql/vue' +/** + * Other 3rd-party imports + */ import { countBy as _countBy } from 'lodash-es' import pattern from 'patternomaly' /** diff --git a/src/views/TargetView.vue b/src/views/TargetView.vue index b9bf64cd3581accc925443b57b8c9bd5a2c86933..6c48a2fe5f3a559ffe67b8f0cc6531ebc9c2504f 100644 --- a/src/views/TargetView.vue +++ b/src/views/TargetView.vue @@ -28,18 +28,25 @@ import IconFa6SolidCirclePlus from '~icons/fa6-solid/circle-plus' import IconFa6SolidCircleQuestion from '~icons/fa6-solid/circle-question' import IconFa6SolidCircleXmark from '~icons/fa6-solid/circle-xmark' /** - * Other 3rd-party imports + * Composables imports */ -import { find as _find, sortBy as _sortBy } from 'lodash-es' import { useQuery } from '@urql/vue' import { useFetch, useTitle } from '@vueuse/core' +/** + * Other 3rd-party imports + */ +import { + find as _find, + sortBy as _sortBy, + mapValues as _mapValues +} from 'lodash-es' /** * Types import */ import type { TailwindDefaultColorNameModel } from '@/typings/styleTypes' import type { InteractionCardModel } from '@/components/InteractionCard.vue' -import type { C4GGraphModel, C4GNodeModel } from '@/typings/Codev4GraphFormat' -import type { highlightGroupModel } from '@/components/SequenceBoard.vue' +import type { C4GGraphModel } from '@/typings/Codev4GraphFormat' +import type { objectModel } from '@/components/SequenceBoard.vue' import { GraphQlType, ModifType, @@ -173,37 +180,28 @@ const secondaryStructureWithModificationsMetadata = secondaryStructure.value && { graph: { ...secondaryStructure.value?.graph, - nodes: Object.entries( - secondaryStructure.value?.graph.nodes || {} - ).reduce<{ [nodeId: string]: C4GNodeModel }>( - (nodes, [nodeId, node]) => { - const nodeModification = _find(target.value?.modifications, [ - 'position', - _find(node.metadata?.data, ['label', 'nucleotidePosition']) - ?.value - ]) - return nodeModification - ? { - ...nodes, - [`${nodeId}`]: { - ...node, - metadata: { - ...node.metadata, - data: [ - ...(node.metadata?.data || []), - { - label: 'modification', - type: 'number', - value: nodeModification - } - ] + nodes: _mapValues(secondaryStructure.value?.graph.nodes, (node) => { + const nodeModification = _find(target.value?.modifications, [ + 'position', + _find(node.metadata?.data, ['label', 'nucleotidePosition'])?.value + ]) + return nodeModification + ? { + ...node, + metadata: { + ...node.metadata, + data: [ + ...(node.metadata?.data || []), + { + label: 'modification', + type: 'number', + value: nodeModification } - } + ] } - : { ...nodes, [`${nodeId}`]: node } - }, - {} - ) + } + : node + }) } } ) @@ -313,7 +311,7 @@ const filteredModifications = computed( */ const sequenceChunks = computed(() => filteredModifications.value.reduce<{ - [groupId: string]: highlightGroupModel + [groupId: string]: objectModel }>( (sequenceChunks, modification) => ({ ...sequenceChunks, @@ -698,7 +696,7 @@ const selectedGraphicsTab = ref(GraphicsTabEnum[props.initialGraphicsPanelTab]) v-if="target?.seq" :sequence="target.seq.replace(/T/g, 'U')" :sequence-id="target.id" - :highlighted-groups="sequenceChunks" + :objects="sequenceChunks" :legend-items="TARGET_LEGEND_ITEMS" > <template #legend-item-title="{ item }"> @@ -710,7 +708,7 @@ const selectedGraphicsTab = ref(GraphicsTabEnum[props.initialGraphicsPanelTab]) /> </template> - <template #tooltip-item="{ group }"> + <template #tooltip-item="{ object: group }"> <Chip v-if="group.type && isInEnum(group.type, ModifType)" :class="[ diff --git a/tailwind.config.js b/tailwind.config.js index 0b7312ea8f7f42178c407698151f0f38fc73ec3a..80e2887f6afbd23c8c3a1c4c5e31e0870390afa0 100644 --- a/tailwind.config.js +++ b/tailwind.config.js @@ -53,6 +53,8 @@ export default { pattern: /^stroke-[a-z]+-600$/ }, 'brightness-90', // ChromosomeMagnify - 'cursor-pointer' // ChromosomeMagnify + 'cursor-pointer', // ChromosomeMagnify + 'text-xs', // SecondaryStructure, SequenceAlignment (tooltip) + 'text-center' // SecondaryStructure, SequenceAlignment (tooltip) ] }