├── .babelrc ├── .editorconfig ├── .github └── ISSUE_TEMPLATE.md ├── .gitignore ├── .npmignore ├── README.md ├── jest.config.json ├── license.md ├── package-lock.json ├── package.json ├── screenshot.png ├── src ├── index.js └── lib │ ├── core │ └── hyperline.js │ ├── plugins │ ├── battery.js │ ├── battery │ │ ├── battery-icon.js │ │ ├── charging.js │ │ ├── critical.js │ │ └── draining.js │ ├── cpu.js │ ├── docker.js │ ├── git-status.js │ ├── hostname.js │ ├── index.js │ ├── ip.js │ ├── memory.js │ ├── network.js │ ├── spotify.js │ ├── time.js │ └── uptime.js │ └── utils │ ├── colors.js │ ├── config.js │ ├── svg-icon.js │ └── time.js ├── tests ├── __snapshots__ │ └── time.spec.js.snap ├── spotify.spec.js └── time.spec.js ├── webpack.config.js └── yarn.lock /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets": ["react"], 3 | "plugins": [ 4 | [ 5 | "styled-jsx/babel", 6 | { 7 | "vendorPrefixes": false 8 | } 9 | ], 10 | "transform-object-rest-spread" 11 | ], 12 | "env": { 13 | "test": { 14 | "presets": ["es2015"] 15 | } 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | # EditorConfig is awesome: http://EditorConfig.org 2 | 3 | # top-most EditorConfig file 4 | root = true 5 | 6 | # Unix-style newlines with a newline ending every file 7 | [*] 8 | end_of_line = lf 9 | insert_final_newline = true 10 | 11 | # Matches multiple files with brace expansion notation 12 | # Set default charset 13 | [*.{js}] 14 | charset = utf-8 15 | 16 | # 4 space indentation 17 | [*.js] 18 | indent_style = space 19 | indent_size = 2 20 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE.md: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | npm-debug.log 3 | dist 4 | .vscode 5 | .DS_STORE 6 | .idea 7 | -------------------------------------------------------------------------------- /.npmignore: -------------------------------------------------------------------------------- 1 | # All 2 | * 3 | 4 | # But not 5 | !dist/** 6 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | HyperLine 2 | ========= 3 | 4 | **HyperLine is a status line plugin for [Hyper.app](https://hyper.is/)**. It shows you useful system information such as free memory, uptime and CPU usage. 5 | 6 | ![Screenshot](./screenshot.png) 7 | 8 | ## Install 9 | 10 | * **NOTE:** HyperLine is not currently supported when using Microsoft Windows. See [this issue](https://github.com/Hyperline/hyperline/issues/57) for additional information. 11 | 12 | To install, edit `~/.hyper.js` and add `"hyperline"` to `plugins`: 13 | 14 | ``` 15 | plugins: [ 16 | "hyperline", 17 | ], 18 | ``` 19 | 20 | ## Styling the line 21 | 22 | We implemented the same mechanism for styling/creating css classes that Hyper uses. 23 | This will allow you to create custom HyperLine themes the same way you would create a Hyper css theme. 24 | 25 | ## Configuring plugins 26 | Add the names of plugins in the order in which you would like them to be displayed to your `~/.hyper.js`: 27 | 28 | ``` 29 | config: { 30 | hyperline: { 31 | plugins: [ 32 | "ip", 33 | "cpu", 34 | "spotify" 35 | ] 36 | }, 37 | } 38 | ``` 39 | You can see a list of all available plugins in [`src/lib/plugins/index.js`](https://github.com/Hyperline/hyperline/blob/master/src/lib/plugins/index.js) 40 | 41 | ## Contributing 42 | 43 | Feel free to contribute to HyperLine by [requesting a feature](https://github.com/hyperline/hyperline/issues/new), [submitting a bug](https://github.com/hyperline/hyperline/issues/new) or contributing code. 44 | 45 | To set up the project for development: 46 | 47 | 1. Clone this repository into `~/.hyper_plugins/local/` 48 | 2. Run `npm install` within the project directory 49 | 3. Run `npm run build` to build the plugin **OR** `npm run dev` to build the plugin and watch for file changes. 50 | 4. Add the name of the directory to `localPlugins` in `~/.hyper.js`. 51 | 5. Reload terminal window 52 | 53 | ## Authors 54 | 55 | - Nick Tikhonov [@nicktikhonov](https://github.com/nicktikhonov) 56 | - Tim Neutkens [@timneutkens](https://github.com/timneutkens) 57 | - Stefan Ivic [@stefanivic](https://github.com/stefanivic) 58 | - Henrik Dahlheim [@henrikdahl](https://github.com/henrikdahl) 59 | 60 | ## Contributors 61 | 62 | This project exists thanks to all the people who contribute. 63 | 64 | -------------------------------------------------------------------------------- /jest.config.json: -------------------------------------------------------------------------------- 1 | { 2 | "testRegex": "(/tests/.*|\\.(test|spec))\\.js$", 3 | "moduleFileExtensions": [ "js" ], 4 | "moduleDirectories": [ "node_modules" ] 5 | } 6 | -------------------------------------------------------------------------------- /license.md: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2016 Nick Tikhonov, Tim Neutkens 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: 6 | 7 | The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. 8 | 9 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 10 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "hyperline", 3 | "version": "1.2.0", 4 | "description": "Handy status line for Hyper.app", 5 | "keywords": [ 6 | "hyper.app", 7 | "hyper", 8 | "hyperterm" 9 | ], 10 | "author": "Nick Tikhonov", 11 | "contributors": [ 12 | { 13 | "name": "Tim Neutkens" 14 | }, 15 | { 16 | "name": "Stefan Ivic" 17 | }, 18 | { 19 | "name": "Henrik Dahlheim" 20 | } 21 | ], 22 | "license": "MIT", 23 | "repository": "hyperline/hyperline", 24 | "main": "dist/hyperline.js", 25 | "files": [ 26 | "dist/hyperline.js" 27 | ], 28 | "dependencies": { 29 | "color": "^0.11.3", 30 | "git-state": "^4.0.0", 31 | "json-loader": "^0.5.4", 32 | "left-pad": "^1.1.3", 33 | "moment": "^2.18.1", 34 | "opencollective": "^1.0.3", 35 | "prop-types": "^15.5.10", 36 | "public-ip": "^2.0.1", 37 | "spotify-node-applescript": "^1.1.0", 38 | "styled-jsx": "2.2.6", 39 | "systeminformation": "^3.4.1" 40 | }, 41 | "devDependencies": { 42 | "babel-core": "^6.11.4", 43 | "babel-jest": "^20.0.3", 44 | "babel-loader": "^6.2.4", 45 | "babel-plugin-transform-object-rest-spread": "6.26.0", 46 | "babel-preset-es2015": "^6.9.0", 47 | "babel-preset-react": "^6.11.1", 48 | "cross-env": "^3.1.4", 49 | "eslint-config-prettier": "^2.3.0", 50 | "eslint-config-xo-react": "^0.13.0", 51 | "eslint-loader": "^1.5.0", 52 | "eslint-plugin-react": "^7.1.0", 53 | "jest": "^20.0.4", 54 | "lint-staged": "^4.0.0", 55 | "prettier": "^1.5.2", 56 | "prop-types": "^15.5.10", 57 | "webpack": "2.2.0-rc.1", 58 | "webpack-node-externals": "^1.3.3", 59 | "xo": "^0.18.2", 60 | "xo-loader": "^0.8.0" 61 | }, 62 | "scripts": { 63 | "build": "cross-env NODE_ENV=production webpack", 64 | "dev": "webpack --watch", 65 | "tdev": "jest --watch", 66 | "test": "jest", 67 | "precommit": "lint-staged", 68 | "postinstall": "opencollective postinstall" 69 | }, 70 | "xo": { 71 | "ignores": [ 72 | "examples/**/*", 73 | "node_modules/**/*" 74 | ], 75 | "extends": "prettier", 76 | "rules": { 77 | "import/no-extraneous-dependencies": 0, 78 | "import/no-unresolved": 0, 79 | "no-unused-vars": 0, 80 | "import/order": 1, 81 | "no-warning-comments": 0, 82 | "prefer-promise-reject-errors": 0, 83 | "import/named": 1, 84 | "space-in-parens": 0, 85 | "object-curly-spacing": 0, 86 | "computed-property-spacing": 0, 87 | "space-infix-ops": 0, 88 | "one-var": 0, 89 | "no-console": 0, 90 | "no-useless-constructor": 1, 91 | "no-useless-escape": 1 92 | } 93 | }, 94 | "lint-staged": { 95 | "*.js": [ 96 | "npm run lint", 97 | "prettier --single-quote --no-semi --write", 98 | "jest" 99 | ] 100 | }, 101 | "collective": { 102 | "type": "opencollective", 103 | "url": "https://opencollective.com/hyperline", 104 | "logo": "https://opencollective.com/opencollective/logo.txt" 105 | } 106 | } 107 | -------------------------------------------------------------------------------- /screenshot.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Hyperline/hyperline/b4e275e6b399abff1540400d719f1169a77e3b30/screenshot.png -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react' 2 | import PropTypes from 'prop-types' 3 | import HyperLine from './lib/core/hyperline' 4 | import { getColorList } from './lib/utils/colors' 5 | import hyperlinePlugins from './lib/plugins' 6 | 7 | export function reduceUI(state, { type, config }) { 8 | switch (type) { 9 | case 'CONFIG_LOAD': 10 | case 'CONFIG_RELOAD': { 11 | return state.set('hyperline', config.hyperline) 12 | } 13 | default: 14 | break 15 | } 16 | 17 | return state 18 | } 19 | 20 | export function mapHyperState({ ui: { colors, fontFamily, hyperline } }, map) { 21 | let userPlugins = [] 22 | if (hyperline !== undefined) { 23 | if (hyperline.plugins !== undefined) { 24 | userPlugins = hyperline.plugins 25 | } 26 | } 27 | 28 | return Object.assign({}, map, { 29 | colors: getColorList(colors), 30 | fontFamily, 31 | userPlugins 32 | }) 33 | } 34 | 35 | function pluginsByName(plugins) { 36 | const dict = {} 37 | plugins.forEach((plugin) => { 38 | dict[plugin.displayName()] = plugin 39 | }) 40 | 41 | return dict 42 | } 43 | 44 | function filterPluginsByConfig(plugins) { 45 | const config = window.config.getConfig().hyperline 46 | if (!config) return plugins 47 | 48 | const userPluginNames = config.plugins 49 | if (!userPluginNames) { 50 | return plugins 51 | } 52 | 53 | plugins = pluginsByName(plugins) 54 | const filtered = [] 55 | 56 | userPluginNames.forEach((name) => { 57 | if (plugins.hasOwnProperty(name)) { 58 | filtered.push(plugins[name]) 59 | } 60 | }) 61 | 62 | return filtered 63 | } 64 | 65 | export function decorateHyperLine(HyperLine) { 66 | return class extends Component { 67 | static displayName() { 68 | return 'HyperLine' 69 | } 70 | 71 | static propTypes() { 72 | return { 73 | plugins: PropTypes.array.isRequired 74 | } 75 | } 76 | 77 | static get defaultProps() { 78 | return { 79 | plugins: [] 80 | } 81 | } 82 | 83 | render() { 84 | const plugins = [...this.props.plugins, ...hyperlinePlugins] 85 | 86 | return 87 | } 88 | } 89 | } 90 | 91 | export function decorateHyper(Hyper) { 92 | return class extends Component { 93 | static displayName() { 94 | return 'Hyper' 95 | } 96 | 97 | static propTypes() { 98 | return { 99 | colors: PropTypes.oneOfType([PropTypes.object, PropTypes.array]), 100 | fontFamily: PropTypes.string, 101 | customChildren: PropTypes.element.isRequired 102 | } 103 | } 104 | 105 | render() { 106 | const customChildren = ( 107 |
108 | {this.props.customChildren} 109 | 110 |
111 | ) 112 | 113 | return 114 | } 115 | } 116 | } 117 | -------------------------------------------------------------------------------- /src/lib/core/hyperline.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import PropTypes from 'prop-types' 3 | import Component from 'hyper/component' 4 | import decorate from 'hyper/decorate' 5 | 6 | class HyperLine extends Component { 7 | static propTypes() { 8 | return { 9 | plugins: PropTypes.array.isRequired 10 | } 11 | } 12 | 13 | render() { 14 | const { plugins, ...props } = this.props 15 | 16 | return ( 17 |
18 | {plugins.map((Component, index) => ( 19 |
20 | 21 |
22 | ))} 23 | 24 | 47 |
48 | ) 49 | } 50 | } 51 | 52 | export default decorate(HyperLine, 'HyperLine') 53 | -------------------------------------------------------------------------------- /src/lib/plugins/battery.js: -------------------------------------------------------------------------------- 1 | /* eslint no-undef: 0 */ 2 | // Note: This is to stop XO from complaining about {navigator} 3 | 4 | import React from 'react' 5 | import Component from 'hyper/component' 6 | import leftPad from 'left-pad' 7 | import BatteryIcon from './battery/battery-icon' 8 | 9 | export default class Battery extends Component { 10 | static displayName() { 11 | return 'battery' 12 | } 13 | 14 | constructor(props) { 15 | super(props) 16 | 17 | this.state = { 18 | charging: false, 19 | percentage: '--' 20 | } 21 | 22 | this.batteryEvents = [ 'chargingchange', 'chargingtimechange', 'dischargingtimechange', 'levelchange' ] 23 | this.handleEvent = this.handleEvent.bind(this) 24 | } 25 | 26 | setBatteryStatus(battery) { 27 | this.setState({ 28 | charging: battery.charging, 29 | percentage: Math.floor(battery.level * 100) 30 | }) 31 | } 32 | 33 | handleEvent(event) { 34 | this.setBatteryStatus(event.target) 35 | } 36 | 37 | componentDidMount() { 38 | navigator.getBattery().then(battery => { 39 | this.setBatteryStatus(battery) 40 | 41 | this.batteryEvents.forEach(event => { 42 | battery.addEventListener(event, this.handleEvent, false) 43 | }) 44 | }) 45 | } 46 | 47 | componentWillUnmount() { 48 | navigator.getBattery().then(battery => { 49 | this.batteryEvents.forEach(event => { 50 | battery.removeEventListener(event, this.handleEvent) 51 | }) 52 | }) 53 | } 54 | 55 | render() { 56 | const { charging, percentage } = this.state 57 | 58 | return ( 59 |
60 | {leftPad(percentage, 2, 0)}% 61 | 62 | 68 |
69 | ) 70 | } 71 | } 72 | -------------------------------------------------------------------------------- /src/lib/plugins/battery/battery-icon.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import PropTypes from 'prop-types'; 3 | import Critical from './critical' 4 | import Charging from './charging' 5 | import Draining from './draining' 6 | 7 | function BatteryIcon({ charging, percentage }) { 8 | if (charging) { 9 | return 10 | } 11 | 12 | if (percentage <= 20) { 13 | return 14 | } 15 | 16 | return 17 | } 18 | 19 | BatteryIcon.propTypes = { 20 | charging: PropTypes.bool, 21 | percentage: PropTypes.number 22 | } 23 | 24 | export default BatteryIcon 25 | -------------------------------------------------------------------------------- /src/lib/plugins/battery/charging.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Component from 'hyper/component' 3 | import SvgIcon from '../../utils/svg-icon' 4 | 5 | export default class Charging extends Component { 6 | render() { 7 | return ( 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 20 | 21 | ) 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /src/lib/plugins/battery/critical.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Component from 'hyper/component' 3 | import SvgIcon from '../../utils/svg-icon' 4 | 5 | export default class Critical extends Component { 6 | render() { 7 | return ( 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 20 | 21 | ) 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /src/lib/plugins/battery/draining.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import PropTypes from 'prop-types' 3 | import Component from 'hyper/component' 4 | import SvgIcon from '../../utils/svg-icon' 5 | 6 | export default class Draining extends Component { 7 | static propTypes() { 8 | return { 9 | percentage: PropTypes.number 10 | } 11 | } 12 | 13 | calculateChargePoint(percent) { 14 | const base = 3.5, 15 | val = Math.round((100 - percent) / 4.5), 16 | point = base + (val / 2) 17 | 18 | return val > 0 ? `M5,3 L11,3 L11,${point} L5,${point} L5,3 Z` : '' 19 | } 20 | 21 | render() { 22 | const chargePoint = this.calculateChargePoint(this.props.percentage) 23 | return ( 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 36 | 37 | ) 38 | } 39 | } 40 | -------------------------------------------------------------------------------- /src/lib/plugins/cpu.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Component from 'hyper/component' 3 | import { currentLoad as cpuLoad } from 'systeminformation' 4 | import leftPad from 'left-pad' 5 | import SvgIcon from '../utils/svg-icon' 6 | 7 | class PluginIcon extends Component { 8 | render() { 9 | return ( 10 | 11 | 12 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 43 | 44 | ) 45 | } 46 | } 47 | 48 | export default class Cpu extends Component { 49 | static displayName() { 50 | return 'cpu' 51 | } 52 | 53 | constructor(props) { 54 | super(props) 55 | 56 | this.state = { 57 | cpuLoad: 0 58 | } 59 | } 60 | 61 | getCpuLoad() { 62 | cpuLoad().then(({ currentload }) => 63 | this.setState({ 64 | cpuLoad: leftPad(currentload.toFixed(2), 2, 0) 65 | }) 66 | ) 67 | } 68 | 69 | componentDidMount() { 70 | this.getCpuLoad() 71 | this.interval = setInterval(() => this.getCpuLoad(), 2500) 72 | } 73 | 74 | componentWillUnmount() { 75 | clearInterval(this.interval) 76 | } 77 | 78 | render() { 79 | return ( 80 |
81 | {this.state.cpuLoad} 82 | 83 | 89 |
90 | ) 91 | } 92 | } 93 | -------------------------------------------------------------------------------- /src/lib/plugins/docker.js: -------------------------------------------------------------------------------- 1 | import { exec as ex } from 'child_process' 2 | import React from 'react' 3 | import Component from 'hyper/component' 4 | import SvgIcon from '../utils/svg-icon' 5 | 6 | class PluginIcon extends Component { 7 | render() { 8 | return ( 9 | 10 | 11 | 12 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 28 | 29 | ) 30 | } 31 | } 32 | 33 | export default class Docker extends Component { 34 | static displayName() { 35 | return 'docker' 36 | } 37 | 38 | constructor(props) { 39 | super(props) 40 | 41 | this.state = { version: 'Not running' } 42 | this.setVersion = this.setVersion.bind(this) 43 | } 44 | 45 | setVersion() { 46 | exec('/usr/local/bin/docker version -f {{.Server.Version}}') 47 | .then(version => { 48 | this.setState({ version }) 49 | }) 50 | .catch(() => { 51 | this.setState({ version: 'Not running' }) 52 | }) 53 | } 54 | 55 | componentDidMount() { 56 | this.setVersion() 57 | this.interval = setInterval(() => this.setVersion(), 15000) 58 | } 59 | 60 | componentWillUnmount() { 61 | clearInterval(this.interval) 62 | } 63 | 64 | render() { 65 | return ( 66 |
67 | {this.state.version} 68 | 69 | 76 |
77 | ) 78 | } 79 | } 80 | 81 | function exec(command, options) { 82 | return new Promise((resolve, reject) => { 83 | ex(command, options, (err, stdout, stderr) => { 84 | if (err) { 85 | reject(`${err}\n${stderr}`) 86 | } else { 87 | resolve(stdout) 88 | } 89 | }) 90 | }) 91 | } 92 | -------------------------------------------------------------------------------- /src/lib/plugins/git-status.js: -------------------------------------------------------------------------------- 1 | import exec from 'child_process' 2 | import shell from 'electron' 3 | import React from 'react' 4 | import Component from 'hyper/component' 5 | 6 | 7 | /* 8 | NOTICE 9 | ============ 10 | This code is essentially a port from Henrik Dahlheim's "hyper-statusline" 11 | https://github.com/henrikdahl/hyper-statusline 12 | 13 | I'm simply the guy that ported it over to work in Hyperline 14 | 15 | 16 | Corbin (basedjux) Matschull 17 | */ 18 | 19 | const dirtyColor = '#FFFBB3' 20 | const pushColor = '#B7E8FF' 21 | 22 | class PluginIcon extends Component { 23 | } 24 | 25 | 26 | let curBranch 27 | let curRemote 28 | let repoDirty 29 | let pushArrow 30 | let pullArrow 31 | 32 | export default class GitStatus extends Component { 33 | static displayName() { 34 | return "git-status" 35 | } 36 | 37 | constructor(props) { 38 | super(props) 39 | 40 | this.state = { 41 | branch: curBranch, 42 | remote: curRemote, 43 | dirty: repoDirty, 44 | push: pushArrow, 45 | pull: pullArrow 46 | } 47 | 48 | this.handleClick = this.handleClick.bind(this) 49 | } 50 | 51 | handleClick(e) { 52 | shell.openExternal(this.state.remote) 53 | } 54 | 55 | checkDirty(actionCwd) { 56 | exec(`git status --porcelain --ignore-submodules -unormal`, { cwd: actionCwd }, (err, branch) => { 57 | repoDirty = true 58 | }) 59 | } 60 | 61 | setRemote(actionCwd) { 62 | exec(`git config --get remote.origin.url`, { cwd: actionCwd }, (err, remote) => { 63 | curRemote = remote.trim().replace(/^git@(.*?):/, 'https://$1/').replace(/[A-z0-9\-]+@/, '').replace(/\.git$/, '') 64 | }) 65 | } 66 | 67 | checkArrows(actionCwd) { 68 | exec(`git rev-list --left-right --count HEAD...@'{u}' 2>/dev/null`, { cwd: actionCwd }, (err, arrows) => { 69 | arrows = arrows.split('\t'); 70 | pushArrow = arrows[0] > 0 ? arrows[0] : ''; 71 | pullArrow = arrows[1] > 0 ? arrows[1] : ''; 72 | }) 73 | } 74 | 75 | setBranch(actionCwd) { 76 | exec(`git symbolic-ref --short HEAD || git rev-parse --short HEAD`, { cwd: actionCwd }, (err,branch) => { 77 | curBranch = branch 78 | 79 | if (branch === '') { 80 | this.setRemote(actionCwd) 81 | this.checkDirty(actionCwd) 82 | this.checkArrows(actionCwd) 83 | } 84 | }) 85 | } 86 | 87 | styles() { 88 | return { 89 | 'item_branch': { 90 | 'padding-left': '30px' 91 | }, 92 | 'item_branch:before': { 93 | 'left': '14.5px', 94 | '-webkit-mask-image': 'url(\'\')', 95 | '-webkit-mask-size': '9px 12px' 96 | }, 97 | 'item_click:hover': { 98 | 'text-decoration': 'underline', 99 | 'cursor': 'pointer' 100 | }, 101 | 'item_folder, .item_text': { 102 | 'line-height': '29px' 103 | }, 104 | 'item_text': { 105 | 'height': '100%' 106 | }, 107 | 'item_icon': { 108 | 'display': 'none', 109 | 'width': '12px', 110 | 'height' : '100%', 111 | 'margin-left': '9px', 112 | '-webkit-mask-size': '12px 12px', 113 | '-webkit-mask-repeat': 'no-repeat', 114 | '-webkit-mast-position': '0 center' 115 | }, 116 | 'icon_active': { 117 | 'display': 'inline-block' 118 | }, 119 | 'icon_dirty': { 120 | '-webkit-mask-image': 'url(\'\')', 121 | 'background-color': `${dirtyColor}` 122 | }, 123 | 'icon_push': { 124 | '-webkit-mask-image': 'url(\'\')', 125 | 'background-color': `${pushColor}` 126 | }, 127 | 'icon_pull': { 128 | 'transform': 'scaleY(-1)', 129 | '-webkit-mask-position': '0 8px' 130 | } 131 | } 132 | } 133 | 134 | 135 | 136 | } 137 | -------------------------------------------------------------------------------- /src/lib/plugins/hostname.js: -------------------------------------------------------------------------------- 1 | import os from 'os' 2 | import React from 'react' 3 | import Component from 'hyper/component' 4 | import SvgIcon from '../utils/svg-icon' 5 | 6 | class PluginIcon extends Component { 7 | render() { 8 | 9 | return ( 10 | 11 | 12 | 16 | 17 | 18 | 19 | 20 | 25 | 26 | ) 27 | } 28 | } 29 | 30 | export default class HostName extends Component { 31 | static displayName() { 32 | return 'hostname' 33 | } 34 | 35 | render() { 36 | const hostname = os.hostname() 37 | const username = process.env.USER 38 | 39 | return ( 40 |
41 | {username}@ 42 | {hostname} 43 | 44 | 50 |
51 | ) 52 | } 53 | } 54 | -------------------------------------------------------------------------------- /src/lib/plugins/index.js: -------------------------------------------------------------------------------- 1 | import hostname from './hostname' 2 | import ip from './ip' 3 | import memory from './memory' 4 | // Import Uptime from './uptime' 5 | import cpu from './cpu' 6 | import network from './network' 7 | import battery from './battery' 8 | // Import Time from './time' 9 | // Import Docker from './docker' 10 | import spotify from './spotify' 11 | 12 | export default [hostname, ip, memory, battery, cpu, network, spotify] 13 | -------------------------------------------------------------------------------- /src/lib/plugins/ip.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Component from 'hyper/component' 3 | import publicIp from 'public-ip' 4 | import SvgIcon from '../utils/svg-icon' 5 | 6 | function getIp() { 7 | return new Promise(resolve => { 8 | publicIp.v4().then(ip => resolve(ip)).catch(() => resolve('?.?.?.?')) 9 | }) 10 | } 11 | 12 | class PluginIcon extends Component { 13 | render() { 14 | return ( 15 | 16 | 17 | 21 | 25 | 26 | 27 | 32 | 33 | ) 34 | } 35 | } 36 | 37 | export default class Ip extends Component { 38 | static displayName() { 39 | return 'ip' 40 | } 41 | 42 | constructor(props) { 43 | super(props) 44 | 45 | this.state = { 46 | ip: '?.?.?.?' 47 | } 48 | 49 | this.setIp = this.setIp.bind(this) 50 | } 51 | 52 | setIp() { 53 | getIp().then(ip => this.setState({ ip })) 54 | } 55 | 56 | componentDidMount() { 57 | // Every 5 seconds 58 | this.setIp() 59 | this.interval = setInterval(() => this.setIp(), 60000 * 5) 60 | } 61 | 62 | componentWillUnmount() { 63 | clearInterval(this.interval) 64 | } 65 | 66 | render() { 67 | return ( 68 |
69 | {this.state.ip} 70 | 71 | 77 |
78 | ) 79 | } 80 | } 81 | -------------------------------------------------------------------------------- /src/lib/plugins/memory.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Component from 'hyper/component' 3 | import { mem as memoryData } from 'systeminformation' 4 | import leftPad from 'left-pad' 5 | import SvgIcon from '../utils/svg-icon' 6 | 7 | class PluginIcon extends Component { 8 | render() { 9 | return ( 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 37 | 38 | 39 | ) 40 | } 41 | } 42 | 43 | export default class Memory extends Component { 44 | static displayName() { 45 | return 'memory' 46 | } 47 | 48 | constructor(props) { 49 | super(props) 50 | 51 | this.state = { 52 | activeMemory: 0, 53 | totalMemory: 0 54 | } 55 | 56 | this.getMemory = this.getMemory.bind(this) 57 | this.setMemory = this.setMemory.bind(this) 58 | } 59 | 60 | componentDidMount() { 61 | this.setMemory() 62 | this.interval = setInterval(() => this.setMemory(), 2500) 63 | } 64 | 65 | componentWillUnmount() { 66 | clearInterval(this.interval) 67 | } 68 | 69 | getMemory() { 70 | return memoryData().then(memory => { 71 | const totalMemory = this.getMb(memory.total) 72 | const activeMemory = this.getMb(memory.active) 73 | const totalWidth = totalMemory.toString().length 74 | 75 | return { 76 | activeMemory: leftPad(activeMemory, totalWidth, 0), 77 | totalMemory 78 | } 79 | }) 80 | } 81 | 82 | setMemory() { 83 | return this.getMemory().then(data => this.setState(data)) 84 | } 85 | 86 | getMb(bytes) { 87 | // 1024 * 1024 = 1048576 88 | return (bytes / 1048576).toFixed(0) 89 | } 90 | 91 | render() { 92 | return ( 93 |
94 | {this.state.activeMemory}MB / {this.state.totalMemory}MB 95 | 96 | 102 |
103 | ) 104 | } 105 | } 106 | -------------------------------------------------------------------------------- /src/lib/plugins/network.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Component from 'hyper/component' 3 | import { networkStats } from 'systeminformation' 4 | import SvgIcon from '../utils/svg-icon' 5 | 6 | class PluginIcon extends Component { 7 | render() { 8 | return ( 9 | 10 | 11 | 12 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 29 | 30 | ) 31 | } 32 | } 33 | 34 | export default class Network extends Component { 35 | static displayName() { 36 | return 'network' 37 | } 38 | 39 | constructor(props) { 40 | super(props) 41 | this.state = { 42 | download: 0, 43 | upload: 0 44 | } 45 | } 46 | 47 | componentDidMount() { 48 | this.getSpeed() 49 | this.interval = setInterval(() => this.getSpeed(), 1500) 50 | } 51 | 52 | componentWillUnmount() { 53 | clearInterval(this.interval) 54 | } 55 | 56 | calculate(data) { 57 | const rawData = data / 1024 58 | return (rawData > 0 ? rawData : 0).toFixed() 59 | } 60 | 61 | getSpeed() { 62 | networkStats().then(data => 63 | this.setState({ 64 | download: this.calculate(data.rx_sec), 65 | upload: this.calculate(data.tx_sec) 66 | }) 67 | ) 68 | } 69 | 70 | render() { 71 | const { download, upload } = this.state 72 | return ( 73 |
74 | {download}kB/s {upload}kB/s 75 | 76 | 82 |
83 | ) 84 | } 85 | } 86 | -------------------------------------------------------------------------------- /src/lib/plugins/spotify.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Component from 'hyper/component' 3 | import spotify from 'spotify-node-applescript' 4 | import SvgIcon from '../utils/svg-icon' 5 | 6 | class PluginIcon extends Component { 7 | render() { 8 | return ( 9 | 10 | 11 | 12 | 16 | 17 | 21 | 22 | 23 | 24 | 25 | 26 | 31 | 32 | ) 33 | } 34 | } 35 | 36 | export default class Spotify extends Component { 37 | static displayName() { 38 | return 'spotify' 39 | } 40 | 41 | constructor(props) { 42 | super(props) 43 | 44 | this.state = { version: 'Not running' } 45 | this.setStatus = this.setStatus.bind(this) 46 | 47 | this.handleSpotifyActivation = this.handleSpotifyActivation.bind(this) 48 | } 49 | 50 | setStatus() { 51 | spotify.isRunning((err, isRunning) => { 52 | if (!isRunning) { 53 | this.setState({ state: 'Not running' }) 54 | return 55 | } 56 | if (err) { 57 | console.log(`Caught exception at setStatus(e): ${err}`) 58 | } 59 | spotify.getState((err, st) => { 60 | if (err) { 61 | console.log(`Caught exception at spotify.getState(e): ${err}`) 62 | } 63 | 64 | spotify.getTrack((err, track) => { 65 | if (err) { 66 | console.log(`Caught exception at spotify.getTrack(e): ${err}`) 67 | } 68 | this.setState({ 69 | state: `${st.state === 'playing' 70 | ? '▶' 71 | : '❚❚'} ${track.artist} - ${track.name}` 72 | }) 73 | }) 74 | }) 75 | }) 76 | } 77 | 78 | /* 79 | TODO: Make this work on Linux and Win 32/64 80 | */ 81 | handleSpotifyActivation(e) { 82 | e.preventDefault() 83 | console.log('HANDLE CLICKED FOR SPOTIFY') 84 | spotify.isRunning((err, isRunning) => { 85 | if (!isRunning) { 86 | spotify.openSpotify() 87 | } 88 | 89 | if (err) { 90 | console.log(`Caught exception at handleSpotifyActivation(e): ${err}`) 91 | } 92 | }) 93 | } 94 | 95 | componentDidMount() { 96 | this.setStatus() 97 | this.interval = setInterval(() => this.setStatus(), 1000) 98 | } 99 | 100 | componentWillUnmount() { 101 | clearInterval(this.interval) 102 | } 103 | 104 | render() { 105 | return ( 106 |
110 | {this.state.state} 111 | 112 | 119 |
120 | ) 121 | } 122 | } 123 | -------------------------------------------------------------------------------- /src/lib/plugins/time.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Component from 'hyper/component' 3 | import moment from 'moment' 4 | import SvgIcon from '../utils/svg-icon' 5 | 6 | class PluginIcon extends Component { 7 | render() { 8 | return ( 9 | 10 | 11 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 27 | 28 | ) 29 | } 30 | } 31 | 32 | export default class Time extends Component { 33 | static displayName() { 34 | return 'time' 35 | } 36 | 37 | constructor(props) { 38 | super(props) 39 | 40 | this.state = { 41 | time: this.getCurrentTime() 42 | } 43 | } 44 | 45 | componentDidMount() { 46 | this.interval = setInterval(() => { 47 | this.setState({ time: this.getCurrentTime() }) 48 | }, 100) 49 | } 50 | 51 | componentWillUnmount() { 52 | clearInterval(this.interval) 53 | } 54 | 55 | getCurrentTime() { 56 | // TODO: Allow for format overriding by the user 57 | return moment().format('LTS') 58 | } 59 | 60 | render() { 61 | return ( 62 |
63 | {this.state.time} 64 | 65 | 71 |
72 | ) 73 | } 74 | } 75 | -------------------------------------------------------------------------------- /src/lib/plugins/uptime.js: -------------------------------------------------------------------------------- 1 | import os from 'os' 2 | import React from 'react' 3 | import Component from 'hyper/component' 4 | import formatUptime from '../utils/time' 5 | import SvgIcon from '../utils/svg-icon' 6 | 7 | class PluginIcon extends Component { 8 | render() { 9 | return ( 10 | 11 | 12 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 28 | 29 | ) 30 | } 31 | } 32 | 33 | export default class Uptime extends Component { 34 | static displayName() { 35 | return 'uptime' 36 | } 37 | 38 | constructor(props) { 39 | super(props) 40 | 41 | this.state = { 42 | uptime: this.getUptime() 43 | } 44 | } 45 | 46 | componentDidMount() { 47 | const uptime = this.getUptime() 48 | // Recheck every 5 minutes 49 | setInterval(() => this.setState({ uptime }), 60000 * 5) 50 | } 51 | 52 | getUptime() { 53 | return formatUptime(os.uptime()) 54 | } 55 | 56 | template(css) { 57 | return ( 58 |
59 | {this.state.uptime} 60 | 61 | 67 |
68 | ) 69 | } 70 | } 71 | -------------------------------------------------------------------------------- /src/lib/utils/colors.js: -------------------------------------------------------------------------------- 1 | // Taken from https://github.com/zeit/hyper/blob/master/lib/utils/colors.js 2 | // Effect of this script is the reverse of colors.js in hyper.app 3 | const colorList = [ 4 | 'black', 5 | 'red', 6 | 'green', 7 | 'yellow', 8 | 'blue', 9 | 'magenta', 10 | 'cyan', 11 | 'white', 12 | 'lightBlack', 13 | 'lightRed', 14 | 'lightGreen', 15 | 'lightYellow', 16 | 'lightBlue', 17 | 'lightMagenta', 18 | 'lightCyan', 19 | 'lightWhite', 20 | 'colorCubes', 21 | 'grayscale' 22 | ]; 23 | 24 | export function getColorList(colors) { 25 | // For forwards compatibility, return early if it's already an object 26 | if (!Array.isArray(colors)) { 27 | return colors; 28 | } 29 | 30 | // For backwards compatibility 31 | const colorsList = {} 32 | colors.forEach( ( color, index ) => { 33 | colorsList[colorList[index]] = color 34 | }); 35 | 36 | return colorsList; 37 | } 38 | 39 | export function colorExists(name) { 40 | return colorList.indexOf(name) !== -1 41 | } 42 | -------------------------------------------------------------------------------- /src/lib/utils/config.js: -------------------------------------------------------------------------------- 1 | // Import plugins from '../plugins' 2 | // import { colorExists } from './colors' 3 | // import notify from 'hyper/notify' 4 | // 5 | // function getPluginFromListByName(pluginList, name) { 6 | // return pluginList.find(each => each.name === name) 7 | // } 8 | // 9 | // function mergeColorConfigs(defaultColor, userColor = false) { 10 | // if (!userColor || !colorExists(userColor)) { 11 | // return defaultColor 12 | // } 13 | // 14 | // return userColor 15 | // } 16 | // 17 | // function mergePluginConfigs(defaultPlugins, userPlugins) { 18 | // if (!userPlugins) { 19 | // return defaultPlugins 20 | // } 21 | // 22 | // return userPlugins.reduce((newPlugins, plugin) => { 23 | // const newPlugin = Object.assign({}, plugin) 24 | // const { name, options = false } = plugin 25 | // 26 | // if (typeof plugin !== 'object' || Array.isArray(plugin)) { 27 | // notify('HyperLine', '\'plugins\' array members in \'.hyper.js\' must be objects.') 28 | // return newPlugins 29 | // } 30 | // 31 | // const { options: defaultOptions = false } = getPluginFromListByName(defaultPlugins, name) 32 | // 33 | // if (!defaultOptions) { 34 | // notify('HyperLine', `Plugin with name "${name}" does not exist.`) 35 | // return newPlugins 36 | // } 37 | // 38 | // if (options) { 39 | // newPlugin.options = defaultOptions 40 | // } 41 | // 42 | // const { validateOptions: validator = false } = plugins[name] 43 | // if (validator) { 44 | // const errors = validator(options) 45 | // if (errors.length > 0) { 46 | // errors.forEach(error => notify(`HyperLine '${name}' plugin`, error)) 47 | // newPlugin.options = defaultOptions 48 | // } 49 | // } 50 | // 51 | // return [ ...newPlugins, plugin ] 52 | // }, []) 53 | // } 54 | // 55 | // export function getDefaultConfig(plugins) { 56 | // return { 57 | // color: 'black', 58 | // plugins: Object.keys(plugins).reduce((pluginsArray, pluginName) => { 59 | // const { defaultOptions } = plugins[pluginName] 60 | // 61 | // const plugin = { 62 | // name: pluginName, 63 | // options: defaultOptions 64 | // } 65 | // 66 | // return [ ...pluginsArray, plugin ] 67 | // }, []) 68 | // } 69 | // } 70 | // 71 | // export function mergeConfigs(defaultConfig, userConfig = false) { 72 | // if (!userConfig) { 73 | // return defaultConfig 74 | // } 75 | // 76 | // return { 77 | // color: mergeColorConfigs(defaultConfig.color, userConfig.color), 78 | // plugins: mergePluginConfigs(defaultConfig.plugins, userConfig.plugins) 79 | // } 80 | // } 81 | -------------------------------------------------------------------------------- /src/lib/utils/svg-icon.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import PropTypes from 'prop-types' 3 | import Component from 'hyper/component' 4 | 5 | export default class SvgIcon extends Component { 6 | static propTypes() { 7 | return { 8 | children: PropTypes.element.isRequired 9 | } 10 | } 11 | 12 | render() { 13 | return ( 14 | 15 | {this.props.children} 16 | 17 | 24 | 25 | ) 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /src/lib/utils/time.js: -------------------------------------------------------------------------------- 1 | import moment from 'moment' 2 | 3 | export default function formatUptime(uptime) { 4 | const uptimeInHours = Number((uptime / 3600).toFixed(0)) 5 | 6 | if (uptimeInHours === 0) { 7 | return '0h' 8 | } 9 | 10 | const uptimeInMoment = moment.duration(uptimeInHours, 'hours') 11 | const days = uptimeInMoment.days() 12 | const hours = uptimeInMoment.hours() 13 | const daysFormatted = days ? days + 'd' : '' 14 | const hoursFormatted = hours ? hours + 'h' : '' 15 | 16 | return [daysFormatted, hoursFormatted].filter(Boolean).join(' ') 17 | } 18 | -------------------------------------------------------------------------------- /tests/__snapshots__/time.spec.js.snap: -------------------------------------------------------------------------------- 1 | // Jest Snapshot v1, https://goo.gl/fbAQLP 2 | 3 | exports[`time formatUptime 1 day 4 hours 1`] = `"1d 4h"`; 4 | 5 | exports[`time formatUptime 3 days 1`] = `"3d"`; 6 | 7 | exports[`time formatUptime 3 days 8 minutes 1`] = `"3d"`; 8 | 9 | exports[`time formatUptime 3 hours 48 minutes 1`] = `"4h"`; 10 | 11 | exports[`time formatUptime 3 seconds 1`] = `"0h"`; 12 | 13 | exports[`time formatUptime 4 hours 1`] = `"4h"`; 14 | 15 | exports[`time formatUptime 23 hours 1`] = `"23h"`; 16 | 17 | exports[`time formatUptime 23 hours 17 minutes 1`] = `"23h"`; 18 | 19 | exports[`time formatUptime 23 hours 49 minutes 1`] = `"1d"`; 20 | 21 | exports[`time formatUptime 23 minutes 1`] = `"0h"`; 22 | 23 | exports[`time formatUptime 29 minutes 1`] = `"0h"`; 24 | 25 | exports[`time formatUptime 30 minutes 1`] = `"1h"`; 26 | 27 | exports[`time formatUptime 31 minutes 1`] = `"1h"`; 28 | 29 | exports[`time formatUptime is a function 1`] = `[Function]`; 30 | -------------------------------------------------------------------------------- /tests/spotify.spec.js: -------------------------------------------------------------------------------- 1 | /* eslint-env jest */ 2 | import spotify from 'spotify-node-applescript' 3 | 4 | describe('spotify', () => { 5 | it('should open spotify', () => { 6 | expect(spotify.openSpotify()).toBeUndefined() 7 | }) 8 | }) 9 | -------------------------------------------------------------------------------- /tests/time.spec.js: -------------------------------------------------------------------------------- 1 | /* eslint-env jest */ 2 | import formatUptime from '../src/lib/utils/time' 3 | 4 | describe('time', () => { 5 | describe('formatUptime', () => { 6 | it('is a function', () => expect(formatUptime).toMatchSnapshot()) 7 | it('3 seconds', () => expect(formatUptime(3)).toMatchSnapshot()) 8 | it('23 minutes', () => expect(formatUptime(1380)).toMatchSnapshot()) 9 | it('29 minutes', () => expect(formatUptime(1740)).toMatchSnapshot()) 10 | it('30 minutes', () => expect(formatUptime(1800)).toMatchSnapshot()) 11 | it('31 minutes', () => expect(formatUptime(1860)).toMatchSnapshot()) 12 | it('3 hours 48 minutes', () => 13 | expect(formatUptime(13680)).toMatchSnapshot()) 14 | it('4 hours', () => expect(formatUptime(14400)).toMatchSnapshot()) 15 | it('23 hours', () => expect(formatUptime(82800)).toMatchSnapshot()) 16 | it('23 hours 17 minutes', () => 17 | expect(formatUptime(83820)).toMatchSnapshot()) 18 | it('23 hours 49 minutes', () => 19 | expect(formatUptime(85740)).toMatchSnapshot()) 20 | it('1 day 4 hours', () => expect(formatUptime(100800)).toMatchSnapshot()) 21 | it('3 days', () => expect(formatUptime(259200)).toMatchSnapshot()) 22 | it('3 days 8 minutes', () => expect(formatUptime(259680)).toMatchSnapshot()) 23 | }) 24 | }) 25 | -------------------------------------------------------------------------------- /webpack.config.js: -------------------------------------------------------------------------------- 1 | const webpack = require('webpack') 2 | const nodeExternals = require('webpack-node-externals') 3 | 4 | module.exports = { 5 | target: 'node', 6 | entry: './src/index.js', 7 | output: { 8 | path: './dist', 9 | filename: 'hyperline.js', 10 | libraryTarget: 'commonjs' 11 | }, 12 | plugins: [ new webpack.DefinePlugin({ 'global.GENTLY': false }) ], 13 | externals: [ nodeExternals(), 'hyper/component', 'hyper/notify', 'hyper/decorate', 'react' ], 14 | module: { 15 | rules: [ 16 | { 17 | test: /\.json$/, 18 | loader: 'json-loader' 19 | }, 20 | { 21 | test: /\.js$/, 22 | loader: 'babel-loader', 23 | exclude: /node_modules/ 24 | } 25 | ] 26 | } 27 | } 28 | --------------------------------------------------------------------------------