forked from railsware/upterm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHistory.ts
110 lines (84 loc) · 2.72 KB
/
History.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
import {lex} from "./CommandExpander";
import * as _ from "lodash";
export class HistoryEntry {
private _raw: string;
constructor(raw: string, private _historyExpanded: string[]) {
this._raw = raw.trim().replace(/\s+/g, " ");
}
get raw(): string {
return this._raw;
}
get historyExpanded(): string[] {
return this._historyExpanded;
}
get lastLexeme(): string {
return _.last(lex(this.raw));
}
toArray(): any[] {
return [this.raw, this.historyExpanded];
}
}
export class History {
static pointer: number = 0;
private static maxEntriesCount: number = 100;
private static storage: HistoryEntry[] = [];
private static defaultEntry: HistoryEntry = new HistoryEntry("", []);
static get all(): HistoryEntry[] {
return this.storage;
}
static get lastEntry(): HistoryEntry {
return this.at(-1);
}
static lastWithPrefix(prefix: string): HistoryEntry {
return this.storage.find(entry => entry.raw.startsWith(prefix)) || this.defaultEntry;
}
static at(position: number): HistoryEntry {
if (position === 0) {
return this.defaultEntry;
}
if (position < 0) {
return this.storage[-(position + 1)] || this.defaultEntry;
}
return this.storage[this.count - 1] || this.defaultEntry;
}
static add(entry: HistoryEntry): void {
this.remove(entry);
this.storage.unshift(entry);
if (this.count > this.maxEntriesCount) {
this.storage.splice(this.maxEntriesCount - 1);
}
this.pointer = 0;
}
static getPrevious(): string {
if (this.pointer < this.count) {
this.pointer += 1;
}
return this.at(-this.pointer).raw;
}
static getNext(): string {
if (this.pointer > 0) {
this.pointer -= 1;
}
return this.at(-this.pointer).raw;
}
private static get count(): number {
return this.storage.length;
}
static serialize(): string {
return `History:${JSON.stringify(History.storage.map(entry => entry.toArray()))}`;
}
static deserialize(serialized: string): void {
this.storage = JSON.parse(serialized).map((entry: any[]) => {
let raw: string = entry[0];
let historyExpanded: string[] = entry[1];
return new HistoryEntry(raw, historyExpanded);
});
}
private static remove(entry: HistoryEntry): void {
const duplicateIndex = this.storage.findIndex(stackedEntry => stackedEntry.raw === entry.raw);
if (duplicateIndex !== -1) {
this.storage.splice(duplicateIndex, 1);
}
}
}
export default History;