2015-06-25 17:32:51 +00:00
|
|
|
# remote
|
|
|
|
|
2016-04-30 16:53:21 +00:00
|
|
|
> 메인 프로세스 모듈을 렌더러 프로세스에서 사용합니다.
|
|
|
|
|
2016-04-17 17:39:05 +00:00
|
|
|
`remote` 모듈은 메인 프로세스와 렌더러 프로세스(웹 페이지) 사이의 inter-process
|
2015-11-19 19:46:05 +00:00
|
|
|
(IPC) 통신을 간단하게 추상화 한 모듈입니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-11-19 19:46:05 +00:00
|
|
|
Electron의 메인 프로세스에선 GUI와 관련 있는(`dialog`, `menu`등) 모듈만 사용할 수
|
2016-04-17 17:39:05 +00:00
|
|
|
있습니다. 렌더러 프로세스에서 이러한 모듈들을 사용하려면 `ipc` 모듈을 통해 메인
|
2016-04-11 01:33:47 +00:00
|
|
|
프로세스와 inter-process 통신을 해야 합니다. 또한, `remote` 모듈을 사용하면
|
2015-11-19 19:46:05 +00:00
|
|
|
inter-process 통신을 하지 않고도 간단한 API를 통해 직접 메인 프로세스의 모듈과
|
|
|
|
메서드를 사용할 수 있습니다. 이 개념은 Java의 [RMI][rmi]와 비슷합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-04-30 16:12:54 +00:00
|
|
|
다음 예시는 렌더러 프로세스에서 브라우저 창을 만드는 예시입니다:
|
2015-06-25 17:32:51 +00:00
|
|
|
|
|
|
|
```javascript
|
2016-05-10 18:14:06 +00:00
|
|
|
const {BrowserWindow} = require('electron').remote;
|
2015-09-02 23:28:12 +00:00
|
|
|
|
2016-05-10 18:14:06 +00:00
|
|
|
let win = new BrowserWindow({width: 800, height: 600});
|
2015-11-16 03:15:21 +00:00
|
|
|
win.loadURL('https://github.com');
|
2015-06-25 17:32:51 +00:00
|
|
|
```
|
|
|
|
|
2016-07-12 16:12:04 +00:00
|
|
|
**참고:** 반대로 메인 프로세스에서 렌더러 프로세스에 접근 하려면 [webContents.executeJavascript](web-contents.md#webcontentsexecutejavascriptcode-usergesture-callback)
|
2015-11-19 19:46:05 +00:00
|
|
|
메서드를 사용하면 됩니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-07-17 00:18:16 +00:00
|
|
|
## Remote 객체
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-11-19 19:46:05 +00:00
|
|
|
`remote` 모듈로부터 반환된 각 객체(메서드 포함)는 메인 프로세스의 객체를 추상화 한
|
|
|
|
객체입니다. (우리는 그것을 remote 객체 또는 remote 함수라고 부릅니다) Remote 모듈의
|
|
|
|
메서드를 호출하거나, 객체에 접근하거나, 생성자로 객체를 생성하는 등의 작업은 실질적으로
|
|
|
|
동기형 inter-process 메시지를 보냅니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-04-30 16:12:54 +00:00
|
|
|
위의 예시에서 사용한 두 `BrowserWindow`와 `win`은 remote 객체입니다. 그리고
|
2016-04-17 17:39:05 +00:00
|
|
|
`new BrowserWindow`이 생성하는 `BrowserWindow` 객체는 렌더러 프로세스에서 생성되지
|
|
|
|
않습니다. 대신에 이 `BrowserWindow` 객체는 메인 프로세스에서 생성되며 렌더러
|
2015-11-19 19:46:05 +00:00
|
|
|
프로세스에 `win` 객체와 같이 이에 대응하는 remote 객체를 반환합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-05-29 16:59:32 +00:00
|
|
|
참고로 remote 객체가 처음 참조될 때 표시되는
|
|
|
|
[enumerable 속성](https://developer.mozilla.org/ko/docs/Web/JavaScript/Enumerability_and_ownership_of_properties)은
|
|
|
|
remote를 통해서만 접근할 수 있습니다.
|
2016-01-11 04:11:10 +00:00
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
## Remote 객체의 생명 주기
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-04-17 17:39:05 +00:00
|
|
|
Electron은 렌더러 프로세스의 remote 객체가 살아있는 한(다시 말해서 GC(garbage
|
2015-11-19 19:46:05 +00:00
|
|
|
collection)가 일어나지 않습니다) 대응하는 메인 프로세스의 객체는 릴리즈되지 않습니다.
|
2015-08-21 02:33:02 +00:00
|
|
|
Remote 객체가 GC 되려면 대응하는 메인 프로세스 내부 객체의 참조가 해제되어야만 합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-04-17 17:39:05 +00:00
|
|
|
만약 remote 객체가 렌더러 프로세스에서 누수가 생겼다면 (예시: 맵에 저장하고 할당
|
2015-11-19 19:46:05 +00:00
|
|
|
해제하지 않음) 대응하는 메인 프로세스의 객체도 누수가 생깁니다. 그래서 remote 객체를
|
|
|
|
사용할 땐 메모리 누수가 생기지 않도록 매우 주의해서 사용해야 합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-08-21 02:33:02 +00:00
|
|
|
참고로 문자열, 숫자와 같은 원시 값 타입은 복사에 의한 참조로 전달됩니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-08-21 02:33:02 +00:00
|
|
|
## 메인 프로세스로 콜백 넘기기
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-04-17 17:39:05 +00:00
|
|
|
메인 프로세스의 코드는 `remote` 모듈을 통해 렌더러 프로세스가 전달하는 콜백 함수를
|
2015-11-19 19:46:05 +00:00
|
|
|
받을 수 있습니다. 하지만 이 작업은 반드시 주의를 기울여 사용해야 합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-11-19 19:46:05 +00:00
|
|
|
첫째, 데드락을 피하기 위해 메인 프로세스로 전달된 콜백들은 비동기로 호출됩니다. 이러한
|
2016-05-10 07:27:14 +00:00
|
|
|
이유로 메인 프로세스에 전달된 콜백의 반환 값을 내부 함수에서 언제나 정상적으로 받을
|
2015-11-19 19:46:05 +00:00
|
|
|
것이라고 예측해선 안됩니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-04-17 17:39:05 +00:00
|
|
|
예를 들어 메인 프로세스에서 `Array.map` 같은 메서드를 사용할 때 렌더러 프로세스에서
|
2015-11-19 19:46:05 +00:00
|
|
|
전달된 함수를 사용해선 안됩니다:
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-08-31 05:48:47 +00:00
|
|
|
```javascript
|
|
|
|
// mapNumbers.js 메인 프로세스
|
2016-05-10 07:27:14 +00:00
|
|
|
exports.withRendererCallback = (mapper) => {
|
2015-08-31 05:48:47 +00:00
|
|
|
return [1,2,3].map(mapper);
|
2016-05-10 18:14:06 +00:00
|
|
|
;
|
2015-08-31 05:48:47 +00:00
|
|
|
|
2016-05-10 07:27:14 +00:00
|
|
|
exports.withLocalCallback = () => {
|
2016-07-12 16:12:04 +00:00
|
|
|
return [1,2,3].map(x => x + 1);
|
2016-05-10 18:14:06 +00:00
|
|
|
};
|
2015-09-02 23:28:12 +00:00
|
|
|
```
|
2015-08-31 05:48:47 +00:00
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
```javascript
|
2016-04-17 17:39:05 +00:00
|
|
|
// 렌더러 프로세스
|
2016-06-10 17:27:25 +00:00
|
|
|
const mapNumbers = require('electron').remote.require('./mapNumbers');
|
2015-08-31 05:48:47 +00:00
|
|
|
|
2016-05-10 07:27:14 +00:00
|
|
|
const withRendererCb = mapNumbers.withRendererCallback(x => x + 1);
|
2015-08-31 05:48:47 +00:00
|
|
|
|
2016-05-10 18:14:06 +00:00
|
|
|
const withLocalCb = mapNumbers.withLocalCallback();
|
2015-08-31 05:48:47 +00:00
|
|
|
|
2016-07-12 16:12:04 +00:00
|
|
|
console.log(withRendererCb, withLocalCb); // [undefined, undefined, undefined], [2, 3, 4]
|
2015-08-31 05:48:47 +00:00
|
|
|
```
|
|
|
|
|
2016-05-10 07:27:14 +00:00
|
|
|
보다시피 동기적인 렌더러 콜백 함수의 반환 값은 예상되지 않은 값입니다. 그리고 메인
|
|
|
|
프로세스에서 처리한 함수의 반환 값과 일치하지 않습니다.
|
2015-08-31 05:48:47 +00:00
|
|
|
|
2015-11-19 19:46:05 +00:00
|
|
|
둘째, 콜백들은 메인 프로세스로 전달, 호출된 이후에도 자동으로 함수의 참조가 릴리즈 되지
|
|
|
|
않습니다. 함수 참조는 메인 프로세스에서 GC가 일어나기 전까지 계속 프로세스에 남아있게
|
|
|
|
됩니다.
|
2015-08-31 05:48:47 +00:00
|
|
|
|
2016-04-30 16:12:54 +00:00
|
|
|
다음 코드를 보면 느낌이 올 것입니다. 이 예시는 remote 객체에 `close` 이벤트 콜백을
|
2016-05-10 07:27:14 +00:00
|
|
|
등록합니다:
|
2015-06-25 17:32:51 +00:00
|
|
|
|
|
|
|
```javascript
|
2016-05-10 07:27:14 +00:00
|
|
|
const remote = require('remote');
|
2015-09-02 23:28:12 +00:00
|
|
|
|
2016-05-10 07:27:14 +00:00
|
|
|
remote.getCurrentWindow().on('close', () => {
|
2015-06-25 17:32:51 +00:00
|
|
|
// blabla...
|
|
|
|
});
|
|
|
|
```
|
|
|
|
|
2016-05-10 07:27:14 +00:00
|
|
|
하지만 이 코드와 같이 등록된 이벤트는 명시적으로 제거하지 않는 이상 콜백 함수의 참조가
|
|
|
|
계속해서 메인 프로세스에 남아있게 됩니다. 만약 명시적으로 콜백을 제거하지 않으면 매 번
|
|
|
|
창을 새로고침 할 때마다 콜백을 새로 설치합니다. 게다가 이전 콜백이 제거되지 않고
|
|
|
|
계속해서 쌓이면서 메모리 누수가 발생합니다.
|
2015-08-31 05:48:47 +00:00
|
|
|
|
2016-05-18 03:15:29 +00:00
|
|
|
설상가상으로 이전에 등록된 콜백의 컨텍스트가 릴리즈 되고 난 후 (e.g. 페이지 새로고침)
|
2015-11-19 19:46:05 +00:00
|
|
|
`close` 이벤트가 발생하면 예외가 발생하고 메인 프로세스가 작동 중지됩니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-04-17 17:39:05 +00:00
|
|
|
이러한 문제를 피하려면 렌더러 프로세스에서 메인 프로세스로 넘긴 함수의 참조를 사용 후
|
2015-11-19 19:46:05 +00:00
|
|
|
확실하게 제거해야 합니다. 작업 후 이벤트 콜백을 포함하여 책임 있게 함수의 참조를
|
2016-04-17 17:39:05 +00:00
|
|
|
제거하거나 메인 프로세스에서 렌더러 프로세스가 종료될 때 내부적으로 함수 참조를
|
2015-11-19 19:46:05 +00:00
|
|
|
제거하도록 설계해야 합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-11-16 03:15:21 +00:00
|
|
|
## 메인 프로세스의 빌트인 모듈에 접근
|
|
|
|
|
2015-11-19 19:46:05 +00:00
|
|
|
메인 프로세스의 빌트인 모듈은 `remote` 모듈에 getter로 등록되어 있습니다. 따라서
|
|
|
|
`remote` 모듈을 `electron` 모듈처럼 직접 사용할 수 있습니다.
|
2015-11-16 03:15:21 +00:00
|
|
|
|
|
|
|
```javascript
|
|
|
|
const app = remote.app;
|
|
|
|
```
|
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
## Methods
|
|
|
|
|
|
|
|
`remote` 모듈은 다음과 같은 메서드를 가지고 있습니다:
|
|
|
|
|
|
|
|
### `remote.require(module)`
|
2015-06-25 17:32:51 +00:00
|
|
|
|
|
|
|
* `module` String
|
|
|
|
|
2015-08-21 02:33:02 +00:00
|
|
|
메인 프로세스의 `require(module)` API를 실행한 후 결과 객체를 반환합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
### `remote.getCurrentWindow()`
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
현재 웹 페이지가 들어있는 [`BrowserWindow`](browser-window.md) 객체를 반환합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
### `remote.getCurrentWebContents()`
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
현재 웹 페이지의 [`WebContents`](web-contents.md) 객체를 반환합니다.
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
### `remote.getGlobal(name)`
|
2015-06-25 17:32:51 +00:00
|
|
|
|
|
|
|
* `name` String
|
|
|
|
|
2015-08-21 02:33:02 +00:00
|
|
|
메인 프로세스의 전역 변수(`name`)를 가져옵니다. (예시: `global[name]`)
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-09-15 16:27:10 +00:00
|
|
|
## Properties
|
|
|
|
|
2015-09-02 23:28:12 +00:00
|
|
|
### `remote.process`
|
2015-06-25 17:32:51 +00:00
|
|
|
|
2016-09-15 16:27:10 +00:00
|
|
|
메인 프로세스의 `process` 객체입니다. `remote.getGlobal('process')`와
|
2015-11-19 19:46:05 +00:00
|
|
|
같습니다. 하지만 캐시 됩니다.
|
2015-11-16 03:15:21 +00:00
|
|
|
|
2015-11-19 19:46:05 +00:00
|
|
|
[rmi]: http://en.wikipedia.org/wiki/Java_remote_method_invocation
|