/* * Copyright (c) 2022 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ let flag = false; let fastQueue = undefined; let arkPritvate = globalThis["ArkPrivate"] || undefined; if (arkPritvate !== undefined) { fastQueue = arkPritvate.Load(arkPritvate.Queue); } else { flag = true; } if (flag || fastQueue == undefined) { class HandlerQueue { private isOutBounds(obj: Queue, prop: any) { let index = Number.parseInt(prop); if (Number.isInteger(index)) { if (index < 0 || index > obj.length) { console.log(index, obj.length) throw new RangeError("the index is out-of-bounds"); } } } get(obj: Queue, prop: any): T { if (typeof prop === "symbol") { return obj[prop]; } this.isOutBounds(obj, prop); return obj[prop]; } set(obj: Queue, prop: any, value: T): boolean { if (prop === "front" || prop === "capacity" || prop === "rear") { obj[prop] = value; return true; } this.isOutBounds(obj, prop); let index = Number(prop); if (index >= 0 && index <= obj.length && Number.isInteger(index)) { obj[index] = value; return true; } return false; } ownKeys(obj: Queue) { let keys = []; for (let i = 0; i < obj.length; i++) { keys.push(i.toString()); } return keys; } defineProperty(obj: Queue, prop: any, desc: any) { return true; } getOwnPropertyDescriptor(obj: Queue, prop: any) { this.isOutBounds(obj, prop); let index = Number.parseInt(prop); if (index >= 0 && index < obj.length && Number.isInteger(index)) { return Object.getOwnPropertyDescriptor(obj, prop); } return } setPrototypeOf(obj: any, prop: any): any { throw new RangeError("Can setPrototype on Queue Object"); } } interface IterableIterator { next: () => { value: T; done: boolean; }; } class Queue { private front: number; private capacity: number; private rear: number; constructor() { this.front = 0; this.capacity = 8; this.rear = 0; return new Proxy(this, new HandlerQueue()); } get length(){ return this.rear - this.front; } add(element: T): boolean { if (this.isFull()) { this.increaseCapacity(); } this[this.rear] = element; this.rear = (this.rear + 1) % (this.capacity + 1); return true; } getFirst(): T { if (this.isEmpty()) { return undefined; } return this[this.front]; } pop(): T { if (this.isEmpty()) { return undefined; } let result = this[this.front]; this.front = (this.front + 1) % (this.capacity + 1); return result; } forEach(callbackfn: (value: T, index?: number, queue?: Queue) => void, thisArg?: Object): void { let k = 0; let i = this.front; if (this.isEmpty()) { return; } else { while (true) { callbackfn.call(thisArg,this[i], k,this); i = (i + 1) % this.capacity; k++; if (i === this.rear) { break; } } } } private isFull(): boolean { return this.length === this.capacity; } private isEmpty(): boolean { return this.length == 0; } [Symbol.iterator](): IterableIterator { let count = this.front; let queue = this; return { next: function () { let done = count == queue.rear; let value = !done ? queue[count] : undefined; count = (count + 1) % queue.capacity; return { done: done, value: value, }; }, }; } private increaseCapacity(): void { this.capacity = 2 * this.capacity; } } Object.freeze(Queue); fastQueue = Queue; } export default fastQueue;