hyper-statusline
Version:
Status Line Plugin for Hyper
359 lines (320 loc) • 16.6 kB
JavaScript
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 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;
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;
width: 14px;
height: 100%;
-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);
};