1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-12-11 23:17:19 +02:00

Compare commits

..

1 Commits

Author SHA1 Message Date
palerdot
e132744a71 android: vosk integration testing screen 2023-04-28 12:47:15 +05:30
362 changed files with 26025 additions and 30636 deletions

View File

@@ -74,8 +74,6 @@ packages/turndown/
packages/pdf-viewer/dist packages/pdf-viewer/dist
plugin_types/ plugin_types/
readme/ readme/
packages/react-native-vosk/lib/
packages/lib/countable/Countable.js
# AUTO-GENERATED - EXCLUDED TYPESCRIPT BUILD # AUTO-GENERATED - EXCLUDED TYPESCRIPT BUILD
packages/app-cli/app/LinkSelector.js packages/app-cli/app/LinkSelector.js
@@ -364,7 +362,6 @@ packages/app-mobile/components/CameraView.js
packages/app-mobile/components/CustomButton.js packages/app-mobile/components/CustomButton.js
packages/app-mobile/components/Dropdown.js packages/app-mobile/components/Dropdown.js
packages/app-mobile/components/ExtendedWebView.js packages/app-mobile/components/ExtendedWebView.js
packages/app-mobile/components/FolderPicker.js
packages/app-mobile/components/NoteBodyViewer/NoteBodyViewer.js packages/app-mobile/components/NoteBodyViewer/NoteBodyViewer.js
packages/app-mobile/components/NoteBodyViewer/hooks/useOnMessage.js packages/app-mobile/components/NoteBodyViewer/hooks/useOnMessage.js
packages/app-mobile/components/NoteBodyViewer/hooks/useOnResourceLongPress.js packages/app-mobile/components/NoteBodyViewer/hooks/useOnResourceLongPress.js
@@ -413,6 +410,7 @@ packages/app-mobile/components/biometrics/biometricAuthenticate.js
packages/app-mobile/components/biometrics/sensorInfo.js packages/app-mobile/components/biometrics/sensorInfo.js
packages/app-mobile/components/getResponsiveValue.js packages/app-mobile/components/getResponsiveValue.js
packages/app-mobile/components/getResponsiveValue.test.js packages/app-mobile/components/getResponsiveValue.test.js
packages/app-mobile/components/screens/VoskScreen.js
packages/app-mobile/components/screens/ConfigScreen.js packages/app-mobile/components/screens/ConfigScreen.js
packages/app-mobile/components/screens/Note.js packages/app-mobile/components/screens/Note.js
packages/app-mobile/components/screens/Notes.js packages/app-mobile/components/screens/Notes.js
@@ -420,15 +418,12 @@ packages/app-mobile/components/screens/UpgradeSyncTargetScreen.js
packages/app-mobile/components/screens/encryption-config.js packages/app-mobile/components/screens/encryption-config.js
packages/app-mobile/components/screens/search.js packages/app-mobile/components/screens/search.js
packages/app-mobile/components/side-menu-content.js packages/app-mobile/components/side-menu-content.js
packages/app-mobile/components/voiceTyping/VoiceTypingDialog.js
packages/app-mobile/gulpfile.js packages/app-mobile/gulpfile.js
packages/app-mobile/root.js packages/app-mobile/root.js
packages/app-mobile/services/AlarmServiceDriver.android.js packages/app-mobile/services/AlarmServiceDriver.android.js
packages/app-mobile/services/AlarmServiceDriver.ios.js packages/app-mobile/services/AlarmServiceDriver.ios.js
packages/app-mobile/services/e2ee/RSA.react-native.js packages/app-mobile/services/e2ee/RSA.react-native.js
packages/app-mobile/services/profiles/index.js packages/app-mobile/services/profiles/index.js
packages/app-mobile/services/voiceTyping/vosk.android.js
packages/app-mobile/services/voiceTyping/vosk.ios.js
packages/app-mobile/setupQuickActions.js packages/app-mobile/setupQuickActions.js
packages/app-mobile/tools/buildInjectedJs.js packages/app-mobile/tools/buildInjectedJs.js
packages/app-mobile/utils/ShareExtension.js packages/app-mobile/utils/ShareExtension.js
@@ -485,7 +480,6 @@ packages/lib/SyncTargetOneDrive.js
packages/lib/SyncTargetRegistry.js packages/lib/SyncTargetRegistry.js
packages/lib/Synchronizer.js packages/lib/Synchronizer.js
packages/lib/TaskQueue.js packages/lib/TaskQueue.js
packages/lib/WelcomeUtils.js
packages/lib/array.js packages/lib/array.js
packages/lib/callbackUrlUtils.js packages/lib/callbackUrlUtils.js
packages/lib/callbackUrlUtils.test.js packages/lib/callbackUrlUtils.test.js
@@ -521,7 +515,6 @@ packages/lib/import-enex-md-gen.js
packages/lib/import-enex-md-gen.test.js packages/lib/import-enex-md-gen.test.js
packages/lib/import-enex.js packages/lib/import-enex.js
packages/lib/locale.js packages/lib/locale.js
packages/lib/locale.test.js
packages/lib/markdownUtils.js packages/lib/markdownUtils.js
packages/lib/markdownUtils.test.js packages/lib/markdownUtils.test.js
packages/lib/markdownUtils2.test.js packages/lib/markdownUtils2.test.js
@@ -848,7 +841,6 @@ packages/renderer/noteStyle.js
packages/renderer/pathUtils.js packages/renderer/pathUtils.js
packages/renderer/utils.js packages/renderer/utils.js
packages/tools/build-release-stats.js packages/tools/build-release-stats.js
packages/tools/build-welcome.js
packages/tools/buildServerDocker.js packages/tools/buildServerDocker.js
packages/tools/buildServerDocker.test.js packages/tools/buildServerDocker.test.js
packages/tools/bundleDefaultPlugins.js packages/tools/bundleDefaultPlugins.js

View File

@@ -141,7 +141,6 @@ module.exports = {
'spaced-comment': ['error', 'always'], 'spaced-comment': ['error', 'always'],
'keyword-spacing': ['error', { 'before': true, 'after': true }], 'keyword-spacing': ['error', { 'before': true, 'after': true }],
'no-multi-spaces': ['error'], 'no-multi-spaces': ['error'],
'prefer-object-spread': ['error'],
// Regarding the keyword blacklist: // Regarding the keyword blacklist:
// - err: We generally avoid using too many abbreviations, so it should // - err: We generally avoid using too many abbreviations, so it should

View File

@@ -183,32 +183,12 @@ if [[ $GIT_TAG_NAME = v* ]]; then
# cd "$ROOT_DIR/packages/tools" # cd "$ROOT_DIR/packages/tools"
# node bundleDefaultPlugins.js # node bundleDefaultPlugins.js
cd "$ROOT_DIR/packages/app-desktop" cd "$ROOT_DIR/packages/app-desktop"
USE_HARD_LINKS=false yarn run dist
if [ "$IS_MACOS" == "1" ]; then
# This is to fix this error:
#
# Exit code: ENOENT. spawn /usr/bin/python ENOENT
#
# Ref: https://github.com/electron-userland/electron-builder/issues/6767#issuecomment-1096589528
#
# It can be removed once we upgrade to electron-builder@23, however we
# cannot currently do this due to this error:
# https://github.com/laurent22/joplin/issues/8149
PYTHON_PATH=$(which python) USE_HARD_LINKS=false yarn run dist
else
USE_HARD_LINKS=false yarn run dist
fi
elif [[ $IS_LINUX = 1 ]] && [[ $GIT_TAG_NAME = $SERVER_TAG_PREFIX-* ]]; then elif [[ $IS_LINUX = 1 ]] && [[ $GIT_TAG_NAME = $SERVER_TAG_PREFIX-* ]]; then
echo "Step: Building Docker Image..." echo "Step: Building Docker Image..."
cd "$ROOT_DIR" cd "$ROOT_DIR"
yarn run buildServerDocker --tag-name $GIT_TAG_NAME --push-images --repository $SERVER_REPOSITORY yarn run buildServerDocker --tag-name $GIT_TAG_NAME --push-images --repository $SERVER_REPOSITORY
else else
echo "Step: Building but *not* publishing desktop application..." echo "Step: Building but *not* publishing desktop application..."
USE_HARD_LINKS=false yarn run dist --publish=never
if [ "$IS_MACOS" == "1" ]; then
# See above why we need to specify Python
PYTHON_PATH=$(which python) USE_HARD_LINKS=false yarn run dist --publish=never
else
USE_HARD_LINKS=false yarn run dist --publish=never
fi
fi fi

View File

@@ -17,8 +17,9 @@ jobs:
concurrent_skipping: 'same_content_newer' concurrent_skipping: 'same_content_newer'
BuildAndroidDebug: BuildAndroidDebug:
if: github.repository == 'laurent22/joplin'
needs: pre_job needs: pre_job
if: github.repository == 'laurent22/joplin' && needs.pre_job.outputs.should_skip != 'true' if: needs.pre_job.outputs.should_skip != 'true'
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Install Linux dependencies - name: Install Linux dependencies

View File

@@ -50,7 +50,6 @@ jobs:
sudo apt-get install -y gettext sudo apt-get install -y gettext
sudo apt-get install -y libsecret-1-dev sudo apt-get install -y libsecret-1-dev
sudo apt-get install -y translate-toolkit sudo apt-get install -y translate-toolkit
sudo apt-get install -y rsync
- name: Install Docker Engine - name: Install Docker Engine
# if: runner.os == 'Linux' && startsWith(github.ref, 'refs/tags/server-v') # if: runner.os == 'Linux' && startsWith(github.ref, 'refs/tags/server-v')

8
.gitignore vendored
View File

@@ -349,7 +349,6 @@ packages/app-mobile/components/CameraView.js
packages/app-mobile/components/CustomButton.js packages/app-mobile/components/CustomButton.js
packages/app-mobile/components/Dropdown.js packages/app-mobile/components/Dropdown.js
packages/app-mobile/components/ExtendedWebView.js packages/app-mobile/components/ExtendedWebView.js
packages/app-mobile/components/FolderPicker.js
packages/app-mobile/components/NoteBodyViewer/NoteBodyViewer.js packages/app-mobile/components/NoteBodyViewer/NoteBodyViewer.js
packages/app-mobile/components/NoteBodyViewer/hooks/useOnMessage.js packages/app-mobile/components/NoteBodyViewer/hooks/useOnMessage.js
packages/app-mobile/components/NoteBodyViewer/hooks/useOnResourceLongPress.js packages/app-mobile/components/NoteBodyViewer/hooks/useOnResourceLongPress.js
@@ -398,6 +397,7 @@ packages/app-mobile/components/biometrics/biometricAuthenticate.js
packages/app-mobile/components/biometrics/sensorInfo.js packages/app-mobile/components/biometrics/sensorInfo.js
packages/app-mobile/components/getResponsiveValue.js packages/app-mobile/components/getResponsiveValue.js
packages/app-mobile/components/getResponsiveValue.test.js packages/app-mobile/components/getResponsiveValue.test.js
packages/app-mobile/components/screens/VoskScreen.js
packages/app-mobile/components/screens/ConfigScreen.js packages/app-mobile/components/screens/ConfigScreen.js
packages/app-mobile/components/screens/Note.js packages/app-mobile/components/screens/Note.js
packages/app-mobile/components/screens/Notes.js packages/app-mobile/components/screens/Notes.js
@@ -405,15 +405,12 @@ packages/app-mobile/components/screens/UpgradeSyncTargetScreen.js
packages/app-mobile/components/screens/encryption-config.js packages/app-mobile/components/screens/encryption-config.js
packages/app-mobile/components/screens/search.js packages/app-mobile/components/screens/search.js
packages/app-mobile/components/side-menu-content.js packages/app-mobile/components/side-menu-content.js
packages/app-mobile/components/voiceTyping/VoiceTypingDialog.js
packages/app-mobile/gulpfile.js packages/app-mobile/gulpfile.js
packages/app-mobile/root.js packages/app-mobile/root.js
packages/app-mobile/services/AlarmServiceDriver.android.js packages/app-mobile/services/AlarmServiceDriver.android.js
packages/app-mobile/services/AlarmServiceDriver.ios.js packages/app-mobile/services/AlarmServiceDriver.ios.js
packages/app-mobile/services/e2ee/RSA.react-native.js packages/app-mobile/services/e2ee/RSA.react-native.js
packages/app-mobile/services/profiles/index.js packages/app-mobile/services/profiles/index.js
packages/app-mobile/services/voiceTyping/vosk.android.js
packages/app-mobile/services/voiceTyping/vosk.ios.js
packages/app-mobile/setupQuickActions.js packages/app-mobile/setupQuickActions.js
packages/app-mobile/tools/buildInjectedJs.js packages/app-mobile/tools/buildInjectedJs.js
packages/app-mobile/utils/ShareExtension.js packages/app-mobile/utils/ShareExtension.js
@@ -470,7 +467,6 @@ packages/lib/SyncTargetOneDrive.js
packages/lib/SyncTargetRegistry.js packages/lib/SyncTargetRegistry.js
packages/lib/Synchronizer.js packages/lib/Synchronizer.js
packages/lib/TaskQueue.js packages/lib/TaskQueue.js
packages/lib/WelcomeUtils.js
packages/lib/array.js packages/lib/array.js
packages/lib/callbackUrlUtils.js packages/lib/callbackUrlUtils.js
packages/lib/callbackUrlUtils.test.js packages/lib/callbackUrlUtils.test.js
@@ -506,7 +502,6 @@ packages/lib/import-enex-md-gen.js
packages/lib/import-enex-md-gen.test.js packages/lib/import-enex-md-gen.test.js
packages/lib/import-enex.js packages/lib/import-enex.js
packages/lib/locale.js packages/lib/locale.js
packages/lib/locale.test.js
packages/lib/markdownUtils.js packages/lib/markdownUtils.js
packages/lib/markdownUtils.test.js packages/lib/markdownUtils.test.js
packages/lib/markdownUtils2.test.js packages/lib/markdownUtils2.test.js
@@ -833,7 +828,6 @@ packages/renderer/noteStyle.js
packages/renderer/pathUtils.js packages/renderer/pathUtils.js
packages/renderer/utils.js packages/renderer/utils.js
packages/tools/build-release-stats.js packages/tools/build-release-stats.js
packages/tools/build-welcome.js
packages/tools/buildServerDocker.js packages/tools/buildServerDocker.js
packages/tools/buildServerDocker.test.js packages/tools/buildServerDocker.test.js
packages/tools/bundleDefaultPlugins.js packages/tools/bundleDefaultPlugins.js

View File

@@ -1,209 +0,0 @@
diff --git a/android/build.gradle b/android/build.gradle
index 6afcbbf0cc8ca2d69dd78077d61e59a90b2136bb..9f8d72b4ec5b2b3d290975d6a255917c95300854 100644
--- a/android/build.gradle
+++ b/android/build.gradle
@@ -67,19 +67,19 @@ repositories {
}
// Generate UUIDs for each models contained in android/src/main/assets/
-tasks.register('genUUID') {
- doLast {
- fileTree(dir: "$rootDir/app/src/main/assets", exclude: ['*/*']).visit { fileDetails ->
- if (fileDetails.directory) {
- def odir = file("$rootDir/app/src/main/assets/$fileDetails.relativePath")
- def ofile = file("$odir/uuid")
- mkdir odir
- ofile.text = UUID.randomUUID().toString()
- }
- }
- }
-}
-preBuild.dependsOn genUUID
+// tasks.register('genUUID') {
+// doLast {
+// fileTree(dir: "$rootDir/app/src/main/assets", exclude: ['*/*']).visit { fileDetails ->
+// if (fileDetails.directory) {
+// def odir = file("$rootDir/app/src/main/assets/$fileDetails.relativePath")
+// def ofile = file("$odir/uuid")
+// mkdir odir
+// ofile.text = UUID.randomUUID().toString()
+// }
+// }
+// }
+// }
+// preBuild.dependsOn genUUID
def kotlin_version = getExtOrDefault('kotlinVersion')
diff --git a/android/src/main/java/com/reactnativevosk/VoskModule.kt b/android/src/main/java/com/reactnativevosk/VoskModule.kt
index 0e2b6595b1b2cf1ee01c6c64239c4b0ea37fce19..5a8539b9cce8951967640dba755e29a4e3ff404a 100644
--- a/android/src/main/java/com/reactnativevosk/VoskModule.kt
+++ b/android/src/main/java/com/reactnativevosk/VoskModule.kt
@@ -19,13 +19,25 @@ class VoskModule(reactContext: ReactApplicationContext) : ReactContextBaseJavaMo
return "Vosk"
}
+ @ReactMethod
+ fun addListener(type: String?) {
+ // Keep: Required for RN built in Event Emitter Calls.
+ }
+
+ @ReactMethod
+ fun removeListeners(type: Int?) {
+ // Keep: Required for RN built in Event Emitter Calls.
+ }
+
override fun onResult(hypothesis: String) {
// Get text data from string object
val text = getHypothesisText(hypothesis)
// Stop recording if data found
if (text != null && text.isNotEmpty()) {
- cleanRecognizer();
+ // Don't auto-stop the recogniser - we want to do that when the user
+ // presses on "stop" only.
+ // cleanRecognizer();
sendEvent("onResult", text)
}
}
@@ -93,12 +105,11 @@ class VoskModule(reactContext: ReactApplicationContext) : ReactContextBaseJavaMo
@ReactMethod
fun loadModel(path: String, promise: Promise) {
cleanModel();
- StorageService.unpack(context, path, "models",
- { model: Model? ->
- this.model = model
- promise.resolve("Model successfully loaded")
- }
- ) { e: IOException ->
+
+ try {
+ this.model = Model(path);
+ promise.resolve("Model successfully loaded")
+ } catch (e: IOException) {
this.model = null
promise.reject(e)
}
@@ -153,6 +164,25 @@ class VoskModule(reactContext: ReactApplicationContext) : ReactContextBaseJavaMo
cleanRecognizer();
}
+ @ReactMethod
+ fun stopOnly() {
+ if (speechService != null) {
+ speechService!!.stop()
+ }
+ }
+
+ @ReactMethod
+ fun cleanup() {
+ if (speechService != null) {
+ speechService!!.shutdown();
+ speechService = null
+ }
+ if (recognizer != null) {
+ recognizer!!.close();
+ recognizer = null;
+ }
+ }
+
@ReactMethod
fun unload() {
cleanRecognizer();
diff --git a/lib/typescript/index.d.ts b/lib/typescript/index.d.ts
index 441e41cc402cca3a60b34978ef4fea976076259c..a173acebb4b314402550442ad471e0f7c706e3c4 100644
--- a/lib/typescript/index.d.ts
+++ b/lib/typescript/index.d.ts
@@ -10,6 +10,8 @@ export default class Vosk {
currentRegisteredEvents: EmitterSubscription[];
start: (grammar?: string[] | null) => Promise<String>;
stop: () => void;
+ stopOnly: () => void;
+ cleanup: () => void;
unload: () => void;
onResult: (onResult: (e: VoskEvent) => void) => EventSubscription;
onFinalResult: (onFinalResult: (e: VoskEvent) => void) => EventSubscription;
diff --git a/package.json b/package.json
index 707eddb8d68007f93071ac659c5b087c935c5f01..90ebe20f224eeec472c377df1fef9b15f2ff8200 100644
--- a/package.json
+++ b/package.json
@@ -11,12 +11,9 @@
"src",
"lib",
"android",
- "ios",
"cpp",
- "react-native-vosk.podspec",
"!lib/typescript/example",
"!android/build",
- "!ios/build",
"!**/__tests__",
"!**/__fixtures__",
"!**/__mocks__"
diff --git a/react-native-vosk.podspec b/react-native-vosk.podspec
deleted file mode 100644
index e3d41b90c5eef890c7a5108aaf16ac07d34a698b..0000000000000000000000000000000000000000
--- a/react-native-vosk.podspec
+++ /dev/null
@@ -1,41 +0,0 @@
-require "json"
-
-package = JSON.parse(File.read(File.join(__dir__, "package.json")))
-folly_version = '2021.06.28.00-v2'
-folly_compiler_flags = '-DFOLLY_NO_CONFIG -DFOLLY_MOBILE=1 -DFOLLY_USE_LIBCPP=1 -Wno-comma -Wno-shorten-64-to-32'
-
-Pod::Spec.new do |s|
- s.name = "react-native-vosk"
- s.version = package["version"]
- s.summary = package["description"]
- s.homepage = package["homepage"]
- s.license = package["license"]
- s.authors = package["author"]
-
- s.platforms = { :ios => "10.0" }
- s.source = { :git => "https://github.com/riderodd/react-native-vosk.git", :tag => "#{s.version}" }
-
- s.source_files = "ios/**/*.{h,m,mm,swift}"
- s.resource_bundles = { 'Vosk' => ['ios/Vosk/*'] }
-
- s.dependency "React-Core"
- s.frameworks = "Accelerate"
- s.library = "c++"
- s.vendored_frameworks = "ios/libvosk.xcframework"
- s.requires_arc = true
-
- # Don't install the dependencies when we run `pod install` in the old architecture.
- if ENV['RCT_NEW_ARCH_ENABLED'] == '1' then
- s.compiler_flags = folly_compiler_flags + " -DRCT_NEW_ARCH_ENABLED=1"
- s.pod_target_xcconfig = {
- "HEADER_SEARCH_PATHS" => "\"$(PODS_ROOT)/boost\"",
- "CLANG_CXX_LANGUAGE_STANDARD" => "c++17"
- }
-
- s.dependency "React-Codegen"
- s.dependency "RCT-Folly", folly_version
- s.dependency "RCTRequired"
- s.dependency "RCTTypeSafety"
- s.dependency "ReactCommon/turbomodule/core"
- end
-end
diff --git a/src/index.tsx b/src/index.tsx
index d9f90c921d89b1b4d85e145443ed3376546a368a..29e4068dbd7500828a73145bd25497a52c9bf638 100644
--- a/src/index.tsx
+++ b/src/index.tsx
@@ -69,6 +69,15 @@ export default class Vosk {
VoskModule.stop();
};
+ stopOnly = () => {
+ VoskModule.stopOnly();
+ };
+
+ cleanup = () => {
+ this.cleanListeners();
+ VoskModule.cleanup();
+ };
+
unload = () => {
this.cleanListeners();
VoskModule.unload();

View File

@@ -0,0 +1,30 @@
diff --git a/android/src/main/java/com/RNFetchBlob/RNFetchBlobReq.java b/android/src/main/java/com/RNFetchBlob/RNFetchBlobReq.java
index a8abd71833879201e3438b2fa51d712a311c4551..ffe9c2c6dfa5c703ba76b65d94d5dd6784102c19 100644
--- a/android/src/main/java/com/RNFetchBlob/RNFetchBlobReq.java
+++ b/android/src/main/java/com/RNFetchBlob/RNFetchBlobReq.java
@@ -591,7 +591,7 @@ public class RNFetchBlobReq extends BroadcastReceiver implements Runnable {
// ignored.printStackTrace();
}
- RNFetchBlobFileResp rnFetchBlobFileResp = (RNFetchBlobFileResp) responseBody;
+ RNFetchBlobFileResp rnFetchBlobFileResp = new RNFetchBlobFileResp(responseBody);
if(rnFetchBlobFileResp != null && !rnFetchBlobFileResp.isDownloadComplete()){
callback.invoke("Download interrupted.", null);
diff --git a/android/src/main/java/com/RNFetchBlob/Response/RNFetchBlobFileResp.java b/android/src/main/java/com/RNFetchBlob/Response/RNFetchBlobFileResp.java
index 2470eef612308c15a89dfea5a1f16937469be29f..965f8becc195965907699182c764ec9e51811450 100644
--- a/android/src/main/java/com/RNFetchBlob/Response/RNFetchBlobFileResp.java
+++ b/android/src/main/java/com/RNFetchBlob/Response/RNFetchBlobFileResp.java
@@ -35,6 +35,12 @@ public class RNFetchBlobFileResp extends ResponseBody {
FileOutputStream ofStream;
boolean isEndMarkerReceived;
+ // ref: https://github.com/joltup/rn-fetch-blob/issues/490#issuecomment-990899440
+ public RNFetchBlobFileResp(ResponseBody body) {
+ super();
+ this.originalBody = body;
+ }
+
public RNFetchBlobFileResp(ReactApplicationContext ctx, String taskId, ResponseBody body, String path, boolean overwrite) throws IOException {
super();
this.rctContext = ctx;

View File

@@ -1,136 +0,0 @@
<mxfile host="Electron" modified="2023-04-29T09:42:39.598Z" agent="Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) draw.io/21.2.1 Chrome/112.0.5615.87 Electron/24.1.2 Safari/537.36" etag="apmX4QvXCQymGu7gtKJn" version="21.2.1" type="device">
<diagram name="Page-1" id="5f0bae14-7c28-e335-631c-24af17079c00">
<mxGraphModel dx="1244" dy="759" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="1100" pageHeight="850" background="none" math="0" shadow="0">
<root>
<mxCell id="0" />
<mxCell id="1" parent="0" />
<mxCell id="93vzSs2z7RmF_nCAYhdf-1" value="Front end" style="rounded=0;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="465" y="120" width="170" height="60" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-4" value="Service" style="ellipse;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="320" y="280" width="120" height="80" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-5" value="Service" style="ellipse;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="490" y="280" width="120" height="80" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-6" value="Service" style="ellipse;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="670" y="280" width="120" height="80" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-10" value="Model" style="ellipse;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="320" y="430" width="120" height="80" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-11" value="Model" style="ellipse;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="490" y="430" width="120" height="80" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-12" value="Model" style="ellipse;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="670" y="430" width="120" height="80" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-14" value="SQLite database" style="shape=cylinder3;whiteSpace=wrap;html=1;boundedLbl=1;backgroundOutline=1;size=15;" vertex="1" parent="1">
<mxGeometry x="490" y="580" width="122.5" height="100" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-19" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-10" target="93vzSs2z7RmF_nCAYhdf-4">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-20" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-11" target="93vzSs2z7RmF_nCAYhdf-4">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-21" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-12" target="93vzSs2z7RmF_nCAYhdf-4">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-22" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-10" target="93vzSs2z7RmF_nCAYhdf-5">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-23" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-11" target="93vzSs2z7RmF_nCAYhdf-5">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-24" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-12" target="93vzSs2z7RmF_nCAYhdf-5">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-25" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-10" target="93vzSs2z7RmF_nCAYhdf-6">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-26" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-11" target="93vzSs2z7RmF_nCAYhdf-6">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-27" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-12" target="93vzSs2z7RmF_nCAYhdf-6">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-28" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-10" target="93vzSs2z7RmF_nCAYhdf-14">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-29" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-14" target="93vzSs2z7RmF_nCAYhdf-11">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-30" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-14" target="93vzSs2z7RmF_nCAYhdf-12">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="700" y="440" as="sourcePoint" />
<mxPoint x="750" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-31" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-4" target="93vzSs2z7RmF_nCAYhdf-1">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="620" y="440" as="sourcePoint" />
<mxPoint x="670" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-32" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-5" target="93vzSs2z7RmF_nCAYhdf-1">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="620" y="440" as="sourcePoint" />
<mxPoint x="670" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-33" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-6" target="93vzSs2z7RmF_nCAYhdf-1">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="620" y="440" as="sourcePoint" />
<mxPoint x="670" y="390" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-37" value="BACKEND" style="swimlane;whiteSpace=wrap;html=1;swimlaneFillColor=none;shadow=0;" vertex="1" parent="1">
<mxGeometry x="280" y="230" width="560" height="480" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-38" value="JSON config file" style="shape=note;whiteSpace=wrap;html=1;backgroundOutline=1;darkOpacity=0.05;" vertex="1" parent="1">
<mxGeometry x="910" y="420" width="80" height="100" as="geometry" />
</mxCell>
<mxCell id="93vzSs2z7RmF_nCAYhdf-39" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;exitX=1;exitY=0.5;exitDx=0;exitDy=0;" edge="1" parent="1" source="93vzSs2z7RmF_nCAYhdf-37" target="93vzSs2z7RmF_nCAYhdf-38">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="830" y="480" as="sourcePoint" />
<mxPoint x="800" y="580" as="targetPoint" />
</mxGeometry>
</mxCell>
</root>
</mxGraphModel>
</diagram>
</mxfile>

View File

@@ -1,70 +0,0 @@
<mxfile host="Electron" modified="2023-04-29T10:24:42.580Z" agent="Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) draw.io/21.2.1 Chrome/112.0.5615.87 Electron/24.1.2 Safari/537.36" etag="kcPEKHJGaBvNGFhEOF2g" version="21.2.1" type="device">
<diagram name="Page-1" id="5f0bae14-7c28-e335-631c-24af17079c00">
<mxGraphModel dx="1306" dy="797" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="1100" pageHeight="850" background="none" math="0" shadow="0">
<root>
<mxCell id="0" />
<mxCell id="1" parent="0" />
<mxCell id="t8PL5avYcYxuv0YEq-6K-7" value="Joplin Server" style="swimlane;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="320" y="40" width="630" height="465" as="geometry">
<mxRectangle x="350" y="300" width="120" height="30" as="alternateBounds" />
</mxGeometry>
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-2" value="Server application" style="rounded=0;whiteSpace=wrap;html=1;" vertex="1" parent="t8PL5avYcYxuv0YEq-6K-7">
<mxGeometry x="270" y="92.5" width="170" height="80" as="geometry" />
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-3" value="PostgreSQL" style="shape=cylinder3;whiteSpace=wrap;html=1;boundedLbl=1;backgroundOutline=1;size=15;" vertex="1" parent="t8PL5avYcYxuv0YEq-6K-7">
<mxGeometry x="190" y="262.5" width="140" height="110" as="geometry" />
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-5" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="t8PL5avYcYxuv0YEq-6K-7" source="t8PL5avYcYxuv0YEq-6K-3" target="t8PL5avYcYxuv0YEq-6K-2">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="280" y="232.5" as="sourcePoint" />
<mxPoint x="330" y="182.5" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-11" value="Note metadata,&lt;br&gt;user accounts, etc." style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];" vertex="1" connectable="0" parent="t8PL5avYcYxuv0YEq-6K-5">
<mxGeometry x="0.0586" y="1" relative="1" as="geometry">
<mxPoint as="offset" />
</mxGeometry>
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-4" value="AWS S3" style="shape=cube;whiteSpace=wrap;html=1;boundedLbl=1;backgroundOutline=1;darkOpacity=0.05;darkOpacity2=0.1;" vertex="1" parent="t8PL5avYcYxuv0YEq-6K-7">
<mxGeometry x="430" y="277.5" width="120" height="80" as="geometry" />
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-6" value="Note and attachment&lt;br&gt;content" style="endArrow=classic;startArrow=classic;html=1;rounded=0;exitX=0;exitY=0;exitDx=50;exitDy=0;exitPerimeter=0;" edge="1" parent="t8PL5avYcYxuv0YEq-6K-7" source="t8PL5avYcYxuv0YEq-6K-4" target="t8PL5avYcYxuv0YEq-6K-2">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="390" y="242.5" as="sourcePoint" />
<mxPoint x="440" y="192.5" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-12" value="Reverse proxy" style="whiteSpace=wrap;html=1;aspect=fixed;" vertex="1" parent="t8PL5avYcYxuv0YEq-6K-7">
<mxGeometry x="70" y="97.5" width="75" height="75" as="geometry" />
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-13" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;" edge="1" parent="t8PL5avYcYxuv0YEq-6K-7" source="t8PL5avYcYxuv0YEq-6K-12" target="t8PL5avYcYxuv0YEq-6K-2">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="180" y="160" as="sourcePoint" />
<mxPoint x="230" y="110" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-18" value="" style="edgeStyle=orthogonalEdgeStyle;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;" edge="1" parent="t8PL5avYcYxuv0YEq-6K-7" source="t8PL5avYcYxuv0YEq-6K-14" target="t8PL5avYcYxuv0YEq-6K-2">
<mxGeometry relative="1" as="geometry" />
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-14" value="Env file (config)" style="shape=note;whiteSpace=wrap;html=1;backgroundOutline=1;darkOpacity=0.05;" vertex="1" parent="t8PL5avYcYxuv0YEq-6K-7">
<mxGeometry x="540" y="90" width="60" height="85" as="geometry" />
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-8" value="Joplin Application (mobile, desktop, ...)" style="whiteSpace=wrap;html=1;aspect=fixed;" vertex="1" parent="1">
<mxGeometry x="40" y="110" width="130" height="130" as="geometry" />
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-9" value="" style="endArrow=classic;startArrow=classic;html=1;rounded=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" edge="1" parent="1" source="t8PL5avYcYxuv0YEq-6K-8" target="t8PL5avYcYxuv0YEq-6K-12">
<mxGeometry width="50" height="50" relative="1" as="geometry">
<mxPoint x="130" y="270" as="sourcePoint" />
<mxPoint x="310" y="230" as="targetPoint" />
</mxGeometry>
</mxCell>
<mxCell id="t8PL5avYcYxuv0YEq-6K-10" value="HTTP REST requests" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];" vertex="1" connectable="0" parent="t8PL5avYcYxuv0YEq-6K-9">
<mxGeometry x="0.0435" relative="1" as="geometry">
<mxPoint as="offset" />
</mxGeometry>
</mxCell>
</root>
</mxGraphModel>
</diagram>
</mxfile>

View File

@@ -42,7 +42,7 @@ export function addJoplinChecklistCommands(editor, ToggleList) {
}); });
editor.addCommand('InsertJoplinChecklist', function (ui, detail) { editor.addCommand('InsertJoplinChecklist', function (ui, detail) {
detail = { ...detail, listType: 'joplinChecklist' }; detail = Object.assign({}, detail, { listType: 'joplinChecklist' });
ToggleList.toggleList(editor, 'UL', detail); ToggleList.toggleList(editor, 'UL', detail);
}); });
} }

Binary file not shown.

Before

Width:  |  Height:  |  Size: 138 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 110 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 10 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 17 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 83 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6.1 KiB

View File

@@ -9,9 +9,7 @@ function getOs() {
function getFilename(path) { function getFilename(path) {
if (!path) return ''; if (!path) return '';
const s = path.split('/'); const s = path.split('/');
const urlWithParams = s.pop(); return s.pop();
const s2 = urlWithParams.split('?');
return s2[0];
} }
function getMobileOs() { function getMobileOs() {

View File

@@ -1,46 +1,4 @@
<?xml version="1.0" encoding="UTF-8"?><rss xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:content="http://purl.org/rss/1.0/modules/content/" xmlns:atom="http://www.w3.org/2005/Atom" version="2.0"><channel><title><![CDATA[Joplin]]></title><description><![CDATA[Joplin, the open source note-taking application]]></description><link>https://joplinapp.org</link><generator>RSS for Node</generator><lastBuildDate>Mon, 08 May 2023 00:00:00 GMT</lastBuildDate><atom:link href="https://joplinapp.org/rss.xml" rel="self" type="application/rss+xml"/><pubDate>Mon, 08 May 2023 00:00:00 GMT</pubDate><item><title><![CDATA[What's new in Joplin 2.10]]></title><description><![CDATA[<p>Great news! Joplin 2.10 is here and we've made some amazing improvements and bug fixes, with a focus on the mobile app.</p> <?xml version="1.0" encoding="UTF-8"?><rss xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:content="http://purl.org/rss/1.0/modules/content/" xmlns:atom="http://www.w3.org/2005/Atom" version="2.0"><channel><title><![CDATA[Joplin]]></title><description><![CDATA[Joplin, the open source note-taking application]]></description><link>https://joplinapp.org</link><generator>RSS for Node</generator><lastBuildDate>Thu, 02 Mar 2023 00:00:00 GMT</lastBuildDate><atom:link href="https://joplinapp.org/rss.xml" rel="self" type="application/rss+xml"/><pubDate>Thu, 02 Mar 2023 00:00:00 GMT</pubDate><item><title><![CDATA[Joplin will participate in JdLL 2023!]]></title><description><![CDATA[<p>On 1 and 2 April 2023, we will have a stand for Joplin at the <a href="https://www.jdll.org/">Journées du Logiciel Libre</a> in Lyon, France. The JdLL has been taking place in Lyon for 24 years and is a popular open source conference in France. We had a stand in 2020 and 2021 but that was cancelled due to Covid, so this year is a first for Joplin!</p>
<h1>New design for &quot;New note&quot; and &quot;New to-do&quot; buttons<a name="new-design-for-new-note-and-new-to-do-buttons" href="#new-design-for-new-note-and-new-to-do-buttons" class="heading-anchor">🔗</a></h1>
<p>We're excited to announce that we've made it even easier to create new notes and to-do lists with new designs for the &quot;New note&quot; and &quot;New to-do&quot; buttons.</p>
<p>If there is enough space, the button labels will be shown in full:</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20230508-new-note-1.png" alt=""></p>
<p>While for those who prefer a more narrow note list, only the button icons will be shown:</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20230508-new-note-2.png" alt=""></p>
<p>It's a small improvement, but we're confident it will make the app even more intuitive for new users.</p>
<h1>Fixes and improvements<a name="fixes-and-improvements" href="#fixes-and-improvements" class="heading-anchor">🔗</a></h1>
<p>This version includes 30 bug fixes and 16 general improvements. Let's dive into some of the highlights:</p>
<ul>
<li>
<p>Self Not Found and Pedro have been working tirelessly on improving the pasting of plain text in the application, and we're happy to say that there is now a brand new &quot;Paste as text&quot; option in the Edit menu.</p>
</li>
<li>
<p>For our amazing plugin developers out there, we have added a few new APIs and fixed a bug that was preventing certain plugins from starting. You can now get even more creative with your plugins!</p>
</li>
<li>
<p>Tao Klerks, has been hard at work fixing and improving the custom sort order of the note list. No more notes in the wrong position when you drop them! Plus, custom sort order is now synchronised too.</p>
</li>
<li>
<p>We've also fixed a few bugs with our Web Clipper, including an issue where certain pages wouldn't import their images. A new Web Clipper has also been released, so you can clip to your heart's content!</p>
</li>
</ul>
<p>Last but not least, we have modernised both the desktop and mobile application modules, just <a href="https://joplinapp.org/news/20221115-renovate/">as we previously announced</a>. Although these changes may not be visible to you, they required a lot of work! But the result is that our applications are now more stable and secure, and it will be easier to maintain them in the long run. We're using a tool called <a href="https://www.mend.io/free-developer-tools/renovate/">Renovate</a>, which will automatically propose package updates that we will review. In total, we've updated a whopping 633 packages so far!</p>
<h1>Android app is available in the Play Store<a name="android-app-is-available-in-the-play-store" href="#android-app-is-available-in-the-play-store" class="heading-anchor">🔗</a></h1>
<p>Our latest version, 2.10, is now back in the Play Store and ready for download! Although we had to skip 2.9 due to some of Google's requirements, we worked hard to ensure that our app complies with their standards, and we are excited to announce that we are back and better than ever! Our iOS version is also available, so you can continue to enjoy the app regardless of the platform you use.</p>
<h2>Biometrics support<a name="biometrics-support" href="#biometrics-support" class="heading-anchor">🔗</a></h2>
<p>To make your experience even more secure, our Android and iOS apps now support biometric unlock! With just a quick scan of your fingerprint or Face ID, you can unlock your app in no time. To enable this beta feature, just head over to the settings and click on &quot;Use biometrics to secure access to the app&quot;.</p>
<p>We've tested this feature thoroughly during prerelease, and have already fixed all known issues. However we still consider it as a beta feature for now, so if you run into any issues please let us know.</p>
<h2>Support for multiple profile<a name="support-for-multiple-profile" href="#support-for-multiple-profile" class="heading-anchor">🔗</a></h2>
<p>We're thrilled to announce that multiple profiles are now supported in our mobile app! To create a new profile, simply go to the Configuration screen and click on &quot;Manage profiles&quot; under the Tools section.</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20230508-biometrics-1.png" alt=""></p>
<p>From there, you can easily add or remove profiles as needed.</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20230508-biometrics-2.png" alt=""></p>
<p>Once multiple profiles are setup, you will see a new option in the sidebar to quickly toggle from one profile to another:</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20230508-biometrics-3.png" alt=""></p>
<p>Once you've set up multiple profiles, you can easily toggle between them using the new option in the sidebar. This feature is perfect for separating your personal and work notes into independent collections.</p>
<h2>Support for realtime search<a name="support-for-realtime-search" href="#support-for-realtime-search" class="heading-anchor">🔗</a></h2>
<p>Our mobile app now has an improved search function that performs text searches in real time! No more waiting for the search results to load, they'll appear instantly as you type.</p>
<h2>Improved filesystem sync performance<a name="improved-filesystem-sync-performance" href="#improved-filesystem-sync-performance" class="heading-anchor">🔗</a></h2>
<p>Thanks to the hard work of jd1378, the sync no longer freezes during filesystem synchronisation. We know how frustrating that can be, and we're thrilled to have solved this issue. Getting filesystem sync to work on Android is never easy due to the restrictions put in place by Google, especially since they frequently change, but we're committed to delivering the best possible experience for our Android users.</p>
]]></description><link>https://joplinapp.org/news/20230508-release-2-10/</link><guid isPermaLink="false">20230508-release-2-10</guid><pubDate>Mon, 08 May 2023 00:00:00 GMT</pubDate><twitter-text>What&apos;s new in Joplin 2.10</twitter-text></item><item><title><![CDATA[Joplin will participate in JdLL 2023!]]></title><description><![CDATA[<p>On 1 and 2 April 2023, we will have a stand for Joplin at the <a href="https://www.jdll.org/">Journées du Logiciel Libre</a> in Lyon, France. The JdLL has been taking place in Lyon for 24 years and is a popular open source conference in France. We had a stand in 2020 and 2021 but that was cancelled due to Covid, so this year is a first for Joplin!</p>
<p>Admission is free, so don't hesitate to come and meet us, exchange ideas and learn more about Joplin!</p> <p>Admission is free, so don't hesitate to come and meet us, exchange ideas and learn more about Joplin!</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20230202-jdll.jpg" alt="Joplin at JdLL 2023"></p> <p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20230202-jdll.jpg" alt="Joplin at JdLL 2023"></p>
]]></description><link>https://joplinapp.org/news/20230302-jdll-2023/</link><guid isPermaLink="false">20230302-jdll-2023</guid><pubDate>Thu, 02 Mar 2023 00:00:00 GMT</pubDate><twitter-text></twitter-text></item><item><title><![CDATA[Introducing the "GitHub Actions Raw Log Viewer" extension for Chrome]]></title><description><![CDATA[<p>If you've ever used GitHub Actions, you will find that they provide by default a nice coloured output for the log. It looks good and it's even interactive! (You can click to collapse/expand blocks of text) But unfortunately it doesn't scale to large workflows, like we have for Joplin - the log can freeze and it will take forever to search for something. Indeed searching is done in &quot;real time&quot;... which mostly means it will freeze for a minute or two for each letter you type in the search box. Not great.</p> ]]></description><link>https://joplinapp.org/news/20230302-jdll-2023/</link><guid isPermaLink="false">20230302-jdll-2023</guid><pubDate>Thu, 02 Mar 2023 00:00:00 GMT</pubDate><twitter-text></twitter-text></item><item><title><![CDATA[Introducing the "GitHub Actions Raw Log Viewer" extension for Chrome]]></title><description><![CDATA[<p>If you've ever used GitHub Actions, you will find that they provide by default a nice coloured output for the log. It looks good and it's even interactive! (You can click to collapse/expand blocks of text) But unfortunately it doesn't scale to large workflows, like we have for Joplin - the log can freeze and it will take forever to search for something. Indeed searching is done in &quot;real time&quot;... which mostly means it will freeze for a minute or two for each letter you type in the search box. Not great.</p>
@@ -313,4 +271,30 @@
<p>- <a href="https://community.letsencrypt.org/t/issues-with-electron-and-expired-root/160991">Issue with Electron and expired root</a> on Let's Encrypt</p> <p>- <a href="https://community.letsencrypt.org/t/issues-with-electron-and-expired-root/160991">Issue with Electron and expired root</a> on Let's Encrypt</p>
<p>- <a href="https://github.com/electron/electron/issues/31212">Let's Encrypt root CA isn't working properly</a> on Electron GitHub repository</p> <p>- <a href="https://github.com/electron/electron/issues/31212">Let's Encrypt root CA isn't working properly</a> on Electron GitHub repository</p>
<p><strong>Update:</strong> I have implemented a temporary fix on Joplin Cloud which should solve the issue for now. If you're still having some issues please let me know. An updated desktop app will be available later on with a more permanent fix.</p> <p><strong>Update:</strong> I have implemented a temporary fix on Joplin Cloud which should solve the issue for now. If you're still having some issues please let me know. An updated desktop app will be available later on with a more permanent fix.</p>
]]></description><link>https://joplinapp.org/news/20210930-163458/</link><guid isPermaLink="false">20210930-163458</guid><pubDate>Thu, 30 Sep 2021 16:34:58 GMT</pubDate><twitter-text></twitter-text></item></channel></rss> ]]></description><link>https://joplinapp.org/news/20210930-163458/</link><guid isPermaLink="false">20210930-163458</guid><pubDate>Thu, 30 Sep 2021 16:34:58 GMT</pubDate><twitter-text></twitter-text></item><item><title><![CDATA[Joplin 2.4 is available!]]></title><description><![CDATA[<p>Joplin 2.4 is now available on desktop, mobile and CLI. Here's what's new in this release:</p>
<h3>Sync Wizard Dialog<a name="sync-wizard-dialog" href="#sync-wizard-dialog" class="heading-anchor">🔗</a></h3>
<p>A new Sync Wizard Dialog has been added to simplify setting up sync on new clients.</p>
<p>The dialog shows the main sync targets, their differences, and makes it easy to choose one and start synchronising. This is mostly aimed at new users or those perhaps less technical. Those who are self hosting or using complex setups will still easily find what they need from a link on that dialog (or in Config &gt; Synchronisation like before).</p>
<p>Sync setup on mobile has been slightly improved too - now on a new client, instead of asking you to sync with Dropbox directly (which may not be what you want), it jumps to the Config &gt; Synchronisation section where you can select the sync target</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20210929-144036_0.png" alt=""></p>
<h3>Disable synchronisation<a name="disable-synchronisation" href="#disable-synchronisation" class="heading-anchor">🔗</a></h3>
<p>It's a small change but something that's been asked many time - it's now possible to disable synchronisation entirely by selecting &quot;None&quot; as a sync target. Previously that could be done in a hacky way, by selecting a non-configured sync target. Now it's clearer and easier to do.</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20210929-144036_1.png" alt=""></p>
<h3>Add back support for deprecated plugins<a name="add-back-support-for-deprecated-plugins" href="#add-back-support-for-deprecated-plugins" class="heading-anchor">🔗</a></h3>
<p>Recently some plugins stopped working because deprecated plugin APIs had been removed. It had been planned for a long time but I suspect the warnings weren't visible enough so plugin developers didn't act on them, and as a result many plugins stopped working.</p>
<p>This is now fixed in the latest version. A selected number of plugins will have access to these old deprecated APIs, which means they will start working again. This was mainly affecting ambrt's plugins such as &quot;Convert Text To New Note&quot; or the popular &quot;Embed Search&quot; plugin.</p>
<h3>Add support for recommended plugins<a name="add-support-for-recommended-plugins" href="#add-support-for-recommended-plugins" class="heading-anchor">🔗</a></h3>
<p>As mentioned in an earlier post, we now support <a href="https://www.patreon.com/posts/introducing-in-55618802">recommended plugins</a>. These recommended plugins appear on top when searching and are identified by a small crown.</p>
<p><img src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/news/20210929-144036_2.png" alt=""></p>
<h3>End to End Encryption improvements<a name="end-to-end-encryption-improvements" href="#end-to-end-encryption-improvements" class="heading-anchor">🔗</a></h3>
<p>Like most recent releases, v2.4 includes a few improvement to the End to End Encryption (E2EE) system. The goal is to make it easier to use, to make it more reliable and to support the future use case of sharing encrypted notebooks or notes.</p>
<p>One important change is the support for a master password. This single password will be responsible to encrypt various keys, including some that will be automatically generated. Thanks to this, it won't be necessary to ask to enter a new password every time a key needs to be encrypted, since the master password can be used. It will also be easier to manage since you'll only have one password to remember instead of a different one for each notebook you might have shared.</p>
<p>Finally, it's now possible to disable a master key. What it means is that it will no longer show up in the list of master keys, and will also no longer generate a warning asking you to enter the password. In some case you might have forgotten it and no longer need it key, so you can now disable it.</p>
<h3>Custom CSS<a name="custom-css" href="#custom-css" class="heading-anchor">🔗</a></h3>
<p>This version also introduces a few internal change to better support custom CSS. In particular the colours now come from a CSS file, which could potentially be overridden, and new UI elements are styled using stylesheets, which likewise could be overridden.</p>
<p>Those are just first steps, but eventually these changes will make it easier to style the UI and create new themes.</p>
<h3>Bug fixes<a name="bug-fixes" href="#bug-fixes" class="heading-anchor">🔗</a></h3>
<p>This release also includes about 30 various bug fixes and improvements.</p>
<p>A notable one is a fix for GotoAnything, which recently wasn't working on first try.</p>
<p>The plugin screen has also been improved so that search works even when GitHub is down or blocked, as it is in China in particular.</p>
]]></description><link>https://joplinapp.org/news/20210929-144036/</link><guid isPermaLink="false">20210929-144036</guid><pubDate>Wed, 29 Sep 2021 14:40:36 GMT</pubDate><twitter-text></twitter-text></item></channel></rss>

View File

@@ -114,20 +114,6 @@ elif [[ $ARCHITECTURE =~ .*i386.*|.*i686.* ]] ; then
exit 1 exit 1
fi fi
#-----------------------------------------------------
print "Checking dependencies..."
## Check if libfuse2 is present.
if [[ $(command -v ldconfig) ]]; then
LIBFUSE=$(ldconfig -p | grep "libfuse.so.2" || echo '')
else
LIBFUSE=$(find /lib /usr/lib /lib64 /usr/lib64 /usr/local/lib -name "libfuse.so.2" 2>/dev/null | grep "libfuse.so.2" || echo '')
fi
if [[ $LIBFUSE == "" ]] ; then
print "${COLOR_RED}Error: Can't get libfuse2 on system, please install libfuse2${COLOR_RESET}"
print "See https://joplinapp.org/faq/#desktop-application-will-not-launch-on-linux and https://github.com/AppImage/AppImageKit/wiki/FUSE for more information"
exit 1
fi
#----------------------------------------------------- #-----------------------------------------------------
# Download Joplin # Download Joplin
#----------------------------------------------------- #-----------------------------------------------------
@@ -148,16 +134,10 @@ else
print "The latest version is ${RELEASE_VERSION}, but you have ${CURRENT_VERSION:-no version} installed." print "The latest version is ${RELEASE_VERSION}, but you have ${CURRENT_VERSION:-no version} installed."
fi fi
# Check if it's an update or a new install
DOWNLOAD_TYPE="New"
if [[ -f ~/.joplin/Joplin.AppImage ]]; then
DOWNLOAD_TYPE="Update"
fi
#----------------------------------------------------- #-----------------------------------------------------
print 'Downloading Joplin...' print 'Downloading Joplin...'
TEMP_DIR=$(mktemp -d) TEMP_DIR=$(mktemp -d)
wget -O "${TEMP_DIR}/Joplin.AppImage" "https://objects.joplinusercontent.com/v${RELEASE_VERSION}/Joplin-${RELEASE_VERSION}.AppImage?source=LinuxInstallScript&type=$DOWNLOAD_TYPE" wget -O "${TEMP_DIR}/Joplin.AppImage" "https://github.com/laurent22/joplin/releases/download/v${RELEASE_VERSION}/Joplin-${RELEASE_VERSION}.AppImage"
wget -O "${TEMP_DIR}/joplin.png" https://joplinapp.org/images/Icon512.png wget -O "${TEMP_DIR}/joplin.png" https://joplinapp.org/images/Icon512.png
#----------------------------------------------------- #-----------------------------------------------------

View File

@@ -1,5 +1,5 @@
<!-- DONATELINKS --> <!-- DONATELINKS -->
[![Donate using PayPal](https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/badges/Donate-PayPal-green.svg)](https://www.paypal.com/donate/?business=E8JMYD2LQ8MMA&no_recurring=0&item_name=I+rely+on+donations+to+maintain+and+improve+the+Joplin+open+source+project.+Thank+you+for+your+help+-+it+makes+a+difference%21&currency_code=EUR) [![Sponsor on GitHub](https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/badges/GitHub-Badge.svg)](https://github.com/sponsors/laurent22/) [![Become a patron](https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/badges/Patreon-Badge.svg)](https://www.patreon.com/joplin) [![Donate using IBAN](https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/badges/Donate-IBAN.svg)](https://joplinapp.org/donate/#donations) [![Donate using PayPal](https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/badges/Donate-PayPal-green.svg)](https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=E8JMYD2LQ8MMA&lc=GB&item_name=Joplin+Development&currency_code=EUR&bn=PP%2dDonationsBF%3abtn_donateCC_LG%2egif%3aNonHosted) [![Sponsor on GitHub](https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/badges/GitHub-Badge.svg)](https://github.com/sponsors/laurent22/) [![Become a patron](https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/badges/Patreon-Badge.svg)](https://www.patreon.com/joplin) [![Donate using IBAN](https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/badges/Donate-IBAN.svg)](https://joplinapp.org/donate/#donations)
<!-- DONATELINKS --> <!-- DONATELINKS -->
<img width="64" src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/LinuxIcons/256x256.png" align="left" /> **Joplin** is a free, open source note taking and to-do application, which can handle a large number of notes organised into notebooks. The notes are searchable, can be copied, tagged and modified either from the applications directly or from your own text editor. The notes are in [Markdown format](#markdown). <img width="64" src="https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/LinuxIcons/256x256.png" align="left" /> **Joplin** is a free, open source note taking and to-do application, which can handle a large number of notes organised into notebooks. The notes are searchable, can be copied, tagged and modified either from the applications directly or from your own text editor. The notes are in [Markdown format](#markdown).
@@ -22,23 +22,21 @@ Three types of applications are available: for **desktop** (Windows, macOS and L
Operating System | Download Operating System | Download
---|--- ---|---
Windows (32 and 64-bit) | <a href='https://objects.joplinusercontent.com/v2.10.19/Joplin-Setup-2.10.19.exe?source=JoplinWebsite&type=New'><img alt='Get it on Windows' width="134px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeWindows.png'/></a> Windows (32 and 64-bit) | <a href='https://github.com/laurent22/joplin/releases/download/v2.9.17/Joplin-Setup-2.9.17.exe'><img alt='Get it on Windows' width="134px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeWindows.png'/></a>
macOS | <a href='https://objects.joplinusercontent.com/v2.10.19/Joplin-2.10.19.dmg?source=JoplinWebsite&type=New'><img alt='Get it on macOS' width="134px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeMacOS.png'/></a> macOS | <a href='https://github.com/laurent22/joplin/releases/download/v2.9.17/Joplin-2.9.17.dmg'><img alt='Get it on macOS' width="134px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeMacOS.png'/></a>
Linux | <a href='https://objects.joplinusercontent.com/v2.10.19/Joplin-2.10.19.AppImage?source=JoplinWebsite&type=New'><img alt='Get it on Linux' width="134px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeLinux.png'/></a> Linux | <a href='https://github.com/laurent22/joplin/releases/download/v2.9.17/Joplin-2.9.17.AppImage'><img alt='Get it on Linux' width="134px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeLinux.png'/></a>
**On Windows**, you may also use the <a href='https://objects.joplinusercontent.com/v2.10.19/JoplinPortable.exe?source=JoplinWebsite&type=New'>Portable version</a>. The [portable application](https://en.wikipedia.org/wiki/Portable_application) allows installing the software on a portable device such as a USB key. Simply copy the file JoplinPortable.exe in any directory on that USB key ; the application will then create a directory called "JoplinProfile" next to the executable file. **On Windows**, you may also use the <a href='https://github.com/laurent22/joplin/releases/download/v2.9.17/JoplinPortable.exe'>Portable version</a>. The [portable application](https://en.wikipedia.org/wiki/Portable_application) allows installing the software on a portable device such as a USB key. Simply copy the file JoplinPortable.exe in any directory on that USB key ; the application will then create a directory called "JoplinProfile" next to the executable file.
**On Linux**, the recommended way is to use the following installation script as it will handle the desktop icon too: **On Linux**, the recommended way is to use the following installation script as it will handle the desktop icon too:
<pre><code style="word-break: break-all">wget -O - https://raw.githubusercontent.com/laurent22/joplin/dev/Joplin_install_and_update.sh | bash</code></pre> <pre><code style="word-break: break-all">wget -O - https://raw.githubusercontent.com/laurent22/joplin/dev/Joplin_install_and_update.sh | bash</code></pre>
The install and update script supports the [following flags](https://github.com/laurent22/joplin/blob/dev/Joplin_install_and_update.sh#L50) (around line 50 at the time of this writing).
## Mobile applications ## Mobile applications
Operating System | Download | Alt. Download Operating System | Download | Alt. Download
---|---|--- ---|---|---
Android | <a href='https://play.google.com/store/apps/details?id=net.cozic.joplin&utm_source=GitHub&utm_campaign=README&pcampaignid=MKT-Other-global-all-co-prtnr-py-PartBadge-Mar2515-1'><img alt='Get it on Google Play' height="40px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeAndroid.png'/></a> | or download the APK file: [64-bit](https://objects.joplinusercontent.com/android-v2.9.8/joplin-v2.9.8.apk?source=JoplinWebsite&type=New) [32-bit](https://objects.joplinusercontent.com/android-v2.9.8/joplin-v2.9.8-32bit.apk?source=JoplinWebsite&type=New) Android | <a href='https://play.google.com/store/apps/details?id=net.cozic.joplin&utm_source=GitHub&utm_campaign=README&pcampaignid=MKT-Other-global-all-co-prtnr-py-PartBadge-Mar2515-1'><img alt='Get it on Google Play' height="40px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeAndroid.png'/></a> | or download the APK file: [64-bit](https://github.com/laurent22/joplin-android/releases/download/android-v2.9.8/joplin-v2.9.8.apk) [32-bit](https://github.com/laurent22/joplin-android/releases/download/android-v2.9.8/joplin-v2.9.8-32bit.apk)
iOS | <a href='https://itunes.apple.com/us/app/joplin/id1315599797'><img alt='Get it on the App Store' height="40px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeIOS.png'/></a> | - iOS | <a href='https://itunes.apple.com/us/app/joplin/id1315599797'><img alt='Get it on the App Store' height="40px" src='https://raw.githubusercontent.com/laurent22/joplin/dev/Assets/WebsiteAssets/images/BadgeIOS.png'/></a> | -
## Terminal application ## Terminal application
@@ -66,7 +64,7 @@ A community maintained list of these distributions can be found here: [Unofficia
# Sponsors # Sponsors
<!-- SPONSORS-ORG --> <!-- SPONSORS-ORG -->
<a href="https://seirei.ne.jp"><img title="Serei Network" width="256" src="https://joplinapp.org/images/sponsors/SeireiNetwork.png"/></a> <a href="https://www.hosting.de/nextcloud/?mtm_campaign=managed-nextcloud&amp;mtm_kwd=joplinapp&amp;mtm_source=joplinapp-webseite&amp;mtm_medium=banner"><img title="Hosting.de" width="256" src="https://joplinapp.org/images/sponsors/HostingDe.png"/></a> <a href="https://residence-greece.com/"><img title="Greece Golden Visa" width="256" src="https://joplinapp.org/images/sponsors/ResidenceGreece.jpg"/></a> <a href="https://grundstueckspreise.info/"><img title="SP Software GmbH" width="256" src="https://joplinapp.org/images/sponsors/Grundstueckspreise.png"/></a> <a href="https://tranio.com/spain/"><img title="Property for sale in Spain" width="256" src="https://joplinapp.org/images/sponsors/TranioOverseasProperty.jpg"/></a> <a href="https://seirei.ne.jp"><img title="Serei Network" width="256" src="https://joplinapp.org/images/sponsors/SeireiNetwork.png"/></a> <a href="https://www.hosting.de/nextcloud/?mtm_campaign=managed-nextcloud&amp;mtm_kwd=joplinapp&amp;mtm_source=joplinapp-webseite&amp;mtm_medium=banner"><img title="Hosting.de" width="256" src="https://joplinapp.org/images/sponsors/HostingDe.png"/></a> <a href="https://residence-greece.com/"><img title="Greece Golden Visa" width="256" src="https://joplinapp.org/images/sponsors/ResidenceGreece.jpg"/></a> <a href="https://grundstueckspreise.info/"><img title="SP Software GmbH" width="256" src="https://joplinapp.org/images/sponsors/Grundstueckspreise.png"/></a>
<!-- SPONSORS-ORG --> <!-- SPONSORS-ORG -->
* * * * * *
@@ -127,11 +125,10 @@ A community maintained list of these distributions can be found here: [Unofficia
- [How to build the apps](https://github.com/laurent22/joplin/blob/dev/BUILD.md) - [How to build the apps](https://github.com/laurent22/joplin/blob/dev/BUILD.md)
- [Writing a technical spec](https://github.com/laurent22/joplin/blob/dev/readme/technical_spec.md) - [Writing a technical spec](https://github.com/laurent22/joplin/blob/dev/readme/technical_spec.md)
- [Desktop application styling](https://github.com/laurent22/joplin/blob/dev/readme/spec/desktop_styling.md) - [Desktop application styling](https://github.com/laurent22/joplin/blob/dev/readme/spec/desktop_styling.md)
- [Note history spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/history.md) - [Note History spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/history.md)
- [Synchronisation spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/sync.md) - [Synchronisation spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/sync.md)
- [Sync Lock spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/sync_lock.md) - [Sync Lock spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/sync_lock.md)
- [Synchronous Scroll spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/sync_scroll.md) - [Synchronous Scroll spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/sync_scroll.md)
- [Overall Architecture spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/architecture.md)
- [Plugin Architecture spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/plugins.md) - [Plugin Architecture spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/plugins.md)
- [Search Sorting spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/search_sorting.md) - [Search Sorting spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/search_sorting.md)
- [E2EE: Technical spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/e2ee.md) - [E2EE: Technical spec](https://github.com/laurent22/joplin/blob/dev/readme/spec/e2ee.md)
@@ -532,47 +529,47 @@ Current translations:
<!-- LOCALE-TABLE-AUTO-GENERATED --> <!-- LOCALE-TABLE-AUTO-GENERATED -->
&nbsp; | Language | Po File | Last translator | Percent done &nbsp; | Language | Po File | Last translator | Percent done
---|---|---|---|--- ---|---|---|---|---
<img src="https://joplinapp.org/images/flags/country-4x3/arableague.png" width="16px"/> | Arabic | [ar](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ar.po) | [Whaell O](mailto:Whaell@protonmail.com) | 79% <img src="https://joplinapp.org/images/flags/country-4x3/arableague.png" width="16px"/> | Arabic | [ar](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ar.po) | [Whaell O](mailto:Whaell@protonmail.com) | 80%
<img src="https://joplinapp.org/images/flags/es/basque_country.png" width="16px"/> | Basque | [eu](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/eu.po) | juan.abasolo@ehu.eus | 22% <img src="https://joplinapp.org/images/flags/es/basque_country.png" width="16px"/> | Basque | [eu](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/eu.po) | juan.abasolo@ehu.eus | 23%
<img src="https://joplinapp.org/images/flags/country-4x3/ba.png" width="16px"/> | Bosnian (Bosna i Hercegovina) | [bs_BA](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/bs_BA.po) | [Derviš T.](mailto:dervis.t@pm.me) | 57% <img src="https://joplinapp.org/images/flags/country-4x3/ba.png" width="16px"/> | Bosnian (Bosna i Hercegovina) | [bs_BA](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/bs_BA.po) | [Derviš T.](mailto:dervis.t@pm.me) | 58%
<img src="https://joplinapp.org/images/flags/country-4x3/bg.png" width="16px"/> | Bulgarian (България) | [bg_BG](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/bg_BG.po) | | 45% <img src="https://joplinapp.org/images/flags/country-4x3/bg.png" width="16px"/> | Bulgarian (България) | [bg_BG](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/bg_BG.po) | | 45%
<img src="https://joplinapp.org/images/flags/es/catalonia.png" width="16px"/> | Catalan | [ca](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ca.po) | [Xavi Ivars](mailto:xavi.ivars@gmail.com) | 88% <img src="https://joplinapp.org/images/flags/es/catalonia.png" width="16px"/> | Catalan | [ca](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ca.po) | [Xavi Ivars](mailto:xavi.ivars@gmail.com) | 89%
<img src="https://joplinapp.org/images/flags/country-4x3/hr.png" width="16px"/> | Croatian (Hrvatska) | [hr_HR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/hr_HR.po) | [Milo Ivir](mailto:mail@milotype.de) | 99% <img src="https://joplinapp.org/images/flags/country-4x3/hr.png" width="16px"/> | Croatian (Hrvatska) | [hr_HR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/hr_HR.po) | [Milo Ivir](mailto:mail@milotype.de) | 89%
<img src="https://joplinapp.org/images/flags/country-4x3/cz.png" width="16px"/> | Czech (Česká republika) | [cs_CZ](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/cs_CZ.po) | Fejby | 98% <img src="https://joplinapp.org/images/flags/country-4x3/cz.png" width="16px"/> | Czech (Česká republika) | [cs_CZ](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/cs_CZ.po) | [Michal Stanke](mailto:michal@stanke.cz) | 77%
<img src="https://joplinapp.org/images/flags/country-4x3/dk.png" width="16px"/> | Dansk (Danmark) | [da_DK](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/da_DK.po) | ERYpTION | 98% <img src="https://joplinapp.org/images/flags/country-4x3/dk.png" width="16px"/> | Dansk (Danmark) | [da_DK](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/da_DK.po) | ERYpTION | 99%
<img src="https://joplinapp.org/images/flags/country-4x3/de.png" width="16px"/> | Deutsch (Deutschland) | [de_DE](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/de_DE.po) | [MrKanister](mailto:pueblos_spatulas@aleeas.com) | 98% <img src="https://joplinapp.org/images/flags/country-4x3/de.png" width="16px"/> | Deutsch (Deutschland) | [de_DE](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/de_DE.po) | [MrKanister](mailto:pueblos_spatulas@aleeas.com) | 99%
<img src="https://joplinapp.org/images/flags/country-4x3/ee.png" width="16px"/> | Eesti Keel (Eesti) | [et_EE](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/et_EE.po) | | 44% <img src="https://joplinapp.org/images/flags/country-4x3/ee.png" width="16px"/> | Eesti Keel (Eesti) | [et_EE](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/et_EE.po) | | 44%
<img src="https://joplinapp.org/images/flags/country-4x3/gb.png" width="16px"/> | English (United Kingdom) | [en_GB](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/en_GB.po) | | 100% <img src="https://joplinapp.org/images/flags/country-4x3/gb.png" width="16px"/> | English (United Kingdom) | [en_GB](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/en_GB.po) | | 100%
<img src="https://joplinapp.org/images/flags/country-4x3/us.png" width="16px"/> | English (United States of America) | [en_US](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/en_US.po) | | 100% <img src="https://joplinapp.org/images/flags/country-4x3/us.png" width="16px"/> | English (United States of America) | [en_US](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/en_US.po) | | 100%
<img src="https://joplinapp.org/images/flags/country-4x3/es.png" width="16px"/> | Español (España) | [es_ES](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/es_ES.po) | [Francisco Villaverde](mailto:teko.gr@gmail.com) | 97% <img src="https://joplinapp.org/images/flags/country-4x3/es.png" width="16px"/> | Español (España) | [es_ES](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/es_ES.po) | [Francisco Villaverde](mailto:teko.gr@gmail.com) | 99%
<img src="https://joplinapp.org/images/flags/esperanto.png" width="16px"/> | Esperanto | [eo](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/eo.po) | Marton Paulo | 25% <img src="https://joplinapp.org/images/flags/esperanto.png" width="16px"/> | Esperanto | [eo](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/eo.po) | Marton Paulo | 25%
<img src="https://joplinapp.org/images/flags/country-4x3/fi.png" width="16px"/> | Finnish (Suomi) | [fi_FI](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/fi_FI.po) | mrkaato0 | 98% <img src="https://joplinapp.org/images/flags/country-4x3/fi.png" width="16px"/> | Finnish (Suomi) | [fi_FI](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/fi_FI.po) | mrkaato0 | 95%
<img src="https://joplinapp.org/images/flags/country-4x3/fr.png" width="16px"/> | Français (France) | [fr_FR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/fr_FR.po) | Laurent Cozic | 100% <img src="https://joplinapp.org/images/flags/country-4x3/fr.png" width="16px"/> | Français (France) | [fr_FR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/fr_FR.po) | Laurent Cozic | 99%
<img src="https://joplinapp.org/images/flags/es/galicia.png" width="16px"/> | Galician (España) | [gl_ES](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/gl_ES.po) | [Marcos Lans](mailto:marcoslansgarza@gmail.com) | 29% <img src="https://joplinapp.org/images/flags/es/galicia.png" width="16px"/> | Galician (España) | [gl_ES](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/gl_ES.po) | [Marcos Lans](mailto:marcoslansgarza@gmail.com) | 29%
<img src="https://joplinapp.org/images/flags/country-4x3/id.png" width="16px"/> | Indonesian (Indonesia) | [id_ID](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/id_ID.po) | [Wisnu Adi Santoso](mailto:waditos@gmail.com) | 88% <img src="https://joplinapp.org/images/flags/country-4x3/id.png" width="16px"/> | Indonesian (Indonesia) | [id_ID](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/id_ID.po) | [Wisnu Adi Santoso](mailto:waditos@gmail.com) | 89%
<img src="https://joplinapp.org/images/flags/country-4x3/it.png" width="16px"/> | Italiano (Italia) | [it_IT](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/it_IT.po) | [Manuel Tassi](mailto:mannivuwiki@gmail.com) | 80% <img src="https://joplinapp.org/images/flags/country-4x3/it.png" width="16px"/> | Italiano (Italia) | [it_IT](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/it_IT.po) | [Manuel Tassi](mailto:mannivuwiki@gmail.com) | 81%
<img src="https://joplinapp.org/images/flags/country-4x3/hu.png" width="16px"/> | Magyar (Magyarország) | [hu_HU](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/hu_HU.po) | [Magyari Balázs](mailto:balmag@gmail.com) | 77% <img src="https://joplinapp.org/images/flags/country-4x3/hu.png" width="16px"/> | Magyar (Magyarország) | [hu_HU](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/hu_HU.po) | [Magyari Balázs](mailto:balmag@gmail.com) | 78%
<img src="https://joplinapp.org/images/flags/country-4x3/be.png" width="16px"/> | Nederlands (België, Belgique, Belgien) | [nl_BE](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/nl_BE.po) | | 78% <img src="https://joplinapp.org/images/flags/country-4x3/be.png" width="16px"/> | Nederlands (België, Belgique, Belgien) | [nl_BE](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/nl_BE.po) | | 79%
<img src="https://joplinapp.org/images/flags/country-4x3/nl.png" width="16px"/> | Nederlands (Nederland) | [nl_NL](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/nl_NL.po) | [MHolkamp](mailto:mholkamp@users.noreply.github.com) | 87% <img src="https://joplinapp.org/images/flags/country-4x3/nl.png" width="16px"/> | Nederlands (Nederland) | [nl_NL](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/nl_NL.po) | [MHolkamp](mailto:mholkamp@users.noreply.github.com) | 88%
<img src="https://joplinapp.org/images/flags/country-4x3/no.png" width="16px"/> | Norwegian (Norge, Noreg) | [nb_NO](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/nb_NO.po) | [Mats Estensen](mailto:code@mxe.no) | 87% <img src="https://joplinapp.org/images/flags/country-4x3/no.png" width="16px"/> | Norwegian (Norge, Noreg) | [nb_NO](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/nb_NO.po) | [Mats Estensen](mailto:code@mxe.no) | 88%
<img src="https://joplinapp.org/images/flags/country-4x3/ir.png" width="16px"/> | Persian | [fa](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/fa.po) | [Kourosh Firoozbakht](mailto:kourox@protonmail.com) | 55% <img src="https://joplinapp.org/images/flags/country-4x3/ir.png" width="16px"/> | Persian | [fa](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/fa.po) | [Kourosh Firoozbakht](mailto:kourox@protonmail.com) | 55%
<img src="https://joplinapp.org/images/flags/country-4x3/pl.png" width="16px"/> | Polski (Polska) | [pl_PL](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/pl_PL.po) | [X3NO](mailto:X3NO@disroot.org) | 90% <img src="https://joplinapp.org/images/flags/country-4x3/pl.png" width="16px"/> | Polski (Polska) | [pl_PL](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/pl_PL.po) | [X3NO](mailto:X3NO@disroot.org) | 91%
<img src="https://joplinapp.org/images/flags/country-4x3/br.png" width="16px"/> | Português (Brasil) | [pt_BR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/pt_BR.po) | [Douglas Leão](mailto:djlsplays@gmail.com) | 87% <img src="https://joplinapp.org/images/flags/country-4x3/br.png" width="16px"/> | Português (Brasil) | [pt_BR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/pt_BR.po) | [Renato Nunes Bastos](mailto:rnbastos@gmail.com) | 88%
<img src="https://joplinapp.org/images/flags/country-4x3/pt.png" width="16px"/> | Português (Portugal) | [pt_PT](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/pt_PT.po) | [Diogo Caveiro](mailto:dcaveiro@yahoo.com) | 72% <img src="https://joplinapp.org/images/flags/country-4x3/pt.png" width="16px"/> | Português (Portugal) | [pt_PT](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/pt_PT.po) | [Diogo Caveiro](mailto:dcaveiro@yahoo.com) | 73%
<img src="https://joplinapp.org/images/flags/country-4x3/ro.png" width="16px"/> | Română | [ro](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ro.po) | [Cristi Duluta](mailto:cristi.duluta@gmail.com) | 50% <img src="https://joplinapp.org/images/flags/country-4x3/ro.png" width="16px"/> | Română | [ro](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ro.po) | [Cristi Duluta](mailto:cristi.duluta@gmail.com) | 51%
<img src="https://joplinapp.org/images/flags/country-4x3/si.png" width="16px"/> | Slovenian (Slovenija) | [sl_SI](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/sl_SI.po) | [Martin Korelič](mailto:martin.korelic@protonmail.com) | 80% <img src="https://joplinapp.org/images/flags/country-4x3/si.png" width="16px"/> | Slovenian (Slovenija) | [sl_SI](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/sl_SI.po) | [Martin Korelič](mailto:martin.korelic@protonmail.com) | 80%
<img src="https://joplinapp.org/images/flags/country-4x3/se.png" width="16px"/> | Svenska | [sv](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/sv.po) | [Jonatan Nyberg](mailto:jonatan@autistici.org) | 99% <img src="https://joplinapp.org/images/flags/country-4x3/se.png" width="16px"/> | Svenska | [sv](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/sv.po) | [Jonatan Nyberg](mailto:jonatan@autistici.org) | 96%
<img src="https://joplinapp.org/images/flags/country-4x3/th.png" width="16px"/> | Thai (ประเทศไทย) | [th_TH](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/th_TH.po) | | 36% <img src="https://joplinapp.org/images/flags/country-4x3/th.png" width="16px"/> | Thai (ประเทศไทย) | [th_TH](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/th_TH.po) | | 36%
<img src="https://joplinapp.org/images/flags/country-4x3/vn.png" width="16px"/> | Tiếng Việt | [vi](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/vi.po) | | 77% <img src="https://joplinapp.org/images/flags/country-4x3/vn.png" width="16px"/> | Tiếng Việt | [vi](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/vi.po) | | 78%
<img src="https://joplinapp.org/images/flags/country-4x3/tr.png" width="16px"/> | Türkçe (Türkiye) | [tr_TR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/tr_TR.po) | [Arda Kılıçdağı](mailto:arda@kilicdagi.com) | 99% <img src="https://joplinapp.org/images/flags/country-4x3/tr.png" width="16px"/> | Türkçe (Türkiye) | [tr_TR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/tr_TR.po) | [Arda Kılıçdağı](mailto:arda@kilicdagi.com) | 89%
<img src="https://joplinapp.org/images/flags/country-4x3/ua.png" width="16px"/> | Ukrainian (Україна) | [uk_UA](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/uk_UA.po) | [Vyacheslav Andreykiv](mailto:vandreykiv@gmail.com) | 72% <img src="https://joplinapp.org/images/flags/country-4x3/ua.png" width="16px"/> | Ukrainian (Україна) | [uk_UA](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/uk_UA.po) | [Vyacheslav Andreykiv](mailto:vandreykiv@gmail.com) | 72%
<img src="https://joplinapp.org/images/flags/country-4x3/gr.png" width="16px"/> | Ελληνικά (Ελλάδα) | [el_GR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/el_GR.po) | [Harris Arvanitis](mailto:xaris@tuta.io) | 87% <img src="https://joplinapp.org/images/flags/country-4x3/gr.png" width="16px"/> | Ελληνικά (Ελλάδα) | [el_GR](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/el_GR.po) | [Harris Arvanitis](mailto:xaris@tuta.io) | 88%
<img src="https://joplinapp.org/images/flags/country-4x3/ru.png" width="16px"/> | Русский (Россия) | [ru_RU](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ru_RU.po) | [Dmitriy K](mailto:dmitry@atsip.ru) | 99% <img src="https://joplinapp.org/images/flags/country-4x3/ru.png" width="16px"/> | Русский (Россия) | [ru_RU](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ru_RU.po) | [Dmitriy Q](mailto:krotesk@mail.ru) | 99%
<img src="https://joplinapp.org/images/flags/country-4x3/rs.png" width="16px"/> | српски језик (Србија) | [sr_RS](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/sr_RS.po) | | 64% <img src="https://joplinapp.org/images/flags/country-4x3/rs.png" width="16px"/> | српски језик (Србија) | [sr_RS](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/sr_RS.po) | | 65%
<img src="https://joplinapp.org/images/flags/country-4x3/cn.png" width="16px"/> | 中文 (简体) | [zh_CN](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/zh_CN.po) | [wh201906](mailto:wh201906@yandex.com) | 96% <img src="https://joplinapp.org/images/flags/country-4x3/cn.png" width="16px"/> | 中文 (简体) | [zh_CN](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/zh_CN.po) | [wh201906](mailto:wh201906@yandex.com) | 97%
<img src="https://joplinapp.org/images/flags/country-4x3/tw.png" width="16px"/> | 中文 (繁體) | [zh_TW](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/zh_TW.po) | [Kevin Hsu](mailto:kevin.hsu.hws@gmail.com) | 88% <img src="https://joplinapp.org/images/flags/country-4x3/tw.png" width="16px"/> | 中文 (繁體) | [zh_TW](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/zh_TW.po) | [Kevin Hsu](mailto:kevin.hsu.hws@gmail.com) | 89%
<img src="https://joplinapp.org/images/flags/country-4x3/jp.png" width="16px"/> | 日本語 (日本) | [ja_JP](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ja_JP.po) | [genneko](mailto:genneko217@gmail.com) | 88% <img src="https://joplinapp.org/images/flags/country-4x3/jp.png" width="16px"/> | 日本語 (日本) | [ja_JP](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ja_JP.po) | [genneko](mailto:genneko217@gmail.com) | 89%
<img src="https://joplinapp.org/images/flags/country-4x3/kr.png" width="16px"/> | 한국어 | [ko](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ko.po) | [Ji-Hyeon Gim](mailto:potatogim@potatogim.net) | 88% <img src="https://joplinapp.org/images/flags/country-4x3/kr.png" width="16px"/> | 한국어 | [ko](https://github.com/laurent22/joplin/blob/dev/packages/tools/locales/ko.po) | [Ji-Hyeon Gim](mailto:potatogim@potatogim.net) | 89%
<!-- LOCALE-TABLE-AUTO-GENERATED --> <!-- LOCALE-TABLE-AUTO-GENERATED -->
# Contributors # Contributors

View File

@@ -107,10 +107,7 @@
".eslintignore": true, ".eslintignore": true,
".gitignore": true, ".gitignore": true,
".vscode/*": true, ".vscode/*": true,
".yarn/cache": true, ".yarn": true,
".yarn/install-state.gz": true,
".yarn/plugins": true,
".yarn/releases": true,
"*.sublime-workspace": true, "*.sublime-workspace": true,
"**/_mydocs": true, "**/_mydocs": true,
"**/_mydocs/EnexSamples/*.enex": true, "**/_mydocs/EnexSamples/*.enex": true,

View File

@@ -66,36 +66,36 @@
"devDependencies": { "devDependencies": {
"@joplin/utils": "~2.11", "@joplin/utils": "~2.11",
"@seiyab/eslint-plugin-react-hooks": "4.5.1-beta.0", "@seiyab/eslint-plugin-react-hooks": "4.5.1-beta.0",
"@typescript-eslint/eslint-plugin": "5.59.0", "@typescript-eslint/eslint-plugin": "5.48.2",
"@typescript-eslint/parser": "5.59.0", "@typescript-eslint/parser": "5.48.2",
"cspell": "5.21.2", "cspell": "5.21.2",
"eslint": "8.39.0", "eslint": "8.31.0",
"eslint-interactive": "10.7.0", "eslint-interactive": "10.3.0",
"eslint-plugin-import": "2.27.5", "eslint-plugin-import": "2.27.4",
"eslint-plugin-jest": "27.2.1", "eslint-plugin-jest": "27.2.1",
"eslint-plugin-promise": "6.1.1", "eslint-plugin-promise": "6.1.1",
"eslint-plugin-react": "7.32.2", "eslint-plugin-react": "7.32.0",
"execa": "5.1.1", "execa": "5.1.1",
"fs-extra": "11.1.1", "fs-extra": "11.1.1",
"glob": "8.1.0", "glob": "8.1.0",
"gulp": "4.0.2", "gulp": "4.0.2",
"husky": "3.1.0", "husky": "3.1.0",
"lerna": "3.22.1", "lerna": "3.22.1",
"lint-staged": "13.2.2", "lint-staged": "13.2.0",
"madge": "6.0.0", "madge": "6.0.0",
"npm-package-json-lint": "6.4.0", "npm-package-json-lint": "6.4.0",
"typedoc": "0.17.8", "typedoc": "0.17.8",
"typescript": "5.0.2" "typescript": "4.9.4"
}, },
"dependencies": { "dependencies": {
"@types/fs-extra": "11.0.1", "@types/fs-extra": "9.0.13",
"http-server": "14.1.1", "http-server": "14.1.1",
"node-gyp": "9.3.1", "node-gyp": "9.3.1",
"nodemon": "2.0.22" "nodemon": "2.0.22"
}, },
"packageManager": "yarn@3.5.0", "packageManager": "yarn@3.3.1",
"resolutions": { "resolutions": {
"react-native-camera@4.2.1": "patch:react-native-camera@npm%3A4.2.1#./.yarn/patches/react-native-camera-npm-4.2.1-24b2600a7e.patch", "react-native-camera@4.2.1": "patch:react-native-camera@npm%3A4.2.1#./.yarn/patches/react-native-camera-npm-4.2.1-24b2600a7e.patch",
"react-native-vosk@0.1.12": "patch:react-native-vosk@npm%3A0.1.12#./.yarn/patches/react-native-vosk-npm-0.1.12-76b1caaae8.patch" "rn-fetch-blob@0.12.0": "patch:rn-fetch-blob@npm%3A0.12.0#./.yarn/patches/rn-fetch-blob-npm-0.12.0-cf02e3c544.patch"
} }
} }

View File

@@ -302,7 +302,7 @@ class AppGui {
const output = []; const output = [];
for (let i = 0; i < keymap.length; i++) { for (let i = 0; i < keymap.length; i++) {
const item = { ...keymap[i] }; const item = Object.assign({}, keymap[i]);
if (!item.command) throw new Error(`Missing command for keymap item: ${JSON.stringify(item)}`); if (!item.command) throw new Error(`Missing command for keymap item: ${JSON.stringify(item)}`);
@@ -427,7 +427,7 @@ class AppGui {
async handleModelAction(action) { async handleModelAction(action) {
this.logger().info('Action:', action); this.logger().info('Action:', action);
const state = { ...defaultState }; const state = Object.assign({}, defaultState);
state.notes = this.widget('noteList').items; state.notes = this.widget('noteList').items;
const newState = reducer(state, action); const newState = reducer(state, action);

View File

@@ -192,7 +192,7 @@ class Application extends BaseApplication {
let output = await this.cache_.getItem('metadata'); let output = await this.cache_.getItem('metadata');
if (output) { if (output) {
this.commandMetadata_ = output; this.commandMetadata_ = output;
return { ...this.commandMetadata_ }; return Object.assign({}, this.commandMetadata_);
} }
const commands = this.commands(); const commands = this.commands();
@@ -207,7 +207,7 @@ class Application extends BaseApplication {
await this.cache_.setItem('metadata', output, 1000 * 60 * 60 * 24); await this.cache_.setItem('metadata', output, 1000 * 60 * 60 * 24);
this.commandMetadata_ = output; this.commandMetadata_ = output;
return { ...this.commandMetadata_ }; return Object.assign({}, this.commandMetadata_);
} }
hasGui() { hasGui() {

View File

@@ -44,7 +44,7 @@
"@joplin/lib": "~2.11", "@joplin/lib": "~2.11",
"@joplin/renderer": "~2.11", "@joplin/renderer": "~2.11",
"@joplin/utils": "~2.11", "@joplin/utils": "~2.11",
"aws-sdk": "2.1340.0", "aws-sdk": "2.1290.0",
"chalk": "4.1.2", "chalk": "4.1.2",
"compare-version": "0.1.2", "compare-version": "0.1.2",
"fs-extra": "11.1.1", "fs-extra": "11.1.1",
@@ -57,7 +57,7 @@
"proper-lockfile": "4.1.2", "proper-lockfile": "4.1.2",
"read-chunk": "2.1.0", "read-chunk": "2.1.0",
"server-destroy": "1.0.1", "server-destroy": "1.0.1",
"sharp": "0.32.1", "sharp": "0.31.3",
"sprintf-js": "1.1.2", "sprintf-js": "1.1.2",
"sqlite3": "5.1.6", "sqlite3": "5.1.6",
"string-padding": "1.0.2", "string-padding": "1.0.2",
@@ -71,13 +71,13 @@
}, },
"devDependencies": { "devDependencies": {
"@joplin/tools": "~2.11", "@joplin/tools": "~2.11",
"@types/fs-extra": "11.0.1", "@types/fs-extra": "9.0.13",
"@types/jest": "29.5.1", "@types/jest": "29.2.6",
"@types/node": "18.15.13", "@types/node": "18.11.18",
"@types/proper-lockfile": "^4.1.2", "@types/proper-lockfile": "^4.1.2",
"gulp": "4.0.2", "gulp": "4.0.2",
"jest": "29.5.0", "jest": "29.4.3",
"temp": "0.9.4", "temp": "0.9.4",
"typescript": "5.0.2" "typescript": "4.9.4"
} }
} }

View File

@@ -1,19 +0,0 @@
<table>
<tbody>
<tr>
<td style="text-align:left">Left</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
<tr>
<td style="text-align:left">Left</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
<tr>
<td style="text-align:left">Left</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
</tbody>
</table>

View File

@@ -1,5 +0,0 @@
| | | |
| :--- | :---: | ---: |
| Left | Centered | Right |
| Left | Centered | Right |
| Left | Centered | Right |

View File

@@ -1,26 +0,0 @@
<table>
<thead>
<tr>
<th style="text-align:left">Left-aligned Column</th>
<th style="text-align:center">Center-aligned Column</th>
<th style="text-align:right">Right-aligned Column</th>
</tr>
</thead>
<tbody>
<tr>
<td style="text-align:left">Left</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
<tr>
<td style="text-align:left">Left</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
<tr>
<td style="text-align:left">Left</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
</tbody>
</table>

View File

@@ -1,5 +0,0 @@
| Left-aligned Column | Center-aligned Column | Right-aligned Column |
| :--- | :---: | ---: |
| Left | Centered | Right |
| Left | Centered | Right |
| Left | Centered | Right |

View File

@@ -1,14 +0,0 @@
<table>
<thead>
<tr>
<th align="center">abc</th>
<th align="right">defghi</th>
</tr>
</thead>
<tbody>
<tr>
<td align="center">bar</td>
<td align="right">baz</td>
</tr>
</tbody>
</table>

View File

@@ -1,3 +0,0 @@
| abc | defghi |
| :---: | ---: |
| bar | baz |

View File

@@ -1,29 +0,0 @@
<table>
<thead>
<tr>
<th style="text-align:left">Left-aligned Column</th>
<th>This header cell's text is unaligned, but a majority of the text in this column is center-aligned so the
column will be center-aligned</th>
<th style="text-align:right">Right-aligned Column</th>
</tr>
</thead>
<tbody>
<tr>
<td style="text-align:left">Left</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
<tr>
<td style="text-align:right">This is the only right-aligned cell in this column. This is possible if a user
edits a cell's alignment using the cell properties dialog.</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
<tr>
<td style="text-align:center">This is the only center-aligned cell in this column. This is possible if a
user edits a cell's alignment using the cell properties dialog.</td>
<td style="text-align:center">Centered</td>
<td style="text-align:right">Right</td>
</tr>
</tbody>
</table>

View File

@@ -1,5 +0,0 @@
| Left-aligned Column | This header cell's text is unaligned, but a majority of the text in this column is center-aligned so the column will be center-aligned | Right-aligned Column |
| :--- | :---: | ---: |
| Left | Centered | Right |
| This is the only right-aligned cell in this column. This is possible if a user edits a cell's alignment using the cell properties dialog. | Centered | Right |
| This is the only center-aligned cell in this column. This is possible if a user edits a cell's alignment using the cell properties dialog. | Centered | Right |

View File

@@ -1 +0,0 @@
<div class="jop-noMdConv">

View File

@@ -1 +0,0 @@
<div><svg><style></svg><iframe srcdoc="<script>top.require('child_process').execSync('calc')</script>"></iframe></div>

View File

@@ -1 +0,0 @@
<a href="#" class="jop-noMdConv">XSS</a>

View File

@@ -1 +0,0 @@
<a data-from-md="" href="javascript:top.require('child_process').execSync('open -a Calculator')">XSS</a>

View File

@@ -1 +0,0 @@
<use href="#" class="jop-noMdConv">

View File

@@ -1 +0,0 @@
<svg><use href="data:image/svg+xml,&lt;svg id='x' xmlns='http://www.w3.org/2000/svg'&gt;&lt;image href='asdf' onerror='top.require(`child_process`).execSync(`calc.exe`)' /&gt;&lt;/svg&gt;#x" />

Before

Width:  |  Height:  |  Size: 193 B

View File

@@ -1 +0,0 @@
<map name="test" class="jop-noMdConv"><area coords="0,0,1000,1000" href="#" class="jop-noMdConv"/></map><img usemap="#test" src="https://github.com/Ry0taK.png" class="jop-noMdConv"/>

View File

@@ -1 +0,0 @@
<map name="test"><area coords="0,0,1000,1000" href="javascript:top.require(`child_process`).execSync(`calc.exe`)"></map><img usemap="#test" src="https://github.com/Ry0taK.png">

View File

@@ -1 +0,0 @@
<a href="#top" class="jop-noMdConv">⬆️</a>

View File

@@ -1 +0,0 @@
<a href="#top">⬆️</a>

View File

@@ -94,12 +94,12 @@ browser_.runtime.onMessage.addListener(async (command) => {
const imageSize = await getImageSize(imageDataUrl); const imageSize = await getImageSize(imageDataUrl);
const imagePixelRatio = imageSize.width / command.content.windowInnerWidth; const imagePixelRatio = imageSize.width / command.content.windowInnerWidth;
const content = { ...command.content }; const content = Object.assign({}, command.content);
content.image_data_url = imageDataUrl; content.image_data_url = imageDataUrl;
if ('url' in content) content.source_url = content.url; if ('url' in content) content.source_url = content.url;
const ratio = browserZoom * imagePixelRatio; const ratio = browserZoom * imagePixelRatio;
const newArea = { ...command.content.crop_rect }; const newArea = Object.assign({}, command.content.crop_rect);
newArea.x *= ratio; newArea.x *= ratio;
newArea.y *= ratio; newArea.y *= ratio;
newArea.width *= ratio; newArea.width *= ratio;

View File

@@ -378,7 +378,7 @@
tags: command.tags || '', tags: command.tags || '',
image_sizes: imageSizes, image_sizes: imageSizes,
anchor_names: anchorNames, anchor_names: anchorNames,
source_command: { ...command }, source_command: Object.assign({}, command),
convert_to: convertToMarkup, convert_to: convertToMarkup,
stylesheets: stylesheets, stylesheets: stylesheets,
}; };
@@ -392,7 +392,7 @@
} catch (error) { } catch (error) {
console.warn(error); console.warn(error);
console.warn('Sending full page HTML instead'); console.warn('Sending full page HTML instead');
const newCommand = { ...command, name: 'completePageHtml' }; const newCommand = Object.assign({}, command, { name: 'completePageHtml' });
const response = await prepareCommandResponse(newCommand); const response = await prepareCommandResponse(newCommand);
response.warning = 'Could not retrieve simplified version of page - full page has been saved instead.'; response.warning = 'Could not retrieve simplified version of page - full page has been saved instead.';
return response; return response;

View File

@@ -67,7 +67,7 @@ class AppComponent extends Component {
}); });
this.confirm_click = async () => { this.confirm_click = async () => {
const content = { ...this.props.clippedContent }; const content = Object.assign({}, this.props.clippedContent);
content.tags = this.state.selectedTags.join(','); content.tags = this.state.selectedTags.join(',');
content.parent_id = this.props.selectedFolderId; content.parent_id = this.props.selectedFolderId;
const response = await bridge().sendContentToJoplin(content); const response = await bridge().sendContentToJoplin(content);

View File

@@ -410,7 +410,7 @@ class Bridge {
if (body) fetchOptions.body = typeof body === 'string' ? body : JSON.stringify(body); if (body) fetchOptions.body = typeof body === 'string' ? body : JSON.stringify(body);
query = { ...query, token: this.token_ }; query = Object.assign(query || {}, { token: this.token_ });
let queryString = ''; let queryString = '';
if (query) { if (query) {

View File

@@ -40,34 +40,34 @@ function reducer(state = defaultState, action) {
if (action.type === 'WARNING_SET') { if (action.type === 'WARNING_SET') {
newState = { ...state }; newState = Object.assign({}, state);
newState.warning = action.text; newState.warning = action.text;
} else if (action.type === 'IS_PROBABLY_READERABLE') { } else if (action.type === 'IS_PROBABLY_READERABLE') {
newState = { ...state }; newState = Object.assign({}, state);
newState.isProbablyReaderable = action.value; newState.isProbablyReaderable = action.value;
} else if (action.type === 'CLIPPED_CONTENT_SET') { } else if (action.type === 'CLIPPED_CONTENT_SET') {
newState = { ...state }; newState = Object.assign({}, state);
newState.clippedContent = action.content; newState.clippedContent = action.content;
} else if (action.type === 'CLIPPED_CONTENT_TITLE_SET') { } else if (action.type === 'CLIPPED_CONTENT_TITLE_SET') {
newState = { ...state }; newState = Object.assign({}, state);
const newContent = newState.clippedContent ? { ...newState.clippedContent } : {}; const newContent = newState.clippedContent ? Object.assign({}, newState.clippedContent) : {};
newContent.title = action.text; newContent.title = action.text;
newState.clippedContent = newContent; newState.clippedContent = newContent;
} else if (action.type === 'CONTENT_UPLOAD') { } else if (action.type === 'CONTENT_UPLOAD') {
newState = { ...state }; newState = Object.assign({}, state);
newState.contentUploadOperation = action.operation; newState.contentUploadOperation = action.operation;
} else if (action.type === 'FOLDERS_SET') { } else if (action.type === 'FOLDERS_SET') {
newState = { ...state }; newState = Object.assign({}, state);
newState.folders = action.folders; newState.folders = action.folders;
if (!newState.selectedFolderId && action.folders.length) { if (!newState.selectedFolderId && action.folders.length) {
@@ -76,30 +76,30 @@ function reducer(state = defaultState, action) {
} else if (action.type === 'TAGS_SET') { } else if (action.type === 'TAGS_SET') {
newState = { ...state }; newState = Object.assign({}, state);
newState.tags = action.tags; newState.tags = action.tags;
} else if (action.type === 'SELECTED_FOLDER_SET') { } else if (action.type === 'SELECTED_FOLDER_SET') {
newState = { ...state }; newState = Object.assign({}, state);
newState.selectedFolderId = action.id; newState.selectedFolderId = action.id;
} else if (action.type === 'CLIPPER_SERVER_SET') { } else if (action.type === 'CLIPPER_SERVER_SET') {
newState = { ...state }; newState = Object.assign({}, state);
const clipperServer = { ...newState.clipperServer }; const clipperServer = Object.assign({}, newState.clipperServer);
if ('foundState' in action) clipperServer.foundState = action.foundState; if ('foundState' in action) clipperServer.foundState = action.foundState;
if ('port' in action) clipperServer.port = action.port; if ('port' in action) clipperServer.port = action.port;
newState.clipperServer = clipperServer; newState.clipperServer = clipperServer;
} else if (action.type === 'ENV_SET') { } else if (action.type === 'ENV_SET') {
newState = { ...state }; newState = Object.assign({}, state);
newState.env = action.env; newState.env = action.env;
} else if (action.type === 'AUTH_STATE_SET') { } else if (action.type === 'AUTH_STATE_SET') {
newState = { ...state }; newState = Object.assign({}, state);
newState.authStatus = action.value; newState.authStatus = action.value;
} }

View File

@@ -29,11 +29,13 @@ export default class InteropServiceHelper {
private static async exportNoteToHtmlFile(noteId: string, exportOptions: ExportNoteOptions) { private static async exportNoteToHtmlFile(noteId: string, exportOptions: ExportNoteOptions) {
const tempFile = `${Setting.value('tempDir')}/${md5(Date.now() + Math.random())}.html`; const tempFile = `${Setting.value('tempDir')}/${md5(Date.now() + Math.random())}.html`;
const fullExportOptions: ExportOptions = { path: tempFile, const fullExportOptions: ExportOptions = Object.assign({}, {
path: tempFile,
format: 'html', format: 'html',
target: FileSystemItem.File, target: FileSystemItem.File,
sourceNoteIds: [noteId], sourceNoteIds: [noteId],
customCss: '', ...exportOptions }; customCss: '',
}, exportOptions);
const service = InteropService.instance(); const service = InteropService.instance();

View File

@@ -4,9 +4,6 @@ import { defaultState, State } from '@joplin/lib/reducer';
import iterateItems from './gui/ResizableLayout/utils/iterateItems'; import iterateItems from './gui/ResizableLayout/utils/iterateItems';
import { LayoutItem } from './gui/ResizableLayout/utils/types'; import { LayoutItem } from './gui/ResizableLayout/utils/types';
import validateLayout from './gui/ResizableLayout/utils/validateLayout'; import validateLayout from './gui/ResizableLayout/utils/validateLayout';
import Logger from '@joplin/lib/Logger';
const logger = Logger.create('app.reducer');
export interface AppStateRoute { export interface AppStateRoute {
type: string; type: string;
@@ -85,7 +82,7 @@ export default function(state: AppState, action: any) {
const currentRoute = state.route; const currentRoute = state.route;
newState = { ...state }; newState = Object.assign({}, state);
const newNavHistory = state.navHistory.slice(); const newNavHistory = state.navHistory.slice();
if (goingBack) { if (goingBack) {
@@ -122,7 +119,7 @@ export default function(state: AppState, action: any) {
case 'WINDOW_CONTENT_SIZE_SET': case 'WINDOW_CONTENT_SIZE_SET':
newState = { ...state }; newState = Object.assign({}, state);
newState.windowContentSize = action.size; newState.windowContentSize = action.size;
break; break;
@@ -150,7 +147,7 @@ export default function(state: AppState, action: any) {
return nextLayout === 'both' ? ['editor', 'viewer'] : [nextLayout]; return nextLayout === 'both' ? ['editor', 'viewer'] : [nextLayout];
}; };
newState = { ...state }; newState = Object.assign({}, state);
const panes = state.noteVisiblePanes.slice(); const panes = state.noteVisiblePanes.slice();
newState.noteVisiblePanes = getNextLayout(panes); newState.noteVisiblePanes = getNextLayout(panes);
@@ -159,7 +156,7 @@ export default function(state: AppState, action: any) {
case 'NOTE_VISIBLE_PANES_SET': case 'NOTE_VISIBLE_PANES_SET':
newState = { ...state }; newState = Object.assign({}, state);
newState.noteVisiblePanes = action.panes; newState.noteVisiblePanes = action.panes;
break; break;
@@ -174,31 +171,22 @@ export default function(state: AppState, action: any) {
case 'MAIN_LAYOUT_SET_ITEM_PROP': case 'MAIN_LAYOUT_SET_ITEM_PROP':
{ {
if (!state.mainLayout) { let newLayout = produce(state.mainLayout, (draftLayout: LayoutItem) => {
logger.warn('MAIN_LAYOUT_SET_ITEM_PROP: Trying to set an item prop on the layout, but layout is empty: ', JSON.stringify(action)); iterateItems(draftLayout, (_itemIndex: number, item: LayoutItem, _parent: LayoutItem) => {
} else { if (item.key === action.itemKey) {
let newLayout = produce(state.mainLayout, (draftLayout: LayoutItem) => { (item as any)[action.propName] = action.propValue;
iterateItems(draftLayout, (_itemIndex: number, item: LayoutItem, _parent: LayoutItem) => { return false;
if (!item) { }
logger.warn('MAIN_LAYOUT_SET_ITEM_PROP: Found an empty item in layout: ', JSON.stringify(state.mainLayout)); return true;
} else {
if (item.key === action.itemKey) {
(item as any)[action.propName] = action.propValue;
return false;
}
}
return true;
});
}); });
});
if (newLayout !== state.mainLayout) newLayout = validateLayout(newLayout); if (newLayout !== state.mainLayout) newLayout = validateLayout(newLayout);
newState = { newState = {
...state, ...state,
mainLayout: newLayout, mainLayout: newLayout,
}; };
}
} }
break; break;
@@ -206,7 +194,7 @@ export default function(state: AppState, action: any) {
case 'NOTE_FILE_WATCHER_ADD': case 'NOTE_FILE_WATCHER_ADD':
if (newState.watchedNoteFiles.indexOf(action.id) < 0) { if (newState.watchedNoteFiles.indexOf(action.id) < 0) {
newState = { ...state }; newState = Object.assign({}, state);
const watchedNoteFiles = newState.watchedNoteFiles.slice(); const watchedNoteFiles = newState.watchedNoteFiles.slice();
watchedNoteFiles.push(action.id); watchedNoteFiles.push(action.id);
newState.watchedNoteFiles = watchedNoteFiles; newState.watchedNoteFiles = watchedNoteFiles;
@@ -216,7 +204,7 @@ export default function(state: AppState, action: any) {
case 'NOTE_FILE_WATCHER_REMOVE': case 'NOTE_FILE_WATCHER_REMOVE':
{ {
newState = { ...state }; newState = Object.assign({}, state);
const idx = newState.watchedNoteFiles.indexOf(action.id); const idx = newState.watchedNoteFiles.indexOf(action.id);
if (idx >= 0) { if (idx >= 0) {
const watchedNoteFiles = newState.watchedNoteFiles.slice(); const watchedNoteFiles = newState.watchedNoteFiles.slice();
@@ -229,7 +217,7 @@ export default function(state: AppState, action: any) {
case 'NOTE_FILE_WATCHER_CLEAR': case 'NOTE_FILE_WATCHER_CLEAR':
if (state.watchedNoteFiles.length) { if (state.watchedNoteFiles.length) {
newState = { ...state }; newState = Object.assign({}, state);
newState.watchedNoteFiles = []; newState.watchedNoteFiles = [];
} }
break; break;
@@ -237,38 +225,38 @@ export default function(state: AppState, action: any) {
case 'EDITOR_SCROLL_PERCENT_SET': case 'EDITOR_SCROLL_PERCENT_SET':
{ {
newState = { ...state }; newState = Object.assign({}, state);
const newPercents = { ...newState.lastEditorScrollPercents }; const newPercents = Object.assign({}, newState.lastEditorScrollPercents);
newPercents[action.noteId] = action.percent; newPercents[action.noteId] = action.percent;
newState.lastEditorScrollPercents = newPercents; newState.lastEditorScrollPercents = newPercents;
} }
break; break;
case 'NOTE_DEVTOOLS_TOGGLE': case 'NOTE_DEVTOOLS_TOGGLE':
newState = { ...state }; newState = Object.assign({}, state);
newState.devToolsVisible = !newState.devToolsVisible; newState.devToolsVisible = !newState.devToolsVisible;
break; break;
case 'NOTE_DEVTOOLS_SET': case 'NOTE_DEVTOOLS_SET':
newState = { ...state }; newState = Object.assign({}, state);
newState.devToolsVisible = action.value; newState.devToolsVisible = action.value;
break; break;
case 'VISIBLE_DIALOGS_ADD': case 'VISIBLE_DIALOGS_ADD':
newState = { ...state }; newState = Object.assign({}, state);
newState.visibleDialogs = { ...newState.visibleDialogs }; newState.visibleDialogs = Object.assign({}, newState.visibleDialogs);
newState.visibleDialogs[action.name] = true; newState.visibleDialogs[action.name] = true;
break; break;
case 'VISIBLE_DIALOGS_REMOVE': case 'VISIBLE_DIALOGS_REMOVE':
newState = { ...state }; newState = Object.assign({}, state);
newState.visibleDialogs = { ...newState.visibleDialogs }; newState.visibleDialogs = Object.assign({}, newState.visibleDialogs);
delete newState.visibleDialogs[action.name]; delete newState.visibleDialogs[action.name];
break; break;
case 'FOCUS_SET': case 'FOCUS_SET':
newState = { ...state }; newState = Object.assign({}, state);
newState.focusedField = action.field; newState.focusedField = action.field;
break; break;
@@ -276,7 +264,7 @@ export default function(state: AppState, action: any) {
// A field can only clear its own state // A field can only clear its own state
if (action.field === state.focusedField) { if (action.field === state.focusedField) {
newState = { ...state }; newState = Object.assign({}, state);
newState.focusedField = null; newState.focusedField = null;
} }
break; break;
@@ -293,7 +281,7 @@ export default function(state: AppState, action: any) {
isOpen = action.isOpen !== false; isOpen = action.isOpen !== false;
} }
newState = { ...state }; newState = Object.assign({}, state);
if (isOpen) { if (isOpen) {
const newDialogs = newState.dialogs.slice(); const newDialogs = newState.dialogs.slice();

View File

@@ -80,7 +80,6 @@ const appDefaultState = createAppDefaultState(
class Application extends BaseApplication { class Application extends BaseApplication {
private checkAllPluginStartedIID_: any = null; private checkAllPluginStartedIID_: any = null;
private initPluginServiceDone_: boolean = false;
public constructor() { public constructor() {
super(); super();
@@ -259,9 +258,6 @@ class Application extends BaseApplication {
} }
private async initPluginService() { private async initPluginService() {
if (this.initPluginServiceDone_) return;
this.initPluginServiceDone_ = true;
const service = PluginService.instance(); const service = PluginService.instance();
const pluginRunner = new PluginRunner(); const pluginRunner = new PluginRunner();

View File

@@ -68,10 +68,6 @@ export class Bridge {
return process.argv; return process.argv;
} }
public getLocale = () => {
return this.electronApp().electronApp().getLocale();
};
// Applies to electron-context-menu@3: // Applies to electron-context-menu@3:
// //
// For now we have to disable spell checking in non-editor text // For now we have to disable spell checking in non-editor text
@@ -201,10 +197,12 @@ export class Bridge {
...options, ...options,
}; };
const result = this.showMessageBox_(this.window(), { type: 'question', const result = this.showMessageBox_(this.window(), Object.assign({}, {
type: 'question',
message: message, message: message,
cancelId: 1, cancelId: 1,
buttons: options.buttons, ...options }); buttons: options.buttons,
}, options));
return result === 0; return result === 0;
} }
@@ -213,17 +211,21 @@ export class Bridge {
public showMessageBox(message: string, options: any = null) { public showMessageBox(message: string, options: any = null) {
if (options === null) options = {}; if (options === null) options = {};
const result = this.showMessageBox_(this.window(), { type: 'question', const result = this.showMessageBox_(this.window(), Object.assign({}, {
type: 'question',
message: message, message: message,
buttons: [_('OK'), _('Cancel')], ...options }); buttons: [_('OK'), _('Cancel')],
}, options));
return result; return result;
} }
public showInfoMessageBox(message: string, options: any = {}) { public showInfoMessageBox(message: string, options: any = {}) {
const result = this.showMessageBox_(this.window(), { type: 'info', const result = this.showMessageBox_(this.window(), Object.assign({}, {
type: 'info',
message: message, message: message,
buttons: [_('OK')], ...options }); buttons: [_('OK')],
}, options));
return result === 0; return result === 0;
} }

View File

@@ -34,7 +34,7 @@ function getMajorMinorTagName(tagName: string) {
} }
async function fetchLatestRelease(options: CheckForUpdateOptions) { async function fetchLatestRelease(options: CheckForUpdateOptions) {
options = { includePreReleases: false, ...options }; options = Object.assign({}, { includePreReleases: false }, options);
const response = await shim.fetch('https://api.github.com/repos/laurent22/joplin/releases'); const response = await shim.fetch('https://api.github.com/repos/laurent22/joplin/releases');
@@ -97,8 +97,7 @@ async function fetchLatestRelease(options: CheckForUpdateOptions) {
} }
if (found) { if (found) {
downloadUrl = asset.browser_download_url.replace('github.com/laurent22/joplin/releases/download', 'objects.joplinusercontent.com'); downloadUrl = asset.browser_download_url;
downloadUrl.concat('?source=DesktopApp&type=Update');
break; break;
} }
} }

View File

@@ -20,7 +20,7 @@ export const runtime = (): CommandRuntime => {
}; };
await saveProfileConfig(`${Setting.value('rootProfileDir')}/profiles.json`, newConfig); await saveProfileConfig(`${Setting.value('rootProfileDir')}/profiles.json`, newConfig);
await restart(); await restart(false);
}, },
}; };
}; };

View File

@@ -46,11 +46,13 @@ class ClipperConfigScreenComponent extends React.Component {
public render() { public render() {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const containerStyle = { ...theme.containerStyle, overflowY: 'scroll', const containerStyle = Object.assign({}, theme.containerStyle, {
overflowY: 'scroll',
// padding: theme.configScreenPadding, // padding: theme.configScreenPadding,
backgroundColor: theme.backgroundColor3 }; backgroundColor: theme.backgroundColor3,
});
const buttonStyle = { ...theme.buttonStyle, marginRight: 10 }; const buttonStyle = Object.assign({}, theme.buttonStyle, { marginRight: 10 });
const stepBoxStyle = { const stepBoxStyle = {
border: '1px solid', border: '1px solid',
@@ -104,16 +106,18 @@ class ClipperConfigScreenComponent extends React.Component {
); );
} }
const apiTokenStyle = { ...theme.textStyle, color: theme.colorFaded, const apiTokenStyle = Object.assign({}, theme.textStyle, {
color: theme.colorFaded,
wordBreak: 'break-all', wordBreak: 'break-all',
paddingTop: 10, paddingTop: 10,
paddingBottom: 10 }; paddingBottom: 10,
});
return ( return (
<div> <div>
<div style={containerStyle}> <div style={containerStyle}>
<div> <div>
<p style={{ ...theme.textStyle, marginTop: 0 }}>{_('Joplin Web Clipper allows saving web pages and screenshots from your browser to Joplin.')}</p> <p style={Object.assign({}, theme.textStyle, { marginTop: 0 })}>{_('Joplin Web Clipper allows saving web pages and screenshots from your browser to Joplin.')}</p>
<p style={theme.textStyle}>{_('In order to use the web clipper, you need to do the following:')}</p> <p style={theme.textStyle}>{_('In order to use the web clipper, you need to do the following:')}</p>
<div style={stepBoxStyle}> <div style={stepBoxStyle}>

View File

@@ -135,7 +135,7 @@ class ConfigScreenComponent extends React.Component<any, any> {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
return ( return (
<div style={{ ...theme.textStyle, marginBottom: 15 }}> <div style={Object.assign({}, theme.textStyle, { marginBottom: 15 })}>
{description} {description}
</div> </div>
); );
@@ -177,7 +177,7 @@ class ConfigScreenComponent extends React.Component<any, any> {
if (section.name === 'sync') { if (section.name === 'sync') {
const syncTargetMd = SyncTargetRegistry.idToMetadata(settings['sync.target']); const syncTargetMd = SyncTargetRegistry.idToMetadata(settings['sync.target']);
const statusStyle = { ...theme.textStyle, marginTop: 10 }; const statusStyle = Object.assign({}, theme.textStyle, { marginTop: 10 });
if (syncTargetMd.supportsConfigCheck) { if (syncTargetMd.supportsConfigCheck) {
const messages = shared.checkSyncConfigMessages(this); const messages = shared.checkSyncConfigMessages(this);
@@ -207,7 +207,7 @@ class ConfigScreenComponent extends React.Component<any, any> {
if (advancedSettingComps.length) { if (advancedSettingComps.length) {
const iconName = this.state.showAdvancedSettings ? 'fa fa-angle-down' : 'fa fa-angle-right'; const iconName = this.state.showAdvancedSettings ? 'fa fa-angle-down' : 'fa fa-angle-right';
// const advancedSettingsButtonStyle = { ...theme.buttonStyle, marginBottom: 10 }; // const advancedSettingsButtonStyle = Object.assign({}, theme.buttonStyle, { marginBottom: 10 });
advancedSettingsButton = ( advancedSettingsButton = (
<div style={{ marginBottom: 10 }}> <div style={{ marginBottom: 10 }}>
<Button <Button
@@ -233,19 +233,23 @@ class ConfigScreenComponent extends React.Component<any, any> {
private labelStyle(themeId: number) { private labelStyle(themeId: number) {
const theme = themeStyle(themeId); const theme = themeStyle(themeId);
return { ...theme.textStyle, display: 'block', return Object.assign({}, theme.textStyle, {
display: 'block',
color: theme.color, color: theme.color,
fontSize: theme.fontSize * 1.083333, fontSize: theme.fontSize * 1.083333,
fontWeight: 500, fontWeight: 500,
marginBottom: theme.mainPadding / 2 }; marginBottom: theme.mainPadding / 2,
});
} }
private descriptionStyle(themeId: number) { private descriptionStyle(themeId: number) {
const theme = themeStyle(themeId); const theme = themeStyle(themeId);
return { ...theme.textStyle, color: theme.colorFaded, return Object.assign({}, theme.textStyle, {
color: theme.colorFaded,
fontStyle: 'italic', fontStyle: 'italic',
maxWidth: '70em', maxWidth: '70em',
marginTop: 5 }; marginTop: 5,
});
} }
private renderLabel(themeId: number, label: string) { private renderLabel(themeId: number, label: string) {
@@ -260,12 +264,14 @@ class ConfigScreenComponent extends React.Component<any, any> {
private renderHeader(themeId: number, label: string, style: any = null) { private renderHeader(themeId: number, label: string, style: any = null) {
const theme = themeStyle(themeId); const theme = themeStyle(themeId);
const labelStyle = { ...theme.textStyle, display: 'block', const labelStyle = Object.assign({}, theme.textStyle, {
display: 'block',
color: theme.color, color: theme.color,
fontSize: theme.fontSize * 1.25, fontSize: theme.fontSize * 1.25,
fontWeight: 500, fontWeight: 500,
marginBottom: theme.mainPadding, marginBottom: theme.mainPadding,
...style }; ...style,
});
return ( return (
<div style={labelStyle}> <div style={labelStyle}>
@@ -289,13 +295,17 @@ class ConfigScreenComponent extends React.Component<any, any> {
const labelStyle = this.labelStyle(this.props.themeId); const labelStyle = this.labelStyle(this.props.themeId);
const subLabel = { ...labelStyle, display: 'block', const subLabel = Object.assign({}, labelStyle, {
display: 'block',
opacity: 0.7, opacity: 0.7,
marginBottom: labelStyle.marginBottom }; marginBottom: labelStyle.marginBottom,
});
const checkboxLabelStyle = { ...labelStyle, marginLeft: 8, const checkboxLabelStyle = Object.assign({}, labelStyle, {
marginLeft: 8,
display: 'inline', display: 'inline',
backgroundColor: 'transparent' }; backgroundColor: 'transparent',
});
const controlStyle = { const controlStyle = {
display: 'inline-block', display: 'inline-block',
@@ -304,7 +314,8 @@ class ConfigScreenComponent extends React.Component<any, any> {
backgroundColor: theme.backgroundColor, backgroundColor: theme.backgroundColor,
}; };
const textInputBaseStyle = { ...controlStyle, fontFamily: theme.fontFamily, const textInputBaseStyle = Object.assign({}, controlStyle, {
fontFamily: theme.fontFamily,
border: '1px solid', border: '1px solid',
padding: '4px 6px', padding: '4px 6px',
boxSizing: 'border-box', boxSizing: 'border-box',
@@ -313,7 +324,8 @@ class ConfigScreenComponent extends React.Component<any, any> {
paddingLeft: 6, paddingLeft: 6,
paddingRight: 6, paddingRight: 6,
paddingTop: 4, paddingTop: 4,
paddingBottom: 4 }; paddingBottom: 4,
});
const updateSettingValue = (key: string, value: any) => { const updateSettingValue = (key: string, value: any) => {
const md = Setting.settingMetadata(key); const md = Setting.settingMetadata(key);
@@ -369,12 +381,14 @@ class ConfigScreenComponent extends React.Component<any, any> {
); );
} }
const selectStyle = { ...controlStyle, paddingLeft: 6, const selectStyle = Object.assign({}, controlStyle, {
paddingLeft: 6,
paddingRight: 6, paddingRight: 6,
paddingTop: 4, paddingTop: 4,
paddingBottom: 4, paddingBottom: 4,
borderColor: theme.borderColor4, borderColor: theme.borderColor4,
borderRadius: 3 }; borderRadius: 3,
});
return ( return (
<div key={key} style={rowStyle}> <div key={key} style={rowStyle}>
@@ -429,8 +443,10 @@ class ConfigScreenComponent extends React.Component<any, any> {
</div> </div>
); );
} else if (md.type === Setting.TYPE_STRING) { } else if (md.type === Setting.TYPE_STRING) {
const inputStyle: any = { ...textInputBaseStyle, width: '50%', const inputStyle: any = Object.assign({}, textInputBaseStyle, {
minWidth: '20em' }; width: '50%',
minWidth: '20em',
});
const inputType = md.secure === true ? 'password' : 'text'; const inputType = md.secure === true ? 'password' : 'text';
if (md.subType === 'file_path_and_args' || md.subType === 'file_path' || md.subType === 'directory_path') { if (md.subType === 'file_path_and_args' || md.subType === 'file_path' || md.subType === 'directory_path') {
@@ -526,7 +542,7 @@ class ConfigScreenComponent extends React.Component<any, any> {
<div style={{ display: 'flex', flexDirection: 'row', alignItems: 'center', marginBottom: inputStyle.marginBottom }}> <div style={{ display: 'flex', flexDirection: 'row', alignItems: 'center', marginBottom: inputStyle.marginBottom }}>
<input <input
type={inputType} type={inputType}
style={{ ...inputStyle, marginBottom: 0, marginRight: 5 }} style={Object.assign({}, inputStyle, { marginBottom: 0, marginRight: 5 })}
onChange={(event: any) => { onChange={(event: any) => {
onPathChange(event); onPathChange(event);
}} }}
@@ -579,7 +595,7 @@ class ConfigScreenComponent extends React.Component<any, any> {
const label = [md.label()]; const label = [md.label()];
if (md.unitLabel) label.push(`(${md.unitLabel()})`); if (md.unitLabel) label.push(`(${md.unitLabel()})`);
const inputStyle: any = { ...textInputBaseStyle }; const inputStyle: any = Object.assign({}, textInputBaseStyle);
return ( return (
<div key={key} style={rowStyle}> <div key={key} style={rowStyle}>
@@ -663,13 +679,15 @@ class ConfigScreenComponent extends React.Component<any, any> {
public render() { public render() {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const style = { const style = Object.assign({},
...this.props.style, this.props.style,
overflow: 'hidden', {
display: 'flex', overflow: 'hidden',
flexDirection: 'column', display: 'flex',
backgroundColor: theme.backgroundColor3, flexDirection: 'column',
}; backgroundColor: theme.backgroundColor3,
}
);
const settings = this.state.settings; const settings = this.state.settings;

View File

@@ -74,7 +74,7 @@ export default function DialogButtonRow(props: Props) {
if (props.cancelButtonShow !== false) { if (props.cancelButtonShow !== false) {
buttonComps.push( buttonComps.push(
<button disabled={props.cancelButtonDisabled} key="cancel" style={{ ...buttonStyle }} onClick={onCancelButtonClick}> <button disabled={props.cancelButtonDisabled} key="cancel" style={Object.assign({}, buttonStyle)} onClick={onCancelButtonClick}>
{props.cancelButtonLabel ? props.cancelButtonLabel : _('Cancel')} {props.cancelButtonLabel ? props.cancelButtonLabel : _('Cancel')}
</button> </button>
); );

View File

@@ -29,13 +29,15 @@ class DropboxLoginScreenComponent extends React.Component<any, any> {
const style = this.props.style; const style = this.props.style;
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const containerStyle = { ...theme.containerStyle, padding: theme.configScreenPadding, const containerStyle = Object.assign({}, theme.containerStyle, {
padding: theme.configScreenPadding,
height: style.height - theme.margin * 2, height: style.height - theme.margin * 2,
flex: 1 }; flex: 1,
});
const inputStyle = { ...theme.inputStyle, width: 500 }; const inputStyle = Object.assign({}, theme.inputStyle, { width: 500 });
const buttonStyle = { ...theme.buttonStyle, marginRight: 10 }; const buttonStyle = Object.assign({}, theme.buttonStyle, { marginRight: 10 });
return ( return (
<div style={{ display: 'flex', flexDirection: 'column', height: '100%' }}> <div style={{ display: 'flex', flexDirection: 'column', height: '100%' }}>

View File

@@ -82,7 +82,7 @@ function ExtensionBadge(props: Props) {
void bridge().openExternal(props.url); void bridge().openExternal(props.url);
}; };
const rootStyle = props.style ? { ...style.root, ...props.style } : style.root; const rootStyle = props.style ? Object.assign({}, style.root, props.style) : style.root;
return ( return (
<a style={rootStyle} onClick={onClick} href="#"> <a style={rootStyle} onClick={onClick} href="#">

View File

@@ -23,7 +23,7 @@ class HelpButtonComponent extends React.Component<Props> {
public render() { public render() {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const style = { ...this.props.style, color: theme.color, textDecoration: 'none' }; const style = Object.assign({}, this.props.style, { color: theme.color, textDecoration: 'none' });
const helpIconStyle = { flex: 0, width: 16, height: 16, marginLeft: 10 }; const helpIconStyle = { flex: 0, width: 16, height: 16, marginLeft: 10 };
const extraProps: any = {}; const extraProps: any = {};
if (this.props.tip) extraProps['data-tip'] = this.props.tip; if (this.props.tip) extraProps['data-tip'] = this.props.tip;

View File

@@ -18,19 +18,21 @@ class IconButton extends React.Component<Props> {
}; };
const icon = <i style={iconStyle} className={`fas ${this.props.iconName}`}></i>; const icon = <i style={iconStyle} className={`fas ${this.props.iconName}`}></i>;
const rootStyle = { const rootStyle = Object.assign(
display: 'flex', {
textDecoration: 'none', display: 'flex',
padding: 10, textDecoration: 'none',
width: theme.buttonMinHeight, padding: 10,
height: theme.buttonMinHeight, width: theme.buttonMinHeight,
boxSizing: 'border-box', height: theme.buttonMinHeight,
alignItems: 'center', boxSizing: 'border-box',
justifyContent: 'center', alignItems: 'center',
backgroundColor: theme.backgroundColor, justifyContent: 'center',
cursor: 'default', backgroundColor: theme.backgroundColor,
...style, cursor: 'default',
}; },
style
);
return ( return (
<a <a

View File

@@ -1,4 +1,7 @@
import * as React from 'react'; import * as React from 'react';
import Logger from '@joplin/lib/Logger';
const logger = Logger.create('ItemList');
interface Props { interface Props {
style: any; style: any;
@@ -47,6 +50,15 @@ class ItemList extends React.Component<Props, State> {
let bottomItemIndex = topItemIndex + (visibleItemCount - 1); let bottomItemIndex = topItemIndex + (visibleItemCount - 1);
if (bottomItemIndex >= props.items.length) bottomItemIndex = props.items.length - 1; if (bottomItemIndex >= props.items.length) bottomItemIndex = props.items.length - 1;
// EDGE CASE:
// ref: https://github.com/laurent22/joplin/issues/4124
// when the note list is hidden, visibleItemCount is negative, and scroll top is positive when a note is selected
if (visibleItemCount < 0 && this.scrollTop_ > 0) {
logger.warn('Resetting scrollTop to 0. visibleItemCount is negative, scrollTop is positive.');
// we will reset the scroll top so that there is no blank space at the top of note list
this.scrollTop_ = 0;
}
this.setState({ this.setState({
topItemIndex: topItemIndex, topItemIndex: topItemIndex,
bottomItemIndex: bottomItemIndex, bottomItemIndex: bottomItemIndex,
@@ -69,6 +81,21 @@ class ItemList extends React.Component<Props, State> {
this.updateStateItemIndexes(newProps); this.updateStateItemIndexes(newProps);
} }
public componentDidUpdate(): void {
// EDGE CASE
// scroll top is not updated when item list visibility is toggled
// if the user was at the bottom of the item list before hiding, blank spaces are added at the bottom of the item list
if (this.offsetScroll() !== this.listRef.current?.scrollTop) {
logger.warn(`scrollTop mismatch. Updating scrollTop with current listRef scrollTop(${this.listRef.current.scrollTop})`);
// update scroll postion once if there is a mismatch in scroll position after showing item list
this.onScroll({
target: {
scrollTop: this.listRef.current.scrollTop,
},
});
}
}
public onScroll(event: any) { public onScroll(event: any) {
this.scrollTop_ = event.target.scrollTop; this.scrollTop_ = event.target.scrollTop;
this.updateStateItemIndexes(); this.updateStateItemIndexes();
@@ -121,8 +148,10 @@ class ItemList extends React.Component<Props, State> {
public render() { public render() {
const items = this.props.items; const items = this.props.items;
const style = { ...this.props.style, overflowX: 'hidden', const style = Object.assign({}, this.props.style, {
overflowY: 'auto' }; overflowX: 'hidden',
overflowY: 'auto',
});
// if (this.props.disabled) style.opacity = 0.5; // if (this.props.disabled) style.opacity = 0.5;

View File

@@ -501,14 +501,16 @@ class MainScreenComponent extends React.Component<Props, State> {
height: height, height: height,
}; };
this.styles_.modalLayer = { ...theme.textStyle, zIndex: 10000, this.styles_.modalLayer = Object.assign({}, theme.textStyle, {
zIndex: 10000,
position: 'absolute', position: 'absolute',
top: 0, top: 0,
left: 0, left: 0,
backgroundColor: theme.backgroundColor, backgroundColor: theme.backgroundColor,
width: width - 20, width: width - 20,
height: height - 20, height: height - 20,
padding: 10 }; padding: 10,
});
return this.styles_; return this.styles_;
} }
@@ -801,11 +803,13 @@ class MainScreenComponent extends React.Component<Props, State> {
public render() { public render() {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const style = { const style = Object.assign(
color: theme.color, {
backgroundColor: theme.backgroundColor, color: theme.color,
...this.props.style, backgroundColor: theme.backgroundColor,
}; },
this.props.style
);
const promptOptions = this.state.promptOptions; const promptOptions = this.state.promptOptions;
const styles = this.styles(this.props.themeId, style.width, style.height, this.messageBoxVisible()); const styles = this.styles(this.props.themeId, style.width, style.height, this.messageBoxVisible());
@@ -820,7 +824,7 @@ class MainScreenComponent extends React.Component<Props, State> {
const dialogInfo = PluginManager.instance().pluginDialogToShow(this.props.pluginsLegacy); const dialogInfo = PluginManager.instance().pluginDialogToShow(this.props.pluginsLegacy);
const pluginDialog = !dialogInfo ? null : <dialogInfo.Dialog {...dialogInfo.props} />; const pluginDialog = !dialogInfo ? null : <dialogInfo.Dialog {...dialogInfo.props} />;
const modalLayerStyle = { ...styles.modalLayer, display: this.state.modalLayer.visible ? 'block' : 'none' }; const modalLayerStyle = Object.assign({}, styles.modalLayer, { display: this.state.modalLayer.visible ? 'block' : 'none' });
const notePropertiesDialogOptions = this.state.notePropertiesDialogOptions; const notePropertiesDialogOptions = this.state.notePropertiesDialogOptions;
const noteContentPropertiesDialogOptions = this.state.noteContentPropertiesDialogOptions; const noteContentPropertiesDialogOptions = this.state.noteContentPropertiesDialogOptions;
@@ -862,7 +866,6 @@ class MainScreenComponent extends React.Component<Props, State> {
const mapStateToProps = (state: AppState) => { const mapStateToProps = (state: AppState) => {
const syncInfo = localSyncInfoFromState(state); const syncInfo = localSyncInfoFromState(state);
const showNeedUpgradingEnabledMasterKeyMessage = !!EncryptionService.instance().masterKeysThatNeedUpgrading(syncInfo.masterKeys.filter((k) => !!k.enabled)).length;
return { return {
themeId: state.settings.theme, themeId: state.settings.theme,
@@ -870,7 +873,7 @@ const mapStateToProps = (state: AppState) => {
hasDisabledSyncItems: state.hasDisabledSyncItems, hasDisabledSyncItems: state.hasDisabledSyncItems,
hasDisabledEncryptionItems: state.hasDisabledEncryptionItems, hasDisabledEncryptionItems: state.hasDisabledEncryptionItems,
showMissingMasterKeyMessage: showMissingMasterKeyMessage(syncInfo, state.notLoadedMasterKeys), showMissingMasterKeyMessage: showMissingMasterKeyMessage(syncInfo, state.notLoadedMasterKeys),
showNeedUpgradingMasterKeyMessage: showNeedUpgradingEnabledMasterKeyMessage, showNeedUpgradingMasterKeyMessage: !!EncryptionService.instance().masterKeysThatNeedUpgrading(syncInfo.masterKeys).length,
showShouldReencryptMessage: state.settings['encryption.shouldReencrypt'] >= Setting.SHOULD_REENCRYPT_YES, showShouldReencryptMessage: state.settings['encryption.shouldReencrypt'] >= Setting.SHOULD_REENCRYPT_YES,
shouldUpgradeSyncTarget: state.settings['sync.upgradeState'] === Setting.SYNC_UPGRADE_STATE_SHOULD_DO, shouldUpgradeSyncTarget: state.settings['sync.upgradeState'] === Setting.SYNC_UPGRADE_STATE_SHOULD_DO,
pluginsLegacy: state.pluginsLegacy, pluginsLegacy: state.pluginsLegacy,

View File

@@ -22,7 +22,7 @@ export const runtime = (comp: any): CommandRuntime => {
const { newConfig, newProfile } = createNewProfile(context.state.profileConfig, answer); const { newConfig, newProfile } = createNewProfile(context.state.profileConfig, answer);
newConfig.currentProfileId = newProfile.id; newConfig.currentProfileId = newProfile.id;
await saveProfileConfig(`${Setting.value('rootProfileDir')}/profiles.json`, newConfig); await saveProfileConfig(`${Setting.value('rootProfileDir')}/profiles.json`, newConfig);
await restart(); await restart(false);
} }
comp.setState({ promptOptions: null }); comp.setState({ promptOptions: null });

View File

@@ -17,9 +17,11 @@ export const runtime = (): CommandRuntime => {
const defaultValues = Note.previewFieldsWithDefaultValues({ includeTimestamps: false }); const defaultValues = Note.previewFieldsWithDefaultValues({ includeTimestamps: false });
let newNote = { ...defaultValues, parent_id: folderId, let newNote = Object.assign({}, defaultValues, {
parent_id: folderId,
is_todo: isTodo ? 1 : 0, is_todo: isTodo ? 1 : 0,
body: body }; body: body,
});
newNote = await Note.save(newNote, { provisional: true }); newNote = await Note.save(newNote, { provisional: true });

View File

@@ -3,7 +3,7 @@ import { useState, useEffect } from 'react';
import { _ } from '@joplin/lib/locale'; import { _ } from '@joplin/lib/locale';
import DialogButtonRow from './DialogButtonRow'; import DialogButtonRow from './DialogButtonRow';
const { themeStyle } = require('@joplin/lib/theme'); const { themeStyle } = require('@joplin/lib/theme');
const Countable = require('@joplin/lib/countable/Countable'); const Countable = require('countable');
import markupLanguageUtils from '../utils/markupLanguageUtils'; import markupLanguageUtils from '../utils/markupLanguageUtils';
interface NoteContentPropertiesDialogProps { interface NoteContentPropertiesDialogProps {

View File

@@ -31,7 +31,7 @@ export default function useExternalPlugins(CodeMirror: any, plugins: PluginState
} }
if (mod.codeMirrorOptions) { if (mod.codeMirrorOptions) {
newOptions = { ...newOptions, ...mod.codeMirrorOptions }; newOptions = Object.assign({}, newOptions, mod.codeMirrorOptions);
} }
if (mod.assets) { if (mod.assets) {

View File

@@ -1544,7 +1544,7 @@
} }
}); });
editor.addCommand('InsertJoplinChecklist', function (ui, detail) { editor.addCommand('InsertJoplinChecklist', function (ui, detail) {
detail = { ...detail, listType: 'joplinChecklist' }; detail = Object.assign({}, detail, { listType: 'joplinChecklist' });
ToggleList.toggleList(editor, 'UL', detail); ToggleList.toggleList(editor, 'UL', detail);
}); });
} }

View File

@@ -364,11 +364,13 @@ function NoteEditor(props: NoteEditorProps) {
}, [props.dispatch, formNote]); }, [props.dispatch, formNote]);
function renderNoNotes(rootStyle: any) { function renderNoNotes(rootStyle: any) {
const emptyDivStyle = { const emptyDivStyle = Object.assign(
backgroundColor: 'black', {
opacity: 0.1, backgroundColor: 'black',
...rootStyle, opacity: 0.1,
}; },
rootStyle
);
return <div style={emptyDivStyle}></div>; return <div style={emptyDivStyle}></div>;
} }
@@ -629,7 +631,7 @@ const mapStateToProps = (state: AppState) => {
], whenClauseContext)[0], ], whenClauseContext)[0],
contentMaxWidth: state.settings['style.editor.contentMaxWidth'], contentMaxWidth: state.settings['style.editor.contentMaxWidth'],
isSafeMode: state.settings.isSafeMode, isSafeMode: state.settings.isSafeMode,
useCustomPdfViewer: false, // state.settings.useCustomPdfViewer, useCustomPdfViewer: state.settings.useCustomPdfViewer,
}; };
}; };

View File

@@ -59,12 +59,14 @@ export default function useMarkupToHtml(deps: HookDependencies) {
delete options.replaceResourceInternalToExternalLinks; delete options.replaceResourceInternalToExternalLinks;
const result = await markupToHtml.render(markupLanguage, md, theme, { codeTheme: theme.codeThemeCss, const result = await markupToHtml.render(markupLanguage, md, theme, Object.assign({}, {
codeTheme: theme.codeThemeCss,
resources: resources, resources: resources,
postMessageSyntax: 'ipcProxySendToHost', postMessageSyntax: 'ipcProxySendToHost',
splitted: true, splitted: true,
externalAssetsOnly: true, externalAssetsOnly: true,
codeHighlightCacheKey: 'useMarkupToHtml', ...options }); codeHighlightCacheKey: 'useMarkupToHtml',
}, options));
return result; return result;
// eslint-disable-next-line @seiyab/react-hooks/exhaustive-deps -- Old code before rule was applied // eslint-disable-next-line @seiyab/react-hooks/exhaustive-deps -- Old code before rule was applied

View File

@@ -62,7 +62,7 @@ export default function useNoteSearchBar({ noteSearchBarRef }: UseNoteSearchBarP
const noteSearchBarNextPrevious = useCallback((inc: number) => { const noteSearchBarNextPrevious = useCallback((inc: number) => {
setLocalSearch((prev: LocalSearch) => { setLocalSearch((prev: LocalSearch) => {
const ls = { ...prev }; const ls = Object.assign({}, prev);
ls.selectedIndex += inc; ls.selectedIndex += inc;
ls.timestamp = Date.now(); ls.timestamp = Date.now();
if (ls.selectedIndex < 0) ls.selectedIndex = ls.resultCount - 1; if (ls.selectedIndex < 0) ls.selectedIndex = ls.resultCount - 1;

View File

@@ -181,7 +181,7 @@ function NoteListControls(props: Props) {
useEffect(() => { useEffect(() => {
if (breakpoint === dynamicBreakpoints.Xl) { if (breakpoint === dynamicBreakpoints.Xl) {
noteControlsRef.current.style.flexDirection = 'row'; noteControlsRef.current.style.flexDirection = 'row';
searchAndSortRef.current.style.flex = '2 1 50%'; searchAndSortRef.current.style.flex = '2 1 auto';
props.onContentHeightChange(true); props.onContentHeightChange(true);
} else { } else {
noteControlsRef.current.style.flexDirection = 'column'; noteControlsRef.current.style.flexDirection = 'column';

View File

@@ -108,10 +108,10 @@ function NoteListItem(props: NoteListItemProps, ref: any) {
); );
} }
let listItemTitleStyle = { ...props.style.listItemTitle }; let listItemTitleStyle = Object.assign({}, props.style.listItemTitle);
listItemTitleStyle.paddingLeft = !item.is_todo ? hPadding : 4; listItemTitleStyle.paddingLeft = !item.is_todo ? hPadding : 4;
if (item.is_shared) listItemTitleStyle.color = theme.colorWarn3; if (item.is_shared) listItemTitleStyle.color = theme.colorWarn3;
if (item.is_todo && !!item.todo_completed) listItemTitleStyle = { ...listItemTitleStyle, ...props.style.listItemTitleCompleted }; if (item.is_todo && !!item.todo_completed) listItemTitleStyle = Object.assign(listItemTitleStyle, props.style.listItemTitleCompleted);
const displayTitle = Note.displayTitle(item); const displayTitle = Note.displayTitle(item);
let titleComp = null; let titleComp = null;

View File

@@ -114,7 +114,7 @@ class NotePropertiesDialog extends React.Component<Props, State> {
} }
public formNoteToNote(formNote: any) { public formNoteToNote(formNote: any) {
const note = { id: formNote.id, ...this.latLongFromLocation(formNote.location) }; const note = Object.assign({ id: formNote.id }, this.latLongFromLocation(formNote.location));
note.user_created_time = time.formatLocalToMs(formNote.user_created_time); note.user_created_time = time.formatLocalToMs(formNote.user_created_time);
note.user_updated_time = time.formatLocalToMs(formNote.user_updated_time); note.user_updated_time = time.formatLocalToMs(formNote.user_updated_time);
@@ -211,7 +211,7 @@ class NotePropertiesDialog extends React.Component<Props, State> {
if (!this.state.editedKey) return; if (!this.state.editedKey) return;
return new Promise((resolve: Function) => { return new Promise((resolve: Function) => {
const newFormNote = { ...this.state.formNote }; const newFormNote = Object.assign({}, this.state.formNote);
if (this.state.editedKey.indexOf('_time') >= 0) { if (this.state.editedKey.indexOf('_time') >= 0) {
const dt = time.anythingToDateTime(this.state.editedValue, new Date()); const dt = time.anythingToDateTime(this.state.editedValue, new Date());
@@ -248,7 +248,7 @@ class NotePropertiesDialog extends React.Component<Props, State> {
public createNoteField(key: string, value: any) { public createNoteField(key: string, value: any) {
const styles = this.styles(this.props.themeId); const styles = this.styles(this.props.themeId);
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const labelComp = <label style={{ ...theme.textStyle, ...theme.controlBoxLabel }}>{this.formatLabel(key)}</label>; const labelComp = <label style={Object.assign({}, theme.textStyle, theme.controlBoxLabel)}>{this.formatLabel(key)}</label>;
let controlComp = null; let controlComp = null;
let editComp = null; let editComp = null;
let editCompHandler = null; let editCompHandler = null;
@@ -317,7 +317,7 @@ class NotePropertiesDialog extends React.Component<Props, State> {
const ll = this.latLongFromLocation(value); const ll = this.latLongFromLocation(value);
url = Note.geoLocationUrlFromLatLong(ll.latitude, ll.longitude); url = Note.geoLocationUrlFromLatLong(ll.latitude, ll.longitude);
} }
const urlStyle = { ...theme.urlStyle, maxWidth: '180px', overflow: 'hidden', whiteSpace: 'nowrap', textOverflow: 'ellipsis' }; const urlStyle = Object.assign({}, theme.urlStyle, { maxWidth: '180px', overflow: 'hidden', whiteSpace: 'nowrap', textOverflow: 'ellipsis' });
controlComp = ( controlComp = (
<a href="#" onClick={() => bridge().openExternal(url)} style={urlStyle}> <a href="#" onClick={() => bridge().openExternal(url)} style={urlStyle}>
{displayedValue} {displayedValue}
@@ -330,7 +330,7 @@ class NotePropertiesDialog extends React.Component<Props, State> {
</a> </a>
); );
} else { } else {
controlComp = <div style={{ ...theme.textStyle, ...theme.controlBoxValue }}>{displayedValue}</div>; controlComp = <div style={Object.assign({}, theme.textStyle, theme.controlBoxValue)}>{displayedValue}</div>;
} }
if (['id', 'revisionsLink', 'markup_language'].indexOf(key) < 0) { if (['id', 'revisionsLink', 'markup_language'].indexOf(key) < 0) {

View File

@@ -67,8 +67,8 @@ class NoteRevisionViewerComponent extends React.PureComponent<Props, State> {
flex: 1, flex: 1,
flexDirection: 'column', flexDirection: 'column',
}, },
titleInput: { ...theme.inputStyle, flex: 1 }, titleInput: Object.assign({}, theme.inputStyle, { flex: 1 }),
revisionList: { ...theme.dropdownList, marginLeft: 10, flex: 0.5 }, revisionList: Object.assign({}, theme.dropdownList, { marginLeft: 10, flex: 0.5 }),
}; };
return style; return style;
@@ -205,14 +205,14 @@ class NoteRevisionViewerComponent extends React.PureComponent<Props, State> {
const titleInput = ( const titleInput = (
<div style={{ display: 'flex', flexDirection: 'row', alignItems: 'center', marginBottom: 10, borderWidth: 1, borderBottomStyle: 'solid', borderColor: theme.dividerColor, paddingBottom: 10 }}> <div style={{ display: 'flex', flexDirection: 'row', alignItems: 'center', marginBottom: 10, borderWidth: 1, borderBottomStyle: 'solid', borderColor: theme.dividerColor, paddingBottom: 10 }}>
<button onClick={this.backButton_click} style={{ ...theme.buttonStyle, marginRight: 10, height: theme.inputStyle.height }}> <button onClick={this.backButton_click} style={Object.assign({}, theme.buttonStyle, { marginRight: 10, height: theme.inputStyle.height })}>
<i style={theme.buttonIconStyle} className={'fa fa-chevron-left'}></i>{_('Back')} <i style={theme.buttonIconStyle} className={'fa fa-chevron-left'}></i>{_('Back')}
</button> </button>
<input readOnly type="text" style={style.titleInput} value={this.state.note ? this.state.note.title : ''} /> <input readOnly type="text" style={style.titleInput} value={this.state.note ? this.state.note.title : ''} />
<select disabled={!this.state.revisions.length} value={this.state.currentRevId} style={style.revisionList} onChange={this.revisionList_onChange}> <select disabled={!this.state.revisions.length} value={this.state.currentRevId} style={style.revisionList} onChange={this.revisionList_onChange}>
{revisionListItems} {revisionListItems}
</select> </select>
<button disabled={!this.state.revisions.length || this.state.restoring} onClick={this.importButton_onClick} style={{ ...theme.buttonStyle, marginLeft: 10, height: theme.inputStyle.height }}> <button disabled={!this.state.revisions.length || this.state.restoring} onClick={this.importButton_onClick} style={Object.assign({}, theme.buttonStyle, { marginLeft: 10, height: theme.inputStyle.height })}>
{restoreButtonTitle} {restoreButtonTitle}
</button> </button>
<HelpButton tip={helpMessage} id="noteRevisionHelpButton" onClick={this.helpButton_onClick} /> <HelpButton tip={helpMessage} id="noteRevisionHelpButton" onClick={this.helpButton_onClick} />

View File

@@ -37,8 +37,10 @@ class NoteSearchBar extends React.Component<Props> {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const style = { const style = {
root: { ...theme.textStyle, backgroundColor: theme.backgroundColor, root: Object.assign({}, theme.textStyle, {
color: theme.colorFaded }, backgroundColor: theme.backgroundColor,
color: theme.colorFaded,
}),
}; };
return style; return style;
@@ -148,10 +150,12 @@ class NoteSearchBar extends React.Component<Props> {
const previousButton = this.buttonIconComponent('fa-chevron-up', this.previousButton_click, buttonEnabled); const previousButton = this.buttonIconComponent('fa-chevron-up', this.previousButton_click, buttonEnabled);
const nextButton = this.buttonIconComponent('fa-chevron-down', this.nextButton_click, buttonEnabled); const nextButton = this.buttonIconComponent('fa-chevron-down', this.nextButton_click, buttonEnabled);
const textStyle = { fontSize: theme.fontSize, const textStyle = Object.assign({
fontSize: theme.fontSize,
fontFamily: theme.fontFamily, fontFamily: theme.fontFamily,
color: theme.colorFaded, color: theme.colorFaded,
backgroundColor: theme.backgroundColor }; backgroundColor: theme.backgroundColor,
});
const matchesFoundString = (query.length > 0) ? ( const matchesFoundString = (query.length > 0) ? (
<div style={textStyle}> <div style={textStyle}>

View File

@@ -15,8 +15,10 @@ class NoteStatusBarComponent extends React.Component<Props> {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const style = { const style = {
root: { ...theme.textStyle, backgroundColor: theme.backgroundColor, root: Object.assign({}, theme.textStyle, {
color: theme.colorFaded }, backgroundColor: theme.backgroundColor,
color: theme.colorFaded,
}),
}; };
return style; return style;

View File

@@ -173,7 +173,7 @@ export default class NoteTextViewerComponent extends React.Component<Props, any>
// ---------------------------------------------------------------- // ----------------------------------------------------------------
public render() { public render() {
const viewerStyle = { border: 'none', ...this.props.viewerStyle }; const viewerStyle = Object.assign({}, { border: 'none' }, this.props.viewerStyle);
return <iframe className="noteTextViewer" ref={this.webviewRef_} style={viewerStyle} src="gui/note-viewer/index.html"></iframe>; return <iframe className="noteTextViewer" ref={this.webviewRef_} style={viewerStyle} src="gui/note-viewer/index.html"></iframe>;
} }
} }

View File

@@ -26,15 +26,11 @@ export default class PromptDialog extends React.Component<Props, any> {
private focusInput_: boolean; private focusInput_: boolean;
private styles_: any; private styles_: any;
private styleKey_: string; private styleKey_: string;
private menuIsOpened_: boolean = false;
public constructor(props: Props) { public constructor(props: Props) {
super(props); super(props);
this.answerInput_ = React.createRef(); this.answerInput_ = React.createRef();
this.select_menuOpen = this.select_menuOpen.bind(this);
this.select_menuClose = this.select_menuClose.bind(this);
} }
public UNSAFE_componentWillMount() { public UNSAFE_componentWillMount() {
@@ -43,7 +39,6 @@ export default class PromptDialog extends React.Component<Props, any> {
answer: this.props.defaultValue ? this.props.defaultValue : '', answer: this.props.defaultValue ? this.props.defaultValue : '',
}); });
this.focusInput_ = true; this.focusInput_ = true;
this.menuIsOpened_ = false;
} }
public UNSAFE_componentWillReceiveProps(newProps: Props) { public UNSAFE_componentWillReceiveProps(newProps: Props) {
@@ -57,14 +52,6 @@ export default class PromptDialog extends React.Component<Props, any> {
} }
} }
private select_menuOpen() {
this.menuIsOpened_ = true;
}
private select_menuClose() {
this.menuIsOpened_ = false;
}
public componentDidUpdate() { public componentDidUpdate() {
if (this.focusInput_ && this.answerInput_.current) this.answerInput_.current.focus(); if (this.focusInput_ && this.answerInput_.current) this.answerInput_.current.focus();
this.focusInput_ = false; this.focusInput_ = false;
@@ -132,49 +119,43 @@ export default class PromptDialog extends React.Component<Props, any> {
}; };
this.styles_.select = { this.styles_.select = {
control: (provided: any) => { control: (provided: any) =>
return { ...provided, Object.assign(provided, {
minWidth: width * 0.2, minWidth: width * 0.2,
maxWidth: width * 0.5, maxWidth: width * 0.5,
fontFamily: theme.fontFamily, fontFamily: theme.fontFamily,
}; }),
}, input: (provided: any) =>
input: (provided: any) => { Object.assign(provided, {
return { ...provided,
minWidth: '20px', minWidth: '20px',
color: theme.color, color: theme.color,
}; }),
}, menu: (provided: any) =>
menu: (provided: any) => { Object.assign(provided, {
return { ...provided,
color: theme.color, color: theme.color,
fontFamily: theme.fontFamily, fontFamily: theme.fontFamily,
backgroundColor: theme.backgroundColor, backgroundColor: theme.backgroundColor,
}; }),
}, option: (provided: any, state: any) =>
option: (provided: any, state: any) => { Object.assign(provided, {
return { ...provided,
color: theme.color, color: theme.color,
fontFamily: theme.fontFamily, fontFamily: theme.fontFamily,
paddingLeft: `${10 + (state.data.indentDepth || 0) * 20}px`, paddingLeft: `${10 + (state.data.indentDepth || 0) * 20}px`,
}; }),
}, multiValueLabel: (provided: any) =>
multiValueLabel: (provided: any) => { Object.assign(provided, {
return { ...provided,
fontFamily: theme.fontFamily, fontFamily: theme.fontFamily,
}; }),
}, multiValueRemove: (provided: any) =>
multiValueRemove: (provided: any) => { Object.assign(provided, {
return { ...provided,
color: theme.color, color: theme.color,
}; }),
},
}; };
this.styles_.selectTheme = (tagTheme: any) => { this.styles_.selectTheme = (tagTheme: any) =>
return { ...tagTheme, Object.assign(tagTheme, {
borderRadius: 2, borderRadius: 2,
colors: { ...tagTheme.colors, colors: Object.assign(tagTheme.colors, {
primary: theme.raisedBackgroundColor, primary: theme.raisedBackgroundColor,
primary25: theme.raisedBackgroundColor, primary25: theme.raisedBackgroundColor,
neutral0: theme.backgroundColor, neutral0: theme.backgroundColor,
@@ -190,11 +171,12 @@ export default class PromptDialog extends React.Component<Props, any> {
neutral90: theme.color, neutral90: theme.color,
danger: theme.backgroundColor, danger: theme.backgroundColor,
dangerLight: theme.colorError2, dangerLight: theme.colorError2,
}, }),
}; });
};
this.styles_.desc = { ...theme.textStyle, marginTop: 10 }; this.styles_.desc = Object.assign({}, theme.textStyle, {
marginTop: 10,
});
return this.styles_; return this.styles_;
} }
@@ -242,14 +224,16 @@ export default class PromptDialog extends React.Component<Props, any> {
const onKeyDown = (event: any) => { const onKeyDown = (event: any) => {
if (event.key === 'Enter') { if (event.key === 'Enter') {
// If the dropdown is open, we don't close the dialog - instead if (this.props.inputType === 'tags' || this.props.inputType === 'dropdown') {
// the currently item will be selcted. If it is closed however
// we confirm the dialog.
if ((this.props.inputType === 'tags' || this.props.inputType === 'dropdown') && this.menuIsOpened_) {
// Do nothing // Do nothing
} else { } else {
onClose(true); onClose(true);
} }
// } else if (this.answerInput_.current && !this.answerInput_.current.state.menuIsOpen) {
// // The menu will be open if the user is selecting a new item
// onClose(true);
// }
} else if (event.key === 'Escape') { } else if (event.key === 'Escape') {
onClose(false); onClose(false);
} }
@@ -262,9 +246,9 @@ export default class PromptDialog extends React.Component<Props, any> {
if (this.props.inputType === 'datetime') { if (this.props.inputType === 'datetime') {
inputComp = <Datetime className="datetime-picker" value={this.state.answer} inputProps={{ style: styles.input }} dateFormat={time.dateFormat()} timeFormat={time.timeFormat()} onChange={(momentObject: any) => onDateTimeChange(momentObject)} />; inputComp = <Datetime className="datetime-picker" value={this.state.answer} inputProps={{ style: styles.input }} dateFormat={time.dateFormat()} timeFormat={time.timeFormat()} onChange={(momentObject: any) => onDateTimeChange(momentObject)} />;
} else if (this.props.inputType === 'tags') { } else if (this.props.inputType === 'tags') {
inputComp = <CreatableSelect className="tag-selector" onMenuOpen={this.select_menuOpen} onMenuClose={this.select_menuClose} styles={styles.select} theme={styles.selectTheme} ref={this.answerInput_} value={this.state.answer} placeholder="" components={makeAnimated()} isMulti={true} isClearable={false} backspaceRemovesValue={true} options={this.props.autocomplete} onChange={onSelectChange} onKeyDown={(event: any) => onKeyDown(event)} />; inputComp = <CreatableSelect className="tag-selector" styles={styles.select} theme={styles.selectTheme} ref={this.answerInput_} value={this.state.answer} placeholder="" components={makeAnimated()} isMulti={true} isClearable={false} backspaceRemovesValue={true} options={this.props.autocomplete} onChange={onSelectChange} onKeyDown={(event: any) => onKeyDown(event)} />;
} else if (this.props.inputType === 'dropdown') { } else if (this.props.inputType === 'dropdown') {
inputComp = <Select className="item-selector" onMenuOpen={this.select_menuOpen} onMenuClose={this.select_menuClose} styles={styles.select} theme={styles.selectTheme} ref={this.answerInput_} components={makeAnimated()} value={this.props.answer} defaultValue={this.props.defaultValue} isClearable={false} options={this.props.autocomplete} onChange={onSelectChange} onKeyDown={(event: any) => onKeyDown(event)} />; inputComp = <Select className="item-selector" styles={styles.select} theme={styles.selectTheme} ref={this.answerInput_} components={makeAnimated()} value={this.props.answer} defaultValue={this.props.defaultValue} isClearable={false} options={this.props.autocomplete} onChange={onSelectChange} onKeyDown={(event: any) => onKeyDown(event)} />;
} else { } else {
inputComp = <input style={styles.input} ref={this.answerInput_} value={this.state.answer} type="text" onChange={event => onChange(event)} onKeyDown={event => onKeyDown(event)} />; inputComp = <input style={styles.input} ref={this.answerInput_} value={this.state.answer} type="text" onChange={event => onChange(event)} onKeyDown={event => onKeyDown(event)} />;
} }

View File

@@ -27,7 +27,7 @@ import StyleSheetContainer from './StyleSheets/StyleSheetContainer';
import ImportScreen from './ImportScreen'; import ImportScreen from './ImportScreen';
const { ResourceScreen } = require('./ResourceScreen.js'); const { ResourceScreen } = require('./ResourceScreen.js');
import Navigator from './Navigator'; import Navigator from './Navigator';
import WelcomeUtils from '@joplin/lib/WelcomeUtils'; const WelcomeUtils = require('@joplin/lib/WelcomeUtils');
const { ThemeProvider, StyleSheetManager, createGlobalStyle } = require('styled-components'); const { ThemeProvider, StyleSheetManager, createGlobalStyle } = require('styled-components');
const bridge = require('@electron/remote').require('./bridge').default; const bridge = require('@electron/remote').require('./bridge').default;
@@ -141,7 +141,7 @@ class RootComponent extends React.Component<Props, any> {
}); });
} }
await WelcomeUtils.install(Setting.value('locale'), this.props.dispatch); await WelcomeUtils.install(this.props.dispatch);
} }
private renderModalMessage(props: ModalDialogProps) { private renderModalMessage(props: ModalDialogProps) {

View File

@@ -1,6 +1,6 @@
import * as React from 'react'; import * as React from 'react';
import { useEffect, useRef, useCallback, useMemo } from 'react'; import { useEffect, useRef, useCallback, useMemo } from 'react';
import styled, { css } from 'styled-components'; import styled from 'styled-components';
import shim from '@joplin/lib/shim'; import shim from '@joplin/lib/shim';
import { StyledRoot, StyledAddButton, StyledShareIcon, StyledHeader, StyledHeaderIcon, StyledAllNotesIcon, StyledHeaderLabel, StyledListItem, StyledListItemAnchor, StyledExpandLink, StyledNoteCount, StyledSyncReportText, StyledSyncReport, StyledSynchronizeButton } from './styles'; import { StyledRoot, StyledAddButton, StyledShareIcon, StyledHeader, StyledHeaderIcon, StyledAllNotesIcon, StyledHeaderLabel, StyledListItem, StyledListItemAnchor, StyledExpandLink, StyledNoteCount, StyledSyncReportText, StyledSyncReport, StyledSynchronizeButton } from './styles';
import { ButtonLevel } from '../Button/Button'; import { ButtonLevel } from '../Button/Button';
@@ -40,14 +40,23 @@ const { clipboard } = require('electron');
const logger = Logger.create('Sidebar'); const logger = Logger.create('Sidebar');
// Workaround sidebar rendering bug on Linux Intel GPU. const StyledFoldersHolder = styled.div`
// https://github.com/laurent22/joplin/issues/7506 // linux bug: https://github.com/laurent22/joplin/issues/7506#issuecomment-1447101057
const StyledSpanFix = styled.span` & a.list-item {
${shim.isLinux() && css` ${shim.isLinux() && {
position: relative; opacity: 1,
`} }}
}
`;
const TagsHolder = styled.div`
// linux bug: https://github.com/laurent22/joplin/issues/8000
// solution ref: https://github.com/laurent22/joplin/issues/7506#issuecomment-1447101057
& a.list-item {
${shim.isLinux() && {
opacity: 1,
}}
}
`; `;
interface Props { interface Props {
themeId: number; themeId: number;
@@ -129,7 +138,7 @@ function FolderItem(props: any) {
}} }}
onDoubleClick={onFolderToggleClick_} onDoubleClick={onFolderToggleClick_}
> >
{showFolderIcon ? renderFolderIcon(folderIcon) : null}<StyledSpanFix className="title" style={{ lineHeight: 0 }}>{folderTitle}</StyledSpanFix> {showFolderIcon ? renderFolderIcon(folderIcon) : null}<span className="title" style={{ lineHeight: 0 }}>{folderTitle}</span>
{shareIcon} {noteCountComp} {shareIcon} {noteCountComp}
</StyledListItemAnchor> </StyledListItemAnchor>
</StyledListItem> </StyledListItem>
@@ -564,7 +573,7 @@ const SidebarComponent = (props: Props) => {
tagItem_click(tag); tagItem_click(tag);
}} }}
> >
<StyledSpanFix className="tag-label">{Tag.displayTitle(tag)}</StyledSpanFix> <span className="tag-label">{Tag.displayTitle(tag)}</span>
{noteCount} {noteCount}
</StyledListItemAnchor> </StyledListItemAnchor>
</StyledListItem> </StyledListItem>
@@ -716,13 +725,13 @@ const SidebarComponent = (props: Props) => {
const folderItems = [renderAllNotesItem(theme, allNotesSelected)].concat(result.items); const folderItems = [renderAllNotesItem(theme, allNotesSelected)].concat(result.items);
folderItemsOrder_.current = result.order; folderItemsOrder_.current = result.order;
items.push( items.push(
<div <StyledFoldersHolder
className={`folders ${props.folderHeaderIsExpanded ? 'expanded' : ''}`} className={`folders ${props.folderHeaderIsExpanded ? 'expanded' : ''}`}
key="folder_items" key="folder_items"
style={foldersStyle} style={foldersStyle}
> >
{folderItems} {folderItems}
</div> </StyledFoldersHolder>
); );
} }
@@ -738,9 +747,9 @@ const SidebarComponent = (props: Props) => {
tagItemsOrder_.current = result.order; tagItemsOrder_.current = result.order;
items.push( items.push(
<div className="tags" key="tag_items" style={{ display: props.tagHeaderIsExpanded ? 'block' : 'none' }}> <TagsHolder className="tags" key="tag_items" style={{ display: props.tagHeaderIsExpanded ? 'block' : 'none' }}>
{tagItems} {tagItems}
</div> </TagsHolder>
); );
} }

View File

@@ -56,13 +56,15 @@ function StatusScreen(props: Props) {
flexDirection: 'column', flexDirection: 'column',
}; };
const retryStyle = { ...theme.urlStyle, marginLeft: 5 }; const retryStyle = Object.assign({}, theme.urlStyle, { marginLeft: 5 });
const retryAllStyle = { ...theme.urlStyle, marginTop: 5, display: 'inline-block' }; const retryAllStyle = Object.assign({}, theme.urlStyle, { marginTop: 5, display: 'inline-block' });
const containerPadding = theme.configScreenPadding; const containerPadding = theme.configScreenPadding;
const containerStyle = { ...theme.containerStyle, padding: containerPadding, const containerStyle = Object.assign({}, theme.containerStyle, {
flex: 1 }; padding: containerPadding,
flex: 1,
});
function renderSectionTitleHtml(key: string, title: string) { function renderSectionTitleHtml(key: string, title: string) {
return ( return (

View File

@@ -7,7 +7,7 @@ import { AppState } from '../app.reducer';
class TagItemComponent extends React.Component { class TagItemComponent extends React.Component {
public render() { public render() {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const style = { ...theme.tagStyle }; const style = Object.assign({}, theme.tagStyle);
const { title, id } = this.props; const { title, id } = this.props;
return <button style={style} onClick={() => CommandService.instance().execute('openTag', id)}>{title}</button>; return <button style={style} onClick={() => CommandService.instance().execute('openTag', id)}>{title}</button>;

View File

@@ -16,12 +16,14 @@ class ToolbarBaseComponent extends React.Component<Props, any> {
public render() { public render() {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const style: any = { display: 'flex', const style: any = Object.assign({
display: 'flex',
flexDirection: 'row', flexDirection: 'row',
boxSizing: 'border-box', boxSizing: 'border-box',
backgroundColor: theme.backgroundColor3, backgroundColor: theme.backgroundColor3,
padding: theme.toolbarPadding, padding: theme.toolbarPadding,
paddingRight: theme.mainPadding, ...this.props.style }; paddingRight: theme.mainPadding,
}, this.props.style);
const groupStyle: any = { const groupStyle: any = {
display: 'flex', display: 'flex',
@@ -43,11 +45,13 @@ class ToolbarBaseComponent extends React.Component<Props, any> {
if (!key) key = `${o.type}_${i}`; if (!key) key = `${o.type}_${i}`;
const props = { const props = Object.assign(
key: key, {
themeId: this.props.themeId, key: key,
...o, themeId: this.props.themeId,
}; },
o
);
if (o.name === 'toggleEditors') { if (o.name === 'toggleEditors') {
rightItemComps.push(<ToggleEditorsButton rightItemComps.push(<ToggleEditorsButton
@@ -73,7 +77,7 @@ class ToolbarBaseComponent extends React.Component<Props, any> {
<div style={groupStyle}> <div style={groupStyle}>
{centerItemComps} {centerItemComps}
</div> </div>
<div style={{ ...groupStyle, flex: 1, justifyContent: 'flex-end' }}> <div style={Object.assign({}, groupStyle, { flex: 1, justifyContent: 'flex-end' })}>
{rightItemComps} {rightItemComps}
</div> </div>
</div> </div>

View File

@@ -8,7 +8,7 @@ interface Props {
class ToolbarSpace extends React.Component<Props> { class ToolbarSpace extends React.Component<Props> {
public render() { public render() {
const theme = themeStyle(this.props.themeId); const theme = themeStyle(this.props.themeId);
const style = { ...theme.toolbarStyle }; const style = Object.assign({}, theme.toolbarStyle);
style.minWidth = style.height / 2; style.minWidth = style.height / 2;
return <span style={style}></span>; return <span style={style}></span>;

View File

@@ -96,23 +96,25 @@ markJsUtils.markKeyword = (mark, keyword, stringUtils, extraOptions = null) => {
mark.mark( mark.mark(
[value], [value],
{ Object.assign(
{},
accuracy: accuracy, {
filter: (node, _term, _totalCounter, _counter) => { accuracy: accuracy,
// We exclude SVG because it creates a "<mark>" tag inside filter: (node, _term, _totalCounter, _counter) => {
// the document, which is not a valid SVG tag. As a result // We exclude SVG because it creates a "<mark>" tag inside
// the content within that tag disappears. // the document, which is not a valid SVG tag. As a result
// // the content within that tag disappears.
// mark.js has an "exclude" parameter, but it doesn't work //
// so we use "filter" instead. // mark.js has an "exclude" parameter, but it doesn't work
// // so we use "filter" instead.
// https://github.com/joplin/plugin-abc-sheet-music //
if (isInsideContainer(node, 'SVG')) return false; // https://github.com/joplin/plugin-abc-sheet-music
return true; if (isInsideContainer(node, 'SVG')) return false;
return true;
},
}, },
...extraOptions, extraOptions
} )
); );
}; };

View File

@@ -41,9 +41,13 @@ const style = createSelector(
}, },
}; };
output.buttonIconSelected = { ...output.buttonIcon, color: theme.highlightedColor }; output.buttonIconSelected = Object.assign({}, output.buttonIcon, {
color: theme.highlightedColor,
});
output.buttonLabelSelected = { ...output.buttonLabel, color: theme.color }; output.buttonLabelSelected = Object.assign({}, output.buttonLabel, {
color: theme.color,
});
return output; return output;
} }

View File

@@ -1,6 +1,6 @@
{ {
"name": "@joplin/app-desktop", "name": "@joplin/app-desktop",
"version": "2.11.9", "version": "2.11.1",
"description": "Joplin for Desktop", "description": "Joplin for Desktop",
"main": "main.js", "main": "main.js",
"private": true, "private": true,
@@ -12,7 +12,7 @@
"electronRebuild": "gulp electronRebuild", "electronRebuild": "gulp electronRebuild",
"tsc": "tsc --project tsconfig.json", "tsc": "tsc --project tsconfig.json",
"watch": "tsc --watch --preserveWatchOutput --project tsconfig.json", "watch": "tsc --watch --preserveWatchOutput --project tsconfig.json",
"start": "gulp build && electron . --env dev --log-level debug --open-dev-tools", "start": "gulp build && electron . --env dev --log-level debug --no-welcome --open-dev-tools",
"test": "jest", "test": "jest",
"test-ci": "yarn test" "test-ci": "yarn test"
}, },
@@ -27,7 +27,7 @@
}, },
"build": { "build": {
"appId": "net.cozic.joplin-desktop", "appId": "net.cozic.joplin-desktop",
"compression": "normal", "compression": "maximum",
"productName": "Joplin", "productName": "Joplin",
"npmRebuild": false, "npmRebuild": false,
"afterSign": "./tools/notarizeMacApp.js", "afterSign": "./tools/notarizeMacApp.js",
@@ -110,23 +110,23 @@
"devDependencies": { "devDependencies": {
"@joplin/tools": "~2.11", "@joplin/tools": "~2.11",
"@testing-library/react-hooks": "8.0.1", "@testing-library/react-hooks": "8.0.1",
"@types/jest": "29.5.1", "@types/jest": "29.2.6",
"@types/node": "18.15.13", "@types/node": "18.11.18",
"@types/react": "16.14.41", "@types/react": "16.14.35",
"@types/react-redux": "7.1.25", "@types/react-redux": "7.1.25",
"@types/styled-components": "5.1.26", "@types/styled-components": "5.1.26",
"electron": "19.1.4", "electron": "19.1.4",
"electron-builder": "22.11.7", "electron-builder": "23.6.0",
"electron-notarize": "1.2.2", "electron-notarize": "1.2.2",
"electron-rebuild": "3.2.9", "electron-rebuild": "3.2.9",
"glob": "8.1.0", "glob": "8.1.0",
"gulp": "4.0.2", "gulp": "4.0.2",
"jest": "29.5.0", "jest": "29.4.3",
"jest-environment-jsdom": "29.5.0", "jest-environment-jsdom": "29.4.3",
"js-sha512": "0.8.0", "js-sha512": "0.8.0",
"nan": "2.17.0", "nan": "2.17.0",
"react-test-renderer": "18.2.0", "react-test-renderer": "18.2.0",
"typescript": "5.0.2" "typescript": "4.9.4"
}, },
"optionalDependencies": { "optionalDependencies": {
"7zip-bin-linux": "^1.0.1", "7zip-bin-linux": "^1.0.1",
@@ -138,6 +138,7 @@
"@fortawesome/fontawesome-free": "5.15.4", "@fortawesome/fontawesome-free": "5.15.4",
"@joeattardi/emoji-button": "4.6.4", "@joeattardi/emoji-button": "4.6.4",
"@joplin/lib": "~2.11", "@joplin/lib": "~2.11",
"@joplin/pdf-viewer": "~2.11",
"@joplin/renderer": "~2.11", "@joplin/renderer": "~2.11",
"async-mutex": "0.4.0", "async-mutex": "0.4.0",
"codemirror": "5.65.9", "codemirror": "5.65.9",
@@ -163,15 +164,15 @@
"react-datetime": "3.2.0", "react-datetime": "3.2.0",
"react-dom": "18.2.0", "react-dom": "18.2.0",
"react-redux": "8.0.5", "react-redux": "8.0.5",
"react-select": "5.7.3", "react-select": "5.7.2",
"react-toggle-button": "2.2.0", "react-toggle-button": "2.2.0",
"react-tooltip": "4.5.1", "react-tooltip": "4.5.1",
"redux": "4.2.1", "redux": "4.2.1",
"reselect": "4.1.8", "reselect": "4.1.7",
"roboto-fontface": "0.10.0", "roboto-fontface": "0.10.0",
"smalltalk": "2.5.1", "smalltalk": "2.5.1",
"sqlite3": "5.1.6", "sqlite3": "5.1.6",
"styled-components": "5.3.10", "styled-components": "5.3.9",
"styled-system": "5.1.5", "styled-system": "5.1.5",
"taboverride": "4.0.3", "taboverride": "4.0.3",
"tinymce": "5.10.6" "tinymce": "5.10.6"

View File

@@ -133,8 +133,8 @@ class Dialog extends React.PureComponent<Props, State> {
} }
this.styles_[styleKey] = { this.styles_[styleKey] = {
dialogBox: { ...theme.dialogBox, minWidth: '50%', maxWidth: '50%' }, dialogBox: Object.assign({}, theme.dialogBox, { minWidth: '50%', maxWidth: '50%' }),
input: { ...theme.inputStyle, flex: 1 }, input: Object.assign({}, theme.inputStyle, { flex: 1 }),
row: { row: {
overflow: 'hidden', overflow: 'hidden',
height: itemHeight, height: itemHeight,
@@ -148,7 +148,7 @@ class Dialog extends React.PureComponent<Props, State> {
borderBottomColor: theme.dividerColor, borderBottomColor: theme.dividerColor,
boxSizing: 'border-box', boxSizing: 'border-box',
}, },
help: { ...theme.textStyle, marginBottom: 10 }, help: Object.assign({}, theme.textStyle, { marginBottom: 10 }),
inputHelpWrapper: { display: 'flex', flexDirection: 'row', alignItems: 'center' }, inputHelpWrapper: { display: 'flex', flexDirection: 'row', alignItems: 'center' },
}; };
@@ -163,15 +163,19 @@ class Dialog extends React.PureComponent<Props, State> {
userSelect: 'none', userSelect: 'none',
}; };
const rowTitleStyle = { ...rowTextStyle, fontSize: rowTextStyle.fontSize * 1.4, const rowTitleStyle = Object.assign({}, rowTextStyle, {
fontSize: rowTextStyle.fontSize * 1.4,
marginBottom: this.state.resultsInBody ? 6 : 4, marginBottom: this.state.resultsInBody ? 6 : 4,
color: theme.colorFaded }; color: theme.colorFaded,
});
const rowFragmentsStyle = { ...rowTextStyle, fontSize: rowTextStyle.fontSize * 1.2, const rowFragmentsStyle = Object.assign({}, rowTextStyle, {
fontSize: rowTextStyle.fontSize * 1.2,
marginBottom: this.state.resultsInBody ? 8 : 6, marginBottom: this.state.resultsInBody ? 8 : 6,
color: theme.colorFaded }; color: theme.colorFaded,
});
this.styles_[styleKey].rowSelected = { ...this.styles_[styleKey].row, backgroundColor: theme.selectedColor }; this.styles_[styleKey].rowSelected = Object.assign({}, this.styles_[styleKey].row, { backgroundColor: theme.selectedColor });
this.styles_[styleKey].rowPath = rowTextStyle; this.styles_[styleKey].rowPath = rowTextStyle;
this.styles_[styleKey].rowTitle = rowTitleStyle; this.styles_[styleKey].rowTitle = rowTitleStyle;
this.styles_[styleKey].rowFragments = rowFragmentsStyle; this.styles_[styleKey].rowFragments = rowFragmentsStyle;
@@ -300,7 +304,7 @@ class Dialog extends React.PureComponent<Props, State> {
for (let i = 0; i < results.length; i++) { for (let i = 0; i < results.length; i++) {
const row = results[i]; const row = results[i];
const path = Folder.folderPathString(this.props.folders, row.parent_id); const path = Folder.folderPathString(this.props.folders, row.parent_id);
results[i] = { ...row, path: path ? path : '/' }; results[i] = Object.assign({}, row, { path: path ? path : '/' });
} }
} else { // Note TITLE or BODY } else { // Note TITLE or BODY
listType = BaseModel.TYPE_NOTE; listType = BaseModel.TYPE_NOTE;
@@ -313,7 +317,7 @@ class Dialog extends React.PureComponent<Props, State> {
for (let i = 0; i < results.length; i++) { for (let i = 0; i < results.length; i++) {
const row = results[i]; const row = results[i];
const path = Folder.folderPathString(this.props.folders, row.parent_id); const path = Folder.folderPathString(this.props.folders, row.parent_id);
results[i] = { ...row, path: path }; results[i] = Object.assign({}, row, { path: path });
} }
} else { } else {
const limit = 20; const limit = 20;
@@ -361,9 +365,9 @@ class Dialog extends React.PureComponent<Props, State> {
} }
results[i] = { ...row, path, fragments }; results[i] = Object.assign({}, row, { path, fragments });
} else { } else {
results[i] = { ...row, path: path, fragments: '' }; results[i] = Object.assign({}, row, { path: path, fragments: '' });
} }
} }

View File

@@ -72,10 +72,10 @@ async function main() {
src: langSourceDir, src: langSourceDir,
dest: `${buildLibDir}/tinymce/langs`, dest: `${buildLibDir}/tinymce/langs`,
}, },
// { {
// src: resolve(__dirname, '../../pdf-viewer/dist'), src: resolve(__dirname, '../../pdf-viewer/dist'),
// dest: `${buildLibDir}/@joplin/pdf-viewer`, dest: `${buildLibDir}/@joplin/pdf-viewer`,
// }, },
]; ];
const files = [ const files = [
@@ -93,10 +93,10 @@ async function main() {
src: resolve(__dirname, '../../lib/services/plugins/sandboxProxy.js'), src: resolve(__dirname, '../../lib/services/plugins/sandboxProxy.js'),
dest: `${buildLibDir}/@joplin/lib/services/plugins/sandboxProxy.js`, dest: `${buildLibDir}/@joplin/lib/services/plugins/sandboxProxy.js`,
}, },
// { {
// src: resolve(__dirname, '../../pdf-viewer/index.html'), src: resolve(__dirname, '../../pdf-viewer/index.html'),
// dest: `${buildLibDir}/@joplin/pdf-viewer/index.html`, dest: `${buildLibDir}/@joplin/pdf-viewer/index.html`,
// }, },
]; ];
// First we delete all the destination directories, then we copy the files. // First we delete all the destination directories, then we copy the files.

View File

@@ -72,4 +72,3 @@ components/NoteEditor/CodeMirror/CodeMirror.bundle.min.js
components/NoteEditor/**/*.bundle.js.md5 components/NoteEditor/**/*.bundle.js.md5
utils/fs-driver-android.js utils/fs-driver-android.js
android/app/build-*

View File

@@ -79,12 +79,6 @@ import org.apache.tools.ant.taskdefs.condition.Os
*/ */
project.ext.react = [ project.ext.react = [
// 2023-05-09: This seems to be optional, but it's not. Without it, the app
// will crash on certain devices with this error:
//
// > java.lang.UnsatisfiedLinkError: couldn't find DSO to load: libhermes.so"
//
// https://github.com/laurent22/joplin/issues/8144#issuecomment-1539629812
enableHermes: true, // clean and rebuild if changing enableHermes: true, // clean and rebuild if changing
] ]
@@ -156,8 +150,8 @@ android {
applicationId "net.cozic.joplin" applicationId "net.cozic.joplin"
minSdkVersion rootProject.ext.minSdkVersion minSdkVersion rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion targetSdkVersion rootProject.ext.targetSdkVersion
versionCode 2097714 versionCode 2097687
versionName "2.11.29" versionName "2.11.2"
// ndk { // ndk {
// abiFilters "armeabi-v7a", "x86", "arm64-v8a", "x86_64" // abiFilters "armeabi-v7a", "x86", "arm64-v8a", "x86_64"
// } // }
@@ -304,11 +298,8 @@ dependencies {
implementation fileTree(dir: "libs", include: ["*.jar"]) implementation fileTree(dir: "libs", include: ["*.jar"])
//noinspection GradleDynamicVersion //noinspection GradleDynamicVersion
// implementation "com.facebook.react:react-native:+" // From node_modules implementation "com.facebook.react:react-native:+" // From node_modules
implementation ("com.facebook.react:react-native") version {
strictly "0.70.6" // pass in your react-native version
}
implementation "androidx.swiperefreshlayout:swiperefreshlayout:1.0.0" implementation "androidx.swiperefreshlayout:swiperefreshlayout:1.0.0"
debugImplementation("com.facebook.flipper:flipper:${FLIPPER_VERSION}") { debugImplementation("com.facebook.flipper:flipper:${FLIPPER_VERSION}") {

View File

@@ -0,0 +1,7 @@
French small model for Vosk
WER
%WER 23.95 [ 37203 / 155330, 5373 ins, 4427 del, 27403 sub ] exp/chain_a/tdnn/decode_test_cv/wer_12_0.0
%WER 19.30 [ 2975 / 15412, 683 ins, 672 del, 1620 sub ] exp/chain_a/tdnn/decode_test_mtedx/wer_10_0.0
%WER 27.25 [ 20208 / 74145, 2647 ins, 5852 del, 11709 sub ] exp/chain_a/tdnn/decode_test_podcast_reseg/wer_10_0.0

View File

@@ -0,0 +1,8 @@
--use-energy=false
--sample-frequency=16000
--num-mel-bins=40
--num-ceps=40
--low-freq=40
--high-freq=-200
--allow-upsample=true
--allow-downsample=true

View File

@@ -0,0 +1,10 @@
--min-active=200
--max-active=7000
--beam=13.0
--lattice-beam=4.0
--acoustic-scale=1.0
--frame-subsampling-factor=3
--endpoint.silence-phones=1:2:3:4:5:6:7:8:9:10
--endpoint.rule2.min-trailing-silence=0.5
--endpoint.rule3.min-trailing-silence=1.0
--endpoint.rule4.min-trailing-silence=2.0

Some files were not shown because too many files have changed in this diff Show More