Compare commits

..

43 Commits

Author SHA1 Message Date
Herbert Reiter dc28ac3271 New release 2021-01-04 12:48:22 +01:00
Herbert Reiter f39b9303b5 New release 2021-01-04 12:44:23 +01:00
Herbert Reiter 95f42dd15f Update copyright year 2021-01-04 12:44:07 +01:00
Herbert Reiter 8fe0c8ec36 Update descriptions for F-Droid 2021-01-04 12:41:15 +01:00
Herbert Reiter 079a4f4907 Bugfix: Disable wiki editor handler 2021-01-04 12:38:10 +01:00
Herbert Reiter df9f81bc01 New release 2021-01-04 12:36:30 +01:00
Herbert Reiter 21896b82ac New release 2020-12-28 13:44:13 +01:00
Herbert Reiter cdfed475a9 Fix lint warnings 2020-12-28 13:38:33 +01:00
Herbert Reiter 87d3b988ff Refactoring: Optimize code 2020-12-28 13:34:54 +01:00
Herbert Reiter d48f208e14 Remove redundant images 2020-12-28 13:19:26 +01:00
Herbert Reiter c66e95b83e New release 2020-12-28 12:31:31 +01:00
Herbert Reiter ee58f8026a Update to moasdawiki-server 2.4.2 2020-12-28 12:30:29 +01:00
Herbert Reiter 267f0c84fb Update to moasdawiki-server 2.4.2 2020-12-28 12:28:54 +01:00
Herbert Reiter 203d225955 Use runOnUiThread to show UI toasts 2020-12-28 11:54:46 +01:00
Herbert Reiter 6d6c7577da Bugfix: Use App config 2020-12-27 22:18:56 +01:00
Herbert Reiter 8cbb78e276 Update to moasdawiki-server 2.4.1 2020-12-27 22:02:05 +01:00
Herbert Reiter d682d780e0 Update to moasdawiki-server 2.4.1, replace deprecated AsyncTask usage 2020-12-27 22:01:52 +01:00
Herbert Reiter d32c546571 Update to moasdawiki-server 2.4.1 2020-12-27 16:36:15 +01:00
Herbert Reiter f5caf4bc50 Update to moasdawiki-server 2.4.1 2020-12-27 16:31:10 +01:00
Herbert Reiter 09a2fb28f7 Upgrade library dependencies 2020-12-27 16:30:08 +01:00
Herbert Reiter 0cb372aa26 New release 2020-10-11 18:53:07 +02:00
Herbert Reiter b25bb0d1d2 - Update to moasdawiki-server 2.3.3
- new release
2020-10-11 18:52:55 +02:00
Herbert Reiter 7abb1788b6 Bugfix: For searches only use the search index cache 2020-10-11 18:52:15 +02:00
Herbert Reiter 2a16dbefb8 Bugfix: Don't show empty page on app startup 2020-10-11 15:57:00 +02:00
Herbert Reiter 7e4d30bbe9 Update Gradle version 2020-10-11 13:28:26 +02:00
Herbert Reiter eae02b9153 Update .gitignore 2020-10-11 13:27:45 +02:00
Herbert Reiter 273901dc44 Update README.md 2020-09-15 20:38:44 +02:00
Herbert Reiter 1b3ee62d8b New release 2020-09-13 22:12:15 +02:00
Herbert Reiter 8c330dd825 Update changelog 2020-09-13 22:07:23 +02:00
Herbert Reiter cde3e690be Update changelog 2020-09-13 22:03:37 +02:00
Herbert Reiter 46698f1ed2 Update to moasdawiki-server 2.3.1 2020-09-13 22:03:06 +02:00
Herbert Reiter 93644569fd Bugfix: Don't go back to start page after App restore 2020-09-13 22:02:27 +02:00
Herbert Reiter 8127747534 Update changelog 2020-09-08 22:44:02 +02:00
Herbert Reiter 306fdd8938 Update dependencies 2020-09-08 22:43:28 +02:00
Herbert Reiter 75b29b5fde Suppress lint warning 2020-09-08 22:41:40 +02:00
Herbert Reiter f70370b797 Add German fastlane metadata for F-Droid repository 2020-09-08 22:41:14 +02:00
Herbert Reiter 0acb84af33 Add German fastlane metadata for F-Droid repository 2020-09-07 22:59:23 +02:00
Herbert Reiter 86394e4408 Add link to F-Droid 2020-08-23 21:37:10 +02:00
Herbert Reiter 6f5dff6925 Link to MoasdaWiki server repository 2020-08-23 21:27:44 +02:00
Herbert Reiter dce4c0a981 Update readme 2020-03-15 18:30:32 +01:00
Herbert Reiter 0ba536d7a3 Gradle update 2020-03-15 18:30:07 +01:00
Herbert Reiter 8a7326dcac Add fastlane metadata for F-Droid repository 2020-03-15 18:29:41 +01:00
Herbert Reiter e40e99a968 Update .gitignore 2020-02-08 18:16:14 +01:00
59 changed files with 382 additions and 424 deletions
+4 -1
View File
@@ -1,7 +1,7 @@
# IntelliJ # IntelliJ
.idea/ .idea/
out/ out/
moasdawiki-app.iml *.iml
# Gradle # Gradle
.gradle/ .gradle/
@@ -10,3 +10,6 @@ release/
# Git # Git
local.properties local.properties
# Project
TODO.md
+34
View File
@@ -1,5 +1,39 @@
# Changelog # Changelog
## 2.4.3.0 (versionCode 21, 2021-01-04)
- Update descriptions for F-Droid
- Update copyright year
- Bugfix: Disable wiki editor handler
- Update to moasdawiki-server 2.4.3
## 2.4.2.0 (versionCode 20, 2020-12-28)
- Update to moasdawiki-server 2.4.2
- Upgrade library dependencies
- Refactoring: Replace deprecated AsyncTask
## 2.3.3.0 (versionCode 19, 2020-10-11)
- Bugfix: Show start page on app startup
- Bugfix: For searches only use the search index cache
- Update to moasdawiki-server 2.3.3
## 2.3.1.0 (versionCode 18, 2020-09-13)
- Speed up full text search by a search index
- Bugfix: Don't go back to start page after App restore
- Update to moasdawiki-server 2.3.1
## 2.2.1.1 (versionCode 17, 2020-09-08)
- Add German fastlane metadata for F-Droid repository
- Update library dependencies
## 2.2.1.0 (versionCode 16, 2020-03-15)
- Add fastlane metadata for F-Droid repository
## 2.2.0.0 (versionCode 15, 2020-02-08) ## 2.2.0.0 (versionCode 15, 2020-02-08)
- Speed up full text search by searching only in the page titles in the first step - Speed up full text search by searching only in the page titles in the first step
+2 -2
View File
@@ -632,7 +632,7 @@ state the exclusion of warranty; and each file should have at least
the "copyright" line and a pointer to where the full notice is found. the "copyright" line and a pointer to where the full notice is found.
MoasdaWiki Server MoasdaWiki Server
Copyright (C) 2008 - 2020 Herbert Reiter Copyright (C) 2008 - 2021 Herbert Reiter
This program is free software: you can redistribute it and/or modify This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by it under the terms of the GNU General Public License as published by
@@ -652,7 +652,7 @@ Also add information on how to contact you by electronic and paper mail.
If the program does terminal interaction, make it output a short If the program does terminal interaction, make it output a short
notice like this when it starts in an interactive mode: notice like this when it starts in an interactive mode:
MoasdaWiki Server Copyright (C) 2008 - 2020 Herbert Reiter MoasdaWiki Server Copyright (C) 2008 - 2021 Herbert Reiter
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
This is free software, and you are welcome to redistribute it This is free software, and you are welcome to redistribute it
under certain conditions; type `show c' for details. under certain conditions; type `show c' for details.
+8 -3
View File
@@ -5,20 +5,25 @@
MoasdaWiki App is an extension of the MoasdaWiki Server to have a copy of MoasdaWiki App is an extension of the MoasdaWiki Server to have a copy of
your Wiki content on your mobile device. your Wiki content on your mobile device.
To download the app see https://moasdawiki.net/. For MoasdaWiki documentation see https://moasdawiki.net/.
[<img src="https://fdroid.gitlab.io/artwork/badge/get-it-on.png"
alt="Get it on F-Droid"
height="80">](https://f-droid.org/packages/net.moasdawiki.app)
## Features ## Features
- Synchronizes the data from a MoasdaWiki Server instance. - Synchronizes the data from a MoasdaWiki Server instance.
- Data privacy: No cloud connection established, directly connects to the server inside your private network. - Data privacy: No cloud connection established, directly connects to the server inside your private network.
- Powerful full text search, supports regular expressions. - Fast full text search.
- Content cannot be modified within the app as it is no fun to type Wiki syntax on the mobile device, - Content cannot be modified within the app as it is no fun to type Wiki syntax on the mobile device,
changes have to be done via the MoasdaWiki Server. changes have to be done via the MoasdaWiki Server.
- Calendar integration, shows birthdays and events in the mobile calendar (German version only). - Calendar integration, shows birthdays and events in the mobile calendar (German version only).
## Synchronize content with a MoasdaWiki Server ## Synchronize content with a MoasdaWiki Server
1. Ensure to have a MoasdaWiki Server instance running on a computer in your LAN. 1. Download MoasdaWiki Server from https://gitlab.com/moasdawiki/moasdawiki-server.
1. Set up a MoasdaWiki Server instance in your LAN.
1. Install and open the MoasdaWiki App. 1. Install and open the MoasdaWiki App.
1. You can see a hint that the App has to be configured first. Press on that hint. 1. You can see a hint that the App has to be configured first. Press on that hint.
1. Press on "Host name" and enter the host name of the server instance, e.g. `192.168.1.101`. Press OK. 1. Press on "Host name" and enter the host name of the server instance, e.g. `192.168.1.101`. Press OK.
-133
View File
@@ -1,133 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<module external.linked.project.id=":app" external.linked.project.path="$MODULE_DIR$" external.root.project.path="$MODULE_DIR$/.." external.system.id="GRADLE" type="JAVA_MODULE" version="4">
<component name="FacetManager">
<facet type="android-gradle" name="Android-Gradle">
<configuration>
<option name="GRADLE_PROJECT_PATH" value=":app" />
<option name="LAST_SUCCESSFUL_SYNC_AGP_VERSION" value="3.5.3" />
<option name="LAST_KNOWN_AGP_VERSION" value="3.5.3" />
</configuration>
</facet>
<facet type="android" name="Android">
<configuration>
<option name="SELECTED_BUILD_VARIANT" value="debug" />
<option name="ASSEMBLE_TASK_NAME" value="assembleDebug" />
<option name="COMPILE_JAVA_TASK_NAME" value="compileDebugSources" />
<afterSyncTasks>
<task>generateDebugSources</task>
</afterSyncTasks>
<option name="ALLOW_USER_CONFIGURATION" value="false" />
<option name="MANIFEST_FILE_RELATIVE_PATH" value="/src/main/AndroidManifest.xml" />
<option name="RES_FOLDER_RELATIVE_PATH" value="/src/main/res" />
<option name="RES_FOLDERS_RELATIVE_PATH" value="file://$MODULE_DIR$/src/main/res;file://$MODULE_DIR$/build/generated/res/resValues/debug" />
<option name="TEST_RES_FOLDERS_RELATIVE_PATH" value="" />
<option name="ASSETS_FOLDER_RELATIVE_PATH" value="/src/main/assets" />
</configuration>
</facet>
</component>
<component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_8">
<output url="file://$MODULE_DIR$/build/intermediates/javac/debug/classes" />
<output-test url="file://$MODULE_DIR$/build/intermediates/javac/debugUnitTest/classes" />
<exclude-output />
<content url="file://$MODULE_DIR$">
<sourceFolder url="file://$MODULE_DIR$/build/generated/ap_generated_sources/debug/out" isTestSource="false" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/aidl_source_output_dir/debug/compileDebugAidl/out" isTestSource="false" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/buildConfig/debug" isTestSource="false" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/renderscript_source_output_dir/debug/compileDebugRenderscript/out" isTestSource="false" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/rs/debug" type="java-resource" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/resValues/debug" type="java-resource" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/ap_generated_sources/debugAndroidTest/out" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/aidl_source_output_dir/debugAndroidTest/compileDebugAndroidTestAidl/out" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/buildConfig/androidTest/debug" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/renderscript_source_output_dir/debugAndroidTest/compileDebugAndroidTestRenderscript/out" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/rs/androidTest/debug" type="java-test-resource" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/resValues/androidTest/debug" type="java-test-resource" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/ap_generated_sources/debugUnitTest/out" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/res" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/resources" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/assets" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/aidl" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/java" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/rs" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/shaders" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/res" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/resources" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/assets" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/aidl" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/java" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/rs" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/shaders" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/res" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/resources" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/assets" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/aidl" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/java" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/rs" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/shaders" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/main/res" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/main/resources" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/main/assets" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/main/aidl" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/main/java" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/main/rs" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/main/shaders" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/res" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/resources" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/assets" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/aidl" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/java" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/rs" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/shaders" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/test/res" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/test/resources" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/test/assets" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/test/aidl" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/test/java" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/test/rs" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/test/shaders" isTestSource="true" />
<excludeFolder url="file://$MODULE_DIR$/build" />
</content>
<orderEntry type="jdk" jdkName="Android API 29 Platform (2)" jdkType="Android SDK" />
<orderEntry type="sourceFolder" forTests="false" />
<orderEntry type="library" scope="TEST" name="Gradle: org.testng:testng:6.14.3@jar" level="project" />
<orderEntry type="library" scope="TEST" name="Gradle: com.beust:jcommander:1.72@jar" level="project" />
<orderEntry type="library" scope="TEST" name="Gradle: org.apache-extras.beanshell:bsh:2.0b6@jar" level="project" />
<orderEntry type="library" name="Gradle: org.jetbrains:annotations:18.0.0@jar" level="project" />
<orderEntry type="library" name="Gradle: androidx.collection:collection:1.1.0@jar" level="project" />
<orderEntry type="library" name="Gradle: androidx.lifecycle:lifecycle-common:2.1.0@jar" level="project" />
<orderEntry type="library" name="Gradle: androidx.arch.core:core-common:2.1.0@jar" level="project" />
<orderEntry type="library" name="Gradle: androidx.annotation:annotation:1.1.0@jar" level="project" />
<orderEntry type="library" name="Gradle: androidx.preference:preference:1.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.appcompat:appcompat:1.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.fragment:fragment:1.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.appcompat:appcompat-resources:1.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.recyclerview:recyclerview:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.legacy:legacy-support-core-ui:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.drawerlayout:drawerlayout:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.viewpager:viewpager:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.legacy:legacy-support-core-utils:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.loader:loader:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.activity:activity:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.vectordrawable:vectordrawable-animated:1.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.vectordrawable:vectordrawable:1.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.coordinatorlayout:coordinatorlayout:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.slidingpanelayout:slidingpanelayout:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.customview:customview:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.swiperefreshlayout:swiperefreshlayout:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.asynclayoutinflater:asynclayoutinflater:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.core:core:1.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.cursoradapter:cursoradapter:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.versionedparcelable:versionedparcelable:1.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.lifecycle:lifecycle-viewmodel:2.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.lifecycle:lifecycle-runtime:2.1.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.savedstate:savedstate:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.lifecycle:lifecycle-livedata:2.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.lifecycle:lifecycle-livedata-core:2.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.interpolator:interpolator:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.arch.core:core-runtime:2.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.documentfile:documentfile:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.localbroadcastmanager:localbroadcastmanager:1.0.0@aar" level="project" />
<orderEntry type="library" name="Gradle: androidx.print:print:1.0.0@aar" level="project" />
<orderEntry type="module" module-name="moasdawiki-app" />
</component>
</module>
+10 -10
View File
@@ -1,17 +1,17 @@
apply plugin: 'com.android.application' apply plugin: 'com.android.application'
android { android {
compileSdkVersion 29 // 29 = Android 10 compileSdkVersion 30 // 30 = Android 11
defaultConfig { defaultConfig {
applicationId "net.moasdawiki.app" applicationId "net.moasdawiki.app"
minSdkVersion 26 // 26 = Oreo 8.0 minSdkVersion 26 // 26 = Oreo 8.0
targetSdkVersion 29 // should be same as compileSdkVersion targetSdkVersion 30 // should be same as compileSdkVersion
versionCode 15 versionCode 21
versionName "2.2.0.0" versionName "2.4.3.0"
archivesBaseName = "moasdawiki-" + versionName + "-" + versionCode archivesBaseName = "moasdawiki-" + versionName + "-" + versionCode
} }
sourceSets { sourceSets {
main{ main {
java { java {
exclude "net/moasdawiki/plugin/sync/SynchronizationPlugin*" exclude "net/moasdawiki/plugin/sync/SynchronizationPlugin*"
} }
@@ -24,10 +24,10 @@ android {
} }
dependencies { dependencies {
implementation 'net.moasdawiki:moasdawiki-server:2.2.0' implementation 'net.moasdawiki:moasdawiki-server:2.4.3'
implementation fileTree(include: ['*.jar'], dir: 'libs') implementation fileTree(include: ['*.jar'], dir: 'libs')
implementation 'androidx.appcompat:appcompat:1.1.0' implementation 'androidx.appcompat:appcompat:1.2.0'
implementation 'androidx.preference:preference:1.1.0' implementation 'androidx.preference:preference:1.1.1'
compileOnly 'org.jetbrains:annotations:18.0.0' compileOnly 'org.jetbrains:annotations:20.1.0'
testImplementation 'org.testng:testng:6.14.3' testImplementation 'org.testng:testng:7.3.0'
} }
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -30,6 +30,7 @@ public class AndroidSettings extends Settings {
super(logger, repositoryService, configFileName); super(logger, repositoryService, configFileName);
} }
@Override
@NotNull @NotNull
public String getVersion() { public String getVersion() {
return BuildConfig.VERSION_NAME; return BuildConfig.VERSION_NAME;
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -43,9 +43,7 @@ import androidx.preference.PreferenceManager;
import android.util.Log; import android.util.Log;
import android.widget.Toast; import android.widget.Toast;
import net.moasdawiki.plugin.Plugin; import net.moasdawiki.service.transform.TerminTransformer;
import net.moasdawiki.plugin.PluginService;
import net.moasdawiki.plugin.TerminPlugin;
import net.moasdawiki.util.PathUtils; import net.moasdawiki.util.PathUtils;
import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.NotNull;
@@ -87,7 +85,7 @@ public class CalendarSyncAdapter extends AbstractThreadedSyncAdapter {
@Override @Override
public void onPerformSync(Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) { public void onPerformSync(Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) {
Log.d(TAG, "Begin of onPerformSync()"); Log.d(TAG, "Begin of onPerformSync()");
List<TerminPlugin.Event> events = getWikiEvents(); List<TerminTransformer.Event> events = getWikiEvents();
String calendarId = createCalendar(); String calendarId = createCalendar();
if (calendarId != null) { if (calendarId != null) {
deleteAllEvents(calendarId); deleteAllEvents(calendarId);
@@ -106,21 +104,15 @@ public class CalendarSyncAdapter extends AbstractThreadedSyncAdapter {
* Imports all events on Wiki pages to the Android calendar. * Imports all events on Wiki pages to the Android calendar.
*/ */
@NotNull @NotNull
private List<TerminPlugin.Event> getWikiEvents() { private List<TerminTransformer.Event> getWikiEvents() {
Log.d(TAG, "Reading Wiki events"); Log.d(TAG, "Reading Wiki events");
WikiEngineApplication app = (WikiEngineApplication) getContext(); WikiEngineApplication app = (WikiEngineApplication) getContext();
PluginService pluginService = app.getServiceLocator().getPluginService(); TerminTransformer terminTransformer = app.getTerminTransformer();
TerminPlugin terminPlugin = null; if (terminTransformer == null) {
for (Plugin plugin : pluginService.getPlugins()) { Log.e(TAG, "TerminTransformer not initialized yet, cannot retrieve event list");
if (plugin instanceof TerminPlugin) {
terminPlugin = (TerminPlugin) plugin;
}
}
if (terminPlugin == null) {
Log.e(TAG, "TerminPlugin not found, cannot retrieve event list");
return Collections.emptyList(); return Collections.emptyList();
} }
List<TerminPlugin.Event> events = terminPlugin.getEvents(); List<TerminTransformer.Event> events = terminTransformer.getEvents();
Log.d(TAG, "Wiki events found: " + events.size()); Log.d(TAG, "Wiki events found: " + events.size());
return events; return events;
} }
@@ -203,14 +195,14 @@ public class CalendarSyncAdapter extends AbstractThreadedSyncAdapter {
* Adds all events from the event list. The first occurrence is in the current year, the events * Adds all events from the event list. The first occurrence is in the current year, the events
* are repeated every year. * are repeated every year.
*/ */
private void addEvents(@NotNull String calendarId, @NotNull List<TerminPlugin.Event> events) { private void addEvents(@NotNull String calendarId, @NotNull List<TerminTransformer.Event> events) {
Log.d(TAG, "Create calendar events"); Log.d(TAG, "Create calendar events");
for (TerminPlugin.Event event : events) { for (TerminTransformer.Event event : events) {
String title = event.description; String title = event.description;
if (title == null) { if (title == null) {
title = PathUtils.extractWebName(event.pagePath); title = PathUtils.extractWebName(event.pagePath);
} }
String description = getContext().getString(R.string.calendar_date) + ": " + TerminPlugin.formatGermanDate(event.dateFields); String description = getContext().getString(R.string.calendar_date) + ": " + TerminTransformer.formatGermanDate(event.dateFields);
String eventId = addEvent(calendarId, event.dateFields.day, event.dateFields.month, event.dateFields.year, title, description); String eventId = addEvent(calendarId, event.dateFields.day, event.dateFields.month, event.dateFields.year, title, description);
if (eventId != null) { if (eventId != null) {
addReminder(eventId); addReminder(eventId);
@@ -294,8 +286,7 @@ public class CalendarSyncAdapter extends AbstractThreadedSyncAdapter {
if (!ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.WRITE_CALENDAR)) { if (!ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.WRITE_CALENDAR)) {
Log.d(TAG, "User has permanently denied permission WRITE_CALENDAR, informing him"); Log.d(TAG, "User has permanently denied permission WRITE_CALENDAR, informing him");
String hint = context.getString(R.string.calendar_permission_request, "WRITE_CALENDAR"); String hint = context.getString(R.string.calendar_permission_request, "WRITE_CALENDAR");
Toast toast = Toast.makeText(context, hint, Toast.LENGTH_SHORT); activity.runOnUiThread(() -> Toast.makeText(context, hint, Toast.LENGTH_SHORT).show());
toast.show();
} }
Log.d(TAG, "Ask for permission WRITE_CALENDAR"); Log.d(TAG, "Ask for permission WRITE_CALENDAR");
ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.WRITE_CALENDAR}, 0); ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.WRITE_CALENDAR}, 0);
@@ -305,8 +296,7 @@ public class CalendarSyncAdapter extends AbstractThreadedSyncAdapter {
if (!ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.READ_CALENDAR)) { if (!ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.READ_CALENDAR)) {
Log.d(TAG, "User has permanently denied permission READ_CALENDAR, informing him"); Log.d(TAG, "User has permanently denied permission READ_CALENDAR, informing him");
String hint = context.getString(R.string.calendar_permission_request, "READ_CALENDAR"); String hint = context.getString(R.string.calendar_permission_request, "READ_CALENDAR");
Toast toast = Toast.makeText(context, hint, Toast.LENGTH_SHORT); activity.runOnUiThread(() -> Toast.makeText(context, hint, Toast.LENGTH_SHORT).show());
toast.show();
} }
Log.d(TAG, "Ask for permission READ_CALENDAR"); Log.d(TAG, "Ask for permission READ_CALENDAR");
ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.READ_CALENDAR}, 0); ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.READ_CALENDAR}, 0);
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -20,8 +20,6 @@ package net.moasdawiki.app;
/** /**
* Enthält zentrale Konstanten. * Enthält zentrale Konstanten.
*
* @author Herbert Reiter
*/ */
public abstract class Constants { public abstract class Constants {
public static final String PREFERENCES_SYNC_SERVER_HOST = "sync_server_host"; public static final String PREFERENCES_SYNC_SERVER_HOST = "sync_server_host";
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -25,7 +25,6 @@ import android.content.Context;
import android.content.Intent; import android.content.Intent;
import android.content.SharedPreferences; import android.content.SharedPreferences;
import android.net.Uri; import android.net.Uri;
import android.os.AsyncTask;
import android.os.Bundle; import android.os.Bundle;
import android.util.Log; import android.util.Log;
import android.view.KeyEvent; import android.view.KeyEvent;
@@ -47,17 +46,16 @@ import android.widget.ProgressBar;
import android.widget.TextView; import android.widget.TextView;
import android.widget.Toast; import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity; import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.view.menu.MenuBuilder; import androidx.appcompat.view.menu.MenuBuilder;
import androidx.preference.PreferenceManager; import androidx.preference.PreferenceManager;
import net.moasdawiki.base.ServiceException; import net.moasdawiki.base.ServiceException;
import net.moasdawiki.base.Settings; import net.moasdawiki.base.Settings;
import net.moasdawiki.plugin.Plugin;
import net.moasdawiki.plugin.PluginService;
import net.moasdawiki.server.HttpRequest; import net.moasdawiki.server.HttpRequest;
import net.moasdawiki.server.HttpResponse; import net.moasdawiki.server.RequestDispatcher;
import net.moasdawiki.service.render.HtmlService; import net.moasdawiki.service.HttpResponse;
import net.moasdawiki.service.repository.RepositoryService; import net.moasdawiki.service.repository.RepositoryService;
import net.moasdawiki.util.EscapeUtils; import net.moasdawiki.util.EscapeUtils;
@@ -74,41 +72,39 @@ import java.util.Calendar;
import java.util.Collections; import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/** /**
* Steuert das Verhalten des Hauptfensters inkl. eingebettetem Browser. * Displays the main window with the embedded wiki browser.
*
* @author Herbert Reiter
*/ */
public class MainActivity extends AppCompatActivity { public class MainActivity extends AppCompatActivity {
private static final String TAG = "MainActivity"; private static final String TAG = "MainActivity";
private static final String SERVER_BASE_URL = "http://localhost:1/"; private static final String SERVER_BASE_URL = "http://localhost:1/";
private Settings settings;
private RepositoryService repositoryService; private RepositoryService repositoryService;
private PluginService pluginService; private Settings settings;
private HtmlService htmlService;
private SynchronizeWikiClient synchronizeWikiClient; private SynchronizeWikiClient synchronizeWikiClient;
private RequestDispatcher requestDispatcher;
private WebView webview; private WebView webview;
private long backButtonPressedTimestamp; private long backButtonPressedTimestamp;
private ExecutorService synchronizationExecutorService;
@Override @Override
protected void onCreate(@Nullable Bundle savedInstanceState) { protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState); super.onCreate(savedInstanceState);
setContentView(R.layout.main_layout); setContentView(R.layout.main_layout);
// AndroidMainService holen
WikiEngineApplication app = (WikiEngineApplication) getApplication(); WikiEngineApplication app = (WikiEngineApplication) getApplication();
settings = app.getServiceLocator().getSettings(); repositoryService = app.getRepositoryService();
repositoryService = app.getServiceLocator().getRepositoryService(); settings = app.getSettings();
pluginService = app.getServiceLocator().getPluginService();
htmlService = app.getServiceLocator().getHtmlService();
synchronizeWikiClient = app.getSynchronizeWikiClient(); synchronizeWikiClient = app.getSynchronizeWikiClient();
requestDispatcher = app.getRequestDispatcher();
synchronizationExecutorService = Executors.newSingleThreadExecutor();
// eingebetteten Browser konfigurieren
initWebView(); initWebView();
EditText searchInput = findViewById(R.id.search_input); EditText searchInput = findViewById(R.id.search_input);
@@ -125,31 +121,29 @@ public class MainActivity extends AppCompatActivity {
} }
/** /**
* Konfiguriert den eingebetteten Browser. * Configure the embedded web browser.
*/ */
@SuppressLint("SetJavaScriptEnabled") @SuppressLint("SetJavaScriptEnabled")
private void initWebView() { private void initWebView() {
// Cookies deaktivieren
CookieManager cookieManager = CookieManager.getInstance(); CookieManager cookieManager = CookieManager.getInstance();
cookieManager.setAcceptCookie(false); cookieManager.setAcceptCookie(false);
// localhost-URLs in WebView anzeigen
webview = findViewById(R.id.web_browser); webview = findViewById(R.id.web_browser);
webview.setWebChromeClient(new WebChromeClient()); webview.setWebChromeClient(new WebChromeClient());
webview.setWebViewClient(new WebViewClient() { webview.setWebViewClient(new WebViewClient() {
/** /**
* Steuert, welche Links im eingebetteten Browser und welche im externen Browser * Dispatch URLs the user clicks on.
* geöffnet werden sollen. * External URLs will be shown in an external browser.
*/ */
@Override @Override
public boolean shouldOverrideUrlLoading(@NotNull WebView view, @NotNull WebResourceRequest webResourceRequest) { public boolean shouldOverrideUrlLoading(@NotNull WebView view, @NotNull WebResourceRequest webResourceRequest) {
Uri uri = webResourceRequest.getUrl(); Uri uri = webResourceRequest.getUrl();
String host = uri.getHost(); String host = uri.getHost();
if ("localhost".equals(host)) { if ("localhost".equals(host)) {
// lokale URL im eingebetteten Browser öffnen // wiki URL, show in embedded browser
return false; return false;
} else { } else {
// externe Links im normalen Browser öffnen // external URL, show in external browser
Intent intent = new Intent(Intent.ACTION_VIEW, uri); Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent); startActivity(intent);
return true; return true;
@@ -157,57 +151,46 @@ public class MainActivity extends AppCompatActivity {
} }
/** /**
* Emuliert den Wikiserver und zu vermeiden, dass ein TCP-Port geöffnet werden muss. * Resolve internal URLs without running a real HTTP server,
* so we don't have to open a port.
*/ */
@Override @Override
@Nullable @Nullable
public WebResourceResponse shouldInterceptRequest (@NotNull WebView view, @NotNull WebResourceRequest request) { public WebResourceResponse shouldInterceptRequest (@NotNull WebView view, @NotNull WebResourceRequest request) {
try { try {
// URL-Pfad ermitteln // determine URL path
Uri uri = request.getUrl(); Uri uri = request.getUrl();
String encodedUrl = uri.toString(); String encodedUrl = uri.toString();
String url = URLDecoder.decode(encodedUrl, "UTF-8"); String url = URLDecoder.decode(encodedUrl, "UTF-8");
if (!url.startsWith(SERVER_BASE_URL)) { if (!url.startsWith(SERVER_BASE_URL)) {
return null; // Daten per HTTP laden return null;
} }
String urlPath = url.substring(SERVER_BASE_URL.length() - 1); // ersten "/" behalten String urlPath = url.substring(SERVER_BASE_URL.length() - 1); // extract URL path
int hashPos = urlPath.indexOf('#'); int hashPos = urlPath.indexOf('#');
if (hashPos >= 0) { if (hashPos >= 0) {
// cut off anchor beginning with '#' // cut off anchor beginning with '#'
urlPath = urlPath.substring(0, hashPos); urlPath = urlPath.substring(0, hashPos);
} }
// per URL-Mapping das zuständige Plugin aufrufen // dispatch URL path
HttpResponse response; HttpRequest httpRequest = new HttpRequest();
Plugin plugin = pluginService.getPluginByUrl(urlPath); httpRequest.clientIP = InetAddress.getLocalHost();
if (plugin != null) { httpRequest.httpHeader = Collections.emptyMap();
HttpRequest httpRequest = new HttpRequest(); httpRequest.method = "GET";
httpRequest.clientIP = InetAddress.getLocalHost(); httpRequest.url = urlPath;
httpRequest.httpHeader = Collections.emptyMap(); httpRequest.urlPath = urlPath;
httpRequest.method = "GET"; httpRequest.urlParameters = convertParameters(uri);
httpRequest.url = urlPath; httpRequest.httpBody = new byte[0];
httpRequest.urlPath = urlPath; HttpResponse response = requestDispatcher.handleRequest(httpRequest);
httpRequest.urlParameters = convertParameters(uri);
httpRequest.httpBody = new byte[0];
response = plugin.handleRequest(httpRequest); // send wiki content to browser
if (response == null) { InputStream responseData = new ByteArrayInputStream(response.content);
response = htmlService.generateErrorPage(404, "wiki.plugin.handleRequest.notsupported", plugin.getClass().getName()); return new WebResourceResponse(response.contentType, "UTF-8", responseData);
}
} else {
// unbekannte URL
response = htmlService.generateErrorPage(404, "wiki.server.url.unmapped", urlPath);
}
// Antwortdaten einspeisen
InputStream responseData = new ByteArrayInputStream(response.getContent());
return new WebResourceResponse(response.getContentType(),
"UTF-8", responseData);
} catch (IOException e) { } catch (IOException e) {
e.printStackTrace(); e.printStackTrace();
} }
return null; // Daten per HTTP laden return null; // load HTTP data
} }
@Override @Override
@@ -217,7 +200,6 @@ public class MainActivity extends AppCompatActivity {
} }
}); });
// weitere Einstellungen
WebSettings webSettings = webview.getSettings(); WebSettings webSettings = webview.getSettings();
webSettings.setAllowFileAccess(false); webSettings.setAllowFileAccess(false);
webSettings.setJavaScriptEnabled(true); webSettings.setJavaScriptEnabled(true);
@@ -233,6 +215,7 @@ public class MainActivity extends AppCompatActivity {
return result; return result;
} }
@SuppressLint("RestrictedApi")
@Override @Override
public boolean onCreateOptionsMenu(@NotNull Menu menu) { public boolean onCreateOptionsMenu(@NotNull Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present. // Inflate the menu; this adds items to the action bar if it is present.
@@ -255,30 +238,29 @@ public class MainActivity extends AppCompatActivity {
// automatically handle clicks on the Home/Up button, so long // automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml. // as you specify a parent activity in AndroidManifest.xml.
switch (item.getItemId()) { int itemId = item.getItemId();
case R.id.action_synchronize: if (itemId == R.id.action_synchronize) {
synchronizeServer(); synchronizeWithServer();
return true; return true;
case R.id.action_startpage: } else if (itemId == R.id.action_startpage) {
loadUrl(SERVER_BASE_URL); loadUrl(SERVER_BASE_URL);
return true; return true;
case R.id.action_settings: } else if (itemId == R.id.action_settings) {
showSettingsDialog(); showSettingsDialog();
return true; return true;
case R.id.action_help: } else if (itemId == R.id.action_help) {
String pagePathHelp = getWikiserverHelpUrl(); String pagePathHelp = getWikiserverHelpUrl();
loadUrl(pagePathHelp); loadUrl(pagePathHelp);
return true; return true;
case R.id.action_about: } else if (itemId == R.id.action_about) {
showAboutDialog(); showAboutDialog();
return true; return true;
default:
return super.onOptionsItemSelected(item);
} }
return super.onOptionsItemSelected(item);
} }
@Override @Override
public void onRequestPermissionsResult(int requestCode, @NotNull String[] permissions, @NotNull int[] grantResults) { public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
Log.d(TAG, "Permission granted by user: requestCode=" + requestCode Log.d(TAG, "Permission granted by user: requestCode=" + requestCode
+ ", permissions=" + Arrays.toString(permissions) + ", grantResults=" + Arrays.toString(grantResults)); + ", permissions=" + Arrays.toString(permissions) + ", grantResults=" + Arrays.toString(grantResults));
super.onRequestPermissionsResult(requestCode, permissions, grantResults); super.onRequestPermissionsResult(requestCode, permissions, grantResults);
@@ -286,7 +268,7 @@ public class MainActivity extends AppCompatActivity {
@Override @Override
public void onBackPressed() { public void onBackPressed() {
// webview.canGoBack() liefert immer false // webview.canGoBack() always returns false
WebBackForwardList backForwardList = webview.copyBackForwardList(); WebBackForwardList backForwardList = webview.copyBackForwardList();
Log.d(TAG, "Back button pressed, backForwardList index == " + backForwardList.getCurrentIndex()); Log.d(TAG, "Back button pressed, backForwardList index == " + backForwardList.getCurrentIndex());
if (backForwardList.getCurrentIndex() > 0) { if (backForwardList.getCurrentIndex() > 0) {
@@ -306,15 +288,15 @@ public class MainActivity extends AppCompatActivity {
} }
} }
public void onConfigurationHintClicked(@SuppressWarnings("unused") View view) { public void onConfigurationHintClicked(View view) {
showSettingsDialog(); showSettingsDialog();
} }
public void onSynchronizeHintClicked(@SuppressWarnings("unused") View view) { public void onSynchronizeHintClicked(View view) {
synchronizeServer(); synchronizeWithServer();
} }
public void onSearch(@SuppressWarnings("unused") View view) { public void onSearch(View view) {
EditText searchInput = findViewById(R.id.search_input); EditText searchInput = findViewById(R.id.search_input);
String query = searchInput.getText().toString(); String query = searchInput.getText().toString();
query = query.trim(); query = query.trim();
@@ -334,8 +316,7 @@ public class MainActivity extends AppCompatActivity {
String url = getWikiserverSearchUrl(query); String url = getWikiserverSearchUrl(query);
loadUrl(url); loadUrl(url);
// Der Wartedialog wird nach dem Anzeigen der Suchergebnisse durch Aufruf von // The wait dialog is closed by closeProgressDialog() after the search result is shown
// closeProgressDialog() geschlossen.
} }
private void showSettingsDialog() { private void showSettingsDialog() {
@@ -344,7 +325,7 @@ public class MainActivity extends AppCompatActivity {
} }
private void showAboutDialog() { private void showAboutDialog() {
// Versionsnummer einsetzen // insert version number
@SuppressLint("InflateParams") @SuppressLint("InflateParams")
View dialogView = getLayoutInflater().inflate(R.layout.about_layout, null); View dialogView = getLayoutInflater().inflate(R.layout.about_layout, null);
TextView versionText = dialogView.findViewById(R.id.label_version); TextView versionText = dialogView.findViewById(R.id.label_version);
@@ -353,7 +334,7 @@ public class MainActivity extends AppCompatActivity {
TextView copyrightText = dialogView.findViewById(R.id.label_copyright); TextView copyrightText = dialogView.findViewById(R.id.label_copyright);
copyrightText.setText(getString(R.string.about_copyright, year)); copyrightText.setText(getString(R.string.about_copyright, year));
// Dialog anzeigen // show dialog
AlertDialog.Builder builder = new AlertDialog.Builder(this); AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setView(dialogView); builder.setView(dialogView);
builder.setTitle(R.string.app_name); builder.setTitle(R.string.app_name);
@@ -363,13 +344,48 @@ public class MainActivity extends AppCompatActivity {
dialog.show(); dialog.show();
} }
private void synchronizeServer() { /**
SyncNowTask syncNowTask = new SyncNowTask(); * Called if user presses the synchronization button.
syncNowTask.execute(); */
private void synchronizeWithServer() {
synchronizationExecutorService.submit(this::runSynchronizationWithServer);
} }
/** /**
* Öffnet die angegebene URL im eingebetteten Browser. * Run synchronization of wiki content with wiki server.
* This method is run in an asynchronous background thread.
*/
private void runSynchronizationWithServer() {
int filesCount;
try {
filesCount = synchronizeWikiClient.synchronizeRepository(this::syncProgress);
} catch (ServiceException e) {
Log.e(TAG, "Error synchronizing repository with server", e);
runOnUiThread(() -> showToast(getString(R.string.settings_synchronize_failed)));
return;
}
if (filesCount > 0) {
runOnUiThread(() -> showToast(getString(R.string.settings_synchronize_successful, filesCount)));
WikiEngineApplication app = (WikiEngineApplication) getApplication();
app.resetServices();
CalendarSyncAdapter.requestCalendarSync(this);
} else {
runOnUiThread(() -> showToast(getString(R.string.settings_synchronize_not_necessary)));
}
runOnUiThread(this::hideProgressBar);
runOnUiThread(this::updateLayoutVisibility);
}
private void syncProgress(int progress, int max) {
runOnUiThread(() -> showProgressBar(progress, max));
}
/**
* Open a URL in the embedded browser.
*/ */
private void loadUrl(@NotNull String url) { private void loadUrl(@NotNull String url) {
Log.d(TAG, "Open URL " + url); Log.d(TAG, "Open URL " + url);
@@ -413,11 +429,11 @@ public class MainActivity extends AppCompatActivity {
boolean webViewVisible = !hostUnconfiguredHintVisible && !repositoryEmptyHintVisible; boolean webViewVisible = !hostUnconfiguredHintVisible && !repositoryEmptyHintVisible;
if (webViewVisible) { if (webViewVisible) {
searchArea.setVisibility(View.VISIBLE); searchArea.setVisibility(View.VISIBLE);
boolean previousVisible = (webView.getVisibility() == View.INVISIBLE); boolean previousVisible = (webView.getVisibility() == View.VISIBLE);
webView.setVisibility(View.VISIBLE); webView.setVisibility(View.VISIBLE);
// if WebView is shown the first time, go to start page // if WebView is shown the first time, go to start page
if (!previousVisible) { if (!previousVisible || webView.getUrl() == null) {
loadUrl(SERVER_BASE_URL); loadUrl(SERVER_BASE_URL);
} }
} else { } else {
@@ -450,61 +466,4 @@ public class MainActivity extends AppCompatActivity {
Toast toast = Toast.makeText(this, message, Toast.LENGTH_SHORT); Toast toast = Toast.makeText(this, message, Toast.LENGTH_SHORT);
toast.show(); toast.show();
} }
/**
* Synchronisiert mit dem Wikiserver in einem separaten Thread.
*/
@SuppressLint("StaticFieldLeak")
@SuppressWarnings("NonStaticInnerClassInSecureContext")
private class SyncNowTask extends AsyncTask<Void, ProgressData, Integer> implements SynchronizeWikiClient.ProgressFeedback {
@Nullable
@Override
protected Integer doInBackground(Void... params) {
try {
return synchronizeWikiClient.synchronizeRepository(this);
} catch (ServiceException e) {
Log.e(TAG, "Error synchronizing repository", e);
}
return null;
}
@Override
public void progress(int step, int total) {
ProgressData progressData = new ProgressData();
progressData.step = step;
progressData.total = total;
publishProgress(progressData);
}
@Override
protected void onProgressUpdate(ProgressData... progressData) {
showProgressBar(progressData[0].step, progressData[0].total);
}
@Override
protected void onPostExecute(@Nullable Integer filesCount) {
CalendarSyncAdapter.requestCalendarSync(MainActivity.this);
if (filesCount == null) {
showToast(getString(R.string.settings_synchronize_failed));
} else if (filesCount > 0) {
showToast(getString(R.string.settings_synchronize_successful, filesCount));
// Reset internal file caches
WikiEngineApplication app = (WikiEngineApplication) getApplication();
app.resetServices();
} else {
showToast(getString(R.string.settings_synchronize_not_necessary));
}
hideProgressBar();
updateLayoutVisibility();
}
}
private static class ProgressData {
public int step;
public int total;
}
} }
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -18,9 +18,7 @@
package net.moasdawiki.app; package net.moasdawiki.app;
import android.annotation.SuppressLint;
import android.content.SharedPreferences; import android.content.SharedPreferences;
import android.os.AsyncTask;
import android.os.Bundle; import android.os.Bundle;
import android.text.InputFilter; import android.text.InputFilter;
import android.text.InputType; import android.text.InputType;
@@ -35,11 +33,11 @@ import androidx.preference.PreferenceFragmentCompat;
import net.moasdawiki.base.Settings; import net.moasdawiki.base.Settings;
import net.moasdawiki.service.repository.RepositoryService; import net.moasdawiki.service.repository.RepositoryService;
import org.jetbrains.annotations.NotNull;
import java.text.SimpleDateFormat; import java.text.SimpleDateFormat;
import java.util.Date; import java.util.Date;
import java.util.Locale; import java.util.Locale;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class SettingsFragment extends PreferenceFragmentCompat implements SharedPreferences.OnSharedPreferenceChangeListener { public class SettingsFragment extends PreferenceFragmentCompat implements SharedPreferences.OnSharedPreferenceChangeListener {
@@ -47,15 +45,16 @@ public class SettingsFragment extends PreferenceFragmentCompat implements Shared
private SynchronizeWikiClient synchronizeWikiClient; private SynchronizeWikiClient synchronizeWikiClient;
private RepositoryService repositoryService; private RepositoryService repositoryService;
private Settings settings; private Settings settings;
private ExecutorService synchronizationExecutorService;
@Override @Override
public void onCreate(@Nullable Bundle savedInstanceState) { public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState); super.onCreate(savedInstanceState);
//noinspection ConstantConditions
WikiEngineApplication app = (WikiEngineApplication) getContext().getApplicationContext(); WikiEngineApplication app = (WikiEngineApplication) getContext().getApplicationContext();
synchronizeWikiClient = app.getSynchronizeWikiClient(); synchronizeWikiClient = app.getSynchronizeWikiClient();
repositoryService = app.getServiceLocator().getRepositoryService(); repositoryService = app.getRepositoryService();
settings = app.getServiceLocator().getSettings(); settings = app.getSettings();
synchronizationExecutorService = Executors.newSingleThreadExecutor();
} }
@Override @Override
@@ -167,37 +166,30 @@ public class SettingsFragment extends PreferenceFragmentCompat implements Shared
} }
} }
/**
* Called if the user changes the connection settings to give fast feedback.
*/
private void checkServerConnection() { private void checkServerConnection() {
new ConnectServerTask().execute(); synchronizationExecutorService.submit(this::runCheckServerConnection);
} }
/** /**
* Checks the connection to the Wiki server. * Run server connection check.
* This method is run in an asynchronous background thread.
*/ */
@SuppressLint("StaticFieldLeak") private void runCheckServerConnection() {
@SuppressWarnings("NonStaticInnerClassInSecureContext") SharedPreferences preferences = getPreferenceManager().getSharedPreferences();
private class ConnectServerTask extends AsyncTask<Void, Void, Boolean> { String host = preferences.getString(Constants.PREFERENCES_SYNC_SERVER_HOST, null);
@Override if (host == null) {
@NotNull Log.d(TAG, "Cancel ConnectServerTask because server host is not configured");
protected Boolean doInBackground(Void... params) { return;
SharedPreferences preferences = getPreferenceManager().getSharedPreferences();
String host = preferences.getString(Constants.PREFERENCES_SYNC_SERVER_HOST, null);
if (host == null) {
Log.d(TAG, "Cancel ConnectServerTask because server host is not configured");
return false;
}
// Connect with server
return synchronizeWikiClient.createAndCheckSession();
} }
@Override boolean success = synchronizeWikiClient.createAndCheckSession();
protected void onPostExecute(@NotNull Boolean success) { updateStatusText();
updateStatusText();
if (!success) { if (!success) {
showToast(getString(R.string.settings_search_failed)); showToast(getString(R.string.settings_search_failed));
}
} }
} }
} }
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -29,20 +29,9 @@ import androidx.preference.PreferenceManager;
import net.moasdawiki.base.Logger; import net.moasdawiki.base.Logger;
import net.moasdawiki.base.ServiceException; import net.moasdawiki.base.ServiceException;
import net.moasdawiki.base.Settings; import net.moasdawiki.base.Settings;
import net.moasdawiki.plugin.ServiceLocator;
import net.moasdawiki.plugin.sync.AbstractSyncXml;
import net.moasdawiki.plugin.sync.CheckSessionResponseXml;
import net.moasdawiki.plugin.sync.CheckSessionXml;
import net.moasdawiki.plugin.sync.CreateSessionResponseXml;
import net.moasdawiki.plugin.sync.CreateSessionXml;
import net.moasdawiki.plugin.sync.ErrorResponseXml;
import net.moasdawiki.plugin.sync.ListModifiedFilesResponseXml;
import net.moasdawiki.plugin.sync.ListModifiedFilesXml;
import net.moasdawiki.plugin.sync.ReadFileResponseXml;
import net.moasdawiki.plugin.sync.ReadFileXml;
import net.moasdawiki.plugin.sync.SingleFileXml;
import net.moasdawiki.service.repository.AnyFile; import net.moasdawiki.service.repository.AnyFile;
import net.moasdawiki.service.repository.RepositoryService; import net.moasdawiki.service.repository.RepositoryService;
import net.moasdawiki.service.sync.*;
import net.moasdawiki.util.DateUtils; import net.moasdawiki.util.DateUtils;
import net.moasdawiki.util.xml.XmlGenerator; import net.moasdawiki.util.xml.XmlGenerator;
import net.moasdawiki.util.xml.XmlParser; import net.moasdawiki.util.xml.XmlParser;
@@ -67,8 +56,6 @@ import java.util.Enumeration;
/** /**
* Sucht einen Wikiserver in Netzwerk und synchronisiert alle Wikidateien * Sucht einen Wikiserver in Netzwerk und synchronisiert alle Wikidateien
* im eigenen Repository.. * im eigenen Repository..
*
* @author Herbert Reiter
*/ */
public class SynchronizeWikiClient { public class SynchronizeWikiClient {
@@ -85,13 +72,15 @@ public class SynchronizeWikiClient {
@NotNull @NotNull
private final RepositoryService repositoryService; private final RepositoryService repositoryService;
@NotNull @NotNull
private final SecureRandom random = new SecureRandom(); private final SecureRandom random;
public SynchronizeWikiClient(@NotNull Context mContext, @NotNull ServiceLocator serviceLocator) { public SynchronizeWikiClient(@NotNull Context mContext, @NotNull Logger logger, @NotNull Settings settings,
@NotNull RepositoryService repositoryService) {
this.mContext = mContext; this.mContext = mContext;
this.logger = serviceLocator.getLogger(); this.logger = logger;
this.settings = serviceLocator.getSettings(); this.settings = settings;
this.repositoryService = serviceLocator.getRepositoryService(); this.repositoryService = repositoryService;
this.random = new SecureRandom();
} }
/** /**
@@ -263,7 +252,7 @@ public class SynchronizeWikiClient {
return 0; return 0;
} }
for (int i = 0; i < fileCount && !feedback.isCancelled(); i++) { for (int i = 0; i < fileCount; i++) {
feedback.progress(i, fileCount); feedback.progress(i, fileCount);
SingleFileXml serverFile = response.fileList.get(i); SingleFileXml serverFile = response.fileList.get(i);
try { try {
@@ -279,14 +268,12 @@ public class SynchronizeWikiClient {
app.resetServices(); app.resetServices();
// Update last sync time // Update last sync time
if (!feedback.isCancelled()) { Date currentServerDate = DateUtils.parseUtcDate(response.currentServerTime);
Date currentServerDate = DateUtils.parseUtcDate(response.currentServerTime); if (currentServerDate != null) {
if (currentServerDate != null) { long newLastSyncServerTimeMs = currentServerDate.getTime();
long newLastSyncServerTimeMs = currentServerDate.getTime(); SharedPreferences.Editor editor = preferences.edit();
SharedPreferences.Editor editor = preferences.edit(); editor.putLong(Constants.PREFERENCES_SYNC_SERVER_TIME, newLastSyncServerTimeMs);
editor.putLong(Constants.PREFERENCES_SYNC_SERVER_TIME, newLastSyncServerTimeMs); editor.apply();
editor.apply();
}
} }
return fileCount; return fileCount;
@@ -471,9 +458,8 @@ public class SynchronizeWikiClient {
* Interface, um den Fortschritt während der Synchronisierung mit dem Server * Interface, um den Fortschritt während der Synchronisierung mit dem Server
* zurückzuübermitteln. * zurückzuübermitteln.
*/ */
@FunctionalInterface
public interface ProgressFeedback { public interface ProgressFeedback {
@SuppressWarnings("BooleanMethodIsAlwaysInverted")
boolean isCancelled();
void progress(int step, int total); void progress(int step, int total);
} }
} }
@@ -1,6 +1,6 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2021 Herbert Reiter (herbert@moasdawiki.net)
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@@ -23,58 +23,89 @@ import android.app.Application;
import net.moasdawiki.base.Logger; import net.moasdawiki.base.Logger;
import net.moasdawiki.base.Messages; import net.moasdawiki.base.Messages;
import net.moasdawiki.base.Settings; import net.moasdawiki.base.Settings;
import net.moasdawiki.plugin.PluginService; import net.moasdawiki.server.RequestDispatcher;
import net.moasdawiki.plugin.ServiceLocator; import net.moasdawiki.service.handler.FileDownloadHandler;
import net.moasdawiki.service.handler.SearchHandler;
import net.moasdawiki.service.handler.ViewPageHandler;
import net.moasdawiki.service.render.HtmlService; import net.moasdawiki.service.render.HtmlService;
import net.moasdawiki.service.repository.FilesystemRepositoryService;
import net.moasdawiki.service.repository.RepositoryService; import net.moasdawiki.service.repository.RepositoryService;
import net.moasdawiki.service.search.SearchService; import net.moasdawiki.service.search.SearchService;
import net.moasdawiki.service.transform.IncludePageTransformer;
import net.moasdawiki.service.transform.KontaktseiteTransformer;
import net.moasdawiki.service.transform.TerminTransformer;
import net.moasdawiki.service.transform.TransformWikiPage;
import net.moasdawiki.service.transform.TransformerService;
import net.moasdawiki.service.transform.WikiTagsTransformer;
import net.moasdawiki.service.wiki.WikiService; import net.moasdawiki.service.wiki.WikiService;
import net.moasdawiki.service.wiki.WikiServiceImpl;
import java.io.File; import java.io.File;
/** /**
* Verwaltet den Lebenszyklus der Wiki Engine. Muss außerhalb der Activities erfolgen, weil diese * Main control of the wiki App.
* z.B. beim Drehen des Bildschirm neu erzeugt werden.
* *
* @author Herbert Reiter * Must be run globally for the App, i.e. outside of an activity.
*/ */
public class WikiEngineApplication extends Application { public class WikiEngineApplication extends Application {
private static final String REPOSITORY_ROOT_PATH_DEFAULT = "repository";
private Logger logger; private Logger logger;
private ServiceLocator serviceLocator; private RepositoryService repositoryService;
private Settings settings;
private Messages messages;
private WikiService wikiService;
private SearchService searchService;
private SynchronizeWikiClient synchronizeWikiClient; private SynchronizeWikiClient synchronizeWikiClient;
private TerminTransformer terminTransformer;
private RequestDispatcher requestDispatcher;
@Override @Override
public void onCreate() { public void onCreate() {
super.onCreate(); super.onCreate();
logger = new Logger(System.out); logger = new Logger(System.out);
logger.write("MoasdaWiki app starting"); logger.write("MoasdaWiki App starting");
File internalStorageRepositoryRoot = new File(getFilesDir(), "repository"); File internalStorageRepositoryRoot = new File(getFilesDir(), REPOSITORY_ROOT_PATH_DEFAULT);
RepositoryService repositoryService = new FilesystemRepositoryService(logger, internalStorageRepositoryRoot);
repositoryService.init();
WikiService wikiService = new WikiServiceImpl(logger, repositoryService);
SearchService searchService = new SearchService(logger, wikiService);
Settings settings = new AndroidSettings(logger, repositoryService, AndroidSettings.getConfigFileApp());
Messages messages = new Messages(logger, settings, repositoryService);
PluginService pluginService = new PluginService(logger, settings);
HtmlService htmlService = new HtmlService(logger, settings, messages, wikiService, pluginService);
serviceLocator = new ServiceLocator(logger, settings, messages, repositoryService, wikiService, htmlService, searchService, pluginService);
pluginService.loadPlugins(serviceLocator); // basic services
repositoryService = new RepositoryService(logger, internalStorageRepositoryRoot, false);
settings = new AndroidSettings(logger, repositoryService, Settings.getConfigFileApp());
messages = new Messages(logger, settings, repositoryService);
wikiService = new WikiService(logger, repositoryService, false);
searchService = new SearchService(logger, repositoryService, wikiService, false);
synchronizeWikiClient = new SynchronizeWikiClient(this, serviceLocator); // App: use SynchronizeWikiClient instead of SynchronizationService
synchronizeWikiClient = new SynchronizeWikiClient(this, logger, settings, repositoryService);
// transformers
// do not run the SynchronizationPageTransformer
IncludePageTransformer includePageTransformer = new IncludePageTransformer(logger, wikiService);
KontaktseiteTransformer kontaktseiteTransformer = new KontaktseiteTransformer();
terminTransformer = new TerminTransformer(logger, messages, repositoryService, wikiService, false);
WikiTagsTransformer wikiTagsTransformer = new WikiTagsTransformer(logger, settings, messages, wikiService);
// list of transformers, the order matters
TransformWikiPage[] transformers = {includePageTransformer, kontaktseiteTransformer, terminTransformer, wikiTagsTransformer};
TransformerService transformerService = new TransformerService(transformers);
// more services
HtmlService htmlService = new HtmlService(logger, settings, messages, wikiService, transformerService);
// HTTP handlers
// do not run the EditorHandler
ViewPageHandler viewPageHandler = new ViewPageHandler(logger, settings, wikiService, htmlService);
SearchHandler searchHandler = new SearchHandler(logger, settings, messages, wikiService, searchService, htmlService);
FileDownloadHandler fileDownloadHandler = new FileDownloadHandler(logger, settings, repositoryService, htmlService);
requestDispatcher = new RequestDispatcher(htmlService, viewPageHandler,
searchHandler, null, fileDownloadHandler, null);
} }
public void resetServices() { public void resetServices() {
serviceLocator.getRepositoryService().rebuildCache(); repositoryService.reset();
serviceLocator.getWikiService().reset(); settings.reset();
serviceLocator.getSettings().reset(); messages.reset();
serviceLocator.getMessages().reset(); wikiService.reset();
serviceLocator.getPluginService().loadPlugins(serviceLocator); searchService.reset();
terminTransformer.reset();
} }
@Override @Override
@@ -83,11 +114,23 @@ public class WikiEngineApplication extends Application {
super.onTerminate(); super.onTerminate();
} }
public ServiceLocator getServiceLocator() { public RepositoryService getRepositoryService() {
return serviceLocator; return repositoryService;
}
public Settings getSettings() {
return settings;
}
public TerminTransformer getTerminTransformer() {
return terminTransformer;
} }
public SynchronizeWikiClient getSynchronizeWikiClient() { public SynchronizeWikiClient getSynchronizeWikiClient() {
return synchronizeWikiClient; return synchronizeWikiClient;
} }
public RequestDispatcher getRequestDispatcher() {
return requestDispatcher;
}
} }
Binary file not shown.

Before

Width:  |  Height:  |  Size: 255 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 503 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 444 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 159 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 334 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 289 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 225 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 623 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 543 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 341 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 918 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 801 B

+1 -1
View File
@@ -6,7 +6,7 @@ buildscript {
google() google()
} }
dependencies { dependencies {
classpath 'com.android.tools.build:gradle:3.5.3' classpath 'com.android.tools.build:gradle:4.1.1'
// NOTE: Do not place your application dependencies here; they belong // NOTE: Do not place your application dependencies here; they belong
// in the individual module build.gradle files // in the individual module build.gradle files
@@ -0,0 +1 @@
- Volltextsuche beschleunigen, indem im ersten Schritt nur noch in den Seitentiteln gesucht wird.
@@ -0,0 +1 @@
- Fastlane Metadaten für das F-Droid-Repository hinzufügen
@@ -0,0 +1,2 @@
- Deutsche Fastlane Metadaten für das F-Droid-Repository hinzufügen
- Versionen der verwendeten Bibliotheken aktualisieren
@@ -0,0 +1,3 @@
- Volltextsuche beschleunigen durch einen Suchindex
- Bugfix: Nicht zur Wiki-Startseite wechseln wenn die App wieder geöffnet wird
- Auf moasdawiki-server 2.3.1 aktualisieren
@@ -0,0 +1,3 @@
- Bugfix: Zeige Startseite beim App-Start
- Bugfix: Bei der Suche ausschließlich den Suchindex-Cache verwenden
- Auf moasdawiki-server 2.3.3 aktualisieren
@@ -0,0 +1,3 @@
- Auf moasdawiki-server 2.4.2 aktualisieren
- Bibliotheks-Abhängigkeiten aktualisieren
- Refaktorierung: Überholtes AsyncTask ersetzen
@@ -0,0 +1,4 @@
- Beschreibung für F-Droid aktualisieren
- Copyright-Jahr aktualisieren
- Bugfix: Wiki-Editor deaktivieren
- Auf moasdawiki-server 2.4.3 aktualisieren
@@ -0,0 +1,22 @@
MoasdaWiki App ist eine Erweiterung zum MoasdaWiki-Server, um eine Kopie deiner Wikiinhalte auf deinem Mobilgerät zu haben.
Funkionalitäten:
- Synchronisiert die Inhalte von einer MoasdaWiki-Server-Instanz.
- Datenschutz: Es wird keine Cloud-Verbindung hergestellt, verbindet sich direkt mit dem Server in deinem privaten Netzwerk.
- Kalender-Integration, zeigt Geburtstage und Termine im Kalender des Mobilgeräts (nur in der deutschen Version).
Synchronisieren des Inhalts mit einem MoasdaWiki-Server:
1. Lade MoasdaWiki-Server von https://moasdawiki.net/ herunter.
2. Setze eine MoasdaWiki-Server-Instance in deinem LAN auf.
3. Installiere und öffne die MoasdaWiki-App.
4. Du kannst einen Hinweis sehen, dass die App zunächst konfiguriert werden muss. Drücke auf den Hinweis.
5. Drücke auf "Hostname" und gib den Hostnamen der Server-Instanz ein, z.B. `192.168.1.101`. Drücke OK.
6. Im Statusbereich darunter solltest du nun "Erfordert Berechtigung am Server" sehen. Andernfalls prüfe erneut Hostname und Port.
7. Öffne auf der Serverseite die Wikiseite im Browser, klicke auf "Hilfe" und "Synchronisierung".
8. Du siehst eine Liste von Geräte und Synchronisierungs-Sitzungen. Überprüfe den Gerätenamen und klicke auf "Erlauben".
9. Zurück in der App drücke den Zurück-Button (&larr;) in der Ecke links oben, um zum Hauptdialog zurückzukommen. Nun kannst du einen Hinweis sehen, dass die App synchronisiert werden muss. Drücke auf diesen Hinweis.
10. Nun solltest du alle Inhalte des Servers auch in der App haben und die Wikiseite "Startseite-App" sehen.
Hinweis: Der Wikiinhalt kann nicht in der App modifiziert werden, da es keinen Spaß macht Wiki-Syntax auf dem Mobilgerät zu tippen. Änderungen müssen über den MoasdaWiki-Server erfolgen.
Binary file not shown.

After

Width:  |  Height:  |  Size: 348 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 508 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 356 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 406 KiB

@@ -0,0 +1 @@
App zum Synchronizieren der Inhalte vom MoasdaWiki-Server.
@@ -0,0 +1 @@
- Speed up full text search by searching only in the page titles in the first step
@@ -0,0 +1 @@
- Add fastlane metadata for F-Droid repository
@@ -0,0 +1,2 @@
- Add German fastlane metadata for F-Droid repository
- Update library dependencies
@@ -0,0 +1,3 @@
- Speed up full text search by a search index
- Bugfix: Don't go back to start page after App restore
- Update to moasdawiki-server 2.3.1
@@ -0,0 +1,3 @@
- Bugfix: Show start page on app startup
- Bugfix: For searches only use the search index cache
- Update to moasdawiki-server 2.3.3
@@ -0,0 +1,3 @@
- Update to moasdawiki-server 2.4.2
- Upgrade library dependencies
- Refactoring: Replace deprecated AsyncTask
@@ -0,0 +1,4 @@
- Update descriptions for F-Droid
- Update copyright year
- Bugfix: Disable wiki editor handler
- Update to moasdawiki-server 2.4.3
@@ -0,0 +1,22 @@
MoasdaWiki App is an extension of the MoasdaWiki Server to have a copy of your Wiki content on your mobile device.
Features:
- Synchronizes the data from a MoasdaWiki Server instance.
- Data privacy: No cloud connection established, directly connects to the server inside your private network.
- Calendar integration, shows birthdays and events in the mobile calendar (German version only).
Synchronize content with a MoasdaWiki Server:
1. Download MoasdaWiki Server from https://moasdawiki.net/.
2. Set up a MoasdaWiki Server instance in your LAN.
3. Install and open the MoasdaWiki App.
4. You can see a hint that the App has to be configured first. Press on that hint.
5. Press on "Host name" and enter the host name of the server instance, e.g. `192.168.1.101`. Press OK.
6. In the status section below you should see "Needs authorization at server". Otherwise check host name and port again.
7. On server side open the Wiki page in a browser, click on "Help" and "Synchronization".
8. You can see a list of devices and synchronization sessions. Check the device name and click on "Grant".
9. Back in the app press the back button (&larr;) on the upper left corner to get back to the main dialog. Now you can see a hint that the app has to be synchronized. Press on that hint.
10. Now you should have all the server content also in the app and you can see the "Home-App" wiki page.
Hint: Content cannot be modified within the app as it is no fun to type Wiki syntax on the mobile device, changes have to be done via the MoasdaWiki Server.
Binary file not shown.

After

Width:  |  Height:  |  Size: 177 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 235 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 182 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 191 KiB

@@ -0,0 +1 @@
App to synchronize data from MoasdaWiki Server.
+2 -2
View File
@@ -1,6 +1,6 @@
#Wed Jan 01 21:15:49 CET 2020 #Sun Dec 27 12:01:31 CET 2020
distributionBase=GRADLE_USER_HOME distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists zipStorePath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-all.zip distributionUrl=https\://services.gradle.org/distributions/gradle-6.5-all.zip