Compare commits

..

57 Commits

Author SHA1 Message Date
Herbert Reiter 81c00ef2da New release 2021-12-25 16:01:44 +01:00
Herbert Reiter 54456f9629 New release 2021-12-25 15:55:55 +01:00
Herbert Reiter da2bab9db5 Small layout optimizations 2021-12-25 15:55:19 +01:00
Herbert Reiter 37e92222d5 Update to moasdawiki-server 3.1.0 2021-12-25 15:54:35 +01:00
Herbert Reiter e2cda6afb1 Upgrade library dependencies 2021-12-25 15:53:55 +01:00
Herbert Reiter bb52a69ff7 Support for Android 12 2021-12-25 15:53:27 +01:00
Herbert Reiter ce33ef40d6 Update copyright year 2021-12-25 15:51:36 +01:00
Herbert Reiter e27651e9dd New release 2021-06-09 22:15:10 +02:00
Herbert Reiter cd01a5390d New release 2021-06-09 22:11:55 +02:00
Herbert Reiter 1597613f72 Upgrade library dependencies 2021-06-09 22:11:17 +02:00
Herbert Reiter ec78f31ce5 Update to moasdawiki-server 2.6.1 2021-06-09 22:10:53 +02:00
Herbert Reiter e8833dd11b Remove double back key press to close the app 2021-06-09 22:09:22 +02:00
Herbert Reiter 07a9fa630c New release 2021-05-13 10:34:12 +02:00
Herbert Reiter 96623d73a0 Make GPL-3.0-only licensing more clear 2021-05-13 10:31:18 +02:00
Herbert Reiter 58ba5f48c5 New release 2021-05-08 22:44:55 +02:00
Herbert Reiter 51a34e4463 Update documentation 2021-05-08 22:44:27 +02:00
Herbert Reiter c3b1c53622 Update library dependencies 2021-05-08 22:37:11 +02:00
Herbert Reiter 5f1ed57f42 Show screenshots in README 2021-04-05 10:38:56 +00:00
Herbert Reiter c77845abf5 Update documentation for F-Droid 2021-03-06 18:26:51 +01:00
Herbert Reiter cb128311c7 Update README.md 2021-03-06 18:26:27 +01:00
Herbert Reiter 7b3a8780f6 Update README.md 2021-03-06 16:00:54 +01:00
Herbert Reiter a7a05cc5fa Update README.md 2021-03-06 15:59:42 +01:00
Herbert Reiter 80af757d4d New release 2021-02-28 21:32:21 +01:00
Herbert Reiter d243402082 New release 2021-01-31 16:21:45 +01:00
Herbert Reiter 297df3da1a New release 2021-01-31 16:15:43 +01:00
Herbert Reiter a0a77f6311 New release 2021-01-31 16:15:19 +01:00
Herbert Reiter db66908534 New release 2021-01-31 15:42:58 +01:00
Herbert Reiter b7f57c4748 New release 2021-01-23 15:52:32 +01:00
Herbert Reiter 88ce02dfda New release 2021-01-23 15:50:09 +01:00
Herbert Reiter 1239f83cd6 Update Gradle version 2021-01-23 15:37:53 +01:00
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
60 changed files with 530 additions and 506 deletions
+3
View File
@@ -10,3 +10,6 @@ release/
# Git # Git
local.properties local.properties
# Project
TODO.md
+60
View File
@@ -1,5 +1,65 @@
# Changelog # Changelog
## 3.1.0.0 (versionCode 28, 2021-12-25)
- Small layout optimizations
- Update to moasdawiki-server 3.1.0
- Support for Android 12
- Upgrade library dependencies
- Update copyright year
## 2.6.1.0 (versionCode 27, 2021-06-09)
- Remove double back key press to close the app
- Update to moasdawiki-server 2.6.1
- Upgrade library dependencies
## 2.6.0.0 (versionCode 26, 2021-05-13)
- Make GPL-3.0-only licensing more clear
- Update to moasdawiki-server 2.6.0
## 2.5.2.1 (versionCode 25, 2021-05-08)
- Update documentation
- Update library dependencies
## 2.5.2.0 (versionCode 24, 2021-02-28)
- Find also words that start with the search string
- Update to moasdawiki-server 2.5.2
## 2.5.1.0 (versionCode 23, 2021-01-31)
- Enhance search index to full words
- Introduce search ignore list
- Add support for SVG images
- Update to moasdawiki-server 2.5.1
## 2.4.4.0 (versionCode 22, 2021-01-23)
- Speed up search
- Update to moasdawiki-server 2.4.4
## 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) ## 2.3.1.0 (versionCode 18, 2020-09-13)
- Speed up full text search by a search index - Speed up full text search by a search index
-53
View File
@@ -619,56 +619,3 @@ Program, unless a warranty or assumption of liability accompanies a
copy of the Program in return for a fee. copy of the Program in return for a fee.
END OF TERMS AND CONDITIONS END OF TERMS AND CONDITIONS
How to Apply These Terms to Your New Programs
If you develop a new program, and you want it to be of the greatest
possible use to the public, the best way to achieve this is to make it
free software which everyone can redistribute and change under these terms.
To do so, attach the following notices to the program. It is safest
to attach them to the start of each source file to most effectively
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.
MoasdaWiki Server
Copyright (C) 2008 - 2020 Herbert Reiter
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
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
Also add information on how to contact you by electronic and paper mail.
If the program does terminal interaction, make it output a short
notice like this when it starts in an interactive mode:
MoasdaWiki Server Copyright (C) 2008 - 2020 Herbert Reiter
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
This is free software, and you are welcome to redistribute it
under certain conditions; type `show c' for details.
The hypothetical commands `show w' and `show c' should show the appropriate
parts of the General Public License. Of course, your program's commands
might be different; for a GUI interface, you would use an "about box".
You should also get your employer (if you work as a programmer) or school,
if any, to sign a "copyright disclaimer" for the program, if necessary.
For more information on this, and how to apply and follow the GNU GPL, see
<http://www.gnu.org/licenses/>.
The GNU General Public License does not permit incorporating your program
into proprietary programs. If your program is a subroutine library, you
may consider it more useful to permit linking proprietary applications with
the library. If this is what you want to do, use the GNU Lesser General
Public License instead of this License. But first, please read
<http://www.gnu.org/philosophy/why-not-lgpl.html>.
+23 -13
View File
@@ -2,37 +2,46 @@
## Description ## Description
MoasdaWiki App is an extension of the MoasdaWiki Server to have a copy of MoasdaWiki App is a privacy-friendly frontend for the MoasdaWiki Server
your Wiki content on your mobile device. knowledge management tool. It mirrors the Wiki content on your mobile device.
For MoasdaWiki documentation see https://moasdawiki.net/. For MoasdaWiki documentation see https://moasdawiki.net/.
## Download
[<img src="https://fdroid.gitlab.io/artwork/badge/get-it-on.png" [<img src="https://fdroid.gitlab.io/artwork/badge/get-it-on.png"
alt="Get it on F-Droid" alt="Get it on F-Droid"
height="80">](https://f-droid.org/packages/net.moasdawiki.app) height="80">](https://f-droid.org/packages/net.moasdawiki.app)
## Features ## Features
- Synchronizes the data from a MoasdaWiki Server instance. - Synchronizes the content from your MoasdaWiki Server instance.
- Data privacy: No cloud connection established, directly connects to the server inside your private network. - Fast full text search.
- Powerful full text search, supports regular expressions. - Calendar integration: Shows birthdays and events in the mobile calendar.
- Content cannot be modified within the app as it is no fun to type Wiki syntax on the mobile device, - Data privacy by design: Directly connects to your server instance in your
changes have to be done via the MoasdaWiki Server. private network. No trackers, never establishes a cloud connection.
- Calendar integration, shows birthdays and events in the mobile calendar (German version only). - Wiki content cannot be modified within the app as it is no fun to type Wiki
syntax on a mobile device, changes have to be done via the MoasdaWiki Server.
## Screenshots
<img src="fastlane/metadata/android/en-US/images/phoneScreenshots/1.png" width="250" height="500" />
<img src="fastlane/metadata/android/en-US/images/phoneScreenshots/2.png" width="250" height="500" />
<img src="fastlane/metadata/android/en-US/images/phoneScreenshots/4.png" width="250" height="500" />
## Synchronize content with a MoasdaWiki Server ## Synchronize content with a MoasdaWiki Server
1. Download MoasdaWiki Server from https://gitlab.com/moasdawiki/moasdawiki-server. 1. Download MoasdaWiki Server from https://gitlab.com/moasdawiki/moasdawiki-server.
1. Set up a MoasdaWiki Server instance in your LAN. 1. Set up a MoasdaWiki Server instance in your LAN.
1. Install and open the MoasdaWiki App. 1. Install the MoasdaWiki App.
1. You can see a hint that the App has to be configured first. Press on that hint. 1. In the app you can see a hint that it 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 or IP address of the server instance, e.g. `192.168.1.101`. Press OK.
1. In the status section below you should see "Needs authorization at server". Otherwise check host name and port again. 1. In the status section below you should see "Needs authorization at server". Otherwise check host name and port again.
1. On server side open the Wiki page in a browser, click on "Help" and "Synchronization". 1. On server side open the Wiki page in a browser, click on "Help" and "Synchronization".
1. You can see a list of devices and synchronization sessions. Check the device name and click on "Grant". 1. You can see a list of devices and synchronization sessions. Check the device name and click on "Grant".
1. Back in the app press the back button (&larr;) on the upper left corner to get back to the main dialog. 1. 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. Now you can see a hint that the app has to be synchronized. Press on that hint.
1. Now you should have all the server content also in the app and you can see the "Home" wiki page. 1. Now you should have all the server content also in the app and you can see the "Home-App" wiki page.
## Support ## Support
@@ -40,6 +49,7 @@ If you have questions or any problems you can contact me via [support@moasdawiki
## License ## License
MoasdaWiki server is licensed under the GPL 3 license - see the LICENSE file for details. MoasdaWiki server is licensed under GPL-3.0-only &ndash; see the
[LICENSE](LICENSE) file for details.
Copyright (C) Herbert Reiter Copyright (C) Herbert Reiter
+12 -12
View File
@@ -1,33 +1,33 @@
apply plugin: 'com.android.application' apply plugin: 'com.android.application'
android { android {
compileSdkVersion 29 // 29 = Android 10 compileSdkVersion 31 // 31 = Android 12
defaultConfig { defaultConfig {
applicationId "net.moasdawiki.app" applicationId "net.moasdawiki.app"
minSdkVersion 26 // 26 = Oreo 8.0 minSdkVersion 28 // 28 = Android 9
targetSdkVersion 29 // should be same as compileSdkVersion targetSdkVersion 31 // should be same as compileSdkVersion
versionCode 18 versionCode 28
versionName "2.3.1.0" versionName "3.1.0.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*"
} }
} }
} }
compileOptions { compileOptions {
sourceCompatibility = 1.8 sourceCompatibility JavaVersion.VERSION_11
targetCompatibility = 1.8 targetCompatibility JavaVersion.VERSION_11
} }
} }
dependencies { dependencies {
implementation 'net.moasdawiki:moasdawiki-server:2.3.1' implementation 'net.moasdawiki:moasdawiki-server:3.1.0'
implementation fileTree(include: ['*.jar'], dir: 'libs') implementation fileTree(include: ['*.jar'], dir: 'libs')
implementation 'androidx.appcompat:appcompat:1.2.0' implementation 'androidx.appcompat:appcompat:1.4.0'
implementation 'androidx.preference:preference:1.1.1' implementation 'androidx.preference:preference:1.1.1'
compileOnly 'org.jetbrains:annotations:19.0.0' compileOnly 'org.jetbrains:annotations:22.0.0'
testImplementation 'org.testng:testng:6.14.3' testImplementation 'org.testng:testng:7.4.0'
} }
+4 -2
View File
@@ -20,7 +20,8 @@
<activity <activity
android:name=".MainActivity" android:name=".MainActivity"
android:label="@string/app_name"> android:label="@string/app_name"
android:exported="true">
<intent-filter> <intent-filter>
<action android:name="android.intent.action.MAIN" /> <action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" /> <category android:name="android.intent.category.LAUNCHER" />
@@ -44,7 +45,8 @@
<service <service
android:name="net.moasdawiki.app.CalendarAccountAuthenticatorService" android:name="net.moasdawiki.app.CalendarAccountAuthenticatorService"
tools:ignore="ExportedService"> tools:ignore="ExportedService"
android:exported="true">
<intent-filter> <intent-filter>
<action android:name="android.accounts.AccountAuthenticator"/> <action android:name="android.accounts.AccountAuthenticator"/>
</intent-filter> </intent-filter>
@@ -1,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -30,6 +29,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,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -1,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -1,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -1,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -43,9 +42,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 +84,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 +103,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 +194,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 +285,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 +295,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,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -1,27 +1,24 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; 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,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -25,7 +24,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 +45,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;
@@ -67,48 +64,44 @@ import org.jetbrains.annotations.Nullable;
import java.io.ByteArrayInputStream; import java.io.ByteArrayInputStream;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.net.InetAddress;
import java.net.URLDecoder; import java.net.URLDecoder;
import java.util.Arrays; import java.util.Arrays;
import java.util.Calendar; 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 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 +118,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 +148,40 @@ 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(Collections.emptyMap(),
Plugin plugin = pluginService.getPluginByUrl(urlPath); "GET", urlPath, urlPath, convertParameters(uri), new byte[0]);
if (plugin != null) { HttpResponse response = requestDispatcher.handleRequest(httpRequest);
HttpRequest httpRequest = new HttpRequest();
httpRequest.clientIP = InetAddress.getLocalHost();
httpRequest.httpHeader = Collections.emptyMap();
httpRequest.method = "GET";
httpRequest.url = urlPath;
httpRequest.urlPath = urlPath;
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 +191,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);
@@ -256,30 +229,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);
@@ -287,35 +259,26 @@ 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) {
webview.goBack(); webview.goBack();
} else { } else {
long currentTimeMillis = System.currentTimeMillis(); // Close app
if (backButtonPressedTimestamp + 5000 < currentTimeMillis) {
// First click on back button or previous click was more than 5 seconds ago
Log.d(TAG, "Back button 1x, show close hint");
showToast(getString(R.string.action_back_close_hint));
backButtonPressedTimestamp = currentTimeMillis;
} else {
// Second click on back button within 5 seconds -> close app
Log.d(TAG, "Back button 2x, closing app");
finish(); finish();
} }
} }
}
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();
@@ -335,8 +298,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() {
@@ -345,7 +307,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);
@@ -354,7 +316,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);
@@ -364,13 +326,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);
@@ -418,7 +415,7 @@ public class MainActivity extends AppCompatActivity {
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 {
@@ -451,61 +448,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,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -1,26 +1,23 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
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 +32,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 +44,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 +165,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")
private class ConnectServerTask extends AsyncTask<Void, Void, Boolean> {
@Override
@NotNull
protected Boolean doInBackground(Void... params) {
SharedPreferences preferences = getPreferenceManager().getSharedPreferences(); SharedPreferences preferences = getPreferenceManager().getSharedPreferences();
String host = preferences.getString(Constants.PREFERENCES_SYNC_SERVER_HOST, null); String host = preferences.getString(Constants.PREFERENCES_SYNC_SERVER_HOST, null);
if (host == null) { if (host == null) {
Log.d(TAG, "Cancel ConnectServerTask because server host is not configured"); Log.d(TAG, "Cancel ConnectServerTask because server host is not configured");
return false; return;
} }
// Connect with server boolean success = synchronizeWikiClient.createAndCheckSession();
return synchronizeWikiClient.createAndCheckSession();
}
@Override
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,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -29,20 +28,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 +55,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 +71,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 +251,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,7 +267,6 @@ 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();
@@ -287,7 +274,6 @@ public class SynchronizeWikiClient {
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 +457,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,19 +1,18 @@
/* /*
* MoasdaWiki App * MoasdaWiki App
* Copyright (C) 2008 - 2020 Herbert Reiter (herbert@moasdawiki.net) * Copyright (C) 2008 - 2022 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
* it under the terms of the GNU General Public License as published by * under the terms of the GNU General Public License version 3 as published
* the Free Software Foundation, either version 3 of the License, or * by the Free Software Foundation (GPL-3.0-only).
* (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful, but
* but WITHOUT ANY WARRANTY; without even the implied warranty of * WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License along with
* along with this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.html>.
*/ */
package net.moasdawiki.app; package net.moasdawiki.app;
@@ -23,58 +22,93 @@ 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.SearchIgnoreList;
import net.moasdawiki.service.search.SearchIndex;
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, repositoryService, 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, null, false);
settings = new AndroidSettings(logger, repositoryService, Settings.getConfigFileApp());
messages = new Messages(logger, settings, repositoryService);
wikiService = new WikiService(logger, repositoryService, false);
SearchIgnoreList searchIgnoreList = new SearchIgnoreList(logger, repositoryService);
SearchIndex searchIndex = new SearchIndex(logger, repositoryService, wikiService, searchIgnoreList, true);
searchService = new SearchService(logger, wikiService, searchIgnoreList, searchIndex, 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 +117,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

+4 -2
View File
@@ -16,13 +16,14 @@
<LinearLayout <LinearLayout
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:padding="20dp"
android:orientation="horizontal"> android:orientation="horizontal">
<View <View
android:layout_width="48dp" android:layout_width="48dp"
android:layout_height="48dp" android:layout_height="48dp"
android:layout_gravity="center_vertical" android:layout_gravity="center_vertical"
android:layout_margin="20dp" android:layout_marginEnd="20dp"
android:background="@drawable/ic_info_outline_black" /> android:background="@drawable/ic_info_outline_black" />
<TextView <TextView
@@ -50,13 +51,14 @@
<LinearLayout <LinearLayout
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:padding="20dp"
android:orientation="horizontal"> android:orientation="horizontal">
<View <View
android:layout_width="48dp" android:layout_width="48dp"
android:layout_height="48dp" android:layout_height="48dp"
android:layout_gravity="center_vertical" android:layout_gravity="center_vertical"
android:layout_margin="20dp" android:layout_marginEnd="20dp"
android:background="@drawable/ic_info_outline_black" /> android:background="@drawable/ic_info_outline_black" />
<TextView <TextView
-1
View File
@@ -4,7 +4,6 @@
<string name="about_version">Version %1$s</string> <string name="about_version">Version %1$s</string>
<string name="action_help">Hilfe</string> <string name="action_help">Hilfe</string>
<string name="action_about">Über</string> <string name="action_about">Über</string>
<string name="action_back_close_hint">Erneut drücken um die App zu beenden.</string>
<string name="action_settings">Einstellungen</string> <string name="action_settings">Einstellungen</string>
<string name="action_startpage">Startseite</string> <string name="action_startpage">Startseite</string>
<string name="action_synchronize">Synchronisieren</string> <string name="action_synchronize">Synchronisieren</string>
-1
View File
@@ -4,7 +4,6 @@
<string name="about_homepage_url" translatable="false">https://www.moasdawiki.net/</string> <string name="about_homepage_url" translatable="false">https://www.moasdawiki.net/</string>
<string name="about_version">Version %1$s</string> <string name="about_version">Version %1$s</string>
<string name="action_about">About</string> <string name="action_about">About</string>
<string name="action_back_close_hint">Press again to exit the app.</string>
<string name="action_help">Help</string> <string name="action_help">Help</string>
<string name="action_settings">Settings</string> <string name="action_settings">Settings</string>
<string name="action_startpage">Start page</string> <string name="action_startpage">Start page</string>
+3 -3
View File
@@ -2,11 +2,11 @@
buildscript { buildscript {
repositories { repositories {
jcenter() mavenCentral()
google() google()
} }
dependencies { dependencies {
classpath 'com.android.tools.build:gradle:4.0.1' classpath 'com.android.tools.build:gradle:7.0.4'
// 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
@@ -15,7 +15,7 @@ buildscript {
allprojects { allprojects {
repositories { repositories {
jcenter() mavenCentral()
google() google()
} }
} }
@@ -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,2 @@
- Suche beschleunigen
- Auf moasdawiki-server 2.4.4 aktualisieren
@@ -0,0 +1,4 @@
- Suchindex auf ganze Wörter erweitern
- Ignorierliste für Suchwörter einführen
- Unterstützung für SVG-Grafiken hinzufügen
- Auf moasdawiki-server 2.5.1 aktualisieren
@@ -0,0 +1,2 @@
- Auch Wörter, die mit dem Suchtext beginnen, finden
- Auf moasdawiki-server 2.5.2 aktualisieren
@@ -0,0 +1,2 @@
- Documentation aktualisieren
- Bibliotheks-Abhängigkeiten aktualisieren
@@ -0,0 +1 @@
- Lizenz GPL-3.0-only klarer herausstellen
@@ -0,0 +1,3 @@
- Entferne doppeltes Drücken der Rückwärtstaste zum Schließen der App
- Auf moasdawiki-server 2.6.1 aktualisieren
- Bibliotheks-Abhängigkeiten aktualisieren
@@ -0,0 +1,4 @@
- Kleine Layoutverbesserungen
- Auf moasdawiki-server 3.1.0 aktualisieren
- Unterstützung für Android 12
- Bibliotheks-Abhängigkeiten aktualisieren
@@ -1,22 +1,23 @@
MoasdaWiki App ist eine Erweiterung zum MoasdaWiki-Server, um eine Kopie deiner Wikiinhalte auf deinem Mobilgerät zu haben. MoasdaWiki App ist eine datenschutzfreundliche Oberfläche der MoasdaWiki-Server-Wissensverwaltung. Sie ist eine Kopie deiner Wikiinhalte auf deinem Mobilgerät.
Funkionalitäten: <b>Funktionen:</b>
- Synchronisiert die Inhalte von einer MoasdaWiki-Server-Instanz. - Synchronisiert die Inhalte von deiner MoasdaWiki-Server-Instanz.
- Datenschutz: Es wird keine Cloud-Verbindung hergestellt, verbindet sich direkt mit dem Server in deinem privaten Netzwerk. - Schnelle Volltextsuche
- Flexible Volltextsuche, unterstützt reguläre Ausdrücke. - Kalender-Integration: Zeigt Geburtstage und Termine im Kalender des Mobilgeräts.
- 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. - Datenschutz: Verbindet sich direkt mit dem Server in deinem privaten Netzwerk. Keine Tracker. Keine Cloud-Verbindung.
- Kalender-Integration, zeigt Geburtstage und Termine im Kalender des Mobilgeräts (nur in der deutschen Version).
Synchronisieren des Inhalts mit einem MoasdaWiki-Server: <b>Synchronisieren des Inhalts mit einem MoasdaWiki-Server:</b>
1. Lade MoasdaWiki-Server von https://moasdawiki.net/ herunter. 1. Lade MoasdaWiki-Server von https://moasdawiki.net/ herunter.
2. Setze eine MoasdaWiki-Server-Instance in deinem LAN auf. 2. Setze eine MoasdaWiki-Server-Instance in deinem LAN auf.
3. Installiere und öffne die MoasdaWiki-App. 3. Installiere die MoasdaWiki-App.
4. Du kannst einen Hinweis sehen, dass die App zunächst konfiguriert werden muss. Drücke auf den Hinweis. 4. In der App kannst du einen Hinweis sehen, dass sie 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. 5. Drücke auf "Hostname" und gib den Hostnamen oder IP-Adresse 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. 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". 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". 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. 9. Zurück in der App drücke den Zurück-Button 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. 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.
@@ -1 +1 @@
App zum Synchronizieren der Inhalte vom MoasdaWiki-Server. Datenschutzfreundliche Wissensverwaltung
@@ -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,2 @@
- Speed up search
- Update to moasdawiki-server 2.4.4
@@ -0,0 +1,4 @@
- Enhance search index to full words
- Introduce search ignore list
- Add support for SVG images
- Update to moasdawiki-server 2.5.1
@@ -0,0 +1,2 @@
- Find also words that start with the search string
- Update to moasdawiki-server 2.5.2
@@ -0,0 +1,2 @@
- Update documentation
- Update library dependencies
@@ -0,0 +1 @@
- Make GPL-3.0-only licensing more clear
@@ -0,0 +1,3 @@
- Remove double back key press to close the app
- Update to moasdawiki-server 2.6.1
- Upgrade library dependencies
@@ -0,0 +1,4 @@
- Small layout optimizations
- Update to moasdawiki-server 3.1.0
- Support for Android 12
- Upgrade library dependencies
@@ -1,22 +1,24 @@
MoasdaWiki App is an extension of the MoasdaWiki Server to have a copy of your Wiki content on your mobile device. MoasdaWiki App is a privacy-friendly frontend for the MoasdaWiki Server
knowledge management tool. It mirrors the Wiki content on your mobile device.
Features: <b>Features:</b>
- Synchronizes the data from a MoasdaWiki Server instance. - Synchronizes the content from your MoasdaWiki Server instance.
- Data privacy: No cloud connection established, directly connects to the server inside your private network. - Fast full text search.
- Powerful full text search, supports regular expressions. - Calendar integration: Shows birthdays and events in the mobile calendar.
- 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. - Data privacy by design: Directly connects to your server instance in your private network. No trackers, never establishes a cloud connection.
- Calendar integration, shows birthdays and events in the mobile calendar (German version only).
Synchronize content with a MoasdaWiki Server: <b>Synchronize content with a MoasdaWiki Server:</b>
1. Download MoasdaWiki Server from https://moasdawiki.net/. 1. Download MoasdaWiki Server from https://moasdawiki.net/.
2. Set up a MoasdaWiki Server instance in your LAN. 2. Set up a MoasdaWiki Server instance in your LAN.
3. Install and open the MoasdaWiki App. 3. Install the MoasdaWiki App.
4. You can see a hint that the App has to be configured first. Press on that hint. 4. In the app you can see a hint that it 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. 5. Press on "Host name" and enter the host name or IP address 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. 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". 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". 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. 9. Back in the app press the back button 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. 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.
@@ -1 +1 @@
App to synchronize data from MoasdaWiki Server. Privacy-Friendly Knowledge Management App
+2 -2
View File
@@ -1,6 +1,6 @@
#Tue Sep 08 22:17:23 CEST 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-6.1.1-all.zip distributionUrl=https\://services.gradle.org/distributions/gradle-7.0.2-all.zip