forked from railsware/upterm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSession.ts
129 lines (106 loc) · 4.08 KB
/
Session.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
117
118
119
120
121
122
123
124
125
126
127
128
129
import {readFileSync, writeFile} from "fs";
import * as _ from "lodash";
import Job from "./Job";
import History from "./History";
import Serializer from "./Serializer";
import EmitterWithUniqueID from "./EmitterWithUniqueID";
import PluginManager from "./PluginManager";
import {Status} from "./Enums";
import ApplicationComponent from "./views/1_ApplicationComponent";
import Environment from "./Environment";
import {homeDirectory, normalizeDirectory} from "./utils/Common";
import {remote} from "electron";
export default class Session extends EmitterWithUniqueID {
jobs: Array<Job> = [];
readonly environment = new Environment();
history: typeof History;
historicalCurrentDirectoriesStack: string[] = [];
private readonly stateFileName = `${homeDirectory()}/.black-screen-state`;
// The value of the dictionary is the default value used if there is no serialized data.
private readonly serializableProperties: Dictionary<string> = {
directory: `String:${homeDirectory()}`,
history: `History:[]`,
};
constructor(private application: ApplicationComponent, private _dimensions: Dimensions) {
super();
// TODO: We want to deserialize properties only for the first instance
// TODO: of Session for the application.
this.deserialize();
this.history = History;
this.on("job", this.serialize);
this.clearJobs();
}
createJob(): void {
const job = new Job(this);
job.once("end", () => {
if (remote.app.dock && !remote.BrowserWindow.getAllWindows().some(window => window.isFocused())) {
remote.app.dock.bounce("informational");
remote.app.dock.setBadge(job.status === Status.Success ? "1" : "✕");
/* tslint:disable:no-unused-expression */
new Notification("Command has been completed", { body: job.prompt.value });
}
this.createJob();
});
this.jobs = this.jobs.concat(job);
this.emit("job");
}
get dimensions(): Dimensions {
return this._dimensions;
}
set dimensions(value: Dimensions) {
this._dimensions = value;
this.jobs.forEach(job => job.winch());
}
clearJobs(): void {
this.jobs = [];
this.createJob();
}
close(): void {
this.application.closeSession(this);
}
get directory(): string {
return this.environment.pwd;
}
set directory(value: string) {
let normalizedDirectory = normalizeDirectory(value);
if (normalizedDirectory === this.directory) {
return;
}
PluginManager.environmentObservers.forEach(observer =>
observer.currentWorkingDirectoryWillChange(this, normalizedDirectory)
);
this.environment.pwd = normalizedDirectory;
this.historicalCurrentDirectoriesStack.push(normalizedDirectory);
PluginManager.environmentObservers.forEach(observer =>
observer.currentWorkingDirectoryDidChange(this, normalizedDirectory)
);
}
private serialize = () => {
let values: Dictionary<string> = {};
_.each(this.serializableProperties, (value: string, key: string) =>
values[key] = Serializer.serialize((<any>this)[key])
);
writeFile(this.stateFileName, JSON.stringify(values), (error: any) => {
if (error) throw error;
});
};
private deserialize(): void {
_.each(this.readSerialized(), (value: string, key: string) => {
const setterName = `set${_.capitalize(key)}`;
const that = (<any>this);
const deserializedValue = Serializer.deserialize(value);
if (that[setterName]) {
that[setterName](deserializedValue);
} else {
that[key] = deserializedValue;
}
});
}
private readSerialized(): Dictionary<any> {
try {
return JSON.parse(readFileSync(this.stateFileName).toString());
} catch (error) {
return this.serializableProperties;
}
};
}