mirror of https://github.com/electron/electron
110 lines
3.4 KiB
Markdown
110 lines
3.4 KiB
Markdown
# Custom Window Interactions
|
|
|
|
## Custom draggable regions
|
|
|
|
By default, windows are dragged using the title bar provided by the OS chrome. Apps
|
|
that remove the default title bar need to use the `app-region` CSS property to define
|
|
specific areas that can be used to drag the window. Setting `app-region: drag` marks
|
|
a rectagular area as draggable.
|
|
|
|
It is important to note that draggable areas ignore all pointer events. For example,
|
|
a button element that overlaps a draggable region will not emit mouse clicks or mouse
|
|
enter/exit events within that overlapping area. Setting `app-region: no-drag` reenables
|
|
pointer events by excluding a rectagular area from a draggable region.
|
|
|
|
To make the whole window draggable, you can add `app-region: drag` as
|
|
`body`'s style:
|
|
|
|
```css title='styles.css'
|
|
body {
|
|
app-region: drag;
|
|
}
|
|
```
|
|
|
|
And note that if you have made the whole window draggable, you must also mark
|
|
buttons as non-draggable, otherwise it would be impossible for users to click on
|
|
them:
|
|
|
|
```css title='styles.css'
|
|
button {
|
|
app-region: no-drag;
|
|
}
|
|
```
|
|
|
|
If you're only setting a custom title bar as draggable, you also need to make all
|
|
buttons in title bar non-draggable.
|
|
|
|
### Tip: disable text selection
|
|
|
|
When creating a draggable region, the dragging behavior may conflict with text selection.
|
|
For example, when you drag the title bar, you may accidentally select its text contents.
|
|
To prevent this, you need to disable text selection within a draggable area like this:
|
|
|
|
```css
|
|
.titlebar {
|
|
user-select: none;
|
|
app-region: drag;
|
|
}
|
|
```
|
|
|
|
### Tip: disable context menus
|
|
|
|
On some platforms, the draggable area will be treated as a non-client frame, so
|
|
when you right click on it, a system menu will pop up. To make the context menu
|
|
behave correctly on all platforms, you should never use a custom context menu on
|
|
draggable areas.
|
|
|
|
## Click-through windows
|
|
|
|
To create a click-through window, i.e. making the window ignore all mouse
|
|
events, you can call the [win.setIgnoreMouseEvents(ignore)][ignore-mouse-events]
|
|
API:
|
|
|
|
```js title='main.js'
|
|
const { BrowserWindow } = require('electron')
|
|
|
|
const win = new BrowserWindow()
|
|
win.setIgnoreMouseEvents(true)
|
|
```
|
|
|
|
### Forward mouse events _macOS_ _Windows_
|
|
|
|
Ignoring mouse messages makes the web contents oblivious to mouse movement,
|
|
meaning that mouse movement events will not be emitted. On Windows and macOS, an
|
|
optional parameter can be used to forward mouse move messages to the web page,
|
|
allowing events such as `mouseleave` to be emitted:
|
|
|
|
```js title='main.js'
|
|
const { BrowserWindow, ipcMain } = require('electron')
|
|
|
|
const path = require('node:path')
|
|
|
|
const win = new BrowserWindow({
|
|
webPreferences: {
|
|
preload: path.join(__dirname, 'preload.js')
|
|
}
|
|
})
|
|
|
|
ipcMain.on('set-ignore-mouse-events', (event, ignore, options) => {
|
|
const win = BrowserWindow.fromWebContents(event.sender)
|
|
win.setIgnoreMouseEvents(ignore, options)
|
|
})
|
|
```
|
|
|
|
```js title='preload.js'
|
|
window.addEventListener('DOMContentLoaded', () => {
|
|
const el = document.getElementById('clickThroughElement')
|
|
el.addEventListener('mouseenter', () => {
|
|
ipcRenderer.send('set-ignore-mouse-events', true, { forward: true })
|
|
})
|
|
el.addEventListener('mouseleave', () => {
|
|
ipcRenderer.send('set-ignore-mouse-events', false)
|
|
})
|
|
})
|
|
```
|
|
|
|
This makes the web page click-through when over the `#clickThroughElement` element,
|
|
and returns to normal outside it.
|
|
|
|
[ignore-mouse-events]: ../api/browser-window.md#winsetignoremouseeventsignore-options
|