2020-11-07 15:59:37 +00:00
import ResourceEditWatcher from '@joplin/lib/services/ResourceEditWatcher/index' ;
import CommandService from '@joplin/lib/services/CommandService' ;
import KeymapService from '@joplin/lib/services/KeymapService' ;
2021-08-05 12:48:39 +01:00
import PluginService , { PluginSettings } from '@joplin/lib/services/plugins/PluginService' ;
2020-11-07 15:59:37 +00:00
import resourceEditWatcherReducer , { defaultState as resourceEditWatcherDefaultState } from '@joplin/lib/services/ResourceEditWatcher/reducer' ;
2020-10-09 18:35:46 +01:00
import PluginRunner from './services/plugins/PluginRunner' ;
import PlatformImplementation from './services/plugins/PlatformImplementation' ;
2020-11-07 15:59:37 +00:00
import shim from '@joplin/lib/shim' ;
import AlarmService from '@joplin/lib/services/AlarmService' ;
import AlarmServiceDriverNode from '@joplin/lib/services/AlarmServiceDriverNode' ;
import Logger , { TargetType } from '@joplin/lib/Logger' ;
import Setting from '@joplin/lib/models/Setting' ;
import actionApi from '@joplin/lib/services/rest/actionApi.desktop' ;
import BaseApplication from '@joplin/lib/BaseApplication' ;
2021-05-13 18:57:37 +02:00
import DebugService from '@joplin/lib/debug/DebugService' ;
2020-11-07 15:59:37 +00:00
import { _ , setLocale } from '@joplin/lib/locale' ;
import SpellCheckerService from '@joplin/lib/services/spellChecker/SpellCheckerService' ;
2020-11-05 16:58:23 +00:00
import SpellCheckerServiceDriverNative from './services/spellChecker/SpellCheckerServiceDriverNative' ;
import bridge from './services/bridge' ;
2020-10-31 16:29:17 +00:00
import menuCommandNames from './gui/menuCommandNames' ;
2020-11-13 17:09:28 +00:00
import stateToWhenClauseContext from './services/commands/stateToWhenClauseContext' ;
import ResourceService from '@joplin/lib/services/ResourceService' ;
2020-11-16 11:03:44 +00:00
import ExternalEditWatcher from '@joplin/lib/services/ExternalEditWatcher' ;
2021-09-04 12:37:22 +01:00
import appReducer , { createAppDefaultState } from './app.reducer' ;
2020-11-07 15:59:37 +00:00
const { FoldersScreenUtils } = require ( '@joplin/lib/folders-screen-utils.js' ) ;
2021-01-22 17:41:11 +00:00
import Folder from '@joplin/lib/models/Folder' ;
2020-10-09 18:35:46 +01:00
const fs = require ( 'fs-extra' ) ;
2021-01-22 17:41:11 +00:00
import Tag from '@joplin/lib/models/Tag' ;
2021-01-29 18:45:11 +00:00
import { reg } from '@joplin/lib/registry' ;
2020-10-09 18:35:46 +01:00
const packageInfo = require ( './packageInfo.js' ) ;
2021-01-23 15:51:19 +00:00
import DecryptionWorker from '@joplin/lib/services/DecryptionWorker' ;
2021-06-22 19:57:04 +01:00
import ClipperServer from '@joplin/lib/ClipperServer' ;
2020-10-09 18:35:46 +01:00
const { webFrame } = require ( 'electron' ) ;
const Menu = bridge ( ) . Menu ;
2020-11-07 15:59:37 +00:00
const PluginManager = require ( '@joplin/lib/services/PluginManager' ) ;
2021-01-23 15:51:19 +00:00
import RevisionService from '@joplin/lib/services/RevisionService' ;
import MigrationService from '@joplin/lib/services/MigrationService' ;
2021-07-23 11:05:21 +01:00
import { loadCustomCss , injectCustomStyles } from '@joplin/lib/CssUtils' ;
2021-09-04 13:43:25 +01:00
import mainScreenCommands from './gui/MainScreen/commands/index' ;
import noteEditorCommands from './gui/NoteEditor/commands/index' ;
import noteListCommands from './gui/NoteList/commands/index' ;
import noteListControlsCommands from './gui/NoteListControls/commands/index' ;
import sidebarCommands from './gui/Sidebar/commands/index' ;
import appCommands from './commands/index' ;
import libCommands from '@joplin/lib/commands/index' ;
2021-10-17 14:54:48 +01:00
import { homedir } from 'os' ;
2021-10-01 19:35:27 +01:00
const electronContextMenu = require ( './services/electron-context-menu' ) ;
2021-01-22 17:41:11 +00:00
// import populateDatabase from '@joplin/lib/services/debug/populateDatabase';
2020-10-09 18:35:46 +01:00
2021-09-04 13:43:25 +01:00
const commands = mainScreenCommands
. concat ( noteEditorCommands )
. concat ( noteListCommands )
. concat ( noteListControlsCommands )
. concat ( sidebarCommands ) ;
2020-10-09 18:35:46 +01:00
// Commands that are not tied to any particular component.
// The runtime for these commands can be loaded when the app starts.
2021-09-04 13:43:25 +01:00
const globalCommands = appCommands . concat ( libCommands ) ;
2020-10-09 18:35:46 +01:00
2021-09-04 13:43:25 +01:00
import editorCommandDeclarations from './gui/NoteEditor/editorCommandDeclarations' ;
2021-11-12 00:33:37 +09:00
import PerFolderSortOrderService from './services/sortOrder/PerFolderSortOrderService' ;
2021-05-13 18:57:37 +02:00
import ShareService from '@joplin/lib/services/share/ShareService' ;
2021-05-14 11:29:06 +02:00
import checkForUpdates from './checkForUpdates' ;
2021-09-04 18:11:29 +01:00
import { AppState } from './app.reducer' ;
2021-10-17 18:50:56 +01:00
import syncDebugLog from '@joplin/lib/services/synchronizer/syncDebugLog' ;
2021-10-03 16:00:49 +01:00
// import { runIntegrationTests } from '@joplin/lib/services/e2ee/ppkTestUtils';
2020-10-09 18:35:46 +01:00
const pluginClasses = [
2020-10-18 21:52:10 +01:00
require ( './plugins/GotoAnything' ) . default ,
2020-10-09 18:35:46 +01:00
] ;
2021-09-04 12:37:22 +01:00
const appDefaultState = createAppDefaultState (
bridge ( ) . windowContentSize ( ) ,
resourceEditWatcherDefaultState
) ;
2020-10-09 18:35:46 +01:00
class Application extends BaseApplication {
2021-02-07 16:47:56 +00:00
private checkAllPluginStartedIID_ : any = null ;
2021-10-03 16:00:49 +01:00
public constructor ( ) {
2020-10-09 18:35:46 +01:00
super ( ) ;
this . bridge_nativeThemeUpdated = this . bridge_nativeThemeUpdated . bind ( this ) ;
}
2021-10-03 16:00:49 +01:00
public hasGui() {
2020-10-09 18:35:46 +01:00
return true ;
}
2021-10-03 16:00:49 +01:00
public reducer ( state : AppState = appDefaultState , action : any ) {
2021-09-04 12:37:22 +01:00
let newState = appReducer ( state , action ) ;
2020-10-09 18:35:46 +01:00
newState = resourceEditWatcherReducer ( newState , action ) ;
newState = super . reducer ( newState , action ) ;
return newState ;
}
2021-10-03 16:00:49 +01:00
public toggleDevTools ( visible : boolean ) {
2020-10-09 18:35:46 +01:00
if ( visible ) {
bridge ( ) . openDevTools ( ) ;
} else {
bridge ( ) . closeDevTools ( ) ;
}
}
2021-10-03 16:00:49 +01:00
protected async generalMiddleware ( store : any , next : any , action : any ) {
2020-10-09 18:35:46 +01:00
if ( action . type == 'SETTING_UPDATE_ONE' && action . key == 'locale' || action . type == 'SETTING_UPDATE_ALL' ) {
setLocale ( Setting . value ( 'locale' ) ) ;
// The bridge runs within the main process, with its own instance of locale.js
// so it needs to be set too here.
bridge ( ) . setLocale ( Setting . value ( 'locale' ) ) ;
}
if ( action . type == 'SETTING_UPDATE_ONE' && action . key == 'showTrayIcon' || action . type == 'SETTING_UPDATE_ALL' ) {
this . updateTray ( ) ;
}
if ( action . type == 'SETTING_UPDATE_ONE' && action . key == 'style.editor.fontFamily' || action . type == 'SETTING_UPDATE_ALL' ) {
this . updateEditorFont ( ) ;
}
if ( action . type == 'SETTING_UPDATE_ONE' && action . key == 'windowContentZoomFactor' || action . type == 'SETTING_UPDATE_ALL' ) {
webFrame . setZoomFactor ( Setting . value ( 'windowContentZoomFactor' ) / 100 ) ;
}
if ( [ 'EVENT_NOTE_ALARM_FIELD_CHANGE' , 'NOTE_DELETE' ] . indexOf ( action . type ) >= 0 ) {
await AlarmService . updateNoteNotification ( action . id , action . type === 'NOTE_DELETE' ) ;
}
const result = await super . generalMiddleware ( store , next , action ) ;
const newState = store . getState ( ) ;
if ( [ 'NOTE_VISIBLE_PANES_TOGGLE' , 'NOTE_VISIBLE_PANES_SET' ] . indexOf ( action . type ) >= 0 ) {
Setting . setValue ( 'noteVisiblePanes' , newState . noteVisiblePanes ) ;
}
if ( [ 'NOTE_DEVTOOLS_TOGGLE' , 'NOTE_DEVTOOLS_SET' ] . indexOf ( action . type ) >= 0 ) {
this . toggleDevTools ( newState . devToolsVisible ) ;
}
if ( action . type === 'FOLDER_AND_NOTE_SELECT' ) {
await Folder . expandTree ( newState . folders , action . folderId ) ;
}
if ( this . hasGui ( ) && ( ( action . type == 'SETTING_UPDATE_ONE' && [ 'themeAutoDetect' , 'theme' , 'preferredLightTheme' , 'preferredDarkTheme' ] . includes ( action . key ) ) || action . type == 'SETTING_UPDATE_ALL' ) ) {
this . handleThemeAutoDetect ( ) ;
}
return result ;
}
2021-10-03 16:00:49 +01:00
public handleThemeAutoDetect() {
2020-10-09 18:35:46 +01:00
if ( ! Setting . value ( 'themeAutoDetect' ) ) return ;
if ( bridge ( ) . shouldUseDarkColors ( ) ) {
Setting . setValue ( 'theme' , Setting . value ( 'preferredDarkTheme' ) ) ;
} else {
Setting . setValue ( 'theme' , Setting . value ( 'preferredLightTheme' ) ) ;
}
}
2021-10-03 16:00:49 +01:00
private bridge_nativeThemeUpdated() {
2020-10-09 18:35:46 +01:00
this . handleThemeAutoDetect ( ) ;
}
2021-10-03 16:00:49 +01:00
public updateTray() {
2020-10-09 18:35:46 +01:00
const app = bridge ( ) . electronApp ( ) ;
if ( app . trayShown ( ) === Setting . value ( 'showTrayIcon' ) ) return ;
if ( ! Setting . value ( 'showTrayIcon' ) ) {
app . destroyTray ( ) ;
} else {
const contextMenu = Menu . buildFromTemplate ( [
{ label : _ ( 'Open %s' , app . electronApp ( ) . name ) , click : ( ) = > { app . window ( ) . show ( ) ; } } ,
{ type : 'separator' } ,
2020-11-25 14:40:25 +00:00
{ label : _ ( 'Quit' ) , click : ( ) = > { void app . quit ( ) ; } } ,
2020-10-09 18:35:46 +01:00
] ) ;
app . createTray ( contextMenu ) ;
}
}
2021-10-03 16:00:49 +01:00
public updateEditorFont() {
2020-10-09 18:35:46 +01:00
const fontFamilies = [ ] ;
if ( Setting . value ( 'style.editor.fontFamily' ) ) fontFamilies . push ( ` " ${ Setting . value ( 'style.editor.fontFamily' ) } " ` ) ;
2021-03-26 03:08:22 -06:00
fontFamilies . push ( 'Avenir, Arial, sans-serif' ) ;
2020-10-09 18:35:46 +01:00
// The '*' and '!important' parts are necessary to make sure Russian text is displayed properly
// https://github.com/laurent22/joplin/issues/155
const css = ` .CodeMirror * { font-family: ${ fontFamilies . join ( ', ' ) } !important; } ` ;
const styleTag = document . createElement ( 'style' ) ;
styleTag . type = 'text/css' ;
styleTag . appendChild ( document . createTextNode ( css ) ) ;
document . head . appendChild ( styleTag ) ;
}
2021-10-03 16:00:49 +01:00
public setupContextMenu() {
2021-10-01 19:35:27 +01:00
// bridge().setupContextMenu((misspelledWord: string, dictionarySuggestions: string[]) => {
// let output = SpellCheckerService.instance().contextMenuItems(misspelledWord, dictionarySuggestions);
// console.info(misspelledWord, dictionarySuggestions);
// console.info(output);
// output = output.map(o => {
// delete o.click;
// return o;
// });
// return output;
// });
2020-11-08 01:08:33 +00:00
const MenuItem = bridge ( ) . MenuItem ;
2020-11-05 16:58:23 +00:00
// The context menu must be setup in renderer process because that's where
// the spell checker service lives.
2021-10-01 19:35:27 +01:00
electronContextMenu ( {
2020-11-12 19:13:28 +00:00
shouldShowMenu : ( _event : any , params : any ) = > {
2020-11-05 16:58:23 +00:00
// params.inputFieldType === 'none' when right-clicking the text editor. This is a bit of a hack to detect it because in this
// case we don't want to use the built-in context menu but a custom one.
return params . isEditable && params . inputFieldType !== 'none' ;
} ,
2020-11-12 19:13:28 +00:00
menu : ( actions : any , props : any ) = > {
const spellCheckerMenuItems = SpellCheckerService . instance ( ) . contextMenuItems ( props . misspelledWord , props . dictionarySuggestions ) . map ( ( item : any ) = > new MenuItem ( item ) ) ;
2020-11-05 16:58:23 +00:00
const output = [
actions . cut ( ) ,
actions . copy ( ) ,
actions . paste ( ) ,
. . . spellCheckerMenuItems ,
] ;
return output ;
} ,
} ) ;
}
2020-11-12 19:13:28 +00:00
async loadCustomCss ( filePath : string ) {
2020-10-09 18:35:46 +01:00
let cssString = '' ;
if ( await fs . pathExists ( filePath ) ) {
try {
cssString = await fs . readFile ( filePath , 'utf-8' ) ;
} catch ( error ) {
let msg = error . message ? error . message : '' ;
msg = ` Could not load custom css from ${ filePath } \ n ${ msg } ` ;
error . message = msg ;
throw error ;
}
}
return cssString ;
}
2021-08-06 10:58:32 +01:00
private async checkForLegacyTemplates() {
const templatesDir = ` ${ Setting . value ( 'profileDir' ) } /templates ` ;
if ( await shim . fsDriver ( ) . exists ( templatesDir ) ) {
try {
const files = await shim . fsDriver ( ) . readDirStats ( templatesDir ) ;
for ( const file of files ) {
if ( file . path . endsWith ( '.md' ) ) {
// There is atleast one template.
this . store ( ) . dispatch ( {
type : 'CONTAINS_LEGACY_TEMPLATES' ,
} ) ;
break ;
}
}
} catch ( error ) {
reg . logger ( ) . error ( ` Failed to read templates directory: ${ error } ` ) ;
}
}
}
2020-11-13 17:09:28 +00:00
private async initPluginService() {
2020-11-19 12:34:49 +00:00
const service = PluginService . instance ( ) ;
2020-11-13 17:09:28 +00:00
const pluginRunner = new PluginRunner ( ) ;
2020-11-19 12:34:49 +00:00
service . initialize ( packageInfo . version , PlatformImplementation . instance ( ) , pluginRunner , this . store ( ) ) ;
2021-04-24 20:23:33 +02:00
service . isSafeMode = Setting . value ( 'isSafeMode' ) ;
2020-11-19 12:34:49 +00:00
const pluginSettings = service . unserializePluginSettings ( Setting . value ( 'plugins.states' ) ) ;
2021-08-05 12:48:39 +01:00
{
// Users can add and remove plugins from the config screen at any
// time, however we only effectively uninstall the plugin the next
// time the app is started. What plugin should be uninstalled is
// stored in the settings.
const newSettings = service . clearUpdateState ( await service . uninstallPlugins ( pluginSettings ) ) ;
Setting . setValue ( 'plugins.states' , newSettings ) ;
}
2020-11-13 17:09:28 +00:00
try {
2020-11-19 12:34:49 +00:00
if ( await shim . fsDriver ( ) . exists ( Setting . value ( 'pluginDir' ) ) ) {
await service . loadAndRunPlugins ( Setting . value ( 'pluginDir' ) , pluginSettings ) ;
}
2020-11-13 17:09:28 +00:00
} catch ( error ) {
this . logger ( ) . error ( ` There was an error loading plugins from ${ Setting . value ( 'pluginDir' ) } : ` , error ) ;
}
try {
if ( Setting . value ( 'plugins.devPluginPaths' ) ) {
const paths = Setting . value ( 'plugins.devPluginPaths' ) . split ( ',' ) . map ( ( p : string ) = > p . trim ( ) ) ;
2020-11-19 12:34:49 +00:00
await service . loadAndRunPlugins ( paths , pluginSettings , true ) ;
2020-11-13 17:09:28 +00:00
}
// Also load dev plugins that have passed via command line arguments
if ( Setting . value ( 'startupDevPlugins' ) ) {
2020-11-19 12:34:49 +00:00
await service . loadAndRunPlugins ( Setting . value ( 'startupDevPlugins' ) , pluginSettings , true ) ;
2020-11-13 17:09:28 +00:00
}
} catch ( error ) {
this . logger ( ) . error ( ` There was an error loading plugins from ${ Setting . value ( 'plugins.devPluginPaths' ) } : ` , error ) ;
}
2021-02-07 16:47:56 +00:00
2021-08-05 12:48:39 +01:00
{
// Users can potentially delete files from /plugins or even delete
// the complete folder. When that happens, we still have the plugin
// info in the state, which can cause various issues, so to sort it
// out we remove from the state any plugin that has *not* been loaded
// above (meaning the file was missing).
// https://github.com/laurent22/joplin/issues/5253
const oldSettings = service . unserializePluginSettings ( Setting . value ( 'plugins.states' ) ) ;
const newSettings : PluginSettings = { } ;
for ( const pluginId of Object . keys ( oldSettings ) ) {
if ( ! service . pluginIds . includes ( pluginId ) ) {
this . logger ( ) . warn ( 'Found a plugin in the state that has not been loaded, which means the plugin might have been deleted outside Joplin - removing it from the state:' , pluginId ) ;
continue ;
}
newSettings [ pluginId ] = oldSettings [ pluginId ] ;
}
Setting . setValue ( 'plugins.states' , newSettings ) ;
}
2021-02-07 16:47:56 +00:00
this . checkAllPluginStartedIID_ = setInterval ( ( ) = > {
if ( service . allPluginsStarted ) {
clearInterval ( this . checkAllPluginStartedIID_ ) ;
this . dispatch ( {
type : 'STARTUP_PLUGINS_LOADED' ,
value : true ,
} ) ;
}
} , 500 ) ;
2020-11-13 17:09:28 +00:00
}
2021-10-03 16:00:49 +01:00
public async start ( argv : string [ ] ) : Promise < any > {
2020-10-09 18:35:46 +01:00
// If running inside a package, the command line, instead of being "node.exe <path> <flags>" is "joplin.exe <flags>" so
// insert an extra argument so that they can be processed in a consistent way everywhere.
2021-10-01 19:35:27 +01:00
if ( ! bridge ( ) . electronIsDev ( ) ) argv . splice ( 1 , 0 , '.' ) ;
2020-10-09 18:35:46 +01:00
argv = await super . start ( argv ) ;
await this . applySettingsSideEffects ( ) ;
if ( Setting . value ( 'sync.upgradeState' ) === Setting . SYNC_UPGRADE_STATE_MUST_DO ) {
reg . logger ( ) . info ( 'app.start: doing upgradeSyncTarget action' ) ;
bridge ( ) . window ( ) . show ( ) ;
return { action : 'upgradeSyncTarget' } ;
}
reg . logger ( ) . info ( 'app.start: doing regular boot' ) ;
2021-10-15 12:24:22 +01:00
const dir : string = Setting . value ( 'profileDir' ) ;
syncDebugLog . enabled = false ;
if ( dir . endsWith ( 'dev-desktop-2' ) ) {
2021-10-17 14:54:48 +01:00
syncDebugLog . addTarget ( TargetType . File , {
path : ` ${ homedir ( ) } /synclog.txt ` ,
} ) ;
2021-10-15 12:24:22 +01:00
syncDebugLog . enabled = true ;
syncDebugLog . info ( ` Profile dir: ${ dir } ` ) ;
}
2020-10-09 18:35:46 +01:00
// Loads app-wide styles. (Markdown preview-specific styles loaded in app.js)
const filename = Setting . custom_css_files . JOPLIN_APP ;
2021-07-23 11:05:21 +01:00
await injectCustomStyles ( 'appStyles' , ` ${ dir } / ${ filename } ` ) ;
2020-10-09 18:35:46 +01:00
AlarmService . setDriver ( new AlarmServiceDriverNode ( { appName : packageInfo.build.appId } ) ) ;
AlarmService . setLogger ( reg . logger ( ) ) ;
2020-11-12 19:13:28 +00:00
reg . setShowErrorMessageBoxHandler ( ( message : string ) = > { bridge ( ) . showErrorMessageBox ( message ) ; } ) ;
2020-10-09 18:35:46 +01:00
if ( Setting . value ( 'flagOpenDevTools' ) ) {
bridge ( ) . openDevTools ( ) ;
}
PluginManager . instance ( ) . dispatch_ = this . dispatch . bind ( this ) ;
PluginManager . instance ( ) . setLogger ( reg . logger ( ) ) ;
PluginManager . instance ( ) . register ( pluginClasses ) ;
this . initRedux ( ) ;
2021-11-12 00:33:37 +09:00
PerFolderSortOrderService . initialize ( ) ;
2020-11-13 17:09:28 +00:00
CommandService . instance ( ) . initialize ( this . store ( ) , Setting . value ( 'env' ) == 'dev' , stateToWhenClauseContext ) ;
2020-10-09 18:35:46 +01:00
for ( const command of commands ) {
CommandService . instance ( ) . registerDeclaration ( command . declaration ) ;
}
for ( const command of globalCommands ) {
CommandService . instance ( ) . registerDeclaration ( command . declaration ) ;
CommandService . instance ( ) . registerRuntime ( command . declaration . name , command . runtime ( ) ) ;
}
for ( const declaration of editorCommandDeclarations ) {
CommandService . instance ( ) . registerDeclaration ( declaration ) ;
}
2020-10-31 12:25:12 +00:00
const keymapService = KeymapService . instance ( ) ;
2020-10-31 16:29:17 +00:00
// We only add the commands that appear in the menu because only
// those can have a shortcut associated with them.
keymapService . initialize ( menuCommandNames ( ) ) ;
2020-10-31 12:25:12 +00:00
try {
await keymapService . loadCustomKeymap ( ` ${ dir } /keymap-desktop.json ` ) ;
} catch ( error ) {
reg . logger ( ) . error ( error ) ;
}
2020-10-31 16:29:17 +00:00
// Since the settings need to be loaded before the store is
// created, it will never receive the SETTING_UPDATE_ALL even,
// which mean state.settings will not be initialised. So we
// manually call dispatchUpdateAll() to force an update.
2020-10-09 18:35:46 +01:00
Setting . dispatchUpdateAll ( ) ;
await FoldersScreenUtils . refreshFolders ( ) ;
const tags = await Tag . allWithNotes ( ) ;
this . dispatch ( {
type : 'TAG_UPDATE_ALL' ,
items : tags ,
} ) ;
2021-08-12 16:54:10 +01:00
// const masterKeys = await MasterKey.all();
2020-10-09 18:35:46 +01:00
2021-08-12 16:54:10 +01:00
// this.dispatch({
// type: 'MASTERKEY_UPDATE_ALL',
// items: masterKeys,
// });
2020-10-09 18:35:46 +01:00
this . store ( ) . dispatch ( {
type : 'FOLDER_SELECT' ,
id : Setting.value ( 'activeFolderId' ) ,
} ) ;
this . store ( ) . dispatch ( {
type : 'FOLDER_SET_COLLAPSED_ALL' ,
ids : Setting.value ( 'collapsedFolderIds' ) ,
} ) ;
// Loads custom Markdown preview styles
2021-07-23 11:05:21 +01:00
const cssString = await loadCustomCss ( ` ${ Setting . value ( 'profileDir' ) } /userstyle.css ` ) ;
2020-10-09 18:35:46 +01:00
this . store ( ) . dispatch ( {
2021-07-23 11:05:21 +01:00
type : 'CUSTOM_CSS_APPEND' ,
2020-10-09 18:35:46 +01:00
css : cssString ,
} ) ;
this . store ( ) . dispatch ( {
type : 'NOTE_DEVTOOLS_SET' ,
value : Setting.value ( 'flagOpenDevTools' ) ,
} ) ;
2021-08-06 10:58:32 +01:00
await this . checkForLegacyTemplates ( ) ;
2020-10-09 18:35:46 +01:00
// Note: Auto-update currently doesn't work in Linux: it downloads the update
// but then doesn't install it on exit.
if ( shim . isWindows ( ) || shim . isMac ( ) ) {
const runAutoUpdateCheck = ( ) = > {
if ( Setting . value ( 'autoUpdateEnabled' ) ) {
2021-05-14 11:29:06 +02:00
void checkForUpdates ( true , bridge ( ) . window ( ) , { includePreReleases : Setting.value ( 'autoUpdate.includePreReleases' ) } ) ;
2020-10-09 18:35:46 +01:00
}
} ;
// Initial check on startup
shim . setTimeout ( ( ) = > { runAutoUpdateCheck ( ) ; } , 5000 ) ;
// Then every x hours
shim . setInterval ( ( ) = > { runAutoUpdateCheck ( ) ; } , 12 * 60 * 60 * 1000 ) ;
}
this . updateTray ( ) ;
shim . setTimeout ( ( ) = > {
2020-11-25 14:40:25 +00:00
void AlarmService . garbageCollect ( ) ;
2020-10-09 18:35:46 +01:00
} , 1000 * 60 * 60 ) ;
if ( Setting . value ( 'startMinimized' ) && Setting . value ( 'showTrayIcon' ) ) {
// Keep it hidden
} else {
bridge ( ) . window ( ) . show ( ) ;
}
2021-05-13 18:57:37 +02:00
void ShareService . instance ( ) . maintenance ( ) ;
2020-10-09 18:35:46 +01:00
ResourceService . runInBackground ( ) ;
if ( Setting . value ( 'env' ) === 'dev' ) {
2020-11-25 14:40:25 +00:00
void AlarmService . updateAllNotifications ( ) ;
2020-10-09 18:35:46 +01:00
} else {
2021-01-29 18:45:11 +00:00
void reg . scheduleSync ( 1000 ) . then ( ( ) = > {
2020-10-09 18:35:46 +01:00
// Wait for the first sync before updating the notifications, since synchronisation
// might change the notifications.
2020-11-25 14:40:25 +00:00
void AlarmService . updateAllNotifications ( ) ;
2020-10-09 18:35:46 +01:00
2021-01-22 17:41:11 +00:00
void DecryptionWorker . instance ( ) . scheduleStart ( ) ;
2020-10-09 18:35:46 +01:00
} ) ;
}
const clipperLogger = new Logger ( ) ;
clipperLogger . addTarget ( TargetType . File , { path : ` ${ Setting . value ( 'profileDir' ) } /log-clipper.txt ` } ) ;
clipperLogger . addTarget ( TargetType . Console ) ;
ClipperServer . instance ( ) . initialize ( actionApi ) ;
ClipperServer . instance ( ) . setLogger ( clipperLogger ) ;
ClipperServer . instance ( ) . setDispatch ( this . store ( ) . dispatch ) ;
if ( Setting . value ( 'clipperServer.autoStart' ) ) {
2021-06-22 19:57:04 +01:00
void ClipperServer . instance ( ) . start ( ) ;
2020-10-09 18:35:46 +01:00
}
ExternalEditWatcher . instance ( ) . setLogger ( reg . logger ( ) ) ;
2020-11-16 11:03:44 +00:00
ExternalEditWatcher . instance ( ) . initialize ( bridge , this . store ( ) . dispatch ) ;
2020-10-09 18:35:46 +01:00
2021-10-01 19:35:27 +01:00
ResourceEditWatcher . instance ( ) . initialize ( reg . logger ( ) , ( action : any ) = > { this . store ( ) . dispatch ( action ) ; } , ( path : string ) = > bridge ( ) . openItem ( path ) ) ;
2020-10-09 18:35:46 +01:00
RevisionService . instance ( ) . runInBackground ( ) ;
// Make it available to the console window - useful to call revisionService.collectRevisions()
2021-05-13 18:57:37 +02:00
if ( Setting . value ( 'env' ) === 'dev' ) {
( window as any ) . joplin = {
2020-10-09 18:35:46 +01:00
revisionService : RevisionService.instance ( ) ,
migrationService : MigrationService.instance ( ) ,
decryptionWorker : DecryptionWorker.instance ( ) ,
2021-01-08 18:06:33 +00:00
commandService : CommandService.instance ( ) ,
2020-10-09 18:35:46 +01:00
bridge : bridge ( ) ,
2021-05-13 18:57:37 +02:00
debug : new DebugService ( reg . db ( ) ) ,
2020-10-09 18:35:46 +01:00
} ;
2021-05-13 18:57:37 +02:00
}
2020-10-09 18:35:46 +01:00
bridge ( ) . addEventListener ( 'nativeThemeUpdated' , this . bridge_nativeThemeUpdated ) ;
2020-11-13 17:09:28 +00:00
await this . initPluginService ( ) ;
2020-10-09 18:35:46 +01:00
2020-11-05 16:58:23 +00:00
this . setupContextMenu ( ) ;
await SpellCheckerService . instance ( ) . initialize ( new SpellCheckerServiceDriverNative ( ) ) ;
2020-10-09 18:35:46 +01:00
// await populateDatabase(reg.db());
// setTimeout(() => {
// console.info(CommandService.instance().commandsToMarkdownTable(this.store().getState()));
// }, 2000);
2021-01-22 00:55:58 +00:00
// setTimeout(() => {
// this.dispatch({
// type: 'NAV_GO',
// routeName: 'Config',
// props: {
2021-08-17 12:03:19 +01:00
// defaultSection: 'encryption',
2021-01-22 00:55:58 +00:00
// },
// });
2021-08-17 12:03:19 +01:00
// }, 2000);
2020-11-19 12:34:49 +00:00
2021-08-16 15:20:14 +01:00
// setTimeout(() => {
// this.dispatch({
// type: 'DIALOG_OPEN',
2021-11-15 17:19:51 +00:00
// name: 'editFolder',
// props: { folderId: '3d90f7da26b947dc9c8c6c65e86cd231' },
2021-08-16 15:20:14 +01:00
// });
// }, 2000);
2021-09-01 12:17:20 +01:00
// setTimeout(() => {
// this.dispatch({
// type: 'NAV_GO',
// routeName: 'Config',
// props: {
// defaultSection: 'plugins',
// },
// });
// }, 2000);
2021-10-03 16:00:49 +01:00
// const testData = {
// "publicKey": "-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAmKpb4JiYiY16pGOabje7uMsFd7DcMnruGxJ9HSpOiOduj3ApKqRu0xWCkGyqpekyOjjooZ98wVkDPUFsyVjN+kG8yKFn2xXC5SeRyhIVbdytjYiGshr6x+T9XVI+HnJKQF3WbrcqSOejlDXJv6u7jKrLAlOT3tkqEb0ZefhcEIajq6kNkH51R0lwsFnzxDIK3MW1wNzmiOfM92f8PFxiOBmUtVIngGPlNgyld1FzKN7Ypz1uS6GOqAtRm325qyfE/+2Jgb7WaDFT7VB5pHnOiojj9+xi1DvQWCbbIYXoMi0XVi9i2ZQfM32aFwiHez5UL61IMWUcqQ0/gldh4HFlAQIDAQAB\n-----END PUBLIC KEY-----",
// "privateKey": "-----BEGIN RSA PRIVATE KEY-----\nMIIEpAIBAAKCAQEAmKpb4JiYiY16pGOabje7uMsFd7DcMnruGxJ9HSpOiOduj3ApKqRu0xWCkGyqpekyOjjooZ98wVkDPUFsyVjN+kG8yKFn2xXC5SeRyhIVbdytjYiGshr6x+T9XVI+HnJKQF3WbrcqSOejlDXJv6u7jKrLAlOT3tkqEb0ZefhcEIajq6kNkH51R0lwsFnzxDIK3MW1wNzmiOfM92f8PFxiOBmUtVIngGPlNgyld1FzKN7Ypz1uS6GOqAtRm325qyfE/+2Jgb7WaDFT7VB5pHnOiojj9+xi1DvQWCbbIYXoMi0XVi9i2ZQfM32aFwiHez5UL61IMWUcqQ0/gldh4HFlAQIDAQABAoIBADFFMffPZ9Nk7MLnPmz54cTnCPGzC63jDLuCAQ0LnWMDxiPW4AJaJUZMt+GioISBOWue+D1JOrsv3iLD3bcxyPBOjP33UYxcfpT0a1Ha+j2FriFygX4zxOIEnlyi8VdkLWCOqGj9BlGXKKzpmx4X76Sbbn9mt9+BGNm2vOUnaZcPTVuOI7K6xZynlzMRYSyhu7J0QdYVK44vZ/TjdD/4pgX+ezrGiwx7OCf/KctjvEoYtXYV2gkBOifOlqYOp0fMEC3mVAZfwpvDTbRchb7h0rxmxfKbWsjPtDblByXBLJZ3PGcKcmJlu4Qsfd2AgrY62r+DbNt3EhK072ZilYIfKD0CgYEAybcDbucr67dWMlFh5b79bvJugw6rj1V59Tp+RX9nKgzaiBUHLun6cK5hbgg9z3ejc2SWlX7D+eOyveVjhDlxUOCFURJLo2oPMRKwBBKJkOJhdtAjPzyceYI6Yj2lvtDeijcZfg8F9YqUTMfisDsEi1MbGnqawWwUerN9P5TjRBcCgYEAwcAfw8KTnQsvXPwWwh6Wabtz0bUAKzA/D6oWTR5IbkBfb3jNU8lmh9H66H0P18Nsa3vozA6buW2LDhHCFFkQ4PUTQVKok1qhAsvJBECxdwMqb5iAXk3Yk3qQYGhR23Zkp1u82wmpSaBLKGr+SL9/q5EamqiR3PQYx/aQTeIaFqcCgYAn/N/xXGKYl/++eeOuZ+5V0DmYQZBBGfDTbIUbweXxsBqiX4jNBBVhwTAPYBLgzhbZCVfQyxCOuVT10EOqMrkED35eVAIqoxvf3pSGOiaLUlV/+EMEhj9+1xI753y0FzQGsmWbV98WjiJYFkgaJ5j/BbqZxTRoo8RrjqmFsT5cgQKBgQCWTc4WlmbfSKMIloOtOf9jrMjvoWOtHXN+WmuMjfaQmR2wI13eJvqEWRA1tXdJ4c/FHk39p0OFOQbL9ljCYknmyhiS72XZUlBgE+kwhGNnuSv9gKftAKUH2+gO8j62awUwk8lRfxA2DsTfaQk1NGH9ncauviDR8QcccRmHYeTtNwKBgQCOvHiVaNw8XJIqt2r3j8pEJcr8LO+WNtLDU+h9NhM5a5NxfeRUlxdrqR0FXS4NkE6E3h9iLIRt2V+0bghzJMhKuwdjC0K6+jCb7ImV+Xcl9LNOQ1mPLBLS1jqdQnBS1ZPtcQpMrVi6dU9vVespylKEyGnQnUUtLgYrbO9OMrP1uQ==\n-----END RSA PRIVATE KEY-----",
// "plaintext": "just testing",
// "ciphertext": "LBicxglLvMyBin8uMpUnF5ARQ+KtAM563RViMepnOcyXa/NOJonNBixm+th+jX44\r\n/rie2ESbWg/FnlR4mHCEpTQJFXt12zpeXvtM8Hy1OQMud1B1Hc9hp1hhd1t6cuDz\r\n/Cs10n1+57V6zwHottYA6tn84cBn678SvPa/WTwgvb9lnBVZbesm3dVIr5uh2hk9\r\nNcVkmqyfi+ilkNQ3FIQfL+ciHvPFUIpljgIOipZhmufubdgMGW1HEUYlsmxLE7ce\r\ndpUQJoIbfKJ1x2dJRoeYsCjvcYFWdMUcg78HkXR+UcObP6zkK8cH33fb6PKKd8Z4\r\nToj4HROza8Dp7uCV5XyBTA=="
// };
// await checkTestData(testData);
// const testData = await createTestData();
// await checkTestData(testData);
// await printTestData();
// await runIntegrationTests();
2020-10-09 18:35:46 +01:00
return null ;
}
}
2020-11-12 19:13:28 +00:00
let application_ : Application = null ;
2020-10-09 18:35:46 +01:00
function app() {
if ( ! application_ ) application_ = new Application ( ) ;
return application_ ;
}
export default app ;