mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-17 18:44:45 +02:00
e11e57f1d8
The implementation uses / symbol as a nesting separator. I.e. tag/subtag is a nested tag, where tag is the parent tag and subtag is its child. Creating a tag named tag/subtag/subsubtag creates three tags, one for each level. The tags are associated using parent_id field. In the app, viewing notes with a tag will also show all notes that are associated with any of the tag's descendant tags (same for the note count). Deleting a tag will also delete all its descendant tags. In the desktop app the tags are shown nested just like the notebooks.
789 lines
25 KiB
JavaScript
789 lines
25 KiB
JavaScript
import setUpQuickActions from './setUpQuickActions';
|
|
import PluginAssetsLoader from './PluginAssetsLoader';
|
|
|
|
const React = require('react');
|
|
const { AppState, Keyboard, NativeModules, BackHandler, Animated, View, StatusBar } = require('react-native');
|
|
const SafeAreaView = require('lib/components/SafeAreaView');
|
|
const { connect, Provider } = require('react-redux');
|
|
const { BackButtonService } = require('lib/services/back-button.js');
|
|
const NavService = require('lib/services/NavService.js');
|
|
const AlarmService = require('lib/services/AlarmService.js');
|
|
const AlarmServiceDriver = require('lib/services/AlarmServiceDriver');
|
|
const Alarm = require('lib/models/Alarm');
|
|
const { createStore, applyMiddleware } = require('redux');
|
|
const reduxSharedMiddleware = require('lib/components/shared/reduxSharedMiddleware');
|
|
const { shimInit } = require('lib/shim-init-react.js');
|
|
const { time } = require('lib/time-utils.js');
|
|
const { AppNav } = require('lib/components/app-nav.js');
|
|
const { Logger } = require('lib/logger.js');
|
|
const Note = require('lib/models/Note.js');
|
|
const Folder = require('lib/models/Folder.js');
|
|
const BaseSyncTarget = require('lib/BaseSyncTarget.js');
|
|
const { FoldersScreenUtils } = require('lib/folders-screen-utils.js');
|
|
const Resource = require('lib/models/Resource.js');
|
|
const Tag = require('lib/models/Tag.js');
|
|
const NoteTag = require('lib/models/NoteTag.js');
|
|
const BaseItem = require('lib/models/BaseItem.js');
|
|
const MasterKey = require('lib/models/MasterKey.js');
|
|
const Revision = require('lib/models/Revision.js');
|
|
const BaseModel = require('lib/BaseModel.js');
|
|
const BaseService = require('lib/services/BaseService.js');
|
|
const ResourceService = require('lib/services/ResourceService');
|
|
const RevisionService = require('lib/services/RevisionService');
|
|
const KvStore = require('lib/services/KvStore');
|
|
const { JoplinDatabase } = require('lib/joplin-database.js');
|
|
const { Database } = require('lib/database.js');
|
|
const { NotesScreen } = require('lib/components/screens/notes.js');
|
|
const { TagsScreen } = require('lib/components/screens/tags.js');
|
|
const { NoteScreen } = require('lib/components/screens/note.js');
|
|
const { ConfigScreen } = require('lib/components/screens/config.js');
|
|
const { FolderScreen } = require('lib/components/screens/folder.js');
|
|
const { LogScreen } = require('lib/components/screens/log.js');
|
|
const { StatusScreen } = require('lib/components/screens/status.js');
|
|
const { SearchScreen } = require('lib/components/screens/search.js');
|
|
const { OneDriveLoginScreen } = require('lib/components/screens/onedrive-login.js');
|
|
const { EncryptionConfigScreen } = require('lib/components/screens/encryption-config.js');
|
|
const { DropboxLoginScreen } = require('lib/components/screens/dropbox-login.js');
|
|
const Setting = require('lib/models/Setting.js');
|
|
const { MenuContext } = require('react-native-popup-menu');
|
|
const { SideMenu } = require('lib/components/side-menu.js');
|
|
const { SideMenuContent } = require('lib/components/side-menu-content.js');
|
|
const { SideMenuContentNote } = require('lib/components/side-menu-content-note.js');
|
|
const { DatabaseDriverReactNative } = require('lib/database-driver-react-native');
|
|
const { reg } = require('lib/registry.js');
|
|
const { setLocale, closestSupportedLocale, defaultLocale } = require('lib/locale.js');
|
|
const RNFetchBlob = require('rn-fetch-blob').default;
|
|
const { PoorManIntervals } = require('lib/poor-man-intervals.js');
|
|
const { reducer, defaultState } = require('lib/reducer.js');
|
|
const { FileApiDriverLocal } = require('lib/file-api-driver-local.js');
|
|
const DropdownAlert = require('react-native-dropdownalert').default;
|
|
const ShareExtension = require('lib/ShareExtension.js').default;
|
|
const handleShared = require('lib/shareHandler').default;
|
|
const ResourceFetcher = require('lib/services/ResourceFetcher');
|
|
const SearchEngine = require('lib/services/SearchEngine');
|
|
const WelcomeUtils = require('lib/WelcomeUtils');
|
|
const { themeStyle } = require('lib/components/global-style.js');
|
|
const { uuid } = require('lib/uuid.js');
|
|
|
|
const { loadKeychainServiceAndSettings } = require('lib/services/SettingUtils');
|
|
const KeychainServiceDriverMobile = require('lib/services/keychain/KeychainServiceDriver.mobile').default;
|
|
|
|
const SyncTargetRegistry = require('lib/SyncTargetRegistry.js');
|
|
const SyncTargetOneDrive = require('lib/SyncTargetOneDrive.js');
|
|
const SyncTargetFilesystem = require('lib/SyncTargetFilesystem.js');
|
|
const SyncTargetOneDriveDev = require('lib/SyncTargetOneDriveDev.js');
|
|
const SyncTargetNextcloud = require('lib/SyncTargetNextcloud.js');
|
|
const SyncTargetWebDAV = require('lib/SyncTargetWebDAV.js');
|
|
const SyncTargetDropbox = require('lib/SyncTargetDropbox.js');
|
|
|
|
SyncTargetRegistry.addClass(SyncTargetOneDrive);
|
|
if (__DEV__) SyncTargetRegistry.addClass(SyncTargetOneDriveDev);
|
|
SyncTargetRegistry.addClass(SyncTargetNextcloud);
|
|
SyncTargetRegistry.addClass(SyncTargetWebDAV);
|
|
SyncTargetRegistry.addClass(SyncTargetDropbox);
|
|
SyncTargetRegistry.addClass(SyncTargetFilesystem);
|
|
|
|
const FsDriverRN = require('lib/fs-driver-rn.js').FsDriverRN;
|
|
const DecryptionWorker = require('lib/services/DecryptionWorker');
|
|
const EncryptionService = require('lib/services/EncryptionService');
|
|
const MigrationService = require('lib/services/MigrationService');
|
|
|
|
let storeDispatch = function() {};
|
|
|
|
const logReducerAction = function(action) {
|
|
if (['SIDE_MENU_OPEN_PERCENT', 'SYNC_REPORT_UPDATE'].indexOf(action.type) >= 0) return;
|
|
|
|
const msg = [action.type];
|
|
if (action.routeName) msg.push(action.routeName);
|
|
|
|
// reg.logger().debug('Reducer action', msg.join(', '));
|
|
};
|
|
|
|
const generalMiddleware = store => next => async (action) => {
|
|
logReducerAction(action);
|
|
PoorManIntervals.update(); // This function needs to be called regularly so put it here
|
|
|
|
const result = next(action);
|
|
const newState = store.getState();
|
|
|
|
await reduxSharedMiddleware(store, next, action);
|
|
|
|
if (action.type == 'NAV_GO') Keyboard.dismiss();
|
|
|
|
if (['NOTE_UPDATE_ONE', 'NOTE_DELETE', 'FOLDER_UPDATE_ONE', 'FOLDER_DELETE'].indexOf(action.type) >= 0) {
|
|
if (!await reg.syncTarget().syncStarted()) reg.scheduleSync(5 * 1000, { syncSteps: ['update_remote', 'delete_remote'] });
|
|
SearchEngine.instance().scheduleSyncTables();
|
|
}
|
|
|
|
if (['EVENT_NOTE_ALARM_FIELD_CHANGE', 'NOTE_DELETE'].indexOf(action.type) >= 0) {
|
|
await AlarmService.updateNoteNotification(action.id, action.type === 'NOTE_DELETE');
|
|
}
|
|
|
|
if (action.type == 'SETTING_UPDATE_ONE' && action.key == 'sync.interval' || action.type == 'SETTING_UPDATE_ALL') {
|
|
reg.setupRecurrentSync();
|
|
}
|
|
|
|
if ((action.type == 'SETTING_UPDATE_ONE' && (action.key == 'dateFormat' || action.key == 'timeFormat')) || (action.type == 'SETTING_UPDATE_ALL')) {
|
|
time.setDateFormat(Setting.value('dateFormat'));
|
|
time.setTimeFormat(Setting.value('timeFormat'));
|
|
}
|
|
|
|
if (action.type == 'SETTING_UPDATE_ONE' && action.key == 'locale' || action.type == 'SETTING_UPDATE_ALL') {
|
|
setLocale(Setting.value('locale'));
|
|
}
|
|
|
|
if ((action.type == 'SETTING_UPDATE_ONE' && (action.key.indexOf('encryption.') === 0)) || (action.type == 'SETTING_UPDATE_ALL')) {
|
|
await EncryptionService.instance().loadMasterKeysFromSettings();
|
|
DecryptionWorker.instance().scheduleStart();
|
|
const loadedMasterKeyIds = EncryptionService.instance().loadedMasterKeyIds();
|
|
|
|
storeDispatch({
|
|
type: 'MASTERKEY_REMOVE_NOT_LOADED',
|
|
ids: loadedMasterKeyIds,
|
|
});
|
|
|
|
// Schedule a sync operation so that items that need to be encrypted
|
|
// are sent to sync target.
|
|
reg.scheduleSync();
|
|
}
|
|
|
|
if (action.type == 'NAV_GO' && action.routeName == 'Notes') {
|
|
Setting.setValue('activeFolderId', newState.selectedFolderId);
|
|
}
|
|
|
|
if (action.type === 'SYNC_GOT_ENCRYPTED_ITEM') {
|
|
DecryptionWorker.instance().scheduleStart();
|
|
}
|
|
|
|
if (action.type === 'SYNC_CREATED_OR_UPDATED_RESOURCE') {
|
|
ResourceFetcher.instance().autoAddResources();
|
|
}
|
|
|
|
return result;
|
|
};
|
|
|
|
const navHistory = [];
|
|
|
|
function historyCanGoBackTo(route) {
|
|
if (route.routeName === 'Note') return false;
|
|
if (route.routeName === 'Folder') return false;
|
|
|
|
// There's no point going back to these screens in general and, at least in OneDrive case,
|
|
// it can be buggy to do so, due to incorrectly relying on global state (reg.syncTarget...)
|
|
if (route.routeName === 'OneDriveLogin') return false;
|
|
if (route.routeName === 'DropboxLogin') return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
const DEFAULT_ROUTE = {
|
|
type: 'NAV_GO',
|
|
routeName: 'Notes',
|
|
smartFilterId: 'c3176726992c11e9ac940492261af972',
|
|
};
|
|
|
|
const appDefaultState = Object.assign({}, defaultState, {
|
|
sideMenuOpenPercent: 0,
|
|
route: DEFAULT_ROUTE,
|
|
noteSelectionEnabled: false,
|
|
noteSideMenuOptions: null,
|
|
});
|
|
|
|
const appReducer = (state = appDefaultState, action) => {
|
|
let newState = state;
|
|
let historyGoingBack = false;
|
|
|
|
try {
|
|
switch (action.type) {
|
|
|
|
case 'NAV_BACK':
|
|
|
|
{
|
|
if (!navHistory.length) break;
|
|
|
|
let newAction = null;
|
|
while (navHistory.length) {
|
|
newAction = navHistory.pop();
|
|
if (newAction.routeName != state.route.routeName) break;
|
|
}
|
|
|
|
action = newAction ? newAction : navHistory.pop();
|
|
|
|
historyGoingBack = true;
|
|
}
|
|
|
|
// Fall throught
|
|
|
|
case 'NAV_GO':
|
|
|
|
{
|
|
const currentRoute = state.route;
|
|
|
|
if (!historyGoingBack && historyCanGoBackTo(currentRoute, action)) {
|
|
// If the route *name* is the same (even if the other parameters are different), we
|
|
// overwrite the last route in the history with the current one. If the route name
|
|
// is different, we push a new history entry.
|
|
if (currentRoute.routeName == action.routeName) {
|
|
// nothing
|
|
} else {
|
|
navHistory.push(currentRoute);
|
|
}
|
|
}
|
|
|
|
// HACK: whenever a new screen is loaded, all the previous screens of that type
|
|
// are overwritten with the new screen parameters. This is because the way notes
|
|
// are currently loaded is not optimal (doesn't retain history properly) so
|
|
// this is a simple fix without doing a big refactoring to change the way notes
|
|
// are loaded. Might be good enough since going back to different folders
|
|
// is probably not a common workflow.
|
|
for (let i = 0; i < navHistory.length; i++) {
|
|
const n = navHistory[i];
|
|
if (n.routeName == action.routeName) {
|
|
navHistory[i] = Object.assign({}, action);
|
|
}
|
|
}
|
|
|
|
newState = Object.assign({}, state);
|
|
|
|
newState.selectedNoteHash = '';
|
|
|
|
if ('noteId' in action) {
|
|
newState.selectedNoteIds = action.noteId ? [action.noteId] : [];
|
|
}
|
|
|
|
if ('folderId' in action) {
|
|
newState.selectedFolderId = action.folderId;
|
|
newState.notesParentType = 'Folder';
|
|
}
|
|
|
|
if ('tagId' in action) {
|
|
newState.selectedTagId = action.tagId;
|
|
newState.notesParentType = 'Tag';
|
|
}
|
|
|
|
if ('smartFilterId' in action) {
|
|
newState.smartFilterId = action.smartFilterId;
|
|
newState.notesParentType = 'SmartFilter';
|
|
}
|
|
|
|
if ('itemType' in action) {
|
|
newState.selectedItemType = action.itemType;
|
|
}
|
|
|
|
if ('noteHash' in action) {
|
|
newState.selectedNoteHash = action.noteHash;
|
|
}
|
|
|
|
if ('sharedData' in action) {
|
|
newState.sharedData = action.sharedData;
|
|
} else {
|
|
newState.sharedData = null;
|
|
}
|
|
|
|
newState.route = action;
|
|
newState.historyCanGoBack = !!navHistory.length;
|
|
}
|
|
break;
|
|
|
|
case 'SIDE_MENU_TOGGLE':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.showSideMenu = !newState.showSideMenu;
|
|
break;
|
|
|
|
case 'SIDE_MENU_OPEN':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.showSideMenu = true;
|
|
break;
|
|
|
|
case 'SIDE_MENU_CLOSE':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.showSideMenu = false;
|
|
break;
|
|
|
|
case 'SIDE_MENU_OPEN_PERCENT':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.sideMenuOpenPercent = action.value;
|
|
break;
|
|
|
|
case 'NOTE_SELECTION_TOGGLE':
|
|
|
|
{
|
|
newState = Object.assign({}, state);
|
|
|
|
const noteId = action.id;
|
|
const newSelectedNoteIds = state.selectedNoteIds.slice();
|
|
const existingIndex = state.selectedNoteIds.indexOf(noteId);
|
|
|
|
if (existingIndex >= 0) {
|
|
newSelectedNoteIds.splice(existingIndex, 1);
|
|
} else {
|
|
newSelectedNoteIds.push(noteId);
|
|
}
|
|
|
|
newState.selectedNoteIds = newSelectedNoteIds;
|
|
newState.noteSelectionEnabled = !!newSelectedNoteIds.length;
|
|
}
|
|
break;
|
|
|
|
case 'NOTE_SELECTION_START':
|
|
|
|
if (!state.noteSelectionEnabled) {
|
|
newState = Object.assign({}, state);
|
|
newState.noteSelectionEnabled = true;
|
|
newState.selectedNoteIds = [action.id];
|
|
}
|
|
break;
|
|
|
|
case 'NOTE_SELECTION_END':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.noteSelectionEnabled = false;
|
|
newState.selectedNoteIds = [];
|
|
break;
|
|
|
|
case 'NOTE_SIDE_MENU_OPTIONS_SET':
|
|
|
|
newState = Object.assign({}, state);
|
|
newState.noteSideMenuOptions = action.options;
|
|
break;
|
|
|
|
}
|
|
} catch (error) {
|
|
error.message = `In reducer: ${error.message} Action: ${JSON.stringify(action)}`;
|
|
throw error;
|
|
}
|
|
|
|
return reducer(newState, action);
|
|
};
|
|
|
|
const store = createStore(appReducer, applyMiddleware(generalMiddleware));
|
|
storeDispatch = store.dispatch;
|
|
|
|
function resourceFetcher_downloadComplete(event) {
|
|
if (event.encrypted) {
|
|
DecryptionWorker.instance().scheduleStart();
|
|
}
|
|
}
|
|
|
|
function decryptionWorker_resourceMetadataButNotBlobDecrypted() {
|
|
ResourceFetcher.instance().scheduleAutoAddResources();
|
|
}
|
|
|
|
async function initialize(dispatch) {
|
|
shimInit();
|
|
|
|
Setting.setConstant('env', __DEV__ ? 'dev' : 'prod');
|
|
Setting.setConstant('appId', 'net.cozic.joplin-mobile');
|
|
Setting.setConstant('appType', 'mobile');
|
|
Setting.setConstant('resourceDir', RNFetchBlob.fs.dirs.DocumentDir);
|
|
|
|
const logDatabase = new Database(new DatabaseDriverReactNative());
|
|
await logDatabase.open({ name: 'log.sqlite' });
|
|
await logDatabase.exec(Logger.databaseCreateTableSql());
|
|
|
|
const mainLogger = new Logger();
|
|
mainLogger.addTarget('database', { database: logDatabase, source: 'm' });
|
|
mainLogger.setLevel(Logger.LEVEL_INFO);
|
|
|
|
if (Setting.value('env') == 'dev') {
|
|
mainLogger.addTarget('console');
|
|
mainLogger.setLevel(Logger.LEVEL_DEBUG);
|
|
}
|
|
|
|
reg.setLogger(mainLogger);
|
|
reg.setShowErrorMessageBoxHandler((message) => { alert(message); });
|
|
|
|
BaseService.logger_ = mainLogger;
|
|
|
|
reg.logger().info('====================================');
|
|
reg.logger().info(`Starting application ${Setting.value('appId')} (${Setting.value('env')})`);
|
|
|
|
const dbLogger = new Logger();
|
|
dbLogger.addTarget('database', { database: logDatabase, source: 'm' });
|
|
if (Setting.value('env') == 'dev') {
|
|
dbLogger.addTarget('console');
|
|
dbLogger.setLevel(Logger.LEVEL_INFO); // Set to LEVEL_DEBUG for full SQL queries
|
|
} else {
|
|
dbLogger.setLevel(Logger.LEVEL_INFO);
|
|
}
|
|
|
|
const db = new JoplinDatabase(new DatabaseDriverReactNative());
|
|
db.setLogger(dbLogger);
|
|
reg.setDb(db);
|
|
|
|
reg.dispatch = dispatch;
|
|
BaseModel.dispatch = dispatch;
|
|
FoldersScreenUtils.dispatch = dispatch;
|
|
BaseSyncTarget.dispatch = dispatch;
|
|
NavService.dispatch = dispatch;
|
|
BaseModel.setDb(db);
|
|
|
|
KvStore.instance().setDb(reg.db());
|
|
|
|
BaseItem.loadClass('Note', Note);
|
|
BaseItem.loadClass('Folder', Folder);
|
|
BaseItem.loadClass('Resource', Resource);
|
|
BaseItem.loadClass('Tag', Tag);
|
|
BaseItem.loadClass('NoteTag', NoteTag);
|
|
BaseItem.loadClass('MasterKey', MasterKey);
|
|
BaseItem.loadClass('Revision', Revision);
|
|
|
|
const fsDriver = new FsDriverRN();
|
|
|
|
Resource.fsDriver_ = fsDriver;
|
|
FileApiDriverLocal.fsDriver_ = fsDriver;
|
|
|
|
AlarmService.setDriver(new AlarmServiceDriver());
|
|
AlarmService.setLogger(mainLogger);
|
|
|
|
try {
|
|
if (Setting.value('env') == 'prod') {
|
|
await db.open({ name: 'joplin.sqlite' });
|
|
} else {
|
|
await db.open({ name: 'joplin-71.sqlite' });
|
|
|
|
// await db.clearForTesting();
|
|
}
|
|
|
|
reg.logger().info('Database is ready.');
|
|
reg.logger().info('Loading settings...');
|
|
|
|
await loadKeychainServiceAndSettings(KeychainServiceDriverMobile);
|
|
|
|
if (!Setting.value('clientId')) Setting.setValue('clientId', uuid.create());
|
|
|
|
if (Setting.value('firstStart')) {
|
|
let locale = NativeModules.I18nManager.localeIdentifier;
|
|
if (!locale) locale = defaultLocale();
|
|
Setting.setValue('locale', closestSupportedLocale(locale));
|
|
if (Setting.value('env') === 'dev') Setting.setValue('sync.target', SyncTargetRegistry.nameToId('onedrive_dev'));
|
|
Setting.setValue('firstStart', 0);
|
|
}
|
|
|
|
if (Setting.value('db.ftsEnabled') === -1) {
|
|
const ftsEnabled = await db.ftsEnabled();
|
|
Setting.setValue('db.ftsEnabled', ftsEnabled ? 1 : 0);
|
|
reg.logger().info('db.ftsEnabled = ', Setting.value('db.ftsEnabled'));
|
|
}
|
|
|
|
if (Setting.value('env') === 'dev') {
|
|
Setting.setValue('welcome.enabled', false);
|
|
}
|
|
|
|
PluginAssetsLoader.instance().setLogger(mainLogger);
|
|
await PluginAssetsLoader.instance().importAssets();
|
|
|
|
// eslint-disable-next-line require-atomic-updates
|
|
BaseItem.revisionService_ = RevisionService.instance();
|
|
|
|
// Note: for now we hard-code the folder sort order as we need to
|
|
// create a UI to allow customisation (started in branch mobile_add_sidebar_buttons)
|
|
Setting.setValue('folders.sortOrder.field', 'title');
|
|
Setting.setValue('folders.sortOrder.reverse', false);
|
|
|
|
reg.logger().info(`Sync target: ${Setting.value('sync.target')}`);
|
|
|
|
setLocale(Setting.value('locale'));
|
|
|
|
// ----------------------------------------------------------------
|
|
// E2EE SETUP
|
|
// ----------------------------------------------------------------
|
|
|
|
EncryptionService.fsDriver_ = fsDriver;
|
|
EncryptionService.instance().setLogger(mainLogger);
|
|
// eslint-disable-next-line require-atomic-updates
|
|
BaseItem.encryptionService_ = EncryptionService.instance();
|
|
DecryptionWorker.instance().dispatch = dispatch;
|
|
DecryptionWorker.instance().setLogger(mainLogger);
|
|
DecryptionWorker.instance().setKvStore(KvStore.instance());
|
|
DecryptionWorker.instance().setEncryptionService(EncryptionService.instance());
|
|
await EncryptionService.instance().loadMasterKeysFromSettings();
|
|
DecryptionWorker.instance().on('resourceMetadataButNotBlobDecrypted', decryptionWorker_resourceMetadataButNotBlobDecrypted);
|
|
|
|
// ----------------------------------------------------------------
|
|
// / E2EE SETUP
|
|
// ----------------------------------------------------------------
|
|
|
|
reg.logger().info('Loading folders...');
|
|
|
|
await FoldersScreenUtils.refreshFolders();
|
|
|
|
const tags = await Tag.allWithNotes();
|
|
|
|
dispatch({
|
|
type: 'TAG_UPDATE_ALL',
|
|
items: tags,
|
|
});
|
|
|
|
const masterKeys = await MasterKey.all();
|
|
|
|
dispatch({
|
|
type: 'MASTERKEY_UPDATE_ALL',
|
|
items: masterKeys,
|
|
});
|
|
|
|
const folderId = Setting.value('activeFolderId');
|
|
let folder = await Folder.load(folderId);
|
|
|
|
if (!folder) folder = await Folder.defaultFolder();
|
|
|
|
dispatch({
|
|
type: 'FOLDER_SET_COLLAPSED_ALL',
|
|
ids: Setting.value('collapsedFolderIds'),
|
|
});
|
|
|
|
dispatch({
|
|
type: 'TAG_SET_COLLAPSED_ALL',
|
|
ids: Setting.value('collapsedTagIds'),
|
|
});
|
|
|
|
if (!folder) {
|
|
dispatch(DEFAULT_ROUTE);
|
|
} else {
|
|
dispatch({
|
|
type: 'NAV_GO',
|
|
routeName: 'Notes',
|
|
folderId: folder.id,
|
|
});
|
|
}
|
|
|
|
setUpQuickActions(dispatch, folderId);
|
|
} catch (error) {
|
|
alert(`Initialization error: ${error.message}`);
|
|
reg.logger().error('Initialization error:', error);
|
|
}
|
|
|
|
reg.setupRecurrentSync();
|
|
|
|
PoorManIntervals.setTimeout(() => {
|
|
AlarmService.garbageCollect();
|
|
}, 1000 * 60 * 60);
|
|
|
|
ResourceService.runInBackground();
|
|
|
|
ResourceFetcher.instance().setFileApi(() => { return reg.syncTarget().fileApi(); });
|
|
ResourceFetcher.instance().setLogger(reg.logger());
|
|
ResourceFetcher.instance().dispatch = dispatch;
|
|
ResourceFetcher.instance().on('downloadComplete', resourceFetcher_downloadComplete);
|
|
ResourceFetcher.instance().start();
|
|
|
|
SearchEngine.instance().setDb(reg.db());
|
|
SearchEngine.instance().setLogger(reg.logger());
|
|
SearchEngine.instance().scheduleSyncTables();
|
|
|
|
await MigrationService.instance().run();
|
|
|
|
reg.scheduleSync().then(() => {
|
|
// Wait for the first sync before updating the notifications, since synchronisation
|
|
// might change the notifications.
|
|
AlarmService.updateAllNotifications();
|
|
|
|
DecryptionWorker.instance().scheduleStart();
|
|
});
|
|
|
|
await WelcomeUtils.install(dispatch);
|
|
|
|
// Collect revisions more frequently on mobile because it doesn't auto-save
|
|
// and it cannot collect anything when the app is not active.
|
|
RevisionService.instance().runInBackground(1000 * 30);
|
|
|
|
reg.logger().info('Application initialized');
|
|
}
|
|
|
|
class AppComponent extends React.Component {
|
|
|
|
constructor() {
|
|
super();
|
|
|
|
this.state = {
|
|
sideMenuContentOpacity: new Animated.Value(0),
|
|
};
|
|
|
|
this.lastSyncStarted_ = defaultState.syncStarted;
|
|
|
|
this.backButtonHandler_ = () => {
|
|
return this.backButtonHandler();
|
|
};
|
|
|
|
this.onAppStateChange_ = () => {
|
|
PoorManIntervals.update();
|
|
};
|
|
}
|
|
|
|
async componentDidMount() {
|
|
if (this.props.appState == 'starting') {
|
|
this.props.dispatch({
|
|
type: 'APP_STATE_SET',
|
|
state: 'initializing',
|
|
});
|
|
|
|
await initialize(this.props.dispatch);
|
|
|
|
this.props.dispatch({
|
|
type: 'APP_STATE_SET',
|
|
state: 'ready',
|
|
});
|
|
}
|
|
|
|
BackButtonService.initialize(this.backButtonHandler_);
|
|
|
|
AlarmService.setInAppNotificationHandler(async (alarmId) => {
|
|
const alarm = await Alarm.load(alarmId);
|
|
const notification = await Alarm.makeNotification(alarm);
|
|
this.dropdownAlert_.alertWithType('info', notification.title, notification.body ? notification.body : '');
|
|
});
|
|
|
|
AppState.addEventListener('change', this.onAppStateChange_);
|
|
|
|
const sharedData = await ShareExtension.data();
|
|
if (sharedData) {
|
|
reg.logger().info('Received shared data');
|
|
if (this.props.selectedFolderId) {
|
|
handleShared(sharedData, this.props.selectedFolderId, this.props.dispatch);
|
|
} else {
|
|
reg.logger.info('Cannot handle share - default folder id is not set');
|
|
}
|
|
}
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
AppState.removeEventListener('change', this.onAppStateChange_);
|
|
}
|
|
|
|
componentDidUpdate(prevProps) {
|
|
if (this.props.showSideMenu !== prevProps.showSideMenu) {
|
|
Animated.timing(this.state.sideMenuContentOpacity, {
|
|
toValue: this.props.showSideMenu ? 0.5 : 0,
|
|
duration: 600,
|
|
}).start();
|
|
}
|
|
}
|
|
|
|
async backButtonHandler() {
|
|
if (this.props.noteSelectionEnabled) {
|
|
this.props.dispatch({ type: 'NOTE_SELECTION_END' });
|
|
return true;
|
|
}
|
|
|
|
if (this.props.showSideMenu) {
|
|
this.props.dispatch({ type: 'SIDE_MENU_CLOSE' });
|
|
return true;
|
|
}
|
|
|
|
if (this.props.historyCanGoBack) {
|
|
this.props.dispatch({ type: 'NAV_BACK' });
|
|
return true;
|
|
}
|
|
|
|
BackHandler.exitApp();
|
|
|
|
return false;
|
|
}
|
|
|
|
UNSAFE_componentWillReceiveProps(newProps) {
|
|
if (newProps.syncStarted != this.lastSyncStarted_) {
|
|
if (!newProps.syncStarted) FoldersScreenUtils.refreshFolders();
|
|
this.lastSyncStarted_ = newProps.syncStarted;
|
|
}
|
|
}
|
|
|
|
sideMenu_change(isOpen) {
|
|
// Make sure showSideMenu property of state is updated
|
|
// when the menu is open/closed.
|
|
this.props.dispatch({
|
|
type: isOpen ? 'SIDE_MENU_OPEN' : 'SIDE_MENU_CLOSE',
|
|
});
|
|
}
|
|
|
|
render() {
|
|
if (this.props.appState != 'ready') return null;
|
|
const theme = themeStyle(this.props.theme);
|
|
|
|
let sideMenuContent = null;
|
|
let menuPosition = 'left';
|
|
|
|
if (this.props.routeName === 'Note') {
|
|
sideMenuContent = <SafeAreaView style={{ flex: 1, backgroundColor: theme.backgroundColor }}><SideMenuContentNote options={this.props.noteSideMenuOptions}/></SafeAreaView>;
|
|
menuPosition = 'right';
|
|
} else {
|
|
sideMenuContent = <SafeAreaView style={{ flex: 1, backgroundColor: theme.backgroundColor }}><SideMenuContent/></SafeAreaView>;
|
|
}
|
|
|
|
const appNavInit = {
|
|
Notes: { screen: NotesScreen },
|
|
Note: { screen: NoteScreen },
|
|
Tags: { screen: TagsScreen },
|
|
Folder: { screen: FolderScreen },
|
|
OneDriveLogin: { screen: OneDriveLoginScreen },
|
|
DropboxLogin: { screen: DropboxLoginScreen },
|
|
EncryptionConfig: { screen: EncryptionConfigScreen },
|
|
Log: { screen: LogScreen },
|
|
Status: { screen: StatusScreen },
|
|
Search: { screen: SearchScreen },
|
|
Config: { screen: ConfigScreen },
|
|
};
|
|
|
|
const statusBarStyle = theme.appearance === 'light' ? 'dark-content' : 'light-content';
|
|
|
|
return (
|
|
<View style={{ flex: 1, backgroundColor: theme.backgroundColor }}>
|
|
<SideMenu
|
|
menu={sideMenuContent}
|
|
edgeHitWidth={5}
|
|
menuPosition={menuPosition}
|
|
onChange={(isOpen) => this.sideMenu_change(isOpen)}
|
|
onSliding={(percent) => {
|
|
this.props.dispatch({
|
|
type: 'SIDE_MENU_OPEN_PERCENT',
|
|
value: percent,
|
|
});
|
|
}}
|
|
>
|
|
<StatusBar barStyle={statusBarStyle} />
|
|
<MenuContext style={{ flex: 1, backgroundColor: theme.backgroundColor }}>
|
|
<SafeAreaView style={{ flex: 1 }}>
|
|
<View style={{ flex: 1, backgroundColor: theme.backgroundColor }}>
|
|
<AppNav screens={appNavInit} />
|
|
</View>
|
|
<DropdownAlert ref={ref => this.dropdownAlert_ = ref} tapToCloseEnabled={true} />
|
|
<Animated.View pointerEvents='none' style={{ position: 'absolute', backgroundColor: 'black', opacity: this.state.sideMenuContentOpacity, width: '100%', height: '120%' }}/>
|
|
</SafeAreaView>
|
|
</MenuContext>
|
|
</SideMenu>
|
|
</View>
|
|
);
|
|
}
|
|
}
|
|
|
|
const mapStateToProps = (state) => {
|
|
return {
|
|
historyCanGoBack: state.historyCanGoBack,
|
|
showSideMenu: state.showSideMenu,
|
|
syncStarted: state.syncStarted,
|
|
appState: state.appState,
|
|
noteSelectionEnabled: state.noteSelectionEnabled,
|
|
selectedFolderId: state.selectedFolderId,
|
|
routeName: state.route.routeName,
|
|
theme: state.settings.theme,
|
|
noteSideMenuOptions: state.noteSideMenuOptions,
|
|
};
|
|
};
|
|
|
|
const App = connect(mapStateToProps)(AppComponent);
|
|
|
|
class Root extends React.Component {
|
|
render() {
|
|
return (
|
|
<Provider store={store}>
|
|
<App/>
|
|
</Provider>
|
|
);
|
|
}
|
|
}
|
|
|
|
module.exports = { Root };
|