@galacean/engine-loader 1.1.0-alpha.0 → 1.1.0-alpha.2
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.
- package/dist/main.js +4843 -3713
- package/dist/main.js.map +1 -1
- package/dist/miniprogram.js +4842 -3712
- package/dist/module.js +4847 -3708
- package/dist/module.js.map +1 -1
- package/package.json +5 -5
- package/types/GLTFContentRestorer.d.ts +17 -7
- package/types/GLTFLoader.d.ts +2 -5
- package/types/gltf/GLTFResource.d.ts +2 -6
- package/types/gltf/GLTFSchema.d.ts +7 -1
- package/types/gltf/GLTFUtil.d.ts +3 -3
- package/types/gltf/GLTFUtils.d.ts +6 -4
- package/types/gltf/extensions/GLTFExtensionParser.d.ts +0 -6
- package/types/gltf/extensions/KHR_texture_basisu.d.ts +1 -0
- package/types/gltf/extensions/index.d.ts +1 -1
- package/types/gltf/index.d.ts +0 -1
- package/types/gltf/parser/GLTFAnimationParser.d.ts +3 -3
- package/types/gltf/parser/GLTFBufferParser.d.ts +2 -3
- package/types/gltf/parser/GLTFEntityParser.d.ts +2 -5
- package/types/gltf/parser/GLTFMaterialParser.d.ts +5 -2
- package/types/gltf/parser/GLTFMeshParser.d.ts +6 -6
- package/types/gltf/parser/GLTFParser.d.ts +3 -9
- package/types/gltf/parser/GLTFParserContext.d.ts +29 -23
- package/types/gltf/parser/GLTFSceneParser.d.ts +4 -3
- package/types/gltf/parser/GLTFSchemaParser.d.ts +7 -0
- package/types/gltf/parser/GLTFSkinParser.d.ts +3 -2
- package/types/gltf/parser/GLTFTextureParser.d.ts +9 -5
- package/types/gltf/parser/GLTFValidator.d.ts +1 -2
- package/types/gltf/parser/index.d.ts +2 -1
- package/types/index.d.ts +6 -3
- package/types/ktx2/BinomialLLCTranscoder/BinomialLLCTranscoder.d.ts +13 -0
- package/types/ktx2/BinomialLLCTranscoder/TranscodeWorkerCode.d.ts +33 -0
- package/types/ktx2/KTX2Container.d.ts +72 -0
- package/types/ktx2/KTX2Loader.d.ts +53 -0
- package/types/ktx2/KTX2TargetFormat.d.ts +21 -0
- package/types/ktx2/KhronosTranscoder/KhronosTranscoder.d.ts +17 -0
- package/types/ktx2/KhronosTranscoder/TranscoderWorkerCode.d.ts +34 -0
- package/types/ktx2/TranscodeResult.d.ts +10 -0
- package/types/ktx2/WorkerPool.d.ts +32 -0
- package/types/ktx2/constants.d.ts +7 -0
- package/types/ktx2/transcoder/AbstractTranscoder.d.ts +55 -0
- package/types/ktx2/transcoder/BinomialLLCTranscoder.d.ts +8 -0
- package/types/ktx2/transcoder/BinomialLLCWorkerCode.d.ts +2 -0
- package/types/ktx2/transcoder/KhronosTranscoder.d.ts +13 -0
- package/types/ktx2/transcoder/KhronosWorkerCode.d.ts +1 -0
- package/types/ktx2/zstddec.d.ts +62 -0
- package/types/resource-deserialize/resources/prefab/PrefabDesign.d.ts +4 -4
- package/types/resource-deserialize/utils/BufferReader.d.ts +6 -10
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { AssetPromise, Engine, EngineConfiguration, LoadItem, Loader, ResourceManager, Texture2D, TextureCube } from "@galacean/engine-core";
|
|
2
|
+
import { KTX2TargetFormat } from "./KTX2TargetFormat";
|
|
3
|
+
import { TranscodeResult } from "./transcoder/AbstractTranscoder";
|
|
4
|
+
export declare class KTX2Loader extends Loader<Texture2D | TextureCube> {
|
|
5
|
+
private static _isBinomialInit;
|
|
6
|
+
private static _binomialLLCTranscoder;
|
|
7
|
+
private static _khronosTranscoder;
|
|
8
|
+
private static _supportedMap;
|
|
9
|
+
/**
|
|
10
|
+
* Destroy ktx2 transcoder worker.
|
|
11
|
+
*/
|
|
12
|
+
static destroy(): void;
|
|
13
|
+
/** @internal */
|
|
14
|
+
static _parseBuffer(buffer: Uint8Array, engine: Engine, params?: KTX2Params): Promise<{
|
|
15
|
+
engine: Engine;
|
|
16
|
+
result: TranscodeResult;
|
|
17
|
+
targetFormat: KTX2TargetFormat;
|
|
18
|
+
params: Uint8Array;
|
|
19
|
+
}>;
|
|
20
|
+
/** @internal */
|
|
21
|
+
static _createTextureByBuffer(engine: Engine, transcodeResult: TranscodeResult, targetFormat: KTX2TargetFormat, params?: Uint8Array): Texture2D | TextureCube;
|
|
22
|
+
private static _decideTargetFormat;
|
|
23
|
+
private static _detectSupportedFormat;
|
|
24
|
+
private static _getBinomialLLCTranscoder;
|
|
25
|
+
private static _getKhronosTranscoder;
|
|
26
|
+
private static _getEngineTextureFormat;
|
|
27
|
+
initialize(engine: Engine, configuration: EngineConfiguration): Promise<void>;
|
|
28
|
+
/**
|
|
29
|
+
* @internal
|
|
30
|
+
*/
|
|
31
|
+
load(item: LoadItem & {
|
|
32
|
+
params?: KTX2Params;
|
|
33
|
+
}, resourceManager: ResourceManager): AssetPromise<Texture2D | TextureCube>;
|
|
34
|
+
private _isKhronosSupported;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* KTX2 loader params interface.
|
|
38
|
+
*/
|
|
39
|
+
export interface KTX2Params {
|
|
40
|
+
/** Priority transcoding format queue, default is ASTC/ETC/DXT/PVRTC/RGBA8. */
|
|
41
|
+
priorityFormats: KTX2TargetFormat[];
|
|
42
|
+
}
|
|
43
|
+
declare module "@galacean/engine-core" {
|
|
44
|
+
interface EngineConfiguration {
|
|
45
|
+
/** KTX2 loader options. */
|
|
46
|
+
ktx2Loader?: {
|
|
47
|
+
/** Worker count for transcoder, default is 4. */
|
|
48
|
+
workerCount?: number;
|
|
49
|
+
/** Pre-initialization according to the priority transcoding format queue, default is ASTC/ETC/DXT/PVRTC/RGBA8. */
|
|
50
|
+
priorityFormats?: KTX2TargetFormat[] | KTX2TargetFormat[][];
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* KTX2 transcode target format.
|
|
3
|
+
*/
|
|
4
|
+
export declare enum KTX2TargetFormat {
|
|
5
|
+
/** RGB(A) compressed format, 128 bits per 4x4 pixel block. */
|
|
6
|
+
ASTC = 0,
|
|
7
|
+
/** RGB(A) compressed format, 128 bits per 4x4 pixel block. */
|
|
8
|
+
BC7 = 1,
|
|
9
|
+
/** RGB(A) compressed format, 4 bits per pixel if no alpha channel, 8 bits per pixel if has alpha channel. */
|
|
10
|
+
BC1_BC3 = 2,
|
|
11
|
+
/** RGB(A) compressed format, 4 bits per pixel. */
|
|
12
|
+
PVRTC = 3,
|
|
13
|
+
/** RGB(A) compressed format, 4 bits per pixel if no alpha channel, 8 bits per pixel if has alpha channel. */
|
|
14
|
+
ETC = 4,
|
|
15
|
+
/** R format, 8 bits per pixel. */
|
|
16
|
+
R8 = 5,
|
|
17
|
+
/** RG format, 16 bits per pixel. */
|
|
18
|
+
R8G8 = 6,
|
|
19
|
+
/** RGBA format, 32 bits per pixel. */
|
|
20
|
+
R8G8B8A8 = 7
|
|
21
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { KTX2Container } from "../KTX2Container";
|
|
2
|
+
import { KTX2TargetFormat } from "../KTX2TargetFormat";
|
|
3
|
+
import { TranscodeResult } from "./TranscoderWorkerCode";
|
|
4
|
+
/** @internal */
|
|
5
|
+
export declare class KhronosTranscoder {
|
|
6
|
+
readonly workerLimit: number;
|
|
7
|
+
readonly type: KTX2TargetFormat;
|
|
8
|
+
static transcoderMap: {
|
|
9
|
+
0: string;
|
|
10
|
+
};
|
|
11
|
+
private _transcodeWorkerPool;
|
|
12
|
+
private _initPromise;
|
|
13
|
+
constructor(workerLimit: number, type: KTX2TargetFormat);
|
|
14
|
+
init(): Promise<any>;
|
|
15
|
+
transcode(ktx2Container: KTX2Container): Promise<TranscodeResult>;
|
|
16
|
+
destroy(): void;
|
|
17
|
+
}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
export interface EncodedData {
|
|
2
|
+
buffer: Uint8Array;
|
|
3
|
+
levelWidth: number;
|
|
4
|
+
levelHeight: number;
|
|
5
|
+
uncompressedByteLength: number;
|
|
6
|
+
}
|
|
7
|
+
type MessageType = "init" | "transcode";
|
|
8
|
+
export interface IBaseMessage {
|
|
9
|
+
type: MessageType;
|
|
10
|
+
}
|
|
11
|
+
export interface IInitMessage extends IBaseMessage {
|
|
12
|
+
type: "init";
|
|
13
|
+
transcoderWasm: ArrayBuffer;
|
|
14
|
+
}
|
|
15
|
+
export interface ITranscodeMessage extends IBaseMessage {
|
|
16
|
+
type: "transcode";
|
|
17
|
+
format: number;
|
|
18
|
+
needZstd: boolean;
|
|
19
|
+
data: EncodedData[][];
|
|
20
|
+
}
|
|
21
|
+
export type IMessage = IInitMessage | ITranscodeMessage;
|
|
22
|
+
export type TranscodeResult = {
|
|
23
|
+
data: Array<{
|
|
24
|
+
data: Uint8Array;
|
|
25
|
+
width: number;
|
|
26
|
+
height: number;
|
|
27
|
+
}>;
|
|
28
|
+
};
|
|
29
|
+
export type TranscodeResponse = {
|
|
30
|
+
id: number;
|
|
31
|
+
type: "transcoded";
|
|
32
|
+
} & TranscodeResult;
|
|
33
|
+
export declare function TranscodeWorkerCode(): void;
|
|
34
|
+
export {};
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @internal
|
|
3
|
+
* WorkerPool, T is is post message type, U is return type.
|
|
4
|
+
*/
|
|
5
|
+
export declare class WorkerPool<T = any, U = any> {
|
|
6
|
+
readonly limitedCount: number;
|
|
7
|
+
private readonly _workerCreator;
|
|
8
|
+
private _taskQueue;
|
|
9
|
+
private _workerStatus;
|
|
10
|
+
private _workerItems;
|
|
11
|
+
/**
|
|
12
|
+
* Constructor of WorkerPool.
|
|
13
|
+
* @param limitedCount - worker limit count
|
|
14
|
+
* @param _workerCreator - creator of worker
|
|
15
|
+
*/
|
|
16
|
+
constructor(limitedCount: number, _workerCreator: () => Worker | Promise<Worker>);
|
|
17
|
+
prepareWorker(): Promise<Worker[]>;
|
|
18
|
+
/**
|
|
19
|
+
* Post message to worker.
|
|
20
|
+
* @param message - Message which posted to worker
|
|
21
|
+
* @returns Return a promise of message
|
|
22
|
+
*/
|
|
23
|
+
postMessage(message: T): Promise<U>;
|
|
24
|
+
/**
|
|
25
|
+
* Destroy the worker pool.
|
|
26
|
+
*/
|
|
27
|
+
destroy(): void;
|
|
28
|
+
private _initWorker;
|
|
29
|
+
private _getIdleWorkerId;
|
|
30
|
+
private _onMessage;
|
|
31
|
+
private _nextTask;
|
|
32
|
+
}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
import { WorkerPool } from "../WorkerPool";
|
|
2
|
+
export declare abstract class AbstractTranscoder {
|
|
3
|
+
readonly workerLimitCount: number;
|
|
4
|
+
protected _transcodeWorkerPool: WorkerPool;
|
|
5
|
+
protected _initPromise: Promise<any>;
|
|
6
|
+
constructor(workerLimitCount: number);
|
|
7
|
+
init(): Promise<any>;
|
|
8
|
+
destroy(): void;
|
|
9
|
+
protected abstract _initTranscodeWorkerPool(): Promise<any>;
|
|
10
|
+
protected _createTranscodePool(workerURL: string, wasmBuffer: ArrayBuffer): Promise<Worker[]>;
|
|
11
|
+
}
|
|
12
|
+
type MessageType = "init" | "transcode";
|
|
13
|
+
export interface BaseMessage {
|
|
14
|
+
type: MessageType;
|
|
15
|
+
}
|
|
16
|
+
export interface InitMessage extends BaseMessage {
|
|
17
|
+
type: "init";
|
|
18
|
+
transcoderWasm: ArrayBuffer;
|
|
19
|
+
}
|
|
20
|
+
export interface BinomialTranscodeMessage extends BaseMessage {
|
|
21
|
+
type: "transcode";
|
|
22
|
+
format: number;
|
|
23
|
+
buffer: ArrayBuffer;
|
|
24
|
+
}
|
|
25
|
+
export type IBinomialMessage = InitMessage | BinomialTranscodeMessage;
|
|
26
|
+
export type TranscodeResult = {
|
|
27
|
+
width: number;
|
|
28
|
+
height: number;
|
|
29
|
+
hasAlpha: boolean;
|
|
30
|
+
format: number;
|
|
31
|
+
faces: Array<{
|
|
32
|
+
data: Uint8Array;
|
|
33
|
+
width: number;
|
|
34
|
+
height: number;
|
|
35
|
+
}>[];
|
|
36
|
+
faceCount: number;
|
|
37
|
+
};
|
|
38
|
+
export type TranscodeResponse = {
|
|
39
|
+
id: number;
|
|
40
|
+
type: "transcoded";
|
|
41
|
+
} & TranscodeResult;
|
|
42
|
+
export interface EncodedData {
|
|
43
|
+
buffer: Uint8Array;
|
|
44
|
+
levelWidth: number;
|
|
45
|
+
levelHeight: number;
|
|
46
|
+
uncompressedByteLength: number;
|
|
47
|
+
}
|
|
48
|
+
export interface KhronosTranscoderMessage extends BaseMessage {
|
|
49
|
+
type: "transcode";
|
|
50
|
+
format: number;
|
|
51
|
+
needZstd: boolean;
|
|
52
|
+
data: EncodedData[][];
|
|
53
|
+
}
|
|
54
|
+
export type IKhronosMessageMessage = InitMessage | KhronosTranscoderMessage;
|
|
55
|
+
export {};
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import { KTX2TargetFormat } from "../KTX2TargetFormat";
|
|
2
|
+
import { AbstractTranscoder, TranscodeResult } from "./AbstractTranscoder";
|
|
3
|
+
/** @internal */
|
|
4
|
+
export declare class BinomialLLCTranscoder extends AbstractTranscoder {
|
|
5
|
+
constructor(workerLimitCount: number);
|
|
6
|
+
_initTranscodeWorkerPool(): Promise<Worker[]>;
|
|
7
|
+
transcode(buffer: ArrayBuffer, format: KTX2TargetFormat): Promise<TranscodeResult>;
|
|
8
|
+
}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { KTX2Container } from "../KTX2Container";
|
|
2
|
+
import { KTX2TargetFormat } from "../KTX2TargetFormat";
|
|
3
|
+
import { AbstractTranscoder, TranscodeResult } from "./AbstractTranscoder";
|
|
4
|
+
/** @internal */
|
|
5
|
+
export declare class KhronosTranscoder extends AbstractTranscoder {
|
|
6
|
+
readonly type: KTX2TargetFormat;
|
|
7
|
+
static transcoderMap: {
|
|
8
|
+
0: string;
|
|
9
|
+
};
|
|
10
|
+
constructor(workerLimitCount: number, type: KTX2TargetFormat);
|
|
11
|
+
_initTranscodeWorkerPool(): Promise<Worker[]>;
|
|
12
|
+
transcode(ktx2Container: KTX2Container): Promise<TranscodeResult>;
|
|
13
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function TranscodeWorkerCode(): void;
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* From https://github.com/donmccurdy/zstddec by Don McCurdy
|
|
3
|
+
*/
|
|
4
|
+
interface DecoderExports {
|
|
5
|
+
memory: Uint8Array;
|
|
6
|
+
ZSTD_findDecompressedSize: (compressedPtr: number, compressedSize: number) => number;
|
|
7
|
+
ZSTD_decompress: (uncompressedPtr: number, uncompressedSize: number, compressedPtr: number, compressedSize: number) => number;
|
|
8
|
+
malloc: (ptr: number) => number;
|
|
9
|
+
free: (ptr: number) => void;
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* ZSTD (Zstandard) decoder.
|
|
13
|
+
*/
|
|
14
|
+
export declare class ZSTDDecoder {
|
|
15
|
+
static heap: Uint8Array;
|
|
16
|
+
static IMPORT_OBJECT: {
|
|
17
|
+
env: {
|
|
18
|
+
emscripten_notify_memory_growth: () => void;
|
|
19
|
+
};
|
|
20
|
+
};
|
|
21
|
+
static instance: {
|
|
22
|
+
exports: DecoderExports;
|
|
23
|
+
};
|
|
24
|
+
static WasmModuleURL: string;
|
|
25
|
+
_initPromise: Promise<any>;
|
|
26
|
+
init(): Promise<void>;
|
|
27
|
+
_init(result: WebAssembly.WebAssemblyInstantiatedSource): void;
|
|
28
|
+
decode(array: Uint8Array, uncompressedSize?: number): Uint8Array;
|
|
29
|
+
}
|
|
30
|
+
export {};
|
|
31
|
+
/**
|
|
32
|
+
* BSD License
|
|
33
|
+
*
|
|
34
|
+
* For Zstandard software
|
|
35
|
+
*
|
|
36
|
+
* Copyright (c) 2016-present, Yann Collet, Facebook, Inc. All rights reserved.
|
|
37
|
+
*
|
|
38
|
+
* Redistribution and use in source and binary forms, with or without modification,
|
|
39
|
+
* are permitted provided that the following conditions are met:
|
|
40
|
+
*
|
|
41
|
+
* * Redistributions of source code must retain the above copyright notice, this
|
|
42
|
+
* list of conditions and the following disclaimer.
|
|
43
|
+
*
|
|
44
|
+
* * Redistributions in binary form must reproduce the above copyright notice,
|
|
45
|
+
* this list of conditions and the following disclaimer in the documentation
|
|
46
|
+
* and/or other materials provided with the distribution.
|
|
47
|
+
*
|
|
48
|
+
* * Neither the name Facebook nor the names of its contributors may be used to
|
|
49
|
+
* endorse or promote products derived from this software without specific
|
|
50
|
+
* prior written permission.
|
|
51
|
+
*
|
|
52
|
+
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
53
|
+
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
54
|
+
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
55
|
+
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
|
|
56
|
+
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
57
|
+
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
58
|
+
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
59
|
+
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
60
|
+
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
61
|
+
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
62
|
+
*/
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import type { BackgroundMode } from "@galacean/engine-core";
|
|
2
|
-
import {
|
|
2
|
+
import { IRefObject } from "@galacean/engine-core/types/asset/IRefObject";
|
|
3
3
|
import { IColor } from "../mesh/IModelMesh";
|
|
4
4
|
export interface IPrefabFile {
|
|
5
5
|
entities: Array<IEntity>;
|
|
@@ -9,11 +9,11 @@ export interface IScene extends IPrefabFile {
|
|
|
9
9
|
background: {
|
|
10
10
|
mode: BackgroundMode;
|
|
11
11
|
color: IColor;
|
|
12
|
-
texture?:
|
|
13
|
-
sky?:
|
|
12
|
+
texture?: IRefObject;
|
|
13
|
+
sky?: IRefObject;
|
|
14
14
|
};
|
|
15
15
|
ambient: {
|
|
16
|
-
ambientLight:
|
|
16
|
+
ambientLight: IRefObject;
|
|
17
17
|
diffuseSolidColor: IColor;
|
|
18
18
|
diffuseIntensity: number;
|
|
19
19
|
specularIntensity: number;
|
|
@@ -1,15 +1,11 @@
|
|
|
1
1
|
export declare class BufferReader {
|
|
2
|
-
|
|
2
|
+
data: Uint8Array;
|
|
3
3
|
private _dataView;
|
|
4
4
|
private _littleEndian;
|
|
5
5
|
private _offset;
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
2: string;
|
|
10
|
-
3: string;
|
|
11
|
-
};
|
|
12
|
-
constructor(buffer: ArrayBuffer, byteOffset?: number, byteLength?: number, littleEndian?: boolean);
|
|
6
|
+
private _baseOffset;
|
|
7
|
+
constructor(data: Uint8Array, byteOffset?: number, byteLength?: number, littleEndian?: boolean);
|
|
8
|
+
get position(): number;
|
|
13
9
|
get offset(): number;
|
|
14
10
|
nextUint8(): number;
|
|
15
11
|
nextUint16(): number;
|
|
@@ -25,8 +21,8 @@ export declare class BufferReader {
|
|
|
25
21
|
/**
|
|
26
22
|
* image data 放在最后
|
|
27
23
|
*/
|
|
28
|
-
nextImageData(count?: number):
|
|
29
|
-
nextImagesData(count: number):
|
|
24
|
+
nextImageData(count?: number): Uint8Array;
|
|
25
|
+
nextImagesData(count: number): Uint8Array[];
|
|
30
26
|
skip(bytes: number): this;
|
|
31
27
|
scan(maxByteLength: number, term?: number): Uint8Array;
|
|
32
28
|
}
|