diff --git a/apps/mobile/html/Web.bundle/site/init.js b/apps/mobile/html/Web.bundle/site/init.js index 845da1a9a..c86c96ba8 100755 --- a/apps/mobile/html/Web.bundle/site/init.js +++ b/apps/mobile/html/Web.bundle/site/init.js @@ -124,6 +124,9 @@ function init_tiny(size) { directionality: EDITOR_SETTINGS.directionality, skin_url: 'dist/skins/notesnook', content_css: 'dist/skins/notesnook', + attachmenthandler_download_attachment:(hash) => { + reactNativeEventHandler('attachment_download',hash); + }, plugins: [ 'checklist advlist autolink textpattern hr lists link noneditable image', 'searchreplace codeblock inlinecode keyboardquirks attachmentshandler', @@ -177,24 +180,6 @@ function init_tiny(size) { .h { display: none; } - .img_float_left { - float:left; - } - .img_float_right { - float:right; - } - .img_float_none { - float:none; - } - .img_size_one { - width:100%; - } - .img_size_two { - width:50%; - } - .img_size_three { - width:25%; - } span.diff-del { background-color: #FDB0C0; } diff --git a/apps/mobile/src/components/AttachmentDialog/index.js b/apps/mobile/src/components/AttachmentDialog/index.js index 54ddc0585..5d956be8a 100644 --- a/apps/mobile/src/components/AttachmentDialog/index.js +++ b/apps/mobile/src/components/AttachmentDialog/index.js @@ -1,6 +1,9 @@ import React, {useEffect, useRef, useState} from 'react'; -import {Platform, ScrollView, Text, View} from 'react-native'; +import {Platform, TouchableOpacity, View} from 'react-native'; import {FlatList} from 'react-native-gesture-handler'; +import * as Progress from 'react-native-progress'; +import * as ScopedStorage from 'react-native-scoped-storage'; +import Sodium from 'react-native-sodium'; import Icon from 'react-native-vector-icons/MaterialCommunityIcons'; import {useTracked} from '../../provider'; import {useAttachmentStore} from '../../provider/stores'; @@ -12,24 +15,15 @@ import { import {db} from '../../utils/database'; import { eCloseAttachmentDialog, - eCloseTagsDialog, - eOpenAttachmentsDialog, - eOpenTagsDialog + eOpenAttachmentsDialog } from '../../utils/Events'; +import filesystem from '../../utils/filesystem'; import {SIZE} from '../../utils/SizeUtils'; -import {sleep} from '../../utils/TimeUtils'; +import Storage from '../../utils/storage'; import {ActionIcon} from '../ActionIcon'; import ActionSheetWrapper from '../ActionSheetComponent/ActionSheetWrapper'; import DialogHeader from '../Dialog/dialog-header'; -import {PressableButton} from '../PressableButton'; -import Heading from '../Typography/Heading'; import Paragraph from '../Typography/Paragraph'; -import * as Progress from 'react-native-progress'; -import filesystem from '../../utils/filesystem'; -import * as ScopedStorage from 'react-native-scoped-storage'; -import RNFetchBlob from 'rn-fetch-blob'; -import Sodium from 'react-native-sodium'; -import Storage from '../../utils/storage'; export const AttachmentDialog = () => { const [state] = useTracked(); @@ -142,7 +136,6 @@ const Attachment = ({attachment, note, setNote}) => { const [state] = useTracked(); const colors = state.colors; const progress = useAttachmentStore(state => state.progress); - const setProgress = useAttachmentStore(state => state.setProgress); const [currentProgress, setCurrentProgress] = useState(null); const onPress = async () => { @@ -151,57 +144,22 @@ const Attachment = ({attachment, note, setNote}) => { useAttachmentStore.getState().remove(attachment.metadata.hash); return; } - - let folder = {}; - if (Platform.OS === 'android') { - folder = await ScopedStorage.openDocumentTree(false); - if (!folder) return; - } else { - folder.uri = await Storage.checkAndCreateDir('/Downloads/'); - } - - try { - setCurrentProgress({ - value: 0, - percent: '0%' - }); - await db.fs.downloadFile( - attachment.metadata.hash, - attachment.metadata.hash - ); - let key = await db.user.getEncryptionKey(); - let info = { - iv: attachment.iv, - salt: attachment.salt, - length: attachment.length, - alg: attachment.alg, - hash: attachment.metadata.hash, - hashType: attachment.metadata.hashType, - mime: attachment.metadata.type, - fileName: attachment.metadata.filename, - uri: folder.uri - }; - await Sodium.decryptFile(key, info, false); - ToastEvent.show({ - heading: 'Download successful', - message: attachment.metadata.filename + 'downloaded', - type: 'success', - context: 'local' - }); - } catch (e) { - console.log('download attachment error: ', e); - useAttachmentStore.getState().remove(attachment.metadata.hash); - } + filesystem.downloadAttachment(attachment.metadata.hash); }; + useEffect(() => { let prog = progress[attachment.metadata.hash]; - if (prog && prog.type === 'download') { - prog = prog.recieved / prog.total; + if (prog) { + let type = prog.type; + let loaded = prog.type === 'download' ? prog.recieved : prog.sent; + prog = loaded / prog.total; prog = (prog * 100).toFixed(0); console.log('progress: ', prog); + console.log(prog); setCurrentProgress({ value: prog, - percent: prog + '%' + percent: prog + '%', + type: type }); } else { setCurrentProgress(null); @@ -263,30 +221,38 @@ const Attachment = ({attachment, note, setNote}) => { - {formatBytes(attachment.length)} ({attachment.metadata.type}) + {formatBytes(attachment.length)}{' '} + {currentProgress?.type ? '(' + currentProgress.type + 'ing - tap to cancel)' : ''} {currentProgress ? ( - { + db.fs.cancel(attachment.metadata.hash); + setCurrentProgress(null); + }} style={{ justifyContent: 'center', - marginLeft: 5 + marginLeft: 5, + marginTop:5, + marginRight:-5 }}> (progress * 100).toFixed(0)} borderWidth={0} thickness={2} /> - + ) : ( onPress(attachment)} diff --git a/apps/mobile/src/provider/stores.ts b/apps/mobile/src/provider/stores.ts index fabb2635e..4984b66cd 100644 --- a/apps/mobile/src/provider/stores.ts +++ b/apps/mobile/src/provider/stores.ts @@ -158,11 +158,10 @@ interface AttachmentStore { hash: string, recieved: number, type: "upload" | "download", - success: boolean } }, remove:(hash:string) => void - setProgress: (sent: number, total: number, hash: string, recieved: number, type: "upload" | "download", success) => void + setProgress: (sent: number, total: number, hash: string, recieved: number, type: "upload" | "download") => void } export const useAttachmentStore = create((set, get) => ({ @@ -170,11 +169,20 @@ export const useAttachmentStore = create((set, get) => ({ remove:(hash) => { let _p = get().progress; _p[hash] = null + tiny.call(EditorWebView, ` + (function() { + let progress = ${JSON.stringify({ + loaded:1, + total:1, + hash + })} + tinymce.activeEditor.execCommand("mceUpdateAttachmentProgress",progress); + })()`); set({ progress: { ..._p } }); }, - setProgress: (sent, total, hash, recieved, type, success) => { + setProgress: (sent, total, hash, recieved, type) => { let _p = get().progress; - _p[hash] = { sent, total, hash, recieved, type, success }; + _p[hash] = { sent, total, hash, recieved, type }; let progress = { total, hash, loaded: type === "download" ? recieved : sent }; tiny.call(EditorWebView, ` (function() { diff --git a/apps/mobile/src/utils/filesystem.js b/apps/mobile/src/utils/filesystem.js index d827e2f7f..40dd5b5cb 100644 --- a/apps/mobile/src/utils/filesystem.js +++ b/apps/mobile/src/utils/filesystem.js @@ -1,11 +1,13 @@ import Sodium from 'react-native-sodium'; import RNFetchBlob from 'rn-fetch-blob'; import {useAttachmentStore} from '../provider/stores'; +import {ToastEvent} from '../services/EventManager'; +import {db} from './database'; +import Storage from './storage'; +import * as ScopedStorage from 'react-native-scoped-storage'; const cacheDir = RNFetchBlob.fs.dirs.CacheDir; -let placeholder = ``; - async function readEncrypted(filename, key, cipherData) { try { let path = `${cacheDir}/${filename}`; @@ -84,7 +86,7 @@ async function downloadFile(filename, {url, headers}, cancelToken) { if (exists) return true; let request = RNFetchBlob.config({ path: path, - IOSBackgroundTask: true, + IOSBackgroundTask: true }) .fetch('GET', url, headers) .progress((recieved, total) => { @@ -129,16 +131,67 @@ function cancelable(operation) { return (filename, {url, headers}) => { return { execute: () => operation(filename, {url, headers}, cancelToken), - cancel: () => cancelToken.cancel() + cancel: async () => { + await cancelToken.cancel(); + RNFetchBlob.fs.unlink(`${cacheDir}/${filename}`); + } }; }; } +async function downloadAttachment(hash) { + let attachment = db.attachments.attachment(hash); + if (!attachment) { + console.log('attachment not found'); + return; + } + let folder = {}; + if (Platform.OS === 'android') { + folder = await ScopedStorage.openDocumentTree(false); + if (!folder) return; + } else { + folder.uri = await Storage.checkAndCreateDir('/Downloads/'); + } + + try { + await db.fs.downloadFile( + attachment.metadata.hash, + attachment.metadata.hash + ); + if ( + !(await RNFetchBlob.fs.exists(`${cacheDir}/${attachment.metadata.hash}`)) + ) + return; + let key = await db.user.getEncryptionKey(); + let info = { + iv: attachment.iv, + salt: attachment.salt, + length: attachment.length, + alg: attachment.alg, + hash: attachment.metadata.hash, + hashType: attachment.metadata.hashType, + mime: attachment.metadata.type, + fileName: attachment.metadata.filename, + uri: folder.uri + }; + await Sodium.decryptFile(key, info, false); + ToastEvent.show({ + heading: 'Download successful', + message: attachment.metadata.filename + ' downloaded', + type: 'success', + }); + } catch (e) { + console.log('download attachment error: ', e); + useAttachmentStore.getState().remove(attachment.metadata.hash); + } +} + export default { readEncrypted, writeEncrypted, uploadFile: cancelable(uploadFile), downloadFile: cancelable(downloadFile), deleteFile, - exists + exists, + downloadAttachment }; diff --git a/apps/mobile/src/views/Editor/Functions.js b/apps/mobile/src/views/Editor/Functions.js index 8e96858a2..3c25ca8dc 100644 --- a/apps/mobile/src/views/Editor/Functions.js +++ b/apps/mobile/src/views/Editor/Functions.js @@ -19,6 +19,7 @@ import { eShowGetPremium, eShowMergeDialog } from '../../utils/Events'; +import filesystem from '../../utils/filesystem'; import {openLinkInBrowser} from '../../utils/functions'; import {MMKV} from '../../utils/mmkv'; import {tabBarRef} from '../../utils/Refs'; @@ -31,6 +32,7 @@ export const editorTitleInput = createRef(); export const sourceUri = Platform.OS === 'android' ? 'file:///android_asset/' : 'Web.bundle/site/'; +let lastEditTime = 0; let EDITOR_SETTINGS = null; let webviewOK = true; let noteEdited = false; @@ -153,6 +155,7 @@ async function setNote(item) { title = note.title; id = note.id; noteEdited = false; + lastEditTime = item.dateEdited; if (note.locked) { content.data = note.content.data; content.type = note.content.type; @@ -188,6 +191,7 @@ export const loadNote = async item => { console.log('.....OPEN NOTE.....'); editing.currentlyEditing = true; editing.movedAway = false; + if (editing.isFocused) { tiny.call(EditorWebView, tiny.blur); } @@ -197,6 +201,7 @@ export const loadNote = async item => { await clearEditor(true, true, true); } disableSaving = false; + lastEditTime = 0; clearNote(); noteEdited = false; isFirstLoad = false; @@ -274,7 +279,6 @@ const checkStatus = async noreset => { }, 1000); }); }; -let lastEditTime = 0; export const _onMessage = async evt => { if (!evt || !evt.nativeEvent || !evt.nativeEvent.data) return; @@ -303,7 +307,6 @@ export const _onMessage = async evt => { case 'title': if (message.value !== title) { noteEdited = true; - lastEditTime = Date.now(); title = message.value; eSendEvent('editorScroll', { @@ -315,6 +318,9 @@ export const _onMessage = async evt => { case 'scroll': eSendEvent('editorScroll', message); break; + case 'attachment_download': + filesystem.downloadAttachment(message.value); + break; case 'noteLoaded': tiny.call(EditorWebView, tiny.notLoading); eSendEvent('loadingNote'); @@ -611,6 +617,7 @@ export async function saveNote(preventUpdate) { Navigation.routeNames.Notes ]); let n = db.notes.note(id)?.data?.dateEdited; + lastEditTime = n + 10; tiny.call(EditorWebView, tiny.updateDateEdited(timeConverter(n))); tiny.call(EditorWebView, tiny.updateSavingState('Saved')); } @@ -678,16 +685,28 @@ export async function updateNoteInEditor() { presentResolveConflictDialog(_note); return; } - let data = await db.content.raw(note.contentId); + let data = await db.content.raw(_note.contentId); if (lastEditTime > _note.dateEdited) return; if (data.data === content.data) return; if (content.data.indexOf(data.data) !== -1) return; if (note.dateEdited === _note.dateEdited) return; + console.log('injecting note in editor', lastEditTime, _note.dateEdited); + title = note.title; + content.data = data.data; + note = _note; + lastEditTime = _note.dateEdited + 10; tiny.call(EditorWebView, tiny.isLoading); await setNote(_note); + tiny.call(EditorWebView, tiny.isLoading); post('title', title); post('inject', content.data); + setTimeout(() => { + tiny.call(EditorWebView, tiny.notLoading); + }, 50); + if (id) { + db.attachments.download(id); + } tiny.call(EditorWebView, tiny.notLoading); } diff --git a/apps/mobile/src/views/Editor/tiny/toolbar/commands.js b/apps/mobile/src/views/Editor/tiny/toolbar/commands.js index acc312746..689ecdd86 100644 --- a/apps/mobile/src/views/Editor/tiny/toolbar/commands.js +++ b/apps/mobile/src/views/Editor/tiny/toolbar/commands.js @@ -94,7 +94,12 @@ export const execCommands = { EditorWebView, ` (function() { - let file = ${JSON.stringify(attachment)} + let file = ${JSON.stringify({ + hash:hash, + filename:file.name, + type:file.type, + size:file.size + })} tinymce.activeEditor.execCommand('mceAttachFile',file); setTimeout(function() { tinymce.activeEditor.nodeChanged({selectionChange:true}) @@ -127,8 +132,7 @@ export const execCommands = { maxHeight: 2000, quality: 0.8, mediaType: 'photo', - encryptToFile: false, - ...key + encryptToFile: false }, handleImageResponse ); @@ -147,8 +151,7 @@ export const execCommands = { maxHeight: 2000, quality: 0.8, mediaType: 'photo', - encryptToFile: false, - ...key + encryptToFile: false }, handleImageResponse ); @@ -170,69 +173,9 @@ export const execCommands = { tablesplitcell: "tinymce.activeEditor.execCommand('mceTableSplitCells');", tablemergecell: "tinymce.activeEditor.execCommand('mceTableMergeCells');", tablerowprops: "tinymce.activeEditor.execCommand('mceTableRowProps');", - imageResize25: `(function() { - let node = tinymce.activeEditor.selection.getNode(); - if (tinymce.activeEditor.selection.getNode().tagName === 'IMG') { - - tinymce.activeEditor.undoManager.transact(function() { - if (tinymce.activeEditor.dom.hasClass(node,"img_size_one")) { - tinymce.activeEditor.dom.removeClass(node,"img_size_one") - } - if (tinymce.activeEditor.dom.hasClass(node,"img_size_two")) { - tinymce.activeEditor.dom.removeClass(node,"img_size_two") - } - tinymce.activeEditor.dom.addClass(node,"img_size_three") - setTimeout(function() { - tinymce.activeEditor.nodeChanged({selectionChange:true}) - },100) - }); - - } - - })(); - - `, - imageResize50: `(function() { - let node = tinymce.activeEditor.selection.getNode(); - if (tinymce.activeEditor.selection.getNode().tagName === 'IMG') { - tinymce.activeEditor.undoManager.transact(function() { - if (tinymce.activeEditor.dom.hasClass(node,"img_size_one")) { - tinymce.activeEditor.dom.removeClass(node,"img_size_one") - } - if (tinymce.activeEditor.dom.hasClass(node,"img_size_three")) { - tinymce.activeEditor.dom.removeClass(node,"img_size_three") - } - tinymce.activeEditor.dom.addClass(node,"img_size_two") - setTimeout(function() { - tinymce.activeEditor.nodeChanged({selectionChange:true}) - },100) - - }); - - } - })() - - `, - imageResize100: `(function() { - let node = tinymce.activeEditor.selection.getNode(); - if (tinymce.activeEditor.selection.getNode().tagName === 'IMG') { - tinymce.activeEditor.undoManager.transact(function() { - if (tinymce.activeEditor.dom.hasClass(node,"img_size_three")) { - tinymce.activeEditor.dom.removeClass(node,"img_size_three") - } - if (tinymce.activeEditor.dom.hasClass(node,"img_size_two")) { - tinymce.activeEditor.dom.removeClass(node,"img_size_two") - } - tinymce.activeEditor.dom.addClass(node,"img_size_one") - setTimeout(function() { - tinymce.activeEditor.nodeChanged({selectionChange:true}) - },100) - - }); - } - })() - - `, + imageResize25: () => setImageSize(0.25), + imageResize50: () => setImageSize(0.5), + imageResize100: () => setImageSize(1), imagepreview: `(function() { if (tinymce.activeEditor.selection.getNode().tagName === 'IMG') { var xhr = new XMLHttpRequest(); @@ -272,67 +215,9 @@ export const execCommands = { } })(); `, - imagefloatleft: `(function () { -let node = tinymce.activeEditor.selection.getNode(); - if (node.tagName === 'IMG') { - - tinymce.activeEditor.undoManager.transact(function() { - if (tinymce.activeEditor.dom.hasClass(node,"img_float_right")) { - tinymce.activeEditor.dom.removeClass(node,"img_float_right") - } - if (tinymce.activeEditor.dom.hasClass(node,"img_float_none")) { - tinymce.activeEditor.dom.removeClass(node,"img_float_none") - } - tinymce.activeEditor.dom.addClass(node,"img_float_left") - setTimeout(function() { - tinymce.activeEditor.nodeChanged({selectionChange:true}) - },100) - }); - } - })(); - - `, - imagefloatright: `(function () { -let node = tinymce.activeEditor.selection.getNode(); - if (node.tagName === 'IMG') { - - tinymce.activeEditor.undoManager.transact(function() { - if (tinymce.activeEditor.dom.hasClass(node,"img_float_left")) { - tinymce.activeEditor.dom.removeClass(node, "img_float_left") - } - if (tinymce.activeEditor.dom.hasClass(node,"img_float_none")) { - tinymce.activeEditor.dom.removeClass(node,"img_float_none") - } - tinymce.activeEditor.dom.addClass(node,"img_float_right") - setTimeout(function() { - tinymce.activeEditor.nodeChanged({selectionChange:true}) - },100) - - }); - } - })() - - `, - imagefloatnone: `(function () { - let node = tinymce.activeEditor.selection.getNode(); - if (node.tagName === 'IMG') { - - tinymce.activeEditor.undoManager.transact(function() { - if (tinymce.activeEditor.dom.hasClass(node,"img_float_left")) { - tinymce.activeEditor.dom.removeClass(node,"img_float_left") - } - if (tinymce.activeEditor.dom.hasClass(node,"img_float_right")) { - tinymce.activeEditor.dom.removeClass(node,"img_float_right") - } - tinymce.activeEditor.dom.addClass(node,"img_float_none") - setTimeout(function() { - tinymce.activeEditor.nodeChanged({selectionChange:true}) - },100) - }); - } - })() - - `, + imagefloatleft: () => setFloat('left'), + imagefloatright: () => setFloat('right'), + imagefloatnone: () => setFloat('none'), 'line-break': ` tinymce.activeEditor.undoManager.transact(function() { tinymce.activeEditor.execCommand('InsertLineBreak'); @@ -358,7 +243,7 @@ const handleImageResponse = async response => { uri: image.uri, type: 'url' }); - console.log('hash: ',hash); + tiny.call( EditorWebView, ` @@ -367,7 +252,8 @@ const handleImageResponse = async response => { hash: hash, type: image.type, filename: image.fileName, - dataurl: b64 + dataurl: b64, + size: image.fileSize })} tinymce.activeEditor.execCommand('mceAttachImage',image); setTimeout(function() { @@ -404,3 +290,39 @@ async function attachFile(uri, hash, type, filename) { return false; } } + +const setFloat = float => `(function () { + let node = tinymce.activeEditor.selection.getNode(); + if (node.tagName === 'IMG') { + tinymce.activeEditor.undoManager.transact(function() { + node.style.float = "${float}"; + setTimeout(function() { + tinymce.activeEditor.nodeChanged({selectionChange:true}) + },100) + }); + } + })()`; + +const setImageSize = size => `(function() { + let node = tinymce.activeEditor.selection.getNode(); +if (tinymce.activeEditor.selection.getNode().tagName === 'IMG') { + tinymce.activeEditor.undoManager.transact(function() { + let rect = node.getBoundingClientRect(); + let originalWidth = rect.width; + let originalHeight = rect.height; + if (node.dataset.width) { + originalWidth = node.dataset.width; + originalHeight = node.dataset.height; + } else { + node.dataset.width = originalWidth; + node.dataset.height = originalHeight; + } + + node.width = originalWidth * ${size} + setTimeout(function() { + tinymce.activeEditor.nodeChanged({selectionChange:true}) + },100) + }); +} +})(); +`;