Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c2b3f1c540 | |||
|
|
8e81e78fb2 | ||
|
|
9239da35f8 |
@@ -119,6 +119,7 @@ export default class RetrieveMetadataLoaderAsync extends RetrieveMetadataLoader
|
|||||||
async preLoad() {
|
async preLoad() {
|
||||||
const preLoaders = this.getPreLoaders();
|
const preLoaders = this.getPreLoaders();
|
||||||
const result = await this.runLoaders(preLoaders);
|
const result = await this.runLoaders(preLoaders);
|
||||||
|
console.log('Raw QIDO series result:', result); // Debug: Check if '0008103E' is present
|
||||||
const sortCriteria = this.sortCriteria;
|
const sortCriteria = this.sortCriteria;
|
||||||
const sortFunction = this.sortFunction;
|
const sortFunction = this.sortFunction;
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { utils, classes } from '@ohif/core';
|
import { utils, classes, DicomMetadataStore } from '@ohif/core';
|
||||||
import { id } from './id';
|
import { id } from './id';
|
||||||
import getDisplaySetMessages from './getDisplaySetMessages';
|
import getDisplaySetMessages from './getDisplaySetMessages';
|
||||||
import getDisplaySetsFromUnsupportedSeries from './getDisplaySetsFromUnsupportedSeries';
|
import getDisplaySetsFromUnsupportedSeries from './getDisplaySetsFromUnsupportedSeries';
|
||||||
@@ -75,6 +75,10 @@ const makeDisplaySet = instances => {
|
|||||||
? DYNAMIC_VOLUME_LOADER_SCHEME
|
? DYNAMIC_VOLUME_LOADER_SCHEME
|
||||||
: DEFAULT_VOLUME_LOADER_SCHEME;
|
: DEFAULT_VOLUME_LOADER_SCHEME;
|
||||||
|
|
||||||
|
// Get series metadata from store
|
||||||
|
const study = DicomMetadataStore.getStudy(instance.StudyInstanceUID);
|
||||||
|
const series = study?.series?.find(s => s.SeriesInstanceUID === instance.SeriesInstanceUID);
|
||||||
|
|
||||||
// set appropriate attributes to image set...
|
// set appropriate attributes to image set...
|
||||||
const messages = getDisplaySetMessages(instances, isReconstructable, isDynamicVolume);
|
const messages = getDisplaySetMessages(instances, isReconstructable, isDynamicVolume);
|
||||||
|
|
||||||
@@ -88,7 +92,7 @@ const makeDisplaySet = instances => {
|
|||||||
SeriesNumber: instance.SeriesNumber || 0,
|
SeriesNumber: instance.SeriesNumber || 0,
|
||||||
FrameRate: instance.FrameTime,
|
FrameRate: instance.FrameTime,
|
||||||
SOPClassUID: instance.SOPClassUID,
|
SOPClassUID: instance.SOPClassUID,
|
||||||
SeriesDescription: instance.SeriesDescription || '',
|
SeriesDescription: series?.SeriesDescription || instance.SeriesDescription || '',
|
||||||
Modality: instance.Modality,
|
Modality: instance.Modality,
|
||||||
isMultiFrame: isMultiFrame(instance),
|
isMultiFrame: isMultiFrame(instance),
|
||||||
countIcon: isReconstructable ? 'icon-mpr' : undefined,
|
countIcon: isReconstructable ? 'icon-mpr' : undefined,
|
||||||
|
|||||||
407
platform/app/public/config/default_ayani.js
Normal file
407
platform/app/public/config/default_ayani.js
Normal file
@@ -0,0 +1,407 @@
|
|||||||
|
/** @type {AppTypes.Config} */
|
||||||
|
|
||||||
|
window.config = {
|
||||||
|
routerBasename: '/',
|
||||||
|
// whiteLabeling: {},
|
||||||
|
extensions: [],
|
||||||
|
modes: [],
|
||||||
|
customizationService: {},
|
||||||
|
showStudyList: true,
|
||||||
|
// some windows systems have issues with more than 3 web workers
|
||||||
|
maxNumberOfWebWorkers: 3,
|
||||||
|
// below flag is for performance reasons, but it might not work for all servers
|
||||||
|
showWarningMessageForCrossOrigin: true,
|
||||||
|
showCPUFallbackMessage: true,
|
||||||
|
showLoadingIndicator: true,
|
||||||
|
experimentalStudyBrowserSort: false,
|
||||||
|
strictZSpacingForVolumeViewport: true,
|
||||||
|
groupEnabledModesFirst: true,
|
||||||
|
maxNumRequests: {
|
||||||
|
interaction: 100,
|
||||||
|
thumbnail: 75,
|
||||||
|
// Prefetch number is dependent on the http protocol. For http 2 or
|
||||||
|
// above, the number of requests can be go a lot higher.
|
||||||
|
prefetch: 25,
|
||||||
|
},
|
||||||
|
expertise: false, //* Tambahan untuk enable expertise (CustomizableViewportOverlay)
|
||||||
|
expertise_host: `http://192.168.1.90`, // IP ke NV di PACS Server, untuk fetch expertise bawaan versi NV
|
||||||
|
pacs_document_host: `192.168.1.90`, // IP ke NV di PACS Server untuk ambil pdf
|
||||||
|
pacs_document_port: 8080,
|
||||||
|
defaultDataSourceName: 'local-proxy',
|
||||||
|
dataSources: [
|
||||||
|
{
|
||||||
|
namespace: '@ohif/extension-default.dataSourcesModule.dicomweb',
|
||||||
|
sourceName: 'local-proxy',
|
||||||
|
configuration: {
|
||||||
|
friendlyName: 'Static WADO Local Data',
|
||||||
|
name: 'DCM4CHEE',
|
||||||
|
qidoRoot: `http://192.168.1.90:5000/rs`, // IP ke dicomweb-proxy PACS Server. URI selalu /rs
|
||||||
|
wadoRoot: `http://192.168.1.90:5000/rs`, // IP ke dicomweb-proxy PACS Server. URI selalu /rs qidoSupportsIncludeField: false,
|
||||||
|
supportsReject: true,
|
||||||
|
supportsStow: true,
|
||||||
|
imageRendering: 'wadors',
|
||||||
|
thumbnailRendering: 'wadors',
|
||||||
|
enableStudyLazyLoad: true,
|
||||||
|
supportsFuzzyMatching: false,
|
||||||
|
supportsWildcard: true,
|
||||||
|
staticWado: true,
|
||||||
|
singlepart: 'video',
|
||||||
|
bulkDataURI: {
|
||||||
|
enabled: true,
|
||||||
|
relativeResolution: 'studies',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
namespace: '@ohif/extension-default.dataSourcesModule.dicomwebproxy',
|
||||||
|
sourceName: 'dicomwebproxy',
|
||||||
|
configuration: {
|
||||||
|
friendlyName: 'dicomweb delegating proxy',
|
||||||
|
name: 'dicomwebproxy',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
namespace: '@ohif/extension-default.dataSourcesModule.dicomjson',
|
||||||
|
sourceName: 'dicomjson',
|
||||||
|
configuration: {
|
||||||
|
friendlyName: 'dicom json',
|
||||||
|
name: 'json',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
namespace: '@ohif/extension-default.dataSourcesModule.dicomlocal',
|
||||||
|
sourceName: 'dicomlocal',
|
||||||
|
configuration: {
|
||||||
|
friendlyName: 'dicom local',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
httpErrorHandler: error => {
|
||||||
|
// This is 429 when rejected from the public idc sandbox too often.
|
||||||
|
console.warn(error.status);
|
||||||
|
|
||||||
|
// Could use services manager here to bring up a dialog/modal if needed.
|
||||||
|
console.warn('test, navigate to https://ohif.org/');
|
||||||
|
},
|
||||||
|
// whiteLabeling: {
|
||||||
|
// /* Optional: Should return a React component to be rendered in the "Logo" section of the application's Top Navigation bar */
|
||||||
|
// createLogoComponentFn: function (React) {
|
||||||
|
// return React.createElement(
|
||||||
|
// 'a',
|
||||||
|
// {
|
||||||
|
// target: '_self',
|
||||||
|
// rel: 'noopener noreferrer',
|
||||||
|
// className: 'text-purple-600 line-through',
|
||||||
|
// href: '/',
|
||||||
|
// },
|
||||||
|
// React.createElement('img',
|
||||||
|
// {
|
||||||
|
// src: './assets/customLogo.svg',
|
||||||
|
// className: 'w-8 h-8',
|
||||||
|
// }
|
||||||
|
// ))
|
||||||
|
// },
|
||||||
|
// },
|
||||||
|
hotkeys: [
|
||||||
|
{
|
||||||
|
commandName: 'incrementActiveViewport',
|
||||||
|
label: 'Next Viewport',
|
||||||
|
keys: ['right'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'decrementActiveViewport',
|
||||||
|
label: 'Previous Viewport',
|
||||||
|
keys: ['left'],
|
||||||
|
},
|
||||||
|
{ commandName: 'rotateViewportCW', label: 'Rotate Right', keys: ['r'] },
|
||||||
|
{ commandName: 'rotateViewportCCW', label: 'Rotate Left', keys: ['l'] },
|
||||||
|
{ commandName: 'invertViewport', label: 'Invert', keys: ['i'] },
|
||||||
|
{
|
||||||
|
commandName: 'flipViewportHorizontal',
|
||||||
|
label: 'Flip Horizontally',
|
||||||
|
keys: ['h'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'flipViewportVertical',
|
||||||
|
label: 'Flip Vertically',
|
||||||
|
keys: ['v'],
|
||||||
|
},
|
||||||
|
{ commandName: 'scaleUpViewport', label: 'Zoom In', keys: ['+'] },
|
||||||
|
{ commandName: 'scaleDownViewport', label: 'Zoom Out', keys: ['-'] },
|
||||||
|
{ commandName: 'fitViewportToWindow', label: 'Zoom to Fit', keys: ['='] },
|
||||||
|
{ commandName: 'resetViewport', label: 'Reset', keys: ['space'] },
|
||||||
|
{ commandName: 'nextImage', label: 'Next Image', keys: ['down'] },
|
||||||
|
{ commandName: 'previousImage', label: 'Previous Image', keys: ['up'] },
|
||||||
|
// {
|
||||||
|
// commandName: 'previousViewportDisplaySet',
|
||||||
|
// label: 'Previous Series',
|
||||||
|
// keys: ['pagedown'],
|
||||||
|
// },
|
||||||
|
// {
|
||||||
|
// commandName: 'nextViewportDisplaySet',
|
||||||
|
// label: 'Next Series',
|
||||||
|
// keys: ['pageup'],
|
||||||
|
// },
|
||||||
|
{
|
||||||
|
commandName: 'setToolActive',
|
||||||
|
commandOptions: { toolName: 'Zoom' },
|
||||||
|
label: 'Zoom',
|
||||||
|
keys: ['z'],
|
||||||
|
},
|
||||||
|
// ~ Window level presets
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset1',
|
||||||
|
label: 'W/L Preset 1',
|
||||||
|
keys: ['1'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset2',
|
||||||
|
label: 'W/L Preset 2',
|
||||||
|
keys: ['2'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset3',
|
||||||
|
label: 'W/L Preset 3',
|
||||||
|
keys: ['3'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset4',
|
||||||
|
label: 'W/L Preset 4',
|
||||||
|
keys: ['4'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset5',
|
||||||
|
label: 'W/L Preset 5',
|
||||||
|
keys: ['5'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset6',
|
||||||
|
label: 'W/L Preset 6',
|
||||||
|
keys: ['6'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset7',
|
||||||
|
label: 'W/L Preset 7',
|
||||||
|
keys: ['7'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset8',
|
||||||
|
label: 'W/L Preset 8',
|
||||||
|
keys: ['8'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
commandName: 'windowLevelPreset9',
|
||||||
|
label: 'W/L Preset 9',
|
||||||
|
keys: ['9'],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
tours: [
|
||||||
|
{
|
||||||
|
id: 'basicViewerTour',
|
||||||
|
route: '/viewer',
|
||||||
|
steps: [
|
||||||
|
{
|
||||||
|
id: 'scroll',
|
||||||
|
title: 'Scrolling Through Images',
|
||||||
|
text: 'You can scroll through the images using the mouse wheel or scrollbar.',
|
||||||
|
attachTo: {
|
||||||
|
element: '.viewport-element',
|
||||||
|
on: 'top',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '.cornerstone-viewport-element',
|
||||||
|
event: 'CORNERSTONE_TOOLS_MOUSE_WHEEL',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('.viewport-element'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'zoom',
|
||||||
|
title: 'Zooming In and Out',
|
||||||
|
text: 'You can zoom the images using the right click.',
|
||||||
|
attachTo: {
|
||||||
|
element: '.viewport-element',
|
||||||
|
on: 'left',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '.cornerstone-viewport-element',
|
||||||
|
event: 'CORNERSTONE_TOOLS_MOUSE_UP',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('.viewport-element'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'pan',
|
||||||
|
title: 'Panning the Image',
|
||||||
|
text: 'You can pan the images using the middle click.',
|
||||||
|
attachTo: {
|
||||||
|
element: '.viewport-element',
|
||||||
|
on: 'top',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '.cornerstone-viewport-element',
|
||||||
|
event: 'CORNERSTONE_TOOLS_MOUSE_UP',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('.viewport-element'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'windowing',
|
||||||
|
title: 'Adjusting Window Level',
|
||||||
|
text: 'You can modify the window level using the left click.',
|
||||||
|
attachTo: {
|
||||||
|
element: '.viewport-element',
|
||||||
|
on: 'left',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '.cornerstone-viewport-element',
|
||||||
|
event: 'CORNERSTONE_TOOLS_MOUSE_UP',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('.viewport-element'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'length',
|
||||||
|
title: 'Using the Measurement Tools',
|
||||||
|
text: 'You can measure the length of a region using the Length tool.',
|
||||||
|
attachTo: {
|
||||||
|
element: '[data-cy="MeasurementTools-split-button-primary"]',
|
||||||
|
on: 'bottom',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '[data-cy="MeasurementTools-split-button-primary"]',
|
||||||
|
event: 'click',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () =>
|
||||||
|
waitForElement('[data-cy="MeasurementTools-split-button-primary]'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'drawAnnotation',
|
||||||
|
title: 'Drawing Length Annotations',
|
||||||
|
text: 'Use the length tool on the viewport to measure the length of a region.',
|
||||||
|
attachTo: {
|
||||||
|
element: '.viewport-element',
|
||||||
|
on: 'right',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: 'body',
|
||||||
|
event: 'event::measurement_added',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('.viewport-element'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'trackMeasurement',
|
||||||
|
title: 'Tracking Measurements in the Panel',
|
||||||
|
text: 'Click yes to track the measurements in the measurement panel.',
|
||||||
|
attachTo: {
|
||||||
|
element: '[data-cy="prompt-begin-tracking-yes-btn"]',
|
||||||
|
on: 'bottom',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '[data-cy="prompt-begin-tracking-yes-btn"]',
|
||||||
|
event: 'click',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('[data-cy="prompt-begin-tracking-yes-btn"]'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'openMeasurementPanel',
|
||||||
|
title: 'Opening the Measurements Panel',
|
||||||
|
text: 'Click the measurements button to open the measurements panel.',
|
||||||
|
attachTo: {
|
||||||
|
element: '#trackedMeasurements-btn',
|
||||||
|
on: 'left-start',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '#trackedMeasurements-btn',
|
||||||
|
event: 'click',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('#trackedMeasurements-btn'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'scrollAwayFromMeasurement',
|
||||||
|
title: 'Scrolling Away from a Measurement',
|
||||||
|
text: 'Scroll the images using the mouse wheel away from the measurement.',
|
||||||
|
attachTo: {
|
||||||
|
element: '.viewport-element',
|
||||||
|
on: 'left',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '.cornerstone-viewport-element',
|
||||||
|
event: 'CORNERSTONE_TOOLS_MOUSE_WHEEL',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('.viewport-element'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'jumpToMeasurement',
|
||||||
|
title: 'Jumping to Measurements in the Panel',
|
||||||
|
text: 'Click the measurement in the measurement panel to jump to it.',
|
||||||
|
attachTo: {
|
||||||
|
element: '[data-cy="data-row"]',
|
||||||
|
on: 'left-start',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '[data-cy="data-row"]',
|
||||||
|
event: 'click',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('[data-cy="data-row"]'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'changeLayout',
|
||||||
|
title: 'Changing Layout',
|
||||||
|
text: 'You can change the layout of the viewer using the layout button.',
|
||||||
|
attachTo: {
|
||||||
|
element: '[data-cy="Layout"]',
|
||||||
|
on: 'bottom',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '[data-cy="Layout"]',
|
||||||
|
event: 'click',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('[data-cy="Layout"]'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'selectLayout',
|
||||||
|
title: 'Selecting the MPR Layout',
|
||||||
|
text: 'Select the MPR layout to view the images in MPR mode.',
|
||||||
|
attachTo: {
|
||||||
|
element: '[data-cy="MPR"]',
|
||||||
|
on: 'left-start',
|
||||||
|
},
|
||||||
|
advanceOn: {
|
||||||
|
selector: '[data-cy="MPR"]',
|
||||||
|
event: 'click',
|
||||||
|
},
|
||||||
|
beforeShowPromise: () => waitForElement('[data-cy="MPR"]'),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
tourOptions: {
|
||||||
|
useModalOverlay: true,
|
||||||
|
defaultStepOptions: {
|
||||||
|
buttons: [
|
||||||
|
{
|
||||||
|
text: 'Skip all',
|
||||||
|
action() {
|
||||||
|
this.complete();
|
||||||
|
},
|
||||||
|
secondary: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
function waitForElement(selector, maxAttempts = 20, interval = 25) {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
let attempts = 0;
|
||||||
|
|
||||||
|
const checkForElement = setInterval(() => {
|
||||||
|
const element = document.querySelector(selector);
|
||||||
|
|
||||||
|
if (element || attempts >= maxAttempts) {
|
||||||
|
clearInterval(checkForElement);
|
||||||
|
resolve();
|
||||||
|
}
|
||||||
|
|
||||||
|
attempts++;
|
||||||
|
}, interval);
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -321,6 +321,8 @@ export default class DisplaySetService extends PubSubService {
|
|||||||
const handler = this.extensionManager.getModuleEntry(SOPClassHandlerId);
|
const handler = this.extensionManager.getModuleEntry(SOPClassHandlerId);
|
||||||
|
|
||||||
if (handler.sopClassUids.includes(instance.SOPClassUID)) {
|
if (handler.sopClassUids.includes(instance.SOPClassUID)) {
|
||||||
|
// Add this debug
|
||||||
|
console.log('Using SOPHandler:', SOPClassHandlerId, 'for SOPClass:', instance.SOPClassUID);
|
||||||
// Check if displaySets are already created using this SeriesInstanceUID/SOPClassHandler pair.
|
// Check if displaySets are already created using this SeriesInstanceUID/SOPClassHandler pair.
|
||||||
let displaySets = existingDisplaySets.filter(
|
let displaySets = existingDisplaySets.filter(
|
||||||
displaySet => displaySet.SOPClassHandlerId === SOPClassHandlerId
|
displaySet => displaySet.SOPClassHandlerId === SOPClassHandlerId
|
||||||
@@ -363,6 +365,16 @@ export default class DisplaySetService extends PubSubService {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Add debug
|
||||||
|
console.log(
|
||||||
|
'Created displaySets from Series:',
|
||||||
|
displaySets.map(ds => ({
|
||||||
|
uid: ds.displaySetInstanceUID,
|
||||||
|
desc: ds.SeriesDescription,
|
||||||
|
seriesUID: ds.SeriesInstanceUID,
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
// applying hp-defined viewport settings to the displaysets
|
// applying hp-defined viewport settings to the displaysets
|
||||||
displaySets.forEach(ds => {
|
displaySets.forEach(ds => {
|
||||||
Object.keys(settings).forEach(key => {
|
Object.keys(settings).forEach(key => {
|
||||||
|
|||||||
Reference in New Issue
Block a user