diff --git a/.github/workflows/docker_img.yml b/.github/workflows/docker_img.yml index 3470644fb386722065358045c226d8989a208b34..76e8788819f6139821b2794a5452ae25ee55d7fe 100644 --- a/.github/workflows/docker_img.yml +++ b/.github/workflows/docker_img.yml @@ -24,23 +24,26 @@ jobs: echo "BRANCH_NAME=${GITHUB_REF#refs/heads/}" >> $GITHUB_ENV + echo "MATOMO_URL=${{ env.MATOMO_URL_PROD }}" >> $GITHUB_ENV + echo "MATOMO_ID=${{ env.MATOMO_ID_PROD }}" >> $GITHUB_ENV + if [[ "$GITHUB_REF" == 'refs/heads/master' ]] then echo "Either master, using prod env..." - echo "MATOMO_URL=${{ env.MATOMO_URL_PROD }}" >> $GITHUB_ENV - echo "MATOMO_ID=${{ env.MATOMO_ID_PROD }}" >> $GITHUB_ENV echo "BS_REST_URL=${{ env.SIIBRA_API_STABLE }}" >> $GITHUB_ENV elif [[ "$GITHUB_REF" == 'refs/heads/staging' ]] then echo "Either staging, using staging env..." - echo "MATOMO_URL=${{ env.MATOMO_URL_PROD }}" >> $GITHUB_ENV - echo "MATOMO_ID=${{ env.MATOMO_ID_PROD }}" >> $GITHUB_ENV echo "BS_REST_URL=${{ env.SIIBRA_API_RC }}" >> $GITHUB_ENV else - echo "Using dev env..." - echo "MATOMO_URL=${{ env.MATOMO_URL_DEV }}" >> $GITHUB_ENV - echo "MATOMO_ID=${{ env.MATOMO_ID_DEV }}" >> $GITHUB_ENV - echo "BS_REST_URL=${{ env.SIIBRA_API_LATEST }}" >> $GITHUB_ENV + if [[ "$GITHUB_REF" == *hotfix* ]] + then + echo "Hotfix branch, using prod env..." + echo "BS_REST_URL=${{ env.SIIBRA_API_STABLE }}" >> $GITHUB_ENV + else + echo "Using dev env..." + echo "BS_REST_URL=${{ env.SIIBRA_API_LATEST }}" >> $GITHUB_ENV + fi fi - name: 'Set version variable & expmt feature flag' diff --git a/Dockerfile b/Dockerfile index 23dea08fc6aa3b0cbc1e837707d7d6765b761e62..bc614aacaf8dcef73da19f4a59285e6a31e51daf 100644 --- a/Dockerfile +++ b/Dockerfile @@ -7,7 +7,7 @@ ARG DATASET_PREVIEW_URL ENV DATASET_PREVIEW_URL=${DATASET_PREVIEW_URL:-https://hbp-kg-dataset-previewer.apps.hbp.eu/v2} ARG BS_REST_URL -ENV BS_REST_URL=${BS_REST_URL:-https://siibra-api-latest.apps-dev.hbp.eu/v1_0} +ENV BS_REST_URL=${BS_REST_URL:-https://siibra-api-stable.apps.hbp.eu/v1_0} ARG STRICT_LOCAL ENV STRICT_LOCAL=${STRICT_LOCAL:-false} diff --git a/build_env.md b/build_env.md index 0cb122e3194e0efbf775678d65a0590bdd4f4063..1582bcdb0a487e3a91a88ecdcec45df445a687b9 100644 --- a/build_env.md +++ b/build_env.md @@ -7,7 +7,7 @@ As interactive atlas viewer uses [webpack define plugin](https://webpack.js.org/ | `VERSION` | printed in console on viewer startup | `GIT_HASH` \|\| unspecificed hash | v2.2.2 | | `PRODUCTION` | if the build is for production, toggles optimisations such as minification | `undefined` | true | | `BACKEND_URL` | backend that the viewer calls to fetch available template spaces, parcellations, plugins, datasets | `null` | https://interactive-viewer.apps.hbp.eu/ | -| `BS_REST_URL` | [siibra-api](https://github.com/FZJ-INM1-BDA/siibra-api) used to fetch different resources | https://siibra-api-latest.apps-dev.hbp.eu/v1_0 | +| `BS_REST_URL` | [siibra-api](https://github.com/FZJ-INM1-BDA/siibra-api) used to fetch different resources | https://siibra-api-stable.apps.hbp.eu/v1_0 | | `DATASET_PREVIEW_URL` | dataset preview url used by component <https://github.com/fzj-inm1-bda/kg-dataset-previewer>. Useful for diagnosing issues with dataset previews.| https://hbp-kg-dataset-previewer.apps.hbp.eu/datasetPreview | http://localhost:1234/datasetPreview | | `MATOMO_URL` | base url for matomo analytics | `null` | https://example.com/matomo/ | | `MATOMO_ID` | application id for matomo analytics | `null` | 6 | diff --git a/common/constants.js b/common/constants.js index ceeb3d027330bb5167bd469abe812569d8be6da0..a4880c3a454d1d505182126643a24fab6c197a79 100644 --- a/common/constants.js +++ b/common/constants.js @@ -22,6 +22,7 @@ PIN_DATASET: 'Toggle pinning dataset', TEXT_INPUT_SEARCH_REGION: 'Search for any region of interest in the atlas selected', CLEAR_SELECTED_REGION: 'Clear selected region', + VIEW_PINNED: `View pinned datasets.`, BULK_DOWNLOAD: `Download all pinned data`, NO_BULK_DOWNLOAD: `No datasets pinned`, @@ -73,7 +74,8 @@ EXIT_ANNOTATION_MODE: 'Exit annotation mode', // volume tuning specific - VOLUME_TUNING_EXPAND: 'Expand volume tuning widget' + VOLUME_TUNING_EXPAND: 'Expand volume tuning widget', + BULK_DELETE_ANNOTATIONS: 'Delete all user annotations', } exports.IDS = { @@ -82,6 +84,8 @@ } exports.CONST = { + LOADING_TXT: `Loading ...`, + CANNOT_DECIPHER_HEMISPHERE: 'Cannot decipher region hemisphere.', DOES_NOT_SUPPORT_MULTI_REGION_SELECTION: `Please only select a single region.`, MULTI_REGION_SELECTION: `Multi region selection`, @@ -106,6 +110,9 @@ QUICKTOUR_OK: `Start`, QUICKTOUR_NEXTTIME: `Not now`, QUICKTOUR_CANCEL: `Dismiss`, + + DELETE_ALL_ANNOTATION_CONFIRMATION_MSG: `Are you sure you want to delete all annotations?`, + LOADING_ANNOTATION_MSG: `Loading annotations... Please wait...` } exports.QUICKTOUR_DESC ={ diff --git a/deploy/app.js b/deploy/app.js index 29d6ab11ead3c319530104720e4708ec1f05986f..0a9aaa02df174ea5d934dddc6c7d998bd647f220 100644 --- a/deploy/app.js +++ b/deploy/app.js @@ -7,7 +7,7 @@ const crypto = require('crypto') const cookieParser = require('cookie-parser') const bkwdMdl = require('./bkwdCompat')() -const { router: regionalFeaturesRouter, regionalFeatureIsReady } = require('./regionalFeatures') +const deprecated = (_req, res) => res.status(410).end() const LOCAL_CDN_FLAG = !!process.env.PRECOMPUTED_SERVER @@ -202,11 +202,9 @@ app.use('/logo', require('./logo')) app.get('/ready', async (req, res) => { const authIsReady = authReady ? await authReady() : false - const regionalFeatureReady = await regionalFeatureIsReady() const allReady = [ authIsReady, - regionalFeatureReady, /** * add other ready endpoints here * call sig is await fn(): boolean @@ -234,24 +232,14 @@ const jsonMiddleware = (req, res, next) => { * resources endpoints */ const pluginRouter = require('./plugins') -const previewRouter = require('./preview') - -const setResLocalMiddleWare = routePathname => (req, res, next) => { - res.locals.routePathname = routePathname - next() -} - -const deprecated = (req, res) => { - res.status(404).send(`Route has been removed.`) -} app.use('/atlases', deprecated) app.use('/templates', deprecated) app.use('/nehubaConfig', deprecated) app.use('/datasets', deprecated) -app.use('/regionalFeatures', jsonMiddleware, regionalFeaturesRouter) +app.use('/regionalFeatures', deprecated) app.use('/plugins', jsonMiddleware, pluginRouter) -app.use('/preview', jsonMiddleware, previewRouter) +app.use('/preview', deprecated) const catchError = require('./catchError') app.use(catchError) diff --git a/deploy/auth/hbp-oidc.js b/deploy/auth/hbp-oidc.js deleted file mode 100644 index a3dfb823b7e500535da14969f8ea966df5957140..0000000000000000000000000000000000000000 --- a/deploy/auth/hbp-oidc.js +++ /dev/null @@ -1,44 +0,0 @@ -const passport = require('passport') -const { configureAuth } = require('./oidc') - -const HOSTNAME = process.env.HOSTNAME || 'http://localhost:3000' -const HOST_PATHNAME = process.env.HOST_PATHNAME || '' -const clientId = process.env.HBP_CLIENTID || 'no hbp id' -const clientSecret = process.env.HBP_CLIENTSECRET || 'no hbp client secret' -const discoveryUrl = 'https://services.humanbrainproject.eu/oidc' -const redirectUri = `${HOSTNAME}${HOST_PATHNAME}/hbp-oidc/cb` -const cb = (tokenset, {sub, given_name, family_name, ...rest}, done) => { - return done(null, { - id: `hbp-oidc:${sub}`, - name: `${given_name} ${family_name}`, - type: `hbp-oidc`, - tokenset, - rest - }) -} - -module.exports = async (app) => { - try { - const { oidcStrategy } = await configureAuth({ - clientId, - clientSecret, - discoveryUrl, - redirectUri, - cb, - scope: 'openid offline_access', - clientConfig: { - redirect_uris: [ redirectUri ], - response_types: [ 'code' ] - } - }) - - passport.use('hbp-oidc', oidcStrategy) - app.get('/hbp-oidc/auth', passport.authenticate('hbp-oidc')) - app.get('/hbp-oidc/cb', passport.authenticate('hbp-oidc', { - successRedirect: `${HOST_PATHNAME}/`, - failureRedirect: `${HOST_PATHNAME}/` - })) - } catch (e) { - console.error(e) - } -} diff --git a/deploy/auth/index.js b/deploy/auth/index.js index ebb1172d85e9abc8b553f2cc0c31f453f238c74f..413afb7088e2cfc62b429c07043a566045461308 100644 --- a/deploy/auth/index.js +++ b/deploy/auth/index.js @@ -10,16 +10,12 @@ const ready = async () => isReady const configureAuth = async (app) => { console.log('configure Auth') - const hbpOidc = require('./hbp-oidc') const { bootstrapApp: boostrapOidcV2 } = require('./hbp-oidc-v2') const { initPassportJs, objStoreDb } = require('./util') initPassportJs(app) - await retry(async () => { - await hbpOidc(app) - }, { timeout: 1000, retries: 3 }) await retry(async () => { await boostrapOidcV2(app) }, { timeout: 1000, retries: 3 }) diff --git a/deploy/auth/index.spec.js b/deploy/auth/index.spec.js index a75affe6ffa1c07c4456cb12d61c71d75e090d33..84691ddd6ccae9169f321858453c2eacf0dcb28c 100644 --- a/deploy/auth/index.spec.js +++ b/deploy/auth/index.spec.js @@ -1,8 +1,6 @@ const sinon = require('sinon') const { assert, expect } = require('chai') const initPassportJsStub = sinon.stub() - -const hbpOidcStub = sinon.stub() const hbpOidcV2Stub = sinon.stub() const appGetStub = sinon.stub() @@ -12,9 +10,6 @@ describe('auth/index.js', () => { require.cache[require.resolve('./util')] = { exports: { initPassportJs: initPassportJsStub } } - require.cache[require.resolve('./hbp-oidc')] = { - exports: hbpOidcStub - } require.cache[require.resolve('./hbp-oidc-v2')] = { exports: { bootstrapApp: hbpOidcV2Stub @@ -24,9 +19,7 @@ describe('auth/index.js', () => { beforeEach(() => { delete require.cache[require.resolve('./index.js')] - hbpOidcStub.returns({}) hbpOidcV2Stub.returns({}) - hbpOidcStub.resetHistory() hbpOidcV2Stub.resetHistory() }) @@ -37,11 +30,6 @@ describe('auth/index.js', () => { const dummyObj = { get: appGetStub } await configureAuth(dummyObj) - assert( - hbpOidcStub.called, - 'hbpOidc called' - ) - assert( hbpOidcV2Stub.called, 'hbpOidcV2 called' @@ -58,7 +46,7 @@ describe('auth/index.js', () => { const { configureAuth } = require('./index.js') const dummyObj = { get: appGetStub } - hbpOidcStub.throws(`throw error`) + hbpOidcV2Stub.throws(`throw error`) try { @@ -76,14 +64,10 @@ describe('auth/index.js', () => { } catch (e) { assert( - hbpOidcStub.calledThrice, - 'hbpOidc called thrice' + hbpOidcV2Stub.calledThrice, + 'hbpOidcv2 called thrice' ) - assert( - !hbpOidcV2Stub.called, - 'hbpOidcV2 not called' - ) } }) }) @@ -105,7 +89,7 @@ describe('auth/index.js', () => { const { configureAuth, ready } = require('./index.js') const dummyObj = { get: appGetStub } - hbpOidcStub.throws(`throw error`) + hbpOidcV2Stub.throws(`throw error`) try { await (() => new Promise((rs, rj) => { diff --git a/deploy/bkwdCompat/urlState.js b/deploy/bkwdCompat/urlState.js index f902a8a9f32a10dd843a0d0e4e561db18f28c7e3..66f92c3dc4f6b375fcfe0c70f16a685b43c30e0d 100644 --- a/deploy/bkwdCompat/urlState.js +++ b/deploy/bkwdCompat/urlState.js @@ -42,7 +42,10 @@ const templateMap = { id: 'minds/core/referencespace/v1.0.0/a1655b99-82f1-420f-a3c2-fe80fd4c8588', parc: { 'Cytoarchitectonic Maps - v2.4': { - id: 'juelich/iav/atlas/v1.0.0/7' + id: 'minds/core/parcellationatlas/v1.0.0/94c1125b-b87e-45e4-901c-00daee7f2579-290', + }, + 'Cytoarchitectonic Maps': { + id: 'minds/core/parcellationatlas/v1.0.0/94c1125b-b87e-45e4-901c-00daee7f2579-290', }, 'Cortical Layers Segmentation': { id: 'juelich/iav/atlas/v1.0.0/3' @@ -57,13 +60,14 @@ const templateMap = { id: 'minds/core/referencespace/v1.0.0/dafcffc5-4826-4bf1-8ff6-46b8a31ff8e2', parc: { 'Cytoarchitectonic Maps - v2.5.1': { - id: 'minds/core/parcellationatlas/v1.0.0/94c1125b-b87e-45e4-901c-00daee7f2579-26' + // redirect julich brain v251 to v290 + id: 'minds/core/parcellationatlas/v1.0.0/94c1125b-b87e-45e4-901c-00daee7f2579-290' }, 'Short Fiber Bundles - HCP': { id: 'juelich/iav/atlas/v1.0.0/79cbeaa4ee96d5d3dfe2876e9f74b3dc3d3ffb84304fb9b965b1776563a1069c' }, 'Cytoarchitectonic maps - v1.18': { - id: 'juelich/iav/atlas/v1.0.0/8' + id: 'minds/core/parcellationatlas/v1.0.0/94c1125b-b87e-45e4-901c-00daee7f2579' }, 'Long Bundle': { id: 'juelich/iav/atlas/v1.0.0/5' @@ -93,7 +97,7 @@ const templateMap = { id: 'minds/core/referencespace/v1.0.0/7f39f7be-445b-47c0-9791-e971c0b6d992', parc: { 'Cytoarchitectonic Maps - v1.18': { - id: 'juelich/iav/atlas/v1.0.0/8' + id: 'minds/core/parcellationatlas/v1.0.0/94c1125b-b87e-45e4-901c-00daee7f2579' } } }, @@ -202,8 +206,8 @@ module.exports = (query, _warningCb) => { // ignore region selected and move on } } - - let redirectUrl = '/#' + const HOST_PATHNAME = process.env.HOST_PATHNAME || '' + let redirectUrl = `${HOST_PATHNAME}/#` if (standaloneVolumes) { searchParam.set('standaloneVolumes', standaloneVolumes) if (nav) redirectUrl += nav @@ -222,7 +226,7 @@ module.exports = (query, _warningCb) => { } const { id: p } = parc[parcellationSelected] || {} if (p) redirectUrl += `/p:${encodeId(p)}` - if (r) redirectUrl += r + if (r && parcellationSelected !== 'Cytoarchitectonic Maps - v2.5.1') redirectUrl += r if (nav) redirectUrl += nav if (dsp) redirectUrl += dsp diff --git a/deploy/csp/index.js b/deploy/csp/index.js index d745b8d58f0a70422b9d48cb481cf533f01401eb..4deecdda1795ff07c9e6f6f68fcf812c94540432 100644 --- a/deploy/csp/index.js +++ b/deploy/csp/index.js @@ -39,12 +39,24 @@ const defaultAllowedSites = [ const connectSrc = [ "'self'", + + // needed by ad hoc generation of URL resources "blob:", + + // siibra-api endpoints + 'siibra-api-latest.apps-dev.hbp.eu', + 'siibra-api-rc.apps.hbp.eu', + 'siibra-api-stable.apps.hbp.eu', + + // chunk servers 'neuroglancer.humanbrainproject.org', 'neuroglancer.humanbrainproject.eu', - 'connectivity-query-v1-1-connectivity.apps-dev.hbp.eu', 'object.cscs.ch', - 'hbp-kg-dataset-previewer.apps.hbp.eu/v2/', // required for dataset previews + + // required for dataset previews + 'hbp-kg-dataset-previewer.apps.hbp.eu/v2/', + + // injected by env var ...CSP_CONNECT_SRC ] @@ -125,6 +137,7 @@ module.exports = { } else { console.warn(`CSP Violation: no data received!`) } + res.status(204).end() }) } } diff --git a/deploy/package.json b/deploy/package.json index 8ac08ced5847f7e9915dabd46bb96e642ad3fcd7..95da8df0bd39afe6c3076fe360260e3b4f81995b 100644 --- a/deploy/package.json +++ b/deploy/package.json @@ -5,7 +5,7 @@ "main": "index.js", "scripts": { "start": "node server.js", - "test": "DISABLE_LIMITER=1 node -r dotenv/config ./node_modules/.bin/mocha './**/*.spec.js' --exclude 'node_modules/*' --timeout 60000 --exit", + "test": "node -r dotenv/config ./node_modules/.bin/mocha './**/*.spec.js' --exclude 'node_modules/*' --timeout 60000 --exit", "mocha": "mocha" }, "keywords": [], @@ -17,17 +17,15 @@ "connect-redis": "^5.0.0", "cookie-parser": "^1.4.5", "express": "^4.16.4", - "express-rate-limit": "^5.1.1", "express-session": "^1.15.6", "got": "^10.5.5", - "hbp-seafile": "^0.1.0", + "hbp-seafile": "^0.2.0", "helmet-csp": "^2.8.0", "lru-cache": "^5.1.1", "memorystore": "^1.6.1", "nomiseco": "0.0.2", "openid-client": "^4.4.0", "passport": "^0.4.0", - "rate-limit-redis": "^1.7.0", "redis": "^3.0.2", "request": "^2.88.0", "showdown": "^1.9.1", diff --git a/deploy/preview/index.js b/deploy/preview/index.js deleted file mode 100644 index 426d04d827b0be5c8305c1466c46f13e071aaaf4..0000000000000000000000000000000000000000 --- a/deploy/preview/index.js +++ /dev/null @@ -1,27 +0,0 @@ -const router = require('express').Router() -const request = require('request') -const url = require('url') -const stream = require('stream') -const { getHandleErrorFn } = require('../util/streamHandleError') - -let PROXY_HOSTNAME_WHITELIST - -try{ - PROXY_HOSTNAME_WHITELIST = JSON.parse(process.env.PROXY_HOSTNAME_WHITELIST) -}catch(e){ - PROXY_HOSTNAME_WHITELIST = [] -} - -const whiteList = new Set([ - 'object.cscs.ch', - ...PROXY_HOSTNAME_WHITELIST -]) - -router.get('/file', (req, res) => { - const { fileUrl } = req.query - const f = url.parse(fileUrl) - if(f && f.hostname && whiteList.has(f.hostname)) return request(fileUrl).pipe(res).on('error', getHandleErrorFn(req, res)) - else res.status(400).send() -}) - -module.exports = router \ No newline at end of file diff --git a/deploy/regionalFeatures/index.js b/deploy/regionalFeatures/index.js deleted file mode 100644 index f642c86afa4b8e1695fde72b9bd600f4c5efbfbc..0000000000000000000000000000000000000000 --- a/deploy/regionalFeatures/index.js +++ /dev/null @@ -1,277 +0,0 @@ -const router = require('express').Router() -const request = require('request') - -/** - * TODO migrate to brainscape in the future - */ - -const REGIONAL_FEATURE_ENDPOINT_ARRAY = process.env.REGIONAL_FEATURE_ENDPOINT_ARRAY || [] - -let arrayToFetch = [] -try { - arrayToFetch = JSON.parse(REGIONAL_FEATURE_ENDPOINT_ARRAY) -} catch (e) { - console.warn(`parsing arrayToFetch parse failed`) -} - -const regionIdToDataIdMap = new Map() -const datasetIdToDataMap = new Map() -const datasetIdDetailMap = new Map() - -let additionalDatasets = [] -const returnAdditionalDatasets = async () => additionalDatasets -let isReady = false - -const ITERABLE_KEY_SYMBOL = Symbol('ITERABLE_KEY_SYMBOL') - -/** - * this pattern allows all of the special data to be fetched in parallel - * async await would mean it is fetched one at a time - */ - -const processRegionalFeatureObj = ({ regions, data, ['@id']: datasetId, type, name }) => { - - datasetIdDetailMap.set(datasetId, { - ['@id']: datasetId, - type, - name - }) - for (const { status, ['@id']: regionId, name, files } of regions) { - if (regionIdToDataIdMap.has(regionId)) { - const existingObj = regionIdToDataIdMap.get(regionId) - /** - * existingObj[datasetId] may be undefined - */ - if (!existingObj[datasetId]) { - existingObj[datasetId] = { - type, - } - existingObj[ITERABLE_KEY_SYMBOL] = existingObj[ITERABLE_KEY_SYMBOL].concat(datasetId) - } - existingObj[datasetId][status] = (existingObj[datasetId][status] || []).concat(files) - existingObj[datasetId][ITERABLE_KEY_SYMBOL] = (existingObj[datasetId][ITERABLE_KEY_SYMBOL] || []).concat(status) - } else { - const datasetObj = { - [status]: files, - type, - } - datasetObj[ITERABLE_KEY_SYMBOL] = [status] - const obj = { - name, - '@id': regionId, - [datasetId]: datasetObj - } - obj[ITERABLE_KEY_SYMBOL] = [datasetId] - regionIdToDataIdMap.set(regionId, obj) - } - } - - const dataIdToDataMap = new Map() - datasetIdToDataMap.set(datasetId, dataIdToDataMap) - - for (const { ['@id']: dataId, contact_points: contactPoints, referenceSpaces, ...rest } of data) { - dataIdToDataMap.set(dataId, { - ['@id']: dataId, - contactPoints, - referenceSpaces, - ...rest - }) - } -} - -Promise.all( - arrayToFetch.map(url => - new Promise((rs, rj) => { - request.get(url, (err, _resp, body) => { - if (err) return rj(err) - const parsedObj = JSON.parse(body) - - if (Array.isArray(parsedObj)) { - parsedObj.map(processRegionalFeatureObj) - } else { - processRegionalFeatureObj(parsedObj) - } - - rs() - }) - }) - ) -).then(() => { - const map = new Map() - for (const [regionId, regionObj] of regionIdToDataIdMap.entries()) { - for (const datasetId of regionObj[ITERABLE_KEY_SYMBOL]) { - const newArr = (map.get(datasetId) || []).concat(regionId) - map.set(datasetId, newArr) - } - } - - for (const [ datasetId, arrRegionIds ] of map.entries()) { - additionalDatasets = additionalDatasets.concat({ - fullId: `https://nexus.humanbrainproject.org/v0/data/${datasetId}`, - parcellationRegion: arrRegionIds.map(id => ({ fullId: id })), - species: [], - kgReference: [ - `https://kg.ebrains.eu/search/instances/Dataset/${datasetId}` - ] - }) - } - - isReady = true -}) - -const getFeatureMiddleware = (req, res, next) => { - const { featureFullId } = req.params - const datasetIdToDataMapToUse = res.locals['overwrite_datasetIdToDataMap'] || datasetIdToDataMap - if (!datasetIdToDataMapToUse.has(featureFullId)) { - return res.status(404).send(`Not found. - getFeatureMiddleware -`) - } - res.locals['getFeatureMiddleware_cache_0'] = datasetIdToDataMapToUse.get(featureFullId) - res.locals['getFeatureMiddleware_cache_1'] = datasetIdDetailMap.get(featureFullId) - next() -} - -const sendFeatureResponse = (req, res) => { - if (!res.locals['getFeatureMiddleware_cache_0']) return res.status(500).send(`getFeatureMiddleware_cache_0 not populated`) - const fullIdMap = res.locals['getFeatureMiddleware_cache_0'] - const featureDetail = res.locals['getFeatureMiddleware_cache_1'] || {} - const dataKeys = Array.from(fullIdMap.keys()) - if (dataKeys.length === 0) return res.status(404).end() - return res.status(200).json({ - ...featureDetail, - data: dataKeys.map(dataId => { - return { - ['@id']: dataId, - } - }) - }) -} - -const getFeatureGetDataMiddleware = (req, res, next) => { - const { dataId } = req.params - if (!res.locals['getFeatureMiddleware_cache_0']) return res.status(500).send(`getFeatureMiddleware_cache_0 not populated`) - const map = res.locals['getFeatureMiddleware_cache_0'] - if (!map.has(dataId)) { - return res.status(404).send(`Not found. - getFeatureGetDataMiddleware -`) - } - res.locals['getFeatureGetDataMiddleware_cache_0'] = map.get(dataId) - next() -} - -const sendFeatureDataResponse = (req, res) => { - if (!res.locals['getFeatureGetDataMiddleware_cache_0']) return res.stauts(500).send(`getFeatureGetDataMiddleware_cache_0 not populated`) - const result = res.locals['getFeatureGetDataMiddleware_cache_0'] - res.status(200).json(result) -} - -router.get( - '/byFeature/:featureFullId', - getFeatureMiddleware, - sendFeatureResponse, -) - -router.get( - '/byFeature/:featureFullId/:dataId', - getFeatureMiddleware, - getFeatureGetDataMiddleware, - sendFeatureDataResponse, -) - -const byRegionMiddleware = (req, res, next) => { - - const { regionFullId } = req.params - const { hemisphere, referenceSpaceId } = req.query - - if (!regionIdToDataIdMap.has(regionFullId)) { - return res.status(404).send(`Not found. - byRegionMiddleware -`) - } - - /** - * datasetIdToDataMap: - * datasetId -> dataId -> { ['@id']: string, contactPoints, referenceSpaces } - */ - const overWriteDatasetIdToMap = new Map() - res.locals['byRegionMiddleware_cache_0'] = overWriteDatasetIdToMap - res.locals['overwrite_datasetIdToDataMap'] = overWriteDatasetIdToMap - - /** - * TODO filter by reference spaces - */ - - const regionObj = regionIdToDataIdMap.get(regionFullId) - - for (const datasetId of regionObj[ITERABLE_KEY_SYMBOL]) { - const returnMap = new Map() - overWriteDatasetIdToMap.set(datasetId, returnMap) - for (const hemisphereKey of regionObj[datasetId][ITERABLE_KEY_SYMBOL]) { - - /** - * if hemisphere is defined, then skip if hemisphereKey does not match - */ - - if (!!hemisphere && hemisphereKey !== hemisphere) continue - for (const { ['@id']: dataId } of regionObj[datasetId][hemisphereKey] || []) { - try { - const dataObj = datasetIdToDataMap.get(datasetId).get(dataId) - if ( - !!referenceSpaceId - && !! dataObj['referenceSpaces'] - && dataObj['referenceSpaces'].every(rs => rs['fullId'] !== '*' && rs['fullId'] !== referenceSpaceId) - ) { - continue - } - returnMap.set( - dataId, - dataObj - ) - } catch (e) { - console.warn(`${datasetId} or ${dataId} could not be found in datasetIdToDataMap`) - } - } - } - } - - next() -} - -router.get( - '/byRegion/:regionFullId', - byRegionMiddleware, - async (req, res) => { - if (!res.locals['byRegionMiddleware_cache_0']) return res.status(500).send(`byRegionMiddleware_cache_0 not populated`) - - const returnMap = res.locals['byRegionMiddleware_cache_0'] - return res.status(200).json({ - features: Array.from( - returnMap.keys() - ).filter(id => { - /** - * do not return where there are no datas - */ - return returnMap.get(id).size || 0 > 0 - }).map(id => ({ ['@id']: id })) - }) - } -) - -router.get( - '/byRegion/:regionFullId/:featureFullId', - byRegionMiddleware, - getFeatureMiddleware, - sendFeatureResponse, -) - -router.get( - '/byRegion/:regionFullId/:featureFullId/:dataId', - byRegionMiddleware, - getFeatureMiddleware, - getFeatureGetDataMiddleware, - sendFeatureDataResponse, -) - -const regionalFeatureIsReady = async () => isReady - -module.exports = { - router, - regionalFeatureIsReady, - returnAdditionalDatasets, -} diff --git a/deploy/saneUrl/index.js b/deploy/saneUrl/index.js index b0033f9acefeb56e79bb379cfc0accad09d7b4d6..d6d5caa2ab0ee69c8568a14ee02bd64bf2df7bbd 100644 --- a/deploy/saneUrl/index.js +++ b/deploy/saneUrl/index.js @@ -1,41 +1,12 @@ const express = require('express') const router = express.Router() -const RateLimit = require('express-rate-limit') -const RedisStore = require('rate-limit-redis') const { FallbackStore: Store, NotFoundError } = require('./store') -const lruStore = require('../lruStore') const { Store: DepcStore } = require('./depcObjStore') -const { readUserData, saveUserData } = require('../user/store') const store = new Store() const depStore = new DepcStore() -const { DISABLE_LIMITER, HOSTNAME, HOST_PATHNAME } = process.env - - -let limiter -const getLimiter = async () => { - if (DISABLE_LIMITER) return passthrough - - if (!!limiter) return limiter - - await lruStore._initPr - if (lruStore.redisURL) { - limiter = new RateLimit({ - windowMs: 1e3 * 5, - max: 5, - store: new RedisStore({ redisURL }) - }) - } else { - limiter = new RateLimit({ - windowMs: 1e3 * 5, - max: 5, - }) - } - return limiter -} - -const passthrough = (_, __, next) => next() +const { HOSTNAME, HOST_PATHNAME } = process.env const acceptHtmlProg = /text\/html/i @@ -105,52 +76,7 @@ router.get('/:name', async (req, res) => { }) router.post('/:name', - getLimiter, - async (req, res, next) => { - const { name } = req.params - try { - const exist = await getFile(name) - if (!exist) throw new NotFoundError() - return res.status(409).send(`filename already exists`) - } catch (e) { - if (e instanceof NotFoundError) return next() - else return res.status(500).send(e) - } - }, - express.json(), - async (req, res) => { - const { name } = req.params - const { body, user } = req - - try { - const payload = { - ...body, - userId: user && user.id, - expiry: !user && (Date.now() + 1e3 * 60 * 60 * 72) - } - - await store.set(name, JSON.stringify(payload)) - res.status(200).end() - - try { - if (!user) return - const { savedCustomLinks = [], ...rest } = await readUserData(user) - await saveUserData(user, { - ...rest, - savedCustomLinks: [ - ...savedCustomLinks, - name - ] - }) - } catch (e) { - console.error(`reading/writing user data error ${user && user.id}, ${name}`, e) - } - } catch (e) { - console.error(`saneUrl /POST error`, e) - const { statusCode, statusMessage } = e - res.status(statusCode || 500).send(statusMessage || 'Error encountered.') - } - } + (_req, res) => res.status(410).end() ) router.use((_, res) => { diff --git a/deploy_env.md b/deploy_env.md index 7e4a55a80c10ba180399e31f30a9b71b83bddc5b..ca6a5e72c6aa542a6b88471f88eb3174a3b2e05f 100644 --- a/deploy_env.md +++ b/deploy_env.md @@ -70,7 +70,6 @@ | `REDIS_PORT` | fall back to `REDIS_RATE_LIMITING_DB_EPHEMERAL_PORT_6379_TCP_PORT` | | `REDIS_USERNAME` | | `REDIS_PASSWORD` | -| `DISABLE_LIMITER` | disable rate limiting (maybe required for automated tests) | ##### SaneUrl diff --git a/docs/releases/v2.4.1.md b/docs/releases/v2.4.1.md new file mode 100644 index 0000000000000000000000000000000000000000..4768d30acb00b7b73b5472fbb48c86800739152a --- /dev/null +++ b/docs/releases/v2.4.1.md @@ -0,0 +1,5 @@ +# v2.4.1 + +## Bugfixes + +- fix JulichBrain v2.5.0/v1.18 URL redirection \ No newline at end of file diff --git a/docs/releases/v2.4.2.md b/docs/releases/v2.4.2.md new file mode 100644 index 0000000000000000000000000000000000000000..b2c4a5b8b3fb9b86c60b318067675b3377b00a6e --- /dev/null +++ b/docs/releases/v2.4.2.md @@ -0,0 +1,7 @@ +# v2.4.2 + +## Bugfixes + +- fix template/parcellation selector vertical scroll on small devices +- fix atlas order (human -> rat -> mouse) +- fsaverage use pial as default viewing mode (if possible) diff --git a/docs/releases/v2.4.3.md b/docs/releases/v2.4.3.md new file mode 100644 index 0000000000000000000000000000000000000000..a91b54fd8df91420a49c0a69415f04782b0a7c12 --- /dev/null +++ b/docs/releases/v2.4.3.md @@ -0,0 +1,6 @@ +# v2.4.3 + +## Bugfixes + +- fix some of the big brain maps +- added UI indication for deprecation/unavailability of functionalities diff --git a/docs/releases/v2.4.4.md b/docs/releases/v2.4.4.md new file mode 100644 index 0000000000000000000000000000000000000000..cadb5d9df79de2f9c768008c4a35d1b705a0bb81 --- /dev/null +++ b/docs/releases/v2.4.4.md @@ -0,0 +1,13 @@ +# v2.4.4 + +# Features + +- Allow name and description of annotations to be exported + +## Bugfixes + +- Fix version of connectivity web component + +## Under the hood stuff + +- Respond in csp violation reports diff --git a/docs/releases/v2.4.5.md b/docs/releases/v2.4.5.md new file mode 100644 index 0000000000000000000000000000000000000000..47df26c7f4c10e832a92e0b0e423203bd2aad301 --- /dev/null +++ b/docs/releases/v2.4.5.md @@ -0,0 +1,4 @@ +# v2.4.5 + +## README +- Modify the readme file with integration into the Siibra naming. diff --git a/docs/releases/v2.4.6.md b/docs/releases/v2.4.6.md new file mode 100644 index 0000000000000000000000000000000000000000..7aa58d68364d393a047bbfd87482ef50dfb3f8e5 --- /dev/null +++ b/docs/releases/v2.4.6.md @@ -0,0 +1,5 @@ +# v2.4.6 + +## Bugfixes + +- Fix space press issue on full view diff --git a/docs/releases/v2.4.7.md b/docs/releases/v2.4.7.md new file mode 100644 index 0000000000000000000000000000000000000000..e5fcfaf6f7acabdd64790c13917249bc7be18669 --- /dev/null +++ b/docs/releases/v2.4.7.md @@ -0,0 +1,11 @@ +# v2.4.7 + +## Bugfixes + +- Fix unnamed point landmark naming +- Fix multiregion legacy links (#1022) + +## Under the hood stuff + +- Remove unused imports in the backend +- Use `stable` branch of siibra-api by default (supercedes #997) diff --git a/docs/releases/v2.4.8.md b/docs/releases/v2.4.8.md new file mode 100644 index 0000000000000000000000000000000000000000..8ffbb5c55b071438e1fffe6b284b291f7ee77e0b --- /dev/null +++ b/docs/releases/v2.4.8.md @@ -0,0 +1,13 @@ +# v2.4.8 + +## Enhancements + +- Added badges to annotation tab button to show number of annotations added (#1007) +- Added some feedbacks when annotations are being loaded +- Added delete all annotation button +- Added feedbacks when annotation from different reference spaces are added + +## Under the hood stuff + +- Tweaked space bar capture +- Allow branches with `hotfix` in its name to target production siibra-api endpoints diff --git a/mkdocs.yml b/mkdocs.yml index f504c13ca845a5dd5b2bcb51a01caeadb3f47615..005eb93634ee3d01cea5258c1a22c0673c8c66cc 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -41,6 +41,14 @@ pages: - Display non-atlas volumes: 'advanced/otherVolumes.md' - Release notes: - v2.5.0: 'releases/v2.5.0.md' + - v2.4.8: 'releases/v2.4.8.md' + - v2.4.7: 'releases/v2.4.7.md' + - v2.4.6: 'releases/v2.4.6.md' + - v2.4.5: 'releases/v2.4.5.md' + - v2.4.4: 'releases/v2.4.4.md' + - v2.4.3: 'releases/v2.4.3.md' + - v2.4.2: 'releases/v2.4.2.md' + - v2.4.1: 'releases/v2.4.1.md' - v2.4.0: 'releases/v2.4.0.md' - v2.3.11: 'releases/v2.3.11.md' - v2.3.10: 'releases/v2.3.10.md' diff --git a/src/atlasComponents/connectivity/hasConnectivity.directive.ts b/src/atlasComponents/connectivity/hasConnectivity.directive.ts index b3cf77233d0a1a991722b42876cc38c60c7834ca..8879de8c093488b90acd5fb81d57820f341da36e 100644 --- a/src/atlasComponents/connectivity/hasConnectivity.directive.ts +++ b/src/atlasComponents/connectivity/hasConnectivity.directive.ts @@ -26,6 +26,10 @@ export class HasConnectivity implements OnInit, OnDestroy { } checkConnectivity(region) { + if (!region.context) { + this.hasConnectivity = false + return + } const {atlas, parcellation, template} = region.context if (region.name) { const connectivityUrl = `${this.siibraApiUrl}/atlases/${encodeURIComponent(atlas['@id'])}/parcellations/${encodeURIComponent(parcellation['@id'])}/regions/${encodeURIComponent(region.name)}/features/ConnectivityProfile` diff --git a/src/atlasComponents/parcellationRegion/region.base.ts b/src/atlasComponents/parcellationRegion/region.base.ts index a16995b9af3a98b69c44f0577e2f6e1226b087df..b6e234f7bd763d31d851663f5fc147a68a0be883 100644 --- a/src/atlasComponents/parcellationRegion/region.base.ts +++ b/src/atlasComponents/parcellationRegion/region.base.ts @@ -34,6 +34,7 @@ export class RegionBase { set region(val) { this._region = val this.region$.next(this._region) + this.hasContext$.next(!!this._region.context) this.position = null // bug the centroid returned is currently nonsense @@ -62,6 +63,7 @@ export class RegionBase { return this._region } + public hasContext$: BehaviorSubject<boolean> = new BehaviorSubject(false) public region$: BehaviorSubject<any> = new BehaviorSubject(null) @Input() @@ -90,7 +92,7 @@ export class RegionBase { this.regionInOtherTemplates$ = this.region$.pipe( distinctUntilChanged(), - filter(v => !!v), + filter(v => !!v && !!v.context), switchMap(region => this.store$.pipe( select( regionInOtherTemplateSelector, @@ -285,7 +287,7 @@ export const regionInOtherTemplateSelector = createSelector( const otherTemplates = fetchedTemplates .filter(({ ['@id']: id }) => id !== regionOfInterest.context.template['@id'] && atlasTemplateSpacesIds.includes(id) - && regionOfInterest.availableIn.map(ai => ai.id).includes(id)) + && (regionOfInterest.availableIn || []).map(ai => ai.id).includes(id)) for (const template of otherTemplates) { const parcellation = template.parcellations.find(p => p['@id'] === regionOfInterest.context.parcellation['@id']) diff --git a/src/atlasComponents/parcellationRegion/regionMenu/regionMenu.template.html b/src/atlasComponents/parcellationRegion/regionMenu/regionMenu.template.html index 7f7a0cc5c1aae01b0747185c83b1c4de4c788a09..9717bea444b93fd92fa83b3ed8d1f381c9da79ff 100644 --- a/src/atlasComponents/parcellationRegion/regionMenu/regionMenu.template.html +++ b/src/atlasComponents/parcellationRegion/regionMenu/regionMenu.template.html @@ -127,7 +127,7 @@ content: kgRegionalFeatureList, desc: '', iconTooltip: 'Regional Features', - iavNgIf: true + iavNgIf: hasContext$ | async }"> </ng-container> diff --git a/src/atlasComponents/regionalFeatures/bsFeatures/regionalFeatureWrapper/regionalFeatureWrapper.template.html b/src/atlasComponents/regionalFeatures/bsFeatures/regionalFeatureWrapper/regionalFeatureWrapper.template.html index 6a62ae958982689268974f1fbfab9047dce5188e..adecf93c84ab9c417168b4588e1aef63e4bb55e0 100644 --- a/src/atlasComponents/regionalFeatures/bsFeatures/regionalFeatureWrapper/regionalFeatureWrapper.template.html +++ b/src/atlasComponents/regionalFeatures/bsFeatures/regionalFeatureWrapper/regionalFeatureWrapper.template.html @@ -52,7 +52,7 @@ <!-- feature template --> <ng-template #itemContainer let-feature> - <div class="d-inline-block pt-4 cursor-default" + <div class="d-block pt-4 cursor-default" (click)="handleFeatureClick(feature)" mat-ripple> diff --git a/src/atlasComponents/uiSelectors/atlasLayerSelector/atlasLayerSelector.style.css b/src/atlasComponents/uiSelectors/atlasLayerSelector/atlasLayerSelector.style.css index 9d43dc845dcf18755d5326699abaf0ce4139011b..56cfffcf79c4db7ee31690dbe90d44a4016ad97f 100644 --- a/src/atlasComponents/uiSelectors/atlasLayerSelector/atlasLayerSelector.style.css +++ b/src/atlasComponents/uiSelectors/atlasLayerSelector/atlasLayerSelector.style.css @@ -38,6 +38,8 @@ .selector-container { + overflow-y:scroll; + max-height: 80vh; width: 21rem; bottom: 4rem; z-index: 5; diff --git a/src/atlasComponents/userAnnotations/annotationList/annotationList.component.ts b/src/atlasComponents/userAnnotations/annotationList/annotationList.component.ts index 525f5eaa5dad6a2ca706664963bda383d8bef9ac..07eae088ce2154228ecc4df9f2f3779ed3390d5a 100644 --- a/src/atlasComponents/userAnnotations/annotationList/annotationList.component.ts +++ b/src/atlasComponents/userAnnotations/annotationList/annotationList.component.ts @@ -1,17 +1,18 @@ -import {Component, ViewChild} from "@angular/core"; -import {ARIA_LABELS} from "common/constants"; +import { Component, Optional, ViewChild } from "@angular/core"; +import { ARIA_LABELS, CONST } from "common/constants"; import { ModularUserAnnotationToolService } from "../tools/service"; import { IAnnotationGeometry, TExportFormats } from "../tools/type"; import { ComponentStore } from "src/viewerModule/componentStore"; -import { map, startWith, tap } from "rxjs/operators"; -import { Observable } from "rxjs"; +import { map, shareReplay, startWith } from "rxjs/operators"; +import { Observable, Subscription } from "rxjs"; import { TZipFileConfig } from "src/zipFilesOutput/type"; import { TFileInputEvent } from "src/getFileInput/type"; import { FileInputDirective } from "src/getFileInput/getFileInput.directive"; import { MatSnackBar } from "@angular/material/snack-bar"; import { unzip } from "src/zipFilesOutput/zipFilesOutput.directive"; +import { DialogService } from "src/services/dialogService.service"; -const README = 'EXAMPLE OF READ ME TEXT' +const README = `{id}.sands.json file contains the data of annotations. {id}.desc.json contains the metadata of annotations.` @Component({ selector: 'annotation-list', @@ -26,11 +27,13 @@ export class AnnotationList { public ARIA_LABELS = ARIA_LABELS - @ViewChild(FileInputDirective) fileInput: FileInputDirective + private subs: Subscription[] = [] + private managedAnnotations: IAnnotationGeometry[] = [] public managedAnnotations$ = this.annotSvc.spaceFilteredManagedAnnotations$ + public annotationInOtherSpaces$ = this.annotSvc.otherSpaceManagedAnnotations$ public manAnnExists$ = this.managedAnnotations$.pipe( map(arr => !!arr && arr.length > 0), @@ -39,6 +42,7 @@ export class AnnotationList { public filesExport$: Observable<TZipFileConfig[]> = this.managedAnnotations$.pipe( startWith([] as IAnnotationGeometry[]), + shareReplay(1), map(manAnns => { const readme = { filename: 'README.md', @@ -50,17 +54,28 @@ export class AnnotationList { filecontent: JSON.stringify(ann.toSands(), null, 2), } }) - return [ readme, ...annotationSands ] + const annotationDesc = manAnns.map(ann => { + return { + filename: `${ann.id}.desc.json`, + filecontent: JSON.stringify(this.annotSvc.exportAnnotationMetadata(ann), null, 2) + } + }) + return [ readme, ...annotationSands, ...annotationDesc ] }) ) constructor( private annotSvc: ModularUserAnnotationToolService, private snackbar: MatSnackBar, cStore: ComponentStore<{ useFormat: TExportFormats }>, + @Optional() private dialogSvc: DialogService, ) { cStore.setState({ useFormat: 'sands' }) + + this.subs.push( + this.managedAnnotations$.subscribe(anns => this.managedAnnotations = anns) + ) } public hiddenAnnotations$ = this.annotSvc.hiddenAnnotations$ @@ -71,10 +86,15 @@ export class AnnotationList { private parseAndAddAnnotation(input: string) { const json = JSON.parse(input) const annotation = this.annotSvc.parseAnnotationObject(json) - this.annotSvc.importAnnotation(annotation) + if (annotation) this.annotSvc.importAnnotation(annotation) } async handleImportEvent(ev: TFileInputEvent<'text' | 'file'>){ + + const { abort } = this.dialogSvc.blockUserInteraction({ + title: CONST.LOADING_TXT, + markdown: CONST.LOADING_ANNOTATION_MSG, + }) try { const clearFileInputAndInform = () => { if (this.fileInput) { @@ -128,6 +148,31 @@ export class AnnotationList { this.snackbar.open(`Error importing: ${e.toString()}`, 'Dismiss', { duration: 3000 }) + } finally { + abort() + } + } + + async deleteAllAnnotation(){ + if (this.dialogSvc) { + try { + await this.dialogSvc.getUserConfirm({ + markdown: CONST.DELETE_ALL_ANNOTATION_CONFIRMATION_MSG + }) + + for (const ann of this.managedAnnotations) { + ann.remove() + } + } catch (e) { + // aborted + } + } else { + if (window.confirm(CONST.DELETE_ALL_ANNOTATION_CONFIRMATION_MSG)) { + + for (const ann of this.managedAnnotations) { + ann.remove() + } + } } } } diff --git a/src/atlasComponents/userAnnotations/annotationList/annotationList.style.css b/src/atlasComponents/userAnnotations/annotationList/annotationList.style.css index 1f57a5f2d7a2155d5fb5c398d7fb079ee9337258..8e3b51c8c14b8e513fa4fb4dd7983b943cc2dc5d 100644 --- a/src/atlasComponents/userAnnotations/annotationList/annotationList.style.css +++ b/src/atlasComponents/userAnnotations/annotationList/annotationList.style.css @@ -8,3 +8,11 @@ :host-context([darktheme="false"]) .hovering-header { background-color: rgb(245, 245, 245); } + +.single-annotation-grid-container +{ + display: grid; + grid-template-columns: 3em auto; + gap: 0px 0px; + grid-template-areas: ". ."; +} diff --git a/src/atlasComponents/userAnnotations/annotationList/annotationList.template.html b/src/atlasComponents/userAnnotations/annotationList/annotationList.template.html index 17ac890eafc6f4049b039ad7c36147a2d481642d..7543d5d598ce9d77956aa6036636a1ba7b8408c5 100644 --- a/src/atlasComponents/userAnnotations/annotationList/annotationList.template.html +++ b/src/atlasComponents/userAnnotations/annotationList/annotationList.template.html @@ -26,13 +26,22 @@ <!-- export --> <button mat-icon-button - [zip-files-output]="filesExport$ | async" + [zip-files-output]="filesExport$" zip-files-output-zip-filename="exported_annotations.zip" [attr.aria-label]="ARIA_LABELS.USER_ANNOTATION_EXPORT" [matTooltip]="ARIA_LABELS.USER_ANNOTATION_EXPORT" [disabled]="!(manAnnExists$ | async)"> <i class="fas fa-download"></i> </button> + + <!-- delete all annotations --> + <button mat-icon-button + color="warn" + (click)="deleteAllAnnotation()" + [matTooltip]="ARIA_LABELS.BULK_DELETE_ANNOTATIONS" + [disabled]="!(manAnnExists$ | async)"> + <i class="fas fa-trash"></i> + </button> </mat-card-subtitle> </div> @@ -68,13 +77,10 @@ <!-- single annotation edit body --> <ng-template matExpansionPanelContent> - <div class="d-flex"> - + <div class="single-annotation-grid-container"> <!-- spacer for inset single-annotation-unit --> - <div class="w-3em flex-grow-0 flex-shrink-0"></div> - - <single-annotation-unit [single-annotation-unit-annotation]="managedAnnotation" - class="flex-grow-1 flex-shrink-1"> + <div></div> + <single-annotation-unit [single-annotation-unit-annotation]="managedAnnotation"> </single-annotation-unit> </div> </ng-template> @@ -82,6 +88,12 @@ </mat-accordion> </ng-template> + <ng-template [ngIf]="annotationInOtherSpaces$ | async" let-annsInOtherSpace> + <div *ngIf="annsInOtherSpace.length > 0" class="p-4 text-muted"> + {{ annsInOtherSpace.length }} annotations found in other reference spaces, and not shown here. + </div> + </ng-template> + <!-- place holder when no annotations exist --> <ng-template #placeholderTmpl> <div class="p-4 text-muted"> diff --git a/src/atlasComponents/userAnnotations/annotationMode/annotationMode.component.ts b/src/atlasComponents/userAnnotations/annotationMode/annotationMode.component.ts index 38eae7e8134c3e7c87fa16722f1a2e248a929718..b316ac2632dd0c6b31cf43bb30345434d7f37941 100644 --- a/src/atlasComponents/userAnnotations/annotationMode/annotationMode.component.ts +++ b/src/atlasComponents/userAnnotations/annotationMode/annotationMode.component.ts @@ -11,10 +11,13 @@ import { MatSnackBar } from "@angular/material/snack-bar"; @Component({ selector: 'annotating-tools-panel', templateUrl: './annotationMode.template.html', - styleUrls: ['./annotationMode.style.css'] + styleUrls: ['./annotationMode.style.css'], + exportAs: 'annoToolsPanel' }) export class AnnotationMode implements OnDestroy{ + public annBadges$ = this.modularToolSvc.badges$ + public ARIA_LABELS = ARIA_LABELS public moduleAnnotationTypes: { diff --git a/src/atlasComponents/userAnnotations/filterAnnotationBySpace.pipe.ts b/src/atlasComponents/userAnnotations/filterAnnotationBySpace.pipe.ts index 69506ed32fd960dbe4da329884fa164d4727469c..fb6500281577a0791f6c82b7d53ce6b7aef636a6 100644 --- a/src/atlasComponents/userAnnotations/filterAnnotationBySpace.pipe.ts +++ b/src/atlasComponents/userAnnotations/filterAnnotationBySpace.pipe.ts @@ -1,13 +1,20 @@ import { Pipe, PipeTransform } from "@angular/core"; import { IAnnotationGeometry } from "./tools/type"; +type TOpts = { + reverse?: boolean +} + @Pipe({ name: 'filterAnnotationsBySpace', pure: true }) export class FilterAnnotationsBySpace implements PipeTransform{ - public transform(annotations: IAnnotationGeometry[], space: { '@id': string }): IAnnotationGeometry[]{ - return annotations.filter(ann => ann.space["@id"] === space["@id"]) + public transform(annotations: IAnnotationGeometry[], space: { '@id': string }, opts?: TOpts): IAnnotationGeometry[]{ + const { reverse = false } = opts || {} + return reverse + ? annotations.filter(ann => ann.space["@id"] !== space["@id"]) + : annotations.filter(ann => ann.space["@id"] === space["@id"]) } } \ No newline at end of file diff --git a/src/atlasComponents/userAnnotations/singleAnnotationUnit/singleAnnotationUnit.component.ts b/src/atlasComponents/userAnnotations/singleAnnotationUnit/singleAnnotationUnit.component.ts index 299e8ccf558d42e793637ff2a3d95ff3f1e12b77..956f23c080c1929b07fc62159a42024c68334490 100644 --- a/src/atlasComponents/userAnnotations/singleAnnotationUnit/singleAnnotationUnit.component.ts +++ b/src/atlasComponents/userAnnotations/singleAnnotationUnit/singleAnnotationUnit.component.ts @@ -115,7 +115,7 @@ export class SingleAnnotationNamePipe implements PipeTransform{ public transform(ann: IAnnotationGeometry, name?: string): string{ if (name) return name if (ann instanceof Polygon) return `Unnamed Polygon` - if (ann instanceof Point) return `Unname Point` + if (ann instanceof Point) return `Unnamed Point` if (ann instanceof Line) return `Unnamed Line` return `Unnamed geometry` } @@ -133,4 +133,4 @@ export class SingleAnnotationClsIconPipe implements PipeTransform{ if (ann instanceof Line) return `fas fa-slash` return `fas fa-mouse-pointer` } -} \ No newline at end of file +} diff --git a/src/atlasComponents/userAnnotations/tools/delete.ts b/src/atlasComponents/userAnnotations/tools/delete.ts index b1658e9cc44f91dc68ed41a5bd1bfe7bd49d0f5c..2e118bcd3dddb46ab2aa8d6ce702b8b14ccdd054 100644 --- a/src/atlasComponents/userAnnotations/tools/delete.ts +++ b/src/atlasComponents/userAnnotations/tools/delete.ts @@ -8,6 +8,7 @@ import { AbsToolClass, IAnnotationEvents, IAnnotationGeometry, IAnnotationTools, export class ToolDelete extends AbsToolClass<Point> implements IAnnotationTools, OnDestroy { public subs: Subscription[] = [] + protected managedAnnotations = [] toolType: TToolType = 'deletion' iconClass = 'fas fa-trash' name = 'Delete' diff --git a/src/atlasComponents/userAnnotations/tools/line.ts b/src/atlasComponents/userAnnotations/tools/line.ts index 644258af23408c6edb3b0c03054ad92c400333d4..606c155f3e995bbf3bba85d6b82cb4040ae7a76e 100644 --- a/src/atlasComponents/userAnnotations/tools/line.ts +++ b/src/atlasComponents/userAnnotations/tools/line.ts @@ -193,7 +193,7 @@ export class ToolLine extends AbsToolClass<Line> implements IAnnotationTools, On subs: Subscription[] = [] - private managedAnnotations: Line[] = [] + protected managedAnnotations: Line[] = [] public managedAnnotations$ = new Subject<Line[]>() onMouseMoveRenderPreview(pos: [number, number, number]) { @@ -312,14 +312,6 @@ export class ToolLine extends AbsToolClass<Line> implements IAnnotationTools, On this.subs.forEach(s => s.unsubscribe()) } - addAnnotation(line: Line) { - const idx = this.managedAnnotations.findIndex(ann => ann.id === line.id) - if (idx >= 0) throw new Error(`Line annotation has already been added`) - line.remove = () => this.removeAnnotation(line.id) - this.managedAnnotations.push(line) - this.managedAnnotations$.next(this.managedAnnotations) - } - removeAnnotation(id: string){ const idx = this.managedAnnotations.findIndex(ann => ann.id === id) if (idx < 0) { diff --git a/src/atlasComponents/userAnnotations/tools/point.ts b/src/atlasComponents/userAnnotations/tools/point.ts index e6fd9fee4f42d8535bd8998b0ffb471e211e0eec..af1543fed9b100ecabda6373bf4055b685c580f7 100644 --- a/src/atlasComponents/userAnnotations/tools/point.ts +++ b/src/atlasComponents/userAnnotations/tools/point.ts @@ -109,7 +109,7 @@ export class ToolPoint extends AbsToolClass<Point> implements IAnnotationTools, public iconClass = POINT_ICON_CLASS public subs: Subscription[] = [] - private managedAnnotations: Point[] = [] + protected managedAnnotations: Point[] = [] public managedAnnotations$ = new Subject<Point[]>() constructor( @@ -177,14 +177,6 @@ export class ToolPoint extends AbsToolClass<Point> implements IAnnotationTools, ) } - addAnnotation(point: Point){ - const found = this.managedAnnotations.find(p => p.id === point.id) - if (found) throw new Error(`Point annotation already added`) - point.remove = () => this.removeAnnotation(point.id) - this.managedAnnotations.push(point) - this.managedAnnotations$.next(this.managedAnnotations) - } - /** * @description remove managed annotation via id * @param id id of annotation diff --git a/src/atlasComponents/userAnnotations/tools/poly.ts b/src/atlasComponents/userAnnotations/tools/poly.ts index d226021de999af7b27ae98815f7fb34176f97ff5..2cd9285b256119f8ec112825b96fc615a6eea407 100644 --- a/src/atlasComponents/userAnnotations/tools/poly.ts +++ b/src/atlasComponents/userAnnotations/tools/poly.ts @@ -89,7 +89,7 @@ export class Polygon extends IAnnotationGeometry{ } toString() { - return `Points: ${JSON.stringify(this.points.map(p => p.toString()))}, edges: ${JSON.stringify(this.edges)}.` + return `Name: ${this.name}, Desc: ${this.desc}, Points: ${JSON.stringify(this.points.map(p => p.toString()))}, edges: ${JSON.stringify(this.edges)}.` } toSands(): TSandsPolyLine{ @@ -234,7 +234,7 @@ export class ToolPolygon extends AbsToolClass<Polygon> implements IAnnotationToo private selectedPoly: Polygon private lastAddedPoint: Point - private managedAnnotations: Polygon[] = [] + protected managedAnnotations: Polygon[] = [] public managedAnnotations$ = new Subject<Polygon[]>() public subs: Subscription[] = [] @@ -404,14 +404,6 @@ export class ToolPolygon extends AbsToolClass<Polygon> implements IAnnotationToo ) } - addAnnotation(poly: Polygon){ - const idx = this.managedAnnotations.findIndex(ann => ann.id === poly.id) - if (idx >= 0) throw new Error(`Polygon already added.`) - poly.remove = () => this.removeAnnotation(poly.id) - this.managedAnnotations.push(poly) - this.managedAnnotations$.next(this.managedAnnotations) - } - removeAnnotation(id: string) { const idx = this.managedAnnotations.findIndex(ann => ann.id === id) if (idx < 0) { diff --git a/src/atlasComponents/userAnnotations/tools/select.ts b/src/atlasComponents/userAnnotations/tools/select.ts index fd947eb34e813a956f21349dfd319898ba469fd0..10befd2dfe4b65ade055f2fcc81d57af5e2d4d3b 100644 --- a/src/atlasComponents/userAnnotations/tools/select.ts +++ b/src/atlasComponents/userAnnotations/tools/select.ts @@ -11,6 +11,7 @@ export class ToolSelect extends AbsToolClass<Point> implements IAnnotationTools, toolType: TToolType = 'selecting' iconClass = 'fas fa-mouse-pointer' name = 'Select' + protected managedAnnotations = [] onMouseMoveRenderPreview(){ return [] diff --git a/src/atlasComponents/userAnnotations/tools/service.ts b/src/atlasComponents/userAnnotations/tools/service.ts index c5489b5cd4e5edce83a612e209f8a86c1ffa6e6a..f77ec97b490bcb52fb27784b28f1202144a2f96e 100644 --- a/src/atlasComponents/userAnnotations/tools/service.ts +++ b/src/atlasComponents/userAnnotations/tools/service.ts @@ -27,6 +27,17 @@ const IAV_VOXEL_SIZES_NM = { 'minds/core/referencespace/v1.0.0/MEBRAINS_T1.masked': [1000000, 1000000, 1000000] } +type TAnnotationMetadata = { + id: string + name: string + desc: string +} + +const descType: 'siibra-ex/meta/desc' = 'siibra-ex/meta/desc' +type TTypedAnnMetadata = { + '@type': 'siibra-ex/meta/desc' +} & TAnnotationMetadata + function scanCollapse<T>(){ return (src: Observable<{ tool: string @@ -93,6 +104,20 @@ export class ModularUserAnnotationToolService implements OnDestroy{ scanCollapse(), shareReplay(1), ) + + public otherSpaceManagedAnnotations$ = combineLatest([ + this.selectedTmpl$, + this.managedAnnotations$ + ]).pipe( + map(([tmpl, annts]) => { + return this.filterAnnotationBySpacePipe.transform( + annts, + tmpl, + { reverse: true } + ) + }) + ) + public spaceFilteredManagedAnnotations$ = combineLatest([ this.selectedTmpl$, this.managedAnnotations$ @@ -105,6 +130,10 @@ export class ModularUserAnnotationToolService implements OnDestroy{ }) ) + public badges$ = this.spaceFilteredManagedAnnotations$.pipe( + map(mann => mann.length > 0 ? mann.length : null) + ) + private registeredTools: { name: string iconClass: string @@ -552,7 +581,7 @@ export class ModularUserAnnotationToolService implements OnDestroy{ const anns: IAnnotationGeometry[] = [] for (const obj of arr) { const geometry = this.parseAnnotationObject(obj) - anns.push(geometry) + if (geometry) anns.push(geometry) } for (const ann of anns) { @@ -560,6 +589,21 @@ export class ModularUserAnnotationToolService implements OnDestroy{ } } + public exportAnnotationMetadata(ann: IAnnotationGeometry): TAnnotationMetadata & { '@type': 'siibra-ex/meta/desc' } { + return { + '@type': descType, + id: ann.id, + name: ann.name, + desc: ann.desc, + } + } + + /** + * stop gap measure when exporting/import annotations in sands format + * metadata (name/desc) will be saved in a separate metadata file + */ + private metadataMap = new Map<string, TAnnotationMetadata>() + private storeAnnotation(anns: IAnnotationGeometry[]){ const arr = [] for (const ann of anns) { @@ -627,25 +671,49 @@ export class ModularUserAnnotationToolService implements OnDestroy{ }) } - parseAnnotationObject(json: TSands | TGeometryJson): IAnnotationGeometry{ + parseAnnotationObject(json: TSands | TGeometryJson | TTypedAnnMetadata): IAnnotationGeometry | null{ + let returnObj: IAnnotationGeometry if (json['@type'] === 'tmp/poly') { - return Polygon.fromSANDS(json) + returnObj = Polygon.fromSANDS(json) } if (json['@type'] === 'tmp/line') { - return Line.fromSANDS(json) + returnObj = Line.fromSANDS(json) } if (json['@type'] === 'https://openminds.ebrains.eu/sands/CoordinatePoint') { - return Point.fromSANDS(json) + returnObj = Point.fromSANDS(json) } if (json['@type'] === 'siibra-ex/annotation/point') { - return Point.fromJSON(json) + returnObj = Point.fromJSON(json) } if (json['@type'] === 'siibra-ex/annotation/line') { - return Line.fromJSON(json) + returnObj = Line.fromJSON(json) } if (json['@type'] === 'siibra-ex/annotation/polyline') { - return Polygon.fromJSON(json) + returnObj = Polygon.fromJSON(json) + } + if (json['@type'] === descType) { + const existingAnn = this.managedAnnotations.find(ann => json.id === ann.id) + if (existingAnn) { + + // potentially overwriting existing name and desc... + // maybe should show warning? + existingAnn.setName(json.name) + existingAnn.setDesc(json.desc) + return existingAnn + } else { + const { id, name, desc } = json + this.metadataMap.set(id, { id, name, desc }) + return + } + } else { + const metadata = this.metadataMap.get(returnObj.id) + if (returnObj && metadata) { + returnObj.setName(metadata?.name || null) + returnObj.setDesc(metadata?.desc || null) + this.metadataMap.delete(returnObj.id) + } } + if (returnObj) return returnObj throw new Error(`cannot parse annotation object`) } diff --git a/src/atlasComponents/userAnnotations/tools/type.ts b/src/atlasComponents/userAnnotations/tools/type.ts index efb1d9136c74e1f55497efc62b5fda6798b9afbb..e74f0ca8ee389775b93f49836a4f504ab88ccfec 100644 --- a/src/atlasComponents/userAnnotations/tools/type.ts +++ b/src/atlasComponents/userAnnotations/tools/type.ts @@ -18,9 +18,9 @@ export abstract class AbsToolClass<T extends IAnnotationGeometry> { public abstract name: string public abstract iconClass: string - public abstract addAnnotation(annotation: T): void public abstract removeAnnotation(id: string): void - public abstract managedAnnotations$: Observable<T[]> + public abstract managedAnnotations$: Subject<T[]> + protected abstract managedAnnotations: T[] = [] abstract subs: Subscription[] protected space: TBaseAnnotationGeomtrySpec['space'] @@ -154,6 +154,14 @@ export abstract class AbsToolClass<T extends IAnnotationGeometry> { }), )) ) + + public addAnnotation(geom: T) { + const found = this.managedAnnotations.find(ann => ann.id === geom.id) + if (found) found.remove() + geom.remove = () => this.removeAnnotation(geom.id) + this.managedAnnotations.push(geom) + this.managedAnnotations$.next(this.managedAnnotations) + } } export type TToolType = 'selecting' | 'drawing' | 'deletion' diff --git a/src/auth/auth.service.ts b/src/auth/auth.service.ts index ae56dcc710806d93c05b0803cc42f0ac656accb1..bb610daceac50e7a2ae11ef8fe0919f025354fa3 100644 --- a/src/auth/auth.service.ts +++ b/src/auth/auth.service.ts @@ -32,10 +32,7 @@ export class AuthService implements OnDestroy { * TODO build it dynamically, or at least possible to configure via env var */ public loginMethods: IAuthMethod[] = [{ - name: 'HBP OIDC', - href: 'hbp-oidc/auth' - }, { - name: 'HBP OIDC v2 (beta)', + name: 'HBP OIDC v2', href: 'hbp-oidc-v2/auth' }] diff --git a/src/components/confirmDialog/confirmDialog.component.ts b/src/components/confirmDialog/confirmDialog.component.ts index 5c3d9eb02a50f5847b8ca78a5fabedfd39192f48..ec2e20c9fb6469c9ec26f88c77b73f50a77c57d3 100644 --- a/src/components/confirmDialog/confirmDialog.component.ts +++ b/src/components/confirmDialog/confirmDialog.component.ts @@ -25,12 +25,15 @@ export class ConfirmDialogComponent { @Input() public markdown: string + public hideActionBar = false + constructor(@Inject(MAT_DIALOG_DATA) data: any) { - const { title = null, message = null, markdown, okBtnText, cancelBtnText} = data || {} + const { title = null, message = null, markdown, okBtnText, cancelBtnText, hideActionBar} = data || {} if (title) this.title = title if (message) this.message = message if (markdown) this.markdown = markdown if (okBtnText) this.okBtnText = okBtnText if (cancelBtnText) this.cancelBtnText = cancelBtnText + if (hideActionBar) this.hideActionBar = hideActionBar } } diff --git a/src/components/confirmDialog/confirmDialog.template.html b/src/components/confirmDialog/confirmDialog.template.html index 401261f1ab1aa9c2324cd8510976e971014db1ea..f5f0549460e034215176f4393e3ea2507c83533d 100644 --- a/src/components/confirmDialog/confirmDialog.template.html +++ b/src/components/confirmDialog/confirmDialog.template.html @@ -17,7 +17,7 @@ <mat-divider></mat-divider> -<mat-dialog-actions class="justify-content-start flex-row-reverse"> +<mat-dialog-actions *ngIf="!hideActionBar" class="justify-content-start flex-row-reverse"> <button [mat-dialog-close]="true" mat-raised-button color="primary">{{ okBtnText }}</button> <button [mat-dialog-close]="false" mat-button>{{ cancelBtnText }}</button> </mat-dialog-actions> diff --git a/src/glue.ts b/src/glue.ts index 4437da6f1b2f9d35d9423888cb7636b2c0d4976c..027858745e6ac5223431e586aa6bd016e1ff3611 100644 --- a/src/glue.ts +++ b/src/glue.ts @@ -238,6 +238,13 @@ export class DatasetPreviewGlue implements IDatasetPreviewGlue, OnDestroy{ shareReplay(1), ) + public _volumePreview$ = this.previewingDatasetFiles$.pipe( + switchMap(arr => arr.length > 0 + ? forkJoin(arr.map(v => this.getDatasetPreviewFromId(v))) + : of([])), + map(arr => arr.filter(v => determinePreviewFileType(v) === EnumPreviewFileTypes.VOLUMES)) + ) + private diffPreviewingDatasetFiles$= this.previewingDatasetFiles$.pipe( debounceTime(100), startWith([] as IDatasetPreviewData[]), @@ -326,7 +333,6 @@ export class DatasetPreviewGlue implements IDatasetPreviewGlue, OnDestroy{ distinctUntilChanged(), )) ).subscribe(([ { prvToShow, prvToDismiss }, templateSelected ]) => { - const filterdPrvs = prvToShow.filter(prv => DatasetPreviewGlue.PreviewFileIsInCorrectSpace(prv, templateSelected)) for (const prv of filterdPrvs) { const { volumes } = prv['data']['iav-registered-volumes'] @@ -461,3 +467,28 @@ export class ClickInterceptorService extends RegDeregController<any, boolean>{ // called when the call has not been intercepted } } + +export type _TPLIVal = { + name: string + filename: string + datasetSchema: string + datasetId: string + data: { + 'iav-registered-volumes': { + volumes: { + name: string + source: string + shader: string + transform: any + opacity: string + }[] + } + } + referenceSpaces: { + name: string + fullId: string + }[] + mimetype: 'application/json' +} + +export const _PLI_VOLUME_INJ_TOKEN = new InjectionToken<Observable<_TPLIVal[]>>('_PLI_VOLUME_INJ_TOKEN') diff --git a/src/main.module.ts b/src/main.module.ts index bfe4020d475c53317d98064505ca26cd3d79ef57..14806e6db813a562e356e7c6e94d47db01d499b5 100644 --- a/src/main.module.ts +++ b/src/main.module.ts @@ -39,7 +39,7 @@ import { LoggingModule } from './logging/logging.module'; import { AuthService } from './auth' import 'src/theme.scss' -import { DatasetPreviewGlue, datasetPreviewMetaReducer, IDatasetPreviewGlue, GlueEffects, ClickInterceptorService } from './glue'; +import { DatasetPreviewGlue, datasetPreviewMetaReducer, IDatasetPreviewGlue, GlueEffects, ClickInterceptorService, _PLI_VOLUME_INJ_TOKEN } from './glue'; import { viewerStateHelperReducer, viewerStateMetaReducers, ViewerStateHelperEffect } from './services/state/viewerState.store.helper'; import { TOS_OBS_INJECTION_TOKEN } from './ui/kgtos'; import { UiEffects } from './services/state/uiState/ui.effects'; @@ -185,7 +185,11 @@ export function debug(reducer: ActionReducer<any>): ActionReducer<any> { }, deps: [ UIService ] }, - + { + provide: _PLI_VOLUME_INJ_TOKEN, + useFactory: (glue: DatasetPreviewGlue) => glue._volumePreview$, + deps: [ DatasetPreviewGlue ] + }, { provide: IAV_DATASET_PREVIEW_ACTIVE, useFactory: (glue: DatasetPreviewGlue) => glue.datasetPreviewDisplayed.bind(glue), @@ -234,7 +238,7 @@ export function debug(reducer: ActionReducer<any>): ActionReducer<any> { }, { provide: BS_ENDPOINT, - useValue: (environment.BS_REST_URL || `https://siibra-api-latest.apps-dev.hbp.eu/v1_0`).replace(/\/$/, '') + useValue: (environment.BS_REST_URL || `https://siibra-api-stable.apps.hbp.eu/v1_0`).replace(/\/$/, '') }, ], bootstrap : [ diff --git a/src/services/dialogService.service.ts b/src/services/dialogService.service.ts index ee06d2a91e6a63d5b3e075d6e7cf7a05676108ca..eb0844dd7f3102b921fbe30de7a21a86280fe577 100644 --- a/src/services/dialogService.service.ts +++ b/src/services/dialogService.service.ts @@ -3,6 +3,10 @@ import { ConfirmDialogComponent } from "src/components/confirmDialog/confirmDial import { DialogComponent } from "src/components/dialog/dialog.component"; import {MatDialog, MatDialogRef} from "@angular/material/dialog"; +type TCancellable = { + abort: () => void +} + @Injectable({ providedIn: 'root', }) @@ -16,6 +20,19 @@ export class DialogService { } + public blockUserInteraction(config: Partial<DialogConfig>): TCancellable { + const dialogRef = this.dialog.open(ConfirmDialogComponent, { + data: { + ...config, + hideActionBar: true + }, + hasBackdrop: true, + disableClose: true + }) + const abort = () => dialogRef.close() + return { abort } + } + public getUserConfirm(config: Partial<DialogConfig> = {}): Promise<string> { this.confirmDialogRef = this.dialog.open(ConfirmDialogComponent, { data: config, diff --git a/src/services/state/ngViewerState.store.ts b/src/services/state/ngViewerState.store.ts index b028a6be9f12ed69d22a7da64de7df40831c7dc9..f0ad5000d6b1e1e295f38c09a0749d30fdc9f03e 100644 --- a/src/services/state/ngViewerState.store.ts +++ b/src/services/state/ngViewerState.store.ts @@ -9,7 +9,7 @@ import { HttpClient } from '@angular/common/http'; import { INgLayerInterface, ngViewerActionAddNgLayer, ngViewerActionRemoveNgLayer, ngViewerActionSetPerspOctantRemoval } from './ngViewerState.store.helper' import { PureContantService } from 'src/util'; import { PANELS } from './ngViewerState.store.helper' -import { ngViewerActionToggleMax, ngViewerActionClearView, ngViewerActionSetPanelOrder, ngViewerActionSwitchPanelMode, ngViewerActionForceShowSegment, ngViewerActionNehubaReady } from './ngViewerState/actions'; +import { ngViewerActionToggleMax, ngViewerActionClearView, ngViewerActionSetPanelOrder, ngViewerActionSwitchPanelMode, ngViewerActionForceShowSegment, ngViewerActionNehubaReady, ngViewerActionCycleViews } from './ngViewerState/actions'; import { generalApplyState } from '../stateStore.helper'; import { ngViewerSelectorPanelMode, ngViewerSelectorPanelOrder } from './ngViewerState/selectors'; import { uiActionSnackbarMessage } from './uiState/actions'; @@ -172,9 +172,6 @@ export class NgViewerUseEffect implements OnDestroy { @Effect() public cycleViews$: Observable<any> - @Effect() - public spacebarListener$: Observable<any> - @Effect() public removeAllNonBaseLayers$: Observable<any> @@ -255,7 +252,7 @@ export class NgViewerUseEffect implements OnDestroy { ) this.cycleViews$ = this.actions.pipe( - ofType(ACTION_TYPES.CYCLE_VIEWS), + ofType(ngViewerActionCycleViews.type), withLatestFrom(this.panelOrder$), map(([_, panelOrder]) => { return ngViewerActionSetPanelOrder({ @@ -351,15 +348,6 @@ export class NgViewerUseEffect implements OnDestroy { })), ) - this.spacebarListener$ = fromEvent(document.body, 'keydown', { capture: true }).pipe( - filter((ev: KeyboardEvent) => ev.key === ' '), - withLatestFrom(this.panelMode$), - filter(([_ , panelMode]) => panelMode === PANELS.SINGLE_PANEL), - mapTo({ - type: ACTION_TYPES.CYCLE_VIEWS, - }), - ) - /** * simplify with layer browser */ @@ -427,7 +415,6 @@ export class NgViewerUseEffect implements OnDestroy { export { INgLayerInterface } const ACTION_TYPES = { - CYCLE_VIEWS: 'CYCLE_VIEWS', REMOVE_ALL_NONBASE_LAYERS: `REMOVE_ALL_NONBASE_LAYERS`, } diff --git a/src/services/state/ngViewerState/actions.ts b/src/services/state/ngViewerState/actions.ts index 7f864defac09d58f556d97d276bd44255a435bc3..c504a085a33cb3a8c67291a3a5c28da57a687838 100644 --- a/src/services/state/ngViewerState/actions.ts +++ b/src/services/state/ngViewerState/actions.ts @@ -66,3 +66,7 @@ export const ngViewerActionClearView = createAction( `[ngViewerAction] clearView`, props<{ payload: { [key: string]: boolean }}>() ) + +export const ngViewerActionCycleViews = createAction( + `[ngViewerAction] cycleView` +) \ No newline at end of file diff --git a/src/ui/layerbrowser/layerBrowserComponent/layerbrowser.component.ts b/src/ui/layerbrowser/layerBrowserComponent/layerbrowser.component.ts index 72a0687e3c67ec3cd078aed5a6bb9d309b9445f4..043e0005a7d84e5a99f7e3acf52a48410df73378 100644 --- a/src/ui/layerbrowser/layerBrowserComponent/layerbrowser.component.ts +++ b/src/ui/layerbrowser/layerBrowserComponent/layerbrowser.component.ts @@ -13,6 +13,16 @@ import { ARIA_LABELS } from 'common/constants' import { INgLayerInterface } from "../index"; +const SHOW_LAYER_NAMES = [ + 'PLI Fiber Orientation Red Channel', + 'PLI Fiber Orientation Green Channel', + 'PLI Fiber Orientation Blue Channel', + 'Blockface Image', + 'PLI Transmittance', + 'T2w MRI', + 'MRI Labels' +] + @Component({ selector : 'layer-browser', templateUrl : './layerbrowser.template.html', @@ -98,7 +108,7 @@ export class LayerBrowser implements OnInit, OnDestroy { ).pipe( map(([baseNgLayerNames, loadedNgLayers]) => { const baseNameSet = new Set(baseNgLayerNames) - return loadedNgLayers.filter(l => !baseNameSet.has(l.name)) + return loadedNgLayers.filter(l => SHOW_LAYER_NAMES.includes(l.name)) }), distinctUntilChanged() ) diff --git a/src/ui/topMenu/topMenuCmp/topMenu.components.ts b/src/ui/topMenu/topMenuCmp/topMenu.components.ts index 3b4ecb5f4aa8bd0446c5bcf11a5eee79b27c582b..0d464d59a1fe19491977f384eb0a3cb83f61788c 100644 --- a/src/ui/topMenu/topMenuCmp/topMenu.components.ts +++ b/src/ui/topMenu/topMenuCmp/topMenu.components.ts @@ -3,13 +3,14 @@ import { Component, Input, TemplateRef, + ViewChild, } from "@angular/core"; import { Observable, of } from "rxjs"; import { map } from "rxjs/operators"; import { AuthService } from "src/auth"; import { MatDialog, MatDialogConfig, MatDialogRef } from "@angular/material/dialog"; import { MatBottomSheet } from "@angular/material/bottom-sheet"; -import { CONST, QUICKTOUR_DESC } from 'common/constants' +import { CONST, QUICKTOUR_DESC, ARIA_LABELS } from 'common/constants' import { IQuickTourData } from "src/ui/quickTour/constrants"; import { environment } from 'src/environments/environment' @@ -26,6 +27,7 @@ export class TopMenuCmp { public EXPERIMENTAL_FEATURE_FLAG = environment.EXPERIMENTAL_FEATURE_FLAG + public ARIA_LABELS = ARIA_LABELS public PINNED_DATASETS_BADGE_DESC = CONST.PINNED_DATASETS_BADGE_DESC public matBtnStyle = 'mat-icon-button' @@ -59,6 +61,14 @@ export class TopMenuCmp { order: 8, } + public pinnedDsNotAvail = 'We are reworking pinned dataset feature. Please check back later.' + @ViewChild('savedDatasets', { read: TemplateRef }) + private savedDatasetTmpl: TemplateRef<any> + + public openPinnedDatasets(){ + // this.bottomSheet.open(this.savedDatasetTmpl) + } + constructor( private authService: AuthService, private dialog: MatDialog, diff --git a/src/ui/topMenu/topMenuCmp/topMenu.template.html b/src/ui/topMenu/topMenuCmp/topMenu.template.html index dc889adc9b46ee6ddd826bd4c76452d55beb743f..f652afdd663843230c4b712d3f28a35739751dfd 100644 --- a/src/ui/topMenu/topMenuCmp/topMenu.template.html +++ b/src/ui/topMenu/topMenuCmp/topMenu.template.html @@ -107,16 +107,19 @@ <!-- pinned dataset btn --> <ng-template #pinnedDatasetBtnTmpl> <div class="btnWrapper" - (click)="bottomSheet.open(savedDatasets)" + (click)="openPinnedDatasets()" [matBadge]="(favDataEntries$ | async)?.length > 0 ? (favDataEntries$ | async)?.length : null " matBadgeColor="accent" matBadgePosition="above after" [matBadgeDescription]="PINNED_DATASETS_BADGE_DESC" - matTooltip="Pinned datasets"> + [matTooltip]="pinnedDsNotAvail" + aria-disabled="true" + role="button"> <iav-dynamic-mat-button [attr.pinned-datasets-length]="(favDataEntries$ | async)?.length" [iav-dynamic-mat-button-style]="matBtnStyle" [iav-dynamic-mat-button-color]="matBtnColor" + [iav-dynamic-mat-button-disabled]="true" iav-dynamic-mat-button-aria-label="Show pinned datasets"> <i class="fas fa-thumbtack"></i> diff --git a/src/util/directives/keyDownListener.directive.spec.ts b/src/util/directives/keyDownListener.directive.spec.ts new file mode 100644 index 0000000000000000000000000000000000000000..7be542f0d33d3569569c8415f1e5dfc4fe3a37f2 --- /dev/null +++ b/src/util/directives/keyDownListener.directive.spec.ts @@ -0,0 +1,102 @@ +import { DOCUMENT } from "@angular/common" +import { Component } from "@angular/core" +import { ComponentFixture, fakeAsync, TestBed, tick } from "@angular/core/testing" +import { By } from "@angular/platform-browser" +import { KeyListenerConfig, KeyListner } from "./keyDownListener.directive" + +@Component({ + template: `` +}) + +class DummyCmp{ + public keyConfig: KeyListenerConfig[]=[{ + type: 'keydown', + key: 'a', + },{ + type: 'keyup', + key: 'a', + },{ + type: 'keydown', + key: 'd', + target: 'document', + capture: true + },{ + type: 'keydown', + key: 'e', + target: 'document' + }] + + // will get spied on + public listener(event: any){ + console.log('lister called') + } +} + +const inputId = `text-input` +describe('KeyListner', () => { + beforeEach(async () => { + TestBed.configureTestingModule({ + imports: [], + declarations: [ + KeyListner, + DummyCmp + ], + }).overrideComponent(DummyCmp, { + set: { + template: ` + <div><input id="${inputId}"></div> + <div> + <div + [iav-key-listener]="keyConfig" + (iav-key-event)="listener($event)"> + </div> + </div> + ` + } + }) + + await TestBed.compileComponents() + }) + + it('> creates component just fine', () => { + const fixture = TestBed.createComponent(DummyCmp) + expect(fixture).toBeTruthy() + }) + it('> Directive is created', () => { + const fixture = TestBed.createComponent(DummyCmp) + const keyListenerDirective = fixture.debugElement.query(By.directive(KeyListner)) + expect(keyListenerDirective).toBeTruthy() + }) + + describe('> directive working as intended', () => { + let eventListSpy: jasmine.Spy + let fixture: ComponentFixture<DummyCmp> + beforeEach(() => { + fixture = TestBed.createComponent(DummyCmp) + eventListSpy = spyOn(fixture.componentInstance, 'listener') + fixture.detectChanges() + }) + describe('> if dispatch element was host element', () => { + it('> should trigger event', () => { + const newKeybEv = new KeyboardEvent('keydown', { + key: 'd' + }) + const nativeEl = fixture.nativeElement as HTMLElement + nativeEl.dispatchEvent(newKeybEv) + + expect(eventListSpy).toHaveBeenCalled() + }) + }) + describe('> if dispatch element was input', () => { + it('> should not trigger event listener', () => { + const newKeybEv = new KeyboardEvent('keydown', { + key: 'd' + }) + const nativeEl = fixture.debugElement.query(By.css(`#${inputId}`)).nativeElement as HTMLElement + nativeEl.dispatchEvent(newKeybEv) + + expect(eventListSpy).not.toHaveBeenCalled() + }) + }) + }) +}) diff --git a/src/util/directives/keyDownListener.directive.ts b/src/util/directives/keyDownListener.directive.ts index f3fc055d1ffbb171c5ae2b9a72fa6af3ef12bb80..e400fcea13e3daf14f43aa03d49b578dd330fe2b 100644 --- a/src/util/directives/keyDownListener.directive.ts +++ b/src/util/directives/keyDownListener.directive.ts @@ -100,7 +100,7 @@ export interface KeyListenerConfig { key: string target?: 'document' capture?: boolean - stop: boolean + stop?: boolean // fromEvent seems to be a passive listener, wheather or not { passive: false } flag is set or not // so preventDefault cannot be called anyway } diff --git a/src/util/fn.ts b/src/util/fn.ts index db3ad47ce6e43a8065bef6acc0459be9a72d5d99..479958c477576d2e34b7cc8f1be6243048024992 100644 --- a/src/util/fn.ts +++ b/src/util/fn.ts @@ -246,6 +246,47 @@ const BACKCOMAP_KEY_DICT = { // fsaverage "minds/core/referencespace/v1.0.0/tmp-fsaverage": fsAverageKeyVal, + }, + // allen mouse + 'juelich/iav/atlas/v1.0.0/2': { + // ccf v3 + "minds/core/referencespace/v1.0.0/265d32a0-3d84-40a5-926f-bf89f68212b9": { + // ccf v3 2017 + "minds/core/parcellationatlas/v1.0.0/05655b58-3b6f-49db-b285-64b5a0276f83": { + "whole brain": "v3_2017", + "left hemisphere": "v3_2017", + "right hemisphere": "v3_2017" + }, + // ccf v3 2015, + "minds/core/parcellationatlas/v1.0.0/39a1384b-8413-4d27-af8d-22432225401f": { + "whole brain": "atlas", + "left hemisphere": "atlas", + "right hemisphere": "atlas" + } + } + }, + // waxholm + "minds/core/parcellationatlas/v1.0.0/522b368e-49a3-49fa-88d3-0870a307974a": { + "minds/core/referencespace/v1.0.0/d5717c4a-0fa1-46e6-918c-b8003069ade8": { + // v1.01 + "minds/core/parcellationatlas/v1.0.0/11017b35-7056-4593-baad-3934d211daba": { + "whole brain": "v1_01", + "left hemisphere": "v1_01", + "right hemisphere": "v1_01" + }, + // v2 + "minds/core/parcellationatlas/v1.0.0/2449a7f0-6dd0-4b5a-8f1e-aec0db03679d": { + "whole brain": "v2", + "left hemisphere": "v2", + "right hemisphere": "v2" + }, + // v3 + "minds/core/parcellationatlas/v1.0.0/ebb923ba-b4d5-4b82-8088-fa9215c2e1fe": { + "whole brain": "v3", + "left hemisphere": "v3", + "right hemisphere": "v3" + } + } } } @@ -287,14 +328,41 @@ export class MultiDimMap{ } } -export function recursiveMutate<T>(arr: T[], getChildren: (obj: T) => T[], mutateFn: (obj: T) => void){ +export function mutateDeepMerge(toObj: any, fromObj: any){ + if (typeof toObj !== 'object') throw new Error(`toObj needs to be object`) + if (typeof fromObj !== 'object') throw new Error(`fromObj needs to be object`) + + for (const key in fromObj) { + if (!toObj[key]) { + toObj[key] = fromObj[key] + continue + } + if (Array.isArray(toObj[key])) { + const objToAppend = Array.isArray(fromObj[key]) + ? fromObj[key] + : [fromObj[key]] + toObj[key].push(...objToAppend) + continue + } + if (typeof toObj[key] === typeof fromObj[key] && typeof toObj[key] === 'object') { + mutateDeepMerge(toObj[key], fromObj[key]) + continue + } + throw new Error(`cannot mutate ${key} typeof ${typeof fromObj[key]}`) + } + + return toObj +} + +export function recursiveMutate<T>(arr: T[], getChildren: (obj: T) => T[], mutateFn: (obj: T) => void, depthFirst = false){ for (const obj of arr) { - mutateFn(obj) + if (!depthFirst) mutateFn(obj) recursiveMutate( getChildren(obj), getChildren, mutateFn ) + if (depthFirst) mutateFn(obj) } } diff --git a/src/util/patchPureConstants.ts b/src/util/patchPureConstants.ts new file mode 100644 index 0000000000000000000000000000000000000000..fb1b4ad5b95c613133be272ba8874d39503f6df4 --- /dev/null +++ b/src/util/patchPureConstants.ts @@ -0,0 +1,197 @@ +/** + * README: the purpose of this file is to monkey patch discrepency between siibra-api + * backend and original backend. + * + * In principle, these should be built into siibra-python, and this file should become obsolete. + */ + +import { IHasId } from "./interfaces"; +import { TRegion } from "./siibraApiConstants/types"; + +type TAppend = { + parent: IHasId | { name: string } + '@type': 'julich/siibra/append-region/v0.0.1' +} + +type TPatch = { + target: IHasId | { name: string } + '@type': 'julich/siibra/patch-region/v0.0.1' +} + +type TPatchRegion = { + '@id': string + targetSpace: IHasId[] | '*' + targetParcellation: IHasId[] | '*' + payload: Partial<TRegion> +} & (TAppend | TPatch) + +const encoder = new TextEncoder() +async function getShaDigest(input: string){ + const digest = await crypto.subtle.digest('SHA-1', encoder.encode(input)) + const array = Array.from(new Uint8Array(digest)) + const hex = array.map(v => v.toString(16)).join('') + return hex +} +async function getInterpolatedPatchObj(targetName: string, labelIndex: number){ + const returnObj: TPatchRegion = { + '@id': '', + "@type": 'julich/siibra/patch-region/v0.0.1', + "target": { + "name": targetName + }, + "targetParcellation": [{ + "@id": 'minds/core/parcellationatlas/v1.0.0/94c1125b-b87e-45e4-901c-00daee7f2579-290' + }], + "targetSpace": [{ + '@id': 'minds/core/referencespace/v1.0.0/a1655b99-82f1-420f-a3c2-fe80fd4c8588' + }], + "payload": { + "volumeSrc": { + 'minds/core/referencespace/v1.0.0/a1655b99-82f1-420f-a3c2-fe80fd4c8588': { + "collect": [{ + "@type": "fzj/tmp/volume_type/v0.0.1" as "fzj/tmp/volume_type/v0.0.1", + "@id": "fzj/tmp/volume_type/v0.0.1/interpolated", + "name": "Julich Brain v2.5 interpolated map", + "volume_type": "neuroglancer/precomputed" as "neuroglancer/precomputed", + "url": "https://neuroglancer.humanbrainproject.org/precomputed/BigBrainRelease.2015/2019_05_22_interpolated_areas", + "detail": { + "neuroglancer/precomputed": { + "labelIndex": labelIndex, + "transform": [[1,0,0,-70677184],[0,1,0,-51990000],[0,0,1,-58788284],[0,0,0,1]] + } + }, + "space_id": "minds/core/referencespace/v1.0.0/a1655b99-82f1-420f-a3c2-fe80fd4c8588", + map_type: 'labelled' + }] + } + } + } + } + const hex = await getShaDigest(JSON.stringify(returnObj)) + return { + ...returnObj, + '@id': hex + } +} + +async function getIndividualMap(parentName: string, regionName: string, url: string, transform: number[][], labelIndex: number){ + const volumeId = await getShaDigest(url) + const returnObj: TPatchRegion = { + '@id': '', + "@type": 'julich/siibra/append-region/v0.0.1', + "parent": { + "name": parentName + }, + "targetParcellation": [{ + "@id": 'minds/core/parcellationatlas/v1.0.0/94c1125b-b87e-45e4-901c-00daee7f2579-290' + }], + "targetSpace": [{ + '@id': 'minds/core/referencespace/v1.0.0/a1655b99-82f1-420f-a3c2-fe80fd4c8588' + }], + "payload": { + "name": regionName, + "volumeSrc": { + 'minds/core/referencespace/v1.0.0/a1655b99-82f1-420f-a3c2-fe80fd4c8588': { + "collect": [{ + "@type": "fzj/tmp/volume_type/v0.0.1" as "fzj/tmp/volume_type/v0.0.1", + "@id": `fzj/tmp/volume_type/v0.0.1/${volumeId}`, + "name": "Julich Brain v2.5 detailed map", + "volume_type": "neuroglancer/precomputed" as "neuroglancer/precomputed", + "url": url, + "detail": { + "neuroglancer/precomputed": { + "labelIndex": labelIndex, + "transform": transform + } + }, + space_id: 'minds/core/referencespace/v1.0.0/a1655b99-82f1-420f-a3c2-fe80fd4c8588', + map_type: 'labelled' + }] + } + } + } + } + const hex = await getShaDigest(JSON.stringify(returnObj)) + return { + ...returnObj, + '@id': hex + } +} + +const bigBrainRegions: Promise<TPatchRegion>[] = [ + getInterpolatedPatchObj('Area IFJ1 (IFS,PreCS)', 9), + getInterpolatedPatchObj('Area IFJ2 (IFS,PreCS)', 10), + getInterpolatedPatchObj('Area IFS1 (IFS)', 11), + getInterpolatedPatchObj('Area IFS2 (IFS)', 12), + getInterpolatedPatchObj('Area IFS3 (IFS)', 13), + getInterpolatedPatchObj('Area IFS4 (IFS)', 14), + + getIndividualMap( + 'lateral geniculate body', + 'LGB-lam1 (CGL, Metathalamus)', + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2020_11_11_LGB-lam/LGB-lam1/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), + getIndividualMap( + 'lateral geniculate body', + 'LGB-lam2 (CGL, Metathalamus)', + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2020_11_11_LGB-lam/LGB-lam2/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), + getIndividualMap( + 'lateral geniculate body', + 'LGB-lam3 (CGL, Metathalamus)', + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2020_11_11_LGB-lam/LGB-lam3/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), + getIndividualMap( + 'lateral geniculate body', + 'LGB-lam4 (CGL, Metathalamus)', + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2020_11_11_LGB-lam/LGB-lam4/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), + getIndividualMap( + 'lateral geniculate body', + 'LGB-lam5 (CGL, Metathalamus)', + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2020_11_11_LGB-lam/LGB-lam5/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), + getIndividualMap( + 'lateral geniculate body', + 'LGB-lam6 (CGL, Metathalamus)', + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2020_11_11_LGB-lam/LGB-lam6/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), + + getIndividualMap( + 'medial geniculate body', + "MGB-MGBd (CGM, Metathalamus)", + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2021_04_27_mgb/2021_04_27_MGBd/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), + getIndividualMap( + 'medial geniculate body', + "MGB-MGBm (CGM, Metathalamus)", + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2021_04_27_mgb/2021_04_27_MGBm/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), + getIndividualMap( + 'medial geniculate body', + "MGB-MGBv (CGM, Metathalamus)", + "https://neuroglancer.humanbrainproject.eu/precomputed/BigBrainRelease.2015/2021_04_27_mgb/2021_04_27_MGBv/", + [[1,0,0,-70677184.0],[0,1,0,-7290000.0],[0,0,1,-58788284.0],[0,0,0,1]], + 1 + ), +] + +export const patchRegions = [ + ...bigBrainRegions +] diff --git a/src/util/pureConstant.service.ts b/src/util/pureConstant.service.ts index b7962a9fe0ed9622e7237cb2b7cd7245ac9539ad..4c4b83719692c209cb80e2e52a217bec5a38d5fe 100644 --- a/src/util/pureConstant.service.ts +++ b/src/util/pureConstant.service.ts @@ -1,6 +1,6 @@ import { Inject, Injectable, OnDestroy } from "@angular/core"; import { Store, select } from "@ngrx/store"; -import { Observable, Subscription, of, forkJoin, combineLatest } from "rxjs"; +import { Observable, Subscription, of, forkJoin, combineLatest, from } from "rxjs"; import { viewerConfigSelectorUseMobileUi } from "src/services/state/viewerConfig.store.helper"; import { shareReplay, tap, scan, catchError, filter, switchMap, map, distinctUntilChanged, mapTo } from "rxjs/operators"; import { HttpClient } from "@angular/common/http"; @@ -10,7 +10,8 @@ import { viewerStateFetchedAtlasesSelector, viewerStateSelectedTemplateSelector import { BS_ENDPOINT, BACKENDURL } from "src/util/constants"; import { flattenReducer } from 'common/util' import { IVolumeTypeDetail, TAtlas, TId, TParc, TRegion, TRegionDetail, TSpaceFull, TSpaceSummary, TVolumeSrc } from "./siibraApiConstants/types"; -import { MultiDimMap, recursiveMutate } from "./fn"; +import { MultiDimMap, recursiveMutate, mutateDeepMerge } from "./fn"; +import { patchRegions } from './patchPureConstants' const validVolumeType = new Set([ @@ -204,6 +205,12 @@ Raise/track issues at github repo: <a target = "_blank" href = "${this.repoUrl}" ) } + private patchRegions$ = forkJoin( + patchRegions.map(patch => from(patch)) + ).pipe( + shareReplay(1) + ) + private getRegions(atlasId: string, parcId: string, spaceId: string){ return this.http.get<TRegion[]>( `${this.bsEndpoint}/atlases/${encodeURIComponent(atlasId)}/parcellations/${encodeURIComponent(parcId)}/regions`, @@ -213,6 +220,53 @@ Raise/track issues at github repo: <a target = "_blank" href = "${this.repoUrl}" }, responseType: 'json' } + ).pipe( + switchMap(regions => this.patchRegions$.pipe( + map(patchRegions => { + for (const p of patchRegions) { + if ( + p.targetParcellation !== '*' + && Array.isArray(p.targetParcellation) + && p.targetParcellation.every(p => p["@id"] !== parcId) + ) { + continue + } + if ( + p.targetSpace !== '*' + && Array.isArray(p.targetSpace) + && p.targetSpace.every(sp => sp['@id'] !== spaceId) + ) { + continue + } + + recursiveMutate( + regions, + r => r.children || [], + region => { + + if (p["@type"] === 'julich/siibra/append-region/v0.0.1') { + if (p.parent['name'] === region.name) { + if (!region.children) region.children = [] + region.children.push( + p.payload as TRegion + ) + } + } + if (p['@type'] === 'julich/siibra/patch-region/v0.0.1') { + if (p.target['name'] === region.name) { + mutateDeepMerge( + region, + p.payload + ) + } + } + }, + true + ) + } + return regions + }) + )) ) } @@ -435,7 +489,7 @@ Raise/track issues at github repo: <a target = "_blank" href = "${this.repoUrl}" }), catchError((err, obs) => of([])), tap((arr: any[]) => this.totalAtlasesLength = arr.length), - scan((acc, curr) => acc.concat(curr).sort((a, b) => (a.order || 1001) - (b.order || 1000)), []), + scan((acc, curr) => acc.concat(curr).sort((a, b) => (a.order || 0) - (b.order || 0)), []), shareReplay(1) ) diff --git a/src/util/siibraApiConstants/types.ts b/src/util/siibraApiConstants/types.ts index 677827288aaff99b13332ee6de7559920d64837e..b73781a0242f120fbe78571fd9ae64e06cddc2c4 100644 --- a/src/util/siibraApiConstants/types.ts +++ b/src/util/siibraApiConstants/types.ts @@ -18,6 +18,7 @@ export interface IVolumeTypeDetail { 'nii': null 'neuroglancer/precomputed': { 'neuroglancer/precomputed': { + 'labelIndex': number 'transform': TNgTransform } } diff --git a/src/viewerModule/module.ts b/src/viewerModule/module.ts index 53d48a2ecc114144ba740b3f25231f2030882df9..8cd43cf6ace30f25be8afea1cffe58be1c175881 100644 --- a/src/viewerModule/module.ts +++ b/src/viewerModule/module.ts @@ -28,6 +28,7 @@ import { API_SERVICE_SET_VIEWER_HANDLE_TOKEN, AtlasViewerAPIServices, setViewerH import { ILoadMesh, LOAD_MESH_TOKEN } from "src/messaging/types"; import { KeyFrameModule } from "src/keyframesModule/module"; import { ViewerInternalStateSvc } from "./viewerInternalState.service"; +import { LayerBrowserModule } from "src/ui/layerbrowser"; @NgModule({ imports: [ @@ -50,6 +51,7 @@ import { ViewerInternalStateSvc } from "./viewerInternalState.service"; ViewerStateBreadCrumbModule, KgRegionalFeatureModule, KeyFrameModule, + LayerBrowserModule, ], declarations: [ ViewerCmp, diff --git a/src/viewerModule/nehuba/layerCtrl.service/layerCtrl.service.ts b/src/viewerModule/nehuba/layerCtrl.service/layerCtrl.service.ts index 56e87c06579b66207560caf92611ca2ab83b6c39..d3232e3e098d260c3771b95858fb428ea7f1caf0 100644 --- a/src/viewerModule/nehuba/layerCtrl.service/layerCtrl.service.ts +++ b/src/viewerModule/nehuba/layerCtrl.service/layerCtrl.service.ts @@ -1,7 +1,7 @@ import { Inject, Injectable, OnDestroy, Optional } from "@angular/core"; import { select, Store } from "@ngrx/store"; -import { BehaviorSubject, combineLatest, from, merge, Observable, of, Subject, Subscription } from "rxjs"; -import { debounceTime, distinctUntilChanged, filter, map, shareReplay, switchMap, withLatestFrom } from "rxjs/operators"; +import { BehaviorSubject, combineLatest, from, merge, NEVER, Observable, of, Subject, Subscription } from "rxjs"; +import { debounceTime, distinctUntilChanged, filter, map, shareReplay, startWith, switchMap, withLatestFrom } from "rxjs/operators"; import { viewerStateCustomLandmarkSelector, viewerStateSelectedParcellationSelector, viewerStateSelectedRegionsSelector, viewerStateSelectedTemplateSelector } from "src/services/state/viewerState/selectors"; import { getRgb, IColorMap, INgLayerCtrl, INgLayerInterface, TNgLayerCtrl } from "./layerCtrl.util"; import { getMultiNgIdsRegionsLabelIndexMap } from "../constants"; @@ -12,6 +12,7 @@ import { EnumColorMapName } from "src/util/colorMaps"; import { getShader, PMAP_DEFAULT_CONFIG } from "src/util/constants"; import { ngViewerActionAddNgLayer, ngViewerActionRemoveNgLayer, ngViewerSelectorClearView, ngViewerSelectorLayers } from "src/services/state/ngViewerState.store.helper"; import { serialiseParcellationRegion } from 'common/util' +import { _PLI_VOLUME_INJ_TOKEN, _TPLIVal } from "src/glue"; export const BACKUP_COLOR = { red: 255, @@ -36,7 +37,9 @@ export function getAuxMeshesAndReturnIColor(auxMeshes: IAuxMesh[]): IColorMap{ return returnVal } -@Injectable() +@Injectable({ + providedIn: 'root' +}) export class NehubaLayerControlService implements OnDestroy{ static PMAP_LAYER_NAME = 'regional-pmap' @@ -144,10 +147,25 @@ export class NehubaLayerControlService implements OnDestroy{ while (this.sub.length > 0) this.sub.pop().unsubscribe() } + private pliVol$: Observable<string[]> = this._pliVol$ + ? this._pliVol$.pipe( + map(arr => { + const output = [] + for (const item of arr) { + for (const volume of item.data["iav-registered-volumes"].volumes) { + output.push(volume.name) + } + } + return output + }) + ) + : NEVER constructor( private store$: Store<any>, + @Optional() @Inject(_PLI_VOLUME_INJ_TOKEN) private _pliVol$: Observable<_TPLIVal[]>, @Optional() @Inject(REGION_OF_INTEREST) roi$: Observable<TRegionDetail> ){ + if (roi$) { this.sub.push( @@ -285,10 +303,10 @@ export class NehubaLayerControlService implements OnDestroy{ shareReplay(1) ) - public visibleLayer$: Observable<string[]> = combineLatest([ + public expectedLayerNames$ = combineLatest([ this.selectedTemplateSelector$, this.auxMeshes$, - this.selParcNgIdMap$ + this.selParcNgIdMap$, ]).pipe( map(([ tmpl, auxMeshes, parcNgIdMap ]) => { const ngIdSet = new Set<string>() @@ -305,6 +323,18 @@ export class NehubaLayerControlService implements OnDestroy{ }) ) + public visibleLayer$: Observable<string[]> = combineLatest([ + this.expectedLayerNames$, + this.pliVol$.pipe( + startWith([]) + ), + ]).pipe( + map(([ expectedLayerNames, layerNames ]) => { + const ngIdSet = new Set<string>([...layerNames, ...expectedLayerNames]) + return Array.from(ngIdSet) + }) + ) + /** * define when shown segments should be updated */ diff --git a/src/viewerModule/nehuba/nehubaViewerGlue/nehubaViewerGlue.component.ts b/src/viewerModule/nehuba/nehubaViewerGlue/nehubaViewerGlue.component.ts index 6256f6b7e8f02b90e4043e35ee6d0c714a510126..15504851b024378aeb96c8c96ef2f04d73489842 100644 --- a/src/viewerModule/nehuba/nehubaViewerGlue/nehubaViewerGlue.component.ts +++ b/src/viewerModule/nehuba/nehubaViewerGlue/nehubaViewerGlue.component.ts @@ -1,7 +1,7 @@ import { AfterViewInit, Component, ElementRef, EventEmitter, Inject, Input, OnChanges, OnDestroy, Optional, Output, SimpleChanges, TemplateRef, ViewChild } from "@angular/core"; import { select, Store } from "@ngrx/store"; import { asyncScheduler, combineLatest, fromEvent, merge, NEVER, Observable, of, Subject } from "rxjs"; -import { ngViewerActionToggleMax } from "src/services/state/ngViewerState/actions"; +import { ngViewerActionCycleViews, ngViewerActionToggleMax } from "src/services/state/ngViewerState/actions"; import { ClickInterceptor, CLICK_INTERCEPTOR_INJECTOR } from "src/util"; import { uiStateMouseOverSegmentsSelector } from "src/services/state/uiState/selectors"; import { debounceTime, distinctUntilChanged, filter, map, mapTo, scan, shareReplay, startWith, switchMap, switchMapTo, take, tap, throttleTime } from "rxjs/operators"; @@ -79,6 +79,8 @@ export class NehubaGlueCmp implements IViewer<'nehuba'>, OnChanges, OnDestroy, A public ARIA_LABELS = ARIA_LABELS public IDS = IDS + private currentPanelMode: PANELS + @ViewChild(NehubaViewerContainerDirective, { static: true }) public nehubaContainerDirective: NehubaViewerContainerDirective @@ -341,6 +343,9 @@ export class NehubaGlueCmp implements IViewer<'nehuba'>, OnChanges, OnDestroy, A ]).pipe( switchMap(this.waitForNehuba.bind(this)) ).subscribe(([mode, panelOrder]) => { + + this.currentPanelMode = mode + const viewPanels = panelOrder.split('').map(v => Number(v)).map(idx => this.viewPanels[idx]) as [HTMLElement, HTMLElement, HTMLElement, HTMLElement] /** @@ -652,6 +657,13 @@ export class NehubaGlueCmp implements IViewer<'nehuba'>, OnChanges, OnDestroy, A this.onDestroyCb.push(() => navSub.unsubscribe()) } + handleCycleViewEvent(){ + if (this.currentPanelMode !== PANELS.SINGLE_PANEL) return + this.store$.dispatch( + ngViewerActionCycleViews() + ) + } + handleViewerLoadedEvent(flag: boolean) { this.viewerEvent.emit({ type: EnumViewerEvt.VIEWERLOADED, diff --git a/src/viewerModule/nehuba/nehubaViewerGlue/nehubaViewerGlue.template.html b/src/viewerModule/nehuba/nehubaViewerGlue/nehubaViewerGlue.template.html index ba41f41c2cc6d2fc711a0ccb60cee8c8f512ae3f..73aeec9212a1f91a1fa8c16ce1c047e655a0177d 100644 --- a/src/viewerModule/nehuba/nehubaViewerGlue/nehubaViewerGlue.template.html +++ b/src/viewerModule/nehuba/nehubaViewerGlue/nehubaViewerGlue.template.html @@ -11,6 +11,8 @@ iav-nehuba-viewer-container #iavContainer="iavNehubaViewerContainer" iav-mouse-hover + [iav-key-listener]="[{ type: 'keydown', key: ' ', target: 'document', capture: true }]" + (iav-key-event)="handleCycleViewEvent()" (iavNehubaViewerContainerViewerLoading)="handleViewerLoadedEvent($event)"> </div> diff --git a/src/viewerModule/nehuba/statusCard/statusCard.component.ts b/src/viewerModule/nehuba/statusCard/statusCard.component.ts index a6daa233d6acdc5197d19d4a9dbc284c3e26f329..76dfa6bbbdb0288b6755984fa651d71b3e50c5df 100644 --- a/src/viewerModule/nehuba/statusCard/statusCard.component.ts +++ b/src/viewerModule/nehuba/statusCard/statusCard.component.ts @@ -57,6 +57,8 @@ export class StatusCardComponent implements OnInit, OnChanges{ order: 6, } + public saneUrlDeprecated = `Custom URL is going away. New custom URLs can no longer be created. Custom URLs you generated in the past will continue to work.` + public SHARE_BTN_ARIA_LABEL = ARIA_LABELS.SHARE_BTN public COPY_URL_TO_CLIPBOARD_ARIA_LABEL = ARIA_LABELS.SHARE_COPY_URL_CLIPBOARD public SHARE_CUSTOM_URL_ARIA_LABEL = ARIA_LABELS.SHARE_CUSTOM_URL diff --git a/src/viewerModule/nehuba/statusCard/statusCard.template.html b/src/viewerModule/nehuba/statusCard/statusCard.template.html index 5e4d793a725e31f41f5bd45406a11b95ef1340a6..5f57d9719055a265dbfb005c5c4d7d3cc46ab1ea 100644 --- a/src/viewerModule/nehuba/statusCard/statusCard.template.html +++ b/src/viewerModule/nehuba/statusCard/statusCard.template.html @@ -154,9 +154,11 @@ Copy link to this view </span> </mat-list-item> - <mat-list-item (click)="openDialog(shareSaneUrl, { ariaLabel: SHARE_CUSTOM_URL_DIALOG_ARIA_LABEL })" + <mat-list-item [attr.aria-label]="SHARE_CUSTOM_URL_ARIA_LABEL" - [attr.tab-index]="10"> + [attr.tab-index]="10" + [matTooltip]="saneUrlDeprecated" + class="text-muted"> <mat-icon class="mr-4" fontSet="fas" diff --git a/src/viewerModule/threeSurfer/threeSurferGlue/threeSurfer.component.ts b/src/viewerModule/threeSurfer/threeSurferGlue/threeSurfer.component.ts index 551bec7a55fc14645db6470dc3920ddaef092a8c..a00b376604e92ed7e4328b4289c227813f475836 100644 --- a/src/viewerModule/threeSurfer/threeSurferGlue/threeSurfer.component.ts +++ b/src/viewerModule/threeSurfer/threeSurferGlue/threeSurfer.component.ts @@ -28,6 +28,7 @@ type TInternalState = { hemisphere: 'left' | 'right' | 'both' } const pZoomFactor = 5e3 +const preferredFsMode = 'pial' type THandlingCustomEv = { regions: ({ name?: string, error?: string })[] @@ -534,8 +535,9 @@ export class ThreeSurferGlueCmp implements IViewer<'threeSurfer'>, OnChanges, Af } } - // load mode0 by default - this.loadMode(this.config.modes[0]) + // load preferredFsMode or mode0 by default + const loadMode = this.config.modes.find(m => m.name === preferredFsMode) || this.config.modes[0] + this.loadMode(loadMode) this.viewerEvent.emit({ type: EnumViewerEvt.VIEWERLOADED, diff --git a/src/viewerModule/viewerCmp/viewerCmp.component.ts b/src/viewerModule/viewerCmp/viewerCmp.component.ts index 1426fc874577952974e87f3ecc347ee632004db2..bb0c20a548bf82504019a4fe307c180d6d498d4b 100644 --- a/src/viewerModule/viewerCmp/viewerCmp.component.ts +++ b/src/viewerModule/viewerCmp/viewerCmp.component.ts @@ -1,6 +1,6 @@ import { Component, ComponentFactory, ComponentFactoryResolver, ElementRef, Inject, Injector, Input, OnDestroy, Optional, TemplateRef, ViewChild, ViewContainerRef } from "@angular/core"; import { select, Store } from "@ngrx/store"; -import {combineLatest, merge, Observable, of, Subject, Subscription} from "rxjs"; +import {combineLatest, merge, NEVER, Observable, of, Subject, Subscription} from "rxjs"; import {catchError, distinctUntilChanged, filter, map, shareReplay, startWith, switchMap } from "rxjs/operators"; import { viewerStateSetSelectedRegions } from "src/services/state/viewerState/actions"; import { @@ -23,6 +23,8 @@ import { ContextMenuService, TContextMenuReg } from "src/contextMenuModule"; import { ComponentStore } from "../componentStore"; import { MAT_DIALOG_DATA } from "@angular/material/dialog"; import { GenericInfoCmp } from "src/atlasComponents/regionalFeatures/bsFeatures/genericInfo"; +import { _PLI_VOLUME_INJ_TOKEN, _TPLIVal } from "src/glue"; +import { uiActionSetPreviewingDatasetFiles } from "src/services/state/uiState.store.helper"; type TCStoreViewerCmp = { overlaySideNav: any @@ -115,7 +117,10 @@ export function ROIFactory(store: Store<any>, svc: PureContantService){ }) export class ViewerCmp implements OnDestroy { - + public _pliTitle = "Fiber structures of a human hippocampus based on joint DMRI, 3D-PLI, and TPFM acquisitions" + public _pliDesc = "The collected datasets provide real multimodal, multiscale structural connectivity insights into the human hippocampus. One post mortem hippocampus was scanned with Anatomical and Diffusion MRI (dMRI) [1], 3D Polarized Light Imaging (3D-PLI) [2], and Two-Photon Fluorescence Microscopy (TPFM) [3] using protocols specifically developed during SGA1 and SGA2, rendering joint tissue imaging possible. MRI scanning was performed with a 11.7 T Preclinical MRI system (gradients: 760 mT/m, slew rate: 9500 T/m/s) yielding T1-w and T2-w maps at 200 µm and dMRI-based maps at 300 µm resolution. During tissue sectioning (60 µm thickness) blockface (en-face) images were acquired from the surface of the frozen brain block, serving as reference for data integration/co-alignment. 530 brain sections were scanned with 3D-PLI. HPC-based image analysis provided transmittance, retardation, and fiber orientation maps at 1.3 µm in-plane resolution. TPFM was finally applied to selected brain sections utilizing autofluorescence properties of the fibrous tissue which appears after PBS washing (MAGIC protocol). The TPFM measurements provide a resolution of 0.44 µm x 0.44 µm x 1 µm." + public _pliLink = "https://doi.org/10.25493/JQ30-E08" + public CONST = CONST public ARIA_LABELS = ARIA_LABELS @@ -215,17 +220,36 @@ export class ViewerCmp implements OnDestroy { private getRegionFromlabelIndexId: (arg: {labelIndexId: string}) => any private genericInfoCF: ComponentFactory<GenericInfoCmp> + + public pliVol$ = this._pliVol$ || NEVER + public clearVoi(){ + this.store$.dispatch( + uiActionSetPreviewingDatasetFiles({ + previewingDatasetFiles: [] + }) + ) + } constructor( private store$: Store<any>, private viewerModuleSvc: ContextMenuService<TContextArg<'threeSurfer' | 'nehuba'>>, private cStore: ComponentStore<TCStoreViewerCmp>, cfr: ComponentFactoryResolver, + @Optional() @Inject(_PLI_VOLUME_INJ_TOKEN) private _pliVol$: Observable<_TPLIVal[]>, @Optional() @Inject(REGION_OF_INTEREST) public regionOfInterest$: Observable<any> ){ this.genericInfoCF = cfr.resolveComponentFactory(GenericInfoCmp) this.subscriptions.push( + this.pliVol$.subscribe(val => { + if (val.length > 0) { + this.sidenavTopSwitch && this.sidenavTopSwitch.open() + this.sidenavLeftSwitch && this.sidenavLeftSwitch.open() + } else { + this.sidenavTopSwitch && this.sidenavTopSwitch.close() + this.sidenavLeftSwitch && this.sidenavLeftSwitch.close() + } + }), this.selectedRegions$.subscribe(() => { this.clearPreviewingDataset() }), diff --git a/src/viewerModule/viewerCmp/viewerCmp.template.html b/src/viewerModule/viewerCmp/viewerCmp.template.html index 30d2c0411ce039d73beba463374fe6a3e2bed884..b77622c13103c80c4ccc5e2c938ccd1bca2a97d1 100644 --- a/src/viewerModule/viewerCmp/viewerCmp.template.html +++ b/src/viewerModule/viewerCmp/viewerCmp.template.html @@ -39,11 +39,13 @@ matColor: 'primary', fontIcon: 'fa-list', tooltip: 'Annotation list', - click: viewerModeDrawer.toggle.bind(viewerModeDrawer) + click: viewerModeDrawer.toggle.bind(viewerModeDrawer), + badge: toolPanel?.annBadges$ | async }"> </ng-container> - <annotating-tools-panel class="z-index-10"> + <annotating-tools-panel class="z-index-10" + #toolPanel="annoToolsPanel"> </annotating-tools-panel> </div> @@ -97,7 +99,6 @@ </mat-drawer-container> - <!-- top drawer --> <mat-drawer-container [hidden]="viewerMode$ | async" @@ -250,9 +251,14 @@ 'invisible overflow-hidden h-0': overlaySidenav$ | async, 'h-100': !(overlaySidenav$ | async) }" class="position-relative d-flex flex-column"> + + <ng-template let-pliVol [ngIf]="pliVol$ | async" [ngIfElse]="sidenavRegionTmpl"> + <ng-template [ngIf]="pliVol.length > 0" [ngIfElse]="sidenavRegionTmpl"> + <ng-template [ngTemplateOutlet]="voiTmpl"> - <ng-container *ngTemplateOutlet="sidenavRegionTmpl"> - </ng-container> + </ng-template> + </ng-template> + </ng-template> <!-- TODO dataset preview will become deprecated in the future. Regional feature/data feature will replace it --> @@ -422,6 +428,8 @@ let-customColor="customColor" let-customColorDarkmode="customColorDarkmode" let-tooltip="tooltip" + let-badge="badge" + let-badgeColor="badgeColor" let-click="click"> <!-- (click)="sideNavMasterSwitch.toggle()" --> <button mat-raised-button @@ -434,7 +442,9 @@ }" (click)="click && click()" [style.backgroundColor]="customColor" - [color]="(!customColor && matColor) ? matColor : null"> + [color]="(!customColor && matColor) ? matColor : null" + [matBadge]="badge" + [matBadgeColor]="badgeColor || 'warn'"> <span [ngClass]="{'iv-custom-comp text': !!customColor}"> <i class="fas" [ngClass]="fontIcon || 'fa-question'"></i> @@ -442,6 +452,59 @@ </button> </ng-template> +<!-- VOI sidenav tmpl --> +<ng-template #voiTmpl> + + <!-- back btn --> + <button mat-button + (click)="clearVoi()" + [attr.aria-label]="ARIA_LABELS.CLOSE" + class="position-absolute z-index-10 m-2"> + <i class="fas fa-chevron-left"></i> + <span class="ml-1"> + Back + </span> + </button> + + <mat-card class="sidenav-cover-header-container"> + <div class="sidenav-cover-header-container"> + <mat-card-title> + {{ _pliTitle }} + </mat-card-title> + + <mat-card-subtitle class="d-inline-flex align-items-center flex-wrap"> + <mat-icon fontSet="fas" fontIcon="fa-database"></mat-icon> + <span> + Dataset preview + </span> + + <mat-divider vertical="true" class="ml-2 h-2rem"></mat-divider> + + <a [href]="_pliLink" + mat-icon-button + matTooltip="Explore in EBRAINS Knowledge Graph" + target="_blank"> + <i class="fas fa-external-link-alt"></i> + </a> + + </mat-card-subtitle> + </div> + + <small class="text-muted iv-custom-comp darker-bg"> + {{ _pliDesc }} + </small> + + <mat-expansion-panel class="sidenav-cover-header-container"> + <mat-expansion-panel-header> + <mat-panel-title> + Registered Volumes + </mat-panel-title> + </mat-expansion-panel-header> + <layer-browser></layer-browser> + </mat-expansion-panel> + </mat-card> +</ng-template> + <!-- region sidenav tmpl --> <ng-template #sidenavRegionTmpl> @@ -572,21 +635,21 @@ <!-- Multi regions include --> <ng-template #multiRegionInclTmpl> - <mat-chip-list> - <mat-chip *ngFor="let r of regions" - iav-region - [region]="r" - [ngClass]="{ - 'darktheme':regionDirective.rgbDarkmode === true, - 'lighttheme': regionDirective.rgbDarkmode === false - }" - [style.backgroundColor]="regionDirective.rgbString" - #regionDirective="iavRegion"> - <span class="iv-custom-comp text text-truncate d-inline pl-4"> - {{ r.name }} - </span> - </mat-chip> - </mat-chip-list> + + <mat-chip *ngFor="let r of regions" + iav-region + [region]="r" + class="m-1" + [ngClass]="{ + 'darktheme':regionDirective.rgbDarkmode === true, + 'lighttheme': regionDirective.rgbDarkmode === false + }" + [style.backgroundColor]="regionDirective.rgbString" + #regionDirective="iavRegion"> + <span class="iv-custom-comp text text-truncate d-inline"> + {{ r.name }} + </span> + </mat-chip> </ng-template> <ng-container *ngTemplateOutlet="ngMatAccordionTmpl; context: { diff --git a/src/zipFilesOutput/zipFilesOutput.directive.ts b/src/zipFilesOutput/zipFilesOutput.directive.ts index bcf70bddfaf8e3e84ab97f887ae93ae8e95fcfc6..dc980728cfd8675ecfaaf57eb1c4317f74556ba0 100644 --- a/src/zipFilesOutput/zipFilesOutput.directive.ts +++ b/src/zipFilesOutput/zipFilesOutput.directive.ts @@ -2,6 +2,8 @@ import { Directive, HostListener, Inject, Input } from "@angular/core"; import { TZipFileConfig } from "./type"; import * as JSZip from "jszip"; import { DOCUMENT } from "@angular/common"; +import { isObservable, Observable } from "rxjs"; +import { take } from "rxjs/operators"; @Directive({ selector: '[zip-files-output]', @@ -10,15 +12,15 @@ import { DOCUMENT } from "@angular/common"; export class ZipFilesOutput { @Input('zip-files-output') - zipFiles: TZipFileConfig[] = [] + zipFiles: Observable<TZipFileConfig[]> | TZipFileConfig[] = [] @Input('zip-files-output-zip-filename') zipFilename = 'archive.zip' - @HostListener('click') - async onClick(){ + private async zipArray(arrZipConfig: TZipFileConfig[]){ + const zip = new JSZip() - for (const zipFile of this.zipFiles) { + for (const zipFile of arrZipConfig) { const { filecontent, filename, base64 } = zipFile zip.file(filename, filecontent, { base64 }) } @@ -32,6 +34,21 @@ export class ZipFilesOutput { this.doc.body.removeChild(anchor) URL.revokeObjectURL(anchor.href) } + + @HostListener('click') + async onClick(){ + if (Array.isArray(this.zipFiles)) { + await this.zipArray(this.zipFiles) + return + } + if (isObservable(this.zipFiles)) { + const zipFiles = await this.zipFiles.pipe( + take(1) + ).toPromise() + await this.zipArray(zipFiles) + return + } + } constructor( @Inject(DOCUMENT) private doc: Document ){