No Contacts Found
\n\t\t\t\t\n\t\t\t\t\t\t\t{{ column.name }}\n\t\t\t\t\t\t | \n\t\t\t\t\t
---|
\n\t\t\t\t\t\t\t\t\t | \n\t\t\t\t\t\t\t
\n\t\t\t\t\t\t |
An Image Viewer window with this study is already open.
\n\t\t\t\t\t\n\t\t\t\t\t\tWould you like to view this study in the existing Window or open this study in a new Viewer Window?\n\t\t\t\t\t
\n\t\t\t\tAn Image Viewer window is already open.
\n\t\t\t\t\t\n\t\t\t\t\t\tWould you like to add this study to the last opened Window, close all Viewer Windows before opening, or open\n\t\t\t\t\t\tthis study in a new Viewer Window?\n\t\t\t\t\t
\n\t\t\t\tThe Keystone Omni desktop application is for Asteris customers only.
\n\t\t\tIf you are interested in learning more or becoming an Asteris customer, please contact us at sales@asteris.com.
\n\t\t\t`\n\t\t\tawait showConfirm(msg, { title: 'Omni Desktop', confirmText: null, cancelText: 'OK' })\n\t\t\treturn\n\t\t}\n\n\t\tclinicAPI.setClinicUrls(clinicUrls)\n\n\t\tdicomPartners.configs = response.data.partnerConfigs\n\n\t\tstorage.setItem('token', state.token)\n\t\tstorage.setItem('expiration', state.expiration)\n\t\tstorage.setItem('lastLogin', getters.isCommunityUser ? 'community' : 'omni')\n\t\t// Will get used on next load :/\n\t\tstorage.setItem('ga:tokenId', googleAnalyticsId)\n\n\t\t// Clear saved cache items on login\n\t\tsessionStorage.removeItem('temp:studies:filter')\n\t\tsessionStorage.removeItem('temp:teleconsultations:filter')\n\n\t\tstorage.setItem('apiDns', apiDns)\n\t\tstorage.setItem('annexDns', annexDns)\n\t\tapi.setApiBaseUrl(apiDns, annexDns)\n\n\t\tomniDesktop.updateConfig(annexDns, state.claims.activeClinicCode, API.defaults.baseURL)\n\t\tapi.auth.setHeader(token)\n\n\t\tif (state.claims.isConsultantUser) {\n\t\t\tlockService.startAutoRefreshing()\n\t\t}\n\t\tclinicAPI.startScanning()\n\t\tawait dispatch('initStatics')\n\t\tif (getters.isRepositoryUser) dispatch('startIdleTimer')\n\t\t// Tracking: Analytics and Sentry\n\t\tdispatch('SET_USER', {\n\t\t\tid: state.claims.userId,\n\t\t\tclinicId: state.claims.activeClinicId,\n\t\t\tclinicCode: state.claims.activeClinicCode,\n\t\t\tusername: state.claims.username,\n\t\t\tisCommunity: getters.isCommunityUser,\n\t\t\tisDesktop: omniDesktop.isConnected,\n\t\t})\n\t\twsData.start()\n\t\tRefreshToken.initializeRefreshToken()\n\t\tRefreshToken.enableRefreshToken()\n\t\tsetTimeout(() => {\n\t\t\tapi.user.getUserSetting('StudyList', 'EnableAntechNotification').then(response => {\n\t\t\t\tif (response.data !== null) {\n\t\t\t\t\tcommit('SET_ENABLE_ANTECH_NOTIFICATION', response.data)\n\t\t\t\t}\n\t\t\t})\n\t\t}, 100)\n\t\teventBus.post('login')\n\t},\n\tasync logIn({ commit, dispatch, getters }, { data }) {\n\t\tcommit('clearErrorNotifications')\n\t\tconst response = await api.auth.logIn(data)\n\t\tif (response.data.needsUpdate) {\n\t\t\tdispatch('showAppUpdateAvailable')\n\t\t}\n\t\tif (response.data.access_token) {\n\t\t\tstorage.clearTokenUses()\n\t\t\tstorage.addTokenUse()\n\t\t}\n\t\tawait dispatch('setAuthenticationAndInit', response)\n\t\ttry {\n\t\t\tGoogleAnalytics.sendGAEvent('LogIn', {\n\t\t\t\tevent_category: 'User',\n\t\t\t\tevent_label: 'LogIn',\n\t\t\t})\n\t\t} catch (err) {\n\t\t\tconsole.log(err)\n\t\t}\n\t},\n\tasync addConsultant({ state, dispatch }, { consultantId }) {\n\t\tconst userId = state.claims.userId\n\t\tif (!userId || !consultantId) return\n\t\tconst response = await api.user.addConsultantToUser({ userId, consultantId })\n\t\tconst consultantWasAdded = response && response.data === true\n\t\tif (consultantWasAdded) {\n\t\t\tawait dispatch('getConsultants')\n\t\t\tdispatch('addNotification', {\n\t\t\t\tmessage: 'A new consultant has been added to your account!',\n\t\t\t\tnotificationType: 'success',\n\t\t\t})\n\t\t}\n\t},\n\tasync createCommunityUser({ dispatch }, { data }) {\n\t\tdispatch('clearAllData')\n\t\tlet result = await api.user.getCommunityDns(data.consultantId)\n\t\tapi.setApiBaseUrl(result.communityDns, result.annexDns)\n\t\tconst response = await api.user.createCommunityUser(data)\n\t\tawait dispatch('setAuthenticationAndInit', response)\n\t\tanalyticsEvent('Global', 'CreateCommunityUser')\n\t},\n\tasync resetPassword({ dispatch }, data) {\n\t\tconst response = await api.auth.resetPassword(data)\n\t\tawait dispatch('setAuthenticationAndInit', response)\n\t},\n\tasync switchClinic({ dispatch, getters }, { id }) {\n\t\trouter.replace('/switching-clinics')\n\t\tconst response = await api.auth.switchClinic({ clinicId: id })\n\t\tawait dispatch('logOut', { changeRoute: false, forceLogout: true })\n\n\t\tawait dispatch('setAuthenticationAndInit', response)\n\t\tif (getters.isAuthenticated) {\n\t\t\tstorage.clearTokenUses()\n\t\t\tstorage.addTokenUse()\n\t\t}\n\t\tdispatch('applyTheme')\n\t\trouter.replace('/')\n\t},\n\tasync logOut({ dispatch }, { forceLogout = false, changeRoute = true } = {}) {\n\t\tif (!forceLogout && !(await checkSafeToLogout())) return\n\t\tdispatch('clearAllData')\n\t\tif (changeRoute) {\n\t\t\trouter.push('/')\n\t\t\tdlg.closeAll()\n\t\t}\n\n\t\teventBus.post(eventBus.type.LOGOUT)\n\t\teventBus.broadcast(eventBus.type.LOGOUT)\n\n\t\t// Disable token auto-refreshing\n\t\tif (_refreshInterval) clearInterval(_refreshInterval)\n\n\t\t// Close WebSockets\n\t\twsData.closeConnection()\n\t\tRefreshToken.stopRefreshToken()\n\t},\n}\n\nfunction getClaimsFromToken(token) {\n\tlet claims = {}\n\tif (!token) return claims\n\n\ttry {\n\t\tconst decoded = jwtDecode(token)\n\n\t\tclaims.userId = decoded['usid']\n\t\tclaims.defaultClinicId = decoded['uspmclid']\n\t\tclaims.defaultClinicCode = decoded['uspmclcd']\n\t\tclaims.permissionMask = decoded['uspmmsk']\n\t\tclaims.username = decoded['sub']\n\t\tclaims.fullname = `${decoded['usfn']} ${decoded['usln']}`\n\t\tclaims.partnerClinicCode = decoded['uspartner']\n\n\t\tclaims.activeClinicId = decoded['actclid']\n\t\tclaims.activeClinicCode = decoded['actclcd']\n\t\tclaims.sessionId = decoded['actssid']\n\n\t\tclaims.isRepositoryUser = decoded['isrpus']\n\t\tclaims.isConsultantUser = decoded['iscsus']\n\t\tclaims.isConsultantMember = decoded['iscsmb']\n\t\tclaims.isConsultantGroup = decoded['iscsgp']\n\t\tclaims.isConsultantIntern = decoded['iscsit']\n\t\tclaims.isConsultantResident = decoded['iscsrs']\n\t\tclaims.isConsultantTranscriptionist = decoded['iscstr']\n\t\tclaims.isCommunityUser = decoded['iscmus']\n\t\tclaims.isAdvancedReportsEnabled = decoded['isare']\n\t\tclaims.isSchedulingEnabled = decoded['isse']\n\n\t\t// Indicates whether the org/group should see UI around resident/intern workflow\n\t\tclaims.isConsultantApprovalSupported = decoded['iscapr']\n\t} catch (err) {}\n\n\treturn claims\n}\n\nasync function clearTempStorage() {\n\ttry {\n\t\tlet keys = Object.keys(localStorage)\n\t\tlet tempKeys = keys.filter(key => key.startsWith('temp:'))\n\t\ttempKeys.forEach(key => storage.removeItem(key))\n\t} catch (err) {\n\t\tconsole.log(err)\n\t}\n}\n\nexport default {\n\tstate,\n\tgetters,\n\tmutations,\n\tactions,\n}\n","export default {\n\tmprActive(state) {\n\t\treturn state.active\n\t},\n\tmprActiveSeries(state) {\n\t\treturn state.activeSeries\n\t},\n\tmprActiveView(state) {\n\t\treturn state.activeSeries && Object.values(state.activeSeries.viewData).find(v => v.active)\n\t},\n\tmprActiveViewKey(state) {\n\t\treturn (\n\t\t\tstate.activeSeries &&\n\t\t\tObject.entries(state.activeSeries.viewData).find(([i, v]) => v.active)[0]\n\t\t)\n\t},\n}\n","import * as cornerstone from 'cornerstone-core'\n\nimport { getImageData, validateMPRImages, loadImageData } from '@vtk/index.js'\nimport wait from '@/utils/wait'\nimport { showConfirm } from '@dialogs/ConfirmDlg.vue'\nimport shouldLoadDicom from '@/utils/shouldLoadDicom.js'\n\nexport default async function loadVolumeFromSeries({ commit, dispatch, state }, { series, completion }) {\n\tcommit('SET_MPR_LOADING', true)\n\tcommit('SET_MPR_LOAD_PROGRESS', 0)\n\n\t// let the loading stuff paint before locking the UI thread with VTK\n\tawait wait(400)\n\n\tconst validImages = validateMPRImages(series.images).images\n\tconst imageIds = validImages.map(i => i.imageId)\n\n\tconst useJpeg = !(shouldLoadDicom(validImages[0].modality) || shouldLoadDicom(series.modality))\n\n\t// TODO: track this in analytics somewhere?\n\t// const startProcess = new Date().getTime()\n\n\t// Get vtk image data for the series image data\n\ttry {\n\t\tconst imageData = getImageData(imageIds, series.seriesId, cornerstone.metaData.get, useJpeg)\n\n\t\tif (imageData.hasMissingSlices) {\n\t\t\tdispatch('addNotification', {\n\t\t\t\tmessage: `Warning:imagePositionPatient
has been used.`,\n\t\t\t\tnotificationType: 'warn',\n\t\t\t\tid: 'extraSlices',\n\t\t\t\tduration: 30000,\n\t\t\t})\n\t\t}\n\n\t\tlet renderFrequencyPct = 3\n\t\tif (imageIds.length > 500) {\n\t\t\trenderFrequencyPct = 7\n\t\t} else if (imageIds.length > 250) {\n\t\t\trenderFrequencyPct = 5\n\t\t}\n\n\t\t// Note: Modifies imageData object!\n\t\tloadImageData(imageData, validImages, series, renderFrequencyPct)\n\n\t\t// Let MPR load after the first loaded image has been processed\n\t\t// Enables Quick MPR loading & faster render, but still is loading the volume in the BG.\n\t\t// NOTE: to require MPR having the images loaded before opening the render view,\n\t\t// move this logic into the Promise.all section\n\t\tPromise.any(imageData.insertPixelDataPromises).then(num => {\n\t\t\tcommit('SET_MPR_VOLUME_DIRECTION', imageData.direction)\n\t\t\tcommit('SET_MPR_VOLUME_SLICE_DIRECTIONALITY', imageData.sliceDirectionality)\n\t\t\tcommit('SET_MPR_VOLUME_DATA', imageData.vtkImageData)\n\t\t\tcommit('SET_MPR_DEFAULT_WINDOW_LEVEL', imageData.voi)\n\t\t\tcommit('SET_MPR_VOLUME_IS_JPEG', useJpeg)\n\t\t\tcommit('UPDATE_AVAILABLE_TOOLS')\n\n\t\t\t// let the caller know we're ready to start image stuff\n\t\t\tcompletion()\n\n\t\t\treturn num\n\t\t})\n\n\t\tconst total = imageData.insertPixelDataPromises.length\n\t\tlet lastpercent = 0\n\t\timageData.insertPixelDataPromises.forEach(promise => {\n\t\t\ttry {\n\t\t\t\tpromise.then(num => {\n\t\t\t\t\tconst pct = Math.floor((num / total) * 100)\n\t\t\t\t\tif (pct > lastpercent + 1) {\n\t\t\t\t\t\tlastpercent = pct\n\t\t\t\t\t\tif (state.activeSeries) commit('SET_MPR_LOAD_PROGRESS', pct)\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t\t} catch (e) {}\n\t\t})\n\n\t\tPromise.all(imageData.insertPixelDataPromises).then(() => {\n\t\t\t// TODO: track this in analytics somewhere?\n\t\t\t// const duration = new Date().getTime() - startProcess\n\t\t\t// console.log(`Conversion Duration: ${duration}ms`)\n\t\t\tif (state.activeSeries) {\n\t\t\t\tcommit('SET_MPR_LOAD_PROGRESS', -1)\n\t\t\t\tcommit('SET_MPR_LOADING', false)\n\t\t\t}\n\t\t})\n\t} catch (e) {\n\t\tif (e instanceof RangeError) {\n\t\t\tconsole.error(e)\n\t\t\tshowConfirm(\n\t\t\t\t`Keystone Omni is not able to generate MPR Views when there are too many images in the series, or if the images are too big.
\n\t\t\t\tYour browser can handle around 900 images with a width and height of 512 pixels, or 200 images with width and height of 1024 pixels. The number of images may be smaller on a Mobile device or Tablet.
\n\t\t\t\tPlease try a smaller study.
`,\n\t\t\t\t{ title: 'Error Generating MPR Volume' }\n\t\t\t)\n\t\t} else {\n\t\t\tconsole.error(e)\n\t\t\tshowConfirm(\n\t\t\t\t`There was an issue generating the MPR volume.
\n\t\t\t\tIf you continue to see this error, please contact the Asteris Support Team.
`,\n\t\t\t\t{ title: 'Error Initializing MPR' }\n\t\t\t)\n\t\t}\n\t\t// Cleanup MPR related things\n\t\tcommit('SET_MPR_LOADING', false)\n\t\tcommit('SET_MPR_ACTIVE', false)\n\t}\n}\n","import api from '@services/api'\n\n/**\n * 1. Generates a base64 image for the active mpr view w/ annotations\n * 2. Kicks off an API call to add the annotated image to the study\n * 3. Dispatches success notification\n * NOTE: Manually updating ViewModel is not needed due to study synchronization\n *\n * @export\n * @param {*} { getters, commit, dispatch }, { reportId, showNotification }\n */\nexport default async function({ getters, commit, dispatch }, { reportId, showNotification = true } = {}) {\n\t// Get component for the active view\n\tconst activeComp = getters.mprActiveView.component\n\t// Render the viewer to a base64 image\n\tconst imgBase64 = await activeComp.renderImage()\n\n\tconst { clinicCode } = getters.activeImage.storageLocation || getters.activeSeries.storageLocation\n\tconst params = {\n\t\tclinicCode,\n\t\tsourceImageId: getters.activeImage.imageId,\n\t\timgBase64,\n\t\treportId,\n\t}\n\n\t// API call to add the annotated image to the study\n\tconst { series } = await api.viewer.renderAnnotatedImage(params)\n\n\t// Show notification\n\tif (showNotification)\n\t\tdispatch('addNotification', {\n\t\t\tmessage: 'New image with burned in annotations successfully created!',\n\t\t\tnotificationType: 'info',\n\t\t})\n\n\treturn series\n}\n","import api from '@services/api'\nimport saveAs from 'file-saver'\n\nexport default async function saveImage(imgBase64, fileName, dispatch) {\n\ttry {\n\t\t// Pure client-side (this will fail if `new Blob` is not supported which is only for very old browsers)\n\t\tsaveAs(imgBase64, fileName)\n\t} catch (e) {\n\t\t// Fallback to network request\n\t\tconst params = {\n\t\t\tfileName,\n\t\t\timgBase64,\n\t\t}\n\n\t\tconst { data } = await api.file.uploadTemporaryImage(params)\n\t\tif (data) {\n\t\t\tconst url = api.file.getDownloadTemporaryImageUrl({\n\t\t\t\tfileName: data,\n\t\t\t})\n\n\t\t\t// https://stackoverflow.com/a/2917/1867984\n\t\t\tconst win = window.open(url)\n\t\t\tif (!win || win.closed || typeof win.closed === 'undefined') {\n\t\t\t\tdispatch('addNotification', {\n\t\t\t\t\tmessage: 'Unable to save image. Pop-up was blocked.',\n\t\t\t\t\tnotificationType: 'warn',\n\t\t\t\t})\n\t\t\t}\n\t\t}\n\t}\n}\n","import saveImage from '@utils/saveImage'\n\nexport default async function saveMprImageAsync({ getters, dispatch }) {\n\t// Get component for the active view\n\tconst activeComp = getters.mprActiveView.component\n\n\t// Render the viewer to a base64 image\n\tconst imgBase64 = await activeComp.renderImage()\n\tconst fileName = `kso-${getters.activeImage.imageId}`\n\tawait saveImage(imgBase64, fileName, dispatch)\n}\n","let iframe = null\n\nexport default async function printImage(imgBase64, dispatch) {\n\tif (!iframe) {\n\t\tlet iframe = document.createElement('iframe')\n\t\tiframe.id = 'printf'\n\t\tiframe.name = 'printf'\n\t\tiframe.style.display = 'none'\n\t\tdocument.body.appendChild(iframe)\n\t}\n\tvar newWin = window.frames['printf']\n\tlet css = 'html,body {height: 100%;\tpadding: 0;\tmargin: 0; }\t@media print {img {\tmax-height: 100%;\tmax-width: 100%;}}'\n\tlet html = `\n\t\t\t\t\tYou may experience slower rendering speeds or poor quality rendering because of the\n\t\t\t\t\tfollowing reason(s):\n\t\t\t\t
\n\t\t\t\t\n\t\t\t\t\tThere are no performance impacting issues that we could detect with your system. You\n\t\t\t\t\tshould experience good MPR performance!\n\t\t\t\t
\n\t\t\t\n\t\t{{ message }}
\n\t\t\t\t\n\t\t\t\t\t\tAE Title:\n\t\t\t\t\t\t{{ service.dicomAeTitle }}\n\t\t\t\t\t
\n\t\t\t\t\t\n\t\t\t\t\t\tHostname:\n\t\t\t\t\t\t{{ service.hostName }}\n\t\t\t\t\t
\n\t\t\t\t\t\n\t\t\t\t\t\tPort:\n\t\t\t\t\t\t{{ service.dicomPort }}\n\t\t\t\t\t
\n\t\t\t\t\n\t\t\t\t\t\tLog Type\n\t\t\t\t\t
\n\t\t\t\t\t\n\t\t\t\t\n\t\t\t
1\" class=\"thumbnail-images\"> ({{ thumbnail.numberOfImages }} Images)
\n\t\t\t\t\n\t\t\t\tThe report for patient {{ info.description }} has been locked by you for over\n\t\t\t\t{{ info.minutesLocked }} minutes. This exceeds the threshold allowed for completion of this\n\t\t\t\tteleconsultation report. You can either complete the report within the next\n\t\t\t\t{{ info.repeatWarningMinutes }} minutes or you can release the report back to the group so\n\t\t\t\tit can be completed.\n\t\t\t
\n\t\t\t\n\t\t\t\tIf you do not respond in {{ secondsUntilRelease }} seconds, Keystone will automatically\n\t\t\t\trelease the report back to the group.\n\t\t\t
\n\t\t\t\n\t\t\t\tNote: If you select to complete this report, Keystone will remind you again in\n\t\t\t\t{{ info.repeatWarningMinutes }} minutes.\n\t\t\t
\n\t\t\t \n\t\t\n\t\t\t\t
This is an addendum.
' },\n\t\t\tDateTime: { sample: '2020-03-02T19:33:50.358Z' },\n\t\t},\n\t\tlayouts: ['Addendum'],\n\t},\n\tBillingCode: {\n\t\titems: {\n\t\t\tName: { sample: 'CODE' },\n\t\t\tValue: { sample: 'Billing Code Value' },\n\t\t},\n\t},\n\tCompletionDate: { sample: '2020-03-02T19:33:50.358Z' },\n\tDraftedBy: {\n\t\titems: {\n\t\t\tLabel: { sample: 'Intern:' },\n\t\t\tName: { sample: 'Bob Drafter' },\n\t\t},\n\t\tsets: ['Teleconsultation'],\n\t},\n\tConsultant: {\n\t\titems: {\n\t\t\tAddress: { sample: '123 Sesame St' },\n\t\t\tCity: { sample: 'New York' },\n\t\t\tState: { sample: 'NY' },\n\t\t\tEmail: { sample: 'example@domain.com' },\n\t\t\tName: { sample: 'Dr. Getwell' },\n\t\t\tOrganization: { sample: 'Sample Clinic' },\n\t\t\tPhone: { sample: '123-456-7890' },\n\t\t\tWebsite: { sample: 'https://www.google.com/' },\n\t\t},\n\t\tsets: ['Teleconsultation'],\n\t},\n\tFinalizedBy: {\n\t\titems: {\n\t\t\tFullName: { sample: 'Dr. Robert Staywell' },\n\t\t\tFirstName: { sample: 'Robert' },\n\t\t\tLastName: { sample: 'Staywell' },\n\n\t\t\tAddress: { sample: '456 Fake St' },\n\t\t\tCity: { sample: 'Beverly Hills' },\n\t\t\tState: { sample: 'CA' },\n\t\t\tZip: { sample: '90210' },\n\t\t\tCountry: { sample: 'United States' },\n\n\t\t\tEmail: { sample: 'example@domain.com' },\n\t\t\tOrganization: { sample: 'Sample Clinic' },\n\t\t\tPhone: { sample: '123-456-7890' },\n\n\t\t\tWebsite: { sample: 'domain.com' },\n\t\t\tTitle: { sample: 'Dr.' },\n\t\t\tDegree: { sample: 'Degree' },\n\t\t\tSuffix: { sample: 'Jr.' },\n\t\t},\n\t\tsets: ['Report'],\n\t},\n\tImageComment: {\n\t\titems: {\n\t\t\tAcquisitionDate: { sample: '2020-03-02T19:33:50.358Z' },\n\t\t\tAnatomy: { sample: 'Tarsus' },\n\t\t\tComment: { sample: 'This is a comment.' },\n\t\t\tEditDateTime: { sample: '2020-03-02T19:33:50.358Z' },\n\t\t\tLimb: { sample: 'Left Leg' },\n\t\t\tView: { sample: 'LEG_VIEW' },\n\t\t},\n\t\tlayouts: ['Image Comment'],\n\t},\n\tPatient: {\n\t\titems: {\n\t\t\tAge: { sample: '15 years' },\n\t\t\tBirthdate: { sample: '2020-03-02T00:00:00.000' },\n\t\t\tBreed: { sample: 'Lab' },\n\t\t\tGender: { sample: 'Male' },\n\t\t\tName: { sample: 'Spike' },\n\t\t\tOwner: { sample: 'Bob Smith' },\n\t\t\tPatientId: { sample: '123' },\n\t\t\tSpecies: { sample: 'Avian' },\n\t\t\tWeight: { sample: 13.3 },\n\t\t\tWeightUnit: { sample: 'LBS' },\n\t\t},\n\t},\n\tPDF: {\n\t\titems: {\n\t\t\tPageNumber: { sample: '1' },\n\t\t\tTotalPages: { sample: '1' },\n\t\t},\n\t\ttemplates: ['PDF Report', 'Standard PDF Report'],\n\t},\n\tReferredBy: {\n\t\titems: {\n\t\t\tAddress: { sample: '456 Fake St' },\n\t\t\tCity: { sample: 'Beverly Hills' },\n\t\t\tState: { sample: 'CA' },\n\t\t\tPostalCode: { sample: '90210' },\n\t\t\tEmail: { sample: 'example@domain.com' },\n\t\t\tName: { sample: 'Dr. Staywell' },\n\t\t\tOrganization: { sample: 'Sample Clinic' },\n\t\t\tPhone: { sample: '123-456-7890' },\n\t\t},\n\t},\n\tStudy: {\n\t\titems: {\n\t\t\tStudyDate: { sample: '2020-03-02T19:33:50.358' },\n\t\t\tModality: { sample: 'DX' },\n\t\t},\n\t},\n}\n\nexport function getPreviewContext(exclude?: string[]) {\n\tconst result = {}\n\tfor (const key in ContextMeta) {\n\t\tif (exclude && exclude.includes(key)) continue\n\t\tresult[key] = getMetaSample(ContextMeta[key])\n\t}\n\treturn result\n}\n\nfunction getMetaSample(metaItem) {\n\tif (metaItem.sample) return metaItem.sample\n\tif (metaItem.items) {\n\t\tconst items = {}\n\t\tfor (const key in metaItem.items) {\n\t\t\titems[key] = getMetaSample(metaItem.items[key])\n\t\t}\n\t\treturn items\n\t}\n}\n","module.exports = __webpack_public_path__ + \"img/vsp-background.3484078c.jpg\";","module.exports = \"\"","import api, { API, paramsSerializer, downloadFile } from '@services/api'\nimport { setLoading } from '@components/Loading.vue'\n\nimport { Layout, Template, Cell, Panel, TemplateSet, TemplateSetStatus, ReportDetail } from '@reporting/classes'\nimport { userData as user } from './userData'\nimport { Dbounce } from '@/utils/dbounce'\nimport { clinicAPI } from '@services/clinicAPI'\n\nexport enum ResidentResponseStatus {\n\tDraft = 0,\n\tComplete = 1,\n\tReviewed = 2,\n}\n\nclass ReportService {\n\tprevRoute = null\n\topenLayouts: Layout[] = []\n\tactiveCell: Cell = null // the selected cell\n\tselLayout: Layout = null\n\tselCell: Cell = null // the mouse over cell\n\tselTool: string = null\n\ttemplate: Template = null\n\tset: TemplateSet = null\n\tfieldMap: { [key: string]: IPartnerField[] } = {}\n\tformulas = ['SUM', 'AVG']\n\tborder = false\n\n\tundoStack: IStateSnapshot[] = []\n\tredoStack: IStateSnapshot[] = []\n\tsavedState: IStateSnapshot = null\n\n\t// drag info\n\tdragCell: Cell = null\n\n\t// cache last loaded meta\n\tmeta: IMeta\n\t_latestMetaVersion?: number\n\tvalidateDbounce: Dbounce\n\n\tclipboard: ICell = null\n\n\tconstructor() {\n\t\tthis.validateDbounce = new Dbounce(2000, this.validateTemplateSet.bind(this))\n\t}\n\n\tget pageHeader() {\n\t\treturn this.selLayout === this.template.root && this.template.pageHeader\n\t}\n\n\tget pageFooter() {\n\t\treturn this.selLayout === this.template.root && this.template.pageFooter\n\t}\n\n\tget selPanel(): Panel {\n\t\treturn this.selCell instanceof Panel ? this.selCell : undefined\n\t}\n\n\tget dragging(): boolean {\n\t\treturn this.dragCell != null\n\t}\n\n\tget stateSnapshot(): IStateSnapshot {\n\t\tif (!this.set) return null\n\t\tconst getLayoutObj = l => ({\n\t\t\ttemplateId: l.template && l.template.id,\n\t\t\tindex: l.template && l.template.layouts.indexOf(l),\n\t\t})\n\t\treturn {\n\t\t\tsetId: this.set.id,\n\t\t\tsetJson: JSON.stringify(this.set.toJSON()),\n\t\t\tsetStatus: this.set.status,\n\t\t\ttemplateId: this.template && this.template.id,\n\t\t\topenLayouts: this.openLayouts.map(getLayoutObj),\n\t\t\tselLayout: getLayoutObj(this.selLayout),\n\t\t\tactiveCell: this.activeCell && this.activeCell.name,\n\t\t}\n\t}\n\tget hasUnsavedChanges(): boolean {\n\t\tif (!this.savedState) return !!this.undoStack.length\n\t\treturn this.savedState.setJson !== this.stateSnapshot.setJson\n\t}\n\n\tcopy() {\n\t\tthis.clipboard = this.activeCell.toJSON()\n\t}\n\n\tget canPaste() {\n\t\treturn !!this.clipboard\n\t}\n\n\tpaste() {\n\t\tif (!this.canPaste) return\n\t\tlet panel = this.activeCell.isPanel ?tags with line breaks\n\thtml = html.replace(/
]*>/gi, '').replace(/<\\/p>/gi, '\\n\\n')\n\n\t// Handle unordered lists\n\thtml = html\n\t\t.replace(/
\n\t\t\t{{ client.name }}\n\t\t
\n\t\t\n\t\t\t{{ client.organization }}\n\t\t\t
\n\t\t\t\n\t\t\t\n\t\t\t\t
\n\t\t\t\t{{ cityStateZip }}\n\t\t\t\n\t\t
\n\t\t\t\n\t\t\t\t{{ client.phone }}\n\t\t\t\t
\n\t\t\t\n\t\t\t{{ client.email }}\n\t\t
{{ consultant.name }}
\n\t\t\n\t\t\t{{ consultant.organization }}\n\t\t\t
\n\t\t\t\n\t\t
\n\t\t\t\n\t\t\t\t{{ consultant.phone }}\n\t\t\t\t
\n\t\t\t\n\t\t\t\n\t\t\t\t{{ consultant.email }}\n\t\t\t\t
\n\t\t\t\n\t\t\t\n\t\t\t\t\n\t\t\t\t\t{{ consultant.website }}\n\t\t\t\t\n\t\t\t\t
\n\t\t\t\n\t\t
\n\t\t\t\n\t\t
\n\tOwner: | \n\t\t\t\t\t{{ patient.owner }} | \n\t\t\t\t
---|---|
Patient ID: | \n\t\t\t\t\t{{ patient.id }} | \n\t\t\t\t
Patient Name: | \n\t\t\t\t\t{{ patient.name }} | \n\t\t\t\t
Gender: | \n\t\t\t\t\t{{ patient.gender }} | \n\t\t\t\t
Birthdate: | \n\t\t\t\t\t\n\t\t\t\t\t\t{{ patient.birthdate | localizeDate({ showTime: false }) }}\n\t\t\t\t\t | \n\t\t\t\t
Species: | \n\t\t\t\t\t{{ patient.species }} | \n\t\t\t\t
Breed: | \n\t\t\t\t\t{{ patient.breed }} | \n\t\t\t\t
Weight: | \n\t\t\t\t\t\n\t\t\t\t\t\t\n\t\t\t\t\t\t\t{{ Number(patient.weight) | formatNumber }}\n\t\t\t\t\t\t\t{{ patient.weightUnit || 'KGS' }}\n\t\t\t\t\t\t\n\t\t\t\t\t | \n\t\t\t\t
\n\t\t\t\t\t\n\t\t\t\t\t\t{{ option.outOfOfficeMessage }}\n\t\t\t\t\t\n\t\t\t\t\tOut of office\n\t\t\t\t
\n\t\t\t\n\t\tThe following email addresses will be notified when the status of your report changes:
\n\t\t\n\t\t\t{{ report.emails.join(', ') }}\n\t\t
\n\t\tDam: {{ saleEntry.motherName }}
\n\t\tSire: {{ saleEntry.fatherName }}
\n\t\tConsigner: {{ saleEntry.consignerName }}
\n\tThe requested location was not found:
\n\t\t\t\t${to.path}`,\n\t\t\t\tnotificationType: 'error',\n\t\t\t})\n\t\t\t// if coming from an Omni route, cancel navigation. Otherwise, redirect to home\n\t\t\tnext(from && from.name ? false : '/')\n\t\t},\n\t},\n\t// optionally include icon tester page\n\t...(process.env.NODE_ENV !== 'production'\n\t\t? [\n\t\t\t\t{\n\t\t\t\t\tpath: '/icons',\n\t\t\t\t\tcomponent: () => import(/* webpackChunkName: \"IconTester\" */ '@/router/views/IconTester'),\n\t\t\t\t},\n\t\t ]\n\t\t: []),\n]\n","export * from \"-!../../node_modules/mini-css-extract-plugin/dist/loader.js??ref--8-oneOf-1-0!../../node_modules/css-loader/index.js??ref--8-oneOf-1-1!../../node_modules/vue-loader/lib/loaders/stylePostLoader.js!../../node_modules/postcss-loader/src/index.js??ref--8-oneOf-1-2!../../node_modules/sass-loader/lib/loader.js??ref--8-oneOf-1-3!../../node_modules/cache-loader/dist/cjs.js??ref--0-0!../../node_modules/vue-loader/lib/index.js??vue-loader-options!./OnBehalfOf.vue?vue&type=style&index=0&id=ff754bd6&prod&lang=scss&scoped=true\"","module.exports = \"\"","module.exports = \"\"","module.exports = \"\"","module.exports = \"\"","export * from \"-!../../../node_modules/mini-css-extract-plugin/dist/loader.js??ref--8-oneOf-1-0!../../../node_modules/css-loader/index.js??ref--8-oneOf-1-1!../../../node_modules/vue-loader/lib/loaders/stylePostLoader.js!../../../node_modules/postcss-loader/src/index.js??ref--8-oneOf-1-2!../../../node_modules/sass-loader/lib/loader.js??ref--8-oneOf-1-3!../../../node_modules/cache-loader/dist/cjs.js??ref--0-0!../../../node_modules/vue-loader/lib/index.js??vue-loader-options!./PatientInfo.vue?vue&type=style&index=0&id=5b862bbf&prod&lang=scss\"","import { API } from '@services/api'\nimport { showAlert } from '@dialogs/MessageDlg.vue'\nimport { eventBus } from './eventBus'\n\nconst DICOM_STATUS_TIME = 1000 * 2\nconst MAX_UPDATES = 500\n\nclass DicomSendItem {\n\tconstructor(item) {\n\t\tthis.queueItemId = item.queueItemId\n\t\tthis.status = item.status\n\t\tthis.imagesSent = item.imagesSent\n\t\tthis.totalImages = item.totalImages\n\t\tthis.dateQueued = item.dateQueued\n\t\tthis.logs = item.logs\n\t\tthis.studyId = item.studyId\n\t\tthis.studyDate = item.studyDate\n\t\tthis.patientId = item.patientId\n\t\tthis.patientName = item.patientName\n\t\tthis.modality = item.modality\n\t}\n\n\tget isError() {\n\t\treturn this.status === 4 || this.status === 5\n\t}\n\n\tget isSuccess() {\n\t\treturn this.status === 3\n\t}\n\n\tget canCancel() {\n\t\treturn this.status <= 2\n\t}\n\n\tget statusName() {\n\t\tswitch (this.status) {\n\t\t\tcase 1:\n\t\t\t\treturn 'Queued'\n\t\t\tcase 2:\n\t\t\t\treturn 'Sending'\n\t\t\tcase 3:\n\t\t\t\treturn 'Successful'\n\t\t\tcase 4:\n\t\t\t\treturn 'Error'\n\t\t\tcase 5:\n\t\t\t\treturn 'Cancelled'\n\t\t}\n\t}\n}\n\nclass DicomSend {\n\tconstructor() {\n\t\tthis.items = [] // DicomSendItem[]\n\t\tthis.interval = 0\n\n\t\teventBus.on('login', this.getUserItems.bind(this))\n\t\tthis.updateCount = 0\n\t\tthis.forceUpdates = false\n\t}\n\n\tget studyItems() {\n\t\treturn this.items.filter(i => i.studyId)\n\t}\n\n\tget sentCount() {\n\t\tlet count = 0\n\t\tthis.items.forEach(s => {\n\t\t\tcount += s.imagesSent\n\t\t})\n\t\treturn count\n\t}\n\n\tget pendingCount() {\n\t\treturn this.items.filter(i => i.status <= 2).length\n\t}\n\n\tget totalImages() {\n\t\tlet count = 0\n\t\tthis.items.forEach(s => {\n\t\t\tcount += s.totalImages\n\t\t})\n\t\treturn count\n\t}\n\n\tenabledUpdates(force = false) {\n\t\tthis.forceUpdates = force\n\n\t\tif (this.pendingCount > 0 && this.interval === 0) {\n\t\t\tthis.interval = setInterval(this.updateList.bind(this), DICOM_STATUS_TIME)\n\t\t}\n\t\tthis.updateCount = 0\n\t}\n\n\tasync updateList() {\n\t\tthis.updateCount++\n\n\t\tfor (let i = this.items.length - 1; i >= 0; i--) {\n\t\t\tlet item = this.items[i]\n\t\t\tif (item.status > 2) continue\n\n\t\t\tlet info = await this.getSendItemInfo(item.queueItemId)\n\t\t\tif (info) {\n\t\t\t\tObject.assign(item, info)\n\t\t\t} else {\n\t\t\t\tthis.items.splice(i, 1)\n\t\t\t}\n\t\t}\n\n\t\tlet stopUpdates = this.pendingCount === 0\n\t\tif (this.updateCount > MAX_UPDATES && !this.forceUpdates) {\n\t\t\tstopUpdates = true\n\t\t}\n\n\t\tif (stopUpdates) {\n\t\t\tclearInterval(this.interval)\n\t\t\tthis.interval = 0\n\t\t\tthis.updateCount = 0\n\t\t}\n\t}\n\n\tvalidateDevice(device) {\n\t\tfunction validateAeTitle(str) {\n\t\t\treturn /^[^\\\\]{1,16}$/.test(str)\n\t\t}\n\n\t\tlet errors = []\n\n\t\tif (device.description !== undefined && !device.description) {\n\t\t\terrors.push('Required Description')\n\t\t}\n\n\t\tif (device.aeTitle) {\n\t\t\tif (!validateAeTitle(device.aeTitle)) {\n\t\t\t\terrors.push('AE Title must be 1-16 characters without backslashes')\n\t\t\t}\n\t\t} else {\n\t\t\terrors.push('Required AE Title')\n\t\t}\n\n\t\tif (device.userAeTitle && !validateAeTitle(device.userAeTitle)) {\n\t\t\terrors.push('User AE Title must be 1-16 characters without backslashes')\n\t\t}\n\n\t\tif (!device.port > 0) {\n\t\t\terrors.push('Required Hostname')\n\t\t}\n\n\t\tif (!device.hostname) {\n\t\t\terrors.push('Required Port number')\n\t\t}\n\n\t\tif (errors.length > 0) {\n\t\t\tshowAlert(errors.join('. '))\n\t\t}\n\t\treturn errors.length === 0\n\t}\n\n\tgetDevices() {\n\t\treturn API.get(`/dicomsend/devices`).then(r => r.data)\n\t}\n\n\tsaveDevice(device) {\n\t\treturn API.post(`/dicomsend/device`, device)\n\t}\n\n\tdeleteDevice(device) {\n\t\treturn API.delete(`/dicomsend/device/${device.id}`)\n\t}\n\n\tsendImages(device, studyId, series, images) {\n\t\treturn API.post(`/dicomsend/send-images`, { device, studyId, series, images }).then(r => {\n\t\t\tthis.getUserItems()\n\t\t\treturn r.data\n\t\t})\n\t}\n\n\tsendEcho(device) {\n\t\treturn API.post(`/dicomsend/echo`, device).then(r => {\n\t\t\tthis.getUserItems()\n\t\t\treturn r.data.queueItemId\n\t\t})\n\t}\n\n\tgetSendItemInfo(queueItemId) {\n\t\treturn API.get(`/dicomsend/info/${queueItemId}`).then(r => r.data)\n\t}\n\n\tgetUserItems() {\n\t\treturn API.get(`/dicomsend/user-items`).then(r => {\n\t\t\tthis.items = []\n\t\t\tr.data.forEach(i => {\n\t\t\t\tthis.items.push(new DicomSendItem(i))\n\t\t\t})\n\n\t\t\tthis.enabledUpdates()\n\t\t\treturn this.items\n\t\t})\n\t}\n\n\tcancelItem(queueItemId) {\n\t\treturn API.get(`/dicomsend/cancel/${queueItemId}`).then(r => this.updateList())\n\t}\n}\n\nexport const dicomSend = new DicomSend()\nwindow.dicomSend = dicomSend\n","module.exports = \"\"","module.exports = \"\"","module.exports = \"\"","var render = function () {var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;return _c('div',[(_vm.cell.widgets.length === 0 && _vm.cell.editorMode)?_c('div',{staticClass:\"editor-placeholder\"},[_c('svg-icon',{staticClass:\"is-warning\",attrs:{\"icon\":\"exclamation-triangle\",\"inline\":\"\",\"fixed\":\"\"}}),_vm._v(\"\\n\\t\\tNo forms are available for this selector. Use the\\n\\t\\t\"),_c('strong',{staticStyle:{\"font-size\":\"0.9em\"}},[_vm._v(\"Add New Form\")]),_vm._v(\" button in the properties pane to add\\n\\t\\tone.\\n\\t\")],1):(_vm.canSelect)?_c('div',{staticStyle:{\"margin-bottom\":\"16px\"}},[_c('div',{staticStyle:{\"font-style\":\"italic\"}},[_vm._v(\"\\n\\t\\t\\tPlease select a response type.\\n\\t\\t\")]),(_vm.canSelect)?_c('v-select',{staticStyle:{\"width\":\"auto\",\"margin\":\"8px 0\"},attrs:{\"options\":_vm.cell.widgets,\"clearable\":false,\"label\":\"name\",\"reduce\":function (o) { return o.name; },\"tabindex\":_vm.cell.template.editorMode ? -1 : 0},nativeOn:{\"keydown\":function($event){if(!$event.type.indexOf('key')&&_vm._k($event.keyCode,\"enter\",13,$event.key,\"Enter\")){ return null; }$event.preventDefault();return _vm.setSubLayout($event)}},model:{value:(_vm.selectedTemplate),callback:function ($$v) {_vm.selectedTemplate=$$v},expression:\"selectedTemplate\"}}):_vm._e(),_c('button',{staticClass:\"btn btn-primary\",attrs:{\"disabled\":!_vm.selectedTemplate},on:{\"click\":function($event){return _vm.setSubLayout()}}},[_vm._v(\"\\n\\t\\t\\tUse the selected type\\n\\t\\t\")])],1):(_vm.canShowResponse && _vm.cell.subLayouts.length)?_c('div',_vm._l((_vm.cell.subLayouts),function(layout,index){return _c('layout',{key:index,attrs:{\"layout\":layout}})}),1):_c('div',{staticStyle:{\"font-style\":\"italic\"}},[_vm._v(\"\\n\\t\\tNo response has been received yet.\\n\\t\")])])}\nvar staticRenderFns = []\n\nexport { render, staticRenderFns }","\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nimport { Cell } from '../classes'\nimport { showConfirm } from '@dialogs/ConfirmDlg.vue'\n\nexport default {\n\tname: 'ResponseSelector',\n\tprops: ['cell'],\n\tdata() {\n\t\treturn {\n\t\t\tselectedTemplate: null,\n\t\t}\n\t},\n\tcomputed: {\n\t\tcanShowResponse() {\n\t\t\tif (this.cell.template.editorMode) return true\n\t\t\telse return this.cell.report.canViewResponse || this.cell.report.canRespond\n\t\t},\n\t\tcanSelect() {\n\t\t\tif (this.cell.value) return false\n\t\t\tlet canRespond = this.cell.report && this.cell.report.canRespond\n\t\t\treturn !this.cell.value && (canRespond || this.cell.editorMode)\n\t\t},\n\t},\n\twatch: {\n\t\t'cell.report.canRespond': {\n\t\t\thandler(canRespond) {\n\t\t\t\tthis.cell.props.required = canRespond\n\t\t\t\tthis.cell.subLayouts.forEach(l => {\n\t\t\t\t\tl.readOnly = !canRespond\n\t\t\t\t})\n\t\t\t},\n\t\t\timmediate: true,\n\t\t},\n\t\t'cell.widgets': {\n\t\t\thandler(widgets) {\n\t\t\t\tif (widgets.length === 1) this.cell.value = widgets[0].name\n\t\t\t\telse if (this.cell.editorMode) {\n\t\t\t\t\tthis.cell.value = null\n\t\t\t\t\tthis.cell.subLayouts = []\n\t\t\t\t}\n\t\t\t\tthis.setSubLayout()\n\t\t\t},\n\t\t\timmediate: true,\n\t\t},\n\t},\n\tcreated() {\n\t\tthis.setSubLayout()\n\t},\n\tmethods: {\n\t\tasync setSubLayout() {\n\t\t\tif (!this.selectedTemplate && !this.cell.value) return\n\t\t\tif (!this.cell.value) {\n\t\t\t\tthis.cell.value = this.selectedTemplate\n\t\t\t}\n\n\t\t\tlet cell = this.cell\n\n\t\t\tif (\n\t\t\t\tcell.subLayout &&\n\t\t\t\tcell.subLayout.name !== cell.value &&\n\t\t\t\t!cell.editorMode &&\n\t\t\t\tcell.widgets.length > 1\n\t\t\t) {\n\t\t\t\tlet msg = 'Are you sure you want to change the response type?'\n\t\t\t\tlet confirmed = await showConfirm(msg)\n\t\t\t\tif (!confirmed) {\n\t\t\t\t\tcell.value = cell.subLayout.name // restore previous selector value\n\t\t\t\t\treturn\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tlet report = cell.report\n\t\t\tlet layout = cell.layout.template.getLayout(cell.value)\n\t\t\tif (layout) {\n\t\t\t\tcell.subLayouts = []\n\t\t\t\tlet sub = layout.createInstance()\n\t\t\t\tsub.readOnly = report && !report.canRespond\n\t\t\t\tsub.forcePath = layout.name\n\t\t\t\tcell.addSubLayout(sub)\n\t\t\t\tif (cell.report) cell.report.validate()\n\t\t\t}\n\t\t},\n\t},\n}\n","import mod from \"-!../../../node_modules/cache-loader/dist/cjs.js??ref--14-0!../../../node_modules/thread-loader/dist/cjs.js!../../../node_modules/babel-loader/lib/index.js!../../../node_modules/ts-loader/index.js??ref--14-3!../../../node_modules/cache-loader/dist/cjs.js??ref--0-0!../../../node_modules/vue-loader/lib/index.js??vue-loader-options!./ResponseSelector.vue?vue&type=script&lang=ts\"; export default mod; export * from \"-!../../../node_modules/cache-loader/dist/cjs.js??ref--14-0!../../../node_modules/thread-loader/dist/cjs.js!../../../node_modules/babel-loader/lib/index.js!../../../node_modules/ts-loader/index.js??ref--14-3!../../../node_modules/cache-loader/dist/cjs.js??ref--0-0!../../../node_modules/vue-loader/lib/index.js??vue-loader-options!./ResponseSelector.vue?vue&type=script&lang=ts\"","import { render, staticRenderFns } from \"./ResponseSelector.vue?vue&type=template&id=11581983\"\nimport script from \"./ResponseSelector.vue?vue&type=script&lang=ts\"\nexport * from \"./ResponseSelector.vue?vue&type=script&lang=ts\"\n\n\n/* normalize component */\nimport normalizer from \"!../../../node_modules/vue-loader/lib/runtime/componentNormalizer.js\"\nvar component = normalizer(\n script,\n render,\n staticRenderFns,\n false,\n null,\n null,\n null\n \n)\n\nexport default component.exports","var render = function () {var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;return _c('transition',{attrs:{\"name\":\"loading-fade\",\"appear\":\"\"}},[(_vm.isLoading)?_c('div',{class:[\"loading-spinner\", { 'is-small': _vm.isSmall }],style:(_vm.alignStyle)},[_c('svg-icon',{attrs:{\"icon\":\"spinner\",\"pulse\":\"\",\"fixed\":\"\"}}),_c('span',[_vm._v(\"Loading...\")])],1):_vm._e()])}\nvar staticRenderFns = []\n\nexport { render, staticRenderFns }","\n\t{{ study.patientId }} - {{ study.attachmentDescription || study.patientName }}
\n\t\t\t\t\t\t\n\t\t\t\t\t\t\tSource:\n\t\t\t\t\t\t\t{{ study.source }}\n\t\t\t\t\t\t
\n\t\t\t\t\t\t\n\t\t\t\t\t\t\t{{ study.studyDateTime | localizeDate }}\n\t\t\t\t\t\t
\n\t\t\t\t\t