Compare commits
	
		
			105 Commits
		
	
	
		
			v123020523
			...
			1e04152687
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1e04152687 | |||
| 1bdfb143ac | |||
| d81ced3964 | |||
| fbafece1fa | |||
| cbed8f07cb | |||
| f54fcc3ba1 | |||
|  | aad93ef722 | ||
| 9e83af0302 | |||
|  | 24b86e66b4 | ||
| 641c444061 | |||
| 0902c61544 | |||
|  | 6790152a0b | ||
|  | 46d1ba418e | ||
| 436373d0ad | |||
|  | 5b9b51c02d | ||
| b81abe384a | |||
|  | 851f862dbe | ||
|  | 8d7e302af8 | ||
| 236e1cca90 | |||
| 3a33cb4510 | |||
| 0bf9ca9a49 | |||
|  | 61e0087894 | ||
|  | 1ec05d9913 | ||
|  | 859bd91bbb | ||
|  | 204b736c53 | ||
|  | f24609c143 | ||
|  | b94d7dc537 | ||
|  | 41910cc4cd | ||
|  | db166ca9d4 | ||
|  | db0d5a4a85 | ||
|  | 3bc0d7cf95 | ||
|  | 8f464d95fd | ||
|  | 5ccd6a3368 | ||
|  | cdbded246e | ||
|  | 750c7758bd | ||
|  | 22f8b14ecd | ||
|  | 6e27d6d4e6 | ||
|  | 14ff4dbd05 | ||
|  | 390c2d0cf3 | ||
|  | e58914ef58 | ||
|  | a03f08fca1 | ||
|  | 8e9b87f00c | ||
|  | f765224a86 | ||
|  | 14d2219eb8 | ||
|  | 137580ccf9 | ||
|  | f101d22f54 | ||
|  | 68aedb7641 | ||
|  | 754d526b49 | ||
|  | c458871569 | ||
| 056825aa0c | |||
| 16b19fc5ce | |||
| 4ad4a23ed8 | |||
| d8c215eacc | |||
| 2b446ab22b | |||
| a029d8a7dc | |||
| 4482234e1a | |||
| b5de30f561 | |||
| 70ad5f322c | |||
| d167092c83 | |||
| c4f4bafe85 | |||
| ed06b22a77 | |||
|  | 172362b533 | ||
|  | ad72cb6f56 | ||
|  | 9057ee0052 | ||
|  | 50d0b44315 | ||
|  | 21b08ed384 | ||
|  | 993c4d2ee9 | ||
|  | 57a9d51027 | ||
|  | 673f0edb8b | ||
|  | 7f96798f13 | ||
|  | 6e5704a45b | ||
|  | 495591159f | ||
|  | 718fe7c5ee | ||
|  | ecd23213f9 | ||
|  | e6baed8cb4 | ||
|  | c87abec0b9 | ||
|  | 0aba41d8bf | ||
|  | 2a2d1047b4 | ||
|  | 66ef1ccf32 | ||
|  | 677ede5bc7 | ||
|  | 996a7ed22c | ||
|  | 85208c4e5a | ||
|  | 5cfec50cba | ||
| 76ad71e1dc | |||
| 0277fb507c | |||
| 8d7d3174aa | |||
|  | 00eb3333fe | ||
|  | 629ca01d99 | ||
|  | c2d8681ce8 | ||
|  | 08f79cb148 | ||
| e21906e70d | |||
|  | 9d2cc32bc9 | ||
|  | d9d057c8dc | ||
|  | 1f3fa0c4a6 | ||
|  | dea3def385 | ||
|  | f72ef2f5d4 | ||
|  | f28cb759df | ||
|  | b9d69c3e64 | ||
|  | c2a1c9eaac | ||
|  | bf37209a15 | ||
|  | 2c558fe6fd | ||
|  | ad88011454 | ||
|  | 559c17bc1d | ||
|  | ab9c46f0eb | ||
|  | aa799d2ca8 | 
							
								
								
									
										142
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										142
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -1,142 +0,0 @@ | |||||||
| kind: pipeline |  | ||||||
| type: docker |  | ||||||
| name: test |  | ||||||
|  |  | ||||||
| steps: |  | ||||||
|   - name: BuildAndTest |  | ||||||
|     image: mingc/android-build-box:latest |  | ||||||
|     commands: |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Configure gradle..." |  | ||||||
|       - mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true\npushCache=false\nsystemProp.org.gradle.internal.http.connectionTimeout=180000\nsystemProp.org.gradle.internal.http.socketTimeout=180000" >> ~/.gradle/gradle.properties |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Building..." |  | ||||||
|       - ./gradlew build -x test |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Testing..." |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - ./gradlew koverMergedXmlReport |  | ||||||
|     environment: |  | ||||||
|       TZ: Europe/Paris |  | ||||||
|       SONAR_HOST_URL: |  | ||||||
|         from_secret: sonarScannerHostUrl |  | ||||||
|       SONAR_LOGIN: |  | ||||||
|         from_secret: sonarScannerLogin |  | ||||||
|   - name: Analyse |  | ||||||
|     image: kytay/sonar-node-plugin |  | ||||||
|     settings: |  | ||||||
|       sonar_host: |  | ||||||
|         from_secret: sonarScannerHostUrl |  | ||||||
|       sonar_token: |  | ||||||
|         from_secret: sonarScannerLogin |  | ||||||
|       use_node_version: 16.18.1 |  | ||||||
|       sonar_debug: true |  | ||||||
|       sonar_project_settings: ./sonar-project.properties |  | ||||||
| trigger: |  | ||||||
|   event: |  | ||||||
|     - push |  | ||||||
|     - pull_request |  | ||||||
|  |  | ||||||
| --- |  | ||||||
| kind: pipeline |  | ||||||
| type: docker |  | ||||||
| name: Publish |  | ||||||
|  |  | ||||||
| steps: |  | ||||||
|   - name: createTagAndChangelog |  | ||||||
|     image: ubuntu:latest |  | ||||||
|     commands: |  | ||||||
|       - apt-get update && apt-get install -y git |  | ||||||
|       - git fetch --tags -p |  | ||||||
|       - PREV=$(git describe --tags --abbrev=0) |  | ||||||
|       - ./build.sh --publish --from-ci |  | ||||||
|       - git remote add pushing https://$GITEA_USR:$GITEA_PASS@gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform.git |  | ||||||
|       - VER=$(git describe --tags --abbrev=0) |  | ||||||
|       - CHANGELOG=$(git log $PREV..HEAD --pretty="- %s") |  | ||||||
|       - echo "**$VER**\n\n$CHANGELOG\n\n--------------------------------------------------------------------\n\n$(cat CHANGELOG.md)" > CHANGELOG.md |  | ||||||
|       - git add CHANGELOG.md |  | ||||||
|       - git commit -m "Changelog for $VER [CI SKIP]" |  | ||||||
|       - git push pushing master |  | ||||||
|       - git push pushing --tags |  | ||||||
|     environment: |  | ||||||
|       TZ: Europe/Paris |  | ||||||
|       GITEA_USR: |  | ||||||
|         from_secret: giteaUsr |  | ||||||
|       GITEA_PASS: |  | ||||||
|         from_secret: giteaPass |  | ||||||
|  |  | ||||||
|   - name: scpFiles |  | ||||||
|     image: appleboy/drone-scp |  | ||||||
|     settings: |  | ||||||
|       host: amine-louveau.fr |  | ||||||
|       username: ubuntu |  | ||||||
|       key: |  | ||||||
|         from_secret: privateKey |  | ||||||
|       port: 22 |  | ||||||
|       target: /home/ubuntu/ |  | ||||||
|       source: version.txt |  | ||||||
|  |  | ||||||
|   - name: deploy |  | ||||||
|     image: appleboy/drone-ssh |  | ||||||
|     settings: |  | ||||||
|       host: amine-louveau.fr |  | ||||||
|       user: ubuntu |  | ||||||
|       key: |  | ||||||
|         from_secret: privateKey |  | ||||||
|       command_timeout: 2m |  | ||||||
|       script: |  | ||||||
|         - cd /home/ubuntu && sudo rm -rf /var/www/amine/version.txt && sudo chown www-data:www-data ./version.txt && sudo mv version.txt /var/www/amine/ |  | ||||||
|  |  | ||||||
| trigger: |  | ||||||
|   event: |  | ||||||
|     - promote |  | ||||||
|   target: |  | ||||||
|     - production |  | ||||||
|  |  | ||||||
| --- |  | ||||||
| kind: pipeline |  | ||||||
| type: docker |  | ||||||
| name: Release |  | ||||||
|  |  | ||||||
| steps: |  | ||||||
|   - name: build |  | ||||||
|     image: mingc/android-build-box:latest |  | ||||||
|     commands: |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Fetch tags..." |  | ||||||
|       - git fetch --tags |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Configure gradle..." |  | ||||||
|       - mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=false\npushCache=false\nsystemProp.org.gradle.internal.http.connectionTimeout=180000\nsystemProp.org.gradle.internal.http.socketTimeout=180000" >> ~/.gradle/gradle.properties |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Generate APK" |  | ||||||
|       - ./gradlew :androidApp:assembleGithubConfigRelease  -P pushCache=false |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Get Key" |  | ||||||
|       - wget https://amine-louveau.fr/key |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Zipalign" |  | ||||||
|       - $ANDROID_HOME/build-tools/31.0.0/zipalign -f -v 4 androidApp/build/outputs/apk/githubConfig/release/androidApp-githubConfig-release-unsigned.apk androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Sign" |  | ||||||
|       - $ANDROID_HOME/build-tools/31.0.0/apksigner sign -v --out signed.apk --ks ./key --ks-key-alias $YOUR_KEY_ALIAS --ks-pass pass:$YOUR_KEYSTORE_PASSWORD --v1-signing-enabled true --v2-signing-enabled true androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk |  | ||||||
|       - echo "---------------------------------------------------------" |  | ||||||
|       - echo "Verify" |  | ||||||
|       - $ANDROID_HOME/build-tools/31.0.0/apksigner verify signed.apk |  | ||||||
|     environment: |  | ||||||
|       TZ: Europe/Paris |  | ||||||
|       YOUR_KEYSTORE_PASSWORD: |  | ||||||
|         from_secret: keyPass |  | ||||||
|       YOUR_KEY_ALIAS: |  | ||||||
|         from_secret: keyAlias |  | ||||||
|  |  | ||||||
|   - name: gitea_release |  | ||||||
|     image: plugins/gitea-release |  | ||||||
|     settings: |  | ||||||
|       api_key: |  | ||||||
|         from_secret: giteaAPI |  | ||||||
|       base_url: https://gitea.amine-louveau.fr |  | ||||||
|       files: signed.apk |  | ||||||
| trigger: |  | ||||||
|   event: |  | ||||||
|     - tag |  | ||||||
							
								
								
									
										10
									
								
								.gitea/workflows/assets/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.gitea/workflows/assets/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | version: '3' | ||||||
|  | services: | ||||||
|  |   selfoss: | ||||||
|  |     container_name: selfoss | ||||||
|  |     image: rsprta/selfoss | ||||||
|  |     network_mode: "host" | ||||||
|  |     ports: | ||||||
|  |       - "8888:8888" | ||||||
|  |  | ||||||
|  |  | ||||||
							
								
								
									
										24
									
								
								.gitea/workflows/common_build.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								.gitea/workflows/common_build.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | name: Build | ||||||
|  | on: | ||||||
|  |   workflow_call: | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   BuildAndTest: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Check out repository code | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           fetch-depth: 0 | ||||||
|  |       - name: Fetch tags | ||||||
|  |         run: git fetch --tags -p | ||||||
|  |       - uses: actions/setup-java@v4 | ||||||
|  |         with: | ||||||
|  |           distribution: 'temurin' | ||||||
|  |           java-version: '17' | ||||||
|  |       - name: Setup Android SDK | ||||||
|  |         uses: android-actions/setup-android@v3 | ||||||
|  |       - name: Configure gradle... | ||||||
|  |         run: mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true" >> ~/.gradle/gradle.properties | ||||||
|  |       - name: Build and test | ||||||
|  |         run: ./gradlew build -x test --stacktrace | ||||||
							
								
								
									
										126
									
								
								.gitea/workflows/on_merge_on_release.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										126
									
								
								.gitea/workflows/on_merge_on_release.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,126 @@ | |||||||
|  | name: Create tag | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - release | ||||||
|  |   workflow_dispatch: | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |     uses: ./.gitea/workflows/common_build.yml | ||||||
|  |   createTagAndChangelog: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: build | ||||||
|  |     steps: | ||||||
|  |       - name: Check out repository code | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           fetch-depth: 0 | ||||||
|  |       - name: Config git | ||||||
|  |         run: | | ||||||
|  |           git config --global user.email aminecmi+giteadrone@pm.me | ||||||
|  |           git config --global user.name giteadrone | ||||||
|  |       - name: Creating the tag and generate changelog | ||||||
|  |         run: | | ||||||
|  |           git fetch --tags -p | ||||||
|  |           PREV=$(git describe --tags --abbrev=0) | ||||||
|  |           ./build.sh --publish --from-ci | ||||||
|  |           VER=$(git describe --tags --abbrev=0) | ||||||
|  |           CHANGELOG=$(git log $PREV..HEAD --pretty="- %s") | ||||||
|  |           echo "**$VER | ||||||
|  |            | ||||||
|  |           $CHANGELOG | ||||||
|  |            | ||||||
|  |           -------------------------------------------------------------------- | ||||||
|  |            | ||||||
|  |           $(cat CHANGELOG.md)" > CHANGELOG.md | ||||||
|  |           git add CHANGELOG.md | ||||||
|  |           touch ./fastlane/metadata/android/en\-US/changelogs/$VER.txt | ||||||
|  |           echo "**$VER** | ||||||
|  |            | ||||||
|  |           $CHANGELOG" > ./fastlane/metadata/android/en\-US/changelogs/$VER.txt | ||||||
|  |           git add ./fastlane/metadata/android/en\-US/changelogs/$VER.txt | ||||||
|  |           git commit -m "Changelog for $VER" | ||||||
|  |       - name: Push changes | ||||||
|  |         uses: appleboy/git-push-action@v1.0.0 | ||||||
|  |         with: | ||||||
|  |           author_name: giteadrone | ||||||
|  |           author_email: aminecmi+giteadrone@pm.me | ||||||
|  |           remote: ${{ secrets.REMOTE_URL }} | ||||||
|  |           followtags: true | ||||||
|  |           ssh_key: ${{ secrets.PRIVATE_KEY }} | ||||||
|  |           tags: true | ||||||
|  |           branch: release | ||||||
|  |       - name: copy file via ssh password | ||||||
|  |         uses: appleboy/scp-action@v0.1.7 | ||||||
|  |         with: | ||||||
|  |           host: amine-bouabdallaoui.fr | ||||||
|  |           username: ubuntu | ||||||
|  |           key: ${{ secrets.PRIVATE_KEY }} | ||||||
|  |           source: "version.txt" | ||||||
|  |           target: "/home/ubuntu/" | ||||||
|  |       - name: deploy version file | ||||||
|  |         uses: appleboy/ssh-action@v1.2.0 | ||||||
|  |         with: | ||||||
|  |           host: amine-bouabdallaoui.fr | ||||||
|  |           username: ubuntu | ||||||
|  |           key: ${{ secrets.PRIVATE_KEY }} | ||||||
|  |           script: cd /home/ubuntu && sudo rm -rf /var/www/amine/version.txt && sudo chown www-data:www-data ./version.txt && sudo mv version.txt /var/www/amine/ | ||||||
|  |   release: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: createTagAndChangelog | ||||||
|  |     steps: | ||||||
|  |       - name: Check out repository code | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           fetch-depth: 0 | ||||||
|  |       - name: Fetch tags | ||||||
|  |         id: version | ||||||
|  |         run: | | ||||||
|  |           git fetch --tags -p | ||||||
|  |           PREV=$(git describe --tags --abbrev=0) | ||||||
|  |           echo $PREV | ||||||
|  |           echo "VERSION=$PREV" >> $GITHUB_OUTPUT | ||||||
|  |       - uses: actions/setup-java@v4 | ||||||
|  |         with: | ||||||
|  |           distribution: 'temurin' | ||||||
|  |           java-version: '17' | ||||||
|  |       - name: Setup Android SDK | ||||||
|  |         uses: android-actions/setup-android@v3 | ||||||
|  |       - name: Configure gradle... | ||||||
|  |         run: mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=false\nsystemProp.org.gradle.internal.http.connectionTimeout=180000\nsystemProp.org.gradle.internal.http.socketTimeout=180000" >> ~/.gradle/gradle.properties | ||||||
|  |       - name: setup go | ||||||
|  |         uses: https://github.com/actions/setup-go@v4 | ||||||
|  |         with: | ||||||
|  |           go-version: '>=1.20.1' | ||||||
|  |       - name: Generate APK | ||||||
|  |         run: ./gradlew :androidApp:assembleGithubConfigRelease | ||||||
|  |       - name: Get Key | ||||||
|  |         run: wget ${{ secrets.KEY_URL }} | ||||||
|  |       - name: Zippalign | ||||||
|  |         run: | | ||||||
|  |           sdkmanager "build-tools;31.0.0" | ||||||
|  |           ls $ANDROID_HOME/build-tools  | ||||||
|  |           $ANDROID_HOME/build-tools/31.0.0/zipalign -f -v 4 androidApp/build/outputs/apk/githubConfig/release/androidApp-githubConfig-release-unsigned.apk androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk | ||||||
|  |       - name: Sigh | ||||||
|  |         run: $ANDROID_HOME/build-tools/31.0.0/apksigner sign -v --out signed.apk --ks ./key --ks-key-alias ${{ secrets.KEY_ALIAS }} --ks-pass pass:${{ secrets.KEYSTORE_PASSWORD }} --v1-signing-enabled true --v2-signing-enabled true androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk | ||||||
|  |       - name: Verify | ||||||
|  |         run: $ANDROID_HOME/build-tools/31.0.0/apksigner verify signed.apk | ||||||
|  |       - name: Release | ||||||
|  |         uses: https://gitea.com/actions/gitea-release-action@main | ||||||
|  |         with: | ||||||
|  |           files: signed.apk | ||||||
|  |           token: ${{ secrets.API_KEY }} | ||||||
|  |           tag_name: ${{ steps.version.outputs.VERSION }} | ||||||
|  |           name: ${{ steps.version.outputs.VERSION }} | ||||||
|  |       - name: Send mail | ||||||
|  |         uses: https://github.com/dawidd6/action-send-mail@v4 | ||||||
|  |         with: | ||||||
|  |           connection_url: ${{ secrets.MAIL_CONNECTION }} | ||||||
|  |           to: ${{ secrets.MAIL_TO }} | ||||||
|  |           from: ${{ secrets.MAIL_FROM }} | ||||||
|  |           subject: Mapping file | ||||||
|  |           priority: high | ||||||
|  |           convert_markdown: true | ||||||
|  |           body: Nouveau fichier de mapping pour la version ${{ steps.version.outputs.VERSION }} | ||||||
|  |           attachments: androidApp/build/outputs/mapping/githubConfigRelease/mapping.txt | ||||||
							
								
								
									
										26
									
								
								.gitea/workflows/on_pr.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								.gitea/workflows/on_pr.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | name: Check PR code | ||||||
|  | on: | ||||||
|  |   pull_request: | ||||||
|  |     branches: | ||||||
|  |       - master | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   Lint: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Check out repository code | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |       - uses: actions/setup-java@v4 | ||||||
|  |         with: | ||||||
|  |           distribution: 'temurin' # See 'Supported distributions' for available options | ||||||
|  |           java-version: '17' | ||||||
|  |       - name: Install klint | ||||||
|  |         run: curl -sSLO https://github.com/pinterest/ktlint/releases/download/1.0.0/ktlint && chmod a+x ktlint && mv ktlint /usr/local/bin/ | ||||||
|  |       - name: Install detekt | ||||||
|  |         run: curl -sSLO https://github.com/detekt/detekt/releases/download/v1.23.1/detekt-cli-1.23.1.zip && unzip detekt-cli-1.23.1.zip | ||||||
|  |       - name: Linting... | ||||||
|  |         run: ktlint 'shared/**/*.kt' 'androidApp/**/*.kt' '!shared/build' || true | ||||||
|  |       - name: Detecting... | ||||||
|  |         run: ./detekt-cli-1.23.1/bin/detekt-cli --all-rules --excludes '**/shared/build/**/*.kt' || true | ||||||
|  |   build: | ||||||
|  |     uses: ./.gitea/workflows/common_build.yml | ||||||
							
								
								
									
										9
									
								
								.gitea/workflows/on_push.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.gitea/workflows/on_push.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | name: Check master code | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - master | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |     uses: ./.gitea/workflows/common_build.yml | ||||||
							
								
								
									
										44
									
								
								.gitea/workflows/on_push_coverage.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								.gitea/workflows/on_push_coverage.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | |||||||
|  | name: Check master code | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - master | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   coverage: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           fetch-depth: 0 | ||||||
|  |       - name: Fetch tags | ||||||
|  |         run: git fetch --tags -p | ||||||
|  |       - uses: KengoTODA/actions-setup-docker-compose@v1 | ||||||
|  |         with: | ||||||
|  |           version: "2.23.3" | ||||||
|  |       - name: run selfoss | ||||||
|  |         run: | | ||||||
|  |           docker compose -f .gitea/workflows/assets/docker-compose.yml up -d | ||||||
|  |       - uses: actions/setup-java@v4 | ||||||
|  |         with: | ||||||
|  |           distribution: 'temurin' | ||||||
|  |           java-version: '17' | ||||||
|  |           cache: gradle | ||||||
|  |       - uses: gradle/actions/setup-gradle@v3 | ||||||
|  |       - uses: android-actions/setup-android@v3 | ||||||
|  |       - name: Configure gradle... | ||||||
|  |         run: mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true" >> ~/.gradle/gradle.properties | ||||||
|  |       - name: coverage | ||||||
|  |         run: | | ||||||
|  |           ./gradlew :koverHtmlReport | ||||||
|  |       - uses: actions/upload-artifact@v3 | ||||||
|  |         with: | ||||||
|  |           name: coverage | ||||||
|  |           path: build/reports/kover/html | ||||||
|  |           retention-days: 1 | ||||||
|  |           overwrite: true | ||||||
|  |           include-hidden-files: true | ||||||
|  |       - name: Clean | ||||||
|  |         if: always() | ||||||
|  |         run: | | ||||||
|  |           docker compose -f .gitea/workflows/assets/docker-compose.yml stop | ||||||
							
								
								
									
										2
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ Please read the guidelines before contributing, and follow them (or try to) when | |||||||
|  |  | ||||||
| There are many ways to contribute to this project, you could [translate the app](https://crowdin.com/project/readerforselfoss), report bugs, request missing features, suggest enhancements and changes to existing ones. You also can improve the README with useful tips that could help the other users. | There are many ways to contribute to this project, you could [translate the app](https://crowdin.com/project/readerforselfoss), report bugs, request missing features, suggest enhancements and changes to existing ones. You also can improve the README with useful tips that could help the other users. | ||||||
|  |  | ||||||
| You can fork the repository, and [help me solve some issues](https://gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform/issues?q=is%3Aissue+is%3Aopen+label%3A%22Up+For+Grabs%22) or [develop new things](https://gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform/issues) | You can fork the repository, and [help me solve some issues](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderForSelfoss-multiplatform/issues?q=is%3Aissue+is%3Aopen+label%3A%22Up+For+Grabs%22) or [develop new things](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderForSelfoss-multiplatform/issues) | ||||||
|  |  | ||||||
| ### What I can't help you with. | ### What I can't help you with. | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -321,3 +321,6 @@ fabric.properties | |||||||
|  |  | ||||||
|  |  | ||||||
| crowdin.properties | crowdin.properties | ||||||
|  |  | ||||||
|  | .kotlin/ | ||||||
|  | build-cache/ | ||||||
							
								
								
									
										231
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										231
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,12 +1,237 @@ | |||||||
| **v123020522** | **v124123641 | ||||||
|  |  | ||||||
| - Changelog for v123020521 [CI SKIP] | - Chore: no tests on build. | ||||||
|  | - Merge pull request 'testing' (#170) from testing into master | ||||||
|  | - fix: Displaying fixes. Fixes #155 | ||||||
|  | - test: coverage | ||||||
|  | - chore: update and use multiplatform datetime | ||||||
|  | - Changelog for v124123421 | ||||||
|  |  | ||||||
| -------------------------------------------------------------------- | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
| **v123020521** | **v124123421 | ||||||
|  |  | ||||||
|  | - fix: Trying to fix the serialization issue. | ||||||
|  | - Changelog for v124113311 | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124113311 | ||||||
|  |  | ||||||
|  | - chore: update versions. (#165) | ||||||
|  | - chore: fastlane changelog. | ||||||
|  | - chore: fastlane fixes. | ||||||
|  | - Changelog for v124113301 | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124113301** | ||||||
|  |  | ||||||
|  | - chore: Gitea Action | ||||||
|  | - Merge pull request 'chore: Gitea Action' (#164) from runner into master | ||||||
|  | - chore: Gitea Action | ||||||
|  | - chore: Readme update. | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124041081** | ||||||
|  |  | ||||||
|  | - chore: comment. | ||||||
|  | - fix: Last time fixing the parsing date hack before moving it to os version. | ||||||
|  | - Changelog for v124030731 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124030731** | ||||||
|  |  | ||||||
|  | - fix: Basic auth and password can have non whitspace characters. Fixes 142. | ||||||
|  | - Changelog for v124020451 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124020451** | ||||||
|  |  | ||||||
|  | - fix: Fixed handling of position in card adapter. | ||||||
|  | - Changelog for v124010301 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124010301** | ||||||
|  |  | ||||||
|  | - fix: This may fix the oom errors. | ||||||
|  | - Changelog for v124010191 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124010191** | ||||||
|  |  | ||||||
|  | - fix: moving listeners. | ||||||
|  | - chore: removed a useless log. | ||||||
|  | - Changelog for v124010032 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124010032** | ||||||
|  |  | ||||||
|  | - fix: Another date format thing. | ||||||
|  | - Changelog for v124010031 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v124010031** | ||||||
|  |  | ||||||
|  | - fix: Checking if selfoss instance. | ||||||
|  | - fix: handle three characters lenght hexcode colors. | ||||||
|  | - Changelog for v123113311 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123113311** | ||||||
|  |  | ||||||
|  | - chore: Source tracker url in the menu. | ||||||
|  | - fix: Handle kodein proguard rules. | ||||||
|  | - Changelog for v123102961 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123102961** | ||||||
|  |  | ||||||
|  | - chore: domain changes. | ||||||
|  | - Changelog for v123102852 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123102852** | ||||||
|  |  | ||||||
|  | - chore: lint cleaning. | ||||||
|  | - Changelog for v123102841 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123102841** | ||||||
|  |  | ||||||
|  | - chore: cleaning ci steps and upgrading dependencies. | ||||||
|  | - feat: Self signed ssl support. | ||||||
|  | - Changelog for v123061811 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123061811** | ||||||
|  |  | ||||||
|  | - feat: Added confirmation dialog for disconnect item menu. | ||||||
|  | - Changelog for v123061651 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123061651** | ||||||
|  |  | ||||||
|  | - i18n: Translation update. | ||||||
|  | - i18n: Translation update. | ||||||
|  | - i18n: Translation update. | ||||||
|  | - fix: avoid trying to open invalid image urls. | ||||||
|  | - Changelog for v123051471 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123051471** | ||||||
|  |  | ||||||
|  | - fix: images could be null. | ||||||
|  | - fix: Check if color is not empty before parsing it. | ||||||
|  | - chore: Removed unused log. | ||||||
|  | - Changelog for v123051331 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123051331** | ||||||
|  |  | ||||||
|  | - fix: illegal input. | ||||||
|  | - Changelog for v123051321 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123051321** | ||||||
|  |  | ||||||
|  | - debug: Debug null context. | ||||||
|  | - Changelog for v123051301 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123051301** | ||||||
|  |  | ||||||
|  | - feat: Basic auth from url. Fixes #142 (#143) | ||||||
|  | - debug: Debug index out of bound exception. | ||||||
|  | - Changelog for v123051211 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123051211** | ||||||
|  |  | ||||||
|  | - fix: Sometimes url isn't even defined. | ||||||
|  | - Changelog for v123041021 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123041021** | ||||||
|  |  | ||||||
|  | - fix: 'Enable Core Library Desugaring to support older Android versions' (#138) from davidoskky/ReaderForSelfoss-multiplatform:desugaring into master | ||||||
|  | - Enable Core Library Desugaring to support older Android versions | ||||||
|  | - Changelog for v123030851 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123030851** | ||||||
|  |  | ||||||
|  | - chore: replace textDrawable library (#136) | ||||||
|  | - refactor: Remove slow login check. Closes #135. | ||||||
|  | - ci: send the mapping file after a release. | ||||||
|  | - Changelog for v123030751 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123030751** | ||||||
|  |  | ||||||
|  | - debug: added a lot to pinpoint the url issue. | ||||||
|  | - feat: Use /sources/stats in the home (#133) | ||||||
|  | - Changelog for v123030681 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123030681** | ||||||
|  |  | ||||||
|  | - fix: Unread and starred can be null. | ||||||
|  | - Fixed version number issue. | ||||||
|  | - Changelog for v123030621 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123030621** | ||||||
|  |  | ||||||
|  | - fix: url required issue. | ||||||
|  | - fix: Canvas reused issue. | ||||||
|  | - Changelog for v123020572 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123020572** | ||||||
|  |  | ||||||
|  | - fix: requirecontext issues ? | ||||||
|  | - debug: activity not found exception. | ||||||
|  | - Changelog for v123020571 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123020571** | ||||||
|  |  | ||||||
|  | - chore: remove errors logging. | ||||||
|  | - fix: quickfix for url param not provided for some sources. | ||||||
|  | - Update 'CHANGELOG.md' | ||||||
|  | - Changelog for v123020523 [CI SKIP] | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | **v123020523** | ||||||
|  |  | ||||||
|  | - fix: Git changelog. | ||||||
|  |  | ||||||
| -------------------------------------------------------------------- | -------------------------------------------------------------------- | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| # ReaderForSelfoss-multiplatform [](https://build.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform) | # ReaderForSelfoss-multiplatform [](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderForSelfoss-multiplatform/actions?workflow=on_push.yml&actor=0&status=0) | ||||||
|  |  | ||||||
| [](https://crowdin.com/project/readerforselfoss) | [](https://crowdin.com/project/readerforselfoss) | ||||||
|  |  | ||||||
| @@ -10,10 +10,6 @@ If you are a user, you can still create new issues. I'll fix them when I can. | |||||||
|  |  | ||||||
| <a href="https://f-droid.org/packages/bou.amine.apps.readerforselfossv2.android"><img src="https://f-droid.org/badge/get-it-on.png" alt="Get it on F-Droid" height="100"></a> | <a href="https://f-droid.org/packages/bou.amine.apps.readerforselfossv2.android"><img src="https://f-droid.org/badge/get-it-on.png" alt="Get it on F-Droid" height="100"></a> | ||||||
|  |  | ||||||
| ## Screen captures |  | ||||||
|  |  | ||||||
| <img src="res//fr-card.png?raw=true" alt="card view" width="400"/> <img src="res//fr-list.png?raw=true" alt="list view" width="400"/> |  | ||||||
|  |  | ||||||
| ## Like my app ? | ## Like my app ? | ||||||
|  |  | ||||||
| <a href="https://www.buymeacoffee.com/aminecmi" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/lato-orange.png" alt="Buy Me A Coffee" style="height: 51px !important;width: 217px !important;" ></a> | <a href="https://www.buymeacoffee.com/aminecmi" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/lato-orange.png" alt="Buy Me A Coffee" style="height: 51px !important;width: 217px !important;" ></a> | ||||||
| @@ -22,15 +18,15 @@ If you are a user, you can still create new issues. I'll fix them when I can. | |||||||
|  |  | ||||||
| 1. **You'll have to have a Selfoss instance running.** You'll find everything you need to install it [here](https://selfoss.aditu.de/). | 1. **You'll have to have a Selfoss instance running.** You'll find everything you need to install it [here](https://selfoss.aditu.de/). | ||||||
|  |  | ||||||
| 2. Check the [Contribution guide](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/.github/CONTRIBUTING.md). | 2. Check the [Contribution guide](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/.github/CONTRIBUTING.md). | ||||||
|  |  | ||||||
| 3. Build the project by following [these steps](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/.github/CONTRIBUTING.md#build-the-project) (you should have read them after the contribution guide) | 3. Build the project by following [these steps](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/.github/CONTRIBUTING.md#build-the-project) (you should have read them after the contribution guide) | ||||||
|  |  | ||||||
| ## Useful links | ## Useful links | ||||||
|  |  | ||||||
| - [Check what changed](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/CHANGELOG.md) | - [Check what changed](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/CHANGELOG.md) | ||||||
| - [See what I'm doing](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/projects/1) | - [See what I'm doing](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/projects/1) | ||||||
| - [Create an issue, or request a new feature](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/issues) | - [Create an issue, or request a new feature](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/issues) | ||||||
| - [Help translation the app](https://crowdin.com/project/readerforselfoss) | - [Help translation the app](https://crowdin.com/project/readerforselfoss) | ||||||
|  |  | ||||||
| ## Contributors (V1) (Alphabetical order) ❤️ | ## Contributors (V1) (Alphabetical order) ❤️ | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								androidApp/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								androidApp/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1,2 @@ | |||||||
| /build | /build | ||||||
|  | .kotlin/ | ||||||
| @@ -8,15 +8,15 @@ plugins { | |||||||
|     kotlin("android") |     kotlin("android") | ||||||
|     kotlin("kapt") |     kotlin("kapt") | ||||||
|     id("com.mikepenz.aboutlibraries.plugin") |     id("com.mikepenz.aboutlibraries.plugin") | ||||||
|     id("org.jetbrains.kotlinx.kover") version "0.6.1" |     id("org.jetbrains.kotlinx.kover") | ||||||
| } | } | ||||||
|  |  | ||||||
| fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | ||||||
|     var result: String = ByteArrayOutputStream().use { outputStream -> |     val result: String = ByteArrayOutputStream().use { outputStream -> | ||||||
|         project.exec { |         project.exec { | ||||||
|             commandLine = cmd.split(" ") |             commandLine = cmd.split(" ") | ||||||
|             standardOutput = outputStream |             standardOutput = outputStream | ||||||
|             isIgnoreExitValue = ignore ?: false |             isIgnoreExitValue = ignore | ||||||
|         } |         } | ||||||
|         outputStream.toString() |         outputStream.toString() | ||||||
|     } |     } | ||||||
| @@ -24,11 +24,10 @@ fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | |||||||
| } | } | ||||||
|  |  | ||||||
| fun gitVersion(): String { | fun gitVersion(): String { | ||||||
|     var process = "" |  | ||||||
|     val maybeTagOfCurrentCommit = execWithOutput("git -C ../ describe --contains HEAD", true) |     val maybeTagOfCurrentCommit = execWithOutput("git -C ../ describe --contains HEAD", true) | ||||||
|     process = if (maybeTagOfCurrentCommit.isEmpty()) { |     val process = if (maybeTagOfCurrentCommit.isEmpty()) { | ||||||
|         println("No tag on current commit. Will take the latest one.") |         println("No tag on current commit. Will take the latest one.") | ||||||
|         execWithOutput("git -C ../ for-each-ref refs/tags --sort=-authordate --format='%(refname:short)' --count=1") |         execWithOutput("git -C ../ for-each-ref refs/tags --sort=-refname --format='%(refname:short)' --count=1") | ||||||
|     } else { |     } else { | ||||||
|         println("Tag found on current commit") |         println("Tag found on current commit") | ||||||
|         execWithOutput("git -C ../ describe --contains HEAD") |         execWithOutput("git -C ../ describe --contains HEAD") | ||||||
| @@ -56,24 +55,24 @@ fun versionNameFromGit(): String { | |||||||
|  |  | ||||||
| android { | android { | ||||||
|     compileOptions { |     compileOptions { | ||||||
|  |         isCoreLibraryDesugaringEnabled = true | ||||||
|         // Flag to enable support for the new language APIs |         // Flag to enable support for the new language APIs | ||||||
|         sourceCompatibility = JavaVersion.VERSION_11 |         sourceCompatibility = JavaVersion.VERSION_17 | ||||||
|         targetCompatibility = JavaVersion.VERSION_11 |         targetCompatibility = JavaVersion.VERSION_17 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // For Kotlin projects |     // For Kotlin projects | ||||||
|     kotlinOptions { |     kotlinOptions { | ||||||
|         jvmTarget = "11" |         jvmTarget = "17" | ||||||
|     } |     } | ||||||
|     compileSdk = 33 |     compileSdk = 34 | ||||||
|     buildToolsVersion = "33.0.0" |  | ||||||
|     buildFeatures { |     buildFeatures { | ||||||
|         viewBinding = true |         viewBinding = true | ||||||
|     } |     } | ||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         applicationId = "bou.amine.apps.readerforselfossv2.android" |         applicationId = "bou.amine.apps.readerforselfossv2.android" | ||||||
|         minSdk = 21 |         minSdk = 25 | ||||||
|         targetSdk = 33 |         targetSdk = 34 | ||||||
|         versionCode = versionCodeFromGit() |         versionCode = versionCodeFromGit() | ||||||
|         versionName = versionNameFromGit() |         versionName = versionNameFromGit() | ||||||
|  |  | ||||||
| @@ -85,8 +84,9 @@ android { | |||||||
|  |  | ||||||
|         // tests |         // tests | ||||||
|         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" |         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" | ||||||
|  |         testInstrumentationRunnerArguments["clearPackageData"] = "true" | ||||||
|     } |     } | ||||||
|     packagingOptions { |     packaging { | ||||||
|         resources { |         resources { | ||||||
|             excludes += "/META-INF/{AL2.0,LGPL2.1}" |             excludes += "/META-INF/{AL2.0,LGPL2.1}" | ||||||
|         } |         } | ||||||
| @@ -108,30 +108,39 @@ android { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     namespace = "bou.amine.apps.readerforselfossv2.android" |     namespace = "bou.amine.apps.readerforselfossv2.android" | ||||||
|  |     testOptions { | ||||||
|  |         animationsDisabled = true | ||||||
|  |         execution = "ANDROIDX_TEST_ORCHESTRATOR" | ||||||
|  |         unitTests { | ||||||
|  |             isIncludeAndroidResources = true | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     implementation(project(":shared")) |     coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:2.0.3") | ||||||
|     implementation("com.google.android.material:material:1.5.0") |  | ||||||
|     implementation("androidx.appcompat:appcompat:1.4.1") |  | ||||||
|     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.6.0") |  | ||||||
|  |  | ||||||
|     implementation("androidx.preference:preference-ktx:1.1.1") |     implementation(project(":shared")) | ||||||
|  |     implementation("com.google.android.material:material:1.9.0") | ||||||
|  |     implementation("androidx.appcompat:appcompat:1.6.1") | ||||||
|  |     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.7.1") | ||||||
|  |  | ||||||
|  |     implementation("androidx.preference:preference-ktx:1.2.1") | ||||||
|  |  | ||||||
|     implementation(fileTree(mapOf("include" to listOf("*.jar"), "dir" to "libs"))) |     implementation(fileTree(mapOf("include" to listOf("*.jar"), "dir" to "libs"))) | ||||||
|  |  | ||||||
|     // Android Support |     // Android Support | ||||||
|     implementation("androidx.appcompat:appcompat:1.4.1") |     implementation("androidx.appcompat:appcompat:1.6.1") | ||||||
|     implementation("com.google.android.material:material:1.5.0") |     implementation("com.google.android.material:material:1.9.0") | ||||||
|     implementation("androidx.recyclerview:recyclerview:1.3.0-alpha01") |     implementation("androidx.recyclerview:recyclerview:1.3.1") | ||||||
|     implementation("androidx.legacy:legacy-support-v4:1.0.0") |     implementation("androidx.legacy:legacy-support-v4:1.0.0") | ||||||
|     implementation("androidx.vectordrawable:vectordrawable:1.2.0-alpha02") |     implementation("androidx.vectordrawable:vectordrawable:1.2.0-beta01") | ||||||
|     implementation("androidx.cardview:cardview:1.0.0") |     implementation("androidx.cardview:cardview:1.0.0") | ||||||
|     implementation("androidx.annotation:annotation:1.3.0") |     implementation("androidx.annotation:annotation:1.7.0") | ||||||
|     implementation("androidx.work:work-runtime-ktx:2.7.1") |     implementation("androidx.work:work-runtime-ktx:2.8.1") | ||||||
|     implementation("androidx.constraintlayout:constraintlayout:2.1.4") |     implementation("androidx.constraintlayout:constraintlayout:2.1.4") | ||||||
|     implementation("org.jsoup:jsoup:1.14.3") |     implementation("org.jsoup:jsoup:1.15.4") | ||||||
|  |  | ||||||
|     //multidex |     //multidex | ||||||
|     implementation("androidx.multidex:multidex:2.0.1") |     implementation("androidx.multidex:multidex:2.0.1") | ||||||
| @@ -142,18 +151,17 @@ dependencies { | |||||||
|  |  | ||||||
|     // Material-ish things |     // Material-ish things | ||||||
|     implementation("com.ashokvarma.android:bottom-navigation-bar:2.2.0") |     implementation("com.ashokvarma.android:bottom-navigation-bar:2.2.0") | ||||||
|     implementation("com.amulyakhare:com.amulyakhare.textdrawable:1.0.1") |  | ||||||
|  |  | ||||||
|     // glide |     // glide | ||||||
|     kapt("com.github.bumptech.glide:compiler:4.14.2") |     kapt("com.github.bumptech.glide:compiler:4.15.0") | ||||||
|     implementation("com.github.bumptech.glide:okhttp3-integration:4.14.2") |     implementation("com.github.bumptech.glide:okhttp3-integration:4.15.0") | ||||||
|  |  | ||||||
|     // Themes |     // Themes | ||||||
|     implementation("com.github.rubensousa:floatingtoolbar:1.5.1") |     implementation("com.github.rubensousa:floatingtoolbar:1.5.1") | ||||||
|  |  | ||||||
|     // Pager |     // Pager | ||||||
|     implementation("me.relex:circleindicator:2.1.6") |     implementation("me.relex:circleindicator:2.1.6") | ||||||
|     implementation("androidx.viewpager2:viewpager2:1.1.0-beta01") |     implementation("androidx.viewpager2:viewpager2:1.1.0-beta02") | ||||||
|  |  | ||||||
|     //Dependency Injection |     //Dependency Injection | ||||||
|     implementation("org.kodein.di:kodein-di:7.14.0") |     implementation("org.kodein.di:kodein-di:7.14.0") | ||||||
| @@ -169,7 +177,7 @@ dependencies { | |||||||
|     //PhotoView |     //PhotoView | ||||||
|     implementation("com.github.chrisbanes:PhotoView:2.3.0") |     implementation("com.github.chrisbanes:PhotoView:2.3.0") | ||||||
|  |  | ||||||
|     implementation("androidx.core:core-ktx:1.8.0") |     implementation("androidx.core:core-ktx:1.12.0") | ||||||
|  |  | ||||||
|     implementation("androidx.lifecycle:lifecycle-extensions:2.2.0") |     implementation("androidx.lifecycle:lifecycle-extensions:2.2.0") | ||||||
|  |  | ||||||
| @@ -182,11 +190,20 @@ dependencies { | |||||||
|     //test |     //test | ||||||
|     testImplementation("junit:junit:4.13.2") |     testImplementation("junit:junit:4.13.2") | ||||||
|     testImplementation("io.mockk:mockk:1.12.0") |     testImplementation("io.mockk:mockk:1.12.0") | ||||||
|     testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.0") |     testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.7.1") | ||||||
|     implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") |     implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") | ||||||
|  |     androidTestImplementation("androidx.test:runner:1.6.2") | ||||||
|  |     androidTestImplementation("androidx.test:rules:1.6.1") | ||||||
|  |     androidTestImplementation("androidx.test.espresso:espresso-core:3.6.1") | ||||||
|  |     implementation("androidx.test.espresso:espresso-idling-resource:3.6.1") | ||||||
|  |     androidTestImplementation("androidx.test.ext:junit-ktx:1.2.1") | ||||||
|  |     androidTestUtil("androidx.test:orchestrator:1.5.1") | ||||||
|  |     testImplementation("org.robolectric:robolectric:4.14.1") | ||||||
|  |     testImplementation("androidx.test:core-ktx:1.6.1") | ||||||
|  |  | ||||||
|     implementation("ch.acra:acra-http:$acraVersion") |     implementation("ch.acra:acra-http:$acraVersion") | ||||||
|     implementation("ch.acra:acra-toast:$acraVersion") |     implementation("ch.acra:acra-toast:$acraVersion") | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| tasks.withType<Test> { | tasks.withType<Test> { | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								androidApp/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								androidApp/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							| @@ -55,6 +55,7 @@ | |||||||
| # maybe remove later ? | # maybe remove later ? | ||||||
| -keep class * extends androidx.fragment.app.Fragment | -keep class * extends androidx.fragment.app.Fragment | ||||||
|  |  | ||||||
|  | -dontwarn org.slf4j.impl.StaticLoggerBinder | ||||||
|  |  | ||||||
| # Keep `Companion` object fields of serializable classes. | # Keep `Companion` object fields of serializable classes. | ||||||
| # This avoids serializer lookup through `getDeclaredClasses` as done for named companion objects. | # This avoids serializer lookup through `getDeclaredClasses` as done for named companion objects. | ||||||
| @@ -85,3 +86,12 @@ | |||||||
|  |  | ||||||
| -dontwarn io.mockk.** | -dontwarn io.mockk.** | ||||||
| -keep class io.mockk.** { *; } | -keep class io.mockk.** { *; } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Kodein | ||||||
|  | -keep, allowobfuscation, allowoptimization class org.kodein.type.TypeReference | ||||||
|  | -keep, allowobfuscation, allowoptimization class org.kodein.type.JVMAbstractTypeToken$Companion$WrappingTest | ||||||
|  |  | ||||||
|  | -keep, allowobfuscation, allowoptimization class * extends org.kodein.type.TypeReference | ||||||
|  | -keep, allowobfuscation, allowoptimization class * extends org.kodein.type.JVMAbstractTypeToken$Companion$WrappingTest | ||||||
| @@ -0,0 +1,90 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.annotation.ArrayRes | ||||||
|  | import androidx.test.espresso.Espresso.onView | ||||||
|  | import androidx.test.espresso.action.ViewActions.click | ||||||
|  | import androidx.test.espresso.action.ViewActions.replaceText | ||||||
|  | import androidx.test.espresso.action.ViewActions.typeTextIntoFocusedView | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.doesNotExist | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isChecked | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isNotChecked | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withId | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import org.hamcrest.CoreMatchers.allOf | ||||||
|  |  | ||||||
|  | fun performLogin(someUrl: String? = null) { | ||||||
|  |     onView(withId(R.id.urlView)).perform(click()).perform( | ||||||
|  |         typeTextIntoFocusedView( | ||||||
|  |             if (!someUrl.isNullOrEmpty()) someUrl else "http://10.0.2.2:8888" | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |     onView(withId(R.id.signInButton)).perform(click()) | ||||||
|  |     Thread.sleep(10000) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun loginAndInitHome() { | ||||||
|  |  | ||||||
|  |     performLogin() | ||||||
|  |     onView(withText(R.string.gdpr_dialog_title)).check(matches(isDisplayed())) | ||||||
|  |     onView(withText("OK")).perform(click()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun changeAndCancelSetting(oldValue: String, newValue: String, openSettingItem: () -> Unit) { | ||||||
|  |     openSettingItem() | ||||||
|  |     onView( | ||||||
|  |         withId(android.R.id.edit) | ||||||
|  |     ).perform(replaceText(newValue)) | ||||||
|  |     onView( | ||||||
|  |         withId(android.R.id.button2) | ||||||
|  |     ).perform(click()) | ||||||
|  |     openSettingItem() | ||||||
|  |     onView( | ||||||
|  |         withId(android.R.id.edit) | ||||||
|  |     ).check(matches(withText(oldValue))) | ||||||
|  |     onView( | ||||||
|  |         withText(newValue) | ||||||
|  |     ).check(doesNotExist()) | ||||||
|  |     onView( | ||||||
|  |         withId(android.R.id.button2) | ||||||
|  |     ).perform(click()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun changeAndSaveSetting(oldValue: String, newValue: String, openSettingItem: () -> Unit) { | ||||||
|  |     openSettingItem() | ||||||
|  |     onView( | ||||||
|  |         withId(android.R.id.edit) | ||||||
|  |     ).perform(replaceText(newValue)) | ||||||
|  |     onView( | ||||||
|  |         withId(android.R.id.button1) | ||||||
|  |     ).perform(click()) | ||||||
|  |     openSettingItem() | ||||||
|  |     onView( | ||||||
|  |         withId(android.R.id.edit) | ||||||
|  |     ).check(matches(withText(newValue))) | ||||||
|  |     if (oldValue.isNotEmpty()) { | ||||||
|  |         onView( | ||||||
|  |             withText(oldValue) | ||||||
|  |         ).check(doesNotExist()) | ||||||
|  |     } | ||||||
|  |     onView( | ||||||
|  |         withId(android.R.id.button2) | ||||||
|  |     ).perform(click()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun testPreferencesFromArray( | ||||||
|  |     context: Context, | ||||||
|  |     @ArrayRes arrayRes: Int, | ||||||
|  |     openSettingItem: () -> Unit | ||||||
|  | ) { | ||||||
|  |     openSettingItem() | ||||||
|  |     context.resources.getStringArray(arrayRes).forEach { res -> | ||||||
|  |         onView(withText(res)).check(matches(allOf(isDisplayed(), isNotChecked()))) | ||||||
|  |         onView(withText(res)).perform(click()) | ||||||
|  |         onView(withText(res)).check(doesNotExist()) | ||||||
|  |         openSettingItem() | ||||||
|  |         onView(withText(res)).check(matches(allOf(isDisplayed(), isChecked()))) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,137 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.test.core.app.ApplicationProvider | ||||||
|  | import androidx.test.espresso.Espresso.onView | ||||||
|  | import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||||
|  | import androidx.test.espresso.action.ViewActions | ||||||
|  | import androidx.test.espresso.action.ViewActions.click | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isClickable | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isFocused | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isRoot | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isSelected | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withId | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||||
|  | import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||||
|  | import androidx.test.filters.LargeTest | ||||||
|  | import org.hamcrest.CoreMatchers.not | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.junit.runner.RunWith | ||||||
|  |  | ||||||
|  | @RunWith(AndroidJUnit4::class) | ||||||
|  | @LargeTest | ||||||
|  | class HomeActivityTest { | ||||||
|  |  | ||||||
|  |     @get:Rule | ||||||
|  |     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||||
|  |  | ||||||
|  |     @Before | ||||||
|  |     fun init() { | ||||||
|  |         loginAndInitHome() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testMenu() { | ||||||
|  |         onView(withId(R.id.action_search)).check(matches(isDisplayed())).check( | ||||||
|  |             matches( | ||||||
|  |                 isClickable() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withId(R.id.action_filter)).check(matches(isDisplayed())).check( | ||||||
|  |             matches( | ||||||
|  |                 isClickable() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.readAll)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.menu_home_sources)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.title_activity_settings)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.menu_home_refresh)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.issue_tracker_link)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.action_disconnect)).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testMenuActions() { | ||||||
|  |         onView(withId(R.id.action_search)).perform(click()) | ||||||
|  |         onView( | ||||||
|  |             withId(R.id.search_src_text) | ||||||
|  |         ).check(matches(isFocused())) | ||||||
|  |         onView(isRoot()).perform(ViewActions.pressBack()) | ||||||
|  |  | ||||||
|  |         onView(withId(R.id.action_filter)).perform(click()) | ||||||
|  |         onView( | ||||||
|  |             withText(R.string.filter_item_sources) | ||||||
|  |         ).check(matches(isDisplayed())) | ||||||
|  |         onView( | ||||||
|  |             withText(R.string.filter_item_tags) | ||||||
|  |         ).check(matches(isDisplayed())) | ||||||
|  |         onView( | ||||||
|  |             withId(R.id.floatingActionButton2) | ||||||
|  |         ).check(matches(isDisplayed())) | ||||||
|  |         onView(isRoot()).perform(ViewActions.pressBack()) | ||||||
|  |  | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.readAll)).perform(click()) | ||||||
|  |         onView(withText(R.string.markall_dialog_message)).check(matches(isDisplayed())) | ||||||
|  |         onView(isRoot()).perform(ViewActions.pressBack()) | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         onView(withText(R.string.menu_home_sources)).perform(click()) | ||||||
|  |         onView(withId(R.id.fab)).check(matches(isDisplayed())) | ||||||
|  |         onView(isRoot()).perform(ViewActions.pressBack()) | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||||
|  |         onView(withText(R.string.pref_header_general)).check(matches(isDisplayed())) | ||||||
|  |         onView(isRoot()).perform(ViewActions.pressBack()) | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         onView(withText(R.string.menu_home_refresh)).perform(click()) | ||||||
|  |         onView(withText(R.string.refresh_dialog_message)).check(matches(isDisplayed())) | ||||||
|  |         onView(isRoot()).perform(ViewActions.pressBack()) | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         /*onView(withText(R.string.issue_tracker_link)).perform(click()) | ||||||
|  |         onView(withText(R.string.markall_dialog_message)).check(matches(isDisplayed())) | ||||||
|  |         onView(isRoot()).perform(ViewActions.pressBack()) | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         )*/ | ||||||
|  |  | ||||||
|  |         onView(withText(R.string.action_disconnect)).perform(click()) | ||||||
|  |         onView(withText(R.string.confirm_disconnect_title)).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testEmptyView() { | ||||||
|  |         onView(withId(R.id.emptyText)).check(matches(isDisplayed())) | ||||||
|  |         onView( | ||||||
|  |             hasBottombarItemText(R.string.tab_new) | ||||||
|  |         ).check(matches(isDisplayed())).check(matches(isSelected())) | ||||||
|  |         onView( | ||||||
|  |             hasBottombarItemText(R.string.tab_read) | ||||||
|  |         ).check(matches(isDisplayed())).check(matches(not(isSelected()))) | ||||||
|  |         onView( | ||||||
|  |             hasBottombarItemText(R.string.tab_favs) | ||||||
|  |         ).check(matches(isDisplayed())).check(matches(not(isSelected()))) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,83 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import android.app.Activity | ||||||
|  | import androidx.test.espresso.Espresso.onView | ||||||
|  | import androidx.test.espresso.IdlingRegistry | ||||||
|  | import androidx.test.espresso.action.ViewActions.click | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isClickable | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isNotChecked | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withId | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||||
|  | import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||||
|  | import androidx.test.filters.LargeTest | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.testing.CountingIdlingResourceSingleton | ||||||
|  | import org.junit.After | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.junit.runner.RunWith | ||||||
|  |  | ||||||
|  | @RunWith(AndroidJUnit4::class) | ||||||
|  | @LargeTest | ||||||
|  | class LoginActivityTest { | ||||||
|  |  | ||||||
|  |     @get:Rule | ||||||
|  |     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||||
|  |  | ||||||
|  |     private fun getActivity(): Activity? { | ||||||
|  |         var activity: Activity? = null | ||||||
|  |         activityRule.scenario.onActivity { | ||||||
|  |             activity = it | ||||||
|  |         } | ||||||
|  |         return activity | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Before | ||||||
|  |     fun registerIdlingResource() { | ||||||
|  |         IdlingRegistry.getInstance() | ||||||
|  |             .register(CountingIdlingResourceSingleton.countingIdlingResource) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @After | ||||||
|  |     fun unregisterIdlingResource() { | ||||||
|  |         IdlingRegistry.getInstance() | ||||||
|  |             .unregister(CountingIdlingResourceSingleton.countingIdlingResource) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun viewIsInitialized() { | ||||||
|  |         onView(withId(R.id.urlView)).check(matches(isDisplayed())) | ||||||
|  |         onView(withId(R.id.selfSigned)).check(matches(isDisplayed())).check(matches(isNotChecked())) | ||||||
|  |             .check( | ||||||
|  |                 matches(isClickable()) | ||||||
|  |             ) | ||||||
|  |         onView(withId(R.id.withLogin)).check(matches(isDisplayed())) | ||||||
|  |             .check(matches(isNotChecked())).check( | ||||||
|  |                 matches(isClickable()) | ||||||
|  |             ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun urlError() { | ||||||
|  |         performLogin("172.17.0.1:8888") | ||||||
|  |         onView(withId(R.id.urlView)).perform(click()) | ||||||
|  |         onView(withId(R.id.urlView)).check(matches(withError(R.string.wrong_infos))) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun multiError() { | ||||||
|  |         onView(withId(R.id.signInButton)).perform(click()) | ||||||
|  |         onView(withId(R.id.signInButton)).perform(click()) | ||||||
|  |         onView(withId(R.id.signInButton)).perform(click()) | ||||||
|  |         onView(withText(R.string.warning_wrong_url)).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun connect() { | ||||||
|  |         performLogin() | ||||||
|  |         onView(withText(R.string.gdpr_dialog_title)).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,172 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import androidx.test.core.app.ApplicationProvider | ||||||
|  | import androidx.test.espresso.Espresso.onView | ||||||
|  | import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||||
|  | import androidx.test.espresso.action.ViewActions | ||||||
|  | import androidx.test.espresso.action.ViewActions.click | ||||||
|  | import androidx.test.espresso.action.ViewActions.replaceText | ||||||
|  | import androidx.test.espresso.action.ViewActions.typeTextIntoFocusedView | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isChecked | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isEnabled | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isFocused | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isRoot | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withId | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||||
|  | import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||||
|  | import androidx.test.filters.LargeTest | ||||||
|  | import org.hamcrest.CoreMatchers.allOf | ||||||
|  | import org.hamcrest.CoreMatchers.not | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.junit.runner.RunWith | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @RunWith(AndroidJUnit4::class) | ||||||
|  | @LargeTest | ||||||
|  | class SettingsActivityGeneralTest { | ||||||
|  |  | ||||||
|  |     @get:Rule | ||||||
|  |     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||||
|  |  | ||||||
|  |     @Before | ||||||
|  |     fun init() { | ||||||
|  |         loginAndInitHome() | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext() | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||||
|  |         onView(withText(R.string.pref_header_general)).perform(click()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testGeneral() { | ||||||
|  |         onView(withText(R.string.pref_api_items_number_title)).check(matches(isDisplayed())) | ||||||
|  |         onView( | ||||||
|  |             withSettingsCheckboxWidget(R.string.pref_general_infinite_loading_title) | ||||||
|  |         ).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.pref_general_category_links)).check(matches(isDisplayed())) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_article_viewer_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), isChecked() | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.reader_static_bar_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.reader_static_bar_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 isEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.pref_general_category_displaying)).check(matches(isDisplayed())) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_card_view_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.card_height_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.card_height_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 not(isEnabled()) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.switch_unread_count_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), isChecked() | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.display_all_counts_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testGeneralActionsNumberItems() { | ||||||
|  |         onView(withText(R.string.pref_api_items_number_title)).perform(click()) | ||||||
|  |         onView(withId(android.R.id.edit)).check(matches(isFocused())) | ||||||
|  |  | ||||||
|  |         // Value check | ||||||
|  |         onView( | ||||||
|  |             withId(android.R.id.edit) | ||||||
|  |         ).perform(replaceText("AVC")) | ||||||
|  |             .check(matches(withText(""))) | ||||||
|  |         // TODO: should check message error. Not working for api level 30+ | ||||||
|  |         onView( | ||||||
|  |             withId(android.R.id.edit) | ||||||
|  |         ).perform(replaceText("-1")) | ||||||
|  |             .check(matches(withText(""))) | ||||||
|  |         // TODO: should check message error. Not working for api level 30+ | ||||||
|  |         onView( | ||||||
|  |             withId(android.R.id.edit) | ||||||
|  |         ).perform(replaceText("300")) | ||||||
|  |             .check(matches(withText(""))) | ||||||
|  |         onView( | ||||||
|  |             withId(android.R.id.edit) | ||||||
|  |         ).perform(typeTextIntoFocusedView("300")) | ||||||
|  |             .check(matches(withText("30"))) | ||||||
|  |         onView( | ||||||
|  |             withId(android.R.id.edit) | ||||||
|  |         ).perform(replaceText("10")) | ||||||
|  |             .check(matches(withText("10"))) | ||||||
|  |         onView(isRoot()).perform(ViewActions.pressBack()) | ||||||
|  |  | ||||||
|  |         // Value saving | ||||||
|  |         changeAndCancelSetting("20", "10") { | ||||||
|  |             onView(withText(R.string.pref_api_items_number_title)).perform(click()) | ||||||
|  |         } | ||||||
|  |         changeAndSaveSetting("20", "10") { | ||||||
|  |             onView(withText(R.string.pref_api_items_number_title)).perform(click()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testGeneralActionsCheckboxes() { | ||||||
|  |         // article viewer settings | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.reader_static_bar_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 isEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_article_viewer_title)).perform(click()) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.reader_static_bar_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 not(isEnabled()) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.card_height_title)).check(matches(not(isEnabled()))) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_card_view_title)).perform(click()) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.card_height_title)).check(matches(isEnabled())) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,169 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.test.core.app.ApplicationProvider | ||||||
|  | import androidx.test.espresso.Espresso.onView | ||||||
|  | import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||||
|  | import androidx.test.espresso.action.ViewActions.click | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isChecked | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isEnabled | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isNotEnabled | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||||
|  | import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||||
|  | import androidx.test.filters.LargeTest | ||||||
|  | import org.hamcrest.CoreMatchers.allOf | ||||||
|  | import org.hamcrest.CoreMatchers.not | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.junit.runner.RunWith | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @RunWith(AndroidJUnit4::class) | ||||||
|  | @LargeTest | ||||||
|  | class SettingsActivityOfflineTest { | ||||||
|  |  | ||||||
|  |     @get:Rule | ||||||
|  |     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||||
|  |  | ||||||
|  |     lateinit var context: Context | ||||||
|  |  | ||||||
|  |     @Before | ||||||
|  |     fun init() { | ||||||
|  |         activityRule.scenario.onActivity { activity -> | ||||||
|  |             context = activity.window.context | ||||||
|  |         } | ||||||
|  |         loginAndInitHome() | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext() | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||||
|  |         onView(withText(R.string.pref_header_offline)).perform(click()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testOffline() { | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_periodic_refresh)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_items_caching)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_items_caching)).check( | ||||||
|  |             matches( | ||||||
|  |                 isEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.pref_periodic_refresh_minutes_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf(isNotEnabled(), isDisplayed()) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_refresh_when_charging)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_refresh_when_charging)).check( | ||||||
|  |             matches( | ||||||
|  |                 isNotEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_notify_new_items)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not(isChecked()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_notify_new_items)).check( | ||||||
|  |             matches( | ||||||
|  |                 isNotEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_update_sources)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), isChecked() | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testOfflineActions() { | ||||||
|  |         onView(withText(R.string.pref_switch_items_caching_off)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.pref_switch_items_caching)).perform(click()) | ||||||
|  |         onView(withText(R.string.pref_switch_items_caching_on)).check(matches(isDisplayed())) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_items_caching)).check( | ||||||
|  |             matches( | ||||||
|  |                 isEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.pref_periodic_refresh_minutes_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 isNotEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_refresh_when_charging)).check( | ||||||
|  |             matches( | ||||||
|  |                 isNotEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_notify_new_items)).check( | ||||||
|  |             matches( | ||||||
|  |                 isNotEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         onView(withText(R.string.pref_switch_periodic_refresh_off)).check( | ||||||
|  |             matches( | ||||||
|  |                 isDisplayed() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_periodic_refresh)).perform(click()) | ||||||
|  |         onView(withText(R.string.pref_switch_periodic_refresh_on)).check( | ||||||
|  |             matches( | ||||||
|  |                 isDisplayed() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_periodic_refresh_minutes_title)).check( | ||||||
|  |             matches( | ||||||
|  |                 isEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_refresh_when_charging)).check( | ||||||
|  |             matches( | ||||||
|  |                 isEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_notify_new_items)).check( | ||||||
|  |             matches( | ||||||
|  |                 isEnabled() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         changeAndCancelSetting("360", "123") { | ||||||
|  |             onView(withText(R.string.pref_periodic_refresh_minutes_title)).perform(click()) | ||||||
|  |         } | ||||||
|  |         changeAndSaveSetting("360", "123") { | ||||||
|  |             onView(withText(R.string.pref_periodic_refresh_minutes_title)).perform(click()) | ||||||
|  |         } | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_refresh_when_charging)).perform(click()) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_notify_new_items)).perform(click()) | ||||||
|  |         onView(withSettingsCheckboxWidget(R.string.pref_switch_update_sources)).perform(click()) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,86 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.test.core.app.ApplicationProvider | ||||||
|  | import androidx.test.espresso.Espresso.onView | ||||||
|  | import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||||
|  | import androidx.test.espresso.action.ViewActions.click | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isChecked | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||||
|  | import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||||
|  | import androidx.test.filters.LargeTest | ||||||
|  | import org.hamcrest.CoreMatchers.allOf | ||||||
|  | import org.hamcrest.CoreMatchers.not | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.junit.runner.RunWith | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @RunWith(AndroidJUnit4::class) | ||||||
|  | @LargeTest | ||||||
|  | class SettingsActivityReaderTest { | ||||||
|  |  | ||||||
|  |     @get:Rule | ||||||
|  |     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||||
|  |  | ||||||
|  |     lateinit var context: Context | ||||||
|  |  | ||||||
|  |     @Before | ||||||
|  |     fun init() { | ||||||
|  |         activityRule.scenario.onActivity { activity -> | ||||||
|  |             context = activity.window.context | ||||||
|  |         } | ||||||
|  |         loginAndInitHome() | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext() | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||||
|  |         onView(withText(R.string.pref_header_viewer)).perform(click()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testReader() { | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_actions_pager_scroll)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), not( | ||||||
|  |                         isChecked() | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.pref_content_reader_font_size)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.settings_reader_font)).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testReaderActions() { | ||||||
|  |         onView(withText(R.string.pref_switch_actions_pager_scroll_off)).check( | ||||||
|  |             matches( | ||||||
|  |                 isDisplayed() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withSettingsCheckboxFrame(R.string.pref_switch_actions_pager_scroll)).perform(click()) | ||||||
|  |         onView(withText(R.string.pref_switch_actions_pager_scroll_on)).check( | ||||||
|  |             matches( | ||||||
|  |                 isDisplayed() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         onView(withText(R.string.pref_content_reader_font_size)).perform(click()) | ||||||
|  |         changeAndCancelSetting("16", "10") { | ||||||
|  |             onView(withText(R.string.pref_content_reader_font_size)).perform(click()) | ||||||
|  |         } | ||||||
|  |         changeAndSaveSetting("16", "10") { | ||||||
|  |             onView(withText(R.string.pref_content_reader_font_size)).perform(click()) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         testPreferencesFromArray(context, R.array.preloaded_fonts_values) { | ||||||
|  |             onView(withText(R.string.settings_reader_font)).perform(click()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,104 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.test.core.app.ApplicationProvider | ||||||
|  | import androidx.test.espresso.Espresso.onView | ||||||
|  | import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||||
|  | import androidx.test.espresso.action.ViewActions.click | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isSelected | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||||
|  | import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||||
|  | import androidx.test.filters.LargeTest | ||||||
|  | import org.hamcrest.CoreMatchers.allOf | ||||||
|  | import org.hamcrest.CoreMatchers.not | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.junit.runner.RunWith | ||||||
|  |  | ||||||
|  | @RunWith(AndroidJUnit4::class) | ||||||
|  | @LargeTest | ||||||
|  | class SettingsActivityTest { | ||||||
|  |  | ||||||
|  |     @get:Rule | ||||||
|  |     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||||
|  |     lateinit var context: Context | ||||||
|  |  | ||||||
|  |     @Before | ||||||
|  |     fun init() { | ||||||
|  |         activityRule.scenario.onActivity { activity -> | ||||||
|  |             context = activity.window.context | ||||||
|  |         } | ||||||
|  |         loginAndInitHome() | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testAllSettings() { | ||||||
|  |  | ||||||
|  |         onView(withText(R.string.pref_header_general)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.pref_header_viewer)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.pref_header_offline)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.pref_header_theme)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.pref_header_links)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.pref_switch_disable_acra)).check( | ||||||
|  |             matches( | ||||||
|  |                 allOf( | ||||||
|  |                     isDisplayed(), | ||||||
|  |                     not(isSelected()) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.action_about)).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testThemes() { | ||||||
|  |         testPreferencesFromArray(context, R.array.ModeTitles) { | ||||||
|  |             onView(withText(R.string.pref_header_theme)).perform(click()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testExperimentail() { | ||||||
|  |         onView(withText(R.string.pref_header_experimental)).perform(click()) | ||||||
|  |         changeAndCancelSetting("", "10") { | ||||||
|  |             onView(withText(R.string.pref_api_timeout)).perform(click()) | ||||||
|  |         } | ||||||
|  |         changeAndSaveSetting("", "10") { | ||||||
|  |             onView(withText(R.string.pref_api_timeout)).perform(click()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testBugReports() { | ||||||
|  |         onView(withText(R.string.pref_switch_disable_acra)).perform(click()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testLinks() { | ||||||
|  |         onView(withText(R.string.pref_header_links)).perform(click()) | ||||||
|  |         onView(withText(R.string.issue_tracker_link)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.issue_tracker_summary)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.source_code)).check(matches(isDisplayed())) | ||||||
|  |         onView(withText(R.string.translation)).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun testAbout() { | ||||||
|  |         onView(withText(R.string.action_about)).perform(click()) | ||||||
|  |         onView(withText("ACRA")).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,65 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.test.core.app.ApplicationProvider | ||||||
|  | import androidx.test.espresso.Espresso.onView | ||||||
|  | import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||||
|  | import androidx.test.espresso.action.ViewActions.click | ||||||
|  | import androidx.test.espresso.action.ViewActions.scrollCompletelyTo | ||||||
|  | import androidx.test.espresso.action.ViewActions.typeTextIntoFocusedView | ||||||
|  | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withId | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||||
|  | import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||||
|  | import androidx.test.filters.LargeTest | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.junit.runner.RunWith | ||||||
|  | import java.util.UUID | ||||||
|  |  | ||||||
|  | @RunWith(AndroidJUnit4::class) | ||||||
|  | @LargeTest | ||||||
|  | class SourcesActivityTest { | ||||||
|  |  | ||||||
|  |     @get:Rule | ||||||
|  |     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||||
|  |  | ||||||
|  |     lateinit var sourceName: String | ||||||
|  |  | ||||||
|  |     @Before | ||||||
|  |     fun init() { | ||||||
|  |         sourceName = UUID.randomUUID().toString().substring(0, 15) | ||||||
|  |  | ||||||
|  |         loginAndInitHome() | ||||||
|  |         openActionBarOverflowOrOptionsMenu( | ||||||
|  |             ApplicationProvider.getApplicationContext<Context>() | ||||||
|  |         ) | ||||||
|  |         onView(withText(R.string.menu_home_sources)) | ||||||
|  |             .perform(click()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun addSource() { | ||||||
|  |         onView(withId(R.id.fab)) | ||||||
|  |             .perform(click()) | ||||||
|  |         onView(withId(R.id.nameInput)) | ||||||
|  |             .perform(click()).perform(typeTextIntoFocusedView(sourceName)) | ||||||
|  |         onView(withId(R.id.sourceUri)) | ||||||
|  |             .perform(click()) | ||||||
|  |             .perform(typeTextIntoFocusedView("https://lorem-rss.herokuapp.com/feed?unit=year&interval=1&length=10")) | ||||||
|  |         onView(withId(R.id.tags)) | ||||||
|  |             .perform(click()).perform(typeTextIntoFocusedView("tag1,tag2,tag3")) | ||||||
|  |         onView(withId(R.id.spoutsSpinner)) | ||||||
|  |             .perform(click()) | ||||||
|  |         onView(withText("RSS Feed")) | ||||||
|  |             .perform(scrollCompletelyTo()) | ||||||
|  |             .perform(click()) | ||||||
|  |         onView(withId(R.id.saveBtn)) | ||||||
|  |             .perform(click()) | ||||||
|  |         onView(withText(sourceName)).check(matches(isDisplayed())) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,101 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import android.view.View | ||||||
|  | import android.widget.EditText | ||||||
|  | import android.widget.ImageView | ||||||
|  | import android.widget.RelativeLayout | ||||||
|  | import androidx.annotation.DrawableRes | ||||||
|  | import androidx.annotation.StringRes | ||||||
|  | import androidx.core.graphics.drawable.toBitmap | ||||||
|  | import androidx.test.espresso.Root | ||||||
|  | import androidx.test.espresso.matcher.RootMatchers.isPlatformPopup | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.hasSibling | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withChild | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withClassName | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withId | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withParent | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withResourceName | ||||||
|  | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
|  | import org.hamcrest.CoreMatchers.allOf | ||||||
|  | import org.hamcrest.Description | ||||||
|  | import org.hamcrest.Matcher | ||||||
|  | import org.hamcrest.Matchers | ||||||
|  | import org.hamcrest.TypeSafeMatcher | ||||||
|  |  | ||||||
|  |  | ||||||
|  | fun withError(@StringRes id: Int): TypeSafeMatcher<View?> { | ||||||
|  |     return object : TypeSafeMatcher<View?>() { | ||||||
|  |         override fun matchesSafely(view: View?): Boolean { | ||||||
|  |             if (view == null) { | ||||||
|  |                 return false | ||||||
|  |             } | ||||||
|  |             val context = view.context | ||||||
|  |             if (view !is EditText) { | ||||||
|  |                 return false | ||||||
|  |             } | ||||||
|  |             if (view.error == null) { | ||||||
|  |                 return false | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return view.error.toString() == context.getString(id) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         override fun describeTo(description: Description?) { | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun isPopupWindow(): Matcher<Root> { | ||||||
|  |     return isPlatformPopup() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun withDrawable(@DrawableRes id: Int) = object : TypeSafeMatcher<View>() { | ||||||
|  |     override fun describeTo(description: Description) { | ||||||
|  |         description.appendText("ImageView with drawable same as drawable with id $id") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun matchesSafely(view: View): Boolean { | ||||||
|  |         val context = view.context | ||||||
|  |         val expectedBitmap = context.getDrawable(id)!!.toBitmap() | ||||||
|  |         try { | ||||||
|  |             return view is ImageView && view.drawable.toBitmap().sameAs(expectedBitmap) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun hasBottombarItemText(@StringRes id: Int): Matcher<View>? { | ||||||
|  |     return allOf( | ||||||
|  |         withResourceName("fixed_bottom_navigation_icon"), | ||||||
|  |         withParent( | ||||||
|  |             allOf( | ||||||
|  |                 withResourceName("fixed_bottom_navigation_icon_container"), | ||||||
|  |                 hasSibling(withText(id)) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun withSettingsCheckboxWidget(@StringRes id: Int): Matcher<View>? { | ||||||
|  |     return allOf( | ||||||
|  |         withId(android.R.id.switch_widget), | ||||||
|  |         withParent( | ||||||
|  |             withSettingsCheckboxFrame(id) | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun withSettingsCheckboxFrame(@StringRes id: Int): Matcher<View>? { | ||||||
|  |     return allOf( | ||||||
|  |         withId(android.R.id.widget_frame), | ||||||
|  |         hasSibling( | ||||||
|  |             allOf( | ||||||
|  |                 withClassName(Matchers.equalTo(RelativeLayout::class.java.name)), | ||||||
|  |                 withChild( | ||||||
|  |                     withText(id) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -1,6 +1,7 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.android | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
|  | import android.net.Uri | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.view.Menu | import android.view.Menu | ||||||
| import android.view.MenuItem | import android.view.MenuItem | ||||||
| @@ -12,7 +13,11 @@ import androidx.appcompat.app.AppCompatActivity | |||||||
| import androidx.appcompat.widget.SearchView | import androidx.appcompat.widget.SearchView | ||||||
| import androidx.core.view.doOnNextLayout | import androidx.core.view.doOnNextLayout | ||||||
| import androidx.lifecycle.lifecycleScope | import androidx.lifecycle.lifecycleScope | ||||||
| import androidx.recyclerview.widget.* | import androidx.recyclerview.widget.DividerItemDecoration | ||||||
|  | import androidx.recyclerview.widget.GridLayoutManager | ||||||
|  | import androidx.recyclerview.widget.ItemTouchHelper | ||||||
|  | import androidx.recyclerview.widget.RecyclerView | ||||||
|  | import androidx.recyclerview.widget.StaggeredGridLayoutManager | ||||||
| import androidx.work.Constraints | import androidx.work.Constraints | ||||||
| import androidx.work.ExistingPeriodicWorkPolicy | import androidx.work.ExistingPeriodicWorkPolicy | ||||||
| import androidx.work.PeriodicWorkRequestBuilder | import androidx.work.PeriodicWorkRequestBuilder | ||||||
| @@ -24,6 +29,7 @@ import bou.amine.apps.readerforselfossv2.android.background.LoadingWorker | |||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityHomeBinding | import bou.amine.apps.readerforselfossv2.android.databinding.ActivityHomeBinding | ||||||
| import bou.amine.apps.readerforselfossv2.android.fragments.FilterSheetFragment | import bou.amine.apps.readerforselfossv2.android.fragments.FilterSheetFragment | ||||||
| import bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity | import bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.testing.CountingIdlingResourceSingleton | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.bottombar.maybeShow | import bou.amine.apps.readerforselfossv2.android.utils.bottombar.maybeShow | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.bottombar.removeBadge | import bou.amine.apps.readerforselfossv2.android.utils.bottombar.removeBadge | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| @@ -43,9 +49,7 @@ import org.kodein.di.instance | |||||||
| import java.security.MessageDigest | import java.security.MessageDigest | ||||||
| import java.util.concurrent.TimeUnit | import java.util.concurrent.TimeUnit | ||||||
|  |  | ||||||
|  |  | ||||||
| class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAware { | class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAware { | ||||||
|  |  | ||||||
|     private var items: ArrayList<SelfossModel.Item> = ArrayList() |     private var items: ArrayList<SelfossModel.Item> = ArrayList() | ||||||
|  |  | ||||||
|     private var elementsShown: ItemType = ItemType.UNREAD |     private var elementsShown: ItemType = ItemType.UNREAD | ||||||
| @@ -59,18 +63,18 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|     private lateinit var recyclerViewScrollListener: RecyclerView.OnScrollListener |     private lateinit var recyclerViewScrollListener: RecyclerView.OnScrollListener | ||||||
|     private lateinit var binding: ActivityHomeBinding |     private lateinit var binding: ActivityHomeBinding | ||||||
|  |  | ||||||
|     private var recyclerAdapter: RecyclerView.Adapter<*>? = null |     private var recyclerAdapter: ItemsAdapter<out RecyclerView.ViewHolder>? = null | ||||||
|  |  | ||||||
|     private var fromTabShortcut: Boolean = false |     private var fromTabShortcut: Boolean = false | ||||||
|  |  | ||||||
|     private val settingsLauncher = registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { |     private val settingsLauncher = | ||||||
|  |         registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { | ||||||
|             appSettingsService.refreshUserSettings() |             appSettingsService.refreshUserSettings() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     override val di by closestDI() |     override val di by closestDI() | ||||||
|     private val repository : Repository by instance() |     private val repository: Repository by instance() | ||||||
|     private val appSettingsService : AppSettingsService by instance() |     private val appSettingsService: AppSettingsService by instance() | ||||||
|  |  | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
| @@ -81,7 +85,8 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         repository.offlineOverride = intent.getBooleanExtra("startOffline", false) |         repository.offlineOverride = intent.getBooleanExtra("startOffline", false) | ||||||
|  |  | ||||||
|         if (fromTabShortcut) { |         if (fromTabShortcut) { | ||||||
|             elementsShown = ItemType.fromInt(intent.getIntExtra("shortcutTab", ItemType.UNREAD.position)) |             elementsShown = | ||||||
|  |                 ItemType.fromInt(intent.getIntExtra("shortcutTab", ItemType.UNREAD.position)) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         setContentView(view) |         setContentView(view) | ||||||
| @@ -92,10 +97,11 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|         handleSwipeRefreshLayout() |         handleSwipeRefreshLayout() | ||||||
|  |  | ||||||
|  |  | ||||||
|         if (appSettingsService.isItemCachingEnabled()) { |         if (appSettingsService.isItemCachingEnabled()) { | ||||||
|  |             CountingIdlingResourceSingleton.increment() | ||||||
|             CoroutineScope(Dispatchers.Main).launch { |             CoroutineScope(Dispatchers.Main).launch { | ||||||
|                 repository.tryToCacheItemsAndGetNewOnes() |                 repository.tryToCacheItemsAndGetNewOnes() | ||||||
|  |                 CountingIdlingResourceSingleton.decrement() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -104,18 +110,21 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         binding.swipeRefreshLayout.setColorSchemeResources( |         binding.swipeRefreshLayout.setColorSchemeResources( | ||||||
|             R.color.refresh_progress_1, |             R.color.refresh_progress_1, | ||||||
|             R.color.refresh_progress_2, |             R.color.refresh_progress_2, | ||||||
|             R.color.refresh_progress_3 |             R.color.refresh_progress_3, | ||||||
|         ) |         ) | ||||||
|         binding.swipeRefreshLayout.setOnRefreshListener { |         binding.swipeRefreshLayout.setOnRefreshListener { | ||||||
|             repository.offlineOverride = false |             repository.offlineOverride = false | ||||||
|             lastFetchDone = false |             lastFetchDone = false | ||||||
|  |             CountingIdlingResourceSingleton.increment() | ||||||
|             CoroutineScope(Dispatchers.Main).launch { |             CoroutineScope(Dispatchers.Main).launch { | ||||||
|                 getElementsAccordingToTab() |                 getElementsAccordingToTab() | ||||||
|                 binding.swipeRefreshLayout.isRefreshing = false |                 binding.swipeRefreshLayout.isRefreshing = false | ||||||
|  |                 CountingIdlingResourceSingleton.decrement() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         val swipeDirs = if (appSettingsService.getPublicAccess()) { |         val swipeDirs = | ||||||
|  |             if (appSettingsService.getPublicAccess()) { | ||||||
|                 0 |                 0 | ||||||
|             } else { |             } else { | ||||||
|                 ItemTouchHelper.LEFT or ItemTouchHelper.RIGHT |                 ItemTouchHelper.LEFT or ItemTouchHelper.RIGHT | ||||||
| @@ -124,28 +133,31 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         val simpleItemTouchCallback = |         val simpleItemTouchCallback = | ||||||
|             object : ItemTouchHelper.SimpleCallback( |             object : ItemTouchHelper.SimpleCallback( | ||||||
|                 0, |                 0, | ||||||
|                 swipeDirs |                 swipeDirs, | ||||||
|             ) { |             ) { | ||||||
|                 override fun getSwipeDirs( |                 override fun getSwipeDirs( | ||||||
|                     recyclerView: RecyclerView, |                     recyclerView: RecyclerView, | ||||||
|                     viewHolder: RecyclerView.ViewHolder |                     viewHolder: RecyclerView.ViewHolder, | ||||||
|                 ): Int = |                 ): Int = | ||||||
|                     if (elementsShown == ItemType.STARRED) { |                     if (elementsShown == ItemType.STARRED) { | ||||||
|                         0 |                         0 | ||||||
|                     } else { |                     } else { | ||||||
|                         super.getSwipeDirs( |                         super.getSwipeDirs( | ||||||
|                             recyclerView, |                             recyclerView, | ||||||
|                             viewHolder |                             viewHolder, | ||||||
|                         ) |                         ) | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                 override fun onMove( |                 override fun onMove( | ||||||
|                     recyclerView: RecyclerView, |                     recyclerView: RecyclerView, | ||||||
|                     viewHolder: RecyclerView.ViewHolder, |                     viewHolder: RecyclerView.ViewHolder, | ||||||
|                     target: RecyclerView.ViewHolder |                     target: RecyclerView.ViewHolder, | ||||||
|                 ): Boolean = false |                 ): Boolean = false | ||||||
|  |  | ||||||
|                 override fun onSwiped(viewHolder: RecyclerView.ViewHolder, swipeDir: Int) { |                 override fun onSwiped( | ||||||
|  |                     viewHolder: RecyclerView.ViewHolder, | ||||||
|  |                     swipeDir: Int, | ||||||
|  |                 ) { | ||||||
|                     val position = viewHolder.bindingAdapterPosition |                     val position = viewHolder.bindingAdapterPosition | ||||||
|                     val i = items.elementAtOrNull(position) |                     val i = items.elementAtOrNull(position) | ||||||
|  |  | ||||||
| @@ -162,7 +174,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|                         Toast.makeText( |                         Toast.makeText( | ||||||
|                             this@HomeActivity, |                             this@HomeActivity, | ||||||
|                             "Found null when swiping at positon $position.", |                             "Found null when swiping at positon $position.", | ||||||
|                             Toast.LENGTH_LONG |                             Toast.LENGTH_LONG, | ||||||
|                         ).show() |                         ).show() | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
| @@ -171,7 +183,10 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         ItemTouchHelper(simpleItemTouchCallback).attachToRecyclerView(binding.recyclerView) |         ItemTouchHelper(simpleItemTouchCallback).attachToRecyclerView(binding.recyclerView) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun updateBottomBarBadgeCount(badge: TextBadgeItem, count: Int) { |     private fun updateBottomBarBadgeCount( | ||||||
|  |         badge: TextBadgeItem, | ||||||
|  |         count: Int, | ||||||
|  |     ) { | ||||||
|         if (count > 0) { |         if (count > 0) { | ||||||
|             badge |             badge | ||||||
|                 .setText(count.toString()) |                 .setText(count.toString()) | ||||||
| @@ -182,14 +197,16 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleBottomBar() { |     private fun handleBottomBar() { | ||||||
|  |         tabNewBadge = | ||||||
|         tabNewBadge = TextBadgeItem() |             TextBadgeItem() | ||||||
|                 .setText("") |                 .setText("") | ||||||
|                 .setHideOnSelect(false).hide(false) |                 .setHideOnSelect(false).hide(false) | ||||||
|         tabArchiveBadge = TextBadgeItem() |         tabArchiveBadge = | ||||||
|  |             TextBadgeItem() | ||||||
|                 .setText("") |                 .setText("") | ||||||
|                 .setHideOnSelect(false).hide(false) |                 .setHideOnSelect(false).hide(false) | ||||||
|         tabStarredBadge = TextBadgeItem() |         tabStarredBadge = | ||||||
|  |             TextBadgeItem() | ||||||
|                 .setText("") |                 .setText("") | ||||||
|                 .setHideOnSelect(false).hide(false) |                 .setHideOnSelect(false).hide(false) | ||||||
|  |  | ||||||
| @@ -218,19 +235,19 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         val tabNew = |         val tabNew = | ||||||
|             BottomNavigationItem( |             BottomNavigationItem( | ||||||
|                 R.drawable.ic_tab_fiber_new_black_24dp, |                 R.drawable.ic_tab_fiber_new_black_24dp, | ||||||
|                 getString(R.string.tab_new) |                 getString(R.string.tab_new), | ||||||
|             ) |             ) | ||||||
|                 .setBadgeItem(tabNewBadge) |                 .setBadgeItem(tabNewBadge) | ||||||
|         val tabArchive = |         val tabArchive = | ||||||
|             BottomNavigationItem( |             BottomNavigationItem( | ||||||
|                 R.drawable.ic_tab_archive_black_24dp, |                 R.drawable.ic_tab_archive_black_24dp, | ||||||
|                 getString(R.string.tab_read) |                 getString(R.string.tab_read), | ||||||
|             ) |             ) | ||||||
|                 .setBadgeItem(tabArchiveBadge) |                 .setBadgeItem(tabArchiveBadge) | ||||||
|         val tabStarred = |         val tabStarred = | ||||||
|             BottomNavigationItem( |             BottomNavigationItem( | ||||||
|                 R.drawable.ic_tab_favorite_black_24dp, |                 R.drawable.ic_tab_favorite_black_24dp, | ||||||
|                 getString(R.string.tab_favs) |                 getString(R.string.tab_favs), | ||||||
|             ).setActiveColorResource(R.color.pink) |             ).setActiveColorResource(R.color.pink) | ||||||
|                 .setBadgeItem(tabStarredBadge) |                 .setBadgeItem(tabStarredBadge) | ||||||
|  |  | ||||||
| @@ -263,15 +280,15 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         handleGDPRDialog(appSettingsService.settings.getBoolean("GDPR_shown", false)) |         handleGDPRDialog(appSettingsService.settings.getBoolean("GDPR_shown", false)) | ||||||
|  |  | ||||||
|         handleRecurringTask() |         handleRecurringTask() | ||||||
|  |         CountingIdlingResourceSingleton.increment() | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|             repository.handleDBActions() |             repository.handleDBActions() | ||||||
|  |             CountingIdlingResourceSingleton.decrement() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         getElementsAccordingToTab() |         getElementsAccordingToTab() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     private fun handleGDPRDialog(GDPRShown: Boolean) { |     private fun handleGDPRDialog(GDPRShown: Boolean) { | ||||||
|         val messageDigest: MessageDigest = MessageDigest.getInstance("SHA-256") |         val messageDigest: MessageDigest = MessageDigest.getInstance("SHA-256") | ||||||
|         messageDigest.update(appSettingsService.getBaseUrl().toByteArray()) |         messageDigest.update(appSettingsService.getBaseUrl().toByteArray()) | ||||||
| @@ -281,7 +298,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|             alertDialog.setMessage(getString(R.string.gdpr_dialog_message)) |             alertDialog.setMessage(getString(R.string.gdpr_dialog_message)) | ||||||
|             alertDialog.setButton( |             alertDialog.setButton( | ||||||
|                 AlertDialog.BUTTON_NEUTRAL, |                 AlertDialog.BUTTON_NEUTRAL, | ||||||
|                 "OK" |                 "OK", | ||||||
|             ) { dialog, _ -> |             ) { dialog, _ -> | ||||||
|                 appSettingsService.settings.putBoolean("GDPR_shown", true) |                 appSettingsService.settings.putBoolean("GDPR_shown", true) | ||||||
|                 dialog.dismiss() |                 dialog.dismiss() | ||||||
| @@ -298,34 +315,40 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         when (currentManager) { |         when (currentManager) { | ||||||
|             is StaggeredGridLayoutManager -> |             is StaggeredGridLayoutManager -> | ||||||
|                 if (!appSettingsService.isCardViewEnabled()) { |                 if (!appSettingsService.isCardViewEnabled()) { | ||||||
|                     layoutManager = GridLayoutManager( |                     layoutManager = | ||||||
|  |                         GridLayoutManager( | ||||||
|                             this, |                             this, | ||||||
|                         calculateNoOfColumns() |                             calculateNoOfColumns(), | ||||||
|                         ) |                         ) | ||||||
|                     binding.recyclerView.layoutManager = layoutManager |                     binding.recyclerView.layoutManager = layoutManager | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|             is GridLayoutManager -> |             is GridLayoutManager -> | ||||||
|                 if (appSettingsService.isCardViewEnabled()) { |                 if (appSettingsService.isCardViewEnabled()) { | ||||||
|                     layoutManager = StaggeredGridLayoutManager( |                     layoutManager = | ||||||
|  |                         StaggeredGridLayoutManager( | ||||||
|                             calculateNoOfColumns(), |                             calculateNoOfColumns(), | ||||||
|                         StaggeredGridLayoutManager.VERTICAL |                             StaggeredGridLayoutManager.VERTICAL, | ||||||
|                         ) |                         ) | ||||||
|                     layoutManager.gapStrategy = |                     layoutManager.gapStrategy = | ||||||
|                         StaggeredGridLayoutManager.GAP_HANDLING_MOVE_ITEMS_BETWEEN_SPANS |                         StaggeredGridLayoutManager.GAP_HANDLING_MOVE_ITEMS_BETWEEN_SPANS | ||||||
|                     binding.recyclerView.layoutManager = layoutManager |                     binding.recyclerView.layoutManager = layoutManager | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|             else -> |             else -> | ||||||
|                 if (currentManager == null) { |                 if (currentManager == null) { | ||||||
|                     if (!appSettingsService.isCardViewEnabled()) { |                     if (!appSettingsService.isCardViewEnabled()) { | ||||||
|                         layoutManager = GridLayoutManager( |                         layoutManager = | ||||||
|  |                             GridLayoutManager( | ||||||
|                                 this, |                                 this, | ||||||
|                             calculateNoOfColumns() |                                 calculateNoOfColumns(), | ||||||
|                             ) |                             ) | ||||||
|                         binding.recyclerView.layoutManager = layoutManager |                         binding.recyclerView.layoutManager = layoutManager | ||||||
|                     } else { |                     } else { | ||||||
|                         layoutManager = StaggeredGridLayoutManager( |                         layoutManager = | ||||||
|  |                             StaggeredGridLayoutManager( | ||||||
|                                 calculateNoOfColumns(), |                                 calculateNoOfColumns(), | ||||||
|                             StaggeredGridLayoutManager.VERTICAL |                                 StaggeredGridLayoutManager.VERTICAL, | ||||||
|                             ) |                             ) | ||||||
|                         layoutManager.gapStrategy = |                         layoutManager.gapStrategy = | ||||||
|                             StaggeredGridLayoutManager.GAP_HANDLING_MOVE_ITEMS_BETWEEN_SPANS |                             StaggeredGridLayoutManager.GAP_HANDLING_MOVE_ITEMS_BETWEEN_SPANS | ||||||
| @@ -336,11 +359,11 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleBottomBarActions() { |     private fun handleBottomBarActions() { | ||||||
|         binding.bottomBar.setTabSelectedListener(object : BottomNavigationBar.OnTabSelectedListener { |         binding.bottomBar.setTabSelectedListener( | ||||||
|  |             object : BottomNavigationBar.OnTabSelectedListener { | ||||||
|                 override fun onTabUnselected(position: Int) = Unit |                 override fun onTabUnselected(position: Int) = Unit | ||||||
|  |  | ||||||
|                 override fun onTabReselected(position: Int) { |                 override fun onTabReselected(position: Int) { | ||||||
|  |  | ||||||
|                     when (val layoutManager = binding.recyclerView.adapter) { |                     when (val layoutManager = binding.recyclerView.adapter) { | ||||||
|                         is StaggeredGridLayoutManager -> |                         is StaggeredGridLayoutManager -> | ||||||
|                             if (layoutManager.findFirstCompletelyVisibleItemPositions(null)[0] == 0) { |                             if (layoutManager.findFirstCompletelyVisibleItemPositions(null)[0] == 0) { | ||||||
| @@ -348,12 +371,14 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|                             } else { |                             } else { | ||||||
|                                 layoutManager.scrollToPositionWithOffset(0, 0) |                                 layoutManager.scrollToPositionWithOffset(0, 0) | ||||||
|                             } |                             } | ||||||
|  |  | ||||||
|                         is GridLayoutManager -> |                         is GridLayoutManager -> | ||||||
|                             if (layoutManager.findFirstCompletelyVisibleItemPosition() == 0) { |                             if (layoutManager.findFirstCompletelyVisibleItemPosition() == 0) { | ||||||
|                                 getElementsAccordingToTab() |                                 getElementsAccordingToTab() | ||||||
|                             } else { |                             } else { | ||||||
|                                 layoutManager.scrollToPositionWithOffset(0, 0) |                                 layoutManager.scrollToPositionWithOffset(0, 0) | ||||||
|                             } |                             } | ||||||
|  |  | ||||||
|                         else -> Unit |                         else -> Unit | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
| @@ -368,7 +393,8 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|                     fetchOnEmptyList() |                     fetchOnEmptyList() | ||||||
|                 } |                 } | ||||||
|         }) |             }, | ||||||
|  |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun fetchOnEmptyList() { |     fun fetchOnEmptyList() { | ||||||
| @@ -378,8 +404,13 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleInfiniteScroll() { |     private fun handleInfiniteScroll() { | ||||||
|         recyclerViewScrollListener = object : RecyclerView.OnScrollListener() { |         recyclerViewScrollListener = | ||||||
|             override fun onScrolled(localRecycler: RecyclerView, dx: Int, dy: Int) { |             object : RecyclerView.OnScrollListener() { | ||||||
|  |                 override fun onScrolled( | ||||||
|  |                     localRecycler: RecyclerView, | ||||||
|  |                     dx: Int, | ||||||
|  |                     dy: Int, | ||||||
|  |                 ) { | ||||||
|                     if (dy > 0) { |                     if (dy > 0) { | ||||||
|                         val lastVisibleItem = getLastVisibleItem() |                         val lastVisibleItem = getLastVisibleItem() | ||||||
|  |  | ||||||
| @@ -394,11 +425,13 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         binding.recyclerView.addOnScrollListener(recyclerViewScrollListener) |         binding.recyclerView.addOnScrollListener(recyclerViewScrollListener) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun getLastVisibleItem() : Int { |     private fun getLastVisibleItem(): Int { | ||||||
|         return when (val manager = binding.recyclerView.layoutManager) { |         return when (val manager = binding.recyclerView.layoutManager) { | ||||||
|             is StaggeredGridLayoutManager -> manager.findLastCompletelyVisibleItemPositions( |             is StaggeredGridLayoutManager -> | ||||||
|                 null |                 manager.findLastCompletelyVisibleItemPositions( | ||||||
|  |                     null, | ||||||
|                 ).last() |                 ).last() | ||||||
|  |  | ||||||
|             is GridLayoutManager -> manager.findLastCompletelyVisibleItemPosition() |             is GridLayoutManager -> manager.findLastCompletelyVisibleItemPosition() | ||||||
|             else -> 0 |             else -> 0 | ||||||
|         } |         } | ||||||
| @@ -411,10 +444,9 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|             binding.emptyText.visibility = View.GONE |             binding.emptyText.visibility = View.GONE | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     fun getElementsAccordingToTab( |     fun getElementsAccordingToTab(appendResults: Boolean = false) { | ||||||
|         appendResults: Boolean = false |         offset = | ||||||
|     ) { |             if (appendResults && items.size > 0) { | ||||||
|         offset = if (appendResults && items.size > 0) { |  | ||||||
|                 items.size - 1 |                 items.size - 1 | ||||||
|             } else { |             } else { | ||||||
|                 0 |                 0 | ||||||
| @@ -424,28 +456,37 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         getItems(appendResults, elementsShown) |         getItems(appendResults, elementsShown) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun getItems(appendResults: Boolean, itemType: ItemType) { |     private fun getItems( | ||||||
|  |         appendResults: Boolean, | ||||||
|  |         itemType: ItemType, | ||||||
|  |     ) { | ||||||
|  |         CountingIdlingResourceSingleton.increment() | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|             binding.swipeRefreshLayout.isRefreshing = true |             binding.swipeRefreshLayout.isRefreshing = true | ||||||
|             repository.displayedItems = itemType |             repository.displayedItems = itemType | ||||||
|             items = if (appendResults) { |             items = | ||||||
|  |                 if (appendResults) { | ||||||
|                     repository.getOlderItems() |                     repository.getOlderItems() | ||||||
|                 } else { |                 } else { | ||||||
|                     repository.getNewerItems() |                     repository.getNewerItems() | ||||||
|                 } |                 } | ||||||
|             binding.swipeRefreshLayout.isRefreshing = false |             binding.swipeRefreshLayout.isRefreshing = false | ||||||
|             handleListResult() |             handleListResult() | ||||||
|  |             CountingIdlingResourceSingleton.decrement() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleListResult(appendResults: Boolean = false) { |     private fun handleListResult(appendResults: Boolean = false) { | ||||||
|         if (appendResults) { |         if (appendResults) { | ||||||
|             val oldManager = binding.recyclerView.layoutManager |             val oldManager = binding.recyclerView.layoutManager | ||||||
|             firstVisible = when (oldManager) { |             firstVisible = | ||||||
|  |                 when (oldManager) { | ||||||
|                     is StaggeredGridLayoutManager -> |                     is StaggeredGridLayoutManager -> | ||||||
|                         oldManager.findFirstCompletelyVisibleItemPositions(null).last() |                         oldManager.findFirstCompletelyVisibleItemPositions(null).last() | ||||||
|  |  | ||||||
|                     is GridLayoutManager -> |                     is GridLayoutManager -> | ||||||
|                         oldManager.findFirstCompletelyVisibleItemPosition() |                         oldManager.findFirstCompletelyVisibleItemPosition() | ||||||
|  |  | ||||||
|                     else -> 0 |                     else -> 0 | ||||||
|                 } |                 } | ||||||
|         } |         } | ||||||
| @@ -471,13 +512,13 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|                 binding.recyclerView.addItemDecoration( |                 binding.recyclerView.addItemDecoration( | ||||||
|                     DividerItemDecoration( |                     DividerItemDecoration( | ||||||
|                         this@HomeActivity, |                         this@HomeActivity, | ||||||
|                         DividerItemDecoration.VERTICAL |                         DividerItemDecoration.VERTICAL, | ||||||
|                     ) |                     ), | ||||||
|                 ) |                 ) | ||||||
|             } |             } | ||||||
|             binding.recyclerView.adapter = recyclerAdapter |             binding.recyclerView.adapter = recyclerAdapter | ||||||
|         } else { |         } else { | ||||||
|                 (recyclerAdapter as ItemsAdapter<*>).updateAllItems(items) |             recyclerAdapter!!.updateAllItems(items) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         reloadBadges() |         reloadBadges() | ||||||
| @@ -486,8 +527,10 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|     private fun reloadBadges() { |     private fun reloadBadges() { | ||||||
|         if (appSettingsService.isDisplayUnreadCountEnabled() || appSettingsService.isDisplayAllCountEnabled()) { |         if (appSettingsService.isDisplayUnreadCountEnabled() || appSettingsService.isDisplayAllCountEnabled()) { | ||||||
|  |             CountingIdlingResourceSingleton.increment() | ||||||
|             CoroutineScope(Dispatchers.IO).launch { |             CoroutineScope(Dispatchers.IO).launch { | ||||||
|                 repository.reloadBadges() |                 repository.reloadBadges() | ||||||
|  |                 CountingIdlingResourceSingleton.decrement() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -523,13 +566,17 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         val searchItem = menu.findItem(R.id.action_search) |         val searchItem = menu.findItem(R.id.action_search) | ||||||
|         val searchView = searchItem.getActionView() as SearchView |         val searchView = searchItem.actionView as SearchView | ||||||
|         searchView.setOnQueryTextListener(this) |         searchView.setOnQueryTextListener(this) | ||||||
|  |  | ||||||
|         return true |         return true | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun needsConfirmation(titleRes: Int, messageRes: Int, doFn: () -> Unit) { |     private fun needsConfirmation( | ||||||
|  |         titleRes: Int, | ||||||
|  |         messageRes: Int, | ||||||
|  |         doFn: () -> Unit, | ||||||
|  |     ) { | ||||||
|         AlertDialog.Builder(this@HomeActivity) |         AlertDialog.Builder(this@HomeActivity) | ||||||
|             .setMessage(messageRes) |             .setMessage(messageRes) | ||||||
|             .setTitle(titleRes) |             .setTitle(titleRes) | ||||||
| @@ -541,45 +588,57 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|         when (item.itemId) { |         when (item.itemId) { | ||||||
|  |             R.id.issue_tracker -> { | ||||||
|  |                 val browserIntent = | ||||||
|  |                     Intent(Intent.ACTION_VIEW, Uri.parse(AppSettingsService.trackerUrl)) | ||||||
|  |                 startActivity(browserIntent) | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |  | ||||||
|             R.id.action_filter -> { |             R.id.action_filter -> { | ||||||
|                 val filterSheetFragment = FilterSheetFragment() |                 val filterSheetFragment = FilterSheetFragment() | ||||||
|                 filterSheetFragment.show(supportFragmentManager, FilterSheetFragment.TAG) |                 filterSheetFragment.show(supportFragmentManager, FilterSheetFragment.TAG) | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             R.id.refresh -> { |             R.id.refresh -> { | ||||||
|                 needsConfirmation(R.string.menu_home_refresh, R.string.refresh_dialog_message) { |                 needsConfirmation(R.string.menu_home_refresh, R.string.refresh_dialog_message) { | ||||||
|                     Toast.makeText(this, R.string.refresh_in_progress, Toast.LENGTH_SHORT).show() |                     Toast.makeText(this, R.string.refresh_in_progress, Toast.LENGTH_SHORT).show() | ||||||
|  |                     CountingIdlingResourceSingleton.increment() | ||||||
|                     CoroutineScope(Dispatchers.Main).launch { |                     CoroutineScope(Dispatchers.Main).launch { | ||||||
|                         val updatedRemote = repository.updateRemote() |                         val updatedRemote = repository.updateRemote() | ||||||
|                         if (updatedRemote) { |                         if (updatedRemote) { | ||||||
|                             Toast.makeText( |                             Toast.makeText( | ||||||
|                                 this@HomeActivity, |                                 this@HomeActivity, | ||||||
|                                 R.string.refresh_success_response, Toast.LENGTH_LONG |                                 R.string.refresh_success_response, | ||||||
|  |                                 Toast.LENGTH_LONG, | ||||||
|                             ) |                             ) | ||||||
|                                 .show() |                                 .show() | ||||||
|                         } else { |                         } else { | ||||||
|                             Toast.makeText( |                             Toast.makeText( | ||||||
|                                 this@HomeActivity, |                                 this@HomeActivity, | ||||||
|                                 R.string.refresh_failer_message, |                                 R.string.refresh_failer_message, | ||||||
|                                 Toast.LENGTH_SHORT |                                 Toast.LENGTH_SHORT, | ||||||
|                             ).show() |                             ).show() | ||||||
|                         } |                         } | ||||||
|  |                         CountingIdlingResourceSingleton.decrement() | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             R.id.readAll -> { |             R.id.readAll -> { | ||||||
|                 if (elementsShown == ItemType.UNREAD) { |                 if (elementsShown == ItemType.UNREAD) { | ||||||
|                     needsConfirmation(R.string.readAll, R.string.markall_dialog_message) { |                     needsConfirmation(R.string.readAll, R.string.markall_dialog_message) { | ||||||
|                         binding.swipeRefreshLayout.isRefreshing = true |                         binding.swipeRefreshLayout.isRefreshing = true | ||||||
|  |                         CountingIdlingResourceSingleton.increment() | ||||||
|                         CoroutineScope(Dispatchers.Main).launch { |                         CoroutineScope(Dispatchers.Main).launch { | ||||||
|                             val success = repository.markAllAsRead(items) |                             val success = repository.markAllAsRead(items) | ||||||
|                             if (success) { |                             if (success) { | ||||||
|                                 Toast.makeText( |                                 Toast.makeText( | ||||||
|                                     this@HomeActivity, |                                     this@HomeActivity, | ||||||
|                                     R.string.all_posts_read, |                                     R.string.all_posts_read, | ||||||
|                                     Toast.LENGTH_SHORT |                                     Toast.LENGTH_SHORT, | ||||||
|                                 ).show() |                                 ).show() | ||||||
|                                 tabNewBadge.removeBadge() |                                 tabNewBadge.removeBadge() | ||||||
|  |  | ||||||
| @@ -588,33 +647,44 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|                                 Toast.makeText( |                                 Toast.makeText( | ||||||
|                                     this@HomeActivity, |                                     this@HomeActivity, | ||||||
|                                     R.string.all_posts_not_read, |                                     R.string.all_posts_not_read, | ||||||
|                                     Toast.LENGTH_SHORT |                                     Toast.LENGTH_SHORT, | ||||||
|                                 ).show() |                                 ).show() | ||||||
|                             } |                             } | ||||||
|                             handleListResult() |                             handleListResult() | ||||||
|                             binding.swipeRefreshLayout.isRefreshing = false |                             binding.swipeRefreshLayout.isRefreshing = false | ||||||
|  |                             CountingIdlingResourceSingleton.decrement() | ||||||
|  |  | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             R.id.action_disconnect -> { |             R.id.action_disconnect -> { | ||||||
|  |                 needsConfirmation( | ||||||
|  |                     R.string.confirm_disconnect_title, | ||||||
|  |                     R.string.confirm_disconnect_description | ||||||
|  |                 ) { | ||||||
|                     runBlocking { |                     runBlocking { | ||||||
|                         repository.logout() |                         repository.logout() | ||||||
|                     } |                     } | ||||||
|                     val intent = Intent(this, LoginActivity::class.java) |                     val intent = Intent(this, LoginActivity::class.java) | ||||||
|                     this.startActivity(intent) |                     this.startActivity(intent) | ||||||
|                     finish() |                     finish() | ||||||
|  |                 } | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             R.id.action_settings -> { |             R.id.action_settings -> { | ||||||
|                 settingsLauncher.launch(Intent(this, SettingsActivity::class.java)) |                 settingsLauncher.launch(Intent(this, SettingsActivity::class.java)) | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             R.id.action_sources -> { |             R.id.action_sources -> { | ||||||
|                 startActivity(Intent(this, SourcesActivity::class.java)) |                 startActivity(Intent(this, SourcesActivity::class.java)) | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             else -> return super.onOptionsItemSelected(item) |             else -> return super.onOptionsItemSelected(item) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -633,20 +703,29 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|     private fun handleRecurringTask() { |     private fun handleRecurringTask() { | ||||||
|         if (appSettingsService.isPeriodicRefreshEnabled()) { |         if (appSettingsService.isPeriodicRefreshEnabled()) { | ||||||
|             val myConstraints = Constraints.Builder() |             val myConstraints = | ||||||
|  |                 Constraints.Builder() | ||||||
|                     .setRequiresBatteryNotLow(true) |                     .setRequiresBatteryNotLow(true) | ||||||
|                     .setRequiresCharging(appSettingsService.isRefreshWhenChargingOnlyEnabled()) |                     .setRequiresCharging(appSettingsService.isRefreshWhenChargingOnlyEnabled()) | ||||||
|                     .setRequiresStorageNotLow(true) |                     .setRequiresStorageNotLow(true) | ||||||
|                     .build() |                     .build() | ||||||
|  |  | ||||||
|             val backgroundWork = |             val backgroundWork = | ||||||
|                 PeriodicWorkRequestBuilder<LoadingWorker>(appSettingsService.getRefreshMinutes(), TimeUnit.MINUTES) |                 PeriodicWorkRequestBuilder<LoadingWorker>( | ||||||
|  |                     appSettingsService.getRefreshMinutes(), | ||||||
|  |                     TimeUnit.MINUTES | ||||||
|  |                 ) | ||||||
|                     .setConstraints(myConstraints) |                     .setConstraints(myConstraints) | ||||||
|                     .addTag("selfoss-loading") |                     .addTag("selfoss-loading") | ||||||
|                     .build() |                     .build() | ||||||
|  |  | ||||||
|             WorkManager.getInstance(baseContext).enqueueUniquePeriodicWork("selfoss-loading", ExistingPeriodicWorkPolicy.KEEP, backgroundWork) |             WorkManager.getInstance( | ||||||
|  |                 baseContext, | ||||||
|  |             ).enqueueUniquePeriodicWork( | ||||||
|  |                 "selfoss-loading", | ||||||
|  |                 ExistingPeriodicWorkPolicy.KEEP, | ||||||
|  |                 backgroundWork | ||||||
|  |             ) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,8 +11,8 @@ import bou.amine.apps.readerforselfossv2.android.databinding.ActivityImageBindin | |||||||
| import bou.amine.apps.readerforselfossv2.android.fragments.ImageFragment | import bou.amine.apps.readerforselfossv2.android.fragments.ImageFragment | ||||||
|  |  | ||||||
| class ImageActivity : AppCompatActivity() { | class ImageActivity : AppCompatActivity() { | ||||||
|     private lateinit var allImages : ArrayList<String> |     private lateinit var allImages: ArrayList<String> | ||||||
|     private var position : Int = 0 |     private var position: Int = 0 | ||||||
|  |  | ||||||
|     private lateinit var binding: ActivityImageBinding |     private lateinit var binding: ActivityImageBinding | ||||||
|  |  | ||||||
| @@ -32,24 +32,41 @@ class ImageActivity : AppCompatActivity() { | |||||||
|         binding.pager.adapter = ScreenSlidePagerAdapter(this) |         binding.pager.adapter = ScreenSlidePagerAdapter(this) | ||||||
|         binding.pager.setCurrentItem(position, false) |         binding.pager.setCurrentItem(position, false) | ||||||
|  |  | ||||||
|         val transitionListener = object : MotionLayout.TransitionListener { |         val transitionListener = | ||||||
|             override fun onTransitionStarted(motionLayout: MotionLayout?, startId: Int, endId: Int) { |             object : MotionLayout.TransitionListener { | ||||||
|                 // Nothing |                 override fun onTransitionStarted( | ||||||
|             } |                     motionLayout: MotionLayout?, | ||||||
|  |                     startId: Int, | ||||||
|             override fun onTransitionChange(motionLayout: MotionLayout?, startId: Int, endId: Int, progress: Float |                     endId: Int, | ||||||
|                 ) { |                 ) { | ||||||
|                     // Nothing |                     // Nothing | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|             override fun onTransitionCompleted(motionLayout: MotionLayout?, currentId: Int) { |                 override fun onTransitionChange( | ||||||
|  |                     motionLayout: MotionLayout?, | ||||||
|  |                     startId: Int, | ||||||
|  |                     endId: Int, | ||||||
|  |                     progress: Float, | ||||||
|  |                 ) { | ||||||
|  |                     // Nothing | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 override fun onTransitionCompleted( | ||||||
|  |                     motionLayout: MotionLayout?, | ||||||
|  |                     currentId: Int, | ||||||
|  |                 ) { | ||||||
|                     if (motionLayout?.currentState == binding.root.endState) { |                     if (motionLayout?.currentState == binding.root.endState) { | ||||||
|                         onBackPressedDispatcher.onBackPressed() |                         onBackPressedDispatcher.onBackPressed() | ||||||
|                         overridePendingTransition(0, 0) |                         overridePendingTransition(0, 0) | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|             override fun onTransitionTrigger(motionLayout: MotionLayout?, triggerId: Int, positive: Boolean, progress: Float) { |                 override fun onTransitionTrigger( | ||||||
|  |                     motionLayout: MotionLayout?, | ||||||
|  |                     triggerId: Int, | ||||||
|  |                     positive: Boolean, | ||||||
|  |                     progress: Float, | ||||||
|  |                 ) { | ||||||
|                     // Nothing |                     // Nothing | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -68,7 +85,6 @@ class ImageActivity : AppCompatActivity() { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : FragmentStateAdapter(fa) { |     private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : FragmentStateAdapter(fa) { | ||||||
|  |  | ||||||
|         override fun getItemCount(): Int = allImages.size |         override fun getItemCount(): Int = allImages.size | ||||||
|  |  | ||||||
|         override fun createFragment(position: Int): Fragment = ImageFragment.newInstance(allImages[position]) |         override fun createFragment(position: Int): Fragment = ImageFragment.newInstance(allImages[position]) | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import android.animation.Animator | |||||||
| import android.animation.AnimatorListenerAdapter | import android.animation.AnimatorListenerAdapter | ||||||
| import android.annotation.SuppressLint | import android.annotation.SuppressLint | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
|  | import android.net.Uri | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.text.TextUtils | import android.text.TextUtils | ||||||
| import android.view.Menu | import android.view.Menu | ||||||
| @@ -16,6 +17,7 @@ import androidx.appcompat.app.AlertDialog | |||||||
| import androidx.appcompat.app.AppCompatActivity | import androidx.appcompat.app.AppCompatActivity | ||||||
| import androidx.appcompat.app.AppCompatDelegate | import androidx.appcompat.app.AppCompatDelegate | ||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityLoginBinding | import bou.amine.apps.readerforselfossv2.android.databinding.ActivityLoginBinding | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.testing.CountingIdlingResourceSingleton | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlInvalid | import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlInvalid | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| @@ -28,9 +30,7 @@ import org.kodein.di.DIAware | |||||||
| import org.kodein.di.android.closestDI | import org.kodein.di.android.closestDI | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|  |  | ||||||
|  |  | ||||||
| class LoginActivity : AppCompatActivity(), DIAware { | class LoginActivity : AppCompatActivity(), DIAware { | ||||||
|  |  | ||||||
|     private var inValidCount: Int = 0 |     private var inValidCount: Int = 0 | ||||||
|     private var isWithLogin = false |     private var isWithLogin = false | ||||||
|  |  | ||||||
| @@ -40,7 +40,6 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|     private val repository: Repository by instance() |     private val repository: Repository by instance() | ||||||
|     private val appSettingsService: AppSettingsService by instance() |     private val appSettingsService: AppSettingsService by instance() | ||||||
|  |  | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|  |  | ||||||
| @@ -57,29 +56,7 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|         if (appSettingsService.getBaseUrl().isNotEmpty()) { |         if (appSettingsService.getBaseUrl().isNotEmpty()) { | ||||||
|             showProgress(true) |             showProgress(true) | ||||||
|             // This should be reverted when "old" users connected with a non-selfoss rss |  | ||||||
|             // are handled. Revert to "simple" way. |  | ||||||
|             CoroutineScope(Dispatchers.Main).launch { |  | ||||||
|                 try { |  | ||||||
|                     val (errorFetching, displaySelfossOnly) = repository.shouldBeSelfossInstance() |  | ||||||
|                     if (!errorFetching && !displaySelfossOnly) { |  | ||||||
|             goToMain() |             goToMain() | ||||||
|                     } else { |  | ||||||
|                         showProgress(false) |  | ||||||
|                         if (displaySelfossOnly) { |  | ||||||
|                             Toast.makeText( |  | ||||||
|                                 applicationContext, |  | ||||||
|                                 R.string.application_selfoss_only, |  | ||||||
|                                 Toast.LENGTH_LONG |  | ||||||
|                             ).show() |  | ||||||
|                         } |  | ||||||
|                         repository.logout() |  | ||||||
|                     } |  | ||||||
|                 } catch (e: Throwable) { |  | ||||||
|                     repository.logout() |  | ||||||
|                     showProgress(false) |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         handleActions() |         handleActions() | ||||||
| @@ -91,7 +68,6 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleActions() { |     private fun handleActions() { | ||||||
|  |  | ||||||
|         binding.passwordView.setOnEditorActionListener( |         binding.passwordView.setOnEditorActionListener( | ||||||
|             TextView.OnEditorActionListener { _, id, _ -> |             TextView.OnEditorActionListener { _, id, _ -> | ||||||
|                 if (id == R.id.loginView || id == EditorInfo.IME_NULL) { |                 if (id == R.id.loginView || id == EditorInfo.IME_NULL) { | ||||||
| @@ -99,7 +75,7 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|                     return@OnEditorActionListener true |                     return@OnEditorActionListener true | ||||||
|                 } |                 } | ||||||
|                 false |                 false | ||||||
|             } |             }, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         binding.signInButton.setOnClickListener { attemptLogin() } |         binding.signInButton.setOnClickListener { attemptLogin() } | ||||||
| @@ -120,16 +96,21 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|             alertDialog.setMessage(getString(R.string.base_url_error)) |             alertDialog.setMessage(getString(R.string.base_url_error)) | ||||||
|             alertDialog.setButton( |             alertDialog.setButton( | ||||||
|                 AlertDialog.BUTTON_NEUTRAL, |                 AlertDialog.BUTTON_NEUTRAL, | ||||||
|                 "OK" |                 "OK", | ||||||
|             ) { dialog, _ -> dialog.dismiss() } |             ) { dialog, _ -> dialog.dismiss() } | ||||||
|             alertDialog.show() |             alertDialog.show() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun goToMain() { |     private fun goToMain() { | ||||||
|  |         CountingIdlingResourceSingleton.increment() | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|             repository.updateApiInformation() |             repository.updateApiInformation() | ||||||
|             ACRA.errorReporter.putCustomData("SELFOSS_API_VERSION", appSettingsService.getApiVersion().toString()) |             ACRA.errorReporter.putCustomData( | ||||||
|  |                 "SELFOSS_API_VERSION", | ||||||
|  |                 appSettingsService.getApiVersion().toString() | ||||||
|  |             ) | ||||||
|  |             CountingIdlingResourceSingleton.decrement() | ||||||
|         } |         } | ||||||
|         val intent = Intent(this, HomeActivity::class.java) |         val intent = Intent(this, HomeActivity::class.java) | ||||||
|         startActivity(intent) |         startActivity(intent) | ||||||
| @@ -145,7 +126,6 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun attemptLogin() { |     private fun attemptLogin() { | ||||||
|  |  | ||||||
|         // Reset errors. |         // Reset errors. | ||||||
|         binding.urlView.error = null |         binding.urlView.error = null | ||||||
|         binding.loginView.error = null |         binding.loginView.error = null | ||||||
| @@ -161,34 +141,44 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|         showProgress(true) |         showProgress(true) | ||||||
|  |  | ||||||
|  |         appSettingsService.updateSelfSigned(binding.selfSigned.isChecked) | ||||||
|  |  | ||||||
|         repository.refreshLoginInformation(url, login, password) |         repository.refreshLoginInformation(url, login, password) | ||||||
|  |  | ||||||
|  |         CountingIdlingResourceSingleton.increment() | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|             val result = repository.login() |             try { | ||||||
|             if (result) { |                 repository.updateApiInformation() | ||||||
|                 val (errorFetching, displaySelfossOnly) = repository.shouldBeSelfossInstance() |             } catch (e: Exception) { | ||||||
|                 if (!errorFetching && !displaySelfossOnly) { |                 if (e.message?.startsWith("No transformation found") == true) { | ||||||
|                     goToMain() |  | ||||||
|                 } else { |  | ||||||
|                     if (displaySelfossOnly) { |  | ||||||
|                     Toast.makeText( |                     Toast.makeText( | ||||||
|                         applicationContext, |                         applicationContext, | ||||||
|                         R.string.application_selfoss_only, |                         R.string.application_selfoss_only, | ||||||
|                             Toast.LENGTH_LONG |                         Toast.LENGTH_LONG, | ||||||
|                     ).show() |                     ).show() | ||||||
|  |                     preferenceError() | ||||||
|  |                     showProgress(false) | ||||||
|                 } |                 } | ||||||
|  |             } | ||||||
|  |             val result = repository.login() | ||||||
|  |             if (result) { | ||||||
|  |                 val errorFetching = repository.checkIfFetchFails() | ||||||
|  |                 if (!errorFetching) { | ||||||
|  |                     goToMain() | ||||||
|  |                 } else { | ||||||
|                     preferenceError() |                     preferenceError() | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|                 preferenceError() |                 preferenceError() | ||||||
|             } |             } | ||||||
|             showProgress(false) |             showProgress(false) | ||||||
|  |             CountingIdlingResourceSingleton.decrement() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun failLoginDetails( |     private fun failLoginDetails( | ||||||
|         password: String, |         password: String, | ||||||
|         login: String |         login: String, | ||||||
|     ) { |     ) { | ||||||
|         var lastFocusedView: View? = null |         var lastFocusedView: View? = null | ||||||
|         var cancel = false |         var cancel = false | ||||||
| @@ -221,7 +211,7 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|                 alertDialog.setMessage(getString(R.string.text_wrong_url)) |                 alertDialog.setMessage(getString(R.string.text_wrong_url)) | ||||||
|                 alertDialog.setButton( |                 alertDialog.setButton( | ||||||
|                     AlertDialog.BUTTON_NEUTRAL, |                     AlertDialog.BUTTON_NEUTRAL, | ||||||
|                     "OK" |                     "OK", | ||||||
|                 ) { dialog, _ -> dialog.dismiss() } |                 ) { dialog, _ -> dialog.dismiss() } | ||||||
|                 alertDialog.show() |                 alertDialog.show() | ||||||
|                 inValidCount = 0 |                 inValidCount = 0 | ||||||
| @@ -230,7 +220,10 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|         maybeCancelAndFocusView(cancel, focusView) |         maybeCancelAndFocusView(cancel, focusView) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun maybeCancelAndFocusView(cancel: Boolean, focusView: View?) { |     private fun maybeCancelAndFocusView( | ||||||
|  |         cancel: Boolean, | ||||||
|  |         focusView: View?, | ||||||
|  |     ) { | ||||||
|         if (cancel) { |         if (cancel) { | ||||||
|             focusView?.requestFocus() |             focusView?.requestFocus() | ||||||
|         } |         } | ||||||
| @@ -244,12 +237,13 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|             .animate() |             .animate() | ||||||
|             .setDuration(shortAnimTime.toLong()) |             .setDuration(shortAnimTime.toLong()) | ||||||
|             .alpha( |             .alpha( | ||||||
|                 if (show) 0F else 1F |                 if (show) 0F else 1F, | ||||||
|             ).setListener(object : AnimatorListenerAdapter() { |             ).setListener( | ||||||
|  |                 object : AnimatorListenerAdapter() { | ||||||
|                     override fun onAnimationEnd(animation: Animator) { |                     override fun onAnimationEnd(animation: Animator) { | ||||||
|                         binding.loginForm.visibility = if (show) View.GONE else View.VISIBLE |                         binding.loginForm.visibility = if (show) View.GONE else View.VISIBLE | ||||||
|                     } |                     } | ||||||
|             } |                 }, | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         binding.loginProgress.visibility = if (show) View.VISIBLE else View.GONE |         binding.loginProgress.visibility = if (show) View.VISIBLE else View.GONE | ||||||
| @@ -257,12 +251,13 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|             .animate() |             .animate() | ||||||
|             .setDuration(shortAnimTime.toLong()) |             .setDuration(shortAnimTime.toLong()) | ||||||
|             .alpha( |             .alpha( | ||||||
|                 if (show) 1F else 0F |                 if (show) 1F else 0F, | ||||||
|             ).setListener(object : AnimatorListenerAdapter() { |             ).setListener( | ||||||
|  |                 object : AnimatorListenerAdapter() { | ||||||
|                     override fun onAnimationEnd(animation: Animator) { |                     override fun onAnimationEnd(animation: Animator) { | ||||||
|                         binding.loginProgress.visibility = if (show) View.VISIBLE else View.GONE |                         binding.loginProgress.visibility = if (show) View.VISIBLE else View.GONE | ||||||
|                     } |                     } | ||||||
|             } |                 }, | ||||||
|             ) |             ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -273,13 +268,25 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|         return when (item.itemId) { |         return when (item.itemId) { | ||||||
|  |             R.id.issue_tracker -> { | ||||||
|  |                 val browserIntent = | ||||||
|  |                     Intent(Intent.ACTION_VIEW, Uri.parse(AppSettingsService.trackerUrl)) | ||||||
|  |                 startActivity(browserIntent) | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |  | ||||||
|             R.id.about -> { |             R.id.about -> { | ||||||
|                 LibsBuilder() |                 LibsBuilder() | ||||||
|                     .withAboutIconShown(true) |                     .withAboutIconShown(true) | ||||||
|                     .withAboutVersionShown(true) |                     .withAboutVersionShown(true) | ||||||
|  |                     .withAboutSpecial2("Bug reports") | ||||||
|  |                     .withAboutSpecial2Description(AppSettingsService.trackerUrl) | ||||||
|  |                     .withAboutSpecial1("Project Page") | ||||||
|  |                     .withAboutSpecial1Description(AppSettingsService.sourceUrl) | ||||||
|                     .start(this) |                     .start(this) | ||||||
|                 true |                 true | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             else -> super.onOptionsItemSelected(item) |             else -> super.onOptionsItemSelected(item) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -8,7 +8,6 @@ import bou.amine.apps.readerforselfossv2.android.databinding.ActivityMainBinding | |||||||
| class MainActivity : AppCompatActivity() { | class MainActivity : AppCompatActivity() { | ||||||
|     private lateinit var binding: ActivityMainBinding |     private lateinit var binding: ActivityMainBinding | ||||||
|  |  | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|         binding = ActivityMainBinding.inflate(layoutInflater) |         binding = ActivityMainBinding.inflate(layoutInflater) | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ import androidx.lifecycle.LifecycleOwner | |||||||
| import androidx.lifecycle.ProcessLifecycleOwner | import androidx.lifecycle.ProcessLifecycleOwner | ||||||
| import androidx.multidex.MultiDexApplication | import androidx.multidex.MultiDexApplication | ||||||
| import bou.amine.apps.readerforselfossv2.DI.networkModule | import bou.amine.apps.readerforselfossv2.DI.networkModule | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.testing.TestingHelper | ||||||
| import bou.amine.apps.readerforselfossv2.android.viewmodel.AppViewModel | import bou.amine.apps.readerforselfossv2.android.viewmodel.AppViewModel | ||||||
| import bou.amine.apps.readerforselfossv2.dao.DriverFactory | import bou.amine.apps.readerforselfossv2.dao.DriverFactory | ||||||
| import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB | import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB | ||||||
| @@ -29,21 +30,25 @@ import org.acra.config.toast | |||||||
| import org.acra.data.StringFormat | import org.acra.data.StringFormat | ||||||
| import org.acra.ktx.initAcra | import org.acra.ktx.initAcra | ||||||
| import org.acra.sender.HttpSender | import org.acra.sender.HttpSender | ||||||
| import org.kodein.di.* | import org.kodein.di.DI | ||||||
|  | import org.kodein.di.DIAware | ||||||
|  | import org.kodein.di.bind | ||||||
|  | import org.kodein.di.instance | ||||||
|  | import org.kodein.di.singleton | ||||||
|  |  | ||||||
| class MyApp : MultiDexApplication(), DIAware { | class MyApp : MultiDexApplication(), DIAware { | ||||||
|  |  | ||||||
|     override val di by DI.lazy { |     override val di by DI.lazy { | ||||||
|         bind<AppSettingsService>() with singleton { AppSettingsService(ACRA.isACRASenderServiceProcess()) } |         bind<AppSettingsService>() with singleton { AppSettingsService(ACRA.isACRASenderServiceProcess() || TestingHelper().isUnitTest()) } | ||||||
|         import(networkModule) |         import(networkModule) | ||||||
|         bind<DriverFactory>() with singleton { DriverFactory(applicationContext) } |         bind<DriverFactory>() with singleton { DriverFactory(applicationContext) } | ||||||
|         bind<ReaderForSelfossDB>() with singleton { ReaderForSelfossDB(driverFactory.createDriver()) } |         bind<ReaderForSelfossDB>() with singleton { ReaderForSelfossDB(driverFactory.createDriver()) } | ||||||
|         bind<Repository>() with singleton { |         bind<Repository>() with | ||||||
|  |                 singleton { | ||||||
|                     Repository( |                     Repository( | ||||||
|                         instance(), |                         instance(), | ||||||
|                         instance(), |                         instance(), | ||||||
|                         isConnectionAvailable, |                         isConnectionAvailable, | ||||||
|                 instance() |                         instance(), | ||||||
|                     ) |                     ) | ||||||
|                 } |                 } | ||||||
|         bind<ConnectivityStatus>() with singleton { ConnectivityStatus(applicationContext) } |         bind<ConnectivityStatus>() with singleton { ConnectivityStatus(applicationContext) } | ||||||
| @@ -70,13 +75,14 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|             ProcessLifecycleOwner.get().lifecycle.addObserver( |             ProcessLifecycleOwner.get().lifecycle.addObserver( | ||||||
|                 AppLifeCycleObserver( |                 AppLifeCycleObserver( | ||||||
|                     connectivityStatus, |                     connectivityStatus, | ||||||
|                     repository |                     repository, | ||||||
|                 ) |                 ), | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|             CoroutineScope(Dispatchers.Main).launch { |             CoroutineScope(Dispatchers.Main).launch { | ||||||
|                 viewModel.networkAvailableProvider.collect { networkAvailable -> |                 viewModel.networkAvailableProvider.collect { networkAvailable -> | ||||||
|                     val toastMessage = if (networkAvailable) { |                     val toastMessage = | ||||||
|  |                         if (networkAvailable) { | ||||||
|                             repository.handleDBActions() |                             repository.handleDBActions() | ||||||
|                             R.string.network_connectivity_retrieved |                             R.string.network_connectivity_retrieved | ||||||
|                         } else { |                         } else { | ||||||
| @@ -86,7 +92,7 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|                     Toast.makeText( |                     Toast.makeText( | ||||||
|                         applicationContext, |                         applicationContext, | ||||||
|                         toastMessage, |                         toastMessage, | ||||||
|                         Toast.LENGTH_SHORT |                         Toast.LENGTH_SHORT, | ||||||
|                     ).show() |                     ).show() | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -100,7 +106,8 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|  |  | ||||||
|         initAcra { |         initAcra { | ||||||
|             reportFormat = StringFormat.JSON |             reportFormat = StringFormat.JSON | ||||||
|             reportContent = listOf( |             reportContent = | ||||||
|  |                 listOf( | ||||||
|                     ReportField.REPORT_ID, |                     ReportField.REPORT_ID, | ||||||
|                     ReportField.INSTALLATION_ID, |                     ReportField.INSTALLATION_ID, | ||||||
|                     ReportField.APP_VERSION_CODE, |                     ReportField.APP_VERSION_CODE, | ||||||
| @@ -121,16 +128,16 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|                     ReportField.USER_COMMENT, |                     ReportField.USER_COMMENT, | ||||||
|                     ReportField.USER_CRASH_DATE, |                     ReportField.USER_CRASH_DATE, | ||||||
|                     ReportField.USER_EMAIL, |                     ReportField.USER_EMAIL, | ||||||
|                 ReportField.CUSTOM_DATA |                     ReportField.CUSTOM_DATA, | ||||||
|                 ) |                 ) | ||||||
|             toast { |             toast { | ||||||
|                 //required |                 // required | ||||||
|                 text = getString(R.string.crash_toast_text) |                 text = getString(R.string.crash_toast_text) | ||||||
|                 length = Toast.LENGTH_SHORT |                 length = Toast.LENGTH_SHORT | ||||||
|             } |             } | ||||||
|             httpSender { |             httpSender { | ||||||
|                 uri = |                 uri = | ||||||
|                     "https://bugs.amine-louveau.fr/report" /*best guess, you may need to adjust this*/ |                     "https://bugs.amine-bouabdallaoui.fr/report" // best guess, you may need to adjust this | ||||||
|                 basicAuthLogin = "qMEscjj89Gwt6cPR" |                 basicAuthLogin = "qMEscjj89Gwt6cPR" | ||||||
|                 basicAuthPassword = "Yo58QFlGzFaWlBzP" |                 basicAuthPassword = "Yo58QFlGzFaWlBzP" | ||||||
|                 httpMethod = HttpSender.Method.POST |                 httpMethod = HttpSender.Method.POST | ||||||
| @@ -148,10 +155,11 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|  |  | ||||||
|             val newItemsChannelname = getString(R.string.new_items_channel_sync) |             val newItemsChannelname = getString(R.string.new_items_channel_sync) | ||||||
|             val newItemsChannelimportance = NotificationManager.IMPORTANCE_DEFAULT |             val newItemsChannelimportance = NotificationManager.IMPORTANCE_DEFAULT | ||||||
|             val newItemsChannelmChannel = NotificationChannel( |             val newItemsChannelmChannel = | ||||||
|  |                 NotificationChannel( | ||||||
|                     AppSettingsService.newItemsChannelId, |                     AppSettingsService.newItemsChannelId, | ||||||
|                     newItemsChannelname, |                     newItemsChannelname, | ||||||
|                 newItemsChannelimportance |                     newItemsChannelimportance, | ||||||
|                 ) |                 ) | ||||||
|  |  | ||||||
|             notificationManager.createNotificationChannel(mChannel) |             notificationManager.createNotificationChannel(mChannel) | ||||||
| @@ -163,9 +171,11 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|         val oldHandler = Thread.getDefaultUncaughtExceptionHandler() |         val oldHandler = Thread.getDefaultUncaughtExceptionHandler() | ||||||
|  |  | ||||||
|         Thread.setDefaultUncaughtExceptionHandler { thread, e -> |         Thread.setDefaultUncaughtExceptionHandler { thread, e -> | ||||||
|             if (e is NoClassDefFoundError && e.stackTrace.asList().any { |             if (e is NoClassDefFoundError && | ||||||
|  |                 e.stackTrace.asList().any { | ||||||
|                     it.toString().contains("android.view.ViewDebug") |                     it.toString().contains("android.view.ViewDebug") | ||||||
|                 }) { |                 } | ||||||
|  |             ) { | ||||||
|                 // Nothing |                 // Nothing | ||||||
|             } else { |             } else { | ||||||
|                 oldHandler.uncaughtException(thread, e) |                 oldHandler.uncaughtException(thread, e) | ||||||
| @@ -175,9 +185,8 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|  |  | ||||||
|     class AppLifeCycleObserver( |     class AppLifeCycleObserver( | ||||||
|         val connectivityStatus: ConnectivityStatus, |         val connectivityStatus: ConnectivityStatus, | ||||||
|         val repository: Repository |         val repository: Repository, | ||||||
|     ) : DefaultLifecycleObserver { |     ) : DefaultLifecycleObserver { | ||||||
|  |  | ||||||
|         override fun onResume(owner: LifecycleOwner) { |         override fun onResume(owner: LifecycleOwner) { | ||||||
|             super.onResume(owner) |             super.onResume(owner) | ||||||
|             repository.connectionMonitored = true |             repository.connectionMonitored = true | ||||||
|   | |||||||
| @@ -23,7 +23,6 @@ import org.kodein.di.android.closestDI | |||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|  |  | ||||||
| class ReaderActivity : AppCompatActivity(), DIAware { | class ReaderActivity : AppCompatActivity(), DIAware { | ||||||
|  |  | ||||||
|     private var currentItem: Int = 0 |     private var currentItem: Int = 0 | ||||||
|  |  | ||||||
|     private lateinit var toolbarMenu: Menu |     private lateinit var toolbarMenu: Menu | ||||||
| @@ -71,7 +70,11 @@ class ReaderActivity : AppCompatActivity(), DIAware { | |||||||
|             finish() |             finish() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         try { | ||||||
|             readItem(allItems[currentItem]) |             readItem(allItems[currentItem]) | ||||||
|  |         } catch (e: IndexOutOfBoundsException) { | ||||||
|  |             finish() | ||||||
|  |         } | ||||||
|  |  | ||||||
|         binding.pager.adapter = ScreenSlidePagerAdapter(this) |         binding.pager.adapter = ScreenSlidePagerAdapter(this) | ||||||
|         binding.pager.setCurrentItem(currentItem, false) |         binding.pager.setCurrentItem(currentItem, false) | ||||||
| @@ -98,15 +101,15 @@ class ReaderActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|     private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : |     private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : | ||||||
|         FragmentStateAdapter(fa) { |         FragmentStateAdapter(fa) { | ||||||
|  |  | ||||||
|         override fun getItemCount(): Int = allItems.size |         override fun getItemCount(): Int = allItems.size | ||||||
|  |  | ||||||
|         override fun createFragment(position: Int): Fragment = |         override fun createFragment(position: Int): Fragment = ArticleFragment.newInstance(allItems[position]) | ||||||
|             ArticleFragment.newInstance(allItems[position]) |  | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onKeyDown(keyCode: Int, event: KeyEvent?): Boolean { |     override fun onKeyDown( | ||||||
|  |         keyCode: Int, | ||||||
|  |         event: KeyEvent?, | ||||||
|  |     ): Boolean { | ||||||
|         return when (keyCode) { |         return when (keyCode) { | ||||||
|             KeyEvent.KEYCODE_VOLUME_DOWN -> { |             KeyEvent.KEYCODE_VOLUME_DOWN -> { | ||||||
|                 val currentFragment = |                 val currentFragment = | ||||||
| @@ -148,10 +151,8 @@ class ReaderActivity : AppCompatActivity(), DIAware { | |||||||
|                 canFavorite() |                 canFavorite() | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |  | ||||||
|             binding.pager.registerOnPageChangeCallback( |             binding.pager.registerOnPageChangeCallback( | ||||||
|                 object : ViewPager2.OnPageChangeCallback() { |                 object : ViewPager2.OnPageChangeCallback() { | ||||||
|  |  | ||||||
|                     override fun onPageSelected(position: Int) { |                     override fun onPageSelected(position: Int) { | ||||||
|                         super.onPageSelected(position) |                         super.onPageSelected(position) | ||||||
|  |  | ||||||
| @@ -162,7 +163,7 @@ class ReaderActivity : AppCompatActivity(), DIAware { | |||||||
|                         } |                         } | ||||||
|                         readItem(allItems[position]) |                         readItem(allItems[position]) | ||||||
|                     } |                     } | ||||||
|                 } |                 }, | ||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ import androidx.appcompat.app.AppCompatActivity | |||||||
| import androidx.recyclerview.widget.LinearLayoutManager | import androidx.recyclerview.widget.LinearLayoutManager | ||||||
| import bou.amine.apps.readerforselfossv2.android.adapters.SourcesListAdapter | import bou.amine.apps.readerforselfossv2.android.adapters.SourcesListAdapter | ||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySourcesBinding | import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySourcesBinding | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.testing.CountingIdlingResourceSingleton | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| @@ -18,11 +19,10 @@ import org.kodein.di.android.closestDI | |||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|  |  | ||||||
| class SourcesActivity : AppCompatActivity(), DIAware { | class SourcesActivity : AppCompatActivity(), DIAware { | ||||||
|  |  | ||||||
|     private lateinit var binding: ActivitySourcesBinding |     private lateinit var binding: ActivitySourcesBinding | ||||||
|  |  | ||||||
|     override val di by closestDI() |     override val di by closestDI() | ||||||
|     private val repository : Repository by instance() |     private val repository: Repository by instance() | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         binding = ActivitySourcesBinding.inflate(layoutInflater) |         binding = ActivitySourcesBinding.inflate(layoutInflater) | ||||||
| @@ -37,7 +37,8 @@ class SourcesActivity : AppCompatActivity(), DIAware { | |||||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) |         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||||
|  |  | ||||||
|         binding.fab.rippleColor = resources.getColor(R.color.colorAccentDark) |         binding.fab.rippleColor = resources.getColor(R.color.colorAccentDark) | ||||||
|         binding.fab.backgroundTintList = ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) |         binding.fab.backgroundTintList = | ||||||
|  |             ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onStop() { |     override fun onStop() { | ||||||
| @@ -49,17 +50,20 @@ class SourcesActivity : AppCompatActivity(), DIAware { | |||||||
|         super.onResume() |         super.onResume() | ||||||
|         val mLayoutManager = LinearLayoutManager(this) |         val mLayoutManager = LinearLayoutManager(this) | ||||||
|  |  | ||||||
|         var items: ArrayList<SelfossModel.Source> |         var items: ArrayList<SelfossModel.SourceDetail> | ||||||
|  |  | ||||||
|         binding.recyclerView.setHasFixedSize(true) |         binding.recyclerView.setHasFixedSize(true) | ||||||
|         binding.recyclerView.layoutManager = mLayoutManager |         binding.recyclerView.layoutManager = mLayoutManager | ||||||
|  |  | ||||||
|  |         CountingIdlingResourceSingleton.increment() | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|             val response = repository.getSources() |             val response = repository.getSourcesDetails() | ||||||
|             if (response.isNotEmpty()) { |             if (response.isNotEmpty()) { | ||||||
|                 items = response |                 items = response | ||||||
|                 val mAdapter = SourcesListAdapter( |                 val mAdapter = | ||||||
|                     this@SourcesActivity, items |                     SourcesListAdapter( | ||||||
|  |                         this@SourcesActivity, | ||||||
|  |                         items, | ||||||
|                     ) |                     ) | ||||||
|                 binding.recyclerView.adapter = mAdapter |                 binding.recyclerView.adapter = mAdapter | ||||||
|                 mAdapter.notifyDataSetChanged() |                 mAdapter.notifyDataSetChanged() | ||||||
| @@ -67,9 +71,10 @@ class SourcesActivity : AppCompatActivity(), DIAware { | |||||||
|                 Toast.makeText( |                 Toast.makeText( | ||||||
|                     this@SourcesActivity, |                     this@SourcesActivity, | ||||||
|                     R.string.cant_get_sources, |                     R.string.cant_get_sources, | ||||||
|                     Toast.LENGTH_SHORT |                     Toast.LENGTH_SHORT, | ||||||
|                 ).show() |                 ).show() | ||||||
|             } |             } | ||||||
|  |             CountingIdlingResourceSingleton.decrement() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         binding.fab.setOnClickListener { |         binding.fab.setOnClickListener { | ||||||
|   | |||||||
| @@ -21,10 +21,8 @@ import org.kodein.di.DIAware | |||||||
| import org.kodein.di.android.closestDI | import org.kodein.di.android.closestDI | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|  |  | ||||||
|  |  | ||||||
| class UpsertSourceActivity : AppCompatActivity(), DIAware { | class UpsertSourceActivity : AppCompatActivity(), DIAware { | ||||||
|  |     private var existingSource: SelfossModel.SourceDetail? = null | ||||||
|     private var existingSource: SelfossModel.Source? = null |  | ||||||
|     private var mSpoutsValue: String? = null |     private var mSpoutsValue: String? = null | ||||||
|  |  | ||||||
|     private lateinit var binding: ActivityUpsertSourceBinding |     private lateinit var binding: ActivityUpsertSourceBinding | ||||||
| @@ -58,7 +56,6 @@ class UpsertSourceActivity : AppCompatActivity(), DIAware { | |||||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) |         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||||
|         supportActionBar?.title = resources.getString(title) |         supportActionBar?.title = resources.getString(title) | ||||||
|  |  | ||||||
|  |  | ||||||
|         maybeGetDetailsFromIntentSharing(intent) |         maybeGetDetailsFromIntentSharing(intent) | ||||||
|  |  | ||||||
|         binding.saveBtn.setOnClickListener { |         binding.saveBtn.setOnClickListener { | ||||||
| @@ -68,7 +65,7 @@ class UpsertSourceActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|     private fun initFields(items: Map<String, SelfossModel.Spout>) { |     private fun initFields(items: Map<String, SelfossModel.Spout>) { | ||||||
|         binding.nameInput.setText(existingSource!!.title) |         binding.nameInput.setText(existingSource!!.title) | ||||||
|         binding.tags.setText(existingSource!!.tags.joinToString(", ")) |         binding.tags.setText(existingSource!!.tags?.joinToString(", ")) | ||||||
|         binding.sourceUri.setText(existingSource!!.params?.url) |         binding.sourceUri.setText(existingSource!!.params?.url) | ||||||
|         binding.spoutsSpinner.setSelection(items.keys.indexOf(existingSource!!.spout)) |         binding.spoutsSpinner.setSelection(items.keys.indexOf(existingSource!!.spout)) | ||||||
|         binding.progress.visibility = View.GONE |         binding.progress.visibility = View.GONE | ||||||
| @@ -88,8 +85,14 @@ class UpsertSourceActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|     private fun handleSpoutsSpinner() { |     private fun handleSpoutsSpinner() { | ||||||
|         val spoutsKV = HashMap<String, String>() |         val spoutsKV = HashMap<String, String>() | ||||||
|         binding.spoutsSpinner.onItemSelectedListener = object : AdapterView.OnItemSelectedListener { |         binding.spoutsSpinner.onItemSelectedListener = | ||||||
|             override fun onItemSelected(adapterView: AdapterView<*>, view: View?, i: Int, l: Long) { |             object : AdapterView.OnItemSelectedListener { | ||||||
|  |                 override fun onItemSelected( | ||||||
|  |                     adapterView: AdapterView<*>, | ||||||
|  |                     view: View?, | ||||||
|  |                     i: Int, | ||||||
|  |                     l: Long, | ||||||
|  |                 ) { | ||||||
|                     if (view != null) { |                     if (view != null) { | ||||||
|                         val spoutName = (view as TextView).text.toString() |                         val spoutName = (view as TextView).text.toString() | ||||||
|                         mSpoutsValue = spoutsKV[spoutName] |                         mSpoutsValue = spoutsKV[spoutName] | ||||||
| @@ -101,12 +104,11 @@ class UpsertSourceActivity : AppCompatActivity(), DIAware { | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |  | ||||||
|         fun handleSpoutFailure(networkIssue: Boolean = false) { |         fun handleSpoutFailure(networkIssue: Boolean = false) { | ||||||
|             Toast.makeText( |             Toast.makeText( | ||||||
|                 this@UpsertSourceActivity, |                 this@UpsertSourceActivity, | ||||||
|                 if (networkIssue) R.string.cant_get_spouts_no_network else R.string.cant_get_spouts, |                 if (networkIssue) R.string.cant_get_spouts_no_network else R.string.cant_get_spouts, | ||||||
|                 Toast.LENGTH_SHORT |                 Toast.LENGTH_SHORT, | ||||||
|             ).show() |             ).show() | ||||||
|             binding.progress.visibility = View.GONE |             binding.progress.visibility = View.GONE | ||||||
|         } |         } | ||||||
| @@ -127,7 +129,7 @@ class UpsertSourceActivity : AppCompatActivity(), DIAware { | |||||||
|                         ArrayAdapter( |                         ArrayAdapter( | ||||||
|                             this@UpsertSourceActivity, |                             this@UpsertSourceActivity, | ||||||
|                             android.R.layout.simple_spinner_item, |                             android.R.layout.simple_spinner_item, | ||||||
|                             itemsStrings |                             itemsStrings, | ||||||
|                         ) |                         ) | ||||||
|                     spinnerArrayAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item) |                     spinnerArrayAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item) | ||||||
|                     binding.spoutsSpinner.adapter = spinnerArrayAdapter |                     binding.spoutsSpinner.adapter = spinnerArrayAdapter | ||||||
| @@ -144,9 +146,7 @@ class UpsertSourceActivity : AppCompatActivity(), DIAware { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun maybeGetDetailsFromIntentSharing( |     private fun maybeGetDetailsFromIntentSharing(intent: Intent) { | ||||||
|         intent: Intent |  | ||||||
|     ) { |  | ||||||
|         if (Intent.ACTION_SEND == intent.action && "text/plain" == intent.type) { |         if (Intent.ACTION_SEND == intent.action && "text/plain" == intent.type) { | ||||||
|             binding.sourceUri.setText(intent.getStringExtra(Intent.EXTRA_TEXT)) |             binding.sourceUri.setText(intent.getStringExtra(Intent.EXTRA_TEXT)) | ||||||
|             binding.nameInput.setText(intent.getStringExtra(Intent.EXTRA_TITLE)) |             binding.nameInput.setText(intent.getStringExtra(Intent.EXTRA_TITLE)) | ||||||
| @@ -172,13 +172,14 @@ class UpsertSourceActivity : AppCompatActivity(), DIAware { | |||||||
|             } |             } | ||||||
|             else -> { |             else -> { | ||||||
|                 CoroutineScope(Dispatchers.Main).launch { |                 CoroutineScope(Dispatchers.Main).launch { | ||||||
|                     val successfullyAddedSource = if (existingSource != null) { |                     val successfullyAddedSource = | ||||||
|  |                         if (existingSource != null) { | ||||||
|                             repository.updateSource( |                             repository.updateSource( | ||||||
|                                 existingSource!!.id, |                                 existingSource!!.id, | ||||||
|                                 binding.nameInput.text.toString(), |                                 binding.nameInput.text.toString(), | ||||||
|                                 url, |                                 url, | ||||||
|                                 mSpoutsValue!!, |                                 mSpoutsValue!!, | ||||||
|                             binding.tags.text.toString() |                                 binding.tags.text.toString(), | ||||||
|                             ) |                             ) | ||||||
|                         } else { |                         } else { | ||||||
|                             repository.createSource( |                             repository.createSource( | ||||||
| @@ -194,7 +195,7 @@ class UpsertSourceActivity : AppCompatActivity(), DIAware { | |||||||
|                         Toast.makeText( |                         Toast.makeText( | ||||||
|                             this@UpsertSourceActivity, |                             this@UpsertSourceActivity, | ||||||
|                             R.string.cant_create_source, |                             R.string.cant_create_source, | ||||||
|                             Toast.LENGTH_SHORT |                             Toast.LENGTH_SHORT, | ||||||
|                         ).show() |                         ).show() | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.android.adapters | package bou.amine.apps.readerforselfossv2.android.adapters | ||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
| import android.content.Context |  | ||||||
| import android.view.LayoutInflater | import android.view.LayoutInflater | ||||||
| import android.view.View | import android.view.View | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
| @@ -9,12 +8,11 @@ import android.widget.ImageView.ScaleType | |||||||
| import androidx.recyclerview.widget.RecyclerView | import androidx.recyclerview.widget.RecyclerView | ||||||
| import bou.amine.apps.readerforselfossv2.android.R | import bou.amine.apps.readerforselfossv2.android.R | ||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.CardItemBinding | import bou.amine.apps.readerforselfossv2.android.databinding.CardItemBinding | ||||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.LinkOnTouchListener | import bou.amine.apps.readerforselfossv2.android.utils.LinkOnTouchListener | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapCenterCrop | import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapCenterCrop | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable | import bou.amine.apps.readerforselfossv2.android.utils.glide.circularDrawable | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrl |  | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| @@ -22,8 +20,6 @@ import bou.amine.apps.readerforselfossv2.service.AppSettingsService | |||||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getThumbnail | import bou.amine.apps.readerforselfossv2.utils.getThumbnail | ||||||
| import com.amulyakhare.textdrawable.TextDrawable |  | ||||||
| import com.amulyakhare.textdrawable.util.ColorGenerator |  | ||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Dispatchers | import kotlinx.coroutines.Dispatchers | ||||||
| @@ -34,11 +30,10 @@ import org.kodein.di.instance | |||||||
|  |  | ||||||
| class ItemCardAdapter( | class ItemCardAdapter( | ||||||
|     override val app: Activity, |     override val app: Activity, | ||||||
|     override var items: ArrayList<SelfossModel.Item>, |     override val items: ArrayList<SelfossModel.Item>, | ||||||
|     override val updateItems: (ArrayList<SelfossModel.Item>) -> Unit |     override val updateHomeItems: (ArrayList<SelfossModel.Item>) -> Unit, | ||||||
| ) : ItemsAdapter<ItemCardAdapter.ViewHolder>() { | ) : ItemsAdapter<ItemCardAdapter.ViewHolder>() { | ||||||
|     private val c: Context = app.baseContext |     override lateinit var binding: CardItemBinding | ||||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL |  | ||||||
|     private val imageMaxHeight: Int = |     private val imageMaxHeight: Int = | ||||||
|         c.resources.getDimension(R.dimen.card_image_max_height).toInt() |         c.resources.getDimension(R.dimen.card_image_max_height).toInt() | ||||||
|  |  | ||||||
| @@ -46,15 +41,50 @@ class ItemCardAdapter( | |||||||
|     override val repository: Repository by instance() |     override val repository: Repository by instance() | ||||||
|     override val appSettingsService: AppSettingsService by instance() |     override val appSettingsService: AppSettingsService by instance() | ||||||
|  |  | ||||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { |     override fun onCreateViewHolder( | ||||||
|         val binding = CardItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) |         parent: ViewGroup, | ||||||
|  |         viewType: Int, | ||||||
|  |     ): ViewHolder { | ||||||
|  |         binding = CardItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||||
|         return ViewHolder(binding) |         return ViewHolder(binding) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onBindViewHolder(holder: ViewHolder, position: Int) { |     private fun handleClickListeners(holderBinding: CardItemBinding, position: Int) { | ||||||
|  |         holderBinding.favButton.setOnClickListener { | ||||||
|  |             val item = items[position] | ||||||
|  |             if (item.starred) { | ||||||
|  |                 CoroutineScope(Dispatchers.IO).launch { | ||||||
|  |                     repository.unstarr(item) | ||||||
|  |                 } | ||||||
|  |                 binding.favButton.isSelected = false | ||||||
|  |             } else { | ||||||
|  |                 CoroutineScope(Dispatchers.IO).launch { | ||||||
|  |                     repository.starr(item) | ||||||
|  |                 } | ||||||
|  |                 binding.favButton.isSelected = true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         binding.shareBtn.setOnClickListener { | ||||||
|  |             val item = items[position] | ||||||
|  |             c.shareLink(item.getLinkDecoded(), item.title.getHtmlDecoded()) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         binding.browserBtn.setOnClickListener { | ||||||
|  |             c.openInBrowserAsNewTask(items[position]) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onBindViewHolder( | ||||||
|  |         holder: ViewHolder, | ||||||
|  |         position: Int, | ||||||
|  |     ) { | ||||||
|         with(holder) { |         with(holder) { | ||||||
|             val itm = items[position] |             val itm = items[position] | ||||||
|  |  | ||||||
|  |             handleClickListeners(binding, position) | ||||||
|  |             handleLinkOpening(binding, position) | ||||||
|  |  | ||||||
|             binding.favButton.isSelected = itm.starred |             binding.favButton.isSelected = itm.starred | ||||||
|             if (appSettingsService.getPublicAccess()) { |             if (appSettingsService.getPublicAccess()) { | ||||||
|                 binding.favButton.visibility = View.GONE |                 binding.favButton.visibility = View.GONE | ||||||
| @@ -66,7 +96,12 @@ class ItemCardAdapter( | |||||||
|  |  | ||||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) |             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||||
|  |  | ||||||
|             binding.sourceTitleAndDate.text = itm.sourceAuthorAndDate() |             binding.sourceTitleAndDate.text = try { | ||||||
|  |                 itm.sourceAuthorAndDate() | ||||||
|  |             } catch (e: Exception) { | ||||||
|  |                 e.sendSilentlyWithAcraWithName("ItemCardAdapter parse date") | ||||||
|  |                 itm.sourceAuthorOnly() | ||||||
|  |             } | ||||||
|  |  | ||||||
|             if (!appSettingsService.isFullHeightCardsEnabled()) { |             if (!appSettingsService.isFullHeightCardsEnabled()) { | ||||||
|                 binding.itemImage.maxHeight = imageMaxHeight |                 binding.itemImage.maxHeight = imageMaxHeight | ||||||
| @@ -83,67 +118,12 @@ class ItemCardAdapter( | |||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (itm.getIcon(repository.baseUrl).isEmpty()) { |             if (itm.getIcon(repository.baseUrl).isEmpty()) { | ||||||
|                 val color = generator.getColor(itm.title.getHtmlDecoded()) |                 binding.sourceImage.setBackgroundAndText(itm.sourcetitle.getHtmlDecoded()) | ||||||
|  |  | ||||||
|                 val drawable = |  | ||||||
|                     TextDrawable |  | ||||||
|                         .builder() |  | ||||||
|                         .round() |  | ||||||
|                         .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) |  | ||||||
|                 binding.sourceImage.setImageDrawable(drawable) |  | ||||||
|             } else { |             } else { | ||||||
|                 c.circularBitmapDrawable(itm.getIcon(repository.baseUrl), binding.sourceImage) |                 c.circularDrawable(itm.getIcon(repository.baseUrl), binding.sourceImage) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun getItemCount(): Int { |     inner class ViewHolder(val binding: CardItemBinding) : RecyclerView.ViewHolder(binding.root) | ||||||
|         return items.size |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     inner class ViewHolder(val binding: CardItemBinding) : RecyclerView.ViewHolder(binding.root) { |  | ||||||
|         init { |  | ||||||
|             handleClickListeners() |  | ||||||
|             handleLinkOpening() |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private fun handleClickListeners() { |  | ||||||
|  |  | ||||||
|             binding.favButton.setOnClickListener { |  | ||||||
|                 val item = items[bindingAdapterPosition] |  | ||||||
|                 if (item.starred) { |  | ||||||
|                     CoroutineScope(Dispatchers.IO).launch { |  | ||||||
|                         repository.unstarr(item) |  | ||||||
|                     } |  | ||||||
|                     binding.favButton.isSelected = false |  | ||||||
|                 } else { |  | ||||||
|                     CoroutineScope(Dispatchers.IO).launch { |  | ||||||
|                         repository.starr(item) |  | ||||||
|                     } |  | ||||||
|                     binding.favButton.isSelected = true |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             binding.shareBtn.setOnClickListener { |  | ||||||
|                 val item = items[bindingAdapterPosition] |  | ||||||
|                 c.shareLink(item.getLinkDecoded(), item.title.getHtmlDecoded()) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             binding.browserBtn.setOnClickListener { |  | ||||||
|                 c.openInBrowserAsNewTask(items[bindingAdapterPosition]) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private fun handleLinkOpening() { |  | ||||||
|             binding.root.setOnClickListener { |  | ||||||
|                 repository.setReaderItems(items) |  | ||||||
|                 c.openItemUrl( |  | ||||||
|                     bindingAdapterPosition, |  | ||||||
|                     items[bindingAdapterPosition].getLinkDecoded(), |  | ||||||
|                     appSettingsService.isArticleViewerEnabled(), |  | ||||||
|                     app |  | ||||||
|                 ) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,97 +1,76 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.android.adapters | package bou.amine.apps.readerforselfossv2.android.adapters | ||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
| import android.content.Context |  | ||||||
| import android.view.LayoutInflater | import android.view.LayoutInflater | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
| import androidx.recyclerview.widget.RecyclerView | import androidx.recyclerview.widget.RecyclerView | ||||||
| import bou.amine.apps.readerforselfossv2.android.R | import bou.amine.apps.readerforselfossv2.android.R | ||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ListItemBinding | import bou.amine.apps.readerforselfossv2.android.databinding.ListItemBinding | ||||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.LinkOnTouchListener | import bou.amine.apps.readerforselfossv2.android.utils.LinkOnTouchListener | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapCenterCrop | import bou.amine.apps.readerforselfossv2.android.utils.glide.circularDrawable | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable |  | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrl |  | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getThumbnail | import bou.amine.apps.readerforselfossv2.utils.getThumbnail | ||||||
| import com.amulyakhare.textdrawable.TextDrawable |  | ||||||
| import com.amulyakhare.textdrawable.util.ColorGenerator |  | ||||||
| import org.kodein.di.DI | import org.kodein.di.DI | ||||||
| import org.kodein.di.android.closestDI | import org.kodein.di.android.closestDI | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|  |  | ||||||
| class ItemListAdapter( | class ItemListAdapter( | ||||||
|     override val app: Activity, |     override val app: Activity, | ||||||
|     override var items: ArrayList<SelfossModel.Item>, |     override val items: ArrayList<SelfossModel.Item>, | ||||||
|     override val updateItems: (ArrayList<SelfossModel.Item>) -> Unit |     override val updateHomeItems: (ArrayList<SelfossModel.Item>) -> Unit, | ||||||
| ) : ItemsAdapter<ItemListAdapter.ViewHolder>() { | ) : ItemsAdapter<ItemListAdapter.ViewHolder>() { | ||||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL |     override lateinit var binding: ListItemBinding | ||||||
|     private val c: Context = app.baseContext |  | ||||||
|  |  | ||||||
|     override val di: DI by closestDI(app) |     override val di: DI by closestDI(app) | ||||||
|     override val repository : Repository by instance() |     override val repository: Repository by instance() | ||||||
|     override val appSettingsService : AppSettingsService by instance() |     override val appSettingsService: AppSettingsService by instance() | ||||||
|  |  | ||||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { |     override fun onCreateViewHolder( | ||||||
|         val binding = ListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) |         parent: ViewGroup, | ||||||
|  |         viewType: Int, | ||||||
|  |     ): ViewHolder { | ||||||
|  |         binding = ListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||||
|         return ViewHolder(binding) |         return ViewHolder(binding) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onBindViewHolder(holder: ViewHolder, position: Int) { |     override fun onBindViewHolder( | ||||||
|  |         holder: ViewHolder, | ||||||
|  |         position: Int, | ||||||
|  |     ) { | ||||||
|         with(holder) { |         with(holder) { | ||||||
|             val itm = items[position] |             val itm = items[position] | ||||||
|  |  | ||||||
|  |             handleLinkOpening(binding, position) | ||||||
|  |  | ||||||
|             binding.title.text = itm.title.getHtmlDecoded() |             binding.title.text = itm.title.getHtmlDecoded() | ||||||
|  |  | ||||||
|             binding.title.setOnTouchListener(LinkOnTouchListener()) |             binding.title.setOnTouchListener(LinkOnTouchListener()) | ||||||
|  |  | ||||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) |             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||||
|  |  | ||||||
|             binding.sourceTitleAndDate.text = itm.sourceAuthorAndDate() |             binding.sourceTitleAndDate.text = try { | ||||||
|  |                 itm.sourceAuthorAndDate() | ||||||
|  |             } catch (e: Exception) { | ||||||
|  |                 e.sendSilentlyWithAcraWithName("ItemListAdapter parse date") | ||||||
|  |                 itm.sourceAuthorOnly() | ||||||
|  |             } | ||||||
|  |  | ||||||
|             if (itm.getThumbnail(repository.baseUrl).isEmpty()) { |             if (itm.getThumbnail(repository.baseUrl).isEmpty()) { | ||||||
|  |  | ||||||
|                 if (itm.getIcon(repository.baseUrl).isEmpty()) { |                 if (itm.getIcon(repository.baseUrl).isEmpty()) { | ||||||
|                     val color = generator.getColor(itm.title.getHtmlDecoded()) |                     binding.itemImage.setBackgroundAndText(itm.sourcetitle.getHtmlDecoded()) | ||||||
|  |  | ||||||
|                     val drawable = |  | ||||||
|                             TextDrawable |  | ||||||
|                                     .builder() |  | ||||||
|                                     .round() |  | ||||||
|                                     .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) |  | ||||||
|  |  | ||||||
|                     binding.itemImage.setImageDrawable(drawable) |  | ||||||
|                 } else { |                 } else { | ||||||
|                     c.circularBitmapDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) |                     c.circularDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|                 c.bitmapCenterCrop(itm.getThumbnail(repository.baseUrl), binding.itemImage) |                 c.circularDrawable(itm.getThumbnail(repository.baseUrl), binding.itemImage) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun getItemCount(): Int = items.size |     inner class ViewHolder(val binding: ListItemBinding) : RecyclerView.ViewHolder(binding.root) | ||||||
|  |  | ||||||
|     inner class ViewHolder(val binding: ListItemBinding) : RecyclerView.ViewHolder(binding.root) { |  | ||||||
|  |  | ||||||
|         init { |  | ||||||
|             handleLinkOpening() |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private fun handleLinkOpening() { |  | ||||||
|             binding.root.setOnClickListener { |  | ||||||
|                 repository.setReaderItems(items) |  | ||||||
|                 c.openItemUrl( |  | ||||||
|                     bindingAdapterPosition, |  | ||||||
|                     items[bindingAdapterPosition].getLinkDecoded(), |  | ||||||
|                     appSettingsService.isArticleViewerEnabled(), |  | ||||||
|                     app |  | ||||||
|                 ) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,10 +1,13 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.android.adapters | package bou.amine.apps.readerforselfossv2.android.adapters | ||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
|  | import android.content.Context | ||||||
| import android.graphics.Color | import android.graphics.Color | ||||||
| import android.widget.TextView | import android.widget.TextView | ||||||
| import androidx.recyclerview.widget.RecyclerView | import androidx.recyclerview.widget.RecyclerView | ||||||
|  | import androidx.viewbinding.ViewBinding | ||||||
| import bou.amine.apps.readerforselfossv2.android.R | import bou.amine.apps.readerforselfossv2.android.R | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.utils.openItemUrl | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| @@ -16,24 +19,32 @@ import kotlinx.coroutines.launch | |||||||
| import org.kodein.di.DIAware | import org.kodein.di.DIAware | ||||||
|  |  | ||||||
| abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapter<VH>(), DIAware { | abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapter<VH>(), DIAware { | ||||||
|     abstract var items: ArrayList<SelfossModel.Item> |     abstract val items: ArrayList<SelfossModel.Item> | ||||||
|     abstract val repository: Repository |     abstract val repository: Repository | ||||||
|  |     abstract val binding: ViewBinding | ||||||
|     abstract val appSettingsService: AppSettingsService |     abstract val appSettingsService: AppSettingsService | ||||||
|     abstract val app: Activity |     abstract val app: Activity | ||||||
|     abstract val updateItems: (ArrayList<SelfossModel.Item>) -> Unit |     abstract val updateHomeItems: (ArrayList<SelfossModel.Item>) -> Unit | ||||||
|  |  | ||||||
|  |     protected val c: Context get() = app.baseContext | ||||||
|  |  | ||||||
|     fun updateAllItems(items: ArrayList<SelfossModel.Item>) { |     fun updateAllItems(items: ArrayList<SelfossModel.Item>) { | ||||||
|         this.items = items |         this.items.clear() | ||||||
|  |         this.items.addAll(items) | ||||||
|  |         updateHomeItems(items) | ||||||
|         notifyDataSetChanged() |         notifyDataSetChanged() | ||||||
|         updateItems(this.items) |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun unmarkSnackbar(item: SelfossModel.Item, position: Int) { |     private fun unmarkSnackbar( | ||||||
|         val s = Snackbar |         item: SelfossModel.Item, | ||||||
|  |         position: Int, | ||||||
|  |     ) { | ||||||
|  |         val s = | ||||||
|  |             Snackbar | ||||||
|                 .make( |                 .make( | ||||||
|                     app.findViewById(R.id.coordLayout), |                     app.findViewById(R.id.coordLayout), | ||||||
|                     R.string.marked_as_read, |                     R.string.marked_as_read, | ||||||
|                 Snackbar.LENGTH_LONG |                     Snackbar.LENGTH_LONG, | ||||||
|                 ) |                 ) | ||||||
|                 .setAction(R.string.undo_string) { |                 .setAction(R.string.undo_string) { | ||||||
|                     unreadItemAtIndex(item, position, false) |                     unreadItemAtIndex(item, position, false) | ||||||
| @@ -45,12 +56,16 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|         s.show() |         s.show() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun markSnackbar(item: SelfossModel.Item, position: Int) { |     private fun markSnackbar( | ||||||
|         val s = Snackbar |         item: SelfossModel.Item, | ||||||
|  |         position: Int, | ||||||
|  |     ) { | ||||||
|  |         val s = | ||||||
|  |             Snackbar | ||||||
|                 .make( |                 .make( | ||||||
|                     app.findViewById(R.id.coordLayout), |                     app.findViewById(R.id.coordLayout), | ||||||
|                     R.string.marked_as_unread, |                     R.string.marked_as_unread, | ||||||
|                 Snackbar.LENGTH_LONG |                     Snackbar.LENGTH_LONG, | ||||||
|                 ) |                 ) | ||||||
|                 .setAction(R.string.undo_string) { |                 .setAction(R.string.undo_string) { | ||||||
|                     readItemAtIndex(item, position, false) |                     readItemAtIndex(item, position, false) | ||||||
| @@ -62,6 +77,18 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|         s.show() |         s.show() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     protected fun handleLinkOpening(holderBinding: ViewBinding, position: Int) { | ||||||
|  |         holderBinding.root.setOnClickListener { | ||||||
|  |             repository.setReaderItems(items) | ||||||
|  |             c.openItemUrl( | ||||||
|  |                 position, | ||||||
|  |                 items[position].getLinkDecoded(), | ||||||
|  |                 appSettingsService.isArticleViewerEnabled(), | ||||||
|  |                 app, | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     fun handleItemAtIndex(position: Int) { |     fun handleItemAtIndex(position: Int) { | ||||||
|         if (items[position].unread) { |         if (items[position].unread) { | ||||||
|             readItemAtIndex(items[position], position) |             readItemAtIndex(items[position], position) | ||||||
| @@ -70,14 +97,19 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun readItemAtIndex(item: SelfossModel.Item, position: Int, showSnackbar: Boolean = true) { |     private fun readItemAtIndex( | ||||||
|  |         item: SelfossModel.Item, | ||||||
|  |         position: Int, | ||||||
|  |         showSnackbar: Boolean = true, | ||||||
|  |     ) { | ||||||
|         CoroutineScope(Dispatchers.IO).launch { |         CoroutineScope(Dispatchers.IO).launch { | ||||||
|             repository.markAsRead(item) |             repository.markAsRead(item) | ||||||
|         } |         } | ||||||
|         if (repository.displayedItems == ItemType.UNREAD) { |         if (repository.displayedItems == ItemType.UNREAD) { | ||||||
|             items.remove(item) |             items.remove(item) | ||||||
|             notifyItemRemoved(position) |             notifyItemRemoved(position) | ||||||
|             updateItems(items) |             notifyItemRangeChanged(position, itemCount) | ||||||
|  |             updateHomeItems(items) | ||||||
|         } else { |         } else { | ||||||
|             notifyItemChanged(position) |             notifyItemChanged(position) | ||||||
|         } |         } | ||||||
| @@ -86,10 +118,13 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun unreadItemAtIndex(item: SelfossModel.Item, position: Int, showSnackbar: Boolean = true) { |     private fun unreadItemAtIndex( | ||||||
|  |         item: SelfossModel.Item, | ||||||
|  |         position: Int, | ||||||
|  |         showSnackbar: Boolean = true, | ||||||
|  |     ) { | ||||||
|         CoroutineScope(Dispatchers.IO).launch { |         CoroutineScope(Dispatchers.IO).launch { | ||||||
|             repository.unmarkAsRead(item) |             repository.unmarkAsRead(item) | ||||||
|  |  | ||||||
|         } |         } | ||||||
|         notifyItemChanged(position) |         notifyItemChanged(position) | ||||||
|         if (showSnackbar) { |         if (showSnackbar) { | ||||||
| @@ -97,18 +132,21 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun addItemAtIndex(item: SelfossModel.Item, position: Int) { |     fun addItemAtIndex( | ||||||
|  |         item: SelfossModel.Item, | ||||||
|  |         position: Int, | ||||||
|  |     ) { | ||||||
|         items.add(position, item) |         items.add(position, item) | ||||||
|         notifyItemInserted(position) |         notifyItemInserted(position) | ||||||
|         updateItems(items) |         updateHomeItems(items) | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun addItemsAtEnd(newItems: List<SelfossModel.Item>) { |     fun addItemsAtEnd(newItems: List<SelfossModel.Item>) { | ||||||
|         val oldSize = items.size |         val oldSize = items.size | ||||||
|         items.addAll(newItems) |         items.addAll(newItems) | ||||||
|         notifyItemRangeInserted(oldSize, newItems.size) |         notifyItemRangeInserted(oldSize, newItems.size) | ||||||
|         updateItems(items) |         updateHomeItems(items) | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     override fun getItemCount(): Int = items.size | ||||||
| } | } | ||||||
| @@ -10,17 +10,14 @@ import android.widget.Button | |||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import androidx.constraintlayout.widget.ConstraintLayout | import androidx.constraintlayout.widget.ConstraintLayout | ||||||
| import androidx.recyclerview.widget.RecyclerView | import androidx.recyclerview.widget.RecyclerView | ||||||
| import bou.amine.apps.readerforselfossv2.android.UpsertSourceActivity |  | ||||||
| import bou.amine.apps.readerforselfossv2.android.R | import bou.amine.apps.readerforselfossv2.android.R | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.UpsertSourceActivity | ||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.SourceListItemBinding | import bou.amine.apps.readerforselfossv2.android.databinding.SourceListItemBinding | ||||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | import bou.amine.apps.readerforselfossv2.android.utils.glide.circularDrawable | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable |  | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||||
| import com.amulyakhare.textdrawable.TextDrawable |  | ||||||
| import com.amulyakhare.textdrawable.util.ColorGenerator |  | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Dispatchers | import kotlinx.coroutines.Dispatchers | ||||||
| import kotlinx.coroutines.launch | import kotlinx.coroutines.launch | ||||||
| @@ -31,37 +28,62 @@ import org.kodein.di.instance | |||||||
|  |  | ||||||
| class SourcesListAdapter( | class SourcesListAdapter( | ||||||
|     private val app: Activity, |     private val app: Activity, | ||||||
|     private val items: ArrayList<SelfossModel.Source> |     private val items: ArrayList<SelfossModel.SourceDetail>, | ||||||
| ) : RecyclerView.Adapter<SourcesListAdapter.ViewHolder>(), DIAware { | ) : RecyclerView.Adapter<SourcesListAdapter.ViewHolder>(), DIAware { | ||||||
|     private val c: Context = app.baseContext |     private val c: Context = app.baseContext | ||||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL |  | ||||||
|     private lateinit var binding: SourceListItemBinding |     private lateinit var binding: SourceListItemBinding | ||||||
|  |  | ||||||
|     override val di: DI by closestDI(app) |     override val di: DI by closestDI(app) | ||||||
|     private val repository : Repository by instance() |     private val repository: Repository by instance() | ||||||
|  |  | ||||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { |     override fun onCreateViewHolder( | ||||||
|  |         parent: ViewGroup, | ||||||
|  |         viewType: Int, | ||||||
|  |     ): ViewHolder { | ||||||
|         binding = SourceListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) |         binding = SourceListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||||
|         return ViewHolder(binding.root) |         return ViewHolder(binding.root) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onBindViewHolder(holder: ViewHolder, position: Int) { |     override fun onBindViewHolder( | ||||||
|  |         holder: ViewHolder, | ||||||
|  |         position: Int, | ||||||
|  |     ) { | ||||||
|         val itm = items[position] |         val itm = items[position] | ||||||
|  |  | ||||||
|         if (itm.getIcon(repository.baseUrl).isEmpty()) { |         val deleteBtn: Button = holder.mView.findViewById(R.id.deleteBtn) | ||||||
|             val color = generator.getColor(itm.title.getHtmlDecoded()) |  | ||||||
|  |  | ||||||
|             val drawable = |         deleteBtn.setOnClickListener { | ||||||
|                 TextDrawable |             val (id, title) = items[position] | ||||||
|                     .builder() |             CoroutineScope(Dispatchers.IO).launch { | ||||||
|                     .round() |                 val successfullyDeletedSource = repository.deleteSource(id, title) | ||||||
|                     .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) |                 if (successfullyDeletedSource) { | ||||||
|             binding.itemImage.setImageDrawable(drawable) |                     items.removeAt(position) | ||||||
|  |                     notifyItemRemoved(position) | ||||||
|  |                     notifyItemRangeChanged(position, itemCount) | ||||||
|                 } else { |                 } else { | ||||||
|             c.circularBitmapDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) |                     Toast.makeText( | ||||||
|  |                         app, | ||||||
|  |                         R.string.can_delete_source, | ||||||
|  |                         Toast.LENGTH_SHORT, | ||||||
|  |                     ).show() | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (itm.error.isNotBlank()) { |         holder.mView.setOnClickListener { | ||||||
|  |             val source = items[position] | ||||||
|  |  | ||||||
|  |             repository.setSelectedSource(source) | ||||||
|  |             app.startActivity(Intent(app, UpsertSourceActivity::class.java)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (itm.getIcon(repository.baseUrl).isEmpty()) { | ||||||
|  |             binding.itemImage.setBackgroundAndText(itm.title.getHtmlDecoded()) | ||||||
|  |         } else { | ||||||
|  |             c.circularDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (!itm.error.isNullOrBlank()) { | ||||||
|             binding.errorText.visibility = View.VISIBLE |             binding.errorText.visibility = View.VISIBLE | ||||||
|             binding.errorText.text = itm.error |             binding.errorText.text = itm.error | ||||||
|         } else { |         } else { | ||||||
| @@ -77,41 +99,5 @@ class SourcesListAdapter( | |||||||
|  |  | ||||||
|     override fun getItemCount(): Int = items.size |     override fun getItemCount(): Int = items.size | ||||||
|  |  | ||||||
|     inner class ViewHolder(private val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) { |     inner class ViewHolder(val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) | ||||||
|  |  | ||||||
|         init { |  | ||||||
|             handleClickListeners() |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private fun handleClickListeners() { |  | ||||||
|  |  | ||||||
|             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) |  | ||||||
|  |  | ||||||
|             deleteBtn.setOnClickListener { |  | ||||||
|                 val (id, title) = items[bindingAdapterPosition] |  | ||||||
|                 CoroutineScope(Dispatchers.IO).launch { |  | ||||||
|                     val successfullyDeletedSource = repository.deleteSource(id, title) |  | ||||||
|                     if (successfullyDeletedSource) { |  | ||||||
|                         items.removeAt(bindingAdapterPosition) |  | ||||||
|                         notifyItemRemoved(bindingAdapterPosition) |  | ||||||
|                         notifyItemRangeChanged(bindingAdapterPosition, itemCount) |  | ||||||
|                     } else { |  | ||||||
|                         Toast.makeText( |  | ||||||
|                             app, |  | ||||||
|                             R.string.can_delete_source, |  | ||||||
|                             Toast.LENGTH_SHORT |  | ||||||
|                         ).show() |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             mView.setOnClickListener { |  | ||||||
|                 val source = items[bindingAdapterPosition] |  | ||||||
|  |  | ||||||
|                 repository.setSelectedSource(source) |  | ||||||
|                 app.startActivity(Intent(app, UpsertSourceActivity::class.java)) |  | ||||||
|  |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -26,16 +26,15 @@ import org.kodein.di.instance | |||||||
| import java.util.* | import java.util.* | ||||||
| import kotlin.concurrent.schedule | import kotlin.concurrent.schedule | ||||||
|  |  | ||||||
| class LoadingWorker(val context: Context, params: WorkerParameters) : Worker(context, params), | class LoadingWorker(val context: Context, params: WorkerParameters) : | ||||||
|  |     Worker(context, params), | ||||||
|     DIAware { |     DIAware { | ||||||
|  |  | ||||||
|     override val di by lazy { (applicationContext as MyApp).di } |     override val di by lazy { (applicationContext as MyApp).di } | ||||||
|     private val repository: Repository by instance() |     private val repository: Repository by instance() | ||||||
|     private val appSettingsService: AppSettingsService by instance() |     private val appSettingsService: AppSettingsService by instance() | ||||||
|  |  | ||||||
|     override fun doWork(): Result { |     override fun doWork(): Result { | ||||||
|         if (appSettingsService.isPeriodicRefreshEnabled() && isNetworkAccessible(context)) { |         if (appSettingsService.isPeriodicRefreshEnabled() && isNetworkAccessible(context)) { | ||||||
|  |  | ||||||
|             CoroutineScope(Dispatchers.IO).launch { |             CoroutineScope(Dispatchers.IO).launch { | ||||||
|                 val notificationManager = |                 val notificationManager = | ||||||
|                     applicationContext.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager |                     applicationContext.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager | ||||||
| @@ -67,19 +66,19 @@ class LoadingWorker(val context: Context, params: WorkerParameters) : Worker(con | |||||||
|  |  | ||||||
|     private fun handleNewItemsNotification( |     private fun handleNewItemsNotification( | ||||||
|         newItems: List<SelfossModel.Item>?, |         newItems: List<SelfossModel.Item>?, | ||||||
|         notificationManager: NotificationManager |         notificationManager: NotificationManager, | ||||||
|     ) { |     ) { | ||||||
|         CoroutineScope(Dispatchers.IO).launch { |         CoroutineScope(Dispatchers.IO).launch { | ||||||
|             val apiItems = newItems.orEmpty() |             val apiItems = newItems.orEmpty() | ||||||
|  |  | ||||||
|  |  | ||||||
|             val newSize = apiItems.filter { it.unread }.size |             val newSize = apiItems.filter { it.unread }.size | ||||||
|             if (newSize > 0) { |             if (newSize > 0) { | ||||||
|  |                 val intent = | ||||||
|                 val intent = Intent(context, MainActivity::class.java).apply { |                     Intent(context, MainActivity::class.java).apply { | ||||||
|                         flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK |                         flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK | ||||||
|                     } |                     } | ||||||
|                 val pflags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { |                 val pflags = | ||||||
|  |                     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||||
|                         PendingIntent.FLAG_IMMUTABLE |                         PendingIntent.FLAG_IMMUTABLE | ||||||
|                     } else { |                     } else { | ||||||
|                         0 |                         0 | ||||||
| @@ -90,14 +89,14 @@ class LoadingWorker(val context: Context, params: WorkerParameters) : Worker(con | |||||||
|                 val newItemsNotification = |                 val newItemsNotification = | ||||||
|                     NotificationCompat.Builder( |                     NotificationCompat.Builder( | ||||||
|                         applicationContext, |                         applicationContext, | ||||||
|                         AppSettingsService.newItemsChannelId |                         AppSettingsService.newItemsChannelId, | ||||||
|                     ) |                     ) | ||||||
|                         .setContentTitle(context.getString(R.string.new_items_notification_title)) |                         .setContentTitle(context.getString(R.string.new_items_notification_title)) | ||||||
|                         .setContentText( |                         .setContentText( | ||||||
|                             context.getString( |                             context.getString( | ||||||
|                                 R.string.new_items_notification_text, |                                 R.string.new_items_notification_text, | ||||||
|                                 newSize |                                 newSize, | ||||||
|                             ) |                             ), | ||||||
|                         ) |                         ) | ||||||
|                         .setPriority(PRIORITY_DEFAULT) |                         .setPriority(PRIORITY_DEFAULT) | ||||||
|                         .setChannelId(AppSettingsService.newItemsChannelId) |                         .setChannelId(AppSettingsService.newItemsChannelId) | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.android.fragments | package bou.amine.apps.readerforselfossv2.android.fragments | ||||||
|  |  | ||||||
|  | import android.content.ActivityNotFoundException | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.content.res.ColorStateList | import android.content.res.ColorStateList | ||||||
| import android.content.res.TypedArray | import android.content.res.TypedArray | ||||||
| @@ -26,11 +27,11 @@ import bou.amine.apps.readerforselfossv2.android.model.toModel | |||||||
| import bou.amine.apps.readerforselfossv2.android.model.toParcelable | import bou.amine.apps.readerforselfossv2.android.model.toParcelable | ||||||
| import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.getBitmapInputStream | import bou.amine.apps.readerforselfossv2.android.utils.glide.getBitmapInputStream | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.utils.isUrlValid | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||||
| import bou.amine.apps.readerforselfossv2.model.MercuryModel | import bou.amine.apps.readerforselfossv2.model.MercuryModel | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.model.StatusAndData |  | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import bou.amine.apps.readerforselfossv2.rest.MercuryApi | import bou.amine.apps.readerforselfossv2.rest.MercuryApi | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| @@ -51,12 +52,10 @@ import org.kodein.di.DIAware | |||||||
| import org.kodein.di.android.x.closestDI | import org.kodein.di.android.x.closestDI | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
| import java.net.MalformedURLException | import java.net.MalformedURLException | ||||||
| import java.net.SocketTimeoutException |  | ||||||
| import java.net.URL | import java.net.URL | ||||||
| import java.util.* | import java.util.* | ||||||
| import java.util.concurrent.ExecutionException | import java.util.concurrent.ExecutionException | ||||||
|  |  | ||||||
|  |  | ||||||
| private const val IMAGE_JPG = "image/jpg" | private const val IMAGE_JPG = "image/jpg" | ||||||
|  |  | ||||||
| class ArticleFragment : Fragment(), DIAware { | class ArticleFragment : Fragment(), DIAware { | ||||||
| @@ -83,7 +82,6 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|  |  | ||||||
|     private val mercuryApi: MercuryApi by instance() |     private val mercuryApi: MercuryApi by instance() | ||||||
|  |  | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|  |  | ||||||
| @@ -95,7 +93,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|     override fun onCreateView( |     override fun onCreateView( | ||||||
|         inflater: LayoutInflater, |         inflater: LayoutInflater, | ||||||
|         container: ViewGroup?, |         container: ViewGroup?, | ||||||
|         savedInstanceState: Bundle? |         savedInstanceState: Bundle?, | ||||||
|     ): View { |     ): View { | ||||||
|         try { |         try { | ||||||
|             binding = FragmentArticleBinding.inflate(inflater, container, false) |             binding = FragmentArticleBinding.inflate(inflater, container, false) | ||||||
| @@ -104,7 +102,12 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|             contentText = item.content |             contentText = item.content | ||||||
|             contentTitle = item.title.getHtmlDecoded() |             contentTitle = item.title.getHtmlDecoded() | ||||||
|             contentImage = item.getThumbnail(repository.baseUrl) |             contentImage = item.getThumbnail(repository.baseUrl) | ||||||
|             contentSource = item.sourceAuthorAndDate() |             contentSource = try { | ||||||
|  |                 item.sourceAuthorAndDate() | ||||||
|  |             } catch (e: Exception) { | ||||||
|  |                 e.sendSilentlyWithAcraWithName("Article Fragment parse date") | ||||||
|  |                 item.sourceAuthorOnly() | ||||||
|  |             } | ||||||
|             allImages = item.getImages() |             allImages = item.getImages() | ||||||
|  |  | ||||||
|             fontSize = appSettingsService.getFontSize() |             fontSize = appSettingsService.getFontSize() | ||||||
| @@ -145,16 +148,16 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                             if (floatingToolbar.isShowing) floatingToolbar.hide() else fab.show() |                             if (floatingToolbar.isShowing) floatingToolbar.hide() else fab.show() | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 } |                 }, | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         } catch (e: InflateException) { |         } catch (e: InflateException) { | ||||||
|             e.sendSilentlyWithAcraWithName("webview not available") |             e.sendSilentlyWithAcraWithName("webview not available") | ||||||
|  |             if (context != null) { | ||||||
|                 AlertDialog.Builder(requireContext()) |                 AlertDialog.Builder(requireContext()) | ||||||
|                     .setMessage(requireContext().getString(R.string.webview_dialog_issue_message)) |                     .setMessage(requireContext().getString(R.string.webview_dialog_issue_message)) | ||||||
|                     .setTitle(requireContext().getString(R.string.webview_dialog_issue_title)) |                     .setTitle(requireContext().getString(R.string.webview_dialog_issue_title)) | ||||||
|                     .setPositiveButton( |                     .setPositiveButton( | ||||||
|                     android.R.string.ok |                         android.R.string.ok, | ||||||
|                     ) { _, _ -> |                     ) { _, _ -> | ||||||
|                         appSettingsService.disableArticleViewer() |                         appSettingsService.disableArticleViewer() | ||||||
|                         requireActivity().finish() |                         requireActivity().finish() | ||||||
| @@ -162,6 +165,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                     .create() |                     .create() | ||||||
|                     .show() |                     .show() | ||||||
|             } |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return binding.root |         return binding.root | ||||||
|     } |     } | ||||||
| @@ -208,7 +212,8 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                     when (item.itemId) { |                     when (item.itemId) { | ||||||
|                         R.id.share_action -> requireActivity().shareLink(url, contentTitle) |                         R.id.share_action -> requireActivity().shareLink(url, contentTitle) | ||||||
|                         R.id.open_action -> requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) |                         R.id.open_action -> requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) | ||||||
|                         R.id.unread_action -> if (context != null) { |                         R.id.unread_action -> | ||||||
|  |                             if (context != null) { | ||||||
|                                 if (this@ArticleFragment.item.unread) { |                                 if (this@ArticleFragment.item.unread) { | ||||||
|                                     CoroutineScope(Dispatchers.IO).launch { |                                     CoroutineScope(Dispatchers.IO).launch { | ||||||
|                                         repository.markAsRead(this@ArticleFragment.item) |                                         repository.markAsRead(this@ArticleFragment.item) | ||||||
| @@ -217,7 +222,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                                     Toast.makeText( |                                     Toast.makeText( | ||||||
|                                         context, |                                         context, | ||||||
|                                         R.string.marked_as_read, |                                         R.string.marked_as_read, | ||||||
|                                     Toast.LENGTH_LONG |                                         Toast.LENGTH_LONG, | ||||||
|                                     ).show() |                                     ).show() | ||||||
|                                 } else { |                                 } else { | ||||||
|                                     CoroutineScope(Dispatchers.IO).launch { |                                     CoroutineScope(Dispatchers.IO).launch { | ||||||
| @@ -227,7 +232,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                                     Toast.makeText( |                                     Toast.makeText( | ||||||
|                                         context, |                                         context, | ||||||
|                                         R.string.marked_as_unread, |                                         R.string.marked_as_unread, | ||||||
|                                     Toast.LENGTH_LONG |                                         Toast.LENGTH_LONG, | ||||||
|                                     ).show() |                                     ).show() | ||||||
|                                 } |                                 } | ||||||
|                             } |                             } | ||||||
| @@ -238,13 +243,14 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 override fun onItemLongClick(item: MenuItem?) { |                 override fun onItemLongClick(item: MenuItem?) { | ||||||
|                     // We do nothing |                     // We do nothing | ||||||
|                 } |                 } | ||||||
|             } |             }, | ||||||
|         ) |         ) | ||||||
|         return floatingToolbar |         return floatingToolbar | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshAlignment() { |     private fun refreshAlignment() { | ||||||
|         textAlignment = when (appSettingsService.getActiveAllignment()) { |         textAlignment = | ||||||
|  |             when (appSettingsService.getActiveAllignment()) { | ||||||
|                 1 -> "justify" |                 1 -> "justify" | ||||||
|                 2 -> "left" |                 2 -> "left" | ||||||
|                 else -> "justify" |                 else -> "justify" | ||||||
| @@ -257,41 +263,46 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|             try { |             try { | ||||||
|                 val response = mercuryApi.query(url) |                 val response = mercuryApi.query(url) | ||||||
|                 if (response.success && response.data != null && !response.data?.content.isNullOrEmpty()) { |                 if (response.success && response.data != null) { | ||||||
|                     binding.titleView.text = response.data!!.title.orEmpty() |                     handleMercuryData(response.data!!) | ||||||
|                     if (typeface != null) { |  | ||||||
|                         binding.titleView.typeface = typeface |  | ||||||
|                     } |  | ||||||
|                     URL(response.data!!.url) |  | ||||||
|                     url = response.data!!.url |  | ||||||
|  |  | ||||||
|                     contentText = response.data!!.content.orEmpty() |  | ||||||
|                     htmlToWebview() |  | ||||||
|  |  | ||||||
|                     handleLeadImage(response) |  | ||||||
|  |  | ||||||
|                     binding.nestedScrollView.scrollTo(0, 0) |  | ||||||
|                     binding.progressBar.visibility = View.GONE |  | ||||||
|                 } else { |                 } else { | ||||||
|                     openInBrowserAfterFailing() |                     openInBrowserAfterFailing() | ||||||
|                 } |                 } | ||||||
|             } catch (e: SocketTimeoutException) { |  | ||||||
|                 openInBrowserAfterFailing() |  | ||||||
|             } catch (e: Exception) { |             } catch (e: Exception) { | ||||||
|                 e.sendSilentlyWithAcraWithName("getContentFromMercury > $url") |  | ||||||
|                 openInBrowserAfterFailing() |                 openInBrowserAfterFailing() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleLeadImage(response: StatusAndData<MercuryModel.ParsedContent>) { |     private fun handleMercuryData(data: MercuryModel.ParsedContent) { | ||||||
|         if (!response.data?.lead_image_url.isNullOrEmpty() && context != null) { |         if (data.error == true || data.failed == true) { | ||||||
|  |             openInBrowserAfterFailing() | ||||||
|  |         } else { | ||||||
|  |             binding.titleView.text = data.title.orEmpty() | ||||||
|  |             if (typeface != null) { | ||||||
|  |                 binding.titleView.typeface = typeface | ||||||
|  |             } | ||||||
|  |             URL(data.url) | ||||||
|  |             url = data.url!! | ||||||
|  |  | ||||||
|  |             contentText = data.content.orEmpty() | ||||||
|  |             htmlToWebview() | ||||||
|  |  | ||||||
|  |             handleLeadImage(data?.lead_image_url) | ||||||
|  |  | ||||||
|  |             binding.nestedScrollView.scrollTo(0, 0) | ||||||
|  |             binding.progressBar.visibility = View.GONE | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun handleLeadImage(lead_image_url: String?) { | ||||||
|  |         if (!lead_image_url.isNullOrEmpty() && context != null) { | ||||||
|             binding.imageView.visibility = View.VISIBLE |             binding.imageView.visibility = View.VISIBLE | ||||||
|             Glide |             Glide | ||||||
|                 .with(requireContext()) |                 .with(requireContext()) | ||||||
|                 .asBitmap() |                 .asBitmap() | ||||||
|                 .load( |                 .load( | ||||||
|                     response.data!!.lead_image_url.orEmpty() |                     lead_image_url, | ||||||
|                 ) |                 ) | ||||||
|                 .apply(RequestOptions.fitCenterTransform()) |                 .apply(RequestOptions.fitCenterTransform()) | ||||||
|                 .into(binding.imageView) |                 .into(binding.imageView) | ||||||
| @@ -301,19 +312,33 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleImageLoading() { |     private fun handleImageLoading() { | ||||||
|         binding.webcontent.webViewClient = object : WebViewClient() { |         binding.webcontent.webViewClient = | ||||||
|  |             object : WebViewClient() { | ||||||
|                 @Deprecated("Deprecated in Java") |                 @Deprecated("Deprecated in Java") | ||||||
|             override fun shouldOverrideUrlLoading(view: WebView?, url: String): Boolean { |                 override fun shouldOverrideUrlLoading( | ||||||
|                 if (binding.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { |                     view: WebView?, | ||||||
|  |                     url: String, | ||||||
|  |                 ): Boolean { | ||||||
|  |                     return if (context != null && url.isUrlValid() && binding.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||||
|  |                         try { | ||||||
|                             requireContext().startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) |                             requireContext().startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) | ||||||
|  |                         } catch (e: ActivityNotFoundException) { | ||||||
|  |                             e.sendSilentlyWithAcraWithName("activityNotFound > $url") | ||||||
|  |                         } | ||||||
|  |                         true | ||||||
|  |                     } else { | ||||||
|  |                         false | ||||||
|                     } |                     } | ||||||
|                 return true |  | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 @Deprecated("Deprecated in Java") |                 @Deprecated("Deprecated in Java") | ||||||
|             override fun shouldInterceptRequest(view: WebView, url: String): WebResourceResponse? { |                 override fun shouldInterceptRequest( | ||||||
|  |                     view: WebView, | ||||||
|  |                     url: String, | ||||||
|  |                 ): WebResourceResponse? { | ||||||
|                     val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) |                     val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|                 if (url.lowercase(Locale.US).contains(".jpg") || url.lowercase(Locale.US) |                     if (url.lowercase(Locale.US).contains(".jpg") || | ||||||
|  |                         url.lowercase(Locale.US) | ||||||
|                             .contains(".jpeg") |                             .contains(".jpeg") | ||||||
|                     ) { |                     ) { | ||||||
|                         try { |                         try { | ||||||
| @@ -322,10 +347,10 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                             return WebResourceResponse( |                             return WebResourceResponse( | ||||||
|                                 IMAGE_JPG, |                                 IMAGE_JPG, | ||||||
|                                 "UTF-8", |                                 "UTF-8", | ||||||
|                             getBitmapInputStream(image, Bitmap.CompressFormat.JPEG) |                                 getBitmapInputStream(image, Bitmap.CompressFormat.JPEG), | ||||||
|                             ) |                             ) | ||||||
|                         } catch (e: ExecutionException) { |                         } catch (e: ExecutionException) { | ||||||
|                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > jpeg > $url") |                             // Do nothing | ||||||
|                         } |                         } | ||||||
|                     } else if (url.lowercase(Locale.US).contains(".png")) { |                     } else if (url.lowercase(Locale.US).contains(".png")) { | ||||||
|                         try { |                         try { | ||||||
| @@ -334,10 +359,10 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                             return WebResourceResponse( |                             return WebResourceResponse( | ||||||
|                                 IMAGE_JPG, |                                 IMAGE_JPG, | ||||||
|                                 "UTF-8", |                                 "UTF-8", | ||||||
|                             getBitmapInputStream(image, Bitmap.CompressFormat.PNG) |                                 getBitmapInputStream(image, Bitmap.CompressFormat.PNG), | ||||||
|                             ) |                             ) | ||||||
|                         } catch (e: ExecutionException) { |                         } catch (e: ExecutionException) { | ||||||
|                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > png > $url") |                             // Do nothing | ||||||
|                         } |                         } | ||||||
|                     } else if (url.lowercase(Locale.US).contains(".webp")) { |                     } else if (url.lowercase(Locale.US).contains(".webp")) { | ||||||
|                         try { |                         try { | ||||||
| @@ -346,10 +371,10 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                             return WebResourceResponse( |                             return WebResourceResponse( | ||||||
|                                 IMAGE_JPG, |                                 IMAGE_JPG, | ||||||
|                                 "UTF-8", |                                 "UTF-8", | ||||||
|                             getBitmapInputStream(image, Bitmap.CompressFormat.WEBP) |                                 getBitmapInputStream(image, Bitmap.CompressFormat.WEBP), | ||||||
|                             ) |                             ) | ||||||
|                         } catch (e: ExecutionException) { |                         } catch (e: ExecutionException) { | ||||||
|                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > webp > $url") |                             // Do nothing | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
| @@ -359,11 +384,10 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun htmlToWebview() { |     private fun htmlToWebview() { | ||||||
|  |         if (context != null) { | ||||||
|             val attrs: IntArray = intArrayOf(android.R.attr.fontFamily) |             val attrs: IntArray = intArrayOf(android.R.attr.fontFamily) | ||||||
|             val a: TypedArray = requireContext().obtainStyledAttributes(resId, attrs) |             val a: TypedArray = requireContext().obtainStyledAttributes(resId, attrs) | ||||||
|  |  | ||||||
|  |  | ||||||
|             binding.webcontent.settings.standardFontFamily = a.getString(0) |             binding.webcontent.settings.standardFontFamily = a.getString(0) | ||||||
|             binding.webcontent.visibility = View.VISIBLE |             binding.webcontent.visibility = View.VISIBLE | ||||||
|  |  | ||||||
| @@ -380,11 +404,14 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|             handleImageLoading() |             handleImageLoading() | ||||||
|  |  | ||||||
|             val gestureDetector = |             val gestureDetector = | ||||||
|             GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { |                 GestureDetector( | ||||||
|  |                     activity, | ||||||
|  |                     object : GestureDetector.SimpleOnGestureListener() { | ||||||
|                         override fun onSingleTapUp(e: MotionEvent): Boolean { |                         override fun onSingleTapUp(e: MotionEvent): Boolean { | ||||||
|                             return performClick() |                             return performClick() | ||||||
|                         } |                         } | ||||||
|             }) |                     }, | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|             binding.webcontent.setOnTouchListener { _, event -> gestureDetector.onTouchEvent(event) } |             binding.webcontent.setOnTouchListener { _, event -> gestureDetector.onTouchEvent(event) } | ||||||
|  |  | ||||||
| @@ -397,21 +424,23 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 val itemUrl = URL(url) |                 val itemUrl = URL(url) | ||||||
|                 baseUrl = itemUrl.protocol + "://" + itemUrl.host |                 baseUrl = itemUrl.protocol + "://" + itemUrl.host | ||||||
|             } catch (e: MalformedURLException) { |             } catch (e: MalformedURLException) { | ||||||
|             e.sendSilentlyWithAcraWithName("htmlToWebview > item url") |                 e.sendSilentlyWithAcraWithName("htmlToWebview > $url") | ||||||
|             } |             } | ||||||
|  |  | ||||||
|         val fontName = when (font) { |             val fontName = | ||||||
|  |                 when (font) { | ||||||
|                     getString(R.string.open_sans_font_id) -> "Open Sans" |                     getString(R.string.open_sans_font_id) -> "Open Sans" | ||||||
|                     getString(R.string.roboto_font_id) -> "Roboto" |                     getString(R.string.roboto_font_id) -> "Roboto" | ||||||
|                     getString(R.string.source_code_pro_font_id) -> "Source Code Pro" |                     getString(R.string.source_code_pro_font_id) -> "Source Code Pro" | ||||||
|                     else -> "" |                     else -> "" | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|         val fontLinkAndStyle = if (font.isNotEmpty()) { |             val fontLinkAndStyle = | ||||||
|  |                 if (font.isNotEmpty()) { | ||||||
|                     """<link href="https://fonts.googleapis.com/css?family=${ |                     """<link href="https://fonts.googleapis.com/css?family=${ | ||||||
|                         fontName.replace( |                         fontName.replace( | ||||||
|                             " ", |                             " ", | ||||||
|                     "+" |                             "+", | ||||||
|                         ) |                         ) | ||||||
|                     }" rel="stylesheet"> |                     }" rel="stylesheet"> | ||||||
|                 |<style> |                 |<style> | ||||||
| @@ -440,7 +469,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 |        color: ${ |                 |        color: ${ | ||||||
|                     String.format( |                     String.format( | ||||||
|                         "#%06X", |                         "#%06X", | ||||||
|                     0xFFFFFF and resources.getColor(R.color.colorAccent) |                         0xFFFFFF and resources.getColor(R.color.colorAccent), | ||||||
|                     ) |                     ) | ||||||
|                 } !important; |                 } !important; | ||||||
|                 |      } |                 |      } | ||||||
| @@ -456,7 +485,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 |        background-color: ${ |                 |        background-color: ${ | ||||||
|                     String.format( |                     String.format( | ||||||
|                         "#%06X", |                         "#%06X", | ||||||
|                     0xFFFFFF and colorSurface.data |                         0xFFFFFF and colorSurface.data, | ||||||
|                     ) |                     ) | ||||||
|                 }; |                 }; | ||||||
|                 |      } |                 |      } | ||||||
| @@ -464,13 +493,13 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 |        background-color: ${ |                 |        background-color: ${ | ||||||
|                     String.format( |                     String.format( | ||||||
|                         "#%06X", |                         "#%06X", | ||||||
|                     0xFFFFFF and colorSurface.data |                         0xFFFFFF and colorSurface.data, | ||||||
|                     ) |                     ) | ||||||
|                 } !important; |                 } !important; | ||||||
|                 |        border-color: ${ |                 |        border-color: ${ | ||||||
|                     String.format( |                     String.format( | ||||||
|                         "#%06X", |                         "#%06X", | ||||||
|                     0xFFFFFF and colorSurface.data |                         0xFFFFFF and colorSurface.data, | ||||||
|                     ) |                     ) | ||||||
|                 }  !important; |                 }  !important; | ||||||
|                 |        padding: 0 !important; |                 |        padding: 0 !important; | ||||||
| @@ -485,7 +514,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 |        background-color: ${ |                 |        background-color: ${ | ||||||
|                     String.format( |                     String.format( | ||||||
|                         "#%06X", |                         "#%06X", | ||||||
|                     0xFFFFFF and colorSurface.data |                         0xFFFFFF and colorSurface.data, | ||||||
|                     ) |                     ) | ||||||
|                 }; |                 }; | ||||||
|                 |      } |                 |      } | ||||||
| @@ -494,12 +523,14 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 |</head> |                 |</head> | ||||||
|                 |<body> |                 |<body> | ||||||
|                 |   $contentText |                 |   $contentText | ||||||
|                 |</body>""".trimMargin(), |                 |</body> | ||||||
|  |                 """.trimMargin(), | ||||||
|                 "text/html", |                 "text/html", | ||||||
|                 "utf-8", |                 "utf-8", | ||||||
|             null |                 null, | ||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     fun scrollDown() { |     fun scrollDown() { | ||||||
|         val height = binding.nestedScrollView.measuredHeight |         val height = binding.nestedScrollView.measuredHeight | ||||||
| @@ -513,15 +544,17 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|  |  | ||||||
|     private fun openInBrowserAfterFailing() { |     private fun openInBrowserAfterFailing() { | ||||||
|         binding.progressBar.visibility = View.GONE |         binding.progressBar.visibility = View.GONE | ||||||
|         requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) |         if (context != null) { | ||||||
|  |             requireContext().openInBrowserAsNewTask(this@ArticleFragment.item) | ||||||
|  |         } else { | ||||||
|  |             Exception("openInBrowserAfterFailing context is null").sendSilentlyWithAcraWithName("openInBrowserAfterFailing > $context") | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     companion object { |     companion object { | ||||||
|         private const val ARG_ITEMS = "items" |         private const val ARG_ITEMS = "items" | ||||||
|  |  | ||||||
|         fun newInstance( |         fun newInstance(item: SelfossModel.Item): ArticleFragment { | ||||||
|             item: SelfossModel.Item |  | ||||||
|         ): ArticleFragment { |  | ||||||
|             val fragment = ArticleFragment() |             val fragment = ArticleFragment() | ||||||
|             val args = Bundle() |             val args = Bundle() | ||||||
|             args.putParcelable(ARG_ITEMS, item.toParcelable()) |             args.putParcelable(ARG_ITEMS, item.toParcelable()) | ||||||
| @@ -531,10 +564,11 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun performClick(): Boolean { |     fun performClick(): Boolean { | ||||||
|         if (binding.webcontent.hitTestResult.type == WebView.HitTestResult.IMAGE_TYPE || |         if (allImages != null && ( | ||||||
|  |                 binding.webcontent.hitTestResult.type == WebView.HitTestResult.IMAGE_TYPE || | ||||||
|                     binding.webcontent.hitTestResult.type == WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE |                     binding.webcontent.hitTestResult.type == WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE | ||||||
|  |             ) | ||||||
|         ) { |         ) { | ||||||
|  |  | ||||||
|             val position: Int = allImages.indexOf(binding.webcontent.hitTestResult.extra) |             val position: Int = allImages.indexOf(binding.webcontent.hitTestResult.extra) | ||||||
|  |  | ||||||
|             val intent = Intent(activity, ImageActivity::class.java) |             val intent = Intent(activity, ImageActivity::class.java) | ||||||
| @@ -545,6 +579,4 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|         } |         } | ||||||
|         return false |         return false | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -17,13 +17,12 @@ import bou.amine.apps.readerforselfossv2.android.R | |||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.FilterFragmentBinding | import bou.amine.apps.readerforselfossv2.android.databinding.FilterFragmentBinding | ||||||
| import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
|  | import bou.amine.apps.readerforselfossv2.utils.getColorHexCode | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
| import com.bumptech.glide.load.DataSource | import com.bumptech.glide.request.target.ViewTarget | ||||||
| import com.bumptech.glide.load.engine.GlideException | import com.bumptech.glide.request.transition.Transition | ||||||
| import com.bumptech.glide.request.RequestListener |  | ||||||
| import com.bumptech.glide.request.target.Target |  | ||||||
| import com.google.android.material.bottomsheet.BottomSheetDialogFragment | import com.google.android.material.bottomsheet.BottomSheetDialogFragment | ||||||
| import com.google.android.material.chip.Chip | import com.google.android.material.chip.Chip | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| @@ -34,9 +33,7 @@ import org.kodein.di.DIAware | |||||||
| import org.kodein.di.android.x.closestDI | import org.kodein.di.android.x.closestDI | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|  |  | ||||||
|  |  | ||||||
| class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | ||||||
|  |  | ||||||
|     private lateinit var binding: FilterFragmentBinding |     private lateinit var binding: FilterFragmentBinding | ||||||
|     override val di: DI by closestDI() |     override val di: DI by closestDI() | ||||||
|     private val repository: Repository by instance() |     private val repository: Repository by instance() | ||||||
| @@ -46,18 +43,17 @@ class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | |||||||
|     override fun onCreateView( |     override fun onCreateView( | ||||||
|         inflater: LayoutInflater, |         inflater: LayoutInflater, | ||||||
|         container: ViewGroup?, |         container: ViewGroup?, | ||||||
|         savedInstanceState: Bundle? |         savedInstanceState: Bundle?, | ||||||
|     ): View { |     ): View { | ||||||
|         binding = |         binding = | ||||||
|             FilterFragmentBinding.inflate( |             FilterFragmentBinding.inflate( | ||||||
|                 inflater, |                 inflater, | ||||||
|                 container, |                 container, | ||||||
|                 false |                 false, | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         val context: Context? = context |         val context: Context? = context | ||||||
|  |  | ||||||
|  |  | ||||||
|         if (context == null) { |         if (context == null) { | ||||||
|             dismiss() |             dismiss() | ||||||
|             Exception("FilterSheetFragment context is null").sendSilentlyWithAcraWithName("FilterSheetFragment > onCreateView") |             Exception("FilterSheetFragment context is null").sendSilentlyWithAcraWithName("FilterSheetFragment > onCreateView") | ||||||
| @@ -79,42 +75,29 @@ class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | |||||||
|         return binding.root |         return binding.root | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private suspend fun handleSourceChips( |     private suspend fun handleSourceChips(context: Context) { | ||||||
|         context: Context |  | ||||||
|     ) { |  | ||||||
|         val sourceGroup = binding.sourcesGroup |         val sourceGroup = binding.sourcesGroup | ||||||
|  |  | ||||||
|         repository.getSources().forEach { source -> |         repository.getSourcesDetailsOrStats().forEachIndexed { _, source -> | ||||||
|             val c = Chip(context) |             val c = Chip(context) | ||||||
|             c.ellipsize = TextUtils.TruncateAt.END |             c.ellipsize = TextUtils.TruncateAt.END | ||||||
|  |  | ||||||
|             Glide.with(context) |             Glide.with(context) | ||||||
|                 .load(source.getIcon(repository.baseUrl)) |                 .load(source.getIcon(repository.baseUrl)) | ||||||
|                 .listener(object : RequestListener<Drawable?> { |                 .into( | ||||||
|                     override fun onLoadFailed( |                     object : ViewTarget<Chip?, Drawable?>(c) { | ||||||
|                         e: GlideException?, |  | ||||||
|                         model: Any?, |  | ||||||
|                         target: Target<Drawable?>?, |  | ||||||
|                         isFirstResource: Boolean |  | ||||||
|                     ): Boolean { |  | ||||||
|                         return false |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                         override fun onResourceReady( |                         override fun onResourceReady( | ||||||
|                         resource: Drawable?, |                             resource: Drawable, | ||||||
|                         model: Any?, |                             transition: Transition<in Drawable?>?, | ||||||
|                         target: Target<Drawable?>?, |                         ) { | ||||||
|                         dataSource: DataSource?, |  | ||||||
|                         isFirstResource: Boolean |  | ||||||
|                     ): Boolean { |  | ||||||
|                             try { |                             try { | ||||||
|                                 c.chipIcon = resource |                                 c.chipIcon = resource | ||||||
|                             } catch (e: Exception) { |                             } catch (e: Exception) { | ||||||
|                                 e.sendSilentlyWithAcraWithName("sources > onResourceReady") |                                 e.sendSilentlyWithAcraWithName("sources > onResourceReady") | ||||||
|                             } |                             } | ||||||
|                         return false |  | ||||||
|                         } |                         } | ||||||
|                 }).preload() |                     }, | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|             c.text = source.title.getHtmlDecoded() |             c.text = source.title.getHtmlDecoded() | ||||||
|  |  | ||||||
| @@ -135,15 +118,14 @@ class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | |||||||
|                 repository.setTagFilter(null) |                 repository.setTagFilter(null) | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |  | ||||||
|             if (repository.sourceFilter.value?.equals(source) == true) { |             if (repository.sourceFilter.value?.equals(source) == true) { | ||||||
|                 c.isCloseIconVisible = true |                 c.isCloseIconVisible = true | ||||||
|                 selectedChip = c |                 selectedChip = c | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             c.isEnabled = source.error.isBlank() |             c.isEnabled = source.error.isNullOrBlank() | ||||||
|  |  | ||||||
|             if (source.error.isNotBlank() && Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { |             if (!source.error.isNullOrBlank() && Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { | ||||||
|                 c.tooltipText = source.error |                 c.tooltipText = source.error | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -151,24 +133,24 @@ class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private suspend fun handleTagChips( |     private suspend fun handleTagChips(context: Context) { | ||||||
|         context: Context, |  | ||||||
|     ) { |  | ||||||
|         val tagGroup = binding.tagsGroup |         val tagGroup = binding.tagsGroup | ||||||
|  |  | ||||||
|         val tags = repository.getTags() |         val tags = repository.getTags() | ||||||
|  |  | ||||||
|         tags.forEach { tag -> |         tags.forEachIndexed { _, tag -> | ||||||
|             val c = Chip(context) |             val c = Chip(context) | ||||||
|             c.ellipsize = TextUtils.TruncateAt.END |             c.ellipsize = TextUtils.TruncateAt.END | ||||||
|             c.text = tag.tag |             c.text = tag.tag | ||||||
|  |  | ||||||
|  |             if (tag.color.isNotEmpty()) { | ||||||
|                 try { |                 try { | ||||||
|                     val gd = GradientDrawable() |                     val gd = GradientDrawable() | ||||||
|                 val gdColor = try { |                     val gdColor = | ||||||
|                     Color.parseColor(tag.color) |                         try { | ||||||
|  |                             Color.parseColor(tag.getColorHexCode()) | ||||||
|                         } catch (e: IllegalArgumentException) { |                         } catch (e: IllegalArgumentException) { | ||||||
|                     e.sendSilentlyWithAcraWithName("color issue " + tag.color) |                             e.sendSilentlyWithAcraWithName("color issue " + tag.color + " / " + tag.getColorHexCode()) | ||||||
|                             resources.getColor(R.color.colorPrimary) |                             resources.getColor(R.color.colorPrimary) | ||||||
|                         } |                         } | ||||||
|                     gd.setColor(gdColor) |                     gd.setColor(gdColor) | ||||||
| @@ -179,6 +161,7 @@ class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | |||||||
|                 } catch (e: Exception) { |                 } catch (e: Exception) { | ||||||
|                     e.sendSilentlyWithAcraWithName("tags > GradientDrawable") |                     e.sendSilentlyWithAcraWithName("tags > GradientDrawable") | ||||||
|                 } |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|             c.setOnCloseIconClickListener { |             c.setOnCloseIconClickListener { | ||||||
|                 (it as Chip).isCloseIconVisible = false |                 (it as Chip).isCloseIconVisible = false | ||||||
| @@ -209,6 +192,4 @@ class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | |||||||
|     companion object { |     companion object { | ||||||
|         const val TAG = "FilterModalBottomSheet" |         const val TAG = "FilterModalBottomSheet" | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -11,8 +11,7 @@ import com.bumptech.glide.load.engine.DiskCacheStrategy | |||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
|  |  | ||||||
| class ImageFragment : Fragment() { | class ImageFragment : Fragment() { | ||||||
|  |     private lateinit var imageUrl: String | ||||||
|     private lateinit var imageUrl : String |  | ||||||
|     private val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) |     private val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|     private var _binding: FragmentImageBinding? = null |     private var _binding: FragmentImageBinding? = null | ||||||
|     private val binding get() = _binding |     private val binding get() = _binding | ||||||
| @@ -23,7 +22,11 @@ class ImageFragment : Fragment() { | |||||||
|         imageUrl = requireArguments().getString("imageUrl")!! |         imageUrl = requireArguments().getString("imageUrl")!! | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { |     override fun onCreateView( | ||||||
|  |         inflater: LayoutInflater, | ||||||
|  |         container: ViewGroup?, | ||||||
|  |         savedInstanceState: Bundle?, | ||||||
|  |     ): View? { | ||||||
|         _binding = FragmentImageBinding.inflate(inflater, container, false) |         _binding = FragmentImageBinding.inflate(inflater, container, false) | ||||||
|         val view = binding?.root |         val view = binding?.root | ||||||
|  |  | ||||||
| @@ -45,9 +48,7 @@ class ImageFragment : Fragment() { | |||||||
|     companion object { |     companion object { | ||||||
|         private const val ARG_IMAGE = "imageUrl" |         private const val ARG_IMAGE = "imageUrl" | ||||||
|  |  | ||||||
|         fun newInstance( |         fun newInstance(imageUrl: String): ImageFragment { | ||||||
|                 imageUrl : String |  | ||||||
|         ): ImageFragment { |  | ||||||
|             val fragment = ImageFragment() |             val fragment = ImageFragment() | ||||||
|             val args = Bundle() |             val args = Bundle() | ||||||
|             args.putString(ARG_IMAGE, imageUrl) |             args.putString(ARG_IMAGE, imageUrl) | ||||||
|   | |||||||
| @@ -9,21 +9,20 @@ import com.bumptech.glide.Glide | |||||||
| import com.bumptech.glide.load.engine.DiskCacheStrategy | import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
|  |  | ||||||
| fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | fun SelfossModel.Item.preloadImages(context: Context): Boolean { | ||||||
|     val imageUrls = this.getImages() |     val imageUrls = this.getImages() | ||||||
|  |  | ||||||
|     val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL).timeout(10000) |     val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL).timeout(10000) | ||||||
|  |  | ||||||
|  |  | ||||||
|     try { |     try { | ||||||
|         for (url in imageUrls) { |         for (url in imageUrls) { | ||||||
|             if ( URLUtil.isValidUrl(url)) { |             if (URLUtil.isValidUrl(url)) { | ||||||
|                 Glide.with(context).asBitmap() |                 Glide.with(context).asBitmap() | ||||||
|                     .apply(glideOptions) |                     .apply(glideOptions) | ||||||
|                     .load(url).submit() |                     .load(url).submit() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } catch (e : Error) { |     } catch (e: Error) { | ||||||
|         e.sendSilentlyWithAcraWithName("preloadImages") |         e.sendSilentlyWithAcraWithName("preloadImages") | ||||||
|         return false |         return false | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import android.os.Parcel | |||||||
| import android.os.Parcelable | import android.os.Parcelable | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
|  |  | ||||||
| fun SelfossModel.Item.toParcelable() : ParecelableItem = | fun SelfossModel.Item.toParcelable(): ParecelableItem = | ||||||
|     ParecelableItem( |     ParecelableItem( | ||||||
|         this.id, |         this.id, | ||||||
|         this.datetime, |         this.datetime, | ||||||
| @@ -17,9 +17,10 @@ fun SelfossModel.Item.toParcelable() : ParecelableItem = | |||||||
|         this.link, |         this.link, | ||||||
|         this.sourcetitle, |         this.sourcetitle, | ||||||
|         this.tags.joinToString(","), |         this.tags.joinToString(","), | ||||||
|         this.author |         this.author, | ||||||
|     ) |     ) | ||||||
| fun ParecelableItem.toModel() : SelfossModel.Item = |  | ||||||
|  | fun ParecelableItem.toModel(): SelfossModel.Item = | ||||||
|     SelfossModel.Item( |     SelfossModel.Item( | ||||||
|         this.id, |         this.id, | ||||||
|         this.datetime, |         this.datetime, | ||||||
| @@ -32,8 +33,9 @@ fun ParecelableItem.toModel() : SelfossModel.Item = | |||||||
|         this.link, |         this.link, | ||||||
|         this.sourcetitle, |         this.sourcetitle, | ||||||
|         this.tags.split(","), |         this.tags.split(","), | ||||||
|         this.author |         this.author, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
| data class ParecelableItem( | data class ParecelableItem( | ||||||
|     val id: Int, |     val id: Int, | ||||||
|     val datetime: String, |     val datetime: String, | ||||||
| @@ -46,13 +48,14 @@ data class ParecelableItem( | |||||||
|     val link: String, |     val link: String, | ||||||
|     val sourcetitle: String, |     val sourcetitle: String, | ||||||
|     val tags: String, |     val tags: String, | ||||||
|     val author: String? |     val author: String?, | ||||||
| ) : Parcelable { | ) : Parcelable { | ||||||
|  |  | ||||||
|     companion object { |     companion object { | ||||||
|         @JvmField |         @JvmField | ||||||
|         val CREATOR: Parcelable.Creator<ParecelableItem> = object : Parcelable.Creator<ParecelableItem> { |         val CREATOR: Parcelable.Creator<ParecelableItem> = | ||||||
|  |             object : Parcelable.Creator<ParecelableItem> { | ||||||
|                 override fun createFromParcel(source: Parcel): ParecelableItem = ParecelableItem(source) |                 override fun createFromParcel(source: Parcel): ParecelableItem = ParecelableItem(source) | ||||||
|  |  | ||||||
|                 override fun newArray(size: Int): Array<ParecelableItem?> = arrayOfNulls(size) |                 override fun newArray(size: Int): Array<ParecelableItem?> = arrayOfNulls(size) | ||||||
|             } |             } | ||||||
|     } |     } | ||||||
| @@ -69,12 +72,15 @@ data class ParecelableItem( | |||||||
|         link = source.readString().orEmpty(), |         link = source.readString().orEmpty(), | ||||||
|         sourcetitle = source.readString().orEmpty(), |         sourcetitle = source.readString().orEmpty(), | ||||||
|         tags = source.readString().orEmpty(), |         tags = source.readString().orEmpty(), | ||||||
|         author = source.readString().orEmpty() |         author = source.readString().orEmpty(), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     override fun describeContents() = 0 |     override fun describeContents() = 0 | ||||||
|  |  | ||||||
|     override fun writeToParcel(dest: Parcel, flags: Int) { |     override fun writeToParcel( | ||||||
|  |         dest: Parcel, | ||||||
|  |         flags: Int, | ||||||
|  |     ) { | ||||||
|         dest.writeInt(id) |         dest.writeInt(id) | ||||||
|         dest.writeString(datetime) |         dest.writeString(datetime) | ||||||
|         dest.writeString(title) |         dest.writeString(title) | ||||||
|   | |||||||
| @@ -24,8 +24,10 @@ import org.kodein.di.android.closestDI | |||||||
|  |  | ||||||
| private const val TITLE_TAG = "settingsActivityTitle" | private const val TITLE_TAG = "settingsActivityTitle" | ||||||
|  |  | ||||||
| class SettingsActivity : AppCompatActivity(), | class SettingsActivity : | ||||||
|         PreferenceFragmentCompat.OnPreferenceStartFragmentCallback, DIAware { |     AppCompatActivity(), | ||||||
|  |     PreferenceFragmentCompat.OnPreferenceStartFragmentCallback, | ||||||
|  |     DIAware { | ||||||
|     override val di by closestDI() |     override val di by closestDI() | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
| @@ -72,13 +74,14 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|  |  | ||||||
|     override fun onPreferenceStartFragment( |     override fun onPreferenceStartFragment( | ||||||
|         caller: PreferenceFragmentCompat, |         caller: PreferenceFragmentCompat, | ||||||
|             pref: Preference |         pref: Preference, | ||||||
|     ): Boolean { |     ): Boolean { | ||||||
|         // Instantiate the new Fragment |         // Instantiate the new Fragment | ||||||
|         val args = pref.extras |         val args = pref.extras | ||||||
|         val fragment = supportFragmentManager.fragmentFactory.instantiate( |         val fragment = | ||||||
|  |             supportFragmentManager.fragmentFactory.instantiate( | ||||||
|                 classLoader, |                 classLoader, | ||||||
|                 pref.fragment |                 pref.fragment.toString(), | ||||||
|             ).apply { |             ).apply { | ||||||
|                 arguments = args |                 arguments = args | ||||||
|                 setTargetFragment(caller, 0) |                 setTargetFragment(caller, 0) | ||||||
| @@ -94,15 +97,20 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     class MainPreferenceFragment : PreferenceFragmentCompat() { |     class MainPreferenceFragment : PreferenceFragmentCompat() { | ||||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { |         override fun onCreatePreferences( | ||||||
|  |             savedInstanceState: Bundle?, | ||||||
|  |             rootKey: String?, | ||||||
|  |         ) { | ||||||
|             setPreferencesFromResource(R.xml.pref_main, rootKey) |             setPreferencesFromResource(R.xml.pref_main, rootKey) | ||||||
|  |  | ||||||
|             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = Preference.OnPreferenceChangeListener { _, newValue -> |             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = | ||||||
|  |                 Preference.OnPreferenceChangeListener { _, newValue -> | ||||||
|                     AppCompatDelegate.setDefaultNightMode(newValue.toString().toInt()) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ |                     AppCompatDelegate.setDefaultNightMode(newValue.toString().toInt()) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ | ||||||
|                     true |                     true | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|             preferenceManager.findPreference<Preference>("action_about")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { _ -> |             preferenceManager.findPreference<Preference>("action_about")?.onPreferenceClickListener = | ||||||
|  |                 Preference.OnPreferenceClickListener { _ -> | ||||||
|                     context?.let { |                     context?.let { | ||||||
|                         LibsBuilder() |                         LibsBuilder() | ||||||
|                             .withAboutIconShown(true) |                             .withAboutIconShown(true) | ||||||
| @@ -115,13 +123,17 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     class GeneralPreferenceFragment : PreferenceFragmentCompat() { |     class GeneralPreferenceFragment : PreferenceFragmentCompat() { | ||||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { |         override fun onCreatePreferences( | ||||||
|  |             savedInstanceState: Bundle?, | ||||||
|  |             rootKey: String?, | ||||||
|  |         ) { | ||||||
|             setPreferencesFromResource(R.xml.pref_general, rootKey) |             setPreferencesFromResource(R.xml.pref_general, rootKey) | ||||||
|  |  | ||||||
|             val editTextPreference = preferenceManager.findPreference<EditTextPreference>("prefer_api_items_number") |             val editTextPreference = preferenceManager.findPreference<EditTextPreference>("prefer_api_items_number") | ||||||
|             editTextPreference?.setOnBindEditTextListener { editText -> |             editTextPreference?.setOnBindEditTextListener { editText -> | ||||||
|                 editText.inputType = InputType.TYPE_CLASS_NUMBER |                 editText.inputType = InputType.TYPE_CLASS_NUMBER | ||||||
|                 editText.filters = arrayOf( |                 editText.filters = | ||||||
|  |                     arrayOf( | ||||||
|                         InputFilter { source, _, _, dest, _, _ -> |                         InputFilter { source, _, _, dest, _, _ -> | ||||||
|                             try { |                             try { | ||||||
|                                 val input: Int = (dest.toString() + source.toString()).toInt() |                                 val input: Int = (dest.toString() + source.toString()).toInt() | ||||||
| @@ -131,26 +143,42 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|                                 Toast.makeText(activity, R.string.items_number_should_be_number, Toast.LENGTH_LONG).show() |                                 Toast.makeText(activity, R.string.items_number_should_be_number, Toast.LENGTH_LONG).show() | ||||||
|                             } |                             } | ||||||
|                             "" |                             "" | ||||||
|                         } |                         }, | ||||||
|                     ) |                     ) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     class ArticleViewerPreferenceFragment : PreferenceFragmentCompat() { |     class ArticleViewerPreferenceFragment : PreferenceFragmentCompat() { | ||||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { |         override fun onCreatePreferences( | ||||||
|  |             savedInstanceState: Bundle?, | ||||||
|  |             rootKey: String?, | ||||||
|  |         ) { | ||||||
|             setPreferencesFromResource(R.xml.pref_viewer, rootKey) |             setPreferencesFromResource(R.xml.pref_viewer, rootKey) | ||||||
|  |  | ||||||
|             val fontSize = preferenceManager.findPreference<EditTextPreference>("reader_font_size") |             val fontSize = preferenceManager.findPreference<EditTextPreference>("reader_font_size") | ||||||
|             fontSize?.setOnBindEditTextListener { editText -> |             fontSize?.setOnBindEditTextListener { editText -> | ||||||
|                 editText.inputType = InputType.TYPE_CLASS_NUMBER |                 editText.inputType = InputType.TYPE_CLASS_NUMBER | ||||||
|                 editText.addTextChangedListener { object : TextWatcher { |                 editText.addTextChangedListener { | ||||||
|                     override fun beforeTextChanged(charSequence: CharSequence, i: Int, i1: Int, i2: Int) { |                     object : TextWatcher { | ||||||
|  |                         override fun beforeTextChanged( | ||||||
|  |                             charSequence: CharSequence, | ||||||
|  |                             i: Int, | ||||||
|  |                             i1: Int, | ||||||
|  |                             i2: Int, | ||||||
|  |                         ) { | ||||||
|                             // We do nothing |                             // We do nothing | ||||||
|                         } |                         } | ||||||
|                     override fun onTextChanged(charSequence: CharSequence, i: Int, i1: Int, i2: Int) { |  | ||||||
|  |                         override fun onTextChanged( | ||||||
|  |                             charSequence: CharSequence, | ||||||
|  |                             i: Int, | ||||||
|  |                             i1: Int, | ||||||
|  |                             i2: Int, | ||||||
|  |                         ) { | ||||||
|                             // We do nothing |                             // We do nothing | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|                         override fun afterTextChanged(editable: Editable) { |                         override fun afterTextChanged(editable: Editable) { | ||||||
|                             try { |                             try { | ||||||
|                                 editText.textSize = editable.toString().toInt().toFloat() |                                 editText.textSize = editable.toString().toInt().toFloat() | ||||||
| @@ -158,8 +186,10 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|                                 e.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > afterTextChanged") |                                 e.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > afterTextChanged") | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                 } } |                     } | ||||||
|                 editText.filters = arrayOf( |                 } | ||||||
|  |                 editText.filters = | ||||||
|  |                     arrayOf( | ||||||
|                         InputFilter { source, _, _, dest, _, _ -> |                         InputFilter { source, _, _, dest, _, _ -> | ||||||
|                             try { |                             try { | ||||||
|                                 val input = (dest.toString() + source.toString()).toInt() |                                 val input = (dest.toString() + source.toString()).toInt() | ||||||
| @@ -168,23 +198,30 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|                                 nfe.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > filters") |                                 nfe.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > filters") | ||||||
|                             } |                             } | ||||||
|                             "" |                             "" | ||||||
|                         } |                         }, | ||||||
|                     ) |                     ) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     class OfflinePreferenceFragment : PreferenceFragmentCompat() { |     class OfflinePreferenceFragment : PreferenceFragmentCompat() { | ||||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { |         override fun onCreatePreferences( | ||||||
|  |             savedInstanceState: Bundle?, | ||||||
|  |             rootKey: String?, | ||||||
|  |         ) { | ||||||
|             setPreferencesFromResource(R.xml.pref_offline, rootKey) |             setPreferencesFromResource(R.xml.pref_offline, rootKey) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     class ThemePreferenceFragment : PreferenceFragmentCompat() { |     class ThemePreferenceFragment : PreferenceFragmentCompat() { | ||||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { |         override fun onCreatePreferences( | ||||||
|  |             savedInstanceState: Bundle?, | ||||||
|  |             rootKey: String?, | ||||||
|  |         ) { | ||||||
|             setPreferencesFromResource(R.xml.pref_theme, rootKey) |             setPreferencesFromResource(R.xml.pref_theme, rootKey) | ||||||
|  |  | ||||||
|             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = Preference.OnPreferenceChangeListener { _, newValue -> |             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = | ||||||
|  |                 Preference.OnPreferenceChangeListener { _, newValue -> | ||||||
|                     AppCompatDelegate.setDefaultNightMode(newValue.toString().toInt()) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ |                     AppCompatDelegate.setDefaultNightMode(newValue.toString().toInt()) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ | ||||||
|                     true |                     true | ||||||
|                 } |                 } | ||||||
| @@ -197,20 +234,26 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|             startActivity(browserIntent) |             startActivity(browserIntent) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { |         override fun onCreatePreferences( | ||||||
|  |             savedInstanceState: Bundle?, | ||||||
|  |             rootKey: String?, | ||||||
|  |         ) { | ||||||
|             setPreferencesFromResource(R.xml.pref_links, rootKey) |             setPreferencesFromResource(R.xml.pref_links, rootKey) | ||||||
|  |  | ||||||
|             preferenceManager.findPreference<Preference>("trackerLink")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { |             preferenceManager.findPreference<Preference>("trackerLink")?.onPreferenceClickListener = | ||||||
|  |                 Preference.OnPreferenceClickListener { | ||||||
|                     openUrl(Uri.parse(AppSettingsService.trackerUrl)) |                     openUrl(Uri.parse(AppSettingsService.trackerUrl)) | ||||||
|                     true |                     true | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|             preferenceManager.findPreference<Preference>("sourceLink")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { |             preferenceManager.findPreference<Preference>("sourceLink")?.onPreferenceClickListener = | ||||||
|  |                 Preference.OnPreferenceClickListener { | ||||||
|                     openUrl(Uri.parse(AppSettingsService.sourceUrl)) |                     openUrl(Uri.parse(AppSettingsService.sourceUrl)) | ||||||
|                     false |                     false | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|             preferenceManager.findPreference<Preference>("translation")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { |             preferenceManager.findPreference<Preference>("translation")?.onPreferenceClickListener = | ||||||
|  |                 Preference.OnPreferenceClickListener { | ||||||
|                     openUrl(Uri.parse(AppSettingsService.translationUrl)) |                     openUrl(Uri.parse(AppSettingsService.translationUrl)) | ||||||
|                     false |                     false | ||||||
|                 } |                 } | ||||||
| @@ -218,7 +261,10 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     class ExperimentalPreferenceFragment : PreferenceFragmentCompat() { |     class ExperimentalPreferenceFragment : PreferenceFragmentCompat() { | ||||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { |         override fun onCreatePreferences( | ||||||
|  |             savedInstanceState: Bundle?, | ||||||
|  |             rootKey: String?, | ||||||
|  |         ) { | ||||||
|             setPreferencesFromResource(R.xml.pref_experimental, rootKey) |             setPreferencesFromResource(R.xml.pref_experimental, rootKey) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -0,0 +1,21 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android.testing | ||||||
|  |  | ||||||
|  | import androidx.test.espresso.idling.CountingIdlingResource | ||||||
|  |  | ||||||
|  | object CountingIdlingResourceSingleton { | ||||||
|  |  | ||||||
|  |     private const val RESOURCE = "GLOBAL" | ||||||
|  |  | ||||||
|  |     @JvmField | ||||||
|  |     val countingIdlingResource = CountingIdlingResource(RESOURCE) | ||||||
|  |  | ||||||
|  |     fun increment() { | ||||||
|  |         countingIdlingResource.increment() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun decrement() { | ||||||
|  |         if (!countingIdlingResource.isIdleNow) { | ||||||
|  |             countingIdlingResource.decrement() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,19 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android.testing | ||||||
|  |  | ||||||
|  | import android.os.Build | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestingHelper { | ||||||
|  |     fun isUnitTest(): Boolean { | ||||||
|  |         var device = Build.DEVICE | ||||||
|  |         var product = Build.PRODUCT | ||||||
|  |         if (device == null) { | ||||||
|  |             device = "" | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (product == null) { | ||||||
|  |             product = "" | ||||||
|  |         } | ||||||
|  |         return device == "robolectric" && product == "robolectric" | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -5,7 +5,10 @@ import android.content.Intent | |||||||
| import bou.amine.apps.readerforselfossv2.android.R | import bou.amine.apps.readerforselfossv2.android.R | ||||||
| import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | ||||||
|  |  | ||||||
| fun Context.shareLink(itemUrl: String, itemTitle: String) { | fun Context.shareLink( | ||||||
|  |     itemUrl: String, | ||||||
|  |     itemTitle: String, | ||||||
|  | ) { | ||||||
|     val sendIntent = Intent() |     val sendIntent = Intent() | ||||||
|     sendIntent.flags = Intent.FLAG_ACTIVITY_NEW_TASK |     sendIntent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||||
|     sendIntent.action = Intent.ACTION_SEND |     sendIntent.action = Intent.ACTION_SEND | ||||||
| @@ -15,7 +18,7 @@ fun Context.shareLink(itemUrl: String, itemTitle: String) { | |||||||
|     startActivity( |     startActivity( | ||||||
|         Intent.createChooser( |         Intent.createChooser( | ||||||
|             sendIntent, |             sendIntent, | ||||||
|             getString(R.string.share) |             getString(R.string.share), | ||||||
|         ).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK) |         ).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK), | ||||||
|     ) |     ) | ||||||
| } | } | ||||||
| @@ -0,0 +1,65 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android.utils | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import android.graphics.drawable.GradientDrawable | ||||||
|  | import android.util.AttributeSet | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
|  | import android.widget.RelativeLayout | ||||||
|  | import android.widget.TextView | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.R | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||||
|  | import com.google.android.material.imageview.ShapeableImageView | ||||||
|  | import kotlin.math.abs | ||||||
|  |  | ||||||
|  | class CircleImageView | ||||||
|  |     @JvmOverloads | ||||||
|  |     constructor( | ||||||
|  |         context: Context, | ||||||
|  |         attrs: AttributeSet? = null, | ||||||
|  |         defStyleAttr: Int = 0, | ||||||
|  |     ) : RelativeLayout(context, attrs, defStyleAttr) { | ||||||
|  |         val view: View | ||||||
|  |         val imageView: ShapeableImageView | ||||||
|  |         val textView: TextView | ||||||
|  |  | ||||||
|  |         private val colorScheme = | ||||||
|  |             listOf( | ||||||
|  |                 -0x1a8c8d, | ||||||
|  |                 -0xf9d6e, | ||||||
|  |                 -0x459738, | ||||||
|  |                 -0x6a8a33, | ||||||
|  |                 -0x867935, | ||||||
|  |                 -0x9b4a0a, | ||||||
|  |                 -0xb03c09, | ||||||
|  |                 -0xb22f1f, | ||||||
|  |                 -0xb24954, | ||||||
|  |                 -0x7e387c, | ||||||
|  |                 -0x512a7f, | ||||||
|  |                 -0x759b, | ||||||
|  |                 -0x2b1ea9, | ||||||
|  |                 -0x2ab1, | ||||||
|  |                 -0x48b3, | ||||||
|  |                 -0x5e7781, | ||||||
|  |                 -0x6f5b52, | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |         init { | ||||||
|  |             view = LayoutInflater.from(context).inflate(R.layout.circle_image_view, this, true) | ||||||
|  |             imageView = view.findViewById(R.id.circleImage) | ||||||
|  |             textView = view.findViewById(R.id.circleText) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         fun setBackgroundAndText(text: String) { | ||||||
|  |             val circleDrawable = GradientDrawable() | ||||||
|  |             val color = colorFromIdentifier(text) | ||||||
|  |             circleDrawable.setColor(color) | ||||||
|  |             imageView.setImageDrawable(circleDrawable) | ||||||
|  |  | ||||||
|  |             textView.text = text.toTextDrawableString() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private fun colorFromIdentifier(key: String): Int { | ||||||
|  |             return colorScheme[abs(key.hashCode()) % colorScheme.size] | ||||||
|  |         } | ||||||
|  |     } | ||||||
| @@ -21,14 +21,13 @@ fun Context.openItemUrl( | |||||||
|     currentItem: Int, |     currentItem: Int, | ||||||
|     linkDecoded: String, |     linkDecoded: String, | ||||||
|     articleViewer: Boolean, |     articleViewer: Boolean, | ||||||
|     app: Activity |     app: Activity, | ||||||
| ) { | ) { | ||||||
|  |  | ||||||
|     if (!linkDecoded.isUrlValid()) { |     if (!linkDecoded.isUrlValid()) { | ||||||
|         Toast.makeText( |         Toast.makeText( | ||||||
|             this, |             this, | ||||||
|             this.getString(R.string.cant_open_invalid_url), |             this.getString(R.string.cant_open_invalid_url), | ||||||
|             Toast.LENGTH_LONG |             Toast.LENGTH_LONG, | ||||||
|         ).show() |         ).show() | ||||||
|     } else { |     } else { | ||||||
|         if (articleViewer) { |         if (articleViewer) { | ||||||
| @@ -44,8 +43,7 @@ fun Context.openItemUrl( | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| fun String.isUrlValid(): Boolean = | fun String.isUrlValid(): Boolean = this.toHttpUrlOrNull() != null && Patterns.WEB_URL.matcher(this).matches() | ||||||
|     this.toHttpUrlOrNull() != null && Patterns.WEB_URL.matcher(this).matches() |  | ||||||
|  |  | ||||||
| fun String.isBaseUrlInvalid(): Boolean { | fun String.isBaseUrlInvalid(): Boolean { | ||||||
|     val baseUrl = this.toHttpUrlOrNull() |     val baseUrl = this.toHttpUrlOrNull() | ||||||
| @@ -66,7 +64,10 @@ fun Context.openInBrowserAsNewTask(i: SelfossModel.Item) { | |||||||
| } | } | ||||||
|  |  | ||||||
| class LinkOnTouchListener : View.OnTouchListener { | class LinkOnTouchListener : View.OnTouchListener { | ||||||
|     override fun onTouch(v: View?, event: MotionEvent?): Boolean { |     override fun onTouch( | ||||||
|  |         v: View?, | ||||||
|  |         event: MotionEvent?, | ||||||
|  |     ): Boolean { | ||||||
|         var ret = false |         var ret = false | ||||||
|         val widget: TextView = v as TextView |         val widget: TextView = v as TextView | ||||||
|         val text: CharSequence = widget.text |         val text: CharSequence = widget.text | ||||||
|   | |||||||
| @@ -8,5 +8,4 @@ fun TextBadgeItem.removeBadge(): TextBadgeItem { | |||||||
|     return this |     return this | ||||||
| } | } | ||||||
|  |  | ||||||
| fun TextBadgeItem.maybeShow(): TextBadgeItem = | fun TextBadgeItem.maybeShow(): TextBadgeItem = if (this.isHidden) this.show() else this | ||||||
|     if (this.isHidden) this.show() else this |  | ||||||
|   | |||||||
| @@ -3,38 +3,37 @@ package bou.amine.apps.readerforselfossv2.android.utils.glide | |||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.graphics.Bitmap | import android.graphics.Bitmap | ||||||
| import android.widget.ImageView | import android.widget.ImageView | ||||||
| import androidx.core.graphics.drawable.RoundedBitmapDrawableFactory | import bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
| import com.bumptech.glide.request.target.BitmapImageViewTarget |  | ||||||
| import java.io.ByteArrayInputStream | import java.io.ByteArrayInputStream | ||||||
| import java.io.ByteArrayOutputStream | import java.io.ByteArrayOutputStream | ||||||
| import java.io.InputStream | import java.io.InputStream | ||||||
|  |  | ||||||
| fun Context.bitmapCenterCrop(url: String, iv: ImageView) = | fun Context.bitmapCenterCrop( | ||||||
|     Glide.with(this) |     url: String, | ||||||
|  |     iv: ImageView, | ||||||
|  | ) = Glide.with(this) | ||||||
|     .asBitmap() |     .asBitmap() | ||||||
|     .load(url) |     .load(url) | ||||||
|     .apply(RequestOptions.centerCropTransform()) |     .apply(RequestOptions.centerCropTransform()) | ||||||
|     .into(iv) |     .into(iv) | ||||||
|  |  | ||||||
| fun Context.circularBitmapDrawable(url: String, iv: ImageView) = | fun Context.circularDrawable( | ||||||
|     Glide.with(this) |     url: String, | ||||||
|         .asBitmap() |     view: CircleImageView, | ||||||
|         .load(url) | ) { | ||||||
|         .apply(RequestOptions.centerCropTransform()) |     view.textView.text = "" | ||||||
|         .into(object : BitmapImageViewTarget(iv) { |  | ||||||
|             override fun setResource(resource: Bitmap?) { |  | ||||||
|                 val circularBitmapDrawable = RoundedBitmapDrawableFactory.create( |  | ||||||
|                     resources, |  | ||||||
|                     resource |  | ||||||
|                 ) |  | ||||||
|                 circularBitmapDrawable.isCircular = true |  | ||||||
|                 iv.setImageDrawable(circularBitmapDrawable) |  | ||||||
|             } |  | ||||||
|         }) |  | ||||||
|  |  | ||||||
| fun getBitmapInputStream(bitmap:Bitmap,compressFormat: Bitmap.CompressFormat): InputStream { |     Glide.with(this) | ||||||
|  |         .load(url) | ||||||
|  |         .into(view.imageView) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun getBitmapInputStream( | ||||||
|  |     bitmap: Bitmap, | ||||||
|  |     compressFormat: Bitmap.CompressFormat, | ||||||
|  | ): InputStream { | ||||||
|     val byteArrayOutputStream = ByteArrayOutputStream() |     val byteArrayOutputStream = ByteArrayOutputStream() | ||||||
|     bitmap.compress(compressFormat, 80, byteArrayOutputStream) |     bitmap.compress(compressFormat, 80, byteArrayOutputStream) | ||||||
|     val bitmapData: ByteArray = byteArrayOutputStream.toByteArray() |     val bitmapData: ByteArray = byteArrayOutputStream.toByteArray() | ||||||
|   | |||||||
| @@ -19,7 +19,8 @@ class AppViewModel(private val repository: Repository) : ViewModel() { | |||||||
|                     if (isConnected && !wasConnected && repository.connectionMonitored) { |                     if (isConnected && !wasConnected && repository.connectionMonitored) { | ||||||
|                         _networkAvailableProvider.emit(true) |                         _networkAvailableProvider.emit(true) | ||||||
|                         wasConnected = true |                         wasConnected = true | ||||||
|                     } else if (!isConnected && wasConnected && repository.connectionMonitored){ |                     } else if (!isConnected && wasConnected && repository.connectionMonitored) | ||||||
|  |                         { | ||||||
|                             _networkAvailableProvider.emit(false) |                             _networkAvailableProvider.emit(false) | ||||||
|                             wasConnected = false |                             wasConnected = false | ||||||
|                         } |                         } | ||||||
|   | |||||||
| @@ -51,6 +51,13 @@ | |||||||
|                 android:maxLines="1" |                 android:maxLines="1" | ||||||
|                 android:minHeight="48dp" /> |                 android:minHeight="48dp" /> | ||||||
|  |  | ||||||
|  |             <com.google.android.material.switchmaterial.SwitchMaterial | ||||||
|  |                 android:id="@+id/selfSigned" | ||||||
|  |                 android:layout_width="match_parent" | ||||||
|  |                 android:layout_height="wrap_content" | ||||||
|  |                 android:text="@string/disable_ssl" | ||||||
|  |                 android:textAlignment="viewStart" /> | ||||||
|  |  | ||||||
|             <com.google.android.material.switchmaterial.SwitchMaterial |             <com.google.android.material.switchmaterial.SwitchMaterial | ||||||
|                 android:id="@+id/withLogin" |                 android:id="@+id/withLogin" | ||||||
|                 android:layout_width="match_parent" |                 android:layout_width="match_parent" | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ | |||||||
|             app:layout_constraintStart_toStartOf="parent" |             app:layout_constraintStart_toStartOf="parent" | ||||||
|             app:layout_constraintTop_toBottomOf="@+id/itemImage"> |             app:layout_constraintTop_toBottomOf="@+id/itemImage"> | ||||||
|  |  | ||||||
|             <ImageView |             <bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||||
|                 android:id="@+id/sourceImage" |                 android:id="@+id/sourceImage" | ||||||
|                 android:layout_width="40dp" |                 android:layout_width="40dp" | ||||||
|                 android:layout_height="40dp" |                 android:layout_height="40dp" | ||||||
| @@ -54,19 +54,24 @@ | |||||||
|                 android:layout_width="0dp" |                 android:layout_width="0dp" | ||||||
|                 android:layout_height="wrap_content" |                 android:layout_height="wrap_content" | ||||||
|                 android:layout_margin="8dp" |                 android:layout_margin="8dp" | ||||||
|  |                 android:layout_marginStart="8dp" | ||||||
|  |                 android:layout_marginTop="8dp" | ||||||
|  |                 android:layout_marginEnd="8dp" | ||||||
|                 android:textAlignment="viewStart" |                 android:textAlignment="viewStart" | ||||||
|                 android:textColor="?android:textColorPrimary" |                 android:textColor="?android:textColorPrimary" | ||||||
|                 android:textStyle="bold" |                 android:textStyle="bold" | ||||||
|                 app:layout_constraintEnd_toEndOf="parent" |                 app:layout_constraintEnd_toEndOf="parent" | ||||||
|                 app:layout_constraintStart_toEndOf="@+id/sourceImage" |                 app:layout_constraintStart_toEndOf="@+id/sourceImage" | ||||||
|                 app:layout_constraintTop_toTopOf="@+id/sourceImage" |                 app:layout_constraintTop_toTopOf="parent" | ||||||
|                 tools:text="Titre" /> |                 tools:text="Titre" /> | ||||||
|  |  | ||||||
|             <TextView |             <TextView | ||||||
|                 android:id="@+id/sourceTitleAndDate" |                 android:id="@+id/sourceTitleAndDate" | ||||||
|                 android:layout_width="0dp" |                 android:layout_width="0dp" | ||||||
|                 android:layout_height="wrap_content" |                 android:layout_height="wrap_content" | ||||||
|  |                 android:layout_marginStart="8dp" | ||||||
|                 android:layout_marginTop="8dp" |                 android:layout_marginTop="8dp" | ||||||
|  |                 android:layout_marginEnd="8dp" | ||||||
|                 android:textAlignment="viewStart" |                 android:textAlignment="viewStart" | ||||||
|                 android:textColor="?android:textColorPrimary" |                 android:textColor="?android:textColorPrimary" | ||||||
|                 android:textSize="14sp" |                 android:textSize="14sp" | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								androidApp/src/main/res/layout/circle_image_view.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								androidApp/src/main/res/layout/circle_image_view.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|  |     android:layout_width="wrap_content" | ||||||
|  |     android:layout_height="wrap_content"> | ||||||
|  |  | ||||||
|  |     <com.google.android.material.imageview.ShapeableImageView | ||||||
|  |         android:id="@+id/circleImage" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:scaleType="centerCrop" | ||||||
|  |         app:shapeAppearanceOverlay="@style/circleImageView" | ||||||
|  |         app:srcCompat="@drawable/background_splash" /> | ||||||
|  |  | ||||||
|  |     <TextView | ||||||
|  |         android:id="@+id/circleText" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:ellipsize="none" | ||||||
|  |         android:gravity="center" | ||||||
|  |         android:singleLine="true" | ||||||
|  |         android:textColor="@color/white" | ||||||
|  |         android:textIsSelectable="false" | ||||||
|  |         android:textSize="20sp" | ||||||
|  |         android:typeface="normal" /> | ||||||
|  | </RelativeLayout> | ||||||
| @@ -5,11 +5,13 @@ | |||||||
|     android:layout_width="match_parent" |     android:layout_width="match_parent" | ||||||
|     android:layout_height="wrap_content"> |     android:layout_height="wrap_content"> | ||||||
|  |  | ||||||
|     <ImageView |     <bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||||
|         android:id="@+id/itemImage" |         android:id="@+id/itemImage" | ||||||
|         android:layout_width="46dp" |         android:layout_width="46dp" | ||||||
|         android:layout_height="46dp" |         android:layout_height="46dp" | ||||||
|         android:layout_marginStart="8dp" |         android:layout_marginStart="8dp" | ||||||
|  |         android:layout_marginTop="8dp" | ||||||
|  |         android:layout_marginBottom="8dp" | ||||||
|         app:layout_constraintBottom_toBottomOf="parent" |         app:layout_constraintBottom_toBottomOf="parent" | ||||||
|         app:layout_constraintStart_toStartOf="parent" |         app:layout_constraintStart_toStartOf="parent" | ||||||
|         app:layout_constraintTop_toTopOf="parent" /> |         app:layout_constraintTop_toTopOf="parent" /> | ||||||
| @@ -20,7 +22,7 @@ | |||||||
|         android:layout_height="wrap_content" |         android:layout_height="wrap_content" | ||||||
|         android:layout_marginStart="8dp" |         android:layout_marginStart="8dp" | ||||||
|         android:layout_marginTop="8dp" |         android:layout_marginTop="8dp" | ||||||
|         android:layout_marginEnd="16dp" |         android:layout_marginEnd="8dp" | ||||||
|         android:ellipsize="end" |         android:ellipsize="end" | ||||||
|         android:fontFamily="sans-serif" |         android:fontFamily="sans-serif" | ||||||
|         android:maxLines="3" |         android:maxLines="3" | ||||||
| @@ -38,15 +40,17 @@ | |||||||
|         android:layout_width="0dp" |         android:layout_width="0dp" | ||||||
|         android:layout_height="wrap_content" |         android:layout_height="wrap_content" | ||||||
|         android:layout_marginStart="8dp" |         android:layout_marginStart="8dp" | ||||||
|         android:layout_marginEnd="16dp" |         android:layout_marginTop="16dp" | ||||||
|  |         android:layout_marginEnd="8dp" | ||||||
|  |         android:layout_marginBottom="8dp" | ||||||
|         android:gravity="start" |         android:gravity="start" | ||||||
|         android:maxLines="1" |  | ||||||
|         android:textAlignment="viewStart" |         android:textAlignment="viewStart" | ||||||
|         android:textColor="?android:textColorPrimary" |         android:textColor="?android:textColorPrimary" | ||||||
|         android:textSize="14sp" |         android:textSize="14sp" | ||||||
|  |         app:layout_constraintBottom_toBottomOf="parent" | ||||||
|         app:layout_constraintEnd_toEndOf="parent" |         app:layout_constraintEnd_toEndOf="parent" | ||||||
|         app:layout_constraintStart_toEndOf="@+id/itemImage" |         app:layout_constraintStart_toEndOf="@+id/itemImage" | ||||||
|         app:layout_constraintTop_toBottomOf="@+id/itemImage" |         app:layout_constraintTop_toBottomOf="@+id/title" | ||||||
|         tools:text="Google Actualité Il y a 5h" /> |         tools:text="Google Actualité Il y a 5h" /> | ||||||
|  |  | ||||||
| </androidx.constraintlayout.widget.ConstraintLayout> | </androidx.constraintlayout.widget.ConstraintLayout> | ||||||
| @@ -25,7 +25,7 @@ | |||||||
|         app:layout_constraintTop_toTopOf="parent" |         app:layout_constraintTop_toTopOf="parent" | ||||||
|         app:layout_constraintVertical_bias="0.0" /> |         app:layout_constraintVertical_bias="0.0" /> | ||||||
|  |  | ||||||
|     <ImageView |     <bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||||
|         android:id="@+id/itemImage" |         android:id="@+id/itemImage" | ||||||
|         android:layout_width="36dp" |         android:layout_width="36dp" | ||||||
|         android:layout_height="36dp" |         android:layout_height="36dp" | ||||||
|   | |||||||
| @@ -36,6 +36,12 @@ | |||||||
|         android:orderInCategory="101" |         android:orderInCategory="101" | ||||||
|         android:title="@string/menu_home_refresh" /> |         android:title="@string/menu_home_refresh" /> | ||||||
|  |  | ||||||
|  |     <item | ||||||
|  |         android:id="@+id/issue_tracker" | ||||||
|  |         app:showAsAction="never" | ||||||
|  |         android:orderInCategory="103" | ||||||
|  |         android:title="@string/issue_tracker_link" /> | ||||||
|  |  | ||||||
|     <item android:id="@+id/action_disconnect" |     <item android:id="@+id/action_disconnect" | ||||||
|           android:title="@string/action_disconnect" |           android:title="@string/action_disconnect" | ||||||
|           android:orderInCategory="104" |           android:orderInCategory="104" | ||||||
|   | |||||||
| @@ -3,6 +3,13 @@ | |||||||
|       xmlns:app="http://schemas.android.com/apk/res-auto"> |       xmlns:app="http://schemas.android.com/apk/res-auto"> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     <item | ||||||
|  |         android:id="@+id/issue_tracker" | ||||||
|  |         app:showAsAction="never" | ||||||
|  |         android:orderInCategory="101" | ||||||
|  |         android:title="@string/issue_tracker_link" /> | ||||||
|  |  | ||||||
|     <item android:id="@+id/about" |     <item android:id="@+id/about" | ||||||
|           android:title="@string/action_about" |           android:title="@string/action_about" | ||||||
|           android:orderInCategory="102" |           android:orderInCategory="102" | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Lector per a Selfoss"</string> |     <string name="app_name">"Lector per a Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Inicia la sessió"</string> |     <string name="title_activity_login">"Inicia la sessió"</string> | ||||||
|     <string name="prompt_password">"Contrasenya"</string> |     <string name="prompt_password">"Contrasenya"</string> | ||||||
| @@ -22,13 +22,6 @@ | |||||||
|     <string name="wrong_infos">"Torneu a comprovar la informació."</string> |     <string name="wrong_infos">"Torneu a comprovar la informació."</string> | ||||||
|     <string name="all_posts_not_read">"No s'han llegit totes les publicacions"</string> |     <string name="all_posts_not_read">"No s'han llegit totes les publicacions"</string> | ||||||
|     <string name="all_posts_read">"S'han llegit totes les publicacions"</string> |     <string name="all_posts_read">"S'han llegit totes les publicacions"</string> | ||||||
|     <string name="nothing_here">"No hi ha res"</string> |  | ||||||
|     <string name="tab_new">"Nou"</string> |  | ||||||
|     <string name="tab_read">"Tot"</string> |  | ||||||
|     <string name="tab_favs">"Preferits"</string> |  | ||||||
|     <string name="action_about">"Quant a"</string> |  | ||||||
|     <string name="marked_as_read">"Element llegit"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Desfés"</string> |     <string name="undo_string">"Desfés"</string> | ||||||
|     <string name="addStringNoUrl">"Inicieu la sessió per afegir fonts."</string> |     <string name="addStringNoUrl">"Inicieu la sessió per afegir fonts."</string> | ||||||
|     <string name="cant_get_sources">"No es pot obtenir la llista de fonts."</string> |     <string name="cant_get_sources">"No es pot obtenir la llista de fonts."</string> | ||||||
| @@ -90,7 +83,7 @@ | |||||||
|     <string name="pref_switch_items_caching">Guarda els elements per utilitzar-los sense connexió</string> |     <string name="pref_switch_items_caching">Guarda els elements per utilitzar-los sense connexió</string> | ||||||
|     <string name="pref_switch_update_sources">Check for new sources and tags</string> |     <string name="pref_switch_update_sources">Check for new sources and tags</string> | ||||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> |     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> | ||||||
|     <string name="network_connectivity_lost">"Network connection lost"</string> |     <string name="network_connectivity_lost">Sense connexió!</string> | ||||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> |     <string name="network_connectivity_retrieved">"Network connection is now available"</string> | ||||||
|     <string name="pref_switch_periodic_refresh">Sincronitza els articles</string> |     <string name="pref_switch_periodic_refresh">Sincronitza els articles</string> | ||||||
|     <string name="pref_switch_periodic_refresh_off">Els articles no se sincronitzaran en segon pla</string> |     <string name="pref_switch_periodic_refresh_off">Els articles no se sincronitzaran en segon pla</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Reader für selfoss"</string> |     <string name="app_name">"Reader für Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Anmelden"</string> |     <string name="title_activity_login">"Anmelden"</string> | ||||||
|     <string name="prompt_password">"Passwort"</string> |     <string name="prompt_password">"Passwort"</string> | ||||||
|     <string name="action_sign_in">"Fortfahren"</string> |     <string name="action_sign_in">"Fortfahren"</string> | ||||||
| @@ -22,22 +22,15 @@ | |||||||
|     <string name="wrong_infos">"Überprüfe deine Angaben noch einmal."</string> |     <string name="wrong_infos">"Überprüfe deine Angaben noch einmal."</string> | ||||||
|     <string name="all_posts_not_read">"Nicht alle Beiträge wurden gelesen"</string> |     <string name="all_posts_not_read">"Nicht alle Beiträge wurden gelesen"</string> | ||||||
|     <string name="all_posts_read">"Alle Beiträge wurden gelesen"</string> |     <string name="all_posts_read">"Alle Beiträge wurden gelesen"</string> | ||||||
|     <string name="nothing_here">"Keine Einträge vorhanden"</string> |  | ||||||
|     <string name="tab_new">"Neu"</string> |  | ||||||
|     <string name="tab_read">"Alle"</string> |  | ||||||
|     <string name="tab_favs">"Favoriten"</string> |  | ||||||
|     <string name="action_about">"Über"</string> |  | ||||||
|     <string name="marked_as_read">"Artikel gelesen"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Rückgängig"</string> |     <string name="undo_string">"Rückgängig"</string> | ||||||
|     <string name="addStringNoUrl">"Melde dich an um Quellen hinzuzufügen."</string> |     <string name="addStringNoUrl">"Melde dich an um Quellen hinzuzufügen."</string> | ||||||
|     <string name="cant_get_sources">"Quellen können nicht abgerufen werden."</string> |     <string name="cant_get_sources">"Quellen können nicht abgerufen werden."</string> | ||||||
|     <string name="cant_create_source">"Quelle kann nicht gespeichert werden."</string> |     <string name="cant_create_source">"Quelle kann nicht gespeichert werden."</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Fehler beim Laden der Spouts-Liste aufgrund von Netzwerkproblemen."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"Fehler beim Laden der Spouts-Liste, möglicherweise aufgrund eines API-Fehlers."</string> | ||||||
|     <string name="form_not_complete">"Das Formular ist nicht vollständig"</string> |     <string name="form_not_complete">"Das Formular ist nicht vollständig"</string> | ||||||
|     <string name="pref_header_links">"Links"</string> |     <string name="pref_header_links">"Links"</string> | ||||||
|     <string name="issue_tracker_link">"Issue Tracker"</string> |     <string name="issue_tracker_link">"Ticketsystem"</string> | ||||||
|     <string name="issue_tracker_summary">"Melde einen Bug oder rege ein neues Feature an"</string> |     <string name="issue_tracker_summary">"Melde einen Bug oder rege ein neues Feature an"</string> | ||||||
|     <string name="warning_wrong_url">"WARNUNG"</string> |     <string name="warning_wrong_url">"WARNUNG"</string> | ||||||
|     <string name="pref_switch_card_view_title">"Kachelansicht"</string> |     <string name="pref_switch_card_view_title">"Kachelansicht"</string> | ||||||
| @@ -64,68 +57,71 @@ | |||||||
|     <string name="filter_item_tags">Tags</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="filter_item_sources">Quellen</string> |     <string name="filter_item_sources">Quellen</string> | ||||||
|     <string name="menu_home_search">Suche</string> |     <string name="menu_home_search">Suche</string> | ||||||
|     <string name="can_delete_source">Can\'t delete the source…</string> |     <string name="can_delete_source">Quelle konnte nicht gelöscht werden…</string> | ||||||
|     <string name="base_url_error">Beim Versuch deine Selfoss-Instanz zu erreichen ist ein Fehler aufgetreten. Solltet dieser Fehler bestehen bleiben, trete bitte mit mir in Kontakt.</string> |     <string name="base_url_error">Beim Versuch deine Selfoss-Instanz zu erreichen ist ein Fehler aufgetreten. Solltet dieser Fehler bestehen bleiben, trete bitte mit mir in Kontakt.</string> | ||||||
|     <string name="pref_header_theme">Designs</string> |     <string name="pref_header_theme">Designs</string> | ||||||
|     <string name="pref_selfoss_category">selfoss API</string> |     <string name="pref_selfoss_category">selfoss API</string> | ||||||
|     <string name="pref_api_items_number_title">Loaded items number</string> |     <string name="pref_api_items_number_title">Anzahl der zu ladenden Artikel</string> | ||||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> |     <string name="pref_general_infinite_loading_title">Weitere Artikel beim Navigieren laden</string> | ||||||
|     <string name="translation">Übersetzung</string> |     <string name="translation">Übersetzung</string> | ||||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> |     <string name="cant_open_invalid_url">Der Artikel-Link ist ungültig. Ich such nach einer Lösung dieses Problems, damit die App nicht abstürzt.</string> | ||||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> |     <string name="items_number_should_be_number">Die Anzahl der Artikel sollte eine Ganzzahl sein.</string> | ||||||
|     <string name="reader_action_open">Im Browser öffnen</string> |     <string name="reader_action_open">Im Browser öffnen</string> | ||||||
|     <string name="reader_action_share">Teilen</string> |     <string name="reader_action_share">Teilen</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Artikel als gelesen markieren, wenn zwischen den Artikeln gewischt wird.</string> | ||||||
|     <string name="add_to_favs_reader">Zu Favoriten hinzufügen</string> |     <string name="add_to_favs_reader">Zu Favoriten hinzufügen</string> | ||||||
|     <string name="pref_content_reader_font_size">Article reader content font size</string> |     <string name="pref_content_reader_font_size">Schriftgröße im Lesemodus</string> | ||||||
|     <string name="pref_header_viewer">Article viewer</string> |     <string name="pref_header_viewer">Lesemodus</string> | ||||||
|     <string name="refresh_dialog_message">This will refresh your Selfoss instance.</string> |     <string name="refresh_dialog_message">Dies wird die Selfoss-Instanz aktualisieren.</string> | ||||||
|     <string name="markall_dialog_message">Dies wird alle Elemente als gelesen markieren.</string> |     <string name="markall_dialog_message">Dies wird alle Elemente als gelesen markieren.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Beim Wischen als gelesen markieren</string> |     <string name="pref_switch_actions_pager_scroll">Beim Wischen als gelesen markieren</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Artikel nicht als gelesen markieren, wenn zwischen den Artikeln gewischt wird.</string> | ||||||
|     <string name="unmark">Eintrag als ungelesen markieren</string> |     <string name="unmark">Eintrag als ungelesen markieren</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline-Modus und Cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Artikel werden nicht lokal zwischengespeichert wodurch die App nicht offline nutzbar ist.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Artikel werden lokal zwischengespeichert wodurch die App offline nutzbar ist.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Artikel lokal zwischenspeichern</string> | ||||||
|     <string name="pref_switch_update_sources">Check for new sources and tags</string> |     <string name="pref_switch_update_sources">Nach neuen Quellen und Tags suchen</string> | ||||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> |     <string name="pref_switch_update_sources_summary">Diese Funktion sollte deaktiviert werden, wenn der Server übermäßig viele Datenbankanfragen erhält.</string> | ||||||
|     <string name="network_connectivity_lost">"Die Netzwerkverbindung wurde unterbrochen"</string> |     <string name="network_connectivity_lost">"Die Netzwerkverbindung wurde unterbrochen"</string> | ||||||
|     <string name="network_connectivity_retrieved">"Netzwerkverbindung ist jetzt verfügbar"</string> |     <string name="network_connectivity_retrieved">"Netzwerkverbindung ist jetzt verfügbar"</string> | ||||||
|     <string name="pref_switch_periodic_refresh">Synchronisiere Artikel</string> |     <string name="pref_switch_periodic_refresh">Synchronisiere Artikel</string> | ||||||
|     <string name="pref_switch_periodic_refresh_off">Artikel werden nicht im Hintergrund synchronisiert</string> |     <string name="pref_switch_periodic_refresh_off">Artikel werden nicht im Hintergrund synchronisiert</string> | ||||||
|     <string name="pref_switch_periodic_refresh_on">Die Artikel werden regelmäßig synchronisiert</string> |     <string name="pref_switch_periodic_refresh_on">Die Artikel werden regelmäßig synchronisiert</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Aktualisierungsintervall (>= 15 Minuten)]]></string> | ||||||
|     <string name="pref_switch_refresh_when_charging">Nur aktualisieren, wenn das Telefon aufgeladen wird</string> |     <string name="pref_switch_refresh_when_charging">Nur aktualisieren, wenn das Telefon aufgeladen wird</string> | ||||||
|     <string name="loading_notification_title">Lädt…</string> |     <string name="loading_notification_title">Lädt…</string> | ||||||
|     <string name="loading_notification_text">Selfoss synchronisiert Ihre Artikel</string> |     <string name="loading_notification_text">Selfoss synchronisiert Ihre Artikel</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Synchronisationsbenachrichtigung</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">Benachrichtigung bei neuen Artikeln</string> | ||||||
|     <string name="new_items_notification_title">New items !</string> |     <string name="new_items_notification_title">Neue Artikel!</string> | ||||||
|     <string name="new_items_notification_text">%1$d new items loaded.</string> |     <string name="new_items_notification_text">%1$d neue Artikel geladen.</string> | ||||||
|     <string name="pref_switch_notify_new_items">Notify on new items synced.</string> |     <string name="pref_switch_notify_new_items">Benachrichtigung bei neuen Artikeln</string> | ||||||
|     <string name="shortcut_offline">Offline</string> |     <string name="shortcut_offline">Offline</string> | ||||||
|     <string name="pref_api_timeout">API-Zeitüberschreitung</string> |     <string name="pref_api_timeout">API-Zeitüberschreitung</string> | ||||||
|     <string name="pref_header_experimental">Experimentell</string> |     <string name="pref_header_experimental">Experimentell</string> | ||||||
|     <string name="webview_dialog_issue_message">Webview not available. Disabling the article viewer to avoid any future crashes. Will load articles inside of your browser from now on.</string> |     <string name="webview_dialog_issue_message">Webview ist nicht verfügbar. Deaktiviere den Lesemodus, um zukünftige Abstürze zu vermeiden. Lade von nun an die Nachrichten in deinen Browser.</string> | ||||||
|     <string name="webview_dialog_issue_title">Webview issue</string> |     <string name="webview_dialog_issue_title">Webview-Probleme</string> | ||||||
|     <string name="reader_text_align_left">Align left</string> |     <string name="reader_text_align_left">Linksbündig</string> | ||||||
|     <string name="reader_text_align_justify">Justify</string> |     <string name="reader_text_align_justify">Blocksatz</string> | ||||||
|     <string name="settings_reader_font">Reader font</string> |     <string name="settings_reader_font">Schriftgröße im Lesemodus</string> | ||||||
|     <string name="reader_static_bar_title">Static bottom bar in the article viewer</string> |     <string name="reader_static_bar_title">Statische untere Leiste im Lesemodus</string> | ||||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> |     <string name="reader_static_bar_on">Die untere Leiste wird dauerhaft angezeigt</string> | ||||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> |     <string name="reader_static_bar_off">Die untere Leiste kann über einen schwebenden Button angezeigt werden</string> | ||||||
|     <string name="remove_source">Remove source</string> |     <string name="remove_source">Quelle entfernen</string> | ||||||
|     <string name="pref_theme_title">Light/Dark mode</string> |     <string name="pref_theme_title">Heller/Dunkler Modus</string> | ||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dunkler Modus</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Systemeinstellungen übernehmen</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Heller Modus</string> | ||||||
|     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> |     <string name="gdpr_dialog_title">Diese App teilt keine persönlichen Daten.</string> | ||||||
|     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> |     <string name="gdpr_dialog_message"><![CDATA[Das Senden von Absturzberichten ist jetzt aktiviert. Die Funktion kann auf der Einstellungsseite deaktiviert werden, beachte aber bitte, dass Absturzberichte für die Anwendungsentwicklung von entscheidender Bedeutung sind.]]></string> | ||||||
|     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> |     <string name="crash_toast_text">Die App ist abgestürzt. Details werden an den Entwickler gesendet.</string> | ||||||
|     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> |     <string name="pref_switch_disable_acra">"Automatische Fehlerberichterstattung deaktivieren. "</string> | ||||||
|     <string name="menu_home_filter">Filters</string> |     <string name="menu_home_filter">Filter</string> | ||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">Diese App funktioniert nur mit einer Selfoss-Instanz, nicht mit einzelnen RSS-Feeds.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Quellen</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Quelle aktualisieren</string> | ||||||
|  |     <string name="confirm_disconnect_title">Verbindung trennen?</string> | ||||||
|  |     <string name="confirm_disconnect_description">Die Verbindung zur Selfoss-Instanz wird getrennt.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Lector para Selfoss"</string> |     <string name="app_name">"Lector para Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Iniciar sesión"</string> |     <string name="title_activity_login">"Iniciar sesión"</string> | ||||||
|     <string name="prompt_password">"Contraseña"</string> |     <string name="prompt_password">"Contraseña"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"Revise sus datos de nuevo."</string> |     <string name="wrong_infos">"Revise sus datos de nuevo."</string> | ||||||
|     <string name="all_posts_not_read">"No todas las publicaciones fueron leídas"</string> |     <string name="all_posts_not_read">"No todas las publicaciones fueron leídas"</string> | ||||||
|     <string name="all_posts_read">"Todas las publicaciones fueron leídas"</string> |     <string name="all_posts_read">"Todas las publicaciones fueron leídas"</string> | ||||||
|     <string name="nothing_here">"Nada aquí"</string> |  | ||||||
|     <string name="tab_new">"Nuevo"</string> |  | ||||||
|     <string name="tab_read">"Todo"</string> |  | ||||||
|     <string name="tab_favs">"Favoritos"</string> |  | ||||||
|     <string name="action_about">"Acerca de"</string> |  | ||||||
|     <string name="marked_as_read">"Artículo leído"</string> |  | ||||||
|     <string name="marked_as_unread">"Artículo no leído"</string> |  | ||||||
|     <string name="undo_string">"Deshacer"</string> |     <string name="undo_string">"Deshacer"</string> | ||||||
|     <string name="addStringNoUrl">"Iniciar sesión para añadir fuentes."</string> |     <string name="addStringNoUrl">"Iniciar sesión para añadir fuentes."</string> | ||||||
|     <string name="cant_get_sources">"No se puede obtener la lista de fuentes."</string> |     <string name="cant_get_sources">"No se puede obtener la lista de fuentes."</string> | ||||||
|     <string name="cant_create_source">"No se puede crear la fuente."</string> |     <string name="cant_create_source">"No se puede crear la fuente."</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"No se puede obtener la lista de fuentes."</string> | ||||||
|     <string name="form_not_complete">"El formulario no está completo"</string> |     <string name="form_not_complete">"El formulario no está completo"</string> | ||||||
|     <string name="pref_header_links">"Enlaces"</string> |     <string name="pref_header_links">"Enlaces"</string> | ||||||
|     <string name="issue_tracker_link">"Rastreador de Incidencias"</string> |     <string name="issue_tracker_link">"Rastreador de Incidencias"</string> | ||||||
| @@ -90,7 +83,7 @@ | |||||||
|     <string name="pref_switch_items_caching">Guardar elementos para uso sin conexión</string> |     <string name="pref_switch_items_caching">Guardar elementos para uso sin conexión</string> | ||||||
|     <string name="pref_switch_update_sources">Check for new sources and tags</string> |     <string name="pref_switch_update_sources">Check for new sources and tags</string> | ||||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> |     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> | ||||||
|     <string name="network_connectivity_lost">"Network connection lost"</string> |     <string name="network_connectivity_lost">Sin conexión!</string> | ||||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> |     <string name="network_connectivity_retrieved">"Network connection is now available"</string> | ||||||
|     <string name="pref_switch_periodic_refresh">Sincronizar artículos</string> |     <string name="pref_switch_periodic_refresh">Sincronizar artículos</string> | ||||||
|     <string name="pref_switch_periodic_refresh_off">Los artículos no se sincronizarán en segundo plano</string> |     <string name="pref_switch_periodic_refresh_off">Los artículos no se sincronizarán en segundo plano</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,131 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> |  | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |  | ||||||
|     <string name="title_activity_login">"Log in"</string> |  | ||||||
|     <string name="prompt_password">"Password"</string> |  | ||||||
|     <string name="action_sign_in">"Go"</string> |  | ||||||
|     <string name="error_invalid_password">"Password not long enough"</string> |  | ||||||
|     <string name="error_field_required">"Field required"</string> |  | ||||||
|     <string name="prompt_url">"Url"</string> |  | ||||||
|     <string name="withLoginSwitch">"Login required ?"</string> |  | ||||||
|     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> |  | ||||||
|     <string name="prompt_login">"Username"</string> |  | ||||||
|     <string name="readAll">"Read all"</string> |  | ||||||
|     <string name="action_disconnect">"Disconnect"</string> |  | ||||||
|     <string name="title_activity_settings">"Settings"</string> |  | ||||||
|     <string name="pref_header_general">"General"</string> |  | ||||||
|     <string name="add_source_hint_tags">"Tag1, Tag2, Tag3"</string> |  | ||||||
|     <string name="add_source_hint_url">"Link"</string> |  | ||||||
|     <string name="add_source_hint_name">"Name"</string> |  | ||||||
|     <string name="add_source">"Add a source"</string> |  | ||||||
|     <string name="add_source_save">"Save"</string> |  | ||||||
|     <string name="wrong_infos">"Check your details again."</string> |  | ||||||
|     <string name="all_posts_not_read">"All posts weren't read"</string> |  | ||||||
|     <string name="all_posts_read">"All posts were read"</string> |  | ||||||
|     <string name="nothing_here">"Nothing here"</string> |  | ||||||
|     <string name="tab_new">"New"</string> |  | ||||||
|     <string name="tab_read">"All"</string> |  | ||||||
|     <string name="tab_favs">"Favorites"</string> |  | ||||||
|     <string name="action_about">"About"</string> |  | ||||||
|     <string name="marked_as_read">"Item read"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Undo"</string> |  | ||||||
|     <string name="addStringNoUrl">"Log in to add sources."</string> |  | ||||||
|     <string name="cant_get_sources">"Can't get sources list."</string> |  | ||||||
|     <string name="cant_create_source">"Can't create source."</string> |  | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |  | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |  | ||||||
|     <string name="form_not_complete">"The form is not complete"</string> |  | ||||||
|     <string name="pref_header_links">"Links"</string> |  | ||||||
|     <string name="issue_tracker_link">"Issue Tracker"</string> |  | ||||||
|     <string name="issue_tracker_summary">"Report a bug or ask for a new feature"</string> |  | ||||||
|     <string name="warning_wrong_url">"WARNING"</string> |  | ||||||
|     <string name="pref_switch_card_view_title">"Card View"</string> |  | ||||||
|     <string name="share">"Share"</string> |  | ||||||
|     <string name="switch_unread_count">"Display the unread count as a badge for the bottom bar."</string> |  | ||||||
|     <string name="switch_unread_count_title">"Display unread count"</string> |  | ||||||
|     <string name="display_all_counts_title">"Display count for favorite and read"</string> |  | ||||||
|     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> |  | ||||||
|     <string name="pref_article_viewer_title">"Open links inside the app"</string> |  | ||||||
|     <string name="pref_article_viewer_on">"Articles will open inside the app"</string> |  | ||||||
|     <string name="pref_article_viewer_off">"Articles will open with your default browser"</string> |  | ||||||
|     <string name="pref_general_category_links">"Link handling"</string> |  | ||||||
|     <string name="pref_general_category_displaying">"Displaying"</string> |  | ||||||
|     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> |  | ||||||
|     <string name="pref_switch_card_view_off">"The articles will be displayed as a list"</string> |  | ||||||
|     <string name="menu_home_refresh">"Update remote"</string> |  | ||||||
|     <string name="refresh_success_response">"The remote is updated, you can now reload the articles list"</string> |  | ||||||
|     <string name="refresh_failer_message">"The update didn't work, try again later, or check your selfoss logs."</string> |  | ||||||
|     <string name="refresh_in_progress">"Refresh in progress"</string> |  | ||||||
|     <string name="card_height_title">Full height cards</string> |  | ||||||
|     <string name="card_height_on">Cards height will adjust to its content</string> |  | ||||||
|     <string name="card_height_off">Card height will be fixed</string> |  | ||||||
|     <string name="source_code">Source code</string> |  | ||||||
|     <string name="filter_item_tags">Tags</string> |  | ||||||
|     <string name="filter_item_sources">Sources</string> |  | ||||||
|     <string name="menu_home_search">Search</string> |  | ||||||
|     <string name="can_delete_source">Can\'t delete the source…</string> |  | ||||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> |  | ||||||
|     <string name="pref_header_theme">Themes</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Api</string> |  | ||||||
|     <string name="pref_api_items_number_title">Loaded items number</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> |  | ||||||
|     <string name="translation">Translation</string> |  | ||||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> |  | ||||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> |  | ||||||
|     <string name="reader_action_open">Open in browser</string> |  | ||||||
|     <string name="reader_action_share">Share</string> |  | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |  | ||||||
|     <string name="add_to_favs_reader">Add to favorites</string> |  | ||||||
|     <string name="pref_content_reader_font_size">Article reader content font size</string> |  | ||||||
|     <string name="pref_header_viewer">Article viewer</string> |  | ||||||
|     <string name="refresh_dialog_message">This will refresh your Selfoss instance.</string> |  | ||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |  | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |  | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |  | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |  | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |  | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |  | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |  | ||||||
|     <string name="pref_switch_update_sources">Check for new sources and tags</string> |  | ||||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> |  | ||||||
|     <string name="network_connectivity_lost">"Network connection lost"</string> |  | ||||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> |  | ||||||
|     <string name="pref_switch_periodic_refresh">Sync articles</string> |  | ||||||
|     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> |  | ||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |  | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> |  | ||||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> |  | ||||||
|     <string name="loading_notification_title">Loading …</string> |  | ||||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> |  | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |  | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |  | ||||||
|     <string name="new_items_notification_title">New items !</string> |  | ||||||
|     <string name="new_items_notification_text">%1$d new items loaded.</string> |  | ||||||
|     <string name="pref_switch_notify_new_items">Notify on new items synced.</string> |  | ||||||
|     <string name="shortcut_offline">Offline</string> |  | ||||||
|     <string name="pref_api_timeout">Api Timeout</string> |  | ||||||
|     <string name="pref_header_experimental">Experimental</string> |  | ||||||
|     <string name="webview_dialog_issue_message">Webview not available. Disabling the article viewer to avoid any future crashes. Will load articles inside of your browser from now on.</string> |  | ||||||
|     <string name="webview_dialog_issue_title">Webview issue</string> |  | ||||||
|     <string name="reader_text_align_left">Align left</string> |  | ||||||
|     <string name="reader_text_align_justify">Justify</string> |  | ||||||
|     <string name="settings_reader_font">Reader font</string> |  | ||||||
|     <string name="reader_static_bar_title">Static bottom bar in the article viewer</string> |  | ||||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> |  | ||||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> |  | ||||||
|     <string name="remove_source">Remove source</string> |  | ||||||
|     <string name="pref_theme_title">Light/Dark mode</string> |  | ||||||
|     <string name="mode_dark">Dark mode</string> |  | ||||||
|     <string name="mode_system">Follow the system setting</string> |  | ||||||
|     <string name="mode_light">Light mode</string> |  | ||||||
|     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> |  | ||||||
|     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> |  | ||||||
|     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> |  | ||||||
|     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> |  | ||||||
|     <string name="menu_home_filter">Filters</string> |  | ||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |  | ||||||
|     <string name="menu_home_sources">Sources</string> |  | ||||||
|     <string name="update_source">Update source</string> |  | ||||||
| </resources> |  | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Login"</string> |     <string name="title_activity_login">"Login"</string> | ||||||
|     <string name="prompt_password">"Mot de passe"</string> |     <string name="prompt_password">"Mot de passe"</string> | ||||||
| @@ -22,13 +22,6 @@ | |||||||
|     <string name="wrong_infos">"Vérifiez vos informations."</string> |     <string name="wrong_infos">"Vérifiez vos informations."</string> | ||||||
|     <string name="all_posts_not_read">"Tous les posts n'ont pas été lus"</string> |     <string name="all_posts_not_read">"Tous les posts n'ont pas été lus"</string> | ||||||
|     <string name="all_posts_read">"Tous les posts sont lus"</string> |     <string name="all_posts_read">"Tous les posts sont lus"</string> | ||||||
|     <string name="nothing_here">"Il n'y a rien ici !"</string> |  | ||||||
|     <string name="tab_new">"Non lus"</string> |  | ||||||
|     <string name="tab_read">"Tous"</string> |  | ||||||
|     <string name="tab_favs">"Favoris"</string> |  | ||||||
|     <string name="action_about">"À propos"</string> |  | ||||||
|     <string name="marked_as_read">"Marqué comme lu"</string> |  | ||||||
|     <string name="marked_as_unread">"Marqué comme non lu"</string> |  | ||||||
|     <string name="undo_string">"Annuler"</string> |     <string name="undo_string">"Annuler"</string> | ||||||
|     <string name="addStringNoUrl">"Identifiez-vous pour ajouter une source."</string> |     <string name="addStringNoUrl">"Identifiez-vous pour ajouter une source."</string> | ||||||
|     <string name="cant_get_sources">"Impossible de récupérer la liste des sources"</string> |     <string name="cant_get_sources">"Impossible de récupérer la liste des sources"</string> | ||||||
| @@ -120,12 +113,15 @@ | |||||||
|     <string name="mode_dark">Thème sombre</string> |     <string name="mode_dark">Thème sombre</string> | ||||||
|     <string name="mode_system">Utiliser les paramètres système</string> |     <string name="mode_system">Utiliser les paramètres système</string> | ||||||
|     <string name="mode_light">Thème clair</string> |     <string name="mode_light">Thème clair</string> | ||||||
|     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> |     <string name="gdpr_dialog_title">L\'application ne partage aucune information personnelle</string> | ||||||
|     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> |     <string name="gdpr_dialog_message"><![CDATA[Le rapport de plantage est activés par défaut. Il peut être désactivé depuis les paramètres de l\'application. Notez que les rapports de plantage sont essentiels pour le développement de l'application.]]></string> | ||||||
|     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> |     <string name="crash_toast_text">Un bug s\'est produit. Le développeur en sera informé.</string> | ||||||
|     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> |     <string name="pref_switch_disable_acra">"Désactiver les rapports de plantage."</string> | ||||||
|     <string name="menu_home_filter">Filters</string> |     <string name="menu_home_filter">Filtres</string> | ||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">Cette application ne fonctionne qu\'avec l\'api de Selfoss, et aucun autre.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Mise à jour des sources</string> | ||||||
|  |     <string name="confirm_disconnect_title">Se déconnecter ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">Vous allez être déconnecté de votre instance Selfoss.</string> | ||||||
|  |     <string name="disable_ssl">Désactiver la vérification SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Lector para selfoss"</string> |     <string name="app_name">"Lector para selfoss"</string> | ||||||
|     <string name="title_activity_login">"Conectar"</string> |     <string name="title_activity_login">"Conectar"</string> | ||||||
|     <string name="prompt_password">"Contrasinal"</string> |     <string name="prompt_password">"Contrasinal"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"Comprobar os teus detalles de novo."</string> |     <string name="wrong_infos">"Comprobar os teus detalles de novo."</string> | ||||||
|     <string name="all_posts_not_read">"Non se leron todas as publicacións"</string> |     <string name="all_posts_not_read">"Non se leron todas as publicacións"</string> | ||||||
|     <string name="all_posts_read">"Leronse todas as publicacións"</string> |     <string name="all_posts_read">"Leronse todas as publicacións"</string> | ||||||
|     <string name="nothing_here">"Non hai nada aquí"</string> |  | ||||||
|     <string name="tab_new">"Novo"</string> |  | ||||||
|     <string name="tab_read">"Todos"</string> |  | ||||||
|     <string name="tab_favs">"Favoritos"</string> |  | ||||||
|     <string name="action_about">"Acerca de"</string> |  | ||||||
|     <string name="marked_as_read">"Elemento lido"</string> |  | ||||||
|     <string name="marked_as_unread">"Elemento non lido"</string> |  | ||||||
|     <string name="undo_string">"Desfacer"</string> |     <string name="undo_string">"Desfacer"</string> | ||||||
|     <string name="addStringNoUrl">"Accede pra engadir fontes."</string> |     <string name="addStringNoUrl">"Accede pra engadir fontes."</string> | ||||||
|     <string name="cant_get_sources">"Non se pode obter a lista de fontes."</string> |     <string name="cant_get_sources">"Non se pode obter a lista de fontes."</string> | ||||||
|     <string name="cant_create_source">"Non se pode crear unha fonte."</string> |     <string name="cant_create_source">"Non se pode crear unha fonte."</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Non se pode obter a lista de spouts por mor dun erro de rede."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"Non se pode obter a lista de spoits. Pode que haxa algún problema coa api."</string> | ||||||
|     <string name="form_not_complete">"O formulario non está completo"</string> |     <string name="form_not_complete">"O formulario non está completo"</string> | ||||||
|     <string name="pref_header_links">"Ligazóns"</string> |     <string name="pref_header_links">"Ligazóns"</string> | ||||||
|     <string name="issue_tracker_link">"Rastrexador de Incidencias"</string> |     <string name="issue_tracker_link">"Rastrexador de Incidencias"</string> | ||||||
| @@ -116,16 +109,19 @@ | |||||||
|     <string name="reader_static_bar_on">A barra inferior mostrarase sempre</string> |     <string name="reader_static_bar_on">A barra inferior mostrarase sempre</string> | ||||||
|     <string name="reader_static_bar_off">A barra inferior pode mostrarse a través do botón flotante</string> |     <string name="reader_static_bar_off">A barra inferior pode mostrarse a través do botón flotante</string> | ||||||
|     <string name="remove_source">Eliminar fonte</string> |     <string name="remove_source">Eliminar fonte</string> | ||||||
|     <string name="pref_theme_title">Light/Dark mode</string> |     <string name="pref_theme_title">Modo Claro/Escuro</string> | ||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Modo escuro</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Seguir axustes do sistema</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Modo claro</string> | ||||||
|     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> |     <string name="gdpr_dialog_title">A aplicación non comparte ningún dato persoal seu.</string> | ||||||
|     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> |     <string name="gdpr_dialog_message"><![CDATA[O envío de informes de erros está habilitado. Pode deshabilitarse dende a páxina de axustes. Ten en conta que os informes de erros son esenciais para o desenvolvemento da aplicación.]]></string> | ||||||
|     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> |     <string name="crash_toast_text">Ocurriu un erro. Enviando os detalles o desenvolvedor.</string> | ||||||
|     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> |     <string name="pref_switch_disable_acra">"Deshabilitar o reporte automático de erros. "</string> | ||||||
|     <string name="menu_home_filter">Filters</string> |     <string name="menu_home_filter">Filtros</string> | ||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">Esta aplicación só funciona cunha instancia de Selfoss, e con ningún outro filtro RSS.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Masuk"</string> |     <string name="title_activity_login">"Masuk"</string> | ||||||
|     <string name="prompt_password">"Kata sandi"</string> |     <string name="prompt_password">"Kata sandi"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"Periksa kembali detail Anda."</string> |     <string name="wrong_infos">"Periksa kembali detail Anda."</string> | ||||||
|     <string name="all_posts_not_read">"Semua pos belum dibaca"</string> |     <string name="all_posts_not_read">"Semua pos belum dibaca"</string> | ||||||
|     <string name="all_posts_read">"Semua pos sudah dibaca"</string> |     <string name="all_posts_read">"Semua pos sudah dibaca"</string> | ||||||
|     <string name="nothing_here">"Tidak ada di sini"</string> |  | ||||||
|     <string name="tab_new">"Baru"</string> |  | ||||||
|     <string name="tab_read">"Semua"</string> |  | ||||||
|     <string name="tab_favs">"Favorit"</string> |  | ||||||
|     <string name="action_about">"Tentang"</string> |  | ||||||
|     <string name="marked_as_read">"Membaca item"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Urung"</string> |     <string name="undo_string">"Urung"</string> | ||||||
|     <string name="addStringNoUrl">"Masuk untuk menambah sumber."</string> |     <string name="addStringNoUrl">"Masuk untuk menambah sumber."</string> | ||||||
|     <string name="cant_get_sources">"Tidak bisa mendapatkan daftar sumber."</string> |     <string name="cant_get_sources">"Tidak bisa mendapatkan daftar sumber."</string> | ||||||
|     <string name="cant_create_source">"Tidak dapat membuat sumber."</string> |     <string name="cant_create_source">"Tidak dapat membuat sumber."</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"Tidak bisa masuk ke daftar Spouts."</string> | ||||||
|     <string name="form_not_complete">"Formulirnya belum selesai"</string> |     <string name="form_not_complete">"Formulirnya belum selesai"</string> | ||||||
|     <string name="pref_header_links">"Tautan"</string> |     <string name="pref_header_links">"Tautan"</string> | ||||||
|     <string name="issue_tracker_link">"Pelacak Masalah"</string> |     <string name="issue_tracker_link">"Pelacak Masalah"</string> | ||||||
| @@ -90,7 +83,7 @@ | |||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|     <string name="pref_switch_update_sources">Check for new sources and tags</string> |     <string name="pref_switch_update_sources">Check for new sources and tags</string> | ||||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> |     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> | ||||||
|     <string name="network_connectivity_lost">"Network connection lost"</string> |     <string name="network_connectivity_lost">"Koneksi jaringan hilang"</string> | ||||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> |     <string name="network_connectivity_retrieved">"Network connection is now available"</string> | ||||||
|     <string name="pref_switch_periodic_refresh">Sync articles</string> |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Lettore RSS per Selfoss"</string> |     <string name="app_name">"Lettore RSS per Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Accedi"</string> |     <string name="title_activity_login">"Accedi"</string> | ||||||
|     <string name="prompt_password">"Password"</string> |     <string name="prompt_password">"Password"</string> | ||||||
| @@ -22,13 +22,6 @@ | |||||||
|     <string name="wrong_infos">"Controlla nuovamente i dati."</string> |     <string name="wrong_infos">"Controlla nuovamente i dati."</string> | ||||||
|     <string name="all_posts_not_read">"All posts weren't read"</string> |     <string name="all_posts_not_read">"All posts weren't read"</string> | ||||||
|     <string name="all_posts_read">"Tutti i messaggi sono stati letti"</string> |     <string name="all_posts_read">"Tutti i messaggi sono stati letti"</string> | ||||||
|     <string name="nothing_here">"Non c'è niente qui"</string> |  | ||||||
|     <string name="tab_new">"Nuovi"</string> |  | ||||||
|     <string name="tab_read">"Tutti"</string> |  | ||||||
|     <string name="tab_favs">"Preferiti"</string> |  | ||||||
|     <string name="action_about">"Informazioni"</string> |  | ||||||
|     <string name="marked_as_read">"Articolo letto"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Annulla"</string> |     <string name="undo_string">"Annulla"</string> | ||||||
|     <string name="addStringNoUrl">"Autenticati per aggiungere fonti."</string> |     <string name="addStringNoUrl">"Autenticati per aggiungere fonti."</string> | ||||||
|     <string name="cant_get_sources">"Can't get sources list."</string> |     <string name="cant_get_sources">"Can't get sources list."</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"로그인"</string> |     <string name="title_activity_login">"로그인"</string> | ||||||
|     <string name="prompt_password">"비밀번호"</string> |     <string name="prompt_password">"비밀번호"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"세부 정보를 다시 확인하세요."</string> |     <string name="wrong_infos">"세부 정보를 다시 확인하세요."</string> | ||||||
|     <string name="all_posts_not_read">"모든 게시물을 읽지 않았습니다."</string> |     <string name="all_posts_not_read">"모든 게시물을 읽지 않았습니다."</string> | ||||||
|     <string name="all_posts_read">"모든 게시물을 읽었습니다."</string> |     <string name="all_posts_read">"모든 게시물을 읽었습니다."</string> | ||||||
|     <string name="nothing_here">"비어있음"</string> |  | ||||||
|     <string name="tab_new">"새로운"</string> |  | ||||||
|     <string name="tab_read">"전체"</string> |  | ||||||
|     <string name="tab_favs">"즐겨찾기"</string> |  | ||||||
|     <string name="action_about">"정보"</string> |  | ||||||
|     <string name="marked_as_read">"항목 읽기"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"실행 취소"</string> |     <string name="undo_string">"실행 취소"</string> | ||||||
|     <string name="addStringNoUrl">"로그인 소스를 추가 해야 합니다."</string> |     <string name="addStringNoUrl">"로그인 소스를 추가 해야 합니다."</string> | ||||||
|     <string name="cant_get_sources">"소스 리스트를 얻을 수 없습니다."</string> |     <string name="cant_get_sources">"소스 리스트를 얻을 수 없습니다."</string> | ||||||
|     <string name="cant_create_source">"소스를 만들 수 없습니다."</string> |     <string name="cant_create_source">"소스를 만들 수 없습니다."</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"Spouts 목록을 가져올 수 없습니다."</string> | ||||||
|     <string name="form_not_complete">"양식이 완료되지 않았습니다."</string> |     <string name="form_not_complete">"양식이 완료되지 않았습니다."</string> | ||||||
|     <string name="pref_header_links">"링크"</string> |     <string name="pref_header_links">"링크"</string> | ||||||
|     <string name="issue_tracker_link">"이슈 트래커"</string> |     <string name="issue_tracker_link">"이슈 트래커"</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <resources> |  | ||||||
|     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> |  | ||||||
|     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> |  | ||||||
|     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> |  | ||||||
|     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> |  | ||||||
|     <string name="menu_home_filter">Filters</string> |  | ||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |  | ||||||
|     <string name="menu_home_sources">Sources</string> |  | ||||||
|     <string name="update_source">Update source</string> |  | ||||||
| </resources> |  | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Selfoss Reader"</string> |     <string name="app_name">"Selfoss Reader"</string> | ||||||
|     <string name="title_activity_login">"Inloggen"</string> |     <string name="title_activity_login">"Inloggen"</string> | ||||||
|     <string name="prompt_password">"Wachtwoord"</string> |     <string name="prompt_password">"Wachtwoord"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"Controleer de gegevens nogmaals."</string> |     <string name="wrong_infos">"Controleer de gegevens nogmaals."</string> | ||||||
|     <string name="all_posts_not_read">"Fout bij markeren als gelezen"</string> |     <string name="all_posts_not_read">"Fout bij markeren als gelezen"</string> | ||||||
|     <string name="all_posts_read">"Alle artikelen gemarkeerd als gelezen"</string> |     <string name="all_posts_read">"Alle artikelen gemarkeerd als gelezen"</string> | ||||||
|     <string name="nothing_here">"Niets gevonden"</string> |  | ||||||
|     <string name="tab_new">"Nieuw"</string> |  | ||||||
|     <string name="tab_read">"Alle"</string> |  | ||||||
|     <string name="tab_favs">"Favorieten"</string> |  | ||||||
|     <string name="action_about">"Over"</string> |  | ||||||
|     <string name="marked_as_read">"Artikel gelezen"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Ongedaan maken"</string> |     <string name="undo_string">"Ongedaan maken"</string> | ||||||
|     <string name="addStringNoUrl">"Login om bronnen toe te voegen"</string> |     <string name="addStringNoUrl">"Login om bronnen toe te voegen"</string> | ||||||
|     <string name="cant_get_sources">"Kan de lijst met bronnen niet ophalen"</string> |     <string name="cant_get_sources">"Kan de lijst met bronnen niet ophalen"</string> | ||||||
|     <string name="cant_create_source">"Kan bron niet creëeren"</string> |     <string name="cant_create_source">"Kan bron niet creëeren"</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"Ophalen spouts mislukt"</string> | ||||||
|     <string name="form_not_complete">"Formulier is niet volledig ingevuld"</string> |     <string name="form_not_complete">"Formulier is niet volledig ingevuld"</string> | ||||||
|     <string name="pref_header_links">"Links"</string> |     <string name="pref_header_links">"Links"</string> | ||||||
|     <string name="issue_tracker_link">"Bug tracker"</string> |     <string name="issue_tracker_link">"Bug tracker"</string> | ||||||
| @@ -92,7 +85,7 @@ | |||||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> |     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> | ||||||
|     <string name="network_connectivity_lost">"Network connection lost"</string> |     <string name="network_connectivity_lost">"Network connection lost"</string> | ||||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> |     <string name="network_connectivity_retrieved">"Network connection is now available"</string> | ||||||
|     <string name="pref_switch_periodic_refresh">Sync articles</string> |     <string name="pref_switch_periodic_refresh">Artikel synchronisieren</string> | ||||||
|     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Entrar"</string> |     <string name="title_activity_login">"Entrar"</string> | ||||||
|     <string name="prompt_password">"Senha"</string> |     <string name="prompt_password">"Senha"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"Verifique os detalhes novamente."</string> |     <string name="wrong_infos">"Verifique os detalhes novamente."</string> | ||||||
|     <string name="all_posts_not_read">"Nenhum post foi lido"</string> |     <string name="all_posts_not_read">"Nenhum post foi lido"</string> | ||||||
|     <string name="all_posts_read">"Todos os posts foram lidos"</string> |     <string name="all_posts_read">"Todos os posts foram lidos"</string> | ||||||
|     <string name="nothing_here">"Nada aqui"</string> |  | ||||||
|     <string name="tab_new">"Novo"</string> |  | ||||||
|     <string name="tab_read">"Todos"</string> |  | ||||||
|     <string name="tab_favs">"Favoritos"</string> |  | ||||||
|     <string name="action_about">"Sobre"</string> |  | ||||||
|     <string name="marked_as_read">"Item lido"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Desfazer"</string> |     <string name="undo_string">"Desfazer"</string> | ||||||
|     <string name="addStringNoUrl">"Faça login para adicionar fontes."</string> |     <string name="addStringNoUrl">"Faça login para adicionar fontes."</string> | ||||||
|     <string name="cant_get_sources">"Não é possível obter a lista de fontes."</string> |     <string name="cant_get_sources">"Não é possível obter a lista de fontes."</string> | ||||||
|     <string name="cant_create_source">"Não é possível criar fonte."</string> |     <string name="cant_create_source">"Não é possível criar fonte."</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"Não é possível obter a lista de spouts."</string> | ||||||
|     <string name="form_not_complete">"O formulário não está completo"</string> |     <string name="form_not_complete">"O formulário não está completo"</string> | ||||||
|     <string name="pref_header_links">"Links"</string> |     <string name="pref_header_links">"Links"</string> | ||||||
|     <string name="issue_tracker_link">"Rastreador de problemas"</string> |     <string name="issue_tracker_link">"Rastreador de problemas"</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Leitor para Selfoss"</string> |     <string name="app_name">"Leitor para Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Iniciar sessão"</string> |     <string name="title_activity_login">"Iniciar sessão"</string> | ||||||
|     <string name="prompt_password">"Palavra passe"</string> |     <string name="prompt_password">"Palavra passe"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"Verifique seus dados novamente."</string> |     <string name="wrong_infos">"Verifique seus dados novamente."</string> | ||||||
|     <string name="all_posts_not_read">"Todas as postagens não foram lidas"</string> |     <string name="all_posts_not_read">"Todas as postagens não foram lidas"</string> | ||||||
|     <string name="all_posts_read">"Todas as postagens foram lidas"</string> |     <string name="all_posts_read">"Todas as postagens foram lidas"</string> | ||||||
|     <string name="nothing_here">"Nada aqui"</string> |  | ||||||
|     <string name="tab_new">"Novo"</string> |  | ||||||
|     <string name="tab_read">"Tudo"</string> |  | ||||||
|     <string name="tab_favs">"Favoritos"</string> |  | ||||||
|     <string name="action_about">"Sobre"</string> |  | ||||||
|     <string name="marked_as_read">"Item lido"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Desfazer"</string> |     <string name="undo_string">"Desfazer"</string> | ||||||
|     <string name="addStringNoUrl">"Logar para adicionar fontes."</string> |     <string name="addStringNoUrl">"Logar para adicionar fontes."</string> | ||||||
|     <string name="cant_get_sources">"Não é possível obter a lista de fontes."</string> |     <string name="cant_get_sources">"Não é possível obter a lista de fontes."</string> | ||||||
|     <string name="cant_create_source">"Não é possível criar a fonte."</string> |     <string name="cant_create_source">"Não é possível criar a fonte."</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"Não é possível obter a lista de bicos."</string> | ||||||
|     <string name="form_not_complete">"O formulário não está completo"</string> |     <string name="form_not_complete">"O formulário não está completo"</string> | ||||||
|     <string name="pref_header_links">"Links"</string> |     <string name="pref_header_links">"Links"</string> | ||||||
|     <string name="issue_tracker_link">"Rastreador de problemas"</string> |     <string name="issue_tracker_link">"Rastreador de problemas"</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"පිවිසෙන්න"</string> |     <string name="title_activity_login">"පිවිසෙන්න"</string> | ||||||
|     <string name="prompt_password">"මුර පදය"</string> |     <string name="prompt_password">"මුර පදය"</string> | ||||||
| @@ -22,13 +22,6 @@ | |||||||
|     <string name="wrong_infos">"Check your details again."</string> |     <string name="wrong_infos">"Check your details again."</string> | ||||||
|     <string name="all_posts_not_read">"All posts weren't read"</string> |     <string name="all_posts_not_read">"All posts weren't read"</string> | ||||||
|     <string name="all_posts_read">"All posts were read"</string> |     <string name="all_posts_read">"All posts were read"</string> | ||||||
|     <string name="nothing_here">"Nothing here"</string> |  | ||||||
|     <string name="tab_new">"New"</string> |  | ||||||
|     <string name="tab_read">"සියල්ල"</string> |  | ||||||
|     <string name="tab_favs">"Favorites"</string> |  | ||||||
|     <string name="action_about">"මේ ගැන"</string> |  | ||||||
|     <string name="marked_as_read">"Item read"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Undo"</string> |     <string name="undo_string">"Undo"</string> | ||||||
|     <string name="addStringNoUrl">"Log in to add sources."</string> |     <string name="addStringNoUrl">"Log in to add sources."</string> | ||||||
|     <string name="cant_get_sources">"Can't get sources list."</string> |     <string name="cant_get_sources">"Can't get sources list."</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Selfoss için okuyucu"</string> |     <string name="app_name">"Selfoss için okuyucu"</string> | ||||||
|     <string name="title_activity_login">"Giriş"</string> |     <string name="title_activity_login">"Giriş"</string> | ||||||
|     <string name="prompt_password">"Şifre"</string> |     <string name="prompt_password">"Şifre"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"Detaylarınızı tekrar kontrol edin."</string> |     <string name="wrong_infos">"Detaylarınızı tekrar kontrol edin."</string> | ||||||
|     <string name="all_posts_not_read">"Tüm mesajlar okunmadı"</string> |     <string name="all_posts_not_read">"Tüm mesajlar okunmadı"</string> | ||||||
|     <string name="all_posts_read">"Tüm mesajlar okundu"</string> |     <string name="all_posts_read">"Tüm mesajlar okundu"</string> | ||||||
|     <string name="nothing_here">"Burada hiçbir şey yok"</string> |  | ||||||
|     <string name="tab_new">"Yeni"</string> |  | ||||||
|     <string name="tab_read">"Tüm"</string> |  | ||||||
|     <string name="tab_favs">"Favoriler"</string> |  | ||||||
|     <string name="action_about">"Hakkında"</string> |  | ||||||
|     <string name="marked_as_read">"Öğeleri oku"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Geri al"</string> |     <string name="undo_string">"Geri al"</string> | ||||||
|     <string name="addStringNoUrl">"Kaynakları eklemek için giriş yapın."</string> |     <string name="addStringNoUrl">"Kaynakları eklemek için giriş yapın."</string> | ||||||
|     <string name="cant_get_sources">"Kaynakları listesi alınamıyor."</string> |     <string name="cant_get_sources">"Kaynakları listesi alınamıyor."</string> | ||||||
|     <string name="cant_create_source">"Kaynak oluşturulamıyor."</string> |     <string name="cant_create_source">"Kaynak oluşturulamıyor."</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"Spouts listesine girilemiyor."</string> | ||||||
|     <string name="form_not_complete">"Form tamamlanamadı"</string> |     <string name="form_not_complete">"Form tamamlanamadı"</string> | ||||||
|     <string name="pref_header_links">"Bağlantılar"</string> |     <string name="pref_header_links">"Bağlantılar"</string> | ||||||
|     <string name="issue_tracker_link">"Sorun İzleyici"</string> |     <string name="issue_tracker_link">"Sorun İzleyici"</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Selfoss 阅读器"</string> |     <string name="app_name">"Selfoss 阅读器"</string> | ||||||
|     <string name="title_activity_login">"登录"</string> |     <string name="title_activity_login">"登录"</string> | ||||||
|     <string name="prompt_password">"密码"</string> |     <string name="prompt_password">"密码"</string> | ||||||
| @@ -22,13 +22,6 @@ | |||||||
|     <string name="wrong_infos">"再次检查您的详细信息。"</string> |     <string name="wrong_infos">"再次检查您的详细信息。"</string> | ||||||
|     <string name="all_posts_not_read">"所有帖子都未读"</string> |     <string name="all_posts_not_read">"所有帖子都未读"</string> | ||||||
|     <string name="all_posts_read">"所有帖子已读"</string> |     <string name="all_posts_read">"所有帖子已读"</string> | ||||||
|     <string name="nothing_here">"暂无内容!"</string> |  | ||||||
|     <string name="tab_new">"新建"</string> |  | ||||||
|     <string name="tab_read">"所有"</string> |  | ||||||
|     <string name="tab_favs">"收藏夹"</string> |  | ||||||
|     <string name="action_about">"关于我们"</string> |  | ||||||
|     <string name="marked_as_read">"已读"</string> |  | ||||||
|     <string name="marked_as_unread">"未读条目"</string> |  | ||||||
|     <string name="undo_string">"撤销"</string> |     <string name="undo_string">"撤销"</string> | ||||||
|     <string name="addStringNoUrl">"登录以添加数据源。"</string> |     <string name="addStringNoUrl">"登录以添加数据源。"</string> | ||||||
|     <string name="cant_get_sources">"无法获取数据列表。"</string> |     <string name="cant_get_sources">"无法获取数据列表。"</string> | ||||||
| @@ -125,7 +118,10 @@ | |||||||
|     <string name="crash_toast_text">发生崩溃。请将细节发送给开发人员。</string> |     <string name="crash_toast_text">发生崩溃。请将细节发送给开发人员。</string> | ||||||
|     <string name="pref_switch_disable_acra">"禁用自动错误报告 "</string> |     <string name="pref_switch_disable_acra">"禁用自动错误报告 "</string> | ||||||
|     <string name="menu_home_filter">筛选器</string> |     <string name="menu_home_filter">筛选器</string> | ||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">此应用只适用于 Selfoss 实例,不适用于其他 RSS 。</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">源</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">更新源</string> | ||||||
|  |     <string name="confirm_disconnect_title">断开连接?</string> | ||||||
|  |     <string name="confirm_disconnect_description">您将断开与 selfoss 实例的连接。</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Selfoss 阅读器"</string> |     <string name="app_name">"Selfoss 阅读器"</string> | ||||||
|     <string name="title_activity_login">"登录"</string> |     <string name="title_activity_login">"登录"</string> | ||||||
|     <string name="prompt_password">"密码"</string> |     <string name="prompt_password">"密码"</string> | ||||||
| @@ -22,19 +22,12 @@ | |||||||
|     <string name="wrong_infos">"再次检查您的详细信息。"</string> |     <string name="wrong_infos">"再次检查您的详细信息。"</string> | ||||||
|     <string name="all_posts_not_read">"所有帖子都未读"</string> |     <string name="all_posts_not_read">"所有帖子都未读"</string> | ||||||
|     <string name="all_posts_read">"所有帖子已读"</string> |     <string name="all_posts_read">"所有帖子已读"</string> | ||||||
|     <string name="nothing_here">"暂无内容!"</string> |  | ||||||
|     <string name="tab_new">"新建"</string> |  | ||||||
|     <string name="tab_read">"所有"</string> |  | ||||||
|     <string name="tab_favs">"收藏夹"</string> |  | ||||||
|     <string name="action_about">"关于我们"</string> |  | ||||||
|     <string name="marked_as_read">"已读"</string> |  | ||||||
|     <string name="marked_as_unread">"未讀項目"</string> |  | ||||||
|     <string name="undo_string">"撤销"</string> |     <string name="undo_string">"撤销"</string> | ||||||
|     <string name="addStringNoUrl">"登录以添加数据源。"</string> |     <string name="addStringNoUrl">"登录以添加数据源。"</string> | ||||||
|     <string name="cant_get_sources">"无法获取数据列表。"</string> |     <string name="cant_get_sources">"无法获取数据列表。"</string> | ||||||
|     <string name="cant_create_source">"无法创建源数据。"</string> |     <string name="cant_create_source">"无法创建源数据。"</string> | ||||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> |     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> |     <string name="cant_get_spouts">"无法获取数据列表"</string> | ||||||
|     <string name="form_not_complete">"窗体未完成"</string> |     <string name="form_not_complete">"窗体未完成"</string> | ||||||
|     <string name="pref_header_links">"链接"</string> |     <string name="pref_header_links">"链接"</string> | ||||||
|     <string name="issue_tracker_link">"问题追踪器"</string> |     <string name="issue_tracker_link">"问题追踪器"</string> | ||||||
| @@ -128,4 +121,7 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
|  |     <string name="disable_ssl">Disable SSL</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Log in"</string> |     <string name="title_activity_login">"Log in"</string> | ||||||
|     <string name="prompt_password">"Password"</string> |     <string name="prompt_password">"Password"</string> | ||||||
| @@ -6,6 +6,7 @@ | |||||||
|     <string name="error_invalid_password">"Password not long enough"</string> |     <string name="error_invalid_password">"Password not long enough"</string> | ||||||
|     <string name="error_field_required">"Field required"</string> |     <string name="error_field_required">"Field required"</string> | ||||||
|     <string name="prompt_url">"Url"</string> |     <string name="prompt_url">"Url"</string> | ||||||
|  |     <string name="disable_ssl">"Disable SSL"</string> | ||||||
|     <string name="withLoginSwitch">"Login required ?"</string> |     <string name="withLoginSwitch">"Login required ?"</string> | ||||||
|     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> |     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> | ||||||
|     <string name="prompt_login">"Username"</string> |     <string name="prompt_login">"Username"</string> | ||||||
| @@ -21,13 +22,6 @@ | |||||||
|     <string name="wrong_infos">"Check your details again."</string> |     <string name="wrong_infos">"Check your details again."</string> | ||||||
|     <string name="all_posts_not_read">"All posts weren't read"</string> |     <string name="all_posts_not_read">"All posts weren't read"</string> | ||||||
|     <string name="all_posts_read">"All posts were read"</string> |     <string name="all_posts_read">"All posts were read"</string> | ||||||
|     <string name="nothing_here">"Nothing here"</string> |  | ||||||
|     <string name="tab_new">"New"</string> |  | ||||||
|     <string name="tab_read">"All"</string> |  | ||||||
|     <string name="tab_favs">"Favorites"</string> |  | ||||||
|     <string name="action_about">"About"</string> |  | ||||||
|     <string name="marked_as_read">"Item read"</string> |  | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |  | ||||||
|     <string name="undo_string">"Undo"</string> |     <string name="undo_string">"Undo"</string> | ||||||
|     <string name="addStringNoUrl">"Log in to add sources."</string> |     <string name="addStringNoUrl">"Log in to add sources."</string> | ||||||
|     <string name="cant_get_sources">"Can't get sources list."</string> |     <string name="cant_get_sources">"Can't get sources list."</string> | ||||||
| @@ -54,8 +48,7 @@ | |||||||
|     <string name="pref_switch_card_view_off">"The articles will be displayed as a list"</string> |     <string name="pref_switch_card_view_off">"The articles will be displayed as a list"</string> | ||||||
|     <string name="menu_home_refresh">"Update remote"</string> |     <string name="menu_home_refresh">"Update remote"</string> | ||||||
|     <string name="refresh_success_response">"The remote is updated, you can now reload the articles list"</string> |     <string name="refresh_success_response">"The remote is updated, you can now reload the articles list"</string> | ||||||
|     <string |     <string name="refresh_failer_message">"The update didn't work, try again later, or check your selfoss logs."</string> | ||||||
|         name="refresh_failer_message">"The update didn't work, try again later, or check your selfoss logs."</string> |  | ||||||
|     <string name="refresh_in_progress">"Refresh in progress"</string> |     <string name="refresh_in_progress">"Refresh in progress"</string> | ||||||
|     <string name="card_height_title">Full height cards</string> |     <string name="card_height_title">Full height cards</string> | ||||||
|     <string name="card_height_on">Cards height will adjust to its content</string> |     <string name="card_height_on">Cards height will adjust to its content</string> | ||||||
| @@ -131,4 +124,6 @@ | |||||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|     <string name="menu_home_sources">Sources</string> |     <string name="menu_home_sources">Sources</string> | ||||||
|     <string name="update_source">Update source</string> |     <string name="update_source">Update source</string> | ||||||
|  |     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||||
|  |     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||||
| </resources> | </resources> | ||||||
| @@ -32,4 +32,10 @@ | |||||||
|         <item name="android:colorBackgroundCacheHint">@null</item> |         <item name="android:colorBackgroundCacheHint">@null</item> | ||||||
|         <item name="android:windowIsTranslucent">true</item> |         <item name="android:windowIsTranslucent">true</item> | ||||||
|     </style> |     </style> | ||||||
|  |  | ||||||
|  |     <style name="circleImageView" parent=""> | ||||||
|  |         <item name="cornerFamily">rounded</item> | ||||||
|  |         <item name="cornerSize">50%</item> | ||||||
|  |     </style> | ||||||
|  |  | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -1,46 +0,0 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.repository |  | ||||||
|  |  | ||||||
| import bou.amine.apps.readerforselfossv2.utils.DateUtils |  | ||||||
| import junit.framework.TestCase.assertEquals |  | ||||||
| import kotlinx.datetime.LocalDateTime |  | ||||||
| import kotlinx.datetime.TimeZone |  | ||||||
| import kotlinx.datetime.toInstant |  | ||||||
| import org.junit.Test |  | ||||||
|  |  | ||||||
| class DatesTest { |  | ||||||
|  |  | ||||||
|     private val v3Date = "2013-04-07T13:43:00+01:00" |  | ||||||
|     private val v4Date = "2013-04-07 13:43:00" |  | ||||||
|     private val bug1Date = "2022-12-24T17:00:08+00" |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun v3_date_should_be_parsed() { |  | ||||||
|         val date = DateUtils.parseDate(v3Date) |  | ||||||
|         val expected = |  | ||||||
|             LocalDateTime(2013, 4, 7, 14, 43, 0, 0).toInstant(TimeZone.currentSystemDefault()) |  | ||||||
|                 .toEpochMilliseconds() |  | ||||||
|  |  | ||||||
|         assertEquals(date, expected) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun v4_date_should_be_parsed() { |  | ||||||
|         val date = DateUtils.parseDate(v4Date) |  | ||||||
|         val expected = |  | ||||||
|             LocalDateTime(2013, 4, 7, 13, 43, 0, 0).toInstant(TimeZone.currentSystemDefault()) |  | ||||||
|                 .toEpochMilliseconds() |  | ||||||
|  |  | ||||||
|         assertEquals(date, expected) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun bug1_date_should_be_parsed() { |  | ||||||
|         val date = DateUtils.parseDate(bug1Date) |  | ||||||
|         val expected = |  | ||||||
|             LocalDateTime(2022, 12, 24, 18, 0, 8, 0).toInstant(TimeZone.currentSystemDefault()) |  | ||||||
|                 .toEpochMilliseconds() |  | ||||||
|  |  | ||||||
|         assertEquals(date, expected) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -0,0 +1,77 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android.tests.robolectric | ||||||
|  |  | ||||||
|  | import android.widget.Button | ||||||
|  | import android.widget.EditText | ||||||
|  | import androidx.core.view.isVisible | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.LoginActivity | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.R | ||||||
|  | import com.google.android.material.switchmaterial.SwitchMaterial | ||||||
|  | import org.junit.Assert.assertEquals | ||||||
|  | import org.junit.Test | ||||||
|  | import org.junit.runner.RunWith | ||||||
|  | import org.robolectric.Robolectric | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @RunWith(RobotElectriqueRunnerclass::class) | ||||||
|  | class LoginActivityTest { | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun login_shouldDisplay() { | ||||||
|  |         Robolectric.buildActivity(LoginActivity::class.java).use { controller -> | ||||||
|  |             controller.setup() // Moves the Activity to the RESUMED state | ||||||
|  |  | ||||||
|  |             val activity = controller.get() | ||||||
|  |             assert(activity.findViewById<EditText>(R.id.urlView).isVisible) | ||||||
|  |             assert(activity.findViewById<SwitchMaterial>(R.id.selfSigned).isVisible) | ||||||
|  |             assert(activity.findViewById<SwitchMaterial>(R.id.selfSigned).isChecked.not()) | ||||||
|  |             assert(activity.findViewById<SwitchMaterial>(R.id.withLogin).isVisible) | ||||||
|  |             assert(activity.findViewById<SwitchMaterial>(R.id.withLogin).isChecked.not()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun urlError() { | ||||||
|  |         Robolectric.buildActivity(LoginActivity::class.java).use { controller -> | ||||||
|  |             controller.setup() // Moves the Activity to the RESUMED state | ||||||
|  |             val activity = controller.get() | ||||||
|  |  | ||||||
|  |             val urlView = activity.findViewById<EditText>(R.id.urlView) | ||||||
|  |             urlView.setText("172.17.0.1:8888") | ||||||
|  |  | ||||||
|  |             activity.findViewById<Button>(R.id.signInButton).performClick() | ||||||
|  |  | ||||||
|  |             urlView.performClick() | ||||||
|  |             assertEquals(activity.getString(R.string.login_url_problem), urlView.error) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun multiError() { | ||||||
|  |         Robolectric.buildActivity(LoginActivity::class.java).use { controller -> | ||||||
|  |             controller.setup() // Moves the Activity to the RESUMED state | ||||||
|  |             val activity = controller.get() | ||||||
|  |  | ||||||
|  |             val signInButton = activity.findViewById<Button>(R.id.signInButton) | ||||||
|  |             repeat(3) { signInButton.performClick() } | ||||||
|  |  | ||||||
|  |             // Vérifie que l'avertissement est affiché | ||||||
|  |             assertEquals(activity.getString(R.string.text_wrong_url), dialogMessage()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* @Test | ||||||
|  |      fun connect() { | ||||||
|  |          Robolectric.buildActivity(LoginActivity::class.java).use { controller -> | ||||||
|  |              controller.setup() // Moves the Activity to the RESUMED state | ||||||
|  |              val activity = controller.get() | ||||||
|  |              val signInButton = activity.findViewById<Button>(R.id.signInButton) | ||||||
|  |              val urlView = activity.findViewById<EditText>(R.id.urlView) | ||||||
|  |              urlView.setText("http://10.0.2.2:8888") | ||||||
|  |              signInButton.performClick() | ||||||
|  |  | ||||||
|  |              val expectedIntent = Intent(activity, HomeActivity::class.java) | ||||||
|  |              val actual = shadowOf(activity).nextStartedActivity | ||||||
|  |              assertEquals(expectedIntent.component, actual.component) | ||||||
|  |          } | ||||||
|  |      }*/ | ||||||
|  | } | ||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android.tests.robolectric | ||||||
|  |  | ||||||
|  | import org.robolectric.RobolectricTestRunner | ||||||
|  | import org.robolectric.annotation.Config | ||||||
|  |  | ||||||
|  | class RobotElectriqueRunnerclass(testClass: Class<*>?) : | ||||||
|  |     RobolectricTestRunner(testClass) { | ||||||
|  |  | ||||||
|  |     override fun buildGlobalConfig(): Config { | ||||||
|  |         return Config.Builder().setSdk(25, 30, 33).build() | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,23 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android.tests.robolectric | ||||||
|  |  | ||||||
|  | import android.view.Menu | ||||||
|  | import android.widget.TextView | ||||||
|  | import androidx.annotation.IdRes | ||||||
|  | import org.junit.Assert.assertTrue | ||||||
|  | import org.robolectric.shadows.ShadowDialog | ||||||
|  |  | ||||||
|  | fun dialogMessage(): String { | ||||||
|  |     val latestDialog = ShadowDialog.getLatestDialog() | ||||||
|  |     return latestDialog.findViewById<TextView>(android.R.id.message)?.text.toString() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun Menu.assertClickable(@IdRes id: Int) { | ||||||
|  |     this.assertVisible(id) | ||||||
|  |     val item = this.findItem(id) | ||||||
|  |     assertTrue(item.isEnabled) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun Menu.assertVisible(@IdRes id: Int) { | ||||||
|  |     val item = this.findItem(id) | ||||||
|  |     assertTrue(item.isVisible) | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.repository | package bou.amine.apps.readerforselfossv2.tests.repository | ||||||
| 
 | 
 | ||||||
| import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB | import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB | ||||||
| import bou.amine.apps.readerforselfossv2.dao.SOURCE | import bou.amine.apps.readerforselfossv2.dao.SOURCE | ||||||
| @@ -6,12 +6,22 @@ import bou.amine.apps.readerforselfossv2.dao.TAG | |||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.model.StatusAndData | import bou.amine.apps.readerforselfossv2.model.StatusAndData | ||||||
| import bou.amine.apps.readerforselfossv2.model.SuccessResponse | import bou.amine.apps.readerforselfossv2.model.SuccessResponse | ||||||
|  | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import bou.amine.apps.readerforselfossv2.rest.SelfossApi | import bou.amine.apps.readerforselfossv2.rest.SelfossApi | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| import bou.amine.apps.readerforselfossv2.utils.ItemType | import bou.amine.apps.readerforselfossv2.utils.ItemType | ||||||
| import bou.amine.apps.readerforselfossv2.utils.toView | import bou.amine.apps.readerforselfossv2.utils.toView | ||||||
| import io.mockk.* | import io.mockk.clearAllMocks | ||||||
| import junit.framework.TestCase.* | import io.mockk.coEvery | ||||||
|  | import io.mockk.coVerify | ||||||
|  | import io.mockk.every | ||||||
|  | import io.mockk.mockk | ||||||
|  | import io.mockk.verify | ||||||
|  | import junit.framework.TestCase.assertEquals | ||||||
|  | import junit.framework.TestCase.assertFalse | ||||||
|  | import junit.framework.TestCase.assertNotSame | ||||||
|  | import junit.framework.TestCase.assertSame | ||||||
|  | import junit.framework.TestCase.assertTrue | ||||||
| import kotlinx.coroutines.flow.MutableStateFlow | import kotlinx.coroutines.flow.MutableStateFlow | ||||||
| import kotlinx.coroutines.runBlocking | import kotlinx.coroutines.runBlocking | ||||||
| import org.junit.Assert.assertNotEquals | import org.junit.Assert.assertNotEquals | ||||||
| @@ -42,11 +52,11 @@ class RepositoryTest { | |||||||
|     private val NUMBER_STARRED = 20 |     private val NUMBER_STARRED = 20 | ||||||
|     private lateinit var repository: Repository |     private lateinit var repository: Repository | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     private fun initializeRepository( |     private fun initializeRepository( | ||||||
|         isConnectionAvailable: MutableStateFlow<Boolean> = MutableStateFlow( |         isConnectionAvailable: MutableStateFlow<Boolean> = | ||||||
|             true |             MutableStateFlow( | ||||||
|         ) |                 true, | ||||||
|  |             ), | ||||||
|     ) { |     ) { | ||||||
|         repository = Repository(api, appSettingsService, isConnectionAvailable, db) |         repository = Repository(api, appSettingsService, isConnectionAvailable, db) | ||||||
| 
 | 
 | ||||||
| @@ -64,13 +74,19 @@ class RepositoryTest { | |||||||
|         every { appSettingsService.isItemCachingEnabled() } returns false |         every { appSettingsService.isItemCachingEnabled() } returns false | ||||||
|         every { appSettingsService.isUpdateSourcesEnabled() } returns false |         every { appSettingsService.isUpdateSourcesEnabled() } returns false | ||||||
| 
 | 
 | ||||||
|         coEvery { api.apiInformation() } returns StatusAndData( |         coEvery { api.apiInformation() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = SelfossModel.ApiInformation("2.19-ba1e8e3", "4.0.0", SelfossModel.ApiConfiguration(false, true)) |                     data = SelfossModel.ApiInformation( | ||||||
|  |                         "2.19-ba1e8e3", | ||||||
|  |                         "4.0.0", | ||||||
|  |                         SelfossModel.ApiConfiguration(false, true) | ||||||
|  |                     ), | ||||||
|                 ) |                 ) | ||||||
|         coEvery { api.stats() } returns StatusAndData( |         coEvery { api.stats() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = SelfossModel.Stats(NUMBER_ARTICLES, NUMBER_UNREAD, NUMBER_STARRED) |                     data = SelfossModel.Stats(NUMBER_ARTICLES, NUMBER_UNREAD, NUMBER_STARRED), | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|         every { db.itemsQueries.deleteItemsWhereSource(any()) } returns Unit |         every { db.itemsQueries.deleteItemsWhereSource(any()) } returns Unit | ||||||
| @@ -116,9 +132,14 @@ class RepositoryTest { | |||||||
|     @Test |     @Test | ||||||
|     fun get_public_access() { |     fun get_public_access() { | ||||||
|         every { appSettingsService.updatePublicAccess(any()) } returns Unit |         every { appSettingsService.updatePublicAccess(any()) } returns Unit | ||||||
|         coEvery { api.apiInformation() } returns StatusAndData( |         coEvery { api.apiInformation() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = SelfossModel.ApiInformation("2.19-ba1e8e3", "4.0.0", SelfossModel.ApiConfiguration(true, true)) |                     data = SelfossModel.ApiInformation( | ||||||
|  |                         "2.19-ba1e8e3", | ||||||
|  |                         "4.0.0", | ||||||
|  |                         SelfossModel.ApiConfiguration(true, true) | ||||||
|  |                     ), | ||||||
|                 ) |                 ) | ||||||
|         every { appSettingsService.getUserName() } returns "" |         every { appSettingsService.getUserName() } returns "" | ||||||
| 
 | 
 | ||||||
| @@ -131,9 +152,14 @@ class RepositoryTest { | |||||||
|     @Test |     @Test | ||||||
|     fun get_public_access_username_not_empty() { |     fun get_public_access_username_not_empty() { | ||||||
|         every { appSettingsService.updatePublicAccess(any()) } returns Unit |         every { appSettingsService.updatePublicAccess(any()) } returns Unit | ||||||
|         coEvery { api.apiInformation() } returns StatusAndData( |         coEvery { api.apiInformation() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = SelfossModel.ApiInformation("2.19-ba1e8e3", "4.0.0", SelfossModel.ApiConfiguration(true, true)) |                     data = SelfossModel.ApiInformation( | ||||||
|  |                         "2.19-ba1e8e3", | ||||||
|  |                         "4.0.0", | ||||||
|  |                         SelfossModel.ApiConfiguration(true, true) | ||||||
|  |                     ), | ||||||
|                 ) |                 ) | ||||||
|         every { appSettingsService.getUserName() } returns "username" |         every { appSettingsService.getUserName() } returns "username" | ||||||
| 
 | 
 | ||||||
| @@ -146,9 +172,14 @@ class RepositoryTest { | |||||||
|     @Test |     @Test | ||||||
|     fun get_public_access_no_auth() { |     fun get_public_access_no_auth() { | ||||||
|         every { appSettingsService.updatePublicAccess(any()) } returns Unit |         every { appSettingsService.updatePublicAccess(any()) } returns Unit | ||||||
|         coEvery { api.apiInformation() } returns StatusAndData( |         coEvery { api.apiInformation() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = SelfossModel.ApiInformation("2.19-ba1e8e3", "4.0.0", SelfossModel.ApiConfiguration(true, false)) |                     data = SelfossModel.ApiInformation( | ||||||
|  |                         "2.19-ba1e8e3", | ||||||
|  |                         "4.0.0", | ||||||
|  |                         SelfossModel.ApiConfiguration(true, false) | ||||||
|  |                     ), | ||||||
|                 ) |                 ) | ||||||
|         every { appSettingsService.getUserName() } returns "" |         every { appSettingsService.getUserName() } returns "" | ||||||
| 
 | 
 | ||||||
| @@ -161,9 +192,14 @@ class RepositoryTest { | |||||||
|     @Test |     @Test | ||||||
|     fun get_public_access_disabled() { |     fun get_public_access_disabled() { | ||||||
|         every { appSettingsService.updatePublicAccess(any()) } returns Unit |         every { appSettingsService.updatePublicAccess(any()) } returns Unit | ||||||
|         coEvery { api.apiInformation() } returns StatusAndData( |         coEvery { api.apiInformation() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = SelfossModel.ApiInformation("2.19-ba1e8e3", "4.0.0", SelfossModel.ApiConfiguration(false, true)) |                     data = SelfossModel.ApiInformation( | ||||||
|  |                         "2.19-ba1e8e3", | ||||||
|  |                         "4.0.0", | ||||||
|  |                         SelfossModel.ApiConfiguration(false, true) | ||||||
|  |                     ), | ||||||
|                 ) |                 ) | ||||||
|         every { appSettingsService.getUserName() } returns "" |         every { appSettingsService.getUserName() } returns "" | ||||||
| 
 | 
 | ||||||
| @@ -182,7 +218,7 @@ class RepositoryTest { | |||||||
|         coEvery { api.getItems(any(), any(), any(), any(), any(), any(), any()) } returns |         coEvery { api.getItems(any(), any(), any(), any(), any(), any(), any()) } returns | ||||||
|                 StatusAndData( |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|                     data = generateTestApiItem(itemParameters) |                     data = generateTestApiItem(itemParameters), | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|         initializeRepository() |         initializeRepository() | ||||||
| @@ -264,7 +300,7 @@ class RepositoryTest { | |||||||
|         itemParameter3.tags = "Other, Tag" |         itemParameter3.tags = "Other, Tag" | ||||||
|         itemParameter3.id = "3" |         itemParameter3.id = "3" | ||||||
|         coEvery { db.itemsQueries.items().executeAsList() } returns generateTestDBItems( |         coEvery { db.itemsQueries.items().executeAsList() } returns generateTestDBItems( | ||||||
|             itemParameter1 |             itemParameter1, | ||||||
|         ) + |         ) + | ||||||
|                 generateTestDBItems(itemParameter2) + |                 generateTestDBItems(itemParameter2) + | ||||||
|                 generateTestDBItems(itemParameter3) |                 generateTestDBItems(itemParameter3) | ||||||
| @@ -292,7 +328,7 @@ class RepositoryTest { | |||||||
|         itemParameter3.sourcetitle = "Other" |         itemParameter3.sourcetitle = "Other" | ||||||
|         itemParameter3.id = "3" |         itemParameter3.id = "3" | ||||||
|         coEvery { db.itemsQueries.items().executeAsList() } returns generateTestDBItems( |         coEvery { db.itemsQueries.items().executeAsList() } returns generateTestDBItems( | ||||||
|             itemParameter1 |             itemParameter1, | ||||||
|         ) + |         ) + | ||||||
|                 generateTestDBItems(itemParameter2) + |                 generateTestDBItems(itemParameter2) + | ||||||
|                 generateTestDBItems(itemParameter3) |                 generateTestDBItems(itemParameter3) | ||||||
| @@ -300,15 +336,18 @@ class RepositoryTest { | |||||||
|         every { appSettingsService.isItemCachingEnabled() } returns true |         every { appSettingsService.isItemCachingEnabled() } returns true | ||||||
| 
 | 
 | ||||||
|         initializeRepository(MutableStateFlow(false)) |         initializeRepository(MutableStateFlow(false)) | ||||||
|         repository.setSourceFilter(SelfossModel.Source( |         repository.setSourceFilter( | ||||||
|  |             SelfossModel.SourceDetail( | ||||||
|                 1, |                 1, | ||||||
|                 "Test", |                 "Test", | ||||||
|  |                 null, | ||||||
|                 listOf("tags"), |                 listOf("tags"), | ||||||
|                 SPOUT, |                 SPOUT, | ||||||
|                 "", |                 "", | ||||||
|                 IMAGE_URL, |                 IMAGE_URL, | ||||||
|             SelfossModel.SourceParams("url") |                 SelfossModel.SourceParams("url"), | ||||||
|         )) |             ), | ||||||
|  |         ) | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             repository.getNewerItems() |             repository.getNewerItems() | ||||||
|         } |         } | ||||||
| @@ -591,13 +630,15 @@ class RepositoryTest { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private fun prepareTags(): Pair<List<SelfossModel.Tag>, List<TAG>> { |     private fun prepareTags(): Pair<List<SelfossModel.Tag>, List<TAG>> { | ||||||
|         val tags = listOf( |         val tags = | ||||||
|  |             listOf( | ||||||
|                 SelfossModel.Tag("test", "red", 6), |                 SelfossModel.Tag("test", "red", 6), | ||||||
|             SelfossModel.Tag("second", "yellow", 0) |                 SelfossModel.Tag("second", "yellow", 0), | ||||||
|             ) |             ) | ||||||
|         val tagsDB = listOf( |         val tagsDB = | ||||||
|  |             listOf( | ||||||
|                 TAG("test_DB", "red", 6), |                 TAG("test_DB", "red", 6), | ||||||
|             TAG("second_DB", "yellow", 0) |                 TAG("second_DB", "yellow", 0), | ||||||
|             ) |             ) | ||||||
| 
 | 
 | ||||||
|         coEvery { api.tags() } returns StatusAndData(success = true, data = tags) |         coEvery { api.tags() } returns StatusAndData(success = true, data = tags) | ||||||
| @@ -609,38 +650,42 @@ class RepositoryTest { | |||||||
|     fun get_sources() { |     fun get_sources() { | ||||||
|         val (sources, sourcesDB) = prepareSources() |         val (sources, sourcesDB) = prepareSources() | ||||||
|         initializeRepository() |         initializeRepository() | ||||||
|         var testSources: List<SelfossModel.Source>? |         var testSources: List<SelfossModel.Source> | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         assertSame(sources, testSources) |         assertEquals(sources, testSources) | ||||||
|         assertNotEquals(sourcesDB.map { it.toView() }, testSources) |         assertNotEquals(sourcesDB.map { it.toView() }, testSources) | ||||||
|         coVerify(exactly = 1) { api.sources() } |         coVerify(exactly = 1) { api.sourcesDetailed() } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private fun prepareSources(): Pair<ArrayList<SelfossModel.Source>, List<SOURCE>> { |     private fun prepareSources(): Pair<ArrayList<SelfossModel.SourceDetail>, List<SOURCE>> { | ||||||
|         val sources = arrayListOf( |         val sources = | ||||||
|             SelfossModel.Source( |             arrayListOf( | ||||||
|  |                 SelfossModel.SourceDetail( | ||||||
|                     1, |                     1, | ||||||
|                     "First source", |                     "First source", | ||||||
|  |                     null, | ||||||
|                     listOf("Test", "second"), |                     listOf("Test", "second"), | ||||||
|                     SPOUT, |                     SPOUT, | ||||||
|                     "", |                     "", | ||||||
|                     IMAGE_URL_2, |                     IMAGE_URL_2, | ||||||
|                 SelfossModel.SourceParams("url") |                     SelfossModel.SourceParams("url"), | ||||||
|                 ), |                 ), | ||||||
|             SelfossModel.Source( |                 SelfossModel.SourceDetail( | ||||||
|                     2, |                     2, | ||||||
|                     "Second source", |                     "Second source", | ||||||
|  |                     null, | ||||||
|                     listOf("second"), |                     listOf("second"), | ||||||
|                     SPOUT, |                     SPOUT, | ||||||
|                     "", |                     "", | ||||||
|                     IMAGE_URL, |                     IMAGE_URL, | ||||||
|                 SelfossModel.SourceParams("url") |                     SelfossModel.SourceParams("url"), | ||||||
|  |                 ), | ||||||
|             ) |             ) | ||||||
|         ) |         val sourcesDB = | ||||||
|         val sourcesDB = listOf( |             listOf( | ||||||
|                 SOURCE( |                 SOURCE( | ||||||
|                     "1", |                     "1", | ||||||
|                     "First DB source", |                     "First DB source", | ||||||
| @@ -648,7 +693,7 @@ class RepositoryTest { | |||||||
|                     SPOUT, |                     SPOUT, | ||||||
|                     "", |                     "", | ||||||
|                     IMAGE_URL_2, |                     IMAGE_URL_2, | ||||||
|                 "url" |                     "url", | ||||||
|                 ), |                 ), | ||||||
|                 SOURCE( |                 SOURCE( | ||||||
|                     "2", |                     "2", | ||||||
| @@ -657,11 +702,11 @@ class RepositoryTest { | |||||||
|                     SPOUT, |                     SPOUT, | ||||||
|                     "", |                     "", | ||||||
|                     IMAGE_URL, |                     IMAGE_URL, | ||||||
|                 "url" |                     "url", | ||||||
|             ) |                 ), | ||||||
|             ) |             ) | ||||||
| 
 | 
 | ||||||
|         coEvery { api.sources() } returns StatusAndData(success = true, data = sources) |         coEvery { api.sourcesDetailed() } returns StatusAndData(success = true, data = sources) | ||||||
|         every { db.sourcesQueries.sources().executeAsList() } returns sourcesDB |         every { db.sourcesQueries.sources().executeAsList() } returns sourcesDB | ||||||
|         return Pair(sources, sourcesDB) |         return Pair(sources, sourcesDB) | ||||||
|     } |     } | ||||||
| @@ -675,13 +720,13 @@ class RepositoryTest { | |||||||
|         initializeRepository() |         initializeRepository() | ||||||
|         var testSources: List<SelfossModel.Source>? |         var testSources: List<SelfossModel.Source>? | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|             // Sources will be fetched from the database on the second call, thus testSources != sources |             // Sources will be fetched from the database on the second call, thus testSources != sources | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         coVerify(exactly = 1) { api.sources() } |         coVerify(exactly = 1) { api.sourcesDetailed() } | ||||||
|         assertNotSame(sources, testSources) |         assertNotEquals(sources, testSources) | ||||||
|         assertEquals(sourcesDB.map { it.toView() }, testSources) |         assertEquals(sourcesDB.map { it.toView() }, testSources) | ||||||
|         verify(atLeast = 1) { db.sourcesQueries.sources().executeAsList() } |         verify(atLeast = 1) { db.sourcesQueries.sources().executeAsList() } | ||||||
|     } |     } | ||||||
| @@ -693,13 +738,13 @@ class RepositoryTest { | |||||||
|         every { appSettingsService.isUpdateSourcesEnabled() } returns true |         every { appSettingsService.isUpdateSourcesEnabled() } returns true | ||||||
|         every { appSettingsService.isItemCachingEnabled() } returns false |         every { appSettingsService.isItemCachingEnabled() } returns false | ||||||
|         initializeRepository() |         initializeRepository() | ||||||
|         var testSources: List<SelfossModel.Source>? |         var testSources: List<SelfossModel.Source> | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         assertSame(sources, testSources) |         assertEquals(sources, testSources) | ||||||
|         coVerify(exactly = 1) { api.sources() } |         coVerify(exactly = 1) { api.sourcesDetailed() } | ||||||
|         verify(exactly = 0) { db.sourcesQueries } |         verify(exactly = 0) { db.sourcesQueries } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -710,13 +755,13 @@ class RepositoryTest { | |||||||
|         every { appSettingsService.isUpdateSourcesEnabled() } returns false |         every { appSettingsService.isUpdateSourcesEnabled() } returns false | ||||||
|         every { appSettingsService.isItemCachingEnabled() } returns false |         every { appSettingsService.isItemCachingEnabled() } returns false | ||||||
|         initializeRepository() |         initializeRepository() | ||||||
|         var testSources: List<SelfossModel.Source>? |         var testSources: List<SelfossModel.Source> | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         assertSame(sources, testSources) |         assertEquals(sources, testSources) | ||||||
|         coVerify(exactly = 1) { api.sources() } |         coVerify(exactly = 1) { api.sourcesDetailed() } | ||||||
|         verify(atLeast = 1) { db.sourcesQueries } |         verify(atLeast = 1) { db.sourcesQueries } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -724,13 +769,13 @@ class RepositoryTest { | |||||||
|     fun get_sources_without_connection() { |     fun get_sources_without_connection() { | ||||||
|         val (_, sourcesDB) = prepareSources() |         val (_, sourcesDB) = prepareSources() | ||||||
|         initializeRepository(MutableStateFlow(false)) |         initializeRepository(MutableStateFlow(false)) | ||||||
|         var testSources: List<SelfossModel.Source>? |         var testSources: List<SelfossModel.Source> | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         assertEquals(sourcesDB.map { it.toView() }, testSources) |         assertEquals(sourcesDB.map { it.toView() }, testSources) | ||||||
|         coVerify(exactly = 0) { api.sources() } |         coVerify(exactly = 0) { api.sourcesDetailed() } | ||||||
|         verify(atLeast = 1) { db.sourcesQueries.sources().executeAsList() } |         verify(atLeast = 1) { db.sourcesQueries.sources().executeAsList() } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -741,13 +786,13 @@ class RepositoryTest { | |||||||
|         every { appSettingsService.isItemCachingEnabled() } returns false |         every { appSettingsService.isItemCachingEnabled() } returns false | ||||||
|         every { appSettingsService.isUpdateSourcesEnabled() } returns true |         every { appSettingsService.isUpdateSourcesEnabled() } returns true | ||||||
|         initializeRepository(MutableStateFlow(false)) |         initializeRepository(MutableStateFlow(false)) | ||||||
|         var testSources: List<SelfossModel.Source>? |         var testSources: List<SelfossModel.Source> | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         assertEquals(emptyList<SelfossModel.Source>(), testSources) |         assertEquals(emptyList<SelfossModel.Source>(), testSources) | ||||||
|         coVerify(exactly = 0) { api.sources() } |         coVerify(exactly = 0) { api.sourcesDetailed() } | ||||||
|         verify(exactly = 0) { db.sourcesQueries.sources().executeAsList() } |         verify(exactly = 0) { db.sourcesQueries.sources().executeAsList() } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -758,13 +803,13 @@ class RepositoryTest { | |||||||
|         every { appSettingsService.isItemCachingEnabled() } returns true |         every { appSettingsService.isItemCachingEnabled() } returns true | ||||||
|         every { appSettingsService.isUpdateSourcesEnabled() } returns false |         every { appSettingsService.isUpdateSourcesEnabled() } returns false | ||||||
|         initializeRepository(MutableStateFlow(false)) |         initializeRepository(MutableStateFlow(false)) | ||||||
|         var testSources: List<SelfossModel.Source>? |         var testSources: List<SelfossModel.Source> | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         assertEquals(sourcesDB.map { it.toView() }, testSources) |         assertEquals(sourcesDB.map { it.toView() }, testSources) | ||||||
|         coVerify(exactly = 0) { api.sources() } |         coVerify(exactly = 0) { api.sourcesDetailed() } | ||||||
|         verify(atLeast = 1) { db.sourcesQueries.sources().executeAsList() } |         verify(atLeast = 1) { db.sourcesQueries.sources().executeAsList() } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -775,13 +820,13 @@ class RepositoryTest { | |||||||
|         every { appSettingsService.isItemCachingEnabled() } returns false |         every { appSettingsService.isItemCachingEnabled() } returns false | ||||||
|         every { appSettingsService.isUpdateSourcesEnabled() } returns false |         every { appSettingsService.isUpdateSourcesEnabled() } returns false | ||||||
|         initializeRepository(MutableStateFlow(false)) |         initializeRepository(MutableStateFlow(false)) | ||||||
|         var testSources: List<SelfossModel.Source>? |         var testSources: List<SelfossModel.Source> | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             testSources = repository.getSources() |             testSources = repository.getSourcesDetails() | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         assertEquals(sourcesDB.map { it.toView() }, testSources) |         assertEquals(sourcesDB.map { it.toView() }, testSources) | ||||||
|         coVerify(exactly = 0) { api.sources() } |         coVerify(exactly = 0) { api.sourcesDetailed() } | ||||||
|         verify(atLeast = 1) { db.sourcesQueries.sources().executeAsList() } |         verify(atLeast = 1) { db.sourcesQueries.sources().executeAsList() } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -793,7 +838,8 @@ class RepositoryTest { | |||||||
|         initializeRepository() |         initializeRepository() | ||||||
|         var response: Boolean |         var response: Boolean | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             response = repository.createSource( |             response = | ||||||
|  |                 repository.createSource( | ||||||
|                     "test", |                     "test", | ||||||
|                     FEED_URL, |                     FEED_URL, | ||||||
|                     SPOUT, |                     SPOUT, | ||||||
| @@ -820,11 +866,12 @@ class RepositoryTest { | |||||||
|         initializeRepository() |         initializeRepository() | ||||||
|         var response: Boolean |         var response: Boolean | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             response = repository.createSource( |             response = | ||||||
|  |                 repository.createSource( | ||||||
|                     "test", |                     "test", | ||||||
|                     FEED_URL, |                     FEED_URL, | ||||||
|                     SPOUT, |                     SPOUT, | ||||||
|                 TAGS |                     TAGS, | ||||||
|                 ) |                 ) | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
| @@ -833,7 +880,7 @@ class RepositoryTest { | |||||||
|                 any(), |                 any(), | ||||||
|                 any(), |                 any(), | ||||||
|                 any(), |                 any(), | ||||||
|                 any() |                 any(), | ||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|         assertSame(false, response) |         assertSame(false, response) | ||||||
| @@ -847,11 +894,12 @@ class RepositoryTest { | |||||||
|         initializeRepository(MutableStateFlow(false)) |         initializeRepository(MutableStateFlow(false)) | ||||||
|         var response: Boolean |         var response: Boolean | ||||||
|         runBlocking { |         runBlocking { | ||||||
|             response = repository.createSource( |             response = | ||||||
|  |                 repository.createSource( | ||||||
|                     "test", |                     "test", | ||||||
|                     FEED_URL, |                     FEED_URL, | ||||||
|                     SPOUT, |                     SPOUT, | ||||||
|                 TAGS |                     TAGS, | ||||||
|                 ) |                 ) | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
| @@ -913,9 +961,10 @@ class RepositoryTest { | |||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     fun update_remote() { |     fun update_remote() { | ||||||
|         coEvery { api.update() } returns StatusAndData( |         coEvery { api.update() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = "finished" |                     data = "finished", | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|         initializeRepository() |         initializeRepository() | ||||||
| @@ -930,9 +979,10 @@ class RepositoryTest { | |||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     fun update_remote_but_response_fails() { |     fun update_remote_but_response_fails() { | ||||||
|         coEvery { api.update() } returns StatusAndData( |         coEvery { api.update() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = false, |                     success = false, | ||||||
|             data = "unallowed access" |                     data = "unallowed access", | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|         initializeRepository() |         initializeRepository() | ||||||
| @@ -947,9 +997,10 @@ class RepositoryTest { | |||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     fun update_remote_with_unallowed_access() { |     fun update_remote_with_unallowed_access() { | ||||||
|         coEvery { api.update() } returns StatusAndData( |         coEvery { api.update() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = "unallowed access" |                     data = "unallowed access", | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|         initializeRepository() |         initializeRepository() | ||||||
| @@ -964,9 +1015,10 @@ class RepositoryTest { | |||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     fun update_remote_without_connection() { |     fun update_remote_without_connection() { | ||||||
|         coEvery { api.update() } returns StatusAndData( |         coEvery { api.update() } returns | ||||||
|  |                 StatusAndData( | ||||||
|                     success = true, |                     success = true, | ||||||
|             data = "undocumented..." |                     data = "undocumented...", | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|         initializeRepository(MutableStateFlow(false)) |         initializeRepository(MutableStateFlow(false)) | ||||||
| @@ -1034,7 +1086,7 @@ class RepositoryTest { | |||||||
|             appSettingsService.refreshLoginInformation( |             appSettingsService.refreshLoginInformation( | ||||||
|                 BASE_URL, |                 BASE_URL, | ||||||
|                 "login", |                 "login", | ||||||
|                 "password" |                 "password", | ||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -1054,9 +1106,10 @@ class RepositoryTest { | |||||||
|                 any(), |                 any(), | ||||||
|                 any(), |                 any(), | ||||||
|                 any(), |                 any(), | ||||||
|                 any() |                 any(), | ||||||
|             ) |             ) | ||||||
|         } returnsMany listOf( |         } returnsMany | ||||||
|  |                 listOf( | ||||||
|                     StatusAndData(success = true, data = generateTestApiItem(itemParameter1)), |                     StatusAndData(success = true, data = generateTestApiItem(itemParameter1)), | ||||||
|                     StatusAndData(success = true, data = generateTestApiItem(itemParameter2)), |                     StatusAndData(success = true, data = generateTestApiItem(itemParameter2)), | ||||||
|                     StatusAndData(success = true, data = generateTestApiItem(itemParameter1)), |                     StatusAndData(success = true, data = generateTestApiItem(itemParameter1)), | ||||||
| @@ -1102,15 +1155,16 @@ class RepositoryTest { | |||||||
|     private fun prepareSearch() { |     private fun prepareSearch() { | ||||||
|         repository.setTagFilter(SelfossModel.Tag("Tag", "read", 0)) |         repository.setTagFilter(SelfossModel.Tag("Tag", "read", 0)) | ||||||
|         repository.setSourceFilter( |         repository.setSourceFilter( | ||||||
|             SelfossModel.Source( |             SelfossModel.SourceDetail( | ||||||
|                 1, |                 1, | ||||||
|                 "First source", |                 "First source", | ||||||
|  |                 5, | ||||||
|                 listOf("Test", "second"), |                 listOf("Test", "second"), | ||||||
|                 SPOUT, |                 SPOUT, | ||||||
|                 "", |                 "", | ||||||
|                 IMAGE_URL_2, |                 IMAGE_URL_2, | ||||||
|                 SelfossModel.SourceParams("url") |                 SelfossModel.SourceParams("url"), | ||||||
|             ) |             ), | ||||||
|         ) |         ) | ||||||
|         repository.searchFilter = "search" |         repository.searchFilter = "search" | ||||||
|     } |     } | ||||||
| @@ -1,9 +1,8 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.repository | package bou.amine.apps.readerforselfossv2.tests.repository | ||||||
| 
 | 
 | ||||||
| import bou.amine.apps.readerforselfossv2.dao.ITEM | import bou.amine.apps.readerforselfossv2.dao.ITEM | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| fun generateTestDBItems(item: FakeItemParameters = FakeItemParameters()): List<ITEM> { | fun generateTestDBItems(item: FakeItemParameters = FakeItemParameters()): List<ITEM> { | ||||||
|     return listOf( |     return listOf( | ||||||
|         ITEM( |         ITEM( | ||||||
| @@ -18,8 +17,8 @@ fun generateTestDBItems(item: FakeItemParameters = FakeItemParameters()): List<I | |||||||
|             link = item.link, |             link = item.link, | ||||||
|             sourcetitle = item.sourcetitle, |             sourcetitle = item.sourcetitle, | ||||||
|             tags = item.tags, |             tags = item.tags, | ||||||
|             author = item.author |             author = item.author, | ||||||
|         ) |         ), | ||||||
|     ) |     ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @@ -37,8 +36,8 @@ fun generateTestApiItem(item: FakeItemParameters = FakeItemParameters()): List<S | |||||||
|             link = item.link, |             link = item.link, | ||||||
|             sourcetitle = item.sourcetitle, |             sourcetitle = item.sourcetitle, | ||||||
|             tags = item.tags.split(','), |             tags = item.tags.split(','), | ||||||
|             author = item.author |             author = item.author, | ||||||
|         ) |         ), | ||||||
|     ) |     ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @@ -1,35 +1,34 @@ | |||||||
| buildscript { | buildscript { | ||||||
|     dependencies { |     dependencies { | ||||||
|         // SqlDelight |         // SqlDelight | ||||||
|         classpath("com.squareup.sqldelight:gradle-plugin:1.5.4") |         classpath("com.squareup.sqldelight:gradle-plugin:1.5.5") | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| plugins { | plugins { | ||||||
|     //trick: for the same plugin versions in all sub-modules |     //trick: for the same plugin versions in all sub-modules | ||||||
|     id("com.android.application").version("7.4.0").apply(false) |     id("com.android.application").version("8.7.3").apply(false) | ||||||
|     id("com.android.library").version("7.4.0").apply(false) |     id("com.android.library").version("8.7.3").apply(false) | ||||||
|     kotlin("android").version("1.7.20").apply(false) |     id("org.jetbrains.kotlin.android").version("2.1.0").apply(false) | ||||||
|     kotlin("multiplatform").version("1.7.20").apply(false) |     kotlin("multiplatform").version("2.1.0").apply(false) | ||||||
|     id("com.mikepenz.aboutlibraries.plugin").version("10.5.1").apply(false) |     id("com.mikepenz.aboutlibraries.plugin").version("10.5.1").apply(false) | ||||||
|     id("org.jetbrains.kotlinx.kover") version "0.6.1" |     id("org.jetbrains.kotlinx.kover") version "0.9.0" apply true | ||||||
| } | } | ||||||
|  |  | ||||||
| allprojects { | allprojects { | ||||||
|     repositories { |     repositories { | ||||||
|         // maven { url = uri("https://nexus.amine-louveau.fr/repository/maven-public/")} |  | ||||||
|         google() |         google() | ||||||
|         mavenCentral() |         mavenCentral() | ||||||
|         jcenter() |  | ||||||
|         maven { url = uri("https://www.jitpack.io") } |         maven { url = uri("https://www.jitpack.io") } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| tasks.register("clean", Delete::class) { | tasks.register("clean", Delete::class) { | ||||||
|     delete(rootProject.buildDir) |     delete(layout.buildDirectory) | ||||||
| } | } | ||||||
|  |  | ||||||
| koverMerged { | dependencies { | ||||||
|     enable() |     kover(project(":shared")) | ||||||
|  |     kover(project(":androidApp")) | ||||||
| } | } | ||||||
| @@ -68,9 +68,9 @@ redirect_from: "/ReaderforSelfoss-multiplatform/" | |||||||
|             <div id="links"> |             <div id="links"> | ||||||
|  |  | ||||||
|  |  | ||||||
|                 <a class="github-button" href="https://gitea.amine-louveau.fr/Louvorg/readerforselfoss-multiplatform" data-size="large" aria-label="Star aminecmi/readerforselfoss-multiplatform on GitHub">Star</a> |                 <a class="github-button" href="https://gitea.amine-bouabdallaoui.fr/Louvorg/readerforselfoss-multiplatform" data-size="large" aria-label="Star aminecmi/readerforselfoss-multiplatform on GitHub">Star</a> | ||||||
|             </div> |             </div> | ||||||
|             <meta itemprop="url" content="https://gitea.amine-louveau.fr/Louvorg/readerforselfoss-multiplatform"> |             <meta itemprop="url" content="https://gitea.amine-bouabdallaoui.fr/Louvorg/readerforselfoss-multiplatform"> | ||||||
|             <meta itemprop="applicationCategory" content="News & Magazines"> |             <meta itemprop="applicationCategory" content="News & Magazines"> | ||||||
|         </div> |         </div> | ||||||
|     </body> |     </body> | ||||||
|   | |||||||
							
								
								
									
										0
									
								
								fastlane/metadata/android/en-US/changelogs/.gitkeep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								fastlane/metadata/android/en-US/changelogs/.gitkeep
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | **v124113311** | ||||||
|  |  | ||||||
|  | - chore: update versions. (#165) | ||||||
|  | - chore: fastlane changelog. | ||||||
|  | - chore: fastlane fixes. | ||||||
|  | - Changelog for v124113301 | ||||||
| @@ -0,0 +1,4 @@ | |||||||
|  | **v124123421** | ||||||
|  |  | ||||||
|  | - fix: Trying to fix the serialization issue. | ||||||
|  | - Changelog for v124113311 | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | **v124123641** | ||||||
|  |  | ||||||
|  | - Chore: no tests on build. | ||||||
|  | - Merge pull request 'testing' (#170) from testing into master | ||||||
|  | - fix: Displaying fixes. Fixes #155 | ||||||
|  | - test: coverage | ||||||
|  | - chore: update and use multiplatform datetime | ||||||
|  | - Changelog for v124123421 | ||||||
							
								
								
									
										
											BIN
										
									
								
								fastlane/metadata/android/en-US/images/icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								fastlane/metadata/android/en-US/images/icon.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 294 KiB | 
| @@ -1 +1 @@ | |||||||
| A new RSS reader for <a href="http://selfoss.aditu.de/">selfoss</a>. | A new RSS reader for selfoss (http://selfoss.aditu.de/) | ||||||
|   | |||||||
| @@ -13,24 +13,17 @@ | |||||||
| #Tue Mar 22 16:50:00 CET 2022 | #Tue Mar 22 16:50:00 CET 2022 | ||||||
| #Gradle | #Gradle | ||||||
| org.gradle.jvmargs=-Xmx2048M -Dfile.encoding=UTF-8 -Dkotlin.daemon.jvm.options\="-Xmx2048M" | org.gradle.jvmargs=-Xmx2048M -Dfile.encoding=UTF-8 -Dkotlin.daemon.jvm.options\="-Xmx2048M" | ||||||
|  |  | ||||||
| #Kotlin | #Kotlin | ||||||
| kotlin.code.style=official | kotlin.code.style=official | ||||||
|  |  | ||||||
| #Android | #Android | ||||||
| android.useAndroidX=true | android.useAndroidX=true | ||||||
| kotlin.native.enableDependencyPropagation=false |  | ||||||
| #android.nonTransitiveRClass=true | #android.nonTransitiveRClass=true | ||||||
| android.enableJetifier=true | android.enableJetifier=true | ||||||
|  | android.nonTransitiveRClass=false | ||||||
|  |  | ||||||
| #MPP | #MPP | ||||||
| kotlin.mpp.enableCInteropCommonization=true | kotlin.mpp.enableCInteropCommonization=true | ||||||
| kotlin.mpp.enableGranularSourceSetsMetadata=true |  | ||||||
|  |  | ||||||
|  |  | ||||||
| org.gradle.parallel=true | org.gradle.parallel=true | ||||||
| org.gradle.caching=true | org.gradle.caching=true | ||||||
| ignoreGitVersion=false | ignoreGitVersion=false | ||||||
| kotlin.native.cacheKind.iosX64=none | kotlin.native.cacheKind.iosX64=none | ||||||
| pushCache=true | org.gradle.configureondemand=true | ||||||
							
								
								
									
										6
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| #Mon Jan 23 20:47:46 CET 2023 | #Mon Nov 25 22:48:24 CET 2024 | ||||||
| distributionBase=GRADLE_USER_HOME | distributionBase=GRADLE_USER_HOME | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-7.5-bin.zip |  | ||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
| zipStorePath=wrapper/dists | distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
|  | zipStorePath=wrapper/dists | ||||||
|   | |||||||
| @@ -173,7 +173,7 @@ | |||||||
| 			); | 			); | ||||||
| 			runOnlyForDeploymentPostprocessing = 0; | 			runOnlyForDeploymentPostprocessing = 0; | ||||||
| 			shellPath = /bin/sh; | 			shellPath = /bin/sh; | ||||||
| 			shellScript = "cd \"$SRCROOT/..\"\n./gradlew :shared:embedAndSignAppleFrameworkForXcode\n"; | 			shellScript = "export JAVA_HOME=/Users/amine/.sdkman/candidates/java/17.0.8.1-jbr\ncd \"$SRCROOT/..\"\n./gradlew :shared:embedAndSignAppleFrameworkForXcode --stacktrace\n"; | ||||||
| 		}; | 		}; | ||||||
| /* End PBXShellScriptBuildPhase section */ | /* End PBXShellScriptBuildPhase section */ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,8 +1,5 @@ | |||||||
| val pushCache: String by settings |  | ||||||
|  |  | ||||||
| pluginManagement { | pluginManagement { | ||||||
|     repositories { |     repositories { | ||||||
|         // maven { url = uri("https://nexus.amine-louveau.fr/repository/maven-public/")} |  | ||||||
|         google() |         google() | ||||||
|         gradlePluginPortal() |         gradlePluginPortal() | ||||||
|         mavenCentral() |         mavenCentral() | ||||||
| @@ -11,23 +8,11 @@ pluginManagement { | |||||||
|  |  | ||||||
| dependencyResolutionManagement { | dependencyResolutionManagement { | ||||||
|     repositories { |     repositories { | ||||||
|         // maven { url = uri("https://nexus.amine-louveau.fr/repository/maven-public/")} |  | ||||||
|         google() |         google() | ||||||
|         mavenCentral() |         mavenCentral() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| buildCache { |  | ||||||
|     remote<HttpBuildCache> { |  | ||||||
|         url = uri("http://18.0.0.7:3071/cache/") |  | ||||||
|         isAllowInsecureProtocol = true |  | ||||||
|         isAllowUntrustedServer = true |  | ||||||
|         isUseExpectContinue = true |  | ||||||
|         isPush = (pushCache == "true") |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| rootProject.name = "ReaderForSelfossV2" | rootProject.name = "ReaderForSelfossV2" | ||||||
| include(":androidApp") | include(":androidApp") | ||||||
| include(":shared") | include(":shared") | ||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | val ktorVersion = "2.3.2" | ||||||
|  |  | ||||||
| object SqlDelight { | object SqlDelight { | ||||||
|     const val runtime = "com.squareup.sqldelight:runtime:1.5.4" |     const val runtime = "com.squareup.sqldelight:runtime:1.5.4" | ||||||
|     const val android = "com.squareup.sqldelight:android-driver:1.5.4" |     const val android = "com.squareup.sqldelight:android-driver:1.5.4" | ||||||
| @@ -9,12 +11,12 @@ plugins { | |||||||
|     kotlin("multiplatform") |     kotlin("multiplatform") | ||||||
|     id("com.android.library") |     id("com.android.library") | ||||||
|     id("com.squareup.sqldelight") |     id("com.squareup.sqldelight") | ||||||
|     kotlin("plugin.serialization") version "1.4.10" |     kotlin("plugin.serialization") version "1.9.0" | ||||||
|     id("org.jetbrains.kotlinx.kover") version "0.6.1" |     id("org.jetbrains.kotlinx.kover") | ||||||
| } | } | ||||||
|  |  | ||||||
| kotlin { | kotlin { | ||||||
|     android() |     androidTarget() | ||||||
|  |  | ||||||
|     listOf( |     listOf( | ||||||
|         iosX64(), |         iosX64(), | ||||||
| @@ -29,16 +31,18 @@ kotlin { | |||||||
|     sourceSets { |     sourceSets { | ||||||
|         val commonMain by getting { |         val commonMain by getting { | ||||||
|             dependencies { |             dependencies { | ||||||
|                 implementation("io.ktor:ktor-client-core:2.1.1") |                 implementation("io.ktor:ktor-client-core:$ktorVersion") | ||||||
|                 implementation("io.ktor:ktor-client-content-negotiation:2.1.1") |                 implementation("io.ktor:ktor-client-content-negotiation:$ktorVersion") | ||||||
|                 implementation("io.ktor:ktor-serialization-kotlinx-json:2.1.1") |                 implementation("io.ktor:ktor-serialization-kotlinx-json:$ktorVersion") | ||||||
|                 implementation("io.ktor:ktor-client-logging:2.1.1") |                 implementation("io.ktor:ktor-client-logging:$ktorVersion") | ||||||
|                 implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.0") |                 implementation("io.ktor:ktor-client-auth:$ktorVersion") | ||||||
|                 implementation("io.ktor:ktor-client-auth:2.1.1") |                 implementation("io.ktor:ktor-client-cio:$ktorVersion") | ||||||
|                 implementation("org.jsoup:jsoup:1.14.3") |                 implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.7.1") | ||||||
|  |  | ||||||
|  |                 implementation("org.jsoup:jsoup:1.15.4") | ||||||
|  |  | ||||||
|                 //Dependency Injection |                 //Dependency Injection | ||||||
|                 implementation("org.kodein.di:kodein-di:7.12.0") |                 implementation("org.kodein.di:kodein-di:7.14.0") | ||||||
|  |  | ||||||
|                 //Settings |                 //Settings | ||||||
|                 implementation("com.russhwolf:multiplatform-settings-no-arg:1.0.0-RC") |                 implementation("com.russhwolf:multiplatform-settings-no-arg:1.0.0-RC") | ||||||
| @@ -48,6 +52,9 @@ kotlin { | |||||||
|  |  | ||||||
|                 // Sql |                 // Sql | ||||||
|                 implementation(SqlDelight.runtime) |                 implementation(SqlDelight.runtime) | ||||||
|  |  | ||||||
|  |                 // Sql | ||||||
|  |                 implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.6.1") | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         val commonTest by getting { |         val commonTest by getting { | ||||||
| @@ -58,14 +65,15 @@ kotlin { | |||||||
|         } |         } | ||||||
|         val androidMain by getting { |         val androidMain by getting { | ||||||
|             dependencies { |             dependencies { | ||||||
|                 implementation("io.ktor:ktor-client-okhttp:2.1.1") |                 implementation("com.squareup.okhttp3:okhttp:4.11.0") | ||||||
|  |                 implementation("io.ktor:ktor-client-okhttp:2.2.4") | ||||||
|                 implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") |                 implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") | ||||||
|  |  | ||||||
|                 // Sql |                 // Sql | ||||||
|                 implementation(SqlDelight.android) |                 implementation(SqlDelight.android) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         val androidTest by getting { |         val androidUnitTest by getting { | ||||||
|             dependencies { |             dependencies { | ||||||
|                 implementation(kotlin("test-junit")) |                 implementation(kotlin("test-junit")) | ||||||
|                 implementation("junit:junit:4.13.2") |                 implementation("junit:junit:4.13.2") | ||||||
| @@ -75,10 +83,6 @@ kotlin { | |||||||
|         val iosArm64Main by getting |         val iosArm64Main by getting | ||||||
|         // val iosSimulatorArm64Main by getting |         // val iosSimulatorArm64Main by getting | ||||||
|         val iosMain by creating { |         val iosMain by creating { | ||||||
|             dependsOn(commonMain) |  | ||||||
|             iosX64Main.dependsOn(this) |  | ||||||
|             iosArm64Main.dependsOn(this) |  | ||||||
|             // iosSimulatorArm64Main.dependsOn(this) |  | ||||||
|  |  | ||||||
|             dependencies { |             dependencies { | ||||||
|                 implementation(SqlDelight.native) |                 implementation(SqlDelight.native) | ||||||
| @@ -89,24 +93,19 @@ kotlin { | |||||||
|         val iosArm64Test by getting |         val iosArm64Test by getting | ||||||
|         // val iosSimulatorArm64Test by getting |         // val iosSimulatorArm64Test by getting | ||||||
|         val iosTest by creating { |         val iosTest by creating { | ||||||
|             dependsOn(commonTest) |  | ||||||
|             iosX64Test.dependsOn(this) |  | ||||||
|             iosArm64Test.dependsOn(this) |  | ||||||
|             // iosSimulatorArm64Test.dependsOn(this) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| android { | android { | ||||||
|     compileSdk = 32 |     compileSdk = 34 | ||||||
|     sourceSets["main"].manifest.srcFile("src/androidMain/AndroidManifest.xml") |     sourceSets["main"].manifest.srcFile("src/androidMain/AndroidManifest.xml") | ||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         minSdk = 21 |         minSdk = 25 | ||||||
|         targetSdk = 32 |  | ||||||
|     } |     } | ||||||
|     compileOptions { |     compileOptions { | ||||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 |         sourceCompatibility = JavaVersion.VERSION_17 | ||||||
|         targetCompatibility = JavaVersion.VERSION_1_8 |         targetCompatibility = JavaVersion.VERSION_17 | ||||||
|     } |     } | ||||||
|     namespace = "bou.amine.apps.readerforselfossv2" |     namespace = "bou.amine.apps.readerforselfossv2" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,23 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.rest | ||||||
|  |  | ||||||
|  | import io.ktor.client.engine.cio.CIOEngineConfig | ||||||
|  | import java.security.cert.X509Certificate | ||||||
|  | import javax.net.ssl.X509TrustManager | ||||||
|  |  | ||||||
|  | class NaiveTrustManager : X509TrustManager { | ||||||
|  |     override fun checkClientTrusted( | ||||||
|  |         chain: Array<out X509Certificate>?, | ||||||
|  |         authType: String?, | ||||||
|  |     ) {} | ||||||
|  |  | ||||||
|  |     override fun checkServerTrusted( | ||||||
|  |         chain: Array<out X509Certificate>?, | ||||||
|  |         authType: String?, | ||||||
|  |     ) {} | ||||||
|  |  | ||||||
|  |     override fun getAcceptedIssuers(): Array<out X509Certificate> = arrayOf() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | actual fun setupInsecureHTTPEngine(config: CIOEngineConfig) { | ||||||
|  |     config.https.trustManager = NaiveTrustManager() | ||||||
|  | } | ||||||
| @@ -1,32 +1,20 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.utils | package bou.amine.apps.readerforselfossv2.utils | ||||||
|  |  | ||||||
| import android.text.format.DateUtils | import android.text.format.DateUtils | ||||||
|  | import io.github.aakira.napier.Napier | ||||||
| import kotlinx.datetime.* | import kotlinx.datetime.* | ||||||
|  |  | ||||||
|  |  | ||||||
| actual class DateUtils { | actual class DateUtils { | ||||||
|     actual companion object { |     actual companion object { | ||||||
|         actual fun parseDate(dateString: String): Long { |  | ||||||
|             return try { |  | ||||||
|                 Instant.parse(dateString).toEpochMilliseconds() |  | ||||||
|             } catch (e: Exception) { |  | ||||||
|                 var str = dateString.replace(" ", "T") |  | ||||||
|                 if (str.matches("\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\+\\d{2}".toRegex())) { |  | ||||||
|                     str = str.split("+")[0] |  | ||||||
|                 } |  | ||||||
|                 LocalDateTime.parse(str).toInstant(TimeZone.currentSystemDefault()).toEpochMilliseconds() |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         actual fun parseRelativeDate(dateString: String): String { |         actual fun parseRelativeDate(dateString: String): String { | ||||||
|  |             val date = dateString.toParsedDate() | ||||||
|  |  | ||||||
|             val date = parseDate(dateString) |             return " " + | ||||||
|  |                 DateUtils.getRelativeTimeSpanString( | ||||||
|             return " " + DateUtils.getRelativeTimeSpanString( |  | ||||||
|                     date, |                     date, | ||||||
|                     Clock.System.now().toEpochMilliseconds(), |                     Clock.System.now().toEpochMilliseconds(), | ||||||
|                     DateUtils.MINUTE_IN_MILLIS, |                     DateUtils.MINUTE_IN_MILLIS, | ||||||
|                 DateUtils.FORMAT_ABBREV_RELATIVE |                     DateUtils.FORMAT_ABBREV_RELATIVE, | ||||||
|                 ) |                 ) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -21,13 +21,13 @@ actual fun SelfossModel.Item.getThumbnail(baseUrl: String): String { | |||||||
| actual fun SelfossModel.Item.getImages(): ArrayList<String> { | actual fun SelfossModel.Item.getImages(): ArrayList<String> { | ||||||
|     val allImages = ArrayList<String>() |     val allImages = ArrayList<String>() | ||||||
|  |  | ||||||
|     for ( image in Jsoup.parse(content).getElementsByTag("img")) { |     for (image in Jsoup.parse(content).getElementsByTag("img")) { | ||||||
|         val url = image.attr("src") |         val url = image.attr("src") | ||||||
|         if (url.lowercase(Locale.US).contains(".jpg") || |         if (url.lowercase(Locale.US).contains(".jpg") || | ||||||
|             url.lowercase(Locale.US).contains(".jpeg") || |             url.lowercase(Locale.US).contains(".jpeg") || | ||||||
|             url.lowercase(Locale.US).contains(".png") || |             url.lowercase(Locale.US).contains(".png") || | ||||||
|             url.lowercase(Locale.US).contains(".webp")) |             url.lowercase(Locale.US).contains(".webp") | ||||||
|         { |         ) { | ||||||
|             allImages.add(url) |             allImages.add(url) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -38,7 +38,11 @@ actual fun SelfossModel.Source.getIcon(baseUrl: String): String { | |||||||
|     return constructUrl(baseUrl, "favicons", icon) |     return constructUrl(baseUrl, "favicons", icon) | ||||||
| } | } | ||||||
|  |  | ||||||
| actual fun constructUrl(baseUrl: String, path: String, file: String?): String { | actual fun constructUrl( | ||||||
|  |     baseUrl: String, | ||||||
|  |     path: String, | ||||||
|  |     file: String?, | ||||||
|  | ): String { | ||||||
|     return if (file == null || file == "null" || file.isEmpty()) { |     return if (file == null || file == "null" || file.isEmpty()) { | ||||||
|         "" |         "" | ||||||
|     } else { |     } else { | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ package bou.amine.apps.readerforselfossv2.DI | |||||||
|  |  | ||||||
| import bou.amine.apps.readerforselfossv2.rest.MercuryApi | import bou.amine.apps.readerforselfossv2.rest.MercuryApi | ||||||
| import bou.amine.apps.readerforselfossv2.rest.SelfossApi | import bou.amine.apps.readerforselfossv2.rest.SelfossApi | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService |  | ||||||
| import org.kodein.di.DI | import org.kodein.di.DI | ||||||
| import org.kodein.di.bind | import org.kodein.di.bind | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|   | |||||||
| @@ -3,12 +3,14 @@ package bou.amine.apps.readerforselfossv2.model | |||||||
| import kotlinx.serialization.Serializable | import kotlinx.serialization.Serializable | ||||||
|  |  | ||||||
| class MercuryModel { | class MercuryModel { | ||||||
|  |  | ||||||
|     @Serializable |     @Serializable | ||||||
|     class ParsedContent( |     class ParsedContent( | ||||||
|         val title: String?, |         val title: String? = null, | ||||||
|         val content: String?, |         val content: String? = null, | ||||||
|         val lead_image_url: String?, // NOSONAR |         val lead_image_url: String? = null, // NOSONAR | ||||||
|         val url: String |         val url: String? = null, | ||||||
|  |         val error: Boolean? = null, | ||||||
|  |         val message: String? = null, | ||||||
|  |         val failed: Boolean? = null, | ||||||
|     ) |     ) | ||||||
| } | } | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user