171 lines
		
	
	
	
		
			5.2 KiB
			
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			171 lines
		
	
	
	
		
			5.2 KiB
			
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 'use strict'
 | |
| 
 | |
| const ipcMain = require('@electron/internal/browser/ipc-main-internal')
 | |
| 
 | |
| // The history operation in renderer is redirected to browser.
 | |
| ipcMain.on('ELECTRON_NAVIGATION_CONTROLLER', function (event, method, ...args) {
 | |
|   event.sender[method](...args)
 | |
| })
 | |
| 
 | |
| ipcMain.on('ELECTRON_SYNC_NAVIGATION_CONTROLLER', function (event, method, ...args) {
 | |
|   event.returnValue = event.sender[method](...args)
 | |
| })
 | |
| 
 | |
| // JavaScript implementation of Chromium's NavigationController.
 | |
| // Instead of relying on Chromium for history control, we compeletely do history
 | |
| // control on user land, and only rely on WebContents.loadURL for navigation.
 | |
| // This helps us avoid Chromium's various optimizations so we can ensure renderer
 | |
| // process is restarted everytime.
 | |
| const NavigationController = (function () {
 | |
|   function NavigationController (webContents) {
 | |
|     this.webContents = webContents
 | |
|     this.clearHistory()
 | |
| 
 | |
|     // webContents may have already navigated to a page.
 | |
|     if (this.webContents._getURL()) {
 | |
|       this.currentIndex++
 | |
|       this.history.push(this.webContents._getURL())
 | |
|     }
 | |
|     this.webContents.on('navigation-entry-commited', (event, url, inPage, replaceEntry) => {
 | |
|       if (this.inPageIndex > -1 && !inPage) {
 | |
|         // Navigated to a new page, clear in-page mark.
 | |
|         this.inPageIndex = -1
 | |
|       } else if (this.inPageIndex === -1 && inPage && !replaceEntry) {
 | |
|         // Started in-page navigations.
 | |
|         this.inPageIndex = this.currentIndex
 | |
|       }
 | |
|       if (this.pendingIndex >= 0) {
 | |
|         // Go to index.
 | |
|         this.currentIndex = this.pendingIndex
 | |
|         this.pendingIndex = -1
 | |
|         this.history[this.currentIndex] = url
 | |
|       } else if (replaceEntry) {
 | |
|         // Non-user initialized navigation.
 | |
|         this.history[this.currentIndex] = url
 | |
|       } else {
 | |
|         // Normal navigation. Clear history.
 | |
|         this.history = this.history.slice(0, this.currentIndex + 1)
 | |
|         this.currentIndex++
 | |
|         this.history.push(url)
 | |
|       }
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.loadURL = function (url, options) {
 | |
|     if (options == null) {
 | |
|       options = {}
 | |
|     }
 | |
|     this.pendingIndex = -1
 | |
|     this.webContents._loadURL(url, options)
 | |
|     return this.webContents.emit('load-url', url, options)
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.getURL = function () {
 | |
|     if (this.currentIndex === -1) {
 | |
|       return ''
 | |
|     } else {
 | |
|       return this.history[this.currentIndex]
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.stop = function () {
 | |
|     this.pendingIndex = -1
 | |
|     return this.webContents._stop()
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.reload = function () {
 | |
|     this.pendingIndex = this.currentIndex
 | |
|     return this.webContents._loadURL(this.getURL(), {})
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.reloadIgnoringCache = function () {
 | |
|     this.pendingIndex = this.currentIndex
 | |
|     return this.webContents._loadURL(this.getURL(), {
 | |
|       extraHeaders: 'pragma: no-cache\n'
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.canGoBack = function () {
 | |
|     return this.getActiveIndex() > 0
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.canGoForward = function () {
 | |
|     return this.getActiveIndex() < this.history.length - 1
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.canGoToIndex = function (index) {
 | |
|     return index >= 0 && index < this.history.length
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.canGoToOffset = function (offset) {
 | |
|     return this.canGoToIndex(this.currentIndex + offset)
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.clearHistory = function () {
 | |
|     this.history = []
 | |
|     this.currentIndex = -1
 | |
|     this.pendingIndex = -1
 | |
|     this.inPageIndex = -1
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.goBack = function () {
 | |
|     if (!this.canGoBack()) {
 | |
|       return
 | |
|     }
 | |
|     this.pendingIndex = this.getActiveIndex() - 1
 | |
|     if (this.inPageIndex > -1 && this.pendingIndex >= this.inPageIndex) {
 | |
|       return this.webContents._goBack()
 | |
|     } else {
 | |
|       return this.webContents._loadURL(this.history[this.pendingIndex], {})
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.goForward = function () {
 | |
|     if (!this.canGoForward()) {
 | |
|       return
 | |
|     }
 | |
|     this.pendingIndex = this.getActiveIndex() + 1
 | |
|     if (this.inPageIndex > -1 && this.pendingIndex >= this.inPageIndex) {
 | |
|       return this.webContents._goForward()
 | |
|     } else {
 | |
|       return this.webContents._loadURL(this.history[this.pendingIndex], {})
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.goToIndex = function (index) {
 | |
|     if (!this.canGoToIndex(index)) {
 | |
|       return
 | |
|     }
 | |
|     this.pendingIndex = index
 | |
|     return this.webContents._loadURL(this.history[this.pendingIndex], {})
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.goToOffset = function (offset) {
 | |
|     if (!this.canGoToOffset(offset)) {
 | |
|       return
 | |
|     }
 | |
|     const pendingIndex = this.currentIndex + offset
 | |
|     if (this.inPageIndex > -1 && pendingIndex >= this.inPageIndex) {
 | |
|       this.pendingIndex = pendingIndex
 | |
|       return this.webContents._goToOffset(offset)
 | |
|     } else {
 | |
|       return this.goToIndex(pendingIndex)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.getActiveIndex = function () {
 | |
|     if (this.pendingIndex === -1) {
 | |
|       return this.currentIndex
 | |
|     } else {
 | |
|       return this.pendingIndex
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   NavigationController.prototype.length = function () {
 | |
|     return this.history.length
 | |
|   }
 | |
| 
 | |
|   return NavigationController
 | |
| })()
 | |
| 
 | |
| module.exports = NavigationController
 | 
