@techsee/techsee-media-service 999.15.0-alpha3 → 999.15.1-camera1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/LocalStreamManager.ts"],"names":[],"mappings":"AAOA,OAAO,EAAC,QAAQ,EAAC,MAAM,yBAAyB,CAAC;AAEjD,OAAO,EACH,iBAAiB,EACjB,qBAAqB,EACrB,2BAA2B,EAE3B,yBAAyB,EAE5B,MAAM,kBAAkB,CAAC;AAY1B,OAAO,CAAC,MAAM,CAAC;IACX,UAAU,MAAM;QACZ,qBAAqB,EAAE,QAAQ,CAAC,yBAAyB,CAAC,CAAC;QAC3D,wBAAwB,EAAE,QAAQ,CAAC,yBAAyB,CAAC,CAAC;QAC9D,2BAA2B,EAAE,QAAQ,CAAC,qBAAqB,CAAC,CAAC;QAC7D,0BAA0B,EAAE,OAAO,CAAC;KACvC;CACJ;AAeD,UAAU,eAAe;IACrB,KAAK,EAAE,QAAQ,CAAC;QAAE,IAAI,CAAC,EAAE,eAAe,CAAC;QAAC,KAAK,CAAC,EAAE,eAAe,CAAA;KAAE,CAAC,CAAC;IACrE,KAAK,EAAE,QAAQ,CAAC,eAAe,CAAC,CAAC;IACjC,YAAY,EAAE,MAAM,CAAC;CACxB;AAED,qBAAa,kBAAkB;IAC3B,OAAO,CAAC,WAAW,CAAoB;IACvC,OAAO,CAAC,eAAe,CAAgE;gBAE3E,WAAW,EAAE,iBAAiB;aAOtC,cAAc;IAIlB,kBAAkB,CAAC,WAAW,EAAE,qBAAqB,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA2C1F,qBAAqB,CAAC,WAAW,EAAE,2BAA2B,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA4CnG,sBAAsB,IAAI,OAAO,CAAC,IAAI,CAAC;IAuBvC,yBAAyB,IAAI,OAAO,CAAC,IAAI,CAAC;IAqB1C,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC;IAShC,OAAO,CAAC,0BAA0B;IAwDlC,OAAO,CAAC,6BAA6B;IAmBrC,OAAO,CAAC,mCAAmC;IAmD3C,iBAAiB,CAAC,cAAc,EAAE;QAAC,KAAK,EAAE,eAAe,EAAE,CAAC;QAAC,KAAK,EAAE,eAAe,EAAE,CAAA;KAAC;IAStF,OAAO,CAAC,oBAAoB;IAuE5B,OAAO,CAAC,wBAAwB;CAwCnC","file":"LocalStreamManager.d.ts","sourcesContent":["import assign from 'lodash/assign';\nimport get from 'lodash/get';\nimport last from 'lodash/last';\nimport first from 'lodash/first';\nimport cloneDeep from 'lodash/cloneDeep';\nimport {Promise} from 'bluebird';\n\nimport {Nullable} from '@techsee/techsee-common';\n\nimport {\n IMediaEnvironment,\n LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult,\n MediaRequestSuccessResult,\n VideoStreamResolution\n} from './MediaContracts';\n\nimport {\n CameraTypes,\n ConstraintType,\n DEFAULT_VIDEO_CONSTRAINT_TYPE,\n DEFAULT_VIDEO_RESOLUTION,\n LocalVideoSourceType,\n MediaRequestErrorCode\n} from './MediaConstants';\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\ndeclare global {\n interface Window {\n latestUserMediaStream: Nullable<MediaRequestSuccessResult>;\n latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;\n latestLocalMediaConstraints: Nullable<LocalMediaConstraints>;\n mediaStreamAlreadyReplaced: boolean;\n }\n}\n\ndeclare const window: Window;\n\nwindow.latestUserMediaStream = null;\nlet userMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.latestDesktopMediaStream = null;\nlet desktopMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.mediaStreamAlreadyReplaced = false;\nwindow.latestLocalMediaConstraints = null;\n\nconst trace = getMediaTracer('LocalStreamManager');\n\ninterface IGroupedDevices {\n video: Nullable<{ back?: MediaDeviceInfo; front?: MediaDeviceInfo }>;\n audio: Nullable<MediaDeviceInfo>;\n camerasCount: number;\n}\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: IGroupedDevices = {video: null, audio: null, camerasCount: 0};\n\n constructor(environment: IMediaEnvironment) {\n this.environment = environment;\n\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n\n get groupedDevices() {\n return this._groupedDevices;\n }\n\n getUserMediaStream(constraints: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n\n if (window.latestUserMediaStream) {\n const videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return Promise.resolve({...window.latestUserMediaStream, isNew: false});\n }\n\n this.destroyUserMediaStream();\n }\n\n userMediaPromise = new Promise((resolve, reject) => {\n const catchHandler = (err: any): void => {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(false, constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep(constraints);\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\n\n resolve({...window.latestUserMediaStream, isNew: true});\n })\n .catch(catchHandler);\n } catch (err) {\n catchHandler(err);\n }\n });\n\n return userMediaPromise;\n }\n\n getDesktopMediaStream(constraints: LocalVideoStreamConstraints): Promise<MediaRequestSuccessResult> {\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n\n trace.info('getDesktopMediaStream', constraints);\n\n if (window.latestDesktopMediaStream) {\n const track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return Promise.resolve({...window.latestDesktopMediaStream, isNew: false});\n }\n this.destroyDesktopMediaStream().catch((error: any) => {\n trace.error('Error destroying desktop media stream', error);\n });\n\n }\n\n desktopMediaPromise = new Promise((resolve, reject) => {\n const mediaCatchHandler = (err: any): void => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(true, {video: constraints, audio: false})\n .then((streamResult: MediaRequestSuccessResult) => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n\n resolve({...window.latestDesktopMediaStream, isNew: true});\n }).catch(mediaCatchHandler);\n } catch (err) {\n mediaCatchHandler(err);\n }\n });\n\n return desktopMediaPromise;\n }\n\n destroyUserMediaStream(): Promise<void> {\n const streamResult = window.latestUserMediaStream;\n\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n window.mediaStreamAlreadyReplaced = false;\n\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\n }\n\n return Promise.resolve();\n }\n\n destroyDesktopMediaStream(): Promise<void> {\n const streamResult = window.latestDesktopMediaStream;\n\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyDesktopMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\n }\n }\n\n return Promise.resolve();\n }\n\n clearAllStreams(): Promise<void> {\n trace.info('LocalStreamManager Clearing all streams');\n\n return Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(() => undefined);\n }\n\n private getHighestResolutionStream(isDesktopMedia: boolean, constraints?: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n const constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n\n if (constraintsCandidates.length === 0) {\n return Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n\n return new Promise((resolve, reject) => {\n let lastSuccessfulResult: Nullable<MediaRequestSuccessResult> = null;\n let lastFailedResult: Nullable<MediaRequestFailResult> = null;\n let isStreamRequestFulfilled = false;\n\n (Promise as any).reduce(constraintsCandidates, (total: any, candidate: MediaStreamConstraints) => {\n if (isStreamRequestFulfilled) {\n return;\n }\n\n const constraintType = get(constraints, 'video.videoResolution.constraintType') || get(constraints, 'video.videoResolution[0].constraintType');\n\n return this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then((stream: MediaStream) => {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch((failResult: MediaRequestFailResult) => {\n lastFailedResult = failResult;\n\n if (lastFailedResult.errorCode === MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(() => {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n }\n\n private getUserMediaStreamMediaDevice(constraints: MediaStreamConstraints, isDesktopMedia: boolean) {\n let streamPromise = null;\n\n if (!isDesktopMedia) {\n //window.OT.getUserMedia()\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n } else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n //@ts-ignore\n : navigator.getDisplayMedia(constraints);\n }\n\n return streamPromise;\n }\n\n //@ts-ignore\n private getStreamPromiseByParsedConstraints(constraints: MediaStreamConstraints, isDesktopMedia: boolean): Promise<MediaStream> {\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch((err: any) => {\n let streamPromise = null;\n\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get(constraints, 'video.facingMode') &&\n get(constraints, 'video.deviceId')) {\n\n const newConstraints = constraints;\n\n trace.info(`getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=${get(constraints, 'video.facingMode')}`, err.name);\n\n // @ts-ignore\n delete newConstraints.video.facingMode;\n ((newConstraints.video as MediaTrackConstraints).deviceId as ConstrainDOMStringParameters) = {\n exact: (constraints.video as MediaTrackConstraints).deviceId\n } as any as ConstrainDOMStringParameters;\n\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n\n return this.parseMediaErrorThenThrow({name: 'audioStreamFailed'});\n } else if (err && err.name && constraints.audio) {\n const newConstraints = constraints;\n\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n\n newConstraints.audio = false;\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n\n streamPromise = Promise.reject(err);\n }\n\n return streamPromise && streamPromise.catch(this.parseMediaErrorThenThrow);\n });\n } catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n }\n\n setGroupedDevices(groupedDevices: {video: MediaDeviceInfo[]; audio: MediaDeviceInfo[]}) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = {video: groupedDevices.video\n ? {[CameraTypes.BACK]: last(groupedDevices.video), [CameraTypes.FRONT]: first(groupedDevices.video)}\n : null,\n audio: last(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length};\n }\n\n private getParsedConstraints(isDesktopMedia: boolean, streamConstraints?: LocalMediaConstraints): MediaStreamConstraints[] {\n const constraintCandidates: MediaStreamConstraints[] = [];\n\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{video: true}];\n }\n\n if (!streamConstraints) {\n return constraintCandidates;\n }\n\n const baseConstraints: MediaStreamConstraints = {};\n\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n\n const cameraType = (streamConstraints.video !== 'boolean') &&\n (streamConstraints.video as LocalVideoStreamConstraints).videoSourceType;\n\n baseConstraints.video = {\n facingMode: cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: {ideal: 15, max: 30}\n };\n\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? CameraTypes.FRONT\n : CameraTypes.BACK}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n\n if (this._groupedDevices.camerasCount > 1) {\n baseConstraints.video.facingMode = {exact: baseConstraints.video.facingMode} as any as ConstrainDOMStringParameters;\n }\n }\n\n const assignConstraint = (videoResolution: VideoStreamResolution): void => {\n const [width, height] = videoResolution.resolution.split('x');\n const constraintCandidate = assign({}, baseConstraints);\n\n constraintCandidate.video = assign({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !((streamConstraints.video as LocalVideoStreamConstraints).videoResolution instanceof Array)) {\n\n const defaultResolution: VideoStreamResolution = {\n resolution: DEFAULT_VIDEO_RESOLUTION,\n constraintType: DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n\n assignConstraint(defaultResolution);\n\n return constraintCandidates;\n }\n\n (streamConstraints.video as any).videoResolution.forEach(assignConstraint);\n\n return constraintCandidates;\n }\n\n private parseMediaErrorThenThrow(error: any): MediaRequestFailResult {\n if (error && error.errorCode) {\n throw error;\n }\n\n trace.error('Get media stream error.', error);\n const errorName = error && error.name ? error.name : '';\n\n const permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n\n ];\n\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n\n const constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n\n ];\n\n const isErrorOfType = (errorsArr: string[]) => errorsArr.filter((err: string) => err === errorName).length > 0;\n\n if (isErrorOfType(permissionErrorNames)) {\n throw {errorCode: MediaRequestErrorCode.PermissionDenied, message: errorName};\n }\n\n if (isErrorOfType(constraintsErrors)) {\n throw {errorCode: MediaRequestErrorCode.Overconstrained, message: errorName};\n }\n\n throw {errorCode: MediaRequestErrorCode.GeneralError, message: errorName};\n }\n}\n\nfunction _constraintByType(value: any, type: ConstraintType): any {\n switch (type) {\n case ConstraintType.MIN:\n return {min: value};\n default:\n return {ideal: value};\n }\n}\n"]}
1
+ {"version":3,"sources":["../src/LocalStreamManager.ts"],"names":[],"mappings":"AAOA,OAAO,EAAC,QAAQ,EAAC,MAAM,yBAAyB,CAAC;AAEjD,OAAO,EACH,iBAAiB,EACjB,qBAAqB,EACrB,2BAA2B,EAE3B,yBAAyB,EAE5B,MAAM,kBAAkB,CAAC;AAY1B,OAAO,CAAC,MAAM,CAAC;IACX,UAAU,MAAM;QACZ,qBAAqB,EAAE,QAAQ,CAAC,yBAAyB,CAAC,CAAC;QAC3D,wBAAwB,EAAE,QAAQ,CAAC,yBAAyB,CAAC,CAAC;QAC9D,2BAA2B,EAAE,QAAQ,CAAC,qBAAqB,CAAC,CAAC;QAC7D,0BAA0B,EAAE,OAAO,CAAC;KACvC;CACJ;AAeD,UAAU,eAAe;IACrB,KAAK,EAAE,QAAQ,CAAC;QAAE,IAAI,CAAC,EAAE,eAAe,CAAC;QAAC,KAAK,CAAC,EAAE,eAAe,CAAA;KAAE,CAAC,CAAC;IACrE,KAAK,EAAE,QAAQ,CAAC,eAAe,CAAC,CAAC;IACjC,YAAY,EAAE,MAAM,CAAC;CACxB;AAED,qBAAa,kBAAkB;IAC3B,OAAO,CAAC,WAAW,CAAoB;IACvC,OAAO,CAAC,eAAe,CAAgE;gBAE3E,WAAW,EAAE,iBAAiB;aAOtC,cAAc;IAIlB,kBAAkB,CAAC,WAAW,EAAE,qBAAqB,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA2C1F,qBAAqB,CAAC,WAAW,EAAE,2BAA2B,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA4CnG,sBAAsB,IAAI,OAAO,CAAC,IAAI,CAAC;IAuBvC,yBAAyB,IAAI,OAAO,CAAC,IAAI,CAAC;IAqB1C,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC;IAShC,OAAO,CAAC,0BAA0B;IAwDlC,OAAO,CAAC,6BAA6B;IAkBrC,OAAO,CAAC,mCAAmC;IAmD3C,iBAAiB,CAAC,cAAc,EAAE;QAAC,KAAK,EAAE,eAAe,EAAE,CAAC;QAAC,KAAK,EAAE,eAAe,EAAE,CAAA;KAAC;IAStF,OAAO,CAAC,oBAAoB;IAuE5B,OAAO,CAAC,wBAAwB;CAwCnC","file":"LocalStreamManager.d.ts","sourcesContent":["import assign from 'lodash/assign';\nimport get from 'lodash/get';\nimport last from 'lodash/last';\nimport first from 'lodash/first';\nimport cloneDeep from 'lodash/cloneDeep';\nimport {Promise} from 'bluebird';\n\nimport {Nullable} from '@techsee/techsee-common';\n\nimport {\n IMediaEnvironment,\n LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult,\n MediaRequestSuccessResult,\n VideoStreamResolution\n} from './MediaContracts';\n\nimport {\n CameraTypes,\n ConstraintType,\n DEFAULT_VIDEO_CONSTRAINT_TYPE,\n DEFAULT_VIDEO_RESOLUTION,\n LocalVideoSourceType,\n MediaRequestErrorCode\n} from './MediaConstants';\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\ndeclare global {\n interface Window {\n latestUserMediaStream: Nullable<MediaRequestSuccessResult>;\n latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;\n latestLocalMediaConstraints: Nullable<LocalMediaConstraints>;\n mediaStreamAlreadyReplaced: boolean;\n }\n}\n\ndeclare const window: Window;\n\nwindow.latestUserMediaStream = null;\nlet userMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.latestDesktopMediaStream = null;\nlet desktopMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.mediaStreamAlreadyReplaced = false;\nwindow.latestLocalMediaConstraints = null;\n\nconst trace = getMediaTracer('LocalStreamManager');\n\ninterface IGroupedDevices {\n video: Nullable<{ back?: MediaDeviceInfo; front?: MediaDeviceInfo }>;\n audio: Nullable<MediaDeviceInfo>;\n camerasCount: number;\n}\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: IGroupedDevices = {video: null, audio: null, camerasCount: 0};\n\n constructor(environment: IMediaEnvironment) {\n this.environment = environment;\n\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n\n get groupedDevices() {\n return this._groupedDevices;\n }\n\n getUserMediaStream(constraints: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n\n if (window.latestUserMediaStream) {\n const videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return Promise.resolve({...window.latestUserMediaStream, isNew: false});\n }\n\n this.destroyUserMediaStream();\n }\n\n userMediaPromise = new Promise((resolve, reject) => {\n const catchHandler = (err: any): void => {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(false, constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep(constraints);\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\n\n resolve({...window.latestUserMediaStream, isNew: true});\n })\n .catch(catchHandler);\n } catch (err) {\n catchHandler(err);\n }\n });\n\n return userMediaPromise;\n }\n\n getDesktopMediaStream(constraints: LocalVideoStreamConstraints): Promise<MediaRequestSuccessResult> {\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n\n trace.info('getDesktopMediaStream', constraints);\n\n if (window.latestDesktopMediaStream) {\n const track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return Promise.resolve({...window.latestDesktopMediaStream, isNew: false});\n }\n this.destroyDesktopMediaStream().catch((error: any) => {\n trace.error('Error destroying desktop media stream', error);\n });\n\n }\n\n desktopMediaPromise = new Promise((resolve, reject) => {\n const mediaCatchHandler = (err: any): void => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(true, {video: constraints, audio: false})\n .then((streamResult: MediaRequestSuccessResult) => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n\n resolve({...window.latestDesktopMediaStream, isNew: true});\n }).catch(mediaCatchHandler);\n } catch (err) {\n mediaCatchHandler(err);\n }\n });\n\n return desktopMediaPromise;\n }\n\n destroyUserMediaStream(): Promise<void> {\n const streamResult = window.latestUserMediaStream;\n\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n window.mediaStreamAlreadyReplaced = false;\n\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\n }\n\n return Promise.resolve();\n }\n\n destroyDesktopMediaStream(): Promise<void> {\n const streamResult = window.latestDesktopMediaStream;\n\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyDesktopMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\n }\n }\n\n return Promise.resolve();\n }\n\n clearAllStreams(): Promise<void> {\n trace.info('LocalStreamManager Clearing all streams');\n\n return Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(() => undefined);\n }\n\n private getHighestResolutionStream(isDesktopMedia: boolean, constraints?: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n const constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n\n if (constraintsCandidates.length === 0) {\n return Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n\n return new Promise((resolve, reject) => {\n let lastSuccessfulResult: Nullable<MediaRequestSuccessResult> = null;\n let lastFailedResult: Nullable<MediaRequestFailResult> = null;\n let isStreamRequestFulfilled = false;\n\n (Promise as any).reduce(constraintsCandidates, (total: any, candidate: MediaStreamConstraints) => {\n if (isStreamRequestFulfilled) {\n return;\n }\n\n const constraintType = get(constraints, 'video.videoResolution.constraintType') || get(constraints, 'video.videoResolution[0].constraintType');\n\n return this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then((stream: MediaStream) => {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch((failResult: MediaRequestFailResult) => {\n lastFailedResult = failResult;\n\n if (lastFailedResult.errorCode === MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(() => {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n }\n\n private getUserMediaStreamMediaDevice(constraints: MediaStreamConstraints, isDesktopMedia: boolean) {\n let streamPromise = null;\n\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n } else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n //@ts-ignore\n : navigator.getDisplayMedia(constraints);\n }\n\n return streamPromise;\n }\n\n //@ts-ignore\n private getStreamPromiseByParsedConstraints(constraints: MediaStreamConstraints, isDesktopMedia: boolean): Promise<MediaStream> {\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch((err: any) => {\n let streamPromise = null;\n\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get(constraints, 'video.facingMode') &&\n get(constraints, 'video.deviceId')) {\n\n const newConstraints = constraints;\n\n trace.info(`getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=${get(constraints, 'video.facingMode')}`, err.name);\n\n // @ts-ignore\n delete newConstraints.video.facingMode;\n ((newConstraints.video as MediaTrackConstraints).deviceId as ConstrainDOMStringParameters) = {\n exact: (constraints.video as MediaTrackConstraints).deviceId\n } as any as ConstrainDOMStringParameters;\n\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n\n return this.parseMediaErrorThenThrow({name: 'audioStreamFailed'});\n } else if (err && err.name && constraints.audio) {\n const newConstraints = constraints;\n\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n\n newConstraints.audio = false;\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n\n streamPromise = Promise.reject(err);\n }\n\n return streamPromise && streamPromise.catch(this.parseMediaErrorThenThrow);\n });\n } catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n }\n\n setGroupedDevices(groupedDevices: {video: MediaDeviceInfo[]; audio: MediaDeviceInfo[]}) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = {video: groupedDevices.video\n ? {[CameraTypes.BACK]: last(groupedDevices.video), [CameraTypes.FRONT]: first(groupedDevices.video)}\n : null,\n audio: last(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length};\n }\n\n private getParsedConstraints(isDesktopMedia: boolean, streamConstraints?: LocalMediaConstraints): MediaStreamConstraints[] {\n const constraintCandidates: MediaStreamConstraints[] = [];\n\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{video: true}];\n }\n\n if (!streamConstraints) {\n return constraintCandidates;\n }\n\n const baseConstraints: MediaStreamConstraints = {};\n\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n\n const cameraType = (streamConstraints.video !== 'boolean') &&\n (streamConstraints.video as LocalVideoStreamConstraints).videoSourceType;\n\n baseConstraints.video = {\n facingMode: cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: {ideal: 15, max: 30}\n };\n\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? CameraTypes.FRONT\n : CameraTypes.BACK}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n\n if (this._groupedDevices.camerasCount > 1) {\n baseConstraints.video.facingMode = {exact: baseConstraints.video.facingMode} as any as ConstrainDOMStringParameters;\n }\n }\n\n const assignConstraint = (videoResolution: VideoStreamResolution): void => {\n const [width, height] = videoResolution.resolution.split('x');\n const constraintCandidate = assign({}, baseConstraints);\n\n constraintCandidate.video = assign({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !((streamConstraints.video as LocalVideoStreamConstraints).videoResolution instanceof Array)) {\n\n const defaultResolution: VideoStreamResolution = {\n resolution: DEFAULT_VIDEO_RESOLUTION,\n constraintType: DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n\n assignConstraint(defaultResolution);\n\n return constraintCandidates;\n }\n\n (streamConstraints.video as any).videoResolution.forEach(assignConstraint);\n\n return constraintCandidates;\n }\n\n private parseMediaErrorThenThrow(error: any): MediaRequestFailResult {\n if (error && error.errorCode) {\n throw error;\n }\n\n trace.error('Get media stream error.', error);\n const errorName = error && error.name ? error.name : '';\n\n const permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n\n ];\n\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n\n const constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n\n ];\n\n const isErrorOfType = (errorsArr: string[]) => errorsArr.filter((err: string) => err === errorName).length > 0;\n\n if (isErrorOfType(permissionErrorNames)) {\n throw {errorCode: MediaRequestErrorCode.PermissionDenied, message: errorName};\n }\n\n if (isErrorOfType(constraintsErrors)) {\n throw {errorCode: MediaRequestErrorCode.Overconstrained, message: errorName};\n }\n\n throw {errorCode: MediaRequestErrorCode.GeneralError, message: errorName};\n }\n}\n\nfunction _constraintByType(value: any, type: ConstraintType): any {\n switch (type) {\n case ConstraintType.MIN:\n return {min: value};\n default:\n return {ideal: value};\n }\n}\n"]}
@@ -203,7 +203,6 @@ var LocalStreamManager = /** @class */function () {
203
203
  LocalStreamManager.prototype.getUserMediaStreamMediaDevice = function (constraints, isDesktopMedia) {
204
204
  var streamPromise = null;
205
205
  if (!isDesktopMedia) {
206
- //window.OT.getUserMedia()
207
206
  streamPromise = navigator.mediaDevices.getUserMedia(constraints);
208
207
  } else {
209
208
  //@ts-ignore
@@ -1 +1 @@
1
- {"version":3,"sources":["LocalStreamManager.js","../src/LocalStreamManager.ts"],"names":["__assign","Object","assign","t","s","i","n","arguments","length","p","prototype","hasOwnProperty","call","apply","__importDefault","mod","__esModule","defineProperty","exports","value","assign_1","require","get_1","last_1","first_1","cloneDeep_1","bluebird_1","MediaConstants_1","MediaTracer_1","window","latestUserMediaStream","userMediaPromise","latestDesktopMediaStream","desktopMediaPromise","mediaStreamAlreadyReplaced","latestLocalMediaConstraints","trace","getMediaTracer","LocalStreamManager","environment","_groupedDevices","video","audio","camerasCount","parseMediaErrorThenThrow","bind","clearAllStreams","get","enumerable","configurable","getUserMediaStream","constraints","_this","info","videoTrack","mediaStream","getVideoTracks","readyState","Promise","resolve","isNew","destroyUserMediaStream","reject","catchHandler","err","error","getHighestResolutionStream","then","streamResult","default","constraint","catch","getDesktopMediaStream","track","destroyDesktopMediaStream","mediaCatchHandler","getTracks","forEach","stop","e","all","undefined","isDesktopMedia","constraintsCandidates","getParsedConstraints","Error","lastSuccessfulResult","lastFailedResult","isStreamRequestFulfilled","reduce","total","candidate","constraintType","getStreamPromiseByParsedConstraints","stream","failResult","errorCode","MediaRequestErrorCode","PermissionDenied","GeneralError","message","getUserMediaStreamMediaDevice","streamPromise","navigator","mediaDevices","getUserMedia","getDisplayMedia","name","newConstraints","facingMode","deviceId","exact","setGroupedDevices","groupedDevices","_a","CameraTypes","BACK","FRONT","streamConstraints","constraintCandidates","baseConstraints","cameraType","videoSourceType","LocalVideoSourceType","CAMERA_FRONT","frameRate","ideal","max","assignConstraint","videoResolution","resolution","split","width","height","constraintCandidate","_constraintByType","push","Array","defaultResolution","DEFAULT_VIDEO_RESOLUTION","DEFAULT_VIDEO_CONSTRAINT_TYPE","errorName","permissionErrorNames","constraintsErrors","isErrorOfType","errorsArr","filter","Overconstrained","type","ConstraintType","MIN","min"],"mappings":"AAAA;;AACA,IAAIA,WAAY,aAAQ,UAAKA,QAAd,IAA2B,YAAY;AAClDA,eAAWC,OAAOC,MAAP,IAAiB,UAASC,CAAT,EAAY;AACpC,aAAK,IAAIC,CAAJ,EAAOC,IAAI,CAAX,EAAcC,IAAIC,UAAUC,MAAjC,EAAyCH,IAAIC,CAA7C,EAAgDD,GAAhD,EAAqD;AACjDD,gBAAIG,UAAUF,CAAV,CAAJ;AACA,iBAAK,IAAII,CAAT,IAAcL,CAAd;AAAiB,oBAAIH,OAAOS,SAAP,CAAiBC,cAAjB,CAAgCC,IAAhC,CAAqCR,CAArC,EAAwCK,CAAxC,CAAJ,EACbN,EAAEM,CAAF,IAAOL,EAAEK,CAAF,CAAP;AADJ;AAEH;AACD,eAAON,CAAP;AACH,KAPD;AAQA,WAAOH,SAASa,KAAT,CAAe,IAAf,EAAqBN,SAArB,CAAP;AACH,CAVD;AAWA,IAAIO,kBAAmB,aAAQ,UAAKA,eAAd,IAAkC,UAAUC,GAAV,EAAe;AACnE,WAAQA,OAAOA,IAAIC,UAAZ,GAA0BD,GAA1B,GAAgC,EAAE,WAAWA,GAAb,EAAvC;AACH,CAFD;AAGAd,OAAOgB,cAAP,CAAsBC,OAAtB,EAA+B,YAA/B,EAA6C,EAAEC,OAAO,IAAT,EAA7C;ACfA,IAAAC,WAAAN,gBAAAO,QAAA,eAAA,CAAA,CAAA;AACA,IAAAC,QAAAR,gBAAAO,QAAA,YAAA,CAAA,CAAA;AACA,IAAAE,SAAAT,gBAAAO,QAAA,aAAA,CAAA,CAAA;AACA,IAAAG,UAAAV,gBAAAO,QAAA,cAAA,CAAA,CAAA;AACA,IAAAI,cAAAX,gBAAAO,QAAA,kBAAA,CAAA,CAAA;AACA,IAAAK,aAAAL,QAAA,UAAA,CAAA;AAaA,IAAAM,mBAAAN,QAAA,kBAAA,CAAA;AAQA,IAAAO,gBAAAP,QAAA,0BAAA,CAAA;AAaAQ,OAAOC,qBAAP,GAA+B,IAA/B;AACA,IAAIC,mBAAiE,IAArE;AAEAF,OAAOG,wBAAP,GAAkC,IAAlC;AACA,IAAIC,sBAAoE,IAAxE;AAEAJ,OAAOK,0BAAP,GAAoC,KAApC;AACAL,OAAOM,2BAAP,GAAqC,IAArC;AAEA,IAAMC,QAAQR,cAAAS,cAAA,CAAe,oBAAf,CAAd;AAQA,IAAAC,qBAAA,aAAA,YAAA;AAII,aAAAA,kBAAA,CAAYC,WAAZ,EAA0C;AAFlC,aAAAC,eAAA,GAAmC,EAACC,OAAO,IAAR,EAAcC,OAAO,IAArB,EAA2BC,cAAc,CAAzC,EAAnC;AAGJ,aAAKJ,WAAL,GAAmBA,WAAnB;AAEA,aAAKK,wBAAL,GAAgC,KAAKA,wBAAL,CAA8BC,IAA9B,CAAmC,IAAnC,CAAhC;AACA,aAAKC,eAAL,GAAuB,KAAKA,eAAL,CAAqBD,IAArB,CAA0B,IAA1B,CAAvB;AACH;AAED5C,WAAAgB,cAAA,CAAIqB,mBAAA5B,SAAJ,EAAI,gBAAJ,EAAkB;AD5BdqC,aC4BJ,eAAA;AACI,mBAAO,KAAKP,eAAZ;AACH,SAFiB;ADzBdQ,oBAAY,ICyBE;ADxBdC,sBAAc;ACwBA,KAAlB;AAIAX,uBAAA5B,SAAA,CAAAwC,kBAAA,GAAA,UAAmBC,WAAnB,EAAqD;AAArD,YAAAC,QAAA,IAAA;AACIhB,cAAMiB,IAAN,CAAW,oBAAX,EAAiCF,WAAjC;AACA,YAAIpB,gBAAJ,EAAsB;AAClB,mBAAOA,gBAAP;AACH;AAED,YAAIF,OAAOC,qBAAX,EAAkC;AAC9B,gBAAMwB,aAAazB,OAAOC,qBAAP,CAA6ByB,WAA7B,CAAyCC,cAAzC,EAAnB;AAEA,gBAAIF,WAAW9C,MAAX,GAAoB,CAApB,IAAyB8C,WAAW,CAAX,EAAcG,UAAd,KAA6B,MAA1D,EAAkE;AAC9D,uBAAO/B,WAAAgC,OAAA,CAAQC,OAAR,CAAe3D,SAAA,EAAA,EAAK6B,OAAOC,qBAAZ,EAAiC,EAAE8B,OAAO,KAAT,EAAjC,CAAf,CAAP;AACH;AAED,iBAAKC,sBAAL;AACH;AAED9B,2BAAmB,IAAIL,WAAAgC,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC3C,gBAAMC,eAAe,SAAfA,YAAe,CAACC,GAAD,EAAS;AAC1BjC,mCAAmB,IAAnB;AACAF,uBAAOC,qBAAP,GAA+B,IAA/B;AACAM,sBAAM6B,KAAN,CAAY,oBAAZ,EAAkCD,GAAlC;AACAF,uBAAOE,GAAP;AACH,aALD;AAOA,gBAAI;AACAZ,sBAAKc,0BAAL,CAAgC,KAAhC,EAAuCf,WAAvC,EACKgB,IADL,CACU,UAACC,YAAD,EAAwC;AAC1CrC,uCAAmB,IAAnB;AACAF,2BAAOC,qBAAP,GAA+BsC,YAA/B;AACAvC,2BAAOM,2BAAP,GAAqCV,YAAA4C,OAAA,CAAUlB,WAAV,CAArC;AACAtB,2BAAOM,2BAAP,CAAmCO,KAAnC,GAA4C0B,aAAaE,UAAb,IAA2BF,aAAaE,UAAb,CAAwB5B,KAA/F;AAEAiB,4BAAO3D,SAAA,EAAA,EAAK6B,OAAOC,qBAAZ,EAAiC,EAAE8B,OAAO,IAAT,EAAjC,CAAP;AACH,iBARL,EASKW,KATL,CASWR,YATX;AAUH,aAXD,CAWE,OAAOC,GAAP,EAAY;AACVD,6BAAaC,GAAb;AACH;AACJ,SAtBkB,CAAnB;AAwBA,eAAOjC,gBAAP;AACH,KAzCD;AA2CAO,uBAAA5B,SAAA,CAAA8D,qBAAA,GAAA,UAAsBrB,WAAtB,EAA8D;AAA9D,YAAAC,QAAA,IAAA;AACI,YAAInB,mBAAJ,EAAyB;AACrB,mBAAOA,mBAAP;AACH;AAEDG,cAAMiB,IAAN,CAAW,uBAAX,EAAoCF,WAApC;AAEA,YAAItB,OAAOG,wBAAX,EAAqC;AACjC,gBAAMyC,QAAQ5C,OAAOG,wBAAP,CAAgCuB,WAAhC,CAA4CC,cAA5C,EAAd;AAEA,gBAAIiB,MAAMjE,MAAN,GAAe,CAAf,IAAoBiE,MAAM,CAAN,EAAShB,UAAT,KAAwB,MAAhD,EAAwD;AACpD;AACA,uBAAO/B,WAAAgC,OAAA,CAAQC,OAAR,CAAe3D,SAAA,EAAA,EAAK6B,OAAOG,wBAAZ,EAAoC,EAAE4B,OAAO,KAAT,EAApC,CAAf,CAAP;AACH;AACD,iBAAKc,yBAAL,GAAiCH,KAAjC,CAAuC,UAACN,KAAD,EAAW;AAC9C7B,sBAAM6B,KAAN,CAAY,uCAAZ,EAAqDA,KAArD;AACH,aAFD;AAIH;AAEDhC,8BAAsB,IAAIP,WAAAgC,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC9C,gBAAMa,oBAAoB,SAApBA,iBAAoB,CAACX,GAAD,EAAS;AAC/B/B,sCAAsB,IAAtB;AACAJ,uBAAOG,wBAAP,GAAkC,IAAlC;AACAI,sBAAM6B,KAAN,CAAY,uBAAZ,EAAqCD,GAArC;AACAF,uBAAOE,GAAP;AACH,aALD;AAOA,gBAAI;AACAZ,sBAAKc,0BAAL,CAAgC,IAAhC,EAAsC,EAACzB,OAAOU,WAAR,EAAqBT,OAAO,KAA5B,EAAtC,EACKyB,IADL,CACU,UAACC,YAAD,EAAwC;AAC1CnC,0CAAsB,IAAtB;AACAJ,2BAAOG,wBAAP,GAAkCoC,YAAlC;AAEAT,4BAAO3D,SAAA,EAAA,EAAK6B,OAAOG,wBAAZ,EAAoC,EAAE4B,OAAO,IAAT,EAApC,CAAP;AACH,iBANL,EAMOW,KANP,CAMaI,iBANb;AAOH,aARD,CAQE,OAAOX,GAAP,EAAY;AACVW,kCAAkBX,GAAlB;AACH;AACJ,SAnBqB,CAAtB;AAqBA,eAAO/B,mBAAP;AACH,KA1CD;AA4CAK,uBAAA5B,SAAA,CAAAmD,sBAAA,GAAA,YAAA;AACI,YAAMO,eAAevC,OAAOC,qBAA5B;AAEAC,2BAAmB,IAAnB;AACAF,eAAOC,qBAAP,GAA+B,IAA/B;AACAD,eAAOM,2BAAP,GAAqC,IAArC;AACAN,eAAOK,0BAAP,GAAoC,KAApC;AAEA,YAAIkC,gBAAgBA,aAAab,WAAjC,EAA8C;AAC1C,gBAAI;AACAa,6BAAab,WAAb,CAAyBqB,SAAzB,GAAqCC,OAArC,CAA6C,UAACJ,KAAD,EAAwB;AACjEA,0BAAMK,IAAN;AACH,iBAFD;AAIA1C,sBAAMiB,IAAN,CAAW,0CAAX;AACH,aAND,CAME,OAAO0B,CAAP,EAAU;AACR3C,sBAAMiB,IAAN,CAAW,oDAAX,EAAiE0B,CAAjE;AACH;AACJ;AAED,eAAOrD,WAAAgC,OAAA,CAAQC,OAAR,EAAP;AACH,KArBD;AAuBArB,uBAAA5B,SAAA,CAAAgE,yBAAA,GAAA,YAAA;AACI,YAAMN,eAAevC,OAAOG,wBAA5B;AAEAH,eAAOG,wBAAP,GAAkC,IAAlC;AACAC,8BAAsB,IAAtB;AAEA,YAAImC,gBAAgBA,aAAab,WAAjC,EAA8C;AAC1C,gBAAI;AACAa,6BAAab,WAAb,CAAyBqB,SAAzB,GAAqCC,OAArC,CAA6C,UAACJ,KAAD,EAAwB;AACjEA,0BAAMK,IAAN;AACH,iBAFD;AAIA1C,sBAAMiB,IAAN,CAAW,6CAAX;AACH,aAND,CAME,OAAO0B,CAAP,EAAU;AACR3C,sBAAMiB,IAAN,CAAW,uDAAX,EAAoE0B,CAApE;AACH;AACJ;AAED,eAAOrD,WAAAgC,OAAA,CAAQC,OAAR,EAAP;AACH,KAnBD;AAqBArB,uBAAA5B,SAAA,CAAAoC,eAAA,GAAA,YAAA;AACIV,cAAMiB,IAAN,CAAW,yCAAX;AAEA,eAAO3B,WAAAgC,OAAA,CAAQsB,GAAR,CAAY,CACf,KAAKN,yBAAL,EADe,EAEf,KAAKb,sBAAL,EAFe,CAAZ,EAGJM,IAHI,CAGC,YAAA;AAAM,mBAAAc,SAAA;AAAS,SAHhB,CAAP;AAIH,KAPD;AASQ3C,uBAAA5B,SAAA,CAAAwD,0BAAA,GAAR,UAAmCgB,cAAnC,EAA4D/B,WAA5D,EAA+F;AAA/F,YAAAC,QAAA,IAAA;AACI,YAAM+B,wBAAwB,KAAKC,oBAAL,CAA0BF,cAA1B,EAA0C/B,WAA1C,CAA9B;AAEAf,cAAMiB,IAAN,CAAW,qDAAX,EAAkE8B,qBAAlE;AAEA,YAAIA,sBAAsB3E,MAAtB,KAAiC,CAArC,EAAwC;AACpC,mBAAOkB,WAAAgC,OAAA,CAAQI,MAAR,CAAe,IAAIuB,KAAJ,CAAU,uCAAV,CAAf,CAAP;AACH;AAED,eAAO,IAAI3D,WAAAgC,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC/B,gBAAIwB,uBAA4D,IAAhE;AACA,gBAAIC,mBAAqD,IAAzD;AACA,gBAAIC,2BAA2B,KAA/B;AAEC9D,uBAAAgC,OAAA,CAAgB+B,MAAhB,CAAuBN,qBAAvB,EAA8C,UAACO,KAAD,EAAaC,SAAb,EAA8C;AACzF,oBAAIH,wBAAJ,EAA8B;AAC1B;AACH;AAED,oBAAMI,iBAAiBtE,MAAA+C,OAAA,CAAIlB,WAAJ,EAAiB,sCAAjB,KAA4D7B,MAAA+C,OAAA,CAAIlB,WAAJ,EAAiB,yCAAjB,CAAnF;AAEA,uBAAOC,MAAKyC,mCAAL,CAAyCF,SAAzC,EAAoDT,cAApD,EACFf,IADE,CACG,UAAC2B,MAAD,EAAoB;AACtBR,2CAAuB;AACnB/B,qCAAauC,MADM;AAEnBxB,oCAAYqB,SAFO;AAGnBC,wCAAcA,cAHK;AAInBhC,+BAAO;AAJY,qBAAvB;AAMAxB,0BAAMiB,IAAN,CAAW,oCAAX,EAAiDiC,oBAAjD;AACAE,+CAA2B,IAA3B;AACA7B,4BAAQ2B,oBAAR;AACH,iBAXE,EAYFf,KAZE,CAYI,UAACwB,UAAD,EAAmC;AACtCR,uCAAmBQ,UAAnB;AAEA,wBAAIR,iBAAiBS,SAAjB,KAA+BrE,iBAAAsE,qBAAA,CAAsBC,gBAAzD,EAA2E;AACvEV,mDAA2B,IAA3B;AACA1B,+BAAOyB,gBAAP;AACH;AACJ,iBAnBE,CAAP;AAoBH,aA3BA,EA2BE,CA3BF,EA4BIpB,IA5BJ,CA4BS,YAAA;AACF,oBAAI,CAACqB,wBAAL,EAA+B;AAC3B,wBAAI,CAACD,gBAAL,EAAuB;AACnBA,2CAAmB;AACfS,uCAAWrE,iBAAAsE,qBAAA,CAAsBE,YADlB;AAEfC,qCAAS;AAFM,yBAAnB;AAIH;AACDtC,2BAAOyB,gBAAP;AACH;AACJ,aAtCJ;AAuCJ,SA5CM,CAAP;AA6CH,KAtDO;AAwDAjD,uBAAA5B,SAAA,CAAA2F,6BAAA,GAAR,UAAsClD,WAAtC,EAA2E+B,cAA3E,EAAkG;AAC9F,YAAIoB,gBAAgB,IAApB;AAEA,YAAI,CAACpB,cAAL,EAAqB;AACjB;AACAoB,4BAAgBC,UAAUC,YAAV,CAAuBC,YAAvB,CAAoCtD,WAApC,CAAhB;AACH,SAHD,MAGO;AACH;AACAmD,4BAAgBC,UAAUC,YAAV,CAAuBE;AACnC;AADY,cAEVH,UAAUC,YAAV,CAAuBE,eAAvB,CAAuCvD,WAAvC;AACF;AAHY,cAIVoD,UAAUG,eAAV,CAA0BvD,WAA1B,CAJN;AAKH;AAED,eAAOmD,aAAP;AACH,KAhBO;AAkBR;AACQhE,uBAAA5B,SAAA,CAAAmF,mCAAA,GAAR,UAA4C1C,WAA5C,EAAiF+B,cAAjF,EAAwG;AAAxG,YAAA9B,QAAA,IAAA;AACIhB,cAAMiB,IAAN,CAAW,qCAAX,EAAkDF,WAAlD;AAEA,YAAI;AACA,mBAAO,KAAKkD,6BAAL,CAAmClD,WAAnC,EAAgD+B,cAAhD,EAAgEX,KAAhE,CAAsE,UAACP,GAAD,EAAS;AAClF,oBAAIsC,gBAAgB,IAApB;AAEAlE,sBAAMiB,IAAN,CAAW,6DAAX,EAA0EW,IAAI2C,IAA9E;AAEA;AACA;AACA;AACA;AACA,oBAAI3C,OAAQA,IAAI2C,IAAJ,KAAa,kBAArB,IAA4CrF,MAAA+C,OAAA,CAAIlB,WAAJ,EAAiB,kBAAjB,CAA5C,IACA7B,MAAA+C,OAAA,CAAIlB,WAAJ,EAAiB,gBAAjB,CADJ,EACwC;AAEpC,wBAAMyD,iBAAiBzD,WAAvB;AAEAf,0BAAMiB,IAAN,CAAW,oHAAkH/B,MAAA+C,OAAA,CAAIlB,WAAJ,EAAiB,kBAAjB,CAA7H,EAAqKa,IAAI2C,IAAzK;AAEA;AACA,2BAAOC,eAAenE,KAAf,CAAqBoE,UAA5B;AACED,mCAAenE,KAAf,CAA+CqE,QAA/C,GAA2F;AACzFC,+BAAQ5D,YAAYV,KAAZ,CAA4CqE;AADqC,qBAA3F;AAIFR,oCAAgBlD,MAAKyC,mCAAL,CAAyCe,cAAzC,EAAyD1B,cAAzD,CAAhB;AACH,iBAdD,MAcO,IAAIlB,OAAOA,IAAI2C,IAAX,IAAmBxD,YAAYT,KAA/B,IAAwC,CAACS,YAAYV,KAAzD,EAAgE;AACnEL,0BAAMiB,IAAN,CAAW,iGAAX,EAA8GW,IAAI2C,IAAlH;AAEA,2BAAOvD,MAAKR,wBAAL,CAA8B,EAAC+D,MAAM,mBAAP,EAA9B,CAAP;AACH,iBAJM,MAIA,IAAI3C,OAAOA,IAAI2C,IAAX,IAAmBxD,YAAYT,KAAnC,EAA0C;AAC7C,wBAAMkE,iBAAiBzD,WAAvB;AAEAf,0BAAMiB,IAAN,CAAW,2HAAX,EAAwIW,IAAI2C,IAA5I;AAEAC,mCAAelE,KAAf,GAAuB,KAAvB;AACA4D,oCAAgBlD,MAAKyC,mCAAL,CAAyCe,cAAzC,EAAyD1B,cAAzD,CAAhB;AACH,iBAPM,MAOA,IAAIlB,OAAO,CAACsC,aAAZ,EAA2B;AAC9BlE,0BAAMiB,IAAN,CAAW,6EAAX,EAA0FW,OAAOA,IAAI2C,IAArG;AAEAL,oCAAgB5E,WAAAgC,OAAA,CAAQI,MAAR,CAAeE,GAAf,CAAhB;AACH;AAED,uBAAOsC,iBAAiBA,cAAc/B,KAAd,CAAoBnB,MAAKR,wBAAzB,CAAxB;AACH,aAzCM,CAAP;AA0CH,SA3CD,CA2CE,OAAOmC,CAAP,EAAU;AACR,iBAAKnC,wBAAL,CAA8BmC,CAA9B;AACH;AACJ,KAjDO;AAmDRzC,uBAAA5B,SAAA,CAAAsG,iBAAA,GAAA,UAAkBC,cAAlB,EAAsF;ADjElF,YAAIC,EAAJ;ACkEA9E,cAAMiB,IAAN,CAAW,mBAAX,EAAgC4D,cAAhC;AACA,aAAKzE,eAAL,GAAuB,EAACC,OAAOwE,eAAexE,KAAf,IAC1ByE,KAAA,EAAA,EAAEA,GAACvF,iBAAAwF,WAAA,CAAYC,IAAb,IAAoB7F,OAAA8C,OAAA,CAAK4C,eAAexE,KAApB,CAAtB,EAAkDyE,GAACvF,iBAAAwF,WAAA,CAAYE,KAAb,IAAqB7F,QAAA6C,OAAA,CAAM4C,eAAexE,KAArB,CAAvE,EAAkGyE,EADxE,IAEzB,IAFiB;AAGvBxE,mBAAOnB,OAAA8C,OAAA,CAAK4C,eAAevE,KAApB,KAA8B,IAHd;AAIvBC,0BAAcsE,eAAexE,KAAf,CAAqBjC,MAJZ,EAAvB;AAKH,KAPD;AASQ8B,uBAAA5B,SAAA,CAAA0E,oBAAA,GAAR,UAA6BF,cAA7B,EAAsDoC,iBAAtD,EAA+F;AAC3F,YAAMC,uBAAiD,EAAvD;AAEA;AACA,YAAIrC,cAAJ,EAAoB;AAChB,mBAAO,CAAC,EAACzC,OAAO,IAAR,EAAD,CAAP;AACH;AAED,YAAI,CAAC6E,iBAAL,EAAwB;AACpB,mBAAOC,oBAAP;AACH;AAED,YAAMC,kBAA0C,EAAhD;AAEAA,wBAAgB9E,KAAhB,GAAwB,OAAO4E,kBAAkB5E,KAAzB,KAAmC,SAAnC,GAA+C4E,kBAAkB5E,KAAjE,GAAyEuC,SAAjG;AAEA,YAAI,CAACqC,kBAAkB7E,KAAvB,EAA8B;AAC1B,mBAAO,CAAC+E,eAAD,CAAP;AACH;AAED,YAAMC,aAAcH,kBAAkB7E,KAAlB,KAA4B,SAA7B,IACd6E,kBAAkB7E,KAAlB,CAAwDiF,eAD7D;AAGAF,wBAAgB/E,KAAhB,GAAwB;AACpBoE,wBAAYY,cAAcA,eAAe9F,iBAAAgG,oBAAA,CAAqBC,YAAlD,GACN,MADM,GAEN,aAHc;AAIpBC,uBAAW,EAACC,OAAO,EAAR,EAAYC,KAAK,EAAjB;AAJS,SAAxB;AAOA,YAAMjB,WAAWxF,MAAA+C,OAAA,CAAI,KAAK7B,eAAT,EAA0B,YAASiF,eAAe9F,iBAAAgG,oBAAA,CAAqBC,YAApC,GAC9CjG,iBAAAwF,WAAA,CAAYE,KADkC,GAE9C1F,iBAAAwF,WAAA,CAAYC,IAFyB,IAErB,WAFL,CAAjB;AAIA,YAAIN,QAAJ,EAAc;AACVU,4BAAgB/E,KAAhB,CAAsBqE,QAAtB,GAAiCA,QAAjC;AAEA,gBAAI,KAAKtE,eAAL,CAAqBG,YAArB,GAAoC,CAAxC,EAA2C;AACvC6E,gCAAgB/E,KAAhB,CAAsBoE,UAAtB,GAAmC,EAACE,OAAOS,gBAAgB/E,KAAhB,CAAsBoE,UAA9B,EAAnC;AACH;AACJ;AAED,YAAMmB,mBAAmB,SAAnBA,gBAAmB,CAACC,eAAD,EAAuC;AACtD,gBAAAf,KAAAe,gBAAAC,UAAA,CAAAC,KAAA,CAAA,GAAA,CAAA;AAAA,gBAACC,QAAAlB,GAAA,CAAA,CAAD;AAAA,gBAAQmB,SAAAnB,GAAA,CAAA,CAAR;AACN,gBAAMoB,sBAAsBlH,SAAAiD,OAAA,CAAO,EAAP,EAAWmD,eAAX,CAA5B;AAEAc,gCAAoB7F,KAApB,GAA4BrB,SAAAiD,OAAA,CAAO,EAAP,EAAWmD,gBAAgB/E,KAA3B,EAAkC;AAC1D2F,uBAAOG,kBAAkBH,KAAlB,EAAyBH,gBAAgBrC,cAAzC,CADmD;AAE1DyC,wBAAQE,kBAAkBF,MAAlB,EAA0BJ,gBAAgBrC,cAA1C;AAFkD,aAAlC,CAA5B;AAIA2B,iCAAqBiB,IAArB,CAA0BF,mBAA1B;AACH,SATD;AAWA,YAAKd,gBAAgB/E,KAAhB,IAAyB,OAAO6E,kBAAkB7E,KAAzB,KAAmC,SAA7D,IACA,EAAG6E,kBAAkB7E,KAAlB,CAAwDwF,eAAxD,YAAmFQ,KAAtF,CADJ,EACkG;AAE9F,gBAAMC,oBAA2C;AAC7CR,4BAAYvG,iBAAAgH,wBADiC;AAE7C/C,gCAAgBjE,iBAAAiH;AAF6B,aAAjD;AAKAZ,6BAAiBU,iBAAjB;AAEA,mBAAOnB,oBAAP;AACH;AAEAD,0BAAkB7E,KAAlB,CAAgCwF,eAAhC,CAAgDpD,OAAhD,CAAwDmD,gBAAxD;AAED,eAAOT,oBAAP;AACH,KArEO;AAuEAjF,uBAAA5B,SAAA,CAAAkC,wBAAA,GAAR,UAAiCqB,KAAjC,EAA2C;AACvC,YAAIA,SAASA,MAAM+B,SAAnB,EAA8B;AAC1B,kBAAM/B,KAAN;AACH;AAED7B,cAAM6B,KAAN,CAAY,yBAAZ,EAAuCA,KAAvC;AACA,YAAM4E,YAAY5E,SAASA,MAAM0C,IAAf,GAAsB1C,MAAM0C,IAA5B,GAAmC,EAArD;AAEA,YAAMmC,uBAAuB,CACzB,iBADyB,EAEzB,0BAFyB,EAGzB,uBAHyB,CAA7B;AAOA;AACA;AACA;AACA;AACA;AACA;AAEA,YAAMC,oBAAoB,CACtB,sBADsB,EAEtB,6BAFsB,CAA1B;AAMA,YAAMC,gBAAgB,SAAhBA,aAAgB,CAACC,SAAD,EAAoB;AAAK,mBAAAA,UAAUC,MAAV,CAAiB,UAAClF,GAAD,EAAY;AAAK,uBAAAA,QAAQ6E,SAAR;AAAiB,aAAnD,EAAqDrI,MAArD,GAA8D,CAA9D;AAA+D,SAA9G;AAEA,YAAIwI,cAAcF,oBAAd,CAAJ,EAAyC;AACrC,kBAAM,EAAC9C,WAAWrE,iBAAAsE,qBAAA,CAAsBC,gBAAlC,EAAoDE,SAASyC,SAA7D,EAAN;AACH;AAED,YAAIG,cAAcD,iBAAd,CAAJ,EAAsC;AAClC,kBAAM,EAAC/C,WAAWrE,iBAAAsE,qBAAA,CAAsBkD,eAAlC,EAAmD/C,SAASyC,SAA5D,EAAN;AACH;AAED,cAAM,EAAC7C,WAAWrE,iBAAAsE,qBAAA,CAAsBE,YAAlC,EAAgDC,SAASyC,SAAzD,EAAN;AACH,KAvCO;AAwCZ,WAAAvG,kBAAA;AAjZA,CAAA,EAAA;AAAapB,QAAAoB,kBAAA,GAAAA,kBAAA;AAmZb,SAASiG,iBAAT,CAA2BpH,KAA3B,EAAuCiI,IAAvC,EAA2D;AACvD,YAAQA,IAAR;AACI,aAAKzH,iBAAA0H,cAAA,CAAeC,GAApB;AACI,mBAAO,EAACC,KAAKpI,KAAN,EAAP;AACJ;AACI,mBAAO,EAAC2G,OAAO3G,KAAR,EAAP;AAJR;AAMH;;AD7FD","file":"LocalStreamManager.js","sourcesContent":["\"use strict\";\nvar __assign = (this && this.__assign) || function () {\n __assign = Object.assign || function(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))\n t[p] = s[p];\n }\n return t;\n };\n return __assign.apply(this, arguments);\n};\nvar __importDefault = (this && this.__importDefault) || function (mod) {\n return (mod && mod.__esModule) ? mod : { \"default\": mod };\n};\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar assign_1 = __importDefault(require(\"lodash/assign\"));\nvar get_1 = __importDefault(require(\"lodash/get\"));\nvar last_1 = __importDefault(require(\"lodash/last\"));\nvar first_1 = __importDefault(require(\"lodash/first\"));\nvar cloneDeep_1 = __importDefault(require(\"lodash/cloneDeep\"));\nvar bluebird_1 = require(\"bluebird\");\nvar MediaConstants_1 = require(\"./MediaConstants\");\nvar MediaTracer_1 = require(\"./MediaUtils/MediaTracer\");\nwindow.latestUserMediaStream = null;\nvar userMediaPromise = null;\nwindow.latestDesktopMediaStream = null;\nvar desktopMediaPromise = null;\nwindow.mediaStreamAlreadyReplaced = false;\nwindow.latestLocalMediaConstraints = null;\nvar trace = MediaTracer_1.getMediaTracer('LocalStreamManager');\nvar LocalStreamManager = /** @class */ (function () {\n function LocalStreamManager(environment) {\n this._groupedDevices = { video: null, audio: null, camerasCount: 0 };\n this.environment = environment;\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n Object.defineProperty(LocalStreamManager.prototype, \"groupedDevices\", {\n get: function () {\n return this._groupedDevices;\n },\n enumerable: true,\n configurable: true\n });\n LocalStreamManager.prototype.getUserMediaStream = function (constraints) {\n var _this = this;\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n if (window.latestUserMediaStream) {\n var videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return bluebird_1.Promise.resolve(__assign({}, window.latestUserMediaStream, { isNew: false }));\n }\n this.destroyUserMediaStream();\n }\n userMediaPromise = new bluebird_1.Promise(function (resolve, reject) {\n var catchHandler = function (err) {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n try {\n _this.getHighestResolutionStream(false, constraints)\n .then(function (streamResult) {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep_1.default(constraints);\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio);\n resolve(__assign({}, window.latestUserMediaStream, { isNew: true }));\n })\n .catch(catchHandler);\n }\n catch (err) {\n catchHandler(err);\n }\n });\n return userMediaPromise;\n };\n LocalStreamManager.prototype.getDesktopMediaStream = function (constraints) {\n var _this = this;\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n trace.info('getDesktopMediaStream', constraints);\n if (window.latestDesktopMediaStream) {\n var track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return bluebird_1.Promise.resolve(__assign({}, window.latestDesktopMediaStream, { isNew: false }));\n }\n this.destroyDesktopMediaStream().catch(function (error) {\n trace.error('Error destroying desktop media stream', error);\n });\n }\n desktopMediaPromise = new bluebird_1.Promise(function (resolve, reject) {\n var mediaCatchHandler = function (err) {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n try {\n _this.getHighestResolutionStream(true, { video: constraints, audio: false })\n .then(function (streamResult) {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n resolve(__assign({}, window.latestDesktopMediaStream, { isNew: true }));\n }).catch(mediaCatchHandler);\n }\n catch (err) {\n mediaCatchHandler(err);\n }\n });\n return desktopMediaPromise;\n };\n LocalStreamManager.prototype.destroyUserMediaStream = function () {\n var streamResult = window.latestUserMediaStream;\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n window.mediaStreamAlreadyReplaced = false;\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\n }\n return bluebird_1.Promise.resolve();\n };\n LocalStreamManager.prototype.destroyDesktopMediaStream = function () {\n var streamResult = window.latestDesktopMediaStream;\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyDesktopMediaStream - stop all tracks');\n }\n catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\n }\n }\n return bluebird_1.Promise.resolve();\n };\n LocalStreamManager.prototype.clearAllStreams = function () {\n trace.info('LocalStreamManager Clearing all streams');\n return bluebird_1.Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(function () { return undefined; });\n };\n LocalStreamManager.prototype.getHighestResolutionStream = function (isDesktopMedia, constraints) {\n var _this = this;\n var constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n if (constraintsCandidates.length === 0) {\n return bluebird_1.Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n return new bluebird_1.Promise(function (resolve, reject) {\n var lastSuccessfulResult = null;\n var lastFailedResult = null;\n var isStreamRequestFulfilled = false;\n bluebird_1.Promise.reduce(constraintsCandidates, function (total, candidate) {\n if (isStreamRequestFulfilled) {\n return;\n }\n var constraintType = get_1.default(constraints, 'video.videoResolution.constraintType') || get_1.default(constraints, 'video.videoResolution[0].constraintType');\n return _this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then(function (stream) {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType: constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch(function (failResult) {\n lastFailedResult = failResult;\n if (lastFailedResult.errorCode === MediaConstants_1.MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(function () {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaConstants_1.MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n };\n LocalStreamManager.prototype.getUserMediaStreamMediaDevice = function (constraints, isDesktopMedia) {\n var streamPromise = null;\n if (!isDesktopMedia) {\n //window.OT.getUserMedia()\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n }\n else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n //@ts-ignore\n : navigator.getDisplayMedia(constraints);\n }\n return streamPromise;\n };\n //@ts-ignore\n LocalStreamManager.prototype.getStreamPromiseByParsedConstraints = function (constraints, isDesktopMedia) {\n var _this = this;\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch(function (err) {\n var streamPromise = null;\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get_1.default(constraints, 'video.facingMode') &&\n get_1.default(constraints, 'video.deviceId')) {\n var newConstraints = constraints;\n trace.info(\"getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=\" + get_1.default(constraints, 'video.facingMode'), err.name);\n // @ts-ignore\n delete newConstraints.video.facingMode;\n newConstraints.video.deviceId = {\n exact: constraints.video.deviceId\n };\n streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n }\n else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n return _this.parseMediaErrorThenThrow({ name: 'audioStreamFailed' });\n }\n else if (err && err.name && constraints.audio) {\n var newConstraints = constraints;\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n newConstraints.audio = false;\n streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n }\n else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n streamPromise = bluebird_1.Promise.reject(err);\n }\n return streamPromise && streamPromise.catch(_this.parseMediaErrorThenThrow);\n });\n }\n catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n };\n LocalStreamManager.prototype.setGroupedDevices = function (groupedDevices) {\n var _a;\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = { video: groupedDevices.video\n ? (_a = {}, _a[MediaConstants_1.CameraTypes.BACK] = last_1.default(groupedDevices.video), _a[MediaConstants_1.CameraTypes.FRONT] = first_1.default(groupedDevices.video), _a) : null,\n audio: last_1.default(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length };\n };\n LocalStreamManager.prototype.getParsedConstraints = function (isDesktopMedia, streamConstraints) {\n var constraintCandidates = [];\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{ video: true }];\n }\n if (!streamConstraints) {\n return constraintCandidates;\n }\n var baseConstraints = {};\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n var cameraType = (streamConstraints.video !== 'boolean') &&\n streamConstraints.video.videoSourceType;\n baseConstraints.video = {\n facingMode: cameraType && cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: { ideal: 15, max: 30 }\n };\n var deviceId = get_1.default(this._groupedDevices, \"video.\" + (cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n ? MediaConstants_1.CameraTypes.FRONT\n : MediaConstants_1.CameraTypes.BACK) + \".deviceId\");\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n if (this._groupedDevices.camerasCount > 1) {\n baseConstraints.video.facingMode = { exact: baseConstraints.video.facingMode };\n }\n }\n var assignConstraint = function (videoResolution) {\n var _a = videoResolution.resolution.split('x'), width = _a[0], height = _a[1];\n var constraintCandidate = assign_1.default({}, baseConstraints);\n constraintCandidate.video = assign_1.default({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !(streamConstraints.video.videoResolution instanceof Array)) {\n var defaultResolution = {\n resolution: MediaConstants_1.DEFAULT_VIDEO_RESOLUTION,\n constraintType: MediaConstants_1.DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n assignConstraint(defaultResolution);\n return constraintCandidates;\n }\n streamConstraints.video.videoResolution.forEach(assignConstraint);\n return constraintCandidates;\n };\n LocalStreamManager.prototype.parseMediaErrorThenThrow = function (error) {\n if (error && error.errorCode) {\n throw error;\n }\n trace.error('Get media stream error.', error);\n var errorName = error && error.name ? error.name : '';\n var permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n ];\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n var constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n ];\n var isErrorOfType = function (errorsArr) { return errorsArr.filter(function (err) { return err === errorName; }).length > 0; };\n if (isErrorOfType(permissionErrorNames)) {\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.PermissionDenied, message: errorName };\n }\n if (isErrorOfType(constraintsErrors)) {\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.Overconstrained, message: errorName };\n }\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.GeneralError, message: errorName };\n };\n return LocalStreamManager;\n}());\nexports.LocalStreamManager = LocalStreamManager;\nfunction _constraintByType(value, type) {\n switch (type) {\n case MediaConstants_1.ConstraintType.MIN:\n return { min: value };\n default:\n return { ideal: value };\n }\n}\n\n//# sourceMappingURL=LocalStreamManager.js.map\n","import assign from 'lodash/assign';\nimport get from 'lodash/get';\nimport last from 'lodash/last';\nimport first from 'lodash/first';\nimport cloneDeep from 'lodash/cloneDeep';\nimport {Promise} from 'bluebird';\n\nimport {Nullable} from '@techsee/techsee-common';\n\nimport {\n IMediaEnvironment,\n LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult,\n MediaRequestSuccessResult,\n VideoStreamResolution\n} from './MediaContracts';\n\nimport {\n CameraTypes,\n ConstraintType,\n DEFAULT_VIDEO_CONSTRAINT_TYPE,\n DEFAULT_VIDEO_RESOLUTION,\n LocalVideoSourceType,\n MediaRequestErrorCode\n} from './MediaConstants';\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\ndeclare global {\n interface Window {\n latestUserMediaStream: Nullable<MediaRequestSuccessResult>;\n latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;\n latestLocalMediaConstraints: Nullable<LocalMediaConstraints>;\n mediaStreamAlreadyReplaced: boolean;\n }\n}\n\ndeclare const window: Window;\n\nwindow.latestUserMediaStream = null;\nlet userMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.latestDesktopMediaStream = null;\nlet desktopMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.mediaStreamAlreadyReplaced = false;\nwindow.latestLocalMediaConstraints = null;\n\nconst trace = getMediaTracer('LocalStreamManager');\n\ninterface IGroupedDevices {\n video: Nullable<{ back?: MediaDeviceInfo; front?: MediaDeviceInfo }>;\n audio: Nullable<MediaDeviceInfo>;\n camerasCount: number;\n}\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: IGroupedDevices = {video: null, audio: null, camerasCount: 0};\n\n constructor(environment: IMediaEnvironment) {\n this.environment = environment;\n\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n\n get groupedDevices() {\n return this._groupedDevices;\n }\n\n getUserMediaStream(constraints: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n\n if (window.latestUserMediaStream) {\n const videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return Promise.resolve({...window.latestUserMediaStream, isNew: false});\n }\n\n this.destroyUserMediaStream();\n }\n\n userMediaPromise = new Promise((resolve, reject) => {\n const catchHandler = (err: any): void => {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(false, constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep(constraints);\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\n\n resolve({...window.latestUserMediaStream, isNew: true});\n })\n .catch(catchHandler);\n } catch (err) {\n catchHandler(err);\n }\n });\n\n return userMediaPromise;\n }\n\n getDesktopMediaStream(constraints: LocalVideoStreamConstraints): Promise<MediaRequestSuccessResult> {\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n\n trace.info('getDesktopMediaStream', constraints);\n\n if (window.latestDesktopMediaStream) {\n const track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return Promise.resolve({...window.latestDesktopMediaStream, isNew: false});\n }\n this.destroyDesktopMediaStream().catch((error: any) => {\n trace.error('Error destroying desktop media stream', error);\n });\n\n }\n\n desktopMediaPromise = new Promise((resolve, reject) => {\n const mediaCatchHandler = (err: any): void => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(true, {video: constraints, audio: false})\n .then((streamResult: MediaRequestSuccessResult) => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n\n resolve({...window.latestDesktopMediaStream, isNew: true});\n }).catch(mediaCatchHandler);\n } catch (err) {\n mediaCatchHandler(err);\n }\n });\n\n return desktopMediaPromise;\n }\n\n destroyUserMediaStream(): Promise<void> {\n const streamResult = window.latestUserMediaStream;\n\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n window.mediaStreamAlreadyReplaced = false;\n\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\n }\n\n return Promise.resolve();\n }\n\n destroyDesktopMediaStream(): Promise<void> {\n const streamResult = window.latestDesktopMediaStream;\n\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyDesktopMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\n }\n }\n\n return Promise.resolve();\n }\n\n clearAllStreams(): Promise<void> {\n trace.info('LocalStreamManager Clearing all streams');\n\n return Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(() => undefined);\n }\n\n private getHighestResolutionStream(isDesktopMedia: boolean, constraints?: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n const constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n\n if (constraintsCandidates.length === 0) {\n return Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n\n return new Promise((resolve, reject) => {\n let lastSuccessfulResult: Nullable<MediaRequestSuccessResult> = null;\n let lastFailedResult: Nullable<MediaRequestFailResult> = null;\n let isStreamRequestFulfilled = false;\n\n (Promise as any).reduce(constraintsCandidates, (total: any, candidate: MediaStreamConstraints) => {\n if (isStreamRequestFulfilled) {\n return;\n }\n\n const constraintType = get(constraints, 'video.videoResolution.constraintType') || get(constraints, 'video.videoResolution[0].constraintType');\n\n return this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then((stream: MediaStream) => {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch((failResult: MediaRequestFailResult) => {\n lastFailedResult = failResult;\n\n if (lastFailedResult.errorCode === MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(() => {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n }\n\n private getUserMediaStreamMediaDevice(constraints: MediaStreamConstraints, isDesktopMedia: boolean) {\n let streamPromise = null;\n\n if (!isDesktopMedia) {\n //window.OT.getUserMedia()\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n } else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n //@ts-ignore\n : navigator.getDisplayMedia(constraints);\n }\n\n return streamPromise;\n }\n\n //@ts-ignore\n private getStreamPromiseByParsedConstraints(constraints: MediaStreamConstraints, isDesktopMedia: boolean): Promise<MediaStream> {\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch((err: any) => {\n let streamPromise = null;\n\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get(constraints, 'video.facingMode') &&\n get(constraints, 'video.deviceId')) {\n\n const newConstraints = constraints;\n\n trace.info(`getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=${get(constraints, 'video.facingMode')}`, err.name);\n\n // @ts-ignore\n delete newConstraints.video.facingMode;\n ((newConstraints.video as MediaTrackConstraints).deviceId as ConstrainDOMStringParameters) = {\n exact: (constraints.video as MediaTrackConstraints).deviceId\n } as any as ConstrainDOMStringParameters;\n\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n\n return this.parseMediaErrorThenThrow({name: 'audioStreamFailed'});\n } else if (err && err.name && constraints.audio) {\n const newConstraints = constraints;\n\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n\n newConstraints.audio = false;\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n\n streamPromise = Promise.reject(err);\n }\n\n return streamPromise && streamPromise.catch(this.parseMediaErrorThenThrow);\n });\n } catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n }\n\n setGroupedDevices(groupedDevices: {video: MediaDeviceInfo[]; audio: MediaDeviceInfo[]}) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = {video: groupedDevices.video\n ? {[CameraTypes.BACK]: last(groupedDevices.video), [CameraTypes.FRONT]: first(groupedDevices.video)}\n : null,\n audio: last(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length};\n }\n\n private getParsedConstraints(isDesktopMedia: boolean, streamConstraints?: LocalMediaConstraints): MediaStreamConstraints[] {\n const constraintCandidates: MediaStreamConstraints[] = [];\n\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{video: true}];\n }\n\n if (!streamConstraints) {\n return constraintCandidates;\n }\n\n const baseConstraints: MediaStreamConstraints = {};\n\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n\n const cameraType = (streamConstraints.video !== 'boolean') &&\n (streamConstraints.video as LocalVideoStreamConstraints).videoSourceType;\n\n baseConstraints.video = {\n facingMode: cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: {ideal: 15, max: 30}\n };\n\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? CameraTypes.FRONT\n : CameraTypes.BACK}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n\n if (this._groupedDevices.camerasCount > 1) {\n baseConstraints.video.facingMode = {exact: baseConstraints.video.facingMode} as any as ConstrainDOMStringParameters;\n }\n }\n\n const assignConstraint = (videoResolution: VideoStreamResolution): void => {\n const [width, height] = videoResolution.resolution.split('x');\n const constraintCandidate = assign({}, baseConstraints);\n\n constraintCandidate.video = assign({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !((streamConstraints.video as LocalVideoStreamConstraints).videoResolution instanceof Array)) {\n\n const defaultResolution: VideoStreamResolution = {\n resolution: DEFAULT_VIDEO_RESOLUTION,\n constraintType: DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n\n assignConstraint(defaultResolution);\n\n return constraintCandidates;\n }\n\n (streamConstraints.video as any).videoResolution.forEach(assignConstraint);\n\n return constraintCandidates;\n }\n\n private parseMediaErrorThenThrow(error: any): MediaRequestFailResult {\n if (error && error.errorCode) {\n throw error;\n }\n\n trace.error('Get media stream error.', error);\n const errorName = error && error.name ? error.name : '';\n\n const permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n\n ];\n\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n\n const constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n\n ];\n\n const isErrorOfType = (errorsArr: string[]) => errorsArr.filter((err: string) => err === errorName).length > 0;\n\n if (isErrorOfType(permissionErrorNames)) {\n throw {errorCode: MediaRequestErrorCode.PermissionDenied, message: errorName};\n }\n\n if (isErrorOfType(constraintsErrors)) {\n throw {errorCode: MediaRequestErrorCode.Overconstrained, message: errorName};\n }\n\n throw {errorCode: MediaRequestErrorCode.GeneralError, message: errorName};\n }\n}\n\nfunction _constraintByType(value: any, type: ConstraintType): any {\n switch (type) {\n case ConstraintType.MIN:\n return {min: value};\n default:\n return {ideal: value};\n }\n}\n"]}
1
+ {"version":3,"sources":["LocalStreamManager.js","../src/LocalStreamManager.ts"],"names":["__assign","Object","assign","t","s","i","n","arguments","length","p","prototype","hasOwnProperty","call","apply","__importDefault","mod","__esModule","defineProperty","exports","value","assign_1","require","get_1","last_1","first_1","cloneDeep_1","bluebird_1","MediaConstants_1","MediaTracer_1","window","latestUserMediaStream","userMediaPromise","latestDesktopMediaStream","desktopMediaPromise","mediaStreamAlreadyReplaced","latestLocalMediaConstraints","trace","getMediaTracer","LocalStreamManager","environment","_groupedDevices","video","audio","camerasCount","parseMediaErrorThenThrow","bind","clearAllStreams","get","enumerable","configurable","getUserMediaStream","constraints","_this","info","videoTrack","mediaStream","getVideoTracks","readyState","Promise","resolve","isNew","destroyUserMediaStream","reject","catchHandler","err","error","getHighestResolutionStream","then","streamResult","default","constraint","catch","getDesktopMediaStream","track","destroyDesktopMediaStream","mediaCatchHandler","getTracks","forEach","stop","e","all","undefined","isDesktopMedia","constraintsCandidates","getParsedConstraints","Error","lastSuccessfulResult","lastFailedResult","isStreamRequestFulfilled","reduce","total","candidate","constraintType","getStreamPromiseByParsedConstraints","stream","failResult","errorCode","MediaRequestErrorCode","PermissionDenied","GeneralError","message","getUserMediaStreamMediaDevice","streamPromise","navigator","mediaDevices","getUserMedia","getDisplayMedia","name","newConstraints","facingMode","deviceId","exact","setGroupedDevices","groupedDevices","_a","CameraTypes","BACK","FRONT","streamConstraints","constraintCandidates","baseConstraints","cameraType","videoSourceType","LocalVideoSourceType","CAMERA_FRONT","frameRate","ideal","max","assignConstraint","videoResolution","resolution","split","width","height","constraintCandidate","_constraintByType","push","Array","defaultResolution","DEFAULT_VIDEO_RESOLUTION","DEFAULT_VIDEO_CONSTRAINT_TYPE","errorName","permissionErrorNames","constraintsErrors","isErrorOfType","errorsArr","filter","Overconstrained","type","ConstraintType","MIN","min"],"mappings":"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","file":"LocalStreamManager.js","sourcesContent":["\"use strict\";\nvar __assign = (this && this.__assign) || function () {\n __assign = Object.assign || function(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))\n t[p] = s[p];\n }\n return t;\n };\n return __assign.apply(this, arguments);\n};\nvar __importDefault = (this && this.__importDefault) || function (mod) {\n return (mod && mod.__esModule) ? mod : { \"default\": mod };\n};\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar assign_1 = __importDefault(require(\"lodash/assign\"));\nvar get_1 = __importDefault(require(\"lodash/get\"));\nvar last_1 = __importDefault(require(\"lodash/last\"));\nvar first_1 = __importDefault(require(\"lodash/first\"));\nvar cloneDeep_1 = __importDefault(require(\"lodash/cloneDeep\"));\nvar bluebird_1 = require(\"bluebird\");\nvar MediaConstants_1 = require(\"./MediaConstants\");\nvar MediaTracer_1 = require(\"./MediaUtils/MediaTracer\");\nwindow.latestUserMediaStream = null;\nvar userMediaPromise = null;\nwindow.latestDesktopMediaStream = null;\nvar desktopMediaPromise = null;\nwindow.mediaStreamAlreadyReplaced = false;\nwindow.latestLocalMediaConstraints = null;\nvar trace = MediaTracer_1.getMediaTracer('LocalStreamManager');\nvar LocalStreamManager = /** @class */ (function () {\n function LocalStreamManager(environment) {\n this._groupedDevices = { video: null, audio: null, camerasCount: 0 };\n this.environment = environment;\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n Object.defineProperty(LocalStreamManager.prototype, \"groupedDevices\", {\n get: function () {\n return this._groupedDevices;\n },\n enumerable: true,\n configurable: true\n });\n LocalStreamManager.prototype.getUserMediaStream = function (constraints) {\n var _this = this;\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n if (window.latestUserMediaStream) {\n var videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return bluebird_1.Promise.resolve(__assign({}, window.latestUserMediaStream, { isNew: false }));\n }\n this.destroyUserMediaStream();\n }\n userMediaPromise = new bluebird_1.Promise(function (resolve, reject) {\n var catchHandler = function (err) {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n try {\n _this.getHighestResolutionStream(false, constraints)\n .then(function (streamResult) {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep_1.default(constraints);\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio);\n resolve(__assign({}, window.latestUserMediaStream, { isNew: true }));\n })\n .catch(catchHandler);\n }\n catch (err) {\n catchHandler(err);\n }\n });\n return userMediaPromise;\n };\n LocalStreamManager.prototype.getDesktopMediaStream = function (constraints) {\n var _this = this;\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n trace.info('getDesktopMediaStream', constraints);\n if (window.latestDesktopMediaStream) {\n var track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return bluebird_1.Promise.resolve(__assign({}, window.latestDesktopMediaStream, { isNew: false }));\n }\n this.destroyDesktopMediaStream().catch(function (error) {\n trace.error('Error destroying desktop media stream', error);\n });\n }\n desktopMediaPromise = new bluebird_1.Promise(function (resolve, reject) {\n var mediaCatchHandler = function (err) {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n try {\n _this.getHighestResolutionStream(true, { video: constraints, audio: false })\n .then(function (streamResult) {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n resolve(__assign({}, window.latestDesktopMediaStream, { isNew: true }));\n }).catch(mediaCatchHandler);\n }\n catch (err) {\n mediaCatchHandler(err);\n }\n });\n return desktopMediaPromise;\n };\n LocalStreamManager.prototype.destroyUserMediaStream = function () {\n var streamResult = window.latestUserMediaStream;\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n window.mediaStreamAlreadyReplaced = false;\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\n }\n return bluebird_1.Promise.resolve();\n };\n LocalStreamManager.prototype.destroyDesktopMediaStream = function () {\n var streamResult = window.latestDesktopMediaStream;\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyDesktopMediaStream - stop all tracks');\n }\n catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\n }\n }\n return bluebird_1.Promise.resolve();\n };\n LocalStreamManager.prototype.clearAllStreams = function () {\n trace.info('LocalStreamManager Clearing all streams');\n return bluebird_1.Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(function () { return undefined; });\n };\n LocalStreamManager.prototype.getHighestResolutionStream = function (isDesktopMedia, constraints) {\n var _this = this;\n var constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n if (constraintsCandidates.length === 0) {\n return bluebird_1.Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n return new bluebird_1.Promise(function (resolve, reject) {\n var lastSuccessfulResult = null;\n var lastFailedResult = null;\n var isStreamRequestFulfilled = false;\n bluebird_1.Promise.reduce(constraintsCandidates, function (total, candidate) {\n if (isStreamRequestFulfilled) {\n return;\n }\n var constraintType = get_1.default(constraints, 'video.videoResolution.constraintType') || get_1.default(constraints, 'video.videoResolution[0].constraintType');\n return _this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then(function (stream) {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType: constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch(function (failResult) {\n lastFailedResult = failResult;\n if (lastFailedResult.errorCode === MediaConstants_1.MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(function () {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaConstants_1.MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n };\n LocalStreamManager.prototype.getUserMediaStreamMediaDevice = function (constraints, isDesktopMedia) {\n var streamPromise = null;\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n }\n else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n //@ts-ignore\n : navigator.getDisplayMedia(constraints);\n }\n return streamPromise;\n };\n //@ts-ignore\n LocalStreamManager.prototype.getStreamPromiseByParsedConstraints = function (constraints, isDesktopMedia) {\n var _this = this;\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch(function (err) {\n var streamPromise = null;\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get_1.default(constraints, 'video.facingMode') &&\n get_1.default(constraints, 'video.deviceId')) {\n var newConstraints = constraints;\n trace.info(\"getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=\" + get_1.default(constraints, 'video.facingMode'), err.name);\n // @ts-ignore\n delete newConstraints.video.facingMode;\n newConstraints.video.deviceId = {\n exact: constraints.video.deviceId\n };\n streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n }\n else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n return _this.parseMediaErrorThenThrow({ name: 'audioStreamFailed' });\n }\n else if (err && err.name && constraints.audio) {\n var newConstraints = constraints;\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n newConstraints.audio = false;\n streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n }\n else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n streamPromise = bluebird_1.Promise.reject(err);\n }\n return streamPromise && streamPromise.catch(_this.parseMediaErrorThenThrow);\n });\n }\n catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n };\n LocalStreamManager.prototype.setGroupedDevices = function (groupedDevices) {\n var _a;\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = { video: groupedDevices.video\n ? (_a = {}, _a[MediaConstants_1.CameraTypes.BACK] = last_1.default(groupedDevices.video), _a[MediaConstants_1.CameraTypes.FRONT] = first_1.default(groupedDevices.video), _a) : null,\n audio: last_1.default(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length };\n };\n LocalStreamManager.prototype.getParsedConstraints = function (isDesktopMedia, streamConstraints) {\n var constraintCandidates = [];\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{ video: true }];\n }\n if (!streamConstraints) {\n return constraintCandidates;\n }\n var baseConstraints = {};\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n var cameraType = (streamConstraints.video !== 'boolean') &&\n streamConstraints.video.videoSourceType;\n baseConstraints.video = {\n facingMode: cameraType && cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: { ideal: 15, max: 30 }\n };\n var deviceId = get_1.default(this._groupedDevices, \"video.\" + (cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n ? MediaConstants_1.CameraTypes.FRONT\n : MediaConstants_1.CameraTypes.BACK) + \".deviceId\");\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n if (this._groupedDevices.camerasCount > 1) {\n baseConstraints.video.facingMode = { exact: baseConstraints.video.facingMode };\n }\n }\n var assignConstraint = function (videoResolution) {\n var _a = videoResolution.resolution.split('x'), width = _a[0], height = _a[1];\n var constraintCandidate = assign_1.default({}, baseConstraints);\n constraintCandidate.video = assign_1.default({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !(streamConstraints.video.videoResolution instanceof Array)) {\n var defaultResolution = {\n resolution: MediaConstants_1.DEFAULT_VIDEO_RESOLUTION,\n constraintType: MediaConstants_1.DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n assignConstraint(defaultResolution);\n return constraintCandidates;\n }\n streamConstraints.video.videoResolution.forEach(assignConstraint);\n return constraintCandidates;\n };\n LocalStreamManager.prototype.parseMediaErrorThenThrow = function (error) {\n if (error && error.errorCode) {\n throw error;\n }\n trace.error('Get media stream error.', error);\n var errorName = error && error.name ? error.name : '';\n var permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n ];\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n var constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n ];\n var isErrorOfType = function (errorsArr) { return errorsArr.filter(function (err) { return err === errorName; }).length > 0; };\n if (isErrorOfType(permissionErrorNames)) {\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.PermissionDenied, message: errorName };\n }\n if (isErrorOfType(constraintsErrors)) {\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.Overconstrained, message: errorName };\n }\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.GeneralError, message: errorName };\n };\n return LocalStreamManager;\n}());\nexports.LocalStreamManager = LocalStreamManager;\nfunction _constraintByType(value, type) {\n switch (type) {\n case MediaConstants_1.ConstraintType.MIN:\n return { min: value };\n default:\n return { ideal: value };\n }\n}\n\n//# sourceMappingURL=LocalStreamManager.js.map\n","import assign from 'lodash/assign';\nimport get from 'lodash/get';\nimport last from 'lodash/last';\nimport first from 'lodash/first';\nimport cloneDeep from 'lodash/cloneDeep';\nimport {Promise} from 'bluebird';\n\nimport {Nullable} from '@techsee/techsee-common';\n\nimport {\n IMediaEnvironment,\n LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult,\n MediaRequestSuccessResult,\n VideoStreamResolution\n} from './MediaContracts';\n\nimport {\n CameraTypes,\n ConstraintType,\n DEFAULT_VIDEO_CONSTRAINT_TYPE,\n DEFAULT_VIDEO_RESOLUTION,\n LocalVideoSourceType,\n MediaRequestErrorCode\n} from './MediaConstants';\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\ndeclare global {\n interface Window {\n latestUserMediaStream: Nullable<MediaRequestSuccessResult>;\n latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;\n latestLocalMediaConstraints: Nullable<LocalMediaConstraints>;\n mediaStreamAlreadyReplaced: boolean;\n }\n}\n\ndeclare const window: Window;\n\nwindow.latestUserMediaStream = null;\nlet userMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.latestDesktopMediaStream = null;\nlet desktopMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.mediaStreamAlreadyReplaced = false;\nwindow.latestLocalMediaConstraints = null;\n\nconst trace = getMediaTracer('LocalStreamManager');\n\ninterface IGroupedDevices {\n video: Nullable<{ back?: MediaDeviceInfo; front?: MediaDeviceInfo }>;\n audio: Nullable<MediaDeviceInfo>;\n camerasCount: number;\n}\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: IGroupedDevices = {video: null, audio: null, camerasCount: 0};\n\n constructor(environment: IMediaEnvironment) {\n this.environment = environment;\n\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n\n get groupedDevices() {\n return this._groupedDevices;\n }\n\n getUserMediaStream(constraints: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n\n if (window.latestUserMediaStream) {\n const videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return Promise.resolve({...window.latestUserMediaStream, isNew: false});\n }\n\n this.destroyUserMediaStream();\n }\n\n userMediaPromise = new Promise((resolve, reject) => {\n const catchHandler = (err: any): void => {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(false, constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep(constraints);\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\n\n resolve({...window.latestUserMediaStream, isNew: true});\n })\n .catch(catchHandler);\n } catch (err) {\n catchHandler(err);\n }\n });\n\n return userMediaPromise;\n }\n\n getDesktopMediaStream(constraints: LocalVideoStreamConstraints): Promise<MediaRequestSuccessResult> {\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n\n trace.info('getDesktopMediaStream', constraints);\n\n if (window.latestDesktopMediaStream) {\n const track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return Promise.resolve({...window.latestDesktopMediaStream, isNew: false});\n }\n this.destroyDesktopMediaStream().catch((error: any) => {\n trace.error('Error destroying desktop media stream', error);\n });\n\n }\n\n desktopMediaPromise = new Promise((resolve, reject) => {\n const mediaCatchHandler = (err: any): void => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(true, {video: constraints, audio: false})\n .then((streamResult: MediaRequestSuccessResult) => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n\n resolve({...window.latestDesktopMediaStream, isNew: true});\n }).catch(mediaCatchHandler);\n } catch (err) {\n mediaCatchHandler(err);\n }\n });\n\n return desktopMediaPromise;\n }\n\n destroyUserMediaStream(): Promise<void> {\n const streamResult = window.latestUserMediaStream;\n\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n window.mediaStreamAlreadyReplaced = false;\n\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\n }\n\n return Promise.resolve();\n }\n\n destroyDesktopMediaStream(): Promise<void> {\n const streamResult = window.latestDesktopMediaStream;\n\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyDesktopMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\n }\n }\n\n return Promise.resolve();\n }\n\n clearAllStreams(): Promise<void> {\n trace.info('LocalStreamManager Clearing all streams');\n\n return Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(() => undefined);\n }\n\n private getHighestResolutionStream(isDesktopMedia: boolean, constraints?: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n const constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n\n if (constraintsCandidates.length === 0) {\n return Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n\n return new Promise((resolve, reject) => {\n let lastSuccessfulResult: Nullable<MediaRequestSuccessResult> = null;\n let lastFailedResult: Nullable<MediaRequestFailResult> = null;\n let isStreamRequestFulfilled = false;\n\n (Promise as any).reduce(constraintsCandidates, (total: any, candidate: MediaStreamConstraints) => {\n if (isStreamRequestFulfilled) {\n return;\n }\n\n const constraintType = get(constraints, 'video.videoResolution.constraintType') || get(constraints, 'video.videoResolution[0].constraintType');\n\n return this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then((stream: MediaStream) => {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch((failResult: MediaRequestFailResult) => {\n lastFailedResult = failResult;\n\n if (lastFailedResult.errorCode === MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(() => {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n }\n\n private getUserMediaStreamMediaDevice(constraints: MediaStreamConstraints, isDesktopMedia: boolean) {\n let streamPromise = null;\n\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n } else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n //@ts-ignore\n : navigator.getDisplayMedia(constraints);\n }\n\n return streamPromise;\n }\n\n //@ts-ignore\n private getStreamPromiseByParsedConstraints(constraints: MediaStreamConstraints, isDesktopMedia: boolean): Promise<MediaStream> {\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch((err: any) => {\n let streamPromise = null;\n\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get(constraints, 'video.facingMode') &&\n get(constraints, 'video.deviceId')) {\n\n const newConstraints = constraints;\n\n trace.info(`getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=${get(constraints, 'video.facingMode')}`, err.name);\n\n // @ts-ignore\n delete newConstraints.video.facingMode;\n ((newConstraints.video as MediaTrackConstraints).deviceId as ConstrainDOMStringParameters) = {\n exact: (constraints.video as MediaTrackConstraints).deviceId\n } as any as ConstrainDOMStringParameters;\n\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n\n return this.parseMediaErrorThenThrow({name: 'audioStreamFailed'});\n } else if (err && err.name && constraints.audio) {\n const newConstraints = constraints;\n\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n\n newConstraints.audio = false;\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n\n streamPromise = Promise.reject(err);\n }\n\n return streamPromise && streamPromise.catch(this.parseMediaErrorThenThrow);\n });\n } catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n }\n\n setGroupedDevices(groupedDevices: {video: MediaDeviceInfo[]; audio: MediaDeviceInfo[]}) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = {video: groupedDevices.video\n ? {[CameraTypes.BACK]: last(groupedDevices.video), [CameraTypes.FRONT]: first(groupedDevices.video)}\n : null,\n audio: last(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length};\n }\n\n private getParsedConstraints(isDesktopMedia: boolean, streamConstraints?: LocalMediaConstraints): MediaStreamConstraints[] {\n const constraintCandidates: MediaStreamConstraints[] = [];\n\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{video: true}];\n }\n\n if (!streamConstraints) {\n return constraintCandidates;\n }\n\n const baseConstraints: MediaStreamConstraints = {};\n\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n\n const cameraType = (streamConstraints.video !== 'boolean') &&\n (streamConstraints.video as LocalVideoStreamConstraints).videoSourceType;\n\n baseConstraints.video = {\n facingMode: cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: {ideal: 15, max: 30}\n };\n\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? CameraTypes.FRONT\n : CameraTypes.BACK}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n\n if (this._groupedDevices.camerasCount > 1) {\n baseConstraints.video.facingMode = {exact: baseConstraints.video.facingMode} as any as ConstrainDOMStringParameters;\n }\n }\n\n const assignConstraint = (videoResolution: VideoStreamResolution): void => {\n const [width, height] = videoResolution.resolution.split('x');\n const constraintCandidate = assign({}, baseConstraints);\n\n constraintCandidate.video = assign({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !((streamConstraints.video as LocalVideoStreamConstraints).videoResolution instanceof Array)) {\n\n const defaultResolution: VideoStreamResolution = {\n resolution: DEFAULT_VIDEO_RESOLUTION,\n constraintType: DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n\n assignConstraint(defaultResolution);\n\n return constraintCandidates;\n }\n\n (streamConstraints.video as any).videoResolution.forEach(assignConstraint);\n\n return constraintCandidates;\n }\n\n private parseMediaErrorThenThrow(error: any): MediaRequestFailResult {\n if (error && error.errorCode) {\n throw error;\n }\n\n trace.error('Get media stream error.', error);\n const errorName = error && error.name ? error.name : '';\n\n const permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n\n ];\n\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n\n const constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n\n ];\n\n const isErrorOfType = (errorsArr: string[]) => errorsArr.filter((err: string) => err === errorName).length > 0;\n\n if (isErrorOfType(permissionErrorNames)) {\n throw {errorCode: MediaRequestErrorCode.PermissionDenied, message: errorName};\n }\n\n if (isErrorOfType(constraintsErrors)) {\n throw {errorCode: MediaRequestErrorCode.Overconstrained, message: errorName};\n }\n\n throw {errorCode: MediaRequestErrorCode.GeneralError, message: errorName};\n }\n}\n\nfunction _constraintByType(value: any, type: ConstraintType): any {\n switch (type) {\n case ConstraintType.MIN:\n return {min: value};\n default:\n return {ideal: value};\n }\n}\n"]}
@@ -104,7 +104,6 @@ export interface IMediaSession {
104
104
  onMediaStreamRenewed(clientRole: SessionClientRole, mediaStream: MediaStream): Promise<void>;
105
105
  replaceStreamTracks(mediaStream: Nullable<MediaStream>): Promise<void>;
106
106
  registerEventCallback(event: string, cb: (eventArgs: any) => void): void;
107
- switchCamera?(clientRole: SessionClientRole): Promise<any>;
108
107
  }
109
108
  export interface RemoteMediaTrack {
110
109
  mediaTrack: MediaStreamTrack;
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/MediaContracts.ts"],"names":[],"mappings":"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","file":"MediaContracts.d.ts","sourcesContent":["import {Nullable} from '@techsee/techsee-common';\n\nexport interface IMediaEnvironment {\n isIOS: () => boolean;\n isIE11: () => boolean;\n majorVersion: () => number;\n}\n\nimport {\n ConstraintType,\n KnownMediaStream,\n LocalVideoSourceType,\n MediaRequestErrorCode,\n MediaServiceType,\n MediaSessionDisconnectReason,\n MediaStreamUnregisterReason,\n SessionClientRole,\n SessionClientType\n} from './MediaConstants';\n\nexport interface LocalMediaConstraints {\n video: boolean | LocalVideoStreamConstraints;\n audio?: boolean;\n}\n\nexport interface VideoStreamResolution {\n resolution: string;\n constraintType: ConstraintType;\n}\n\nexport interface LocalVideoStreamConstraints {\n videoResolution?: string | VideoStreamResolution[];\n videoSourceType?: LocalVideoSourceType;\n facingMode?: string;\n}\n\nexport interface MediaRequestSuccessResult {\n mediaStream: MediaStream;\n constraint: MediaStreamConstraints;\n constraintType: ConstraintType;\n isNew: boolean;\n}\n\nexport interface MediaRequestFailResult {\n message: string;\n errorCode: MediaRequestErrorCode;\n}\n\nexport interface ISessionStreamsManager {\n getMediaStreamForRole(destinationRole: SessionClientRole): Promise<Nullable<MediaStream>>;\n\n addRemoteMediaTrack(mediaTrack: RemoteMediaTrack): Promise<void>;\n\n removeMediaTrack(mediaTrack: MediaStreamTrack): Promise<void>;\n}\n\nexport interface MediaServiceOptions {\n mediaServiceType: MediaServiceType;\n //How much to wait until raise \"peerConnectionTimeout\" if WebRTC connection events not arriving\n peerConnectivityTimeoutSeconds?: number;\n\n //How much to wait totally for video stream to arrive, before making fallback to image upload\n videoNotReceivedTimeoutSeconds?: number;\n}\n\nexport interface MediaSessionParams {\n sessionId: string;\n clientType: SessionClientType;\n clientRole: SessionClientRole;\n credentials: MediaSessionIceCredentials;\n initHandlers?: () => void;\n removeHandlers?: () => void;\n}\n\nexport interface MediaSessionConfiguration {\n //How much to wait until raise \"peerConnectionTimeout\" if WebRTC connection events not arriving\n peerConnectivityTimeoutSeconds: number;\n}\n\nexport interface MediaSubscriberParams {\n container: HTMLDivElement;\n streamType: KnownMediaStream;\n}\n\nexport interface MediaPublisherParams {\n streamTypes: KnownMediaStream[];\n destinationRole: SessionClientRole;\n}\n\nexport interface ClientWebRtcInfo {\n isWebRTCSupported: boolean;\n isWebsiteHasWebcamPermissions: boolean;\n isWebsiteHasMicrophonePermissions: boolean;\n isGetUserMediaSupported: boolean;\n isApplyConstraintsSupported: boolean;\n}\n\nexport interface IMediaSubscriber {\n readonly streamType: KnownMediaStream;\n readonly container: HTMLDivElement;\n readonly isPlaying: boolean;\n readonly isSoundMuted: boolean;\n readonly hasAudio: boolean;\n readonly hasVideo: boolean;\n readonly mediaElement: Nullable<HTMLVideoElement>;\n readonly videoWidth: number;\n readonly videoHeight: number;\n readonly renderWidth: number;\n readonly renderHeight: number;\n\n muteSound(isMuted: boolean): void;\n\n onStateChanged(callback: () => void): void;\n\n onDispose(callback: () => void): void;\n}\n\nexport type MediaSessionIceCredentials = any;//TODO - Define type for credentials\n\nexport interface MediaSessionHandlers {\n onDisconnectedHandler: MediaSessionDisconnectedHandler;\n}\n\nexport type MediaSessionDisconnectedHandler = (reason: MediaSessionDisconnectReason) => void\n\nexport interface DeviceSupportInfo {\n webRtcSupportInfo: ClientWebRtcInfo;\n hasCamera: boolean;\n hasMicrophone: boolean;\n videoPlayback: boolean;\n}\n\nexport interface RemoteTrackStats {\n trackId: string;\n trackStats: any;\n}\n\nexport interface IMediaSession {\n //Boolean parameter is used for supporting \"RTC Test\" application\n connect(connectOnly?: boolean): Promise<void>;\n\n disconnect(): Promise<void>;\n\n sessionDisconnect(): void;\n\n getRemoteTrackStats(mediaTrack: MediaStreamTrack): Promise<RemoteTrackStats>;\n\n onMediaStreamDestroyed(clientRole: SessionClientRole): Promise<void>;\n\n onMediaStreamRenewed(clientRole: SessionClientRole, mediaStream: MediaStream): Promise<void>;\n\n replaceStreamTracks(mediaStream: Nullable<MediaStream>): Promise<void>;\n\n registerEventCallback(event: string, cb: (eventArgs: any) => void): void;\n\n switchCamera?(clientRole: SessionClientRole): Promise<any>;\n}\n\nexport interface RemoteMediaTrack {\n mediaTrack: MediaStreamTrack;\n trackType: KnownMediaStream;\n isRegistered: boolean;\n}\n\nexport interface SnapshotResult {\n base64img: string;\n objectUrl: string;\n imageBlob: Blob;\n mimeType: string;\n}\n\nexport interface SnapshotOptions {\n format?: string;\n quality?: number;\n x?: any;\n y?: any;\n w?: any;\n h?: any;\n}\n\nexport interface StreamDestroyedEventArgs {\n streamType: KnownMediaStream;\n reason: MediaStreamUnregisterReason;\n}\n\nexport interface StreamCreatedEventArgs {\n streamType: KnownMediaStream;\n}\n"]}
1
+ {"version":3,"sources":["../src/MediaContracts.ts"],"names":[],"mappings":"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","file":"MediaContracts.d.ts","sourcesContent":["import {Nullable} from '@techsee/techsee-common';\n\nexport interface IMediaEnvironment {\n isIOS: () => boolean;\n isIE11: () => boolean;\n majorVersion: () => number;\n}\n\nimport {\n ConstraintType,\n KnownMediaStream,\n LocalVideoSourceType,\n MediaRequestErrorCode,\n MediaServiceType,\n MediaSessionDisconnectReason,\n MediaStreamUnregisterReason,\n SessionClientRole,\n SessionClientType\n} from './MediaConstants';\n\nexport interface LocalMediaConstraints {\n video: boolean | LocalVideoStreamConstraints;\n audio?: boolean;\n}\n\nexport interface VideoStreamResolution {\n resolution: string;\n constraintType: ConstraintType;\n}\n\nexport interface LocalVideoStreamConstraints {\n videoResolution?: string | VideoStreamResolution[];\n videoSourceType?: LocalVideoSourceType;\n facingMode?: string;\n}\n\nexport interface MediaRequestSuccessResult {\n mediaStream: MediaStream;\n constraint: MediaStreamConstraints;\n constraintType: ConstraintType;\n isNew: boolean;\n}\n\nexport interface MediaRequestFailResult {\n message: string;\n errorCode: MediaRequestErrorCode;\n}\n\nexport interface ISessionStreamsManager {\n getMediaStreamForRole(destinationRole: SessionClientRole): Promise<Nullable<MediaStream>>;\n\n addRemoteMediaTrack(mediaTrack: RemoteMediaTrack): Promise<void>;\n\n removeMediaTrack(mediaTrack: MediaStreamTrack): Promise<void>;\n}\n\nexport interface MediaServiceOptions {\n mediaServiceType: MediaServiceType;\n //How much to wait until raise \"peerConnectionTimeout\" if WebRTC connection events not arriving\n peerConnectivityTimeoutSeconds?: number;\n\n //How much to wait totally for video stream to arrive, before making fallback to image upload\n videoNotReceivedTimeoutSeconds?: number;\n}\n\nexport interface MediaSessionParams {\n sessionId: string;\n clientType: SessionClientType;\n clientRole: SessionClientRole;\n credentials: MediaSessionIceCredentials;\n initHandlers?: () => void;\n removeHandlers?: () => void;\n}\n\nexport interface MediaSessionConfiguration {\n //How much to wait until raise \"peerConnectionTimeout\" if WebRTC connection events not arriving\n peerConnectivityTimeoutSeconds: number;\n}\n\nexport interface MediaSubscriberParams {\n container: HTMLDivElement;\n streamType: KnownMediaStream;\n}\n\nexport interface MediaPublisherParams {\n streamTypes: KnownMediaStream[];\n destinationRole: SessionClientRole;\n}\n\nexport interface ClientWebRtcInfo {\n isWebRTCSupported: boolean;\n isWebsiteHasWebcamPermissions: boolean;\n isWebsiteHasMicrophonePermissions: boolean;\n isGetUserMediaSupported: boolean;\n isApplyConstraintsSupported: boolean;\n}\n\nexport interface IMediaSubscriber {\n readonly streamType: KnownMediaStream;\n readonly container: HTMLDivElement;\n readonly isPlaying: boolean;\n readonly isSoundMuted: boolean;\n readonly hasAudio: boolean;\n readonly hasVideo: boolean;\n readonly mediaElement: Nullable<HTMLVideoElement>;\n readonly videoWidth: number;\n readonly videoHeight: number;\n readonly renderWidth: number;\n readonly renderHeight: number;\n\n muteSound(isMuted: boolean): void;\n\n onStateChanged(callback: () => void): void;\n\n onDispose(callback: () => void): void;\n}\n\nexport type MediaSessionIceCredentials = any;//TODO - Define type for credentials\n\nexport interface MediaSessionHandlers {\n onDisconnectedHandler: MediaSessionDisconnectedHandler;\n}\n\nexport type MediaSessionDisconnectedHandler = (reason: MediaSessionDisconnectReason) => void\n\nexport interface DeviceSupportInfo {\n webRtcSupportInfo: ClientWebRtcInfo;\n hasCamera: boolean;\n hasMicrophone: boolean;\n videoPlayback: boolean;\n}\n\nexport interface RemoteTrackStats {\n trackId: string;\n trackStats: any;\n}\n\nexport interface IMediaSession {\n //Boolean parameter is used for supporting \"RTC Test\" application\n connect(connectOnly?: boolean): Promise<void>;\n\n disconnect(): Promise<void>;\n\n sessionDisconnect(): void;\n\n getRemoteTrackStats(mediaTrack: MediaStreamTrack): Promise<RemoteTrackStats>;\n\n onMediaStreamDestroyed(clientRole: SessionClientRole): Promise<void>;\n\n onMediaStreamRenewed(clientRole: SessionClientRole, mediaStream: MediaStream): Promise<void>;\n\n replaceStreamTracks(mediaStream: Nullable<MediaStream>): Promise<void>;\n\n registerEventCallback(event: string, cb: (eventArgs: any) => void): void;\n}\n\nexport interface RemoteMediaTrack {\n mediaTrack: MediaStreamTrack;\n trackType: KnownMediaStream;\n isRegistered: boolean;\n}\n\nexport interface SnapshotResult {\n base64img: string;\n objectUrl: string;\n imageBlob: Blob;\n mimeType: string;\n}\n\nexport interface SnapshotOptions {\n format?: string;\n quality?: number;\n x?: any;\n y?: any;\n w?: any;\n h?: any;\n}\n\nexport interface StreamDestroyedEventArgs {\n streamType: KnownMediaStream;\n reason: MediaStreamUnregisterReason;\n}\n\nexport interface StreamCreatedEventArgs {\n streamType: KnownMediaStream;\n}\n"]}
@@ -39,7 +39,7 @@ export declare abstract class TechseeMediaServiceBase {
39
39
  onRecordStarted(callback: (eventArgs: any) => void): void;
40
40
  onRecordStopped(callback: (eventArgs: any) => void): void;
41
41
  onReconnecting(callback: (eventArgs: any) => void): void;
42
- private getSwitchCameraConstraints;
42
+ getSwitchCameraConstraints(): LocalMediaConstraints;
43
43
  switchCamera(revertCameraWhenFailed?: boolean): Promise<any>;
44
44
  private replaceStreamTracks;
45
45
  protected registerStreamResult(constraints: Nullable<LocalMediaConstraints>, streamResult: MediaRequestSuccessResult, switchCamera?: boolean, addStreamType?: KnownMediaStream): Promise<any[]>;