I’ve been working on updating a electron app from 11.x to 12.x and have run into an issue where the apis declared by contextBridge.exposeInMainWorld
come as undefined when called upon through window.
This is my preload.js file
JavaScript
x
26
26
1
const { contextBridge, ipcRenderer } = require('electron');
2
const { path } = require('path')
3
4
contextBridge.exposeInMainWorld('api',{
5
getPath: (filePath) => {
6
return path.parse(filePath).base;
7
},
8
9
removeAllListeners: (ListenerType) => {
10
ipcRenderer.removeAllListeners(ListenerType);
11
},
12
13
openNewPDF: (pdf) => {
14
ipcRenderer.send('openNewPDF',pdf);
15
},
16
newWindow: (file) => {
17
ipcRenderer.send('newWindow',file);
18
},
19
togglePrinting: (value) => {
20
ipcRenderer.send('togglePrinting',value)
21
},
22
resizeWindow: (value) => {
23
ipcRenderer.send('resizeWindow', value)
24
}
25
});
26
my app.js
JavaScript
1
149
149
1
function createWindow(filename = null) {
2
// Create the browser window.
3
let win = new BrowserWindow({
4
width: 550,
5
height: 420,
6
minWidth: 565,
7
minHeight: 200,
8
preload: path.resolve(path.join(__dirname, 'app/preload.js')),
9
resizable: true,
10
titleBarStyle: 'default',
11
show: false
12
});
13
wins.push(win);
14
15
// and load the index.html of the app.
16
17
win.loadFile('app/index.html');
18
win.openDevTools();
19
let wc = win.webContents
20
wc.on('will-navigate', function (e, url) {
21
if (url != wc.getURL()) {
22
e.preventDefault()
23
shell.openExternal(url)
24
}
25
})
26
27
win.once('closed', () => {
28
// Dereference the window object, usually you would store windows
29
// in an array if your app supports multi windows, this is the time
30
// when you should delete the corresponding element.
31
wins = [];
32
});
33
34
win.webContents.removeAllListeners('did-finish-load');
35
win.webContents.once('did-finish-load', () => {
36
if (filename) {
37
win.webContents.send('file-open', filename);
38
win.show();
39
} else {
40
win.show();
41
}
42
});
43
44
45
if (!menuIsConfigured) {
46
const menu = Menu.buildFromTemplate(menuTemplate);
47
48
menu.getMenuItemById('file-open').click = () => {
49
openNewPDF();
50
};
51
52
menu.getMenuItemById('file-print').click = () => {
53
const focusedWin = BrowserWindow.getFocusedWindow();
54
focusedWin.webContents.send('file-print');
55
};
56
57
Menu.setApplicationMenu(menu);
58
menuIsConfigured = true;
59
}
60
61
const openNewPDF = () => {
62
dialog
63
.showOpenDialog(null, {
64
properties: ['openFile'],
65
filters: [{ name: 'PDF Files', extensions: ['pdf'] }]
66
})
67
.then((dialogReturn) => {
68
const filename = dialogReturn['filePaths'][0];
69
if (filename) {
70
if (wins.length === 0) {
71
createWindow(filename.toString());
72
} else {
73
const focusedWin = BrowserWindow.getFocusedWindow();
74
if (focusedWin) {
75
focusedWin.webContents.send('file-open', filename.toString());
76
}
77
}
78
}
79
});
80
}
81
82
ipcMain.removeAllListeners('togglePrinting');
83
ipcMain.once('togglePrinting', (e, msg) => {
84
const menu = Menu.getApplicationMenu();
85
menu.getMenuItemById('file-print').enabled = Boolean(msg);
86
});
87
88
ipcMain.removeAllListeners('newWindow');
89
ipcMain.once('newWindow', (e, msg) => {
90
console.log('opening ', msg, ' in new window');
91
createWindow(msg);
92
});
93
94
ipcMain.removeAllListeners('resizeWindow');
95
ipcMain.once('resizeWindow', (e, msg) => {
96
const { width, height } = win.getBounds();
97
if (width < 1000 || height < 650) {
98
win.setResizable(true);
99
win.setSize(1000, 650);
100
win.center();
101
}
102
});
103
104
ipcMain.removeAllListeners('openNewPDF');
105
ipcMain.once('openNewPDF', (e, msg) => {
106
openNewPDF();
107
});
108
}
109
110
let fileToOpen = '';
111
112
const args = process.argv;
113
const argsLength = args.length;
114
if (argsLength > 1 && args[argsLength - 1].endsWith('.pdf')) {
115
fileToOpen = args[argsLength - 1];
116
}
117
118
app.on('open-file', (event, path) => {
119
event.preventDefault();
120
if (app.isReady()) {
121
if (wins.length === 0) {
122
createWindow(path.toString());
123
} else {
124
const focusedWin = BrowserWindow.getFocusedWindow();
125
focusedWin.webContents.send('file-open', path.toString());
126
}
127
}
128
fileToOpen = path.toString();
129
});
130
131
app.whenReady().then(() => {
132
if (fileToOpen) {
133
createWindow(fileToOpen);
134
} else {
135
createWindow();
136
}
137
});
138
139
140
app.on('window-all-closed', () => {
141
app.quit()
142
});
143
144
app.on('activate', () => {
145
if (BrowserWindow.getAllWindows().length === 0) {
146
createWindow();
147
}
148
});
149
I’m lost on why contextBridge
doesn’t work.
Advertisement
Answer
The object passed to the BrowserWindow
constructor is not correct. The preload
option should be a property of webPreferences
JavaScript
1
6
1
const win = new BrowserWindow({
2
webPreferences: {
3
preload: YOUR_PRELOAD_SCRIPT_PATH
4
}
5
});
6
See the docs