Skip to content

Instantly share code, notes, and snippets.

@kukiron
Created June 25, 2018 04:14
Show Gist options
  • Save kukiron/8027c55f197feec57f9064813e5b4e5b to your computer and use it in GitHub Desktop.
Save kukiron/8027c55f197feec57f9064813e5b4e5b to your computer and use it in GitHub Desktop.
Settings & other plugin customization files for Hyper
// Future versions of Hyper may add additional config options,
// which will not automatically be merged into this file.
// See https://hyper.is#cfg for all currently supported options.
module.exports = {
config: {
// choose either `'stable'` for receiving highly polished,
// or `'canary'` for less polished but more frequent updates
updateChannel: 'stable',
// default font size in pixels for all tabs
fontSize: 12,
// font family with optional fallbacks
fontFamily: 'Fira Code Medium, Consolas, monospace',
// default font weight: 'normal' or 'bold'
fontWeight: 'normal',
// font weight for bold characters: 'normal' or 'bold'
fontWeightBold: 'bold',
// terminal cursor background color and opacity (hex, rgb, hsl, hsv, hwb or cmyk)
cursorColor: '#FFC600',
// terminal text color under BLOCK cursor
cursorAccentColor: '#000',
// `'BEAM'` for |, `'UNDERLINE'` for _, `'BLOCK'` for █
cursorShape: 'BLOCK',
// set to `true` (without backticks and without quotes) for blinking cursor
cursorBlink: true,
// color of the text
foregroundColor: '#fff',
// terminal background color
// opacity is only supported on macOS
backgroundColor: '#000',
// terminal selection color
selectionColor: 'rgba(76,175,80,.3)',
// border color (window, tabs)
borderColor: '#333',
// custom CSS to embed in the main window
css: '',
// custom CSS to embed in the terminal window
termCSS: '',
// if you're using a Linux setup which show native menus, set to false
// default: `true` on Linux, `true` on Windows, ignored on macOS
showHamburgerMenu: '',
// set to `false` (without backticks and without quotes) if you want to hide the minimize, maximize and close buttons
// additionally, set to `'left'` if you want them on the left, like in Ubuntu
// default: `true` (without backticks and without quotes) on Windows and Linux, ignored on macOS
showWindowControls: '',
// custom padding (CSS format, i.e.: `top right bottom left`)
padding: '12px 14px',
// the full list. if you're going to provide the full color palette,
// including the 6 x 6 color cubes and the grayscale map, just provide
// an array here instead of a color map object
colors: {
black: '#000000',
red: '#C51E14',
green: '#1DC121',
yellow: '#C7C329',
blue: '#0A2FC4',
magenta: '#C839C5',
cyan: '#20C5C6',
white: '#C7C7C7',
lightBlack: '#686868',
lightRed: '#FD6F6B',
lightGreen: '#67F86F',
lightYellow: '#FFFA72',
lightBlue: '#6A76FB',
lightMagenta: '#FD7CFC',
lightCyan: '#68FDFE',
lightWhite: '#FFFFFF',
},
// the shell to run when spawning a new session (i.e. /usr/local/bin/fish)
// if left empty, your system's login shell will be used by default
//
// Windows
// - Make sure to use a full path if the binary name doesn't work
// - Remove `--login` in shellArgs
//
// Bash on Windows
// - Example: `C:\\Windows\\System32\\bash.exe`
//
// PowerShell on Windows
// - Example: `C:\\WINDOWS\\System32\\WindowsPowerShell\\v1.0\\powershell.exe`
shell: 'cmd.exe',
// for setting shell arguments (i.e. for using interactive shellArgs: `['-i']`)
// by default `['--login']` will be used
shellArgs: ['/k', 'C:\\Users\\kukiron\\vendor\\init.bat'],
// for environment variables
env: { 'TERM': 'cygwin' }, // required for git-for-windows
// set to `false` for no bell
bell: 'SOUND',
// if `true` (without backticks and without quotes), selected text will automatically be copied to the clipboard
copyOnSelect: false,
// if `true` (without backticks and without quotes), hyper will be set as the default protocol client for SSH
defaultSSHApp: true,
// if `true` (without backticks and without quotes), on right click selected text will be copied or pasted if no
// selection is present (`true` by default on Windows and disables the context menu feature)
// quickEdit: true,
// URL to custom bell
// bellSoundURL: 'http://example.com/bell.mp3',
// for advanced config flags please refer to https://hyper.is/#cfg
},
// a list of plugins to fetch and install from npm
// format: [@org/]project[#version]
// examples:
// `hyperpower`
// `@company/project`
// `project#1.0.1`
plugins: ["hyper-chesterish", "hyper-statusline", "hyper-tabs-enhanced"],
/** Custom Settings STARTS */
hyperTabs: {
trafficButtons: true,
border: true,
closeAlign: 'right',
tabIconsColored: true,
activityColor: 'yellow'
},
hyperStatusLine: {
dirtyColor: 'salmon',
aheadColor: 'ivory',
footerTransparent: false,
},
/** Custom Settings ENDS */
// in development, you can create a directory under
// `~/.hyper_plugins/local/` and include it here
// to load it and avoid it being `npm install`ed
localPlugins: [],
keymaps: {
// Example
// 'window:devtools': 'cmd+alt+o',
},
};
// Syntax scheme
// const backgroundColor = '#09202B';
const backgroundColor = "#19242F";
const foregroundColor = '#F1F1F1';
const cursorColor = '#FFC600';
const borderColor = '#323E4D';
// Config
exports.decorateConfig = config => {
return Object.assign({}, config, {
foregroundColor,
backgroundColor,
borderColor,
// colors,
cursorColor: config.cursorColor || cursorColor,
cursorShape: config.cursorShape || 'BEAM',
fontSize: config.fontSize || 12,
fontFamily: config.fontFamily || '"Fira Code"',
termCSS: `
${config.termCSS || ''}
::selection {
background: #00FF00 !important;
}
x-screen x-row {
font-variant-ligatures: initial;
}
span {
font-weight: normal !important;
}
`,
css: `
${config.css || ''}
::selection {
background: #00FF00 !important;
}
`
});
};
const { shell } = require('electron');
const { exec } = require('child_process');
const color = require('color');
const afterAll = require('after-all-results');
const tildify = require('tildify');
exports.decorateConfig = (config) => {
const colorForeground = color(config.foregroundColor || '#fff');
// const colorBackground = color(config.backgroundColor || '#000');
const colorBackground = color('#15232D');
const colors = {
foreground: colorForeground.string(),
background: colorBackground.lighten(0.3).string()
};
const configColors = Object.assign({
black: '#000000',
red: '#ff0000',
green: '#33ff00',
yellow: '#ffff00',
blue: '#0066ff',
magenta: '#cc00ff',
cyan: '#00ffff',
white: '#d0d0d0',
lightBlack: '#808080',
lightRed: '#ff0000',
lightGreen: '#33ff00',
lightYellow: '#ffff00',
lightBlue: '#0066ff',
lightMagenta: '#cc00ff',
lightCyan: '#00ffff',
lightWhite: '#ffffff'
}, config.colors);
const hyperStatusLine = Object.assign({
footerTransparent: true,
dirtyColor: configColors.lightYellow,
aheadColor: configColors.blue
}, config.hyperStatusLine);
return Object.assign({}, config, {
css: `
${config.css || ''}
.terms_terms {
margin-bottom: 30px;
}
.footer_footer {
display: flex;
justify-content: space-between;
position: absolute;
bottom: 0;
left: 0;
right: 0;
z-index: 100;
font-size: 12px;
height: 30px;
font-family: 'Lekton';
border-top: 1px solid #0B3B5E;
background-color: ${colors.background};
opacity: ${hyperStatusLine.footerTransparent ? '0.5' : '1'};
cursor: default;
-webkit-user-select: none;
transition: opacity 250ms ease;
}
.footer_footer:hover {
opacity: 1;
}
.footer_footer .footer_group {
display: flex;
color: ${colors.foreground};
white-space: nowrap;
margin: 0 14px;
}
.footer_footer .group_overflow {
overflow: hidden;
}
.footer_footer .component_component {
display: flex;
}
.footer_footer .component_item {
position: relative;
line-height: 30px;
margin-left: 9px;
}
.footer_footer .component_item:first-of-type {
margin-left: 0;
}
.footer_footer .item_clickable:hover {
text-decoration: underline;
cursor: pointer;
}
.footer_footer .item_icon:before {
content: '';
position: absolute;
top: 0;
left: 0;
padding-top: 8px;
width: 12px;
height: 60%;
-webkit-mask-repeat: no-repeat;
-webkit-mask-position: 0 center;
background-color: ${colors.foreground};
}
.footer_footer .item_number {
font-size: 10.5px;
font-weight: 500;
}
.footer_footer .item_cwd {
padding-left: 21px;
}
.footer_footer .item_cwd:before {
-webkit-mask-image: url('');
-webkit-mask-size: 14px 12px;
}
.footer_footer .item_branch {
padding-left: 16px;
}
.footer_footer .item_branch:before {
-webkit-mask-image: url('');
-webkit-mask-size: 9px 12px;
}
.footer_footer .item_dirty {
color: ${hyperStatusLine.dirtyColor};
padding-left: 16px;
}
.footer_footer .item_dirty:before {
-webkit-mask-image: url('');
-webkit-mask-size: 12px 12px;
background-color: ${hyperStatusLine.dirtyColor};
}
.footer_footer .item_ahead {
color: ${hyperStatusLine.aheadColor};
padding-left: 16px;
}
.footer_footer .item_ahead:before {
-webkit-mask-image: url('');
-webkit-mask-size: 12px 12px;
background-color: ${hyperStatusLine.aheadColor};
}
.notifications_view {
bottom: 50px;
}
`
});
};
let pid;
let cwd;
let git = {
branch: '',
remote: '',
dirty: 0,
ahead: 0
}
const setCwd = (pid, action) => {
if (process.platform == 'win32') {
let directoryRegex = /([a-zA-Z]:[^\:\[\]\?\"\<\>\|]+)/mi;
if (action && action.data) {
let path = directoryRegex.exec(action.data);
if (path) {
cwd = path[0];
setGit(cwd);
}
}
} else {
exec(`lsof -p ${pid} | awk '$4=="cwd"' | tr -s ' ' | cut -d ' ' -f9-`, (err, stdout) => {
cwd = stdout.trim();
setGit(cwd);
});
}
};
const isGit = (dir, cb) => {
exec(`git rev-parse --is-inside-work-tree`, { cwd: dir }, (err) => {
cb(!err);
});
}
const gitBranch = (repo, cb) => {
exec(`git symbolic-ref --short HEAD || git rev-parse --short HEAD`, { cwd: repo }, (err, stdout) => {
if (err) {
return cb(err);
}
cb(null, stdout.trim());
});
}
const gitRemote = (repo, cb) => {
exec(`git ls-remote --get-url`, { cwd: repo }, (err, stdout) => {
cb(null, stdout.trim().replace(/^git@(.*?):/, 'https://$1/').replace(/[A-z0-9\-]+@/, '').replace(/\.git$/, ''));
});
}
const gitDirty = (repo, cb) => {
exec(`git status --porcelain --ignore-submodules -uno`, { cwd: repo }, (err, stdout) => {
if (err) {
return cb(err);
}
cb(null, !stdout ? 0 : parseInt(stdout.trim().split('\n').length, 10));
});
}
const gitAhead = (repo, cb) => {
exec(`git rev-list --left-only --count HEAD...@'{u}' 2>/dev/null`, { cwd: repo }, (err, stdout) => {
cb(null, parseInt(stdout, 10));
});
}
const gitCheck = (repo, cb) => {
const next = afterAll((err, results) => {
if (err) {
return cb(err);
}
const branch = results[0];
const remote = results[1];
const dirty = results[2];
const ahead = results[3];
cb(null, {
branch: branch,
remote: remote,
dirty: dirty,
ahead: ahead
});
});
gitBranch(repo, next());
gitRemote(repo, next());
gitDirty(repo, next());
gitAhead(repo, next());
}
const setGit = (repo) => {
isGit(repo, (exists) => {
if (!exists) {
git = {
branch: '',
remote: '',
dirty: 0,
ahead: 0
}
return;
}
gitCheck(repo, (err, result) => {
if (err) {
throw err;
}
git = {
branch: result.branch,
remote: result.remote,
dirty: result.dirty,
ahead: result.ahead
}
})
});
}
exports.decorateHyper = (Hyper, { React }) => {
return class extends React.PureComponent {
constructor(props) {
super(props);
this.state = {
cwd: '',
branch: '',
remote: '',
dirty: 0,
ahead: 0
}
this.handleCwdClick = this.handleCwdClick.bind(this);
this.handleBranchClick = this.handleBranchClick.bind(this);
}
handleCwdClick(event) {
shell.openExternal('file://' + this.state.cwd);
}
handleBranchClick(event) {
shell.openExternal(this.state.remote);
}
render() {
const { customChildren } = this.props
const existingChildren = customChildren ? customChildren instanceof Array ? customChildren : [customChildren] : [];
return (
React.createElement(Hyper, Object.assign({}, this.props, {
customInnerChildren: existingChildren.concat(React.createElement('footer', { className: 'footer_footer' },
React.createElement('div', { className: 'footer_group group_overflow' },
React.createElement('div', { className: 'component_component component_cwd' },
React.createElement('div', { className: 'component_item item_icon item_cwd item_clickable', title: this.state.cwd, onClick: this.handleCwdClick, hidden: !this.state.cwd }, this.state.cwd ? tildify(String(this.state.cwd)) : '')
)
),
React.createElement('div', { className: 'footer_group' },
React.createElement('div', { className: 'component_component component_git' },
React.createElement('div', { className: `component_item item_icon item_branch ${this.state.remote ? 'item_clickable' : ''}`, title: this.state.remote, onClick: this.handleBranchClick, hidden: !this.state.branch }, this.state.branch),
React.createElement('div', { className: 'component_item item_icon item_number item_dirty', title: `${this.state.dirty} dirty ${this.state.dirty > 1 ? 'files' : 'file'}`, hidden: !this.state.dirty }, this.state.dirty),
React.createElement('div', { className: 'component_item item_icon item_number item_ahead', title: `${this.state.ahead} ${this.state.ahead > 1 ? 'commits' : 'commit'} ahead`, hidden: !this.state.ahead }, this.state.ahead)
)
)
))
}))
);
}
componentDidMount() {
this.interval = setInterval(() => {
this.setState({
cwd: cwd,
branch: git.branch,
remote: git.remote,
dirty: git.dirty,
ahead: git.ahead
});
}, 100);
}
componentWillUnmount() {
clearInterval(this.interval);
}
};
};
exports.middleware = (store) => (next) => (action) => {
const uids = store.getState().sessions.sessions;
switch (action.type) {
case 'SESSION_SET_XTERM_TITLE':
pid = uids[action.uid].pid;
break;
case 'SESSION_ADD':
pid = action.pid;
setCwd(pid);
break;
case 'SESSION_ADD_DATA':
const { data } = action;
const enterKey = data.indexOf('\n') > 0;
if (enterKey) {
setCwd(pid, action);
}
break;
case 'SESSION_SET_ACTIVE':
pid = uids[action.uid].pid;
setCwd(pid);
break;
}
next(action);
};
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment