hyper/lib/actions/sessions.js
Martin Ek a7595c1a45 Split Panes (#693)
* 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
2016-10-03 19:00:50 -07:00

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
};
}