Compare commits

..

37 Commits

Author SHA1 Message Date
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
51 changed files with 354 additions and 410 deletions
+4 -1
View File
@@ -1,7 +1,7 @@
# IntelliJ
.idea/
out/
moasdawiki-app.iml
*.iml
# Gradle
.gradle/
@@ -10,3 +10,6 @@ release/
# Git
local.properties
# Project
TODO.md
+27
View File
@@ -1,5 +1,32 @@
# Changelog
## 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)
- Speed up full text search by searching only in the page titles in the first step
+8 -3
View File
@@ -5,20 +5,25 @@
MoasdaWiki App is an extension of the MoasdaWiki Server to have a copy of
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
- Synchronizes the data from a MoasdaWiki Server instance.
- 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,
changes have to be done via the MoasdaWiki Server.
- Calendar integration, shows birthdays and events in the mobile calendar (German version only).
## 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. 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.
-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'
android {
compileSdkVersion 29 // 29 = Android 10
compileSdkVersion 30 // 30 = Android 11
defaultConfig {
applicationId "net.moasdawiki.app"
minSdkVersion 26 // 26 = Oreo 8.0
targetSdkVersion 29 // should be same as compileSdkVersion
versionCode 15
versionName "2.2.0.0"
targetSdkVersion 30 // should be same as compileSdkVersion
versionCode 20
versionName "2.4.2.0"
archivesBaseName = "moasdawiki-" + versionName + "-" + versionCode
}
sourceSets {
main{
main {
java {
exclude "net/moasdawiki/plugin/sync/SynchronizationPlugin*"
}
@@ -24,10 +24,10 @@ android {
}
dependencies {
implementation 'net.moasdawiki:moasdawiki-server:2.2.0'
implementation 'net.moasdawiki:moasdawiki-server:2.4.2'
implementation fileTree(include: ['*.jar'], dir: 'libs')
implementation 'androidx.appcompat:appcompat:1.1.0'
implementation 'androidx.preference:preference:1.1.0'
compileOnly 'org.jetbrains:annotations:18.0.0'
testImplementation 'org.testng:testng:6.14.3'
implementation 'androidx.appcompat:appcompat:1.2.0'
implementation 'androidx.preference:preference:1.1.1'
compileOnly 'org.jetbrains:annotations:20.1.0'
testImplementation 'org.testng:testng:7.3.0'
}
@@ -30,6 +30,7 @@ public class AndroidSettings extends Settings {
super(logger, repositoryService, configFileName);
}
@Override
@NotNull
public String getVersion() {
return BuildConfig.VERSION_NAME;
@@ -43,9 +43,7 @@ import androidx.preference.PreferenceManager;
import android.util.Log;
import android.widget.Toast;
import net.moasdawiki.plugin.Plugin;
import net.moasdawiki.plugin.PluginService;
import net.moasdawiki.plugin.TerminPlugin;
import net.moasdawiki.service.transform.TerminTransformer;
import net.moasdawiki.util.PathUtils;
import org.jetbrains.annotations.NotNull;
@@ -87,7 +85,7 @@ public class CalendarSyncAdapter extends AbstractThreadedSyncAdapter {
@Override
public void onPerformSync(Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) {
Log.d(TAG, "Begin of onPerformSync()");
List<TerminPlugin.Event> events = getWikiEvents();
List<TerminTransformer.Event> events = getWikiEvents();
String calendarId = createCalendar();
if (calendarId != null) {
deleteAllEvents(calendarId);
@@ -106,21 +104,15 @@ public class CalendarSyncAdapter extends AbstractThreadedSyncAdapter {
* Imports all events on Wiki pages to the Android calendar.
*/
@NotNull
private List<TerminPlugin.Event> getWikiEvents() {
private List<TerminTransformer.Event> getWikiEvents() {
Log.d(TAG, "Reading Wiki events");
WikiEngineApplication app = (WikiEngineApplication) getContext();
PluginService pluginService = app.getServiceLocator().getPluginService();
TerminPlugin terminPlugin = null;
for (Plugin plugin : pluginService.getPlugins()) {
if (plugin instanceof TerminPlugin) {
terminPlugin = (TerminPlugin) plugin;
}
}
if (terminPlugin == null) {
Log.e(TAG, "TerminPlugin not found, cannot retrieve event list");
TerminTransformer terminTransformer = app.getTerminTransformer();
if (terminTransformer == null) {
Log.e(TAG, "TerminTransformer not initialized yet, cannot retrieve event list");
return Collections.emptyList();
}
List<TerminPlugin.Event> events = terminPlugin.getEvents();
List<TerminTransformer.Event> events = terminTransformer.getEvents();
Log.d(TAG, "Wiki events found: " + events.size());
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
* 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");
for (TerminPlugin.Event event : events) {
for (TerminTransformer.Event event : events) {
String title = event.description;
if (title == null) {
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);
if (eventId != null) {
addReminder(eventId);
@@ -294,8 +286,7 @@ public class CalendarSyncAdapter extends AbstractThreadedSyncAdapter {
if (!ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.WRITE_CALENDAR)) {
Log.d(TAG, "User has permanently denied permission WRITE_CALENDAR, informing him");
String hint = context.getString(R.string.calendar_permission_request, "WRITE_CALENDAR");
Toast toast = Toast.makeText(context, hint, Toast.LENGTH_SHORT);
toast.show();
activity.runOnUiThread(() -> Toast.makeText(context, hint, Toast.LENGTH_SHORT).show());
}
Log.d(TAG, "Ask for permission WRITE_CALENDAR");
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)) {
Log.d(TAG, "User has permanently denied permission READ_CALENDAR, informing him");
String hint = context.getString(R.string.calendar_permission_request, "READ_CALENDAR");
Toast toast = Toast.makeText(context, hint, Toast.LENGTH_SHORT);
toast.show();
activity.runOnUiThread(() -> Toast.makeText(context, hint, Toast.LENGTH_SHORT).show());
}
Log.d(TAG, "Ask for permission READ_CALENDAR");
ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.READ_CALENDAR}, 0);
@@ -20,8 +20,6 @@ package net.moasdawiki.app;
/**
* Enthält zentrale Konstanten.
*
* @author Herbert Reiter
*/
public abstract class Constants {
public static final String PREFERENCES_SYNC_SERVER_HOST = "sync_server_host";
@@ -25,7 +25,6 @@ import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
import android.net.Uri;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.KeyEvent;
@@ -47,17 +46,16 @@ import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.view.menu.MenuBuilder;
import androidx.preference.PreferenceManager;
import net.moasdawiki.base.ServiceException;
import net.moasdawiki.base.Settings;
import net.moasdawiki.plugin.Plugin;
import net.moasdawiki.plugin.PluginService;
import net.moasdawiki.server.HttpRequest;
import net.moasdawiki.server.HttpResponse;
import net.moasdawiki.service.render.HtmlService;
import net.moasdawiki.server.RequestDispatcher;
import net.moasdawiki.service.HttpResponse;
import net.moasdawiki.service.repository.RepositoryService;
import net.moasdawiki.util.EscapeUtils;
@@ -74,41 +72,39 @@ import java.util.Calendar;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/**
* Steuert das Verhalten des Hauptfensters inkl. eingebettetem Browser.
*
* @author Herbert Reiter
* Displays the main window with the embedded wiki browser.
*/
public class MainActivity extends AppCompatActivity {
private static final String TAG = "MainActivity";
private static final String SERVER_BASE_URL = "http://localhost:1/";
private Settings settings;
private RepositoryService repositoryService;
private PluginService pluginService;
private HtmlService htmlService;
private Settings settings;
private SynchronizeWikiClient synchronizeWikiClient;
private RequestDispatcher requestDispatcher;
private WebView webview;
private long backButtonPressedTimestamp;
private ExecutorService synchronizationExecutorService;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main_layout);
// AndroidMainService holen
WikiEngineApplication app = (WikiEngineApplication) getApplication();
settings = app.getServiceLocator().getSettings();
repositoryService = app.getServiceLocator().getRepositoryService();
pluginService = app.getServiceLocator().getPluginService();
htmlService = app.getServiceLocator().getHtmlService();
repositoryService = app.getRepositoryService();
settings = app.getSettings();
synchronizeWikiClient = app.getSynchronizeWikiClient();
requestDispatcher = app.getRequestDispatcher();
synchronizationExecutorService = Executors.newSingleThreadExecutor();
// eingebetteten Browser konfigurieren
initWebView();
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")
private void initWebView() {
// Cookies deaktivieren
CookieManager cookieManager = CookieManager.getInstance();
cookieManager.setAcceptCookie(false);
// localhost-URLs in WebView anzeigen
webview = findViewById(R.id.web_browser);
webview.setWebChromeClient(new WebChromeClient());
webview.setWebViewClient(new WebViewClient() {
/**
* Steuert, welche Links im eingebetteten Browser und welche im externen Browser
* geöffnet werden sollen.
* Dispatch URLs the user clicks on.
* External URLs will be shown in an external browser.
*/
@Override
public boolean shouldOverrideUrlLoading(@NotNull WebView view, @NotNull WebResourceRequest webResourceRequest) {
Uri uri = webResourceRequest.getUrl();
String host = uri.getHost();
if ("localhost".equals(host)) {
// lokale URL im eingebetteten Browser öffnen
// wiki URL, show in embedded browser
return false;
} else {
// externe Links im normalen Browser öffnen
// external URL, show in external browser
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
return true;
@@ -157,30 +151,28 @@ 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
@Nullable
public WebResourceResponse shouldInterceptRequest (@NotNull WebView view, @NotNull WebResourceRequest request) {
try {
// URL-Pfad ermitteln
// determine URL path
Uri uri = request.getUrl();
String encodedUrl = uri.toString();
String url = URLDecoder.decode(encodedUrl, "UTF-8");
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('#');
if (hashPos >= 0) {
// cut off anchor beginning with '#'
urlPath = urlPath.substring(0, hashPos);
}
// per URL-Mapping das zuständige Plugin aufrufen
HttpResponse response;
Plugin plugin = pluginService.getPluginByUrl(urlPath);
if (plugin != null) {
// dispatch URL path
HttpRequest httpRequest = new HttpRequest();
httpRequest.clientIP = InetAddress.getLocalHost();
httpRequest.httpHeader = Collections.emptyMap();
@@ -189,25 +181,16 @@ public class MainActivity extends AppCompatActivity {
httpRequest.urlPath = urlPath;
httpRequest.urlParameters = convertParameters(uri);
httpRequest.httpBody = new byte[0];
HttpResponse response = requestDispatcher.handleRequest(httpRequest);
response = plugin.handleRequest(httpRequest);
if (response == null) {
response = htmlService.generateErrorPage(404, "wiki.plugin.handleRequest.notsupported", plugin.getClass().getName());
}
} 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);
// send wiki content to browser
InputStream responseData = new ByteArrayInputStream(response.content);
return new WebResourceResponse(response.contentType, "UTF-8", responseData);
} catch (IOException e) {
e.printStackTrace();
}
return null; // Daten per HTTP laden
return null; // load HTTP data
}
@Override
@@ -217,7 +200,6 @@ public class MainActivity extends AppCompatActivity {
}
});
// weitere Einstellungen
WebSettings webSettings = webview.getSettings();
webSettings.setAllowFileAccess(false);
webSettings.setJavaScriptEnabled(true);
@@ -233,6 +215,7 @@ public class MainActivity extends AppCompatActivity {
return result;
}
@SuppressLint("RestrictedApi")
@Override
public boolean onCreateOptionsMenu(@NotNull Menu menu) {
// 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
// as you specify a parent activity in AndroidManifest.xml.
switch (item.getItemId()) {
case R.id.action_synchronize:
synchronizeServer();
int itemId = item.getItemId();
if (itemId == R.id.action_synchronize) {
synchronizeWithServer();
return true;
case R.id.action_startpage:
} else if (itemId == R.id.action_startpage) {
loadUrl(SERVER_BASE_URL);
return true;
case R.id.action_settings:
} else if (itemId == R.id.action_settings) {
showSettingsDialog();
return true;
case R.id.action_help:
} else if (itemId == R.id.action_help) {
String pagePathHelp = getWikiserverHelpUrl();
loadUrl(pagePathHelp);
return true;
case R.id.action_about:
} else if (itemId == R.id.action_about) {
showAboutDialog();
return true;
default:
return super.onOptionsItemSelected(item);
}
return super.onOptionsItemSelected(item);
}
@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
+ ", permissions=" + Arrays.toString(permissions) + ", grantResults=" + Arrays.toString(grantResults));
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
@@ -286,7 +268,7 @@ public class MainActivity extends AppCompatActivity {
@Override
public void onBackPressed() {
// webview.canGoBack() liefert immer false
// webview.canGoBack() always returns false
WebBackForwardList backForwardList = webview.copyBackForwardList();
Log.d(TAG, "Back button pressed, backForwardList index == " + backForwardList.getCurrentIndex());
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();
}
public void onSynchronizeHintClicked(@SuppressWarnings("unused") View view) {
synchronizeServer();
public void onSynchronizeHintClicked(View view) {
synchronizeWithServer();
}
public void onSearch(@SuppressWarnings("unused") View view) {
public void onSearch(View view) {
EditText searchInput = findViewById(R.id.search_input);
String query = searchInput.getText().toString();
query = query.trim();
@@ -334,8 +316,7 @@ public class MainActivity extends AppCompatActivity {
String url = getWikiserverSearchUrl(query);
loadUrl(url);
// Der Wartedialog wird nach dem Anzeigen der Suchergebnisse durch Aufruf von
// closeProgressDialog() geschlossen.
// The wait dialog is closed by closeProgressDialog() after the search result is shown
}
private void showSettingsDialog() {
@@ -344,7 +325,7 @@ public class MainActivity extends AppCompatActivity {
}
private void showAboutDialog() {
// Versionsnummer einsetzen
// insert version number
@SuppressLint("InflateParams")
View dialogView = getLayoutInflater().inflate(R.layout.about_layout, null);
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);
copyrightText.setText(getString(R.string.about_copyright, year));
// Dialog anzeigen
// show dialog
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setView(dialogView);
builder.setTitle(R.string.app_name);
@@ -363,13 +344,48 @@ public class MainActivity extends AppCompatActivity {
dialog.show();
}
private void synchronizeServer() {
SyncNowTask syncNowTask = new SyncNowTask();
syncNowTask.execute();
/**
* Called if user presses the synchronization button.
*/
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) {
Log.d(TAG, "Open URL " + url);
@@ -413,11 +429,11 @@ public class MainActivity extends AppCompatActivity {
boolean webViewVisible = !hostUnconfiguredHintVisible && !repositoryEmptyHintVisible;
if (webViewVisible) {
searchArea.setVisibility(View.VISIBLE);
boolean previousVisible = (webView.getVisibility() == View.INVISIBLE);
boolean previousVisible = (webView.getVisibility() == View.VISIBLE);
webView.setVisibility(View.VISIBLE);
// if WebView is shown the first time, go to start page
if (!previousVisible) {
if (!previousVisible || webView.getUrl() == null) {
loadUrl(SERVER_BASE_URL);
}
} else {
@@ -450,61 +466,4 @@ public class MainActivity extends AppCompatActivity {
Toast toast = Toast.makeText(this, message, Toast.LENGTH_SHORT);
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;
}
}
@@ -18,9 +18,7 @@
package net.moasdawiki.app;
import android.annotation.SuppressLint;
import android.content.SharedPreferences;
import android.os.AsyncTask;
import android.os.Bundle;
import android.text.InputFilter;
import android.text.InputType;
@@ -35,11 +33,11 @@ import androidx.preference.PreferenceFragmentCompat;
import net.moasdawiki.base.Settings;
import net.moasdawiki.service.repository.RepositoryService;
import org.jetbrains.annotations.NotNull;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class SettingsFragment extends PreferenceFragmentCompat implements SharedPreferences.OnSharedPreferenceChangeListener {
@@ -47,15 +45,16 @@ public class SettingsFragment extends PreferenceFragmentCompat implements Shared
private SynchronizeWikiClient synchronizeWikiClient;
private RepositoryService repositoryService;
private Settings settings;
private ExecutorService synchronizationExecutorService;
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//noinspection ConstantConditions
WikiEngineApplication app = (WikiEngineApplication) getContext().getApplicationContext();
synchronizeWikiClient = app.getSynchronizeWikiClient();
repositoryService = app.getServiceLocator().getRepositoryService();
settings = app.getServiceLocator().getSettings();
repositoryService = app.getRepositoryService();
settings = app.getSettings();
synchronizationExecutorService = Executors.newSingleThreadExecutor();
}
@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() {
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")
@SuppressWarnings("NonStaticInnerClassInSecureContext")
private class ConnectServerTask extends AsyncTask<Void, Void, Boolean> {
@Override
@NotNull
protected Boolean doInBackground(Void... params) {
private void runCheckServerConnection() {
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;
return;
}
// Connect with server
return synchronizeWikiClient.createAndCheckSession();
}
@Override
protected void onPostExecute(@NotNull Boolean success) {
boolean success = synchronizeWikiClient.createAndCheckSession();
updateStatusText();
if (!success) {
showToast(getString(R.string.settings_search_failed));
}
}
}
}
@@ -29,20 +29,9 @@ import androidx.preference.PreferenceManager;
import net.moasdawiki.base.Logger;
import net.moasdawiki.base.ServiceException;
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.RepositoryService;
import net.moasdawiki.service.sync.*;
import net.moasdawiki.util.DateUtils;
import net.moasdawiki.util.xml.XmlGenerator;
import net.moasdawiki.util.xml.XmlParser;
@@ -67,8 +56,6 @@ import java.util.Enumeration;
/**
* Sucht einen Wikiserver in Netzwerk und synchronisiert alle Wikidateien
* im eigenen Repository..
*
* @author Herbert Reiter
*/
public class SynchronizeWikiClient {
@@ -85,13 +72,15 @@ public class SynchronizeWikiClient {
@NotNull
private final RepositoryService repositoryService;
@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.logger = serviceLocator.getLogger();
this.settings = serviceLocator.getSettings();
this.repositoryService = serviceLocator.getRepositoryService();
this.logger = logger;
this.settings = settings;
this.repositoryService = repositoryService;
this.random = new SecureRandom();
}
/**
@@ -263,7 +252,7 @@ public class SynchronizeWikiClient {
return 0;
}
for (int i = 0; i < fileCount && !feedback.isCancelled(); i++) {
for (int i = 0; i < fileCount; i++) {
feedback.progress(i, fileCount);
SingleFileXml serverFile = response.fileList.get(i);
try {
@@ -279,7 +268,6 @@ public class SynchronizeWikiClient {
app.resetServices();
// Update last sync time
if (!feedback.isCancelled()) {
Date currentServerDate = DateUtils.parseUtcDate(response.currentServerTime);
if (currentServerDate != null) {
long newLastSyncServerTimeMs = currentServerDate.getTime();
@@ -287,7 +275,6 @@ public class SynchronizeWikiClient {
editor.putLong(Constants.PREFERENCES_SYNC_SERVER_TIME, newLastSyncServerTimeMs);
editor.apply();
}
}
return fileCount;
}
@@ -471,9 +458,8 @@ public class SynchronizeWikiClient {
* Interface, um den Fortschritt während der Synchronisierung mit dem Server
* zurückzuübermitteln.
*/
@FunctionalInterface
public interface ProgressFeedback {
@SuppressWarnings("BooleanMethodIsAlwaysInverted")
boolean isCancelled();
void progress(int step, int total);
}
}
@@ -23,58 +23,90 @@ import android.app.Application;
import net.moasdawiki.base.Logger;
import net.moasdawiki.base.Messages;
import net.moasdawiki.base.Settings;
import net.moasdawiki.plugin.PluginService;
import net.moasdawiki.plugin.ServiceLocator;
import net.moasdawiki.server.RequestDispatcher;
import net.moasdawiki.service.handler.EditorHandler;
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.repository.FilesystemRepositoryService;
import net.moasdawiki.service.repository.RepositoryService;
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.WikiServiceImpl;
import java.io.File;
/**
* Verwaltet den Lebenszyklus der Wiki Engine. Muss außerhalb der Activities erfolgen, weil diese
* z.B. beim Drehen des Bildschirm neu erzeugt werden.
* Main control of the wiki App.
*
* @author Herbert Reiter
* Must be run globally for the App, i.e. outside of an activity.
*/
public class WikiEngineApplication extends Application {
private static final String REPOSITORY_ROOT_PATH_DEFAULT = "repository";
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 TerminTransformer terminTransformer;
private RequestDispatcher requestDispatcher;
@Override
public void onCreate() {
super.onCreate();
logger = new Logger(System.out);
logger.write("MoasdaWiki app starting");
logger.write("MoasdaWiki App starting");
File internalStorageRepositoryRoot = new File(getFilesDir(), "repository");
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);
File internalStorageRepositoryRoot = new File(getFilesDir(), REPOSITORY_ROOT_PATH_DEFAULT);
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
ViewPageHandler viewPageHandler = new ViewPageHandler(logger, settings, wikiService, htmlService);
SearchHandler searchHandler = new SearchHandler(logger, settings, messages, wikiService, searchService, htmlService);
EditorHandler editorHandler = new EditorHandler(logger, settings, messages, repositoryService, wikiService, transformerService, htmlService);
FileDownloadHandler fileDownloadHandler = new FileDownloadHandler(logger, settings, repositoryService, htmlService);
requestDispatcher = new RequestDispatcher(htmlService, viewPageHandler,
searchHandler, editorHandler, fileDownloadHandler, null);
}
public void resetServices() {
serviceLocator.getRepositoryService().rebuildCache();
serviceLocator.getWikiService().reset();
serviceLocator.getSettings().reset();
serviceLocator.getMessages().reset();
serviceLocator.getPluginService().loadPlugins(serviceLocator);
repositoryService.reset();
settings.reset();
messages.reset();
wikiService.reset();
searchService.reset();
terminTransformer.reset();
}
@Override
@@ -83,11 +115,23 @@ public class WikiEngineApplication extends Application {
super.onTerminate();
}
public ServiceLocator getServiceLocator() {
return serviceLocator;
public RepositoryService getRepositoryService() {
return repositoryService;
}
public Settings getSettings() {
return settings;
}
public TerminTransformer getTerminTransformer() {
return terminTransformer;
}
public SynchronizeWikiClient getSynchronizeWikiClient() {
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()
}
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
// 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,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.
- Flexible Volltextsuche, unterstützt reguläre Ausdrücke.
- 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.
- 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.
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,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.
- Powerful full text search, supports regular expressions.
- 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.
- 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.
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
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
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