-
-
Notifications
You must be signed in to change notification settings - Fork 428
/
Copy pathnative-image-cache.ts
116 lines (104 loc) · 3.6 KB
/
native-image-cache.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
import {
NativeImage,
nativeImage,
Size,
} from '@theia/core/electron-shared/electron';
import { Endpoint } from '@theia/core/lib/browser/endpoint';
import { FrontendApplicationContribution } from '@theia/core/lib/browser/frontend-application';
import { Deferred } from '@theia/core/lib/common/promise-util';
import { injectable } from '@theia/core/shared/inversify';
import fetch from 'cross-fetch';
const nativeImageIdentifierLiterals = ['cloud'] as const;
export type NativeImageIdentifier =
typeof nativeImageIdentifierLiterals[number];
export const nativeImages: Record<
NativeImageIdentifier,
string | { light: string; dark: string }
> = {
cloud: { light: 'cloud-light.png', dark: 'cloud-dark.png' },
};
export interface ThemeNativeImage {
readonly light: NativeImage;
readonly dark: NativeImage;
}
export function isThemeNativeImage(arg: unknown): arg is ThemeNativeImage {
return (
typeof arg === 'object' &&
(<ThemeNativeImage>arg).light !== undefined &&
(<ThemeNativeImage>arg).dark !== undefined
);
}
type Image = NativeImage | ThemeNativeImage;
@injectable()
export class NativeImageCache implements FrontendApplicationContribution {
private readonly cache = new Map<NativeImageIdentifier, Image>();
private readonly loading = new Map<NativeImageIdentifier, Promise<Image>>();
onStart(): void {
Object.keys(nativeImages).forEach((identifier: NativeImageIdentifier) =>
this.getImage(identifier)
);
}
tryGetImage(identifier: NativeImageIdentifier): Image | undefined {
return this.cache.get(identifier);
}
async getImage(identifier: NativeImageIdentifier): Promise<Image> {
const image = this.cache.get(identifier);
if (image) {
return image;
}
let loading = this.loading.get(identifier);
if (!loading) {
const deferred = new Deferred<Image>();
loading = deferred.promise;
this.loading.set(identifier, loading);
this.fetchImage(identifier).then(
(image) => {
if (!this.cache.has(identifier)) {
this.cache.set(identifier, image);
}
this.loading.delete(identifier);
deferred.resolve(image);
},
(err) => {
this.loading.delete(identifier);
deferred.reject(err);
}
);
}
return loading;
}
private async fetchImage(identifier: NativeImageIdentifier): Promise<Image> {
const value = nativeImages[identifier];
if (typeof value === 'string') {
return this.fetchIconData(value);
}
const [light, dark] = await Promise.all([
this.fetchIconData(value.light),
this.fetchIconData(value.dark),
]);
return { light, dark };
}
private async fetchIconData(filename: string): Promise<NativeImage> {
const path = `nativeImage/${filename}`;
const endpoint = new Endpoint({ path }).getRestUrl().toString();
const response = await fetch(endpoint);
const arrayBuffer = await response.arrayBuffer();
const view = new Uint8Array(arrayBuffer);
const buffer = Buffer.alloc(arrayBuffer.byteLength);
buffer.forEach((_, index) => (buffer[index] = view[index]));
const image = nativeImage.createFromBuffer(buffer);
return this.maybeResize(image);
}
private maybeResize(image: NativeImage): NativeImage {
const currentSize = image.getSize();
if (sizeEquals(currentSize, preferredSize)) {
return image;
}
return image.resize(preferredSize);
}
}
const pixel = 16;
const preferredSize: Size = { height: pixel, width: pixel };
function sizeEquals(left: Size, right: Size): boolean {
return left.height === right.height && left.width === right.width;
}