As a Linux and Mac OS user (mostly Mac), I spend a lot of time at the terminal. While there are several amazing terminal emulators available such as Hyper and iTerm2, I wanted to build my own.
@angular/cli
installed
on your system. Please see Angular-cli documentation for further assistance.`git clone https://github.com/payneusmc07/ng-xterm.git
yarn install
Node-pty (what is used by ng-xterm to spawn terminal sessions) is what is known as a Native Module. This means it is written in C/C++ and complied for use with Node/Electron. Native Modules can be very finicky, and Node-pty is notorious for causing development headaches.
If you try to build/run ng-xterm and you see errors in the developer tools which say something along the lines of:
Cannot find file "../Debug/node-pty.node"
then node-pty needs to be rebuild for the current version of Node/Electron you are using. There are two scripts in package.json to help accomplish this.
npm rebuild node-pty --update-binaries
The included .run folder contains pre-configured scripts which you should be able to load and run without any configuration. These scripts contain the needed environment variables to develop NG-Xterm.
The included .run folder also contains VSCode tasks for running in development and building NG-Xterm for production.
You will need to set your environment variables manually.
yarn init
Please make sure to set the following environment variables if you are not using the included scripts.
yarn dev
yarn build:prod
Note: Please make sure you have run build:prod as Electron will look for the static index.html file.
yarn build:electron-only
please make sure to set the following environment variable if you are not using the included scripts.
yarn build:electron-only:all
please make sure to set the following environment variable if you are not using the included scripts.
yarn build:final:all-platforms:no-publish
yarn build:final:all-platforms:publis
yarn compodoc
It has always been a pet peeve of mine when a program or code base is not properly documented (you know who you are).
With this in mind, I know some of you may feel the amount of documentation/comments I included is a bit excessive, but I would rather have "too much" and sleep better knowing everything is properly documented.
compodoc
package.Those of you familiar with Objective C/Swift background know how "verbose" the function names/parameters can be. And seeing as I am also an Objective C/Swift, I brought that convention over with me, mostly out of habit, but I feel it makes it easier to understand what the code is doing since some of its functionality is described in the function/method name.
Due to the nature of the new Angular compiler and rendering system (Ivy), variables that will be used in the html template cannot be marked as private, since they cannot be resolved by the compiler. An example of this can be seen in the code snippet from the app-root component. Since the class member idFromTab: number | string is used in the html template, it must remain public.
/** The id of the currently selected tab in the tabs component. */
idFromTab: number | string
/**
* Angular will throw the following error if idFromTab
* is made private.
*
* Private field idFromTab cannot be resolved from the
* component template when using the AOT compiler
* */
private idFromTab: number | string
<!-- Since tabID is actively used in the template, it MUST be public -->
<ng-xterm-app-topbar [tabID]="idFromTab"></ng-xterm-app-topbar>
<ng-xterm-tabs (tabID)="setID($event)"></ng-xterm-tabs>
Why are public classes/functions not explicitly marked as public?
By default, methods and classes are public in Typescript, so I did not feel the need to repeat the obvious. Aside from that, when you are working with classes which have a lot of memeber methods, it can become easy to mistake public for private if you are not paying attention.
Why is eslint not installed?
Seeing as NG-Xterm was developed soley by me, I did not feel adding something which monitors my already consistent coding style was needed.
The code snippet below was taken from the ng-xterm-tab-settings component. Since Reactive forms in Angular can be a bit confusing, I thought I would provide a bit of a break down as to how the various forms were configured, and how each setting is retrieved from the terminal settings store.
Please note, not all comments or functions will be a complete refection of the actual code. This page is meant as more of a walkthrough/demo.
import { Component, OnInit } from "@angular/core"
import { FormControl, FormGroup } from "@angular/forms"
import { SettingsService } from "@shared/services"
import { Settings } from "@shared/utils"
@Component({
selector: "ng-xterm-tab-settings",
templateUrl: "./tab-settings.component.html",
styleUrls: ["./tab-settings.component.scss"]
})
/** The form which modifies any settings related to the ng-xterm-tab-settings component */
export class TabSettingsComponent implements OnInit {
/** Define the basic form which modifies any settings related to the ng-xterm-tab-settings component */
tabSettingsForm: FormGroup
/**
* A string value which will represent the preferred background color
* for the UI tabs. This value will be retrieved from the settings store.
* */
tabBackgroundColor: string
/** By "declaring" the various services used by this component in the constructor,
* we allow for the Angular dependency injection (DI) mechanisms to properly configure
* and instantiate the service class(es) for us. When using DI, we DO NOT instantiate
* the service class(es) with the "new" keyword.
* */
constructor(private readonly settingsService: SettingsService) {}
/**
* Since Electron Store returns an "unknown" value, each retrieved settings
* must be type cast to a specific value. Otherwise, Typescript will throw error code
* TS2322: Type 'unknown' is not assignable to type '[stored value]'.
* */
ngOnInit() {
// DO NOT DO THIS!
this.settingsService = new SettingsService()
// retrieve the background color from the settings store. and type cast it to a string
this.tabBackgroundColor = this.settingsService.getItem( Settings.TAB_BG_COLOR) as string
this.tabSettingsForm = new FormGroup({
// the name of the form control: instantiate a new FormControl with an optional default value
"tab-background": new FormControl(this.tabBackgroundColor),
})
}
}
Vercel for giving us Hyper, which served as a large inspiration for this project.
maximegris of Github for the awesome boilerplate/starting template
Google for the Angular platform.
And the Electron team for making such a powerful yet easy to use platform.