mirror of
https://github.com/quine-global/hyper.git
synced 2026-01-12 20:18:41 -09:00
* npm: add .npmrc with save-exact=true * split panes: create initial implementation This allows users to split their Hyperterm terms into multiple nested splits, both vertical and horizontal. Fixes #56 * split panes: suport closing tabs and individual panes * split panes: ensure new splits are placed at the correct index New split panes should be placed after the currently active pane, not at the end like they were previously. * split panes: add explicit dependency to uuid * split panes: implement split pane cycling This adds menu buttons for moving back and forward between open split panes in the currect terminal tab. Doesn't add a hotkey yet, needs some bikeshedding. * split panes: move activeSessionUid to its own object It made little sense to have so many objects with `activeSessionUid` set to `null` when it only mattered on the top level. Now it's an object mapping term-group `uid` to `sessionUid` instead. * split panes: make sure closing the last split pane exits the app * split panes: fix a crash after closing specific panes Sometimes the terminal would crash when a specific split pane was closed, because the `activeSessions` mapping wasn't updated correctly. * split panes: fix a bug that caused initial session sizing to be wrong * fix all our focus / blur issues in one fell swoop :O (famous last words) * get rid of react warning * hterm: make sure not to lose focus when VT listens on clicks * term: restore onactive callback * add missing `return` to override (just in case) * split pane: new split pane implementation * goodbye react-split-pane * added term group resizing action and reducer * terms: supply border color so that we can use it for splits * term-group: add resizing hook * term-groups: add resizing constant * remove split pane css side-effect * split panes: pass existing hterm instances to Term * split panes: add keybindings for split pane cycling * split panes: remove unused action * split panes: remove unused styling * split-pane: remove `console.log` * split-pane: remove `console.log` * split panes: rebalance sizes on insert/removal * split panes: pass existing hterm instances to Term * split panes: add keybindings for split pane cycling * split panes: remove unused action * split panes: remove unused styling * split panes: rebalance sizes on insert/removal * split panes: set a minimum size for resizing * split-pane: fix vertical splits * css :| * package: bump electron * split panes: attach onFocus listener to webviews * 1.4.1 and 1.4.2 are broken. they have the following regression: - open google.com on the main window - open a new tab - come back to previous tab. webview is gone :| * split panes: handle PTY exits * split panes: add linux friendly keybindings
183 lines
3.9 KiB
JavaScript
183 lines
3.9 KiB
JavaScript
import rpc from '../rpc';
|
|
import getURL from '../utils/url-command';
|
|
import {keys} from '../utils/object';
|
|
import {findBySession} from '../utils/term-groups';
|
|
import {
|
|
SESSION_ADD,
|
|
SESSION_RESIZE,
|
|
SESSION_REQUEST,
|
|
SESSION_ADD_DATA,
|
|
SESSION_PTY_DATA,
|
|
SESSION_PTY_EXIT,
|
|
SESSION_USER_EXIT,
|
|
SESSION_SET_ACTIVE,
|
|
SESSION_CLEAR_ACTIVE,
|
|
SESSION_USER_DATA,
|
|
SESSION_URL_SET,
|
|
SESSION_URL_UNSET,
|
|
SESSION_SET_XTERM_TITLE,
|
|
SESSION_SET_PROCESS_TITLE
|
|
} from '../constants/sessions';
|
|
|
|
export function addSession({uid, shell, pid, cols, rows, splitDirection}) {
|
|
return (dispatch, getState) => {
|
|
const {sessions} = getState();
|
|
dispatch({
|
|
type: SESSION_ADD,
|
|
uid,
|
|
shell,
|
|
pid,
|
|
cols,
|
|
rows,
|
|
splitDirection,
|
|
activeUid: sessions.activeUid
|
|
});
|
|
};
|
|
}
|
|
|
|
export function requestSession() {
|
|
return (dispatch, getState) => {
|
|
const {ui} = getState();
|
|
const {cols, rows, cwd} = ui;
|
|
dispatch({
|
|
type: SESSION_REQUEST,
|
|
effect: () => {
|
|
rpc.emit('new', {cols, rows, cwd});
|
|
}
|
|
});
|
|
};
|
|
}
|
|
|
|
export function addSessionData(uid, data) {
|
|
return function (dispatch, getState) {
|
|
dispatch({
|
|
type: SESSION_ADD_DATA,
|
|
data,
|
|
effect() {
|
|
const {shell} = getState().sessions.sessions[uid];
|
|
|
|
const enterKey = Boolean(data.match(/\n/));
|
|
const url = enterKey ? getURL(shell, data) : null;
|
|
|
|
if (url) {
|
|
dispatch({
|
|
type: SESSION_URL_SET,
|
|
uid,
|
|
url
|
|
});
|
|
} else {
|
|
dispatch({
|
|
type: SESSION_PTY_DATA,
|
|
uid,
|
|
data
|
|
});
|
|
}
|
|
}
|
|
});
|
|
};
|
|
}
|
|
|
|
function createExitAction(type) {
|
|
return uid => (dispatch, getState) => {
|
|
return dispatch({
|
|
type,
|
|
uid,
|
|
effect() {
|
|
if (type === SESSION_USER_EXIT) {
|
|
rpc.emit('exit', {uid});
|
|
}
|
|
|
|
const sessions = keys(getState().sessions.sessions);
|
|
if (!sessions.length) {
|
|
window.close();
|
|
}
|
|
}
|
|
});
|
|
};
|
|
}
|
|
|
|
// we want to distinguish an exit
|
|
// that's UI initiated vs pty initiated
|
|
export const userExitSession = createExitAction(SESSION_USER_EXIT);
|
|
export const ptyExitSession = createExitAction(SESSION_PTY_EXIT);
|
|
|
|
export function setActiveSession(uid) {
|
|
return (dispatch, getState) => {
|
|
const state = getState();
|
|
const prevUid = state.sessions.activeUid;
|
|
dispatch({
|
|
type: SESSION_SET_ACTIVE,
|
|
uid,
|
|
effect() {
|
|
// TODO: this goes away when we are able to poll
|
|
// for the title ourseleves, instead of relying
|
|
// on Session and focus/blur to subscribe
|
|
if (prevUid) {
|
|
rpc.emit('blur', {uid: prevUid});
|
|
}
|
|
rpc.emit('focus', {uid});
|
|
}
|
|
});
|
|
};
|
|
}
|
|
|
|
export function clearActiveSession() {
|
|
return {
|
|
type: SESSION_CLEAR_ACTIVE
|
|
};
|
|
}
|
|
|
|
export function setSessionProcessTitle(uid, title) {
|
|
return {
|
|
type: SESSION_SET_PROCESS_TITLE,
|
|
uid,
|
|
title
|
|
};
|
|
}
|
|
|
|
export function setSessionXtermTitle(uid, title) {
|
|
return {
|
|
type: SESSION_SET_XTERM_TITLE,
|
|
uid,
|
|
title
|
|
};
|
|
}
|
|
|
|
export function resizeSession(uid, cols, rows) {
|
|
return (dispatch, getState) => {
|
|
const {termGroups} = getState();
|
|
const group = findBySession(termGroups, uid);
|
|
const isStandaloneTerm = !group.parentUid && !group.children.length;
|
|
dispatch({
|
|
type: SESSION_RESIZE,
|
|
uid,
|
|
cols,
|
|
rows,
|
|
isStandaloneTerm,
|
|
effect() {
|
|
rpc.emit('resize', {uid, cols, rows});
|
|
}
|
|
});
|
|
};
|
|
}
|
|
|
|
export function sendSessionData(uid, data) {
|
|
return function (dispatch, getState) {
|
|
dispatch({
|
|
type: SESSION_USER_DATA,
|
|
data,
|
|
effect() {
|
|
// If no uid is passed, data is sended to the active session.
|
|
const targetUid = uid || getState().sessions.activeUid;
|
|
rpc.emit('data', {uid: targetUid, data});
|
|
}
|
|
});
|
|
};
|
|
}
|
|
|
|
export function exitSessionBrowser(uid) {
|
|
return {
|
|
type: SESSION_URL_UNSET,
|
|
uid
|
|
};
|
|
}
|