123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { nextTick } from 'vue'
- import type { ViewerOptions } from '#shared/composables/useImageViewer.ts'
- import type { MockGraphQLInstance } from './mock-graphql-api'
- import type { Mock } from 'vitest'
- import type { Ref } from 'vue'
- const state = Symbol('test-state')
- interface TestState {
- imageViewerOptions: Ref<ViewerOptions>
- }
- export const getTestState = (): TestState => {
- return (globalThis as any)[state] || {}
- }
- export const setTestState = (newState: Partial<TestState>) => {
- ;(globalThis as any)[state] = {
- ...getTestState(),
- ...newState,
- }
- }
- export const waitForTimeout = async (milliseconds = 0) => {
- return new Promise((resolve) => {
- setTimeout(resolve, milliseconds)
- })
- }
- export const waitForNextTick = async (withTimeout = false) => {
- if (withTimeout) {
- await nextTick()
- return new Promise((resolve) => {
- setTimeout(resolve, 0)
- })
- }
- return nextTick()
- }
- export const waitUntil = async (
- condition: () => unknown,
- msThreshold = process.env.CI ? 30_000 : 1_000,
- ) => {
- // point stack trace to the place where "waitUntil" was called
- const err = new Error('Timeout')
- Error.captureStackTrace(err, waitUntil)
- return new Promise<void>((resolve, reject) => {
- const start = Date.now()
- const max = start + msThreshold
- const interval = setInterval(() => {
- if (condition()) {
- clearInterval(interval)
- resolve()
- }
- if (max < Date.now()) {
- clearInterval(interval)
- reject(err)
- }
- }, 30)
- })
- }
- export const waitUntilApisResolved = (...mockApis: MockGraphQLInstance[]) => {
- return waitUntil(() => mockApis.every((mock) => mock.calls.resolve))
- }
- export const waitUntilSpyCalled = (spy: Mock) => {
- return waitUntil(() => spy.mock.calls.length > 0)
- }
- // The apollo cache always asks for a field, even if it's marked as optional
- // this function returns a proxy that will return "null" on properties not defined
- // in the initial object.
- export const nullableMock = <T extends object>(obj: T): T => {
- const skipProperties = new Set(['_id', 'id', Symbol.toStringTag])
- return new Proxy(obj, {
- get(target, prop, receiver) {
- if (!Reflect.has(target, prop) && !skipProperties.has(prop)) {
- return null
- }
- const value = Reflect.get(target, prop, receiver)
- if (
- Array.isArray(value) &&
- value.every((item) => typeof item === 'object' && item !== null)
- ) {
- return value.map(nullableMock)
- }
- if (typeof value === 'object' && value !== null) {
- return nullableMock(value)
- }
- return value
- },
- })
- }
- export const dataURItoBlob = (dataURI: string) => {
- const byteString = atob(dataURI.split(',')[1])
- const mimeString = dataURI.split(',')[0].split(':')[1].split(';')[0]
- const ab = new ArrayBuffer(byteString.length)
- const ia = new Uint8Array(ab)
- for (let i = 0; i < byteString.length; i += 1) {
- ia[i] = byteString.charCodeAt(i)
- }
- return new Blob([ab], { type: mimeString })
- }
|