Compare commits
	
		
			422 Commits
		
	
	
		
			v1.5.4.18
			...
			41c14362a8
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 41c14362a8 | ||
| 6fa8c901fc | |||
|  | db124ab9de | ||
|  | 953940690d | ||
| 918661be2d | |||
|  | 7b8a5c9a56 | ||
| 2d5ab7bf0c | |||
| 9ba281befb | |||
| 00c8eed034 | |||
| a1e4f89cd1 | |||
| 36a43b3861 | |||
|  | aa6d470f40 | ||
|  | 0046a8a477 | ||
|  | 43ff9d186a | ||
|  | 73dae304be | ||
|  | 66103a451b | ||
|  | 600c62316d | ||
|  | d370ddc4d1 | ||
|  | 4049f6a5c7 | ||
|  | 3e96ac207e | ||
|  | 84f1ab12cf | ||
|  | f48f6ed788 | ||
|  | e517803bd8 | ||
|  | 3eaf390790 | ||
|  | 6de54d63e6 | ||
|  | dd7a2f476b | ||
|  | 1485cc05f4 | ||
|  | d1dad3e61a | ||
|  | e5024b0420 | ||
|  | 9b01692c55 | ||
|  | 33aa587d36 | ||
|  | 12e0766803 | ||
|  | a8721ad7a4 | ||
|  | bc5e882894 | ||
|  | e3460322b1 | ||
|  | 7e3288a076 | ||
|  | ddc754ec25 | ||
|  | 134a0766d6 | ||
|  | 69da932ab5 | ||
|  | 592fb6328a | ||
|  | a0aead6491 | ||
|  | 722b6cc06d | ||
|  | 6d7c4b40f6 | ||
|  | f538ed39fc | ||
|  | 65821492ad | ||
|  | 6ede718a9f | ||
|  | f1757937a4 | ||
| 2bd2e0a953 | |||
|  | b5aef28af0 | ||
|  | 45747a1506 | ||
|  | c6e2e08bcb | ||
|  | 25bf18661e | ||
|  | 6b088dcd24 | ||
|  | d2b18e1880 | ||
|  | eec7c94e98 | ||
|  | d1f8fcacc0 | ||
|  | 07e4a33cbd | ||
|  | f6317f566e | ||
|  | 9f51e4e6a5 | ||
|  | 750604a31f | ||
|  | 392eee0ad4 | ||
|  | 37e7b987ee | ||
|  | 9eac51e729 | ||
|  | fa9cce6783 | ||
|  | f0d4b63a97 | ||
|  | 83eeb11388 | ||
|  | 01f746f33d | ||
|  | 200851894b | ||
|  | 862e5cf4ab | ||
|  | 0b07f2a407 | ||
|  | 9ba6feef0b | ||
|  | 63a0638522 | ||
|  | f9a4e6e363 | ||
|  | 6b40fd4bdc | ||
|  | 04c7776466 | ||
|  | 92c335b4e1 | ||
|  | 17251e576b | ||
|  | 62ea782429 | ||
|  | f99474e3c1 | ||
|  | 57ac8f428f | ||
|  | 9cc1adbf15 | ||
|  | 1d9a440ae7 | ||
|  | 511553806c | ||
|  | 87e7d7c4fe | ||
|  | ec87089310 | ||
|  | d8478ebb01 | ||
|  | 600adc81b5 | ||
|  | ddac2870af | ||
| 8d9c8c1394 | |||
|  | b59c3bcb23 | ||
| 7f554adba5 | |||
|  | 21ce061282 | ||
|  | bdb71e9b14 | ||
|  | df22e7de15 | ||
|  | 6b3550396b | ||
|  | c70f1e31a6 | ||
|  | 695670e944 | ||
|  | 1028826788 | ||
|  | 82a8977c96 | ||
|  | 07d9ce1054 | ||
|  | 7da7d49277 | ||
|  | 9b45365441 | ||
|  | 91a7464bce | ||
|  | 51add226eb | ||
|  | 332e9f5108 | ||
|  | 0b91087c07 | ||
|  | ebbb1ba0f8 | ||
|  | e9143ae852 | ||
|  | 42e8ecee78 | ||
|  | 4efd76fcbc | ||
|  | fb1614070e | ||
|  | c473dd7227 | ||
|  | 76bddb195d | ||
|  | 1e02ad2041 | ||
|  | f6ab909f8b | ||
|  | 7e520e9bed | ||
|  | 32e2d05014 | ||
|  | 40d9c97f73 | ||
|  | 1aa68d3449 | ||
|  | aeeac8cccd | ||
|  | 7292edf997 | ||
|  | f49256c72f | ||
|  | d02b28b81f | ||
|  | 08117043dd | ||
|  | 63496c993e | ||
|  | 00ef542e49 | ||
|  | a78c6e6b33 | ||
|  | 363eaf9bf9 | ||
|  | fec6683701 | ||
|  | 1549edb647 | ||
|  | 3de48ba162 | ||
|  | a2a3d6f1a7 | ||
|  | ccab2c7648 | ||
|  | 880dd1db5c | ||
|  | ed18fea356 | ||
|  | 9816b20bf6 | ||
|  | 0bb2195bff | ||
|  | ab2d0c4036 | ||
|  | 99fc417109 | ||
|  | dc304ef8c1 | ||
|  | c5511880bc | ||
|  | 5fe76d735e | ||
|  | 3064b3b835 | ||
|  | 70dc8af3ce | ||
|  | 53c8c241da | ||
|  | bdc4f5680b | ||
|  | ed290573b2 | ||
|  | 1616a97a8a | ||
|  | d090183007 | ||
|  | de337fd260 | ||
|  | 12dc206323 | ||
|  | d47c508dee | ||
|  | ed75f55437 | ||
|  | 5ad3ad4a57 | ||
|  | aeac1bd1d4 | ||
|  | 4d18085072 | ||
|  | 0c9f8214ca | ||
|  | a7ce7ce02e | ||
|  | 820986c7f0 | ||
|  | 8079cae745 | ||
|  | 6f067bd258 | ||
|  | b6ade0f212 | ||
|  | 27dadc1be3 | ||
|  | 95e4162b4c | ||
|  | f75557585e | ||
|  | 1b4c26919b | ||
|  | ad085bf129 | ||
|  | 8fcd551105 | ||
|  | a0954700e2 | ||
|  | 9705560442 | ||
|  | 1f47a13ce5 | ||
|  | 6f0ff2c975 | ||
|  | 76e5477986 | ||
|  | 7f308d5be3 | ||
|  | 54a43c83e8 | ||
|  | 8fe7266c84 | ||
|  | d7a46b27b7 | ||
|  | 2257d09fdd | ||
|  | 047c5481c4 | ||
|  | 8a6719f934 | ||
|  | 51a692f3be | ||
|  | b333f93171 | ||
|  | 89d34a1a71 | ||
|  | 8788e920ce | ||
|  | d306fb53d3 | ||
|  | 374537b5c7 | ||
|  | 598149d4cd | ||
|  | 50bcf18096 | ||
|  | a089ced03f | ||
|  | 1f18dddf8b | ||
|  | f5934e240e | ||
|  | 6b8da2eacf | ||
|  | f4757a67b7 | ||
|  | 6edeb9d840 | ||
|  | 43ce0fd7bc | ||
|  | 5599f5a8fc | ||
|  | 6fd45ceb4f | ||
|  | 05ad8aac29 | ||
|  | fa4f2476b7 | ||
|  | 00818a94e9 | ||
|  | 5d5250e44a | ||
|  | 3052b33132 | ||
|  | 50de6f8b5b | ||
|  | f88a2f415f | ||
|  | 96f9813e01 | ||
|  | fee739cb17 | ||
|  | b1814c63b9 | ||
|  | c1d45678f8 | ||
|  | 3d34e59a94 | ||
|  | f1133bea8b | ||
|  | ec64c88ff1 | ||
|  | be66dbba6c | ||
|  | 8926cdbbf5 | ||
|  | a956870dec | ||
|  | 8ed7951c9b | ||
|  | 5569a47674 | ||
|  | 0dc6981913 | ||
|  | 4984f2f7ad | ||
|  | 3b6891c84a | ||
|  | 4901e7174c | ||
|  | 8d70e68fe2 | ||
|  | d3e1527b70 | ||
|  | 0c201301f2 | ||
|  | 6090590f24 | ||
|  | 06b88c783d | ||
|  | bb75ebf635 | ||
|  | 7d7d0014be | ||
|  | b3f8d44794 | ||
|  | 29d1e38340 | ||
|  | 2be872e61e | ||
|  | 377c5518f7 | ||
|  | 21be7357b5 | ||
|  | d47ba2c820 | ||
|  | a64b14614a | ||
|  | 6a88192e77 | ||
|  | aa7c630818 | ||
|  | 7fb54f14c7 | ||
|  | 3d709c02b7 | ||
|  | 339d384561 | ||
|  | 50338d51af | ||
|  | 92dbabf899 | ||
|  | 0043021390 | ||
|  | 70ba9b20da | ||
|  | 7fda0a04a1 | ||
|  | 3db3157dc9 | ||
|  | 2089fe60ca | ||
|  | 9606d36670 | ||
|  | 869cf64c54 | ||
|  | f57ec1f6c0 | ||
|  | 361eea9a06 | ||
|  | 838b4056ac | ||
|  | 0c0a98510b | ||
|  | be642ed06f | ||
|  | fd77f38e95 | ||
|  | c9baab7267 | ||
|  | 86985cfd5b | ||
|  | 1327a4e069 | ||
|  | c46acbc579 | ||
|  | 4c6a403fae | ||
|  | 78920022bd | ||
|  | 7b16c41e82 | ||
|  | 3389f8bd09 | ||
|  | 8dc25c527d | ||
|  | 46d6bd57c1 | ||
|  | db014fe13d | ||
|  | 6c293f4cac | ||
|  | 91e5d3736f | ||
|  | e11dee220f | ||
|  | fcebf916d2 | ||
|  | 73cc1a7297 | ||
|  | 798f112498 | ||
|  | 38b5e7dc65 | ||
|  | 2799a48f2b | ||
|  | ad5edae6cd | ||
|  | 9cb02f0272 | ||
|  | 6d24fd9336 | ||
|  | a3a7b78c96 | ||
|  | e995286068 | ||
|  | 65fb6d9b7e | ||
|  | eb02d1efad | ||
|  | f8d3e1eefb | ||
|  | 218b8fa843 | ||
|  | 9f94af6239 | ||
|  | d3584ac40e | ||
|  | 90bdb289d0 | ||
|  | 78a08750a2 | ||
|  | baba851e97 | ||
|  | 2a03783623 | ||
|  | 9f2a4438b1 | ||
|  | 5ee5287ffa | ||
|  | 29547c2c94 | ||
|  | 4846c870fa | ||
|  | c17980a032 | ||
|  | a929e419d9 | ||
|  | 487d484bae | ||
|  | 0ca4c04c61 | ||
|  | c857cf2d67 | ||
|  | acb502028b | ||
|  | 533636f3a1 | ||
|  | eb5672901b | ||
|  | 53a8716b51 | ||
|  | 3aaff612af | ||
|  | fdcd8c6c6a | ||
|  | bafd478604 | ||
|  | 987513a88b | ||
|  | a450ab2a3b | ||
|  | db89fe5aad | ||
|  | 67a30b92f6 | ||
|  | c397de8c3e | ||
|  | b4db532c45 | ||
|  | ebecc9c80a | ||
|  | 4f8556fca8 | ||
|  | 68892fb41b | ||
|  | 6d6f6c72ac | ||
|  | df5556b945 | ||
|  | d6c74049c3 | ||
|  | 18946464a2 | ||
|  | edb5eabee7 | ||
|  | 99a305f3e2 | ||
|  | 68dc5a6acf | ||
|  | 6816461502 | ||
|  | 15b93bbd9e | ||
|  | cd61e140f6 | ||
|  | 4d861a84e6 | ||
|  | f24de68618 | ||
|  | 3bcffff444 | ||
|  | 75e9031fa5 | ||
|  | 3b77e24399 | ||
|  | 0a738e895f | ||
|  | 242e5ba035 | ||
|  | c94612106c | ||
|  | 320924b4ed | ||
|  | 403ecc4521 | ||
|  | 6a50b37364 | ||
|  | d9d341ac5d | ||
|  | e9805b731e | ||
|  | c6d4337cd1 | ||
|  | 173f4b2ff7 | ||
|  | 3b9436264c | ||
|  | 35fe87d79d | ||
|  | f1bb7ba9ad | ||
|  | 279f229166 | ||
|  | be1794e27b | ||
|  | 4d4a2039c8 | ||
|  | 3013ae4f35 | ||
|  | bb3f7d3786 | ||
|  | f7cc305e44 | ||
|  | da17f89148 | ||
|  | ec71ab3c6f | ||
|  | 0d007f1492 | ||
|  | 96f8663b8f | ||
|  | 1a4bc1b301 | ||
|  | b51ae58a97 | ||
|  | b126fc32da | ||
|  | b8d234c415 | ||
|  | 2c8902d404 | ||
|  | 80ad65b196 | ||
|  | 744d9ba72b | ||
|  | 0c1d708588 | ||
|  | 95e79e7c5d | ||
|  | 3ce3260d20 | ||
|  | 641f4f34d3 | ||
|  | 99620cb1c5 | ||
|  | 8f5f33f5d2 | ||
|  | 78e9230b82 | ||
|  | 78aa44c007 | ||
|  | 53fd944f00 | ||
|  | 9e6cb4ee3d | ||
|  | 87ad6f2826 | ||
|  | 9050f5a56f | ||
|  | 3437004082 | ||
|  | dcf620af87 | ||
|  | 128085a02e | ||
|  | 302040ec25 | ||
|  | e177c22032 | ||
|  | a11007113a | ||
|  | 5e7897bcf4 | ||
|  | 9559af3637 | ||
|  | 4c499abcdb | ||
|  | 0055a503b3 | ||
|  | 3a189ee4b6 | ||
|  | e25dc49271 | ||
|  | 4208a80db8 | ||
|  | ddb75e0d93 | ||
|  | 8b37e992a2 | ||
|  | bac59036cd | ||
|  | 6c89a3b77c | ||
|  | dc2ef39fc6 | ||
|  | a4806da2c5 | ||
|  | ee30edb214 | ||
|  | e4ed663fb3 | ||
|  | 01629309b0 | ||
|  | 059c2991fb | ||
|  | 686ec5dd90 | ||
|  | eab9df8ed9 | ||
|  | 0107c3d7e2 | ||
|  | 2def2f2e2c | ||
|  | 44c79892a0 | ||
|  | bc96b314c2 | ||
|  | 8dcf749b4e | ||
|  | 6a56ec6442 | ||
|  | 30e46d7eae | ||
|  | 9458b1834b | ||
|  | 297f797b97 | ||
|  | c70e80758c | ||
|  | 3bf1d7c4f9 | ||
|  | 173247041a | ||
|  | 3a28772096 | ||
|  | bd08b8aba3 | ||
|  | 2ceb0f988b | ||
|  | 4ef3b155b8 | ||
|  | 350e24cded | ||
|  | 1bf8a578bc | ||
|  | 4818a101cc | ||
|  | baebf938ef | ||
|  | fea57c7b1e | ||
|  | 113dfa68be | ||
|  | 60c6514fa1 | ||
|  | 114485afc3 | ||
|  | d6a51381b9 | ||
|  | 620f13fd7c | ||
|  | 6577b2c3d7 | 
							
								
								
									
										23
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -8,7 +8,7 @@ Please read the guidelines before contributing, and follow them (or try to) when | |||||||
|  |  | ||||||
| ### What you can do to help. | ### What you can do to help. | ||||||
|  |  | ||||||
| There are many ways to contribute to this project, you could 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://github.com/aminecmi/ReaderforSelfoss/issues?q=is%3Aissue+is%3Aopen+label%3A%22Up+For+Grabs%22) or [develop new things](https://github.com/aminecmi/ReaderforSelfoss/issues) | You can fork the repository, and [help me solve some issues](https://github.com/aminecmi/ReaderforSelfoss/issues?q=is%3Aissue+is%3Aopen+label%3A%22Up+For+Grabs%22) or [develop new things](https://github.com/aminecmi/ReaderforSelfoss/issues) | ||||||
|  |  | ||||||
| @@ -28,6 +28,7 @@ Always check if the web version of your instance is working. | |||||||
|  |  | ||||||
| ### Pull requests | ### Pull requests | ||||||
|  |  | ||||||
|  | * Don't create a PR for translations. See [here](https://github.com/aminecmi/ReaderforSelfoss/pull/170#issuecomment-355715654) for an explanation why. | ||||||
| * Please ask before starting to work on an issue. I may be working on it, or someone else could be doing so. | * Please ask before starting to work on an issue. I may be working on it, or someone else could be doing so. | ||||||
| * Each pull request should implement **ONE** feature or bugfix. Keep in mind that you can submit as many PR as you want. | * Each pull request should implement **ONE** feature or bugfix. Keep in mind that you can submit as many PR as you want. | ||||||
| * Your code must be simple and clear enough to avoid using comments to explain what it does. | * Your code must be simple and clear enough to avoid using comments to explain what it does. | ||||||
| @@ -40,21 +41,20 @@ Always check if the web version of your instance is working. | |||||||
| * Remember that PR review can take time. | * Remember that PR review can take time. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Install Selfoss (if you don't have an instance) | ||||||
|  |  | ||||||
|  | I won't provide any selfoss instance url. If you want to help, but to not have one, you'll have to install one, and use it. | ||||||
|  |  | ||||||
|  | All the details to need are [here](https://selfoss.aditu.de/). | ||||||
|  |  | ||||||
| # Build the project | # Build the project | ||||||
|  |  | ||||||
| You can directly import this project into IntellIJ/Android Studio. | You can directly import this project into IntellIJ/Android Studio. | ||||||
|  |  | ||||||
| You'll have to: | You'll have to: | ||||||
|  |  | ||||||
| - Configure fabric and add your `apiKey` and `apiSecret` in the `fabric.properties` file. |  | ||||||
| - Create a firebase project and add the `google-services.json` to the `app/` folder. |  | ||||||
| - Define some parameters either in `~/.gradle/gradle.properties` or as gradle parameters (see the examples) | - Define some parameters either in `~/.gradle/gradle.properties` or as gradle parameters (see the examples) | ||||||
|  |  | ||||||
|     - mercuryApiKey: A [Mercury](https://mercury.postlight.com/web-parser/) web parser api key for the internal browser |  | ||||||
|     - feedbackEmail: An email to receive users  feedback. |  | ||||||
|     - sourceUrl: an url to the source code, used in the settings. **It can be empty.** |  | ||||||
|     - trackerUrl: an url to the tracker, used in the settings. **It can be empty.** |  | ||||||
|     - githubToken: a github token used to report issues from within the app. [Details  here](https://github.com/heinrichreimer/android-issue-reporter#how-to-create-a-bot-key). **It can be empty.**  |  | ||||||
|     - appLoginUrl, appLoginUsername and appLoginPassword: url, username and password of a selfoss instance. **These are only used for tests. They can be empty if you don't test API calls.** |     - appLoginUrl, appLoginUsername and appLoginPassword: url, username and password of a selfoss instance. **These are only used for tests. They can be empty if you don't test API calls.** | ||||||
|  |  | ||||||
| ### Examples: | ### Examples: | ||||||
| @@ -64,15 +64,10 @@ You'll have to: | |||||||
| appLoginUrl="URL" # It can be empty. | appLoginUrl="URL" # It can be empty. | ||||||
| appLoginUsername="LOGIN" # It can be empty. | appLoginUsername="LOGIN" # It can be empty. | ||||||
| appLoginPassword="PASS" # It can be empty. | appLoginPassword="PASS" # It can be empty. | ||||||
| mercuryApiKey="LONGAPIKEY" |  | ||||||
| feedbackEmail="EMAIL" |  | ||||||
| sourceUrl="URLSOURCE" # It can be empty. |  | ||||||
| trackerUrl="URLTRACKER" # It can be empty. |  | ||||||
| githubToken="GITHUBTOKEN" # It can be empty or use https://github.com/heinrichreimer/android-issue-reporter#how-to-create-a-bot-key to generate one |  | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| #### As gradle parameters | #### As gradle parameters | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ./gradlew .... -P appLoginUrl="URL" -P appLoginUsername="LOGIN" -P appLoginPassword="PASS" -P mercuryApiKey="LONGAPIKEY" -P feedbackEmail="EMAIL" -P sourceUrl="URLSOURCE" -P trackerUrl="URLTRACKER" -P githubToken="GITHUBTOKEN" | ./gradlew .... -P appLoginUrl="URL" -P appLoginUsername="LOGIN" -P appLoginPassword="PASS" | ||||||
| ``` | ``` | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -5,6 +5,7 @@ | |||||||
| - [ ] I have updated the documentation accordingly. | - [ ] I have updated the documentation accordingly. | ||||||
| - [ ] I have added tests to cover my changes. | - [ ] I have added tests to cover my changes. | ||||||
| - [ ] All new and existing tests passed. | - [ ] All new and existing tests passed. | ||||||
|  | - [ ] This is **NOT** translation related. (See [here](https://github.com/aminecmi/ReaderforSelfoss/pull/170#issuecomment-355715654)) | ||||||
|  |  | ||||||
| This closes issue #XXX | This closes issue #XXX | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										122
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						| @@ -1,3 +1,125 @@ | |||||||
|  | **1.7.x** | ||||||
|  |  | ||||||
|  | - Hiding tags with 0 articles | ||||||
|  |  | ||||||
|  | - Fixed issue with basic auth and images loading | ||||||
|  |  | ||||||
|  | - Added the ability to justify or left align the reader text | ||||||
|  |  | ||||||
|  | - Fixed #251 | ||||||
|  |  | ||||||
|  | - Added experimental issue to set a default timeout. Should work for #238. | ||||||
|  |  | ||||||
|  | - Closing #220. | ||||||
|  |  | ||||||
|  | - Start of #238. "Add a quick shortcut to open the app on offline mode ?" | ||||||
|  |  | ||||||
|  | - Closes #216. Issue with selfoss version 2.19. | ||||||
|  |  | ||||||
|  | - Closes #179. Sync of read/unread/star/unstar items on background task or on app reload with network available. | ||||||
|  |  | ||||||
|  | - Closes #33. Background sync with settings. | ||||||
|  |  | ||||||
|  | - Closing #1. Initial article caching. | ||||||
|  |  | ||||||
|  | - Closing #228 by removing the list action bar. Action buttons are exclusively on the card view from now on. | ||||||
|  |  | ||||||
|  | - Closing #38. Only doing api calls on network available. | ||||||
|  |  | ||||||
|  | - Closing #298 and #287. Issues with Listview rendering | ||||||
|  |  | ||||||
|  | - Closing #290. Fixing back button issue in Settings | ||||||
|  |  | ||||||
|  | - Closing #300. Fixing issues when displaying some special characters. | ||||||
|  |  | ||||||
|  | - Closing #310. Some feeds don't have icons nor thumbnails. | ||||||
|  |  | ||||||
|  | - Closing #178. Expending images on tap. | ||||||
|  |  | ||||||
|  | **1.6.x** | ||||||
|  |  | ||||||
|  | - Handling hidden tags. | ||||||
|  |  | ||||||
|  | - Fixed pre-lolipop issue with automatic theme changes. | ||||||
|  |  | ||||||
|  | - Removed all Build config things. | ||||||
|  |  | ||||||
|  | - Removed firebase and fabric. | ||||||
|  |  | ||||||
|  | - Added Acra for optional crash reporting and error logging. | ||||||
|  |  | ||||||
|  | - Dynamic themes ! | ||||||
|  |  | ||||||
|  | - Strings cleaning. | ||||||
|  |  | ||||||
|  | - Versions updates. | ||||||
|  |  | ||||||
|  | - Fixes #215, #208. | ||||||
|  |  | ||||||
|  | **1.5.7.x** | ||||||
|  |  | ||||||
|  | - Added confirmation to the mark as read and update menues. | ||||||
|  |  | ||||||
|  | - Add to favorites from article viewer. | ||||||
|  |  | ||||||
|  | - Added an option to use a webview in the article viewer (see #149) | ||||||
|  |  | ||||||
|  | - Fixes (#151 #152 #155 #157 #160 #174) and more. | ||||||
|  |  | ||||||
|  | - New year fixes !!! | ||||||
|  |  | ||||||
|  | - Changed page indicator position as it was overlaping content. | ||||||
|  |  | ||||||
|  | - Now using slack instead of gitter. | ||||||
|  |  | ||||||
|  | - Moved completely to a webview to fix #161. | ||||||
|  |  | ||||||
|  | - Fixed typos in French ( Thanks @aancel ) | ||||||
|  |  | ||||||
|  | - Updated the Contribution guide about translations. | ||||||
|  |  | ||||||
|  | - Better handling for articles update. (See #169) | ||||||
|  |  | ||||||
|  | - Ability to change the article viewer content font size (see #153) | ||||||
|  |  | ||||||
|  | - Versions updates * 2. | ||||||
|  |  | ||||||
|  | - Added padding to the recyclerview. | ||||||
|  |  | ||||||
|  | **1.5.5.x (didn't last long) AND 1.5.6.x** | ||||||
|  |  | ||||||
|  | - Toolbar in reader activity. | ||||||
|  |  | ||||||
|  | - Marking items as read on scroll (with settings to enable/disable). | ||||||
|  |  | ||||||
|  | - Swapped the title and subtitle in the article viewer. | ||||||
|  |  | ||||||
|  | - Added an animation to the viewpager. | ||||||
|  |  | ||||||
|  | - Completed Dutch, Indonesian and Portuguese translations ! | ||||||
|  |  | ||||||
|  | - Fixed #142, #144, #147. | ||||||
|  |  | ||||||
|  | - Changed versions handling. | ||||||
|  |  | ||||||
|  | - Removed indonesian english as it was causing issues with the english version of the app. | ||||||
|  |  | ||||||
|  | **1.5.4.22** | ||||||
|  |  | ||||||
|  | - You can now scroll through the loaded articles ! | ||||||
|  |  | ||||||
|  | **1.5.4.21** | ||||||
|  |  | ||||||
|  | - Spanish translation and some Indonesian ! | ||||||
|  |  | ||||||
|  | **1.5.4.20** | ||||||
|  |  | ||||||
|  | - Turkish translation ! | ||||||
|  |  | ||||||
|  | **1.5.4.19** | ||||||
|  |  | ||||||
|  | - Fixed an issue with crowdin configuration (and its translations) | ||||||
|  |  | ||||||
| **1.5.4.18** | **1.5.4.18** | ||||||
|  |  | ||||||
| - Typo fix. | - Typo fix. | ||||||
|   | |||||||
							
								
								
									
										34
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,21 +1,34 @@ | |||||||
| # ReaderForSelfoss | # ReaderForSelfoss **(Only available from F-Droid)** | ||||||
|  |  | ||||||
| [](https://crowdin.com/project/readerforselfoss) [](https://gitter.im/amine-bou/ReaderForSelfoss) | [](https://crowdin.com/project/readerforselfoss) | ||||||
|  |  | ||||||
| [](http://jenkins.amine-bou.fr/job/ReaderForSelfoss/) |  | ||||||
|  |  | ||||||
| [](https://www.codetriage.com/aminecmi/readerforselfoss) |  | ||||||
|  |  | ||||||
| This is the repo of [Reader For Selfoss](https://play.google.com/store/apps/details?id=apps.amine.bou.readerforselfoss&hl=en). |  | ||||||
|  |  | ||||||
| It's an RSS Reader for Android, that **only** works with [Selfoss](https://selfoss.aditu.de/) | It's an RSS Reader for Android, that **only** works with [Selfoss](https://selfoss.aditu.de/) | ||||||
|  |  | ||||||
| The last APK built from source is available [here](https://jenkins.amine-bou.fr/job/ReaderForSelfoss/lastSuccessfulBuild/artifact/SignApksBuilder-out/selfoss-key/selfoss/app-githubConfig-release-unsigned.apk/app-githubConfig-release.apk). | **The project is not dead at all.**  | ||||||
|  |  | ||||||
|  | I still want to work on it, but for the last few months, I didn't have that much time to do so.  | ||||||
|  |  | ||||||
|  | If you are a developer, don't hesitate to help with PRs. | ||||||
|  |  | ||||||
|  | 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/apps.amine.bou.readerforselfoss"><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 ? | ||||||
|  |  | ||||||
|  | <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> | ||||||
|  |  | ||||||
| ## Want to help ? | ## Want to help ? | ||||||
|  |  | ||||||
| Check the [Contribution guide](https://github.com/aminecmi/ReaderforSelfoss/blob/master/.github/CONTRIBUTING.md) | 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://github.com/aminecmi/ReaderforSelfoss/blob/master/.github/CONTRIBUTING.md). | ||||||
|  |  | ||||||
|  | 3. Build the project by following [these steps](https://github.com/aminecmi/ReaderforSelfoss/blob/master/.github/CONTRIBUTING.md#build-the-project) (you should have read them after the contribution guide) | ||||||
|  |  | ||||||
| ## Useful links | ## Useful links | ||||||
|  |  | ||||||
| @@ -23,4 +36,3 @@ Check the [Contribution guide](https://github.com/aminecmi/ReaderforSelfoss/blob | |||||||
| - [See what I'm doing](https://github.com/aminecmi/ReaderforSelfoss/projects/1) | - [See what I'm doing](https://github.com/aminecmi/ReaderforSelfoss/projects/1) | ||||||
| - [Create an issue, or request a new feature](https://github.com/aminecmi/ReaderforSelfoss/issues) | - [Create an issue, or request a new feature](https://github.com/aminecmi/ReaderforSelfoss/issues) | ||||||
| - [Help translation the app](https://crowdin.com/project/readerforselfoss) | - [Help translation the app](https://crowdin.com/project/readerforselfoss) | ||||||
| - [Ask for help](https://gitter.im/amine-bou/ReaderForSelfoss) |  | ||||||
|   | |||||||
							
								
								
									
										215
									
								
								app/build.gradle
									
									
									
									
									
								
							
							
						
						| @@ -1,51 +1,48 @@ | |||||||
| buildscript { | buildscript { | ||||||
|     repositories { |  | ||||||
|         maven { url 'https://maven.fabric.io/public' } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     dependencies { |  | ||||||
|         classpath 'io.fabric.tools:gradle:1.+' |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|  |  | ||||||
| def gitVersion() { | def gitVersion() { | ||||||
|     def process = "git describe --abbrev=0 --tags".execute() |     def process | ||||||
|     return process.text.substring(1).replaceAll("\\.", "") |     def maybeTagOfCurrentCommit = 'git describe --contains HEAD'.execute() | ||||||
|  |     if (maybeTagOfCurrentCommit.text.isEmpty()) { | ||||||
|  |         println "No tag on current commit. Will take the latest one." | ||||||
|  |         process = "git for-each-ref refs/tags --sort=-authordate --format='%(refname:short)' --count=1".execute() | ||||||
|  |     } else { | ||||||
|  |         println "Tag found on current commit" | ||||||
|  |         process = 'git describe --contains HEAD'.execute() | ||||||
|  |     } | ||||||
|  |     return process.text.replaceAll("'", "").substring(1).replaceAll("\\.", "").trim() | ||||||
| } | } | ||||||
|  |  | ||||||
| def versionCodeFromGit() { | def versionCodeFromGit() { | ||||||
|     println "version code " + gitVersion().toInteger() |     println "version code " + gitVersion() | ||||||
|     return gitVersion().toInteger() |     return gitVersion().toInteger() | ||||||
| } | } | ||||||
|  |  | ||||||
| def versionNameFromGit() { | def versionNameFromGit() { | ||||||
|     println "version code " + gitVersion().trim() |     println "version name " + gitVersion() | ||||||
|     return gitVersion().trim() |     return gitVersion() | ||||||
| } | } | ||||||
|  |  | ||||||
| apply plugin: 'org.sonarqube' |  | ||||||
|  |  | ||||||
| apply plugin: 'com.android.application' | apply plugin: 'com.android.application' | ||||||
|  |  | ||||||
| apply plugin: 'io.fabric' |  | ||||||
|  |  | ||||||
| apply plugin: 'kotlin-android' | apply plugin: 'kotlin-android' | ||||||
|  |  | ||||||
|  | apply plugin: 'kotlin-kapt' | ||||||
|  |  | ||||||
| apply plugin: 'kotlin-android-extensions' | apply plugin: 'kotlin-android-extensions' | ||||||
|  |  | ||||||
| repositories { |  | ||||||
|     maven { |  | ||||||
|         url 'https://maven.fabric.io/public' |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| android { | android { | ||||||
|     compileSdkVersion 27 |     compileOptions { | ||||||
|     buildToolsVersion '27.0.0' |         sourceCompatibility JavaVersion.VERSION_1_8 | ||||||
|  |         targetCompatibility JavaVersion.VERSION_1_8 | ||||||
|  |     } | ||||||
|  |     compileSdkVersion 30 | ||||||
|  |     buildToolsVersion '30.0.3' | ||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         applicationId "apps.amine.bou.readerforselfoss" |         applicationId "apps.amine.bou.readerforselfoss" | ||||||
|         minSdkVersion 16 |         minSdkVersion 16 | ||||||
|         targetSdkVersion 27 |         targetSdkVersion 30 | ||||||
|         versionCode versionCodeFromGit() |         versionCode versionCodeFromGit() | ||||||
|         versionName versionNameFromGit() |         versionName versionNameFromGit() | ||||||
|  |  | ||||||
| @@ -58,26 +55,23 @@ android { | |||||||
|         vectorDrawables.useSupportLibrary = true |         vectorDrawables.useSupportLibrary = true | ||||||
|  |  | ||||||
|         // tests |         // tests | ||||||
|         testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" |         testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" | ||||||
|  |  | ||||||
|         buildConfigField "String", "MERCURY_KEY", mercuryApiKey |         javaCompileOptions { | ||||||
|         buildConfigField "String", "FEEDBACK_EMAIL", feedbackEmail |             annotationProcessorOptions { | ||||||
|         buildConfigField "String", "SOURCE_URL", sourceUrl |                 arguments = ["room.schemaLocation": | ||||||
|         buildConfigField "String", "TRACKER_URL", trackerUrl |                                      "$projectDir/schemas".toString()] | ||||||
|         buildConfigField "String", "TRANSLATION_URL", translationUrl |             } | ||||||
|         buildConfigField "String", "GITHUB_TOKEN", githubToken |         } | ||||||
|     } |     } | ||||||
|     buildTypes { |     buildTypes { | ||||||
|         release { |         release { | ||||||
|             minifyEnabled true |             minifyEnabled true | ||||||
|             shrinkResources true |             shrinkResources false | ||||||
|             proguardFiles getDefaultProguardFile('proguard-android.txt'), |             proguardFiles getDefaultProguardFile('proguard-android.txt'), | ||||||
|                     'proguard-rules.pro' |                     'proguard-rules.pro' | ||||||
|         } |         } | ||||||
|         debug { |         debug { | ||||||
|             buildConfigField "String", "LOGIN_URL", appLoginUrl |  | ||||||
|             buildConfigField "String", "LOGIN_USERNAME", appLoginUsername |  | ||||||
|             buildConfigField "String", "LOGIN_PASSWORD", appLoginPassword |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     flavorDimensions "build" |     flavorDimensions "build" | ||||||
| @@ -85,136 +79,75 @@ android { | |||||||
|         githubConfig { |         githubConfig { | ||||||
|             versionNameSuffix '-github' |             versionNameSuffix '-github' | ||||||
|             dimension "build" |             dimension "build" | ||||||
|             buildConfigField "boolean", "GITHUB_VERSION", "true" |  | ||||||
|         } |  | ||||||
|         storeConfig { |  | ||||||
|             versionNameSuffix '-store' |  | ||||||
|             dimension "build" |  | ||||||
|             buildConfigField "boolean", "GITHUB_VERSION", "false" |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     // Testing |     // Testing | ||||||
|     androidTestCompile 'com.android.support.test.espresso:espresso-core:3.0.1' |     androidTestImplementation 'androidx.test.espresso:espresso-core:3.4.0-alpha02' | ||||||
|     androidTestCompile 'com.android.support.test:runner:1.0.1' |     androidTestImplementation 'androidx.test:runner:1.3.1-alpha02' | ||||||
|     // Espresso-contrib for DatePicker, RecyclerView, Drawer actions, Accessibility checks, CountingIdlingResource |     // Espresso-contrib for DatePicker, RecyclerView, Drawer actions, Accessibility checks, CountingIdlingResource | ||||||
|     androidTestCompile 'com.android.support.test.espresso:espresso-contrib:3.0.1' |     androidTestImplementation 'androidx.test.espresso:espresso-contrib:3.4.0-alpha02' | ||||||
|     // Espresso-intents for validation and stubbing of Intents |     // Espresso-intents for validation and stubbing of Intents | ||||||
|     androidTestCompile 'com.android.support.test.espresso:espresso-intents:3.0.1' |     androidTestImplementation 'androidx.test.espresso:espresso-intents:3.4.0-alpha02' | ||||||
|  |     implementation fileTree(include: ['*.jar'], dir: 'libs') | ||||||
|  |     implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version" | ||||||
|     compile fileTree(dir: 'libs', include: ['*.jar']) |  | ||||||
|     compile "org.jetbrains.kotlin:kotlin-stdlib-jre7:$kotlin_version" |  | ||||||
|  |  | ||||||
|     // Android Support |     // Android Support | ||||||
|     compile 'com.android.support:appcompat-v7:27.0.0' |     implementation "androidx.appcompat:appcompat:1.3.0-alpha02" | ||||||
|     compile 'com.android.support:design:27.0.0' |     implementation 'com.google.android.material:material:1.3.0-beta01' | ||||||
|     compile 'com.android.support:recyclerview-v7:27.0.0' |     implementation 'androidx.recyclerview:recyclerview:1.2.0-beta01' | ||||||
|     compile 'com.android.support:support-v4:27.0.0' |     implementation "androidx.legacy:legacy-support-v4:$android_version" | ||||||
|     compile 'com.android.support:support-vector-drawable:27.0.0' |     implementation 'androidx.vectordrawable:vectordrawable:1.2.0-alpha02' | ||||||
|     compile 'com.android.support:customtabs:27.0.0' |     implementation "androidx.browser:browser:1.3.0" | ||||||
|     compile 'com.android.support:cardview-v7:27.0.0' |     implementation "androidx.cardview:cardview:$android_version" | ||||||
|     compile 'com.android.support.constraint:constraint-layout:1.0.2' |     implementation 'androidx.constraintlayout:constraintlayout:2.1.0-alpha2' | ||||||
|  |     implementation 'org.jsoup:jsoup:1.13.1' | ||||||
|     // Firebase + crashlytics |  | ||||||
|     compile 'com.google.firebase:firebase-core:11.4.2' |  | ||||||
|     compile 'com.google.firebase:firebase-config:11.4.2' |  | ||||||
|     compile 'com.google.firebase:firebase-invites:11.4.2' |  | ||||||
|     compile('com.crashlytics.sdk.android:crashlytics:2.7.1@aar') { |  | ||||||
|         transitive = true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     //multidex |     //multidex | ||||||
|     compile 'com.android.support:multidex:1.0.2' |     implementation 'androidx.multidex:multidex:2.0.1' | ||||||
|  |  | ||||||
|     // Intro |  | ||||||
|     compile 'agency.tango.android:material-intro-screen:0.0.5' |  | ||||||
|  |  | ||||||
|     // About |     // About | ||||||
|     compile('com.mikepenz:aboutlibraries:6.0.0@aar') { |     implementation('com.mikepenz:aboutlibraries:6.2.0@aar') { | ||||||
|         transitive = true |         transitive = true | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Retrofit + http logging + okhttp |     // Retrofit + http logging + okhttp | ||||||
|     compile 'com.squareup.retrofit2:retrofit:2.3.0' |     implementation 'com.squareup.retrofit2:retrofit:2.3.0' | ||||||
|     compile 'com.squareup.okhttp3:logging-interceptor:3.9.0' |     implementation 'com.squareup.okhttp3:logging-interceptor:3.9.0' | ||||||
|     compile 'com.squareup.retrofit2:converter-gson:2.3.0' |     implementation 'com.squareup.retrofit2:converter-gson:2.3.0' | ||||||
|     compile 'com.burgstaller:okhttp-digest:1.12' |     implementation 'com.burgstaller:okhttp-digest:1.12' | ||||||
|  |  | ||||||
|     // Material-ish things |     // Material-ish things | ||||||
|     compile 'com.ashokvarma.android:bottom-navigation-bar:2.0.3' |     implementation 'com.ashokvarma.android:bottom-navigation-bar:2.1.0' | ||||||
|     compile 'com.github.jd-alexander:LikeButton:0.2.1' |     implementation 'com.github.jd-alexander:LikeButton:0.2.3' | ||||||
|     compile 'com.amulyakhare:com.amulyakhare.textdrawable:1.0.1' |     implementation 'com.amulyakhare:com.amulyakhare.textdrawable:1.0.1' | ||||||
|     compile 'org.sufficientlysecure:html-textview:3.5' |  | ||||||
|  |  | ||||||
|     // glide |     // glide | ||||||
|     compile 'com.github.bumptech.glide:glide:4.1.1' |     implementation 'com.github.bumptech.glide:glide:4.1.1' | ||||||
|     compile 'com.github.bumptech.glide:okhttp3-integration:4.1.1' |     implementation 'com.github.bumptech.glide:okhttp3-integration:4.1.1' | ||||||
|  |  | ||||||
|     // Asking politely users to rate the app |  | ||||||
|     compile 'com.github.stkent:amplify:2.1.0' |  | ||||||
|  |  | ||||||
|     // For the article reader |  | ||||||
|     compile 'com.klinkerapps:drag-dismiss-activity:1.5.0' |  | ||||||
|  |  | ||||||
|     // Drawer |     // Drawer | ||||||
|     implementation 'co.zsmb:materialdrawer-kt:1.2.1' |     implementation 'co.zsmb:materialdrawer-kt:2.0.2' | ||||||
|     compile 'com.anupcowkur:reservoir:3.1.0' |  | ||||||
|  |  | ||||||
|     // Themes |     // Themes | ||||||
|     compile 'com.52inc:scoops:1.0.0' |     implementation 'com.52inc:scoops:1.0.0' | ||||||
|  |     implementation 'com.jaredrummler:colorpicker:1.0.2' | ||||||
|  |     implementation 'com.github.rubensousa:floatingtoolbar:1.5.1' | ||||||
|  |  | ||||||
|     // Github issues reporter |     // Pager | ||||||
|     compile 'com.heinrichreimersoftware:android-issue-reporter:1.3.1' |     implementation 'me.relex:circleindicator:2.0.0@aar' | ||||||
|  |  | ||||||
|     compile 'com.github.rubensousa:floatingtoolbar:1.5.1' |     //PhotoView | ||||||
| } |     implementation 'com.github.chrisbanes:PhotoView:2.0.0' | ||||||
|  |  | ||||||
| apply plugin: 'com.google.gms.google-services' |     implementation 'androidx.core:core-ktx:1.5.0-alpha05' | ||||||
|  |  | ||||||
|  |     implementation "androidx.lifecycle:lifecycle-livedata:2.3.0-rc01" | ||||||
| afterEvaluate { |     implementation "androidx.lifecycle:lifecycle-common-java8:2.3.0-rc01" | ||||||
|     initFabricPropertiesIfNeeded() |  | ||||||
|     initAppLoginPropertiesIfNeeded() |     implementation "androidx.room:room-runtime:2.3.0-alpha04" | ||||||
|     initAppForSecretPropertiesIfNeeded() |     kapt "androidx.room:room-compiler:2.3.0-alpha04" | ||||||
| } |  | ||||||
|  |     implementation "android.arch.work:work-runtime-ktx:$work_version" | ||||||
| def initFabricPropertiesIfNeeded() { |  | ||||||
|     def propertiesFile = file('fabric.properties') |  | ||||||
|     if (!propertiesFile.exists()) { |  | ||||||
|         def commentMessage = "This is autogenerated fabric property from system environment to prevent key to be committed to source control." |  | ||||||
|         ant.propertyfile(file: "fabric.properties", comment: commentMessage) { |  | ||||||
|             entry(key: "apiSecret", value: crashlyticsdemoApisecret) |  | ||||||
|             entry(key: "apiKey", value: crashlyticsdemoApikey) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def initAppLoginPropertiesIfNeeded() { |  | ||||||
|     def propertiesFile = file(System.getProperty("user.home") + '/.gradle/gradle.properties') |  | ||||||
|     if (!propertiesFile.exists()) { |  | ||||||
|         def commentMessage = "This is autogenerated local property from system environment to prevent key to be committed to source control." |  | ||||||
|         ant.propertyfile(file: System.getProperty("user.home") + "/.gradle/gradle.properties", comment: commentMessage) { |  | ||||||
|             entry(key: "appLoginUrl", value: System.getProperty("appLoginUrl")) |  | ||||||
|             entry(key: "appLoginUsername", value: System.getProperty("appLoginUsername")) |  | ||||||
|             entry(key: "appLoginPassword", value: System.getProperty("appLoginPassword")) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def initAppForSecretPropertiesIfNeeded() { |  | ||||||
|     def propertiesFile = file(System.getProperty("user.home") + '/.gradle/gradle.properties') |  | ||||||
|     if (!propertiesFile.exists()) { |  | ||||||
|         def commentMessage = "This is autogenerated local property from system environment to prevent key to be committed to source control." |  | ||||||
|         ant.propertyfile(file: System.getProperty("user.home") + "/.gradle/gradle.properties", comment: commentMessage) { |  | ||||||
|             entry(key: "mercuryApiKey", value: System.getProperty("mercuryApiKey")) |  | ||||||
|             entry(key: "feedbackEmail", value: System.getProperty("feedbackEmail")) |  | ||||||
|             entry(key: "sourceUrl", value: System.getProperty("sourceUrl")) |  | ||||||
|             entry(key: "trackerUrl", value: System.getProperty("trackerUrl")) |  | ||||||
|             entry(key: "translationUrl", value: System.getProperty("translationUrl")) |  | ||||||
|             entry(key: "githubToken", value: System.getProperty("githubToken")) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								app/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -30,25 +30,13 @@ | |||||||
|     <fields>; |     <fields>; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| ##Retrofit |  | ||||||
| #-keep class com.google.gson.** { *; } |  | ||||||
| #-keep class com.google.inject.** { *; } |  | ||||||
| #-keep class org.apache.http.** { *; } |  | ||||||
| #-keep class org.apache.james.mime4j.** { *; } |  | ||||||
| #-keep class javax.inject.** { *; } |  | ||||||
| #-keep class retrofit.** { *; } |  | ||||||
| #-keepclassmembernames interface * { |  | ||||||
| #    @retrofit.http.* <methods>; |  | ||||||
| #} |  | ||||||
| #-keep class retrofit.** { *; } |  | ||||||
| #-keep class apps.amine.bou.readerforselfoss.api.selfoss.model.** { *; } |  | ||||||
| #-keepclassmembernames interface * { |  | ||||||
| #    @retrofit.http.* <methods>; |  | ||||||
| #} |  | ||||||
| -dontwarn okio.** | -dontwarn okio.** | ||||||
| -dontwarn retrofit2.Platform$Java8 | -dontwarn retrofit2.Platform$Java8 | ||||||
| -keepattributes Signature | -keep class retrofit.** { *; } | ||||||
|  | -keepclasseswithmembers class * { | ||||||
|  |     @retrofit.http.* <methods>; | ||||||
|  | } | ||||||
|  | -keepattributes *Annotation*,Signature | ||||||
| -keepattributes Exceptions | -keepattributes Exceptions | ||||||
| -dontwarn okio.** | -dontwarn okio.** | ||||||
| -dontwarn javax.annotation.Nullable | -dontwarn javax.annotation.Nullable | ||||||
| @@ -69,4 +57,9 @@ | |||||||
|  |  | ||||||
| -dontwarn com.anupcowkur.reservoir.** | -dontwarn com.anupcowkur.reservoir.** | ||||||
|  |  | ||||||
| -dontwarn javax.annotation.** | -dontwarn javax.annotation.** | ||||||
|  |  | ||||||
|  | -keep class android.support.v7.widget.SearchView { *; } | ||||||
|  |  | ||||||
|  | # maybe remove later ? | ||||||
|  | -keep class * extends androidx.fragment.app.Fragment | ||||||
|   | |||||||
| @@ -0,0 +1,96 @@ | |||||||
|  | { | ||||||
|  |   "formatVersion": 1, | ||||||
|  |   "database": { | ||||||
|  |     "version": 1, | ||||||
|  |     "identityHash": "08ca537d7ac9d4dd216e8e395d70801a", | ||||||
|  |     "entities": [ | ||||||
|  |       { | ||||||
|  |         "tableName": "tags", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`tag` TEXT NOT NULL, `color` TEXT NOT NULL, `unread` INTEGER NOT NULL, PRIMARY KEY(`tag`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tag", | ||||||
|  |             "columnName": "tag", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "color", | ||||||
|  |             "columnName": "color", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "tag" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "sources", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `title` TEXT NOT NULL, `tags` TEXT NOT NULL, `spout` TEXT NOT NULL, `error` TEXT NOT NULL, `icon` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "spout", | ||||||
|  |             "columnName": "spout", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "error", | ||||||
|  |             "columnName": "error", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "views": [], | ||||||
|  |     "setupQueries": [ | ||||||
|  |       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||||
|  |       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, \"08ca537d7ac9d4dd216e8e395d70801a\")" | ||||||
|  |     ] | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -0,0 +1,176 @@ | |||||||
|  | { | ||||||
|  |   "formatVersion": 1, | ||||||
|  |   "database": { | ||||||
|  |     "version": 2, | ||||||
|  |     "identityHash": "6fa6944b04100d68eab61039876a8804", | ||||||
|  |     "entities": [ | ||||||
|  |       { | ||||||
|  |         "tableName": "tags", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`tag` TEXT NOT NULL, `color` TEXT NOT NULL, `unread` INTEGER NOT NULL, PRIMARY KEY(`tag`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tag", | ||||||
|  |             "columnName": "tag", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "color", | ||||||
|  |             "columnName": "color", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "tag" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "sources", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `title` TEXT NOT NULL, `tags` TEXT NOT NULL, `spout` TEXT NOT NULL, `error` TEXT NOT NULL, `icon` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "spout", | ||||||
|  |             "columnName": "spout", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "error", | ||||||
|  |             "columnName": "error", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "items", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `datetime` TEXT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `thumbnail` TEXT NOT NULL, `icon` TEXT NOT NULL, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "datetime", | ||||||
|  |             "columnName": "datetime", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "content", | ||||||
|  |             "columnName": "content", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "starred", | ||||||
|  |             "columnName": "starred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "thumbnail", | ||||||
|  |             "columnName": "thumbnail", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "link", | ||||||
|  |             "columnName": "link", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "sourcetitle", | ||||||
|  |             "columnName": "sourcetitle", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "views": [], | ||||||
|  |     "setupQueries": [ | ||||||
|  |       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||||
|  |       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, \"6fa6944b04100d68eab61039876a8804\")" | ||||||
|  |     ] | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -0,0 +1,226 @@ | |||||||
|  | { | ||||||
|  |   "formatVersion": 1, | ||||||
|  |   "database": { | ||||||
|  |     "version": 3, | ||||||
|  |     "identityHash": "7ad9c4961992c13b670128485ebb3efc", | ||||||
|  |     "entities": [ | ||||||
|  |       { | ||||||
|  |         "tableName": "tags", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`tag` TEXT NOT NULL, `color` TEXT NOT NULL, `unread` INTEGER NOT NULL, PRIMARY KEY(`tag`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tag", | ||||||
|  |             "columnName": "tag", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "color", | ||||||
|  |             "columnName": "color", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "tag" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "sources", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `title` TEXT NOT NULL, `tags` TEXT NOT NULL, `spout` TEXT NOT NULL, `error` TEXT NOT NULL, `icon` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "spout", | ||||||
|  |             "columnName": "spout", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "error", | ||||||
|  |             "columnName": "error", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "items", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `datetime` TEXT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `thumbnail` TEXT NOT NULL, `icon` TEXT NOT NULL, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "datetime", | ||||||
|  |             "columnName": "datetime", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "content", | ||||||
|  |             "columnName": "content", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "starred", | ||||||
|  |             "columnName": "starred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "thumbnail", | ||||||
|  |             "columnName": "thumbnail", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "link", | ||||||
|  |             "columnName": "link", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "sourcetitle", | ||||||
|  |             "columnName": "sourcetitle", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "actions", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `articleid` TEXT NOT NULL, `read` INTEGER NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `unstarred` INTEGER NOT NULL)", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "articleId", | ||||||
|  |             "columnName": "articleid", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "read", | ||||||
|  |             "columnName": "read", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "starred", | ||||||
|  |             "columnName": "starred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unstarred", | ||||||
|  |             "columnName": "unstarred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": true | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "views": [], | ||||||
|  |     "setupQueries": [ | ||||||
|  |       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||||
|  |       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, \"7ad9c4961992c13b670128485ebb3efc\")" | ||||||
|  |     ] | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -0,0 +1,226 @@ | |||||||
|  | { | ||||||
|  |   "formatVersion": 1, | ||||||
|  |   "database": { | ||||||
|  |     "version": 4, | ||||||
|  |     "identityHash": "9cf8b03d32f80dfd58160599a1df197d", | ||||||
|  |     "entities": [ | ||||||
|  |       { | ||||||
|  |         "tableName": "tags", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`tag` TEXT NOT NULL, `color` TEXT NOT NULL, `unread` INTEGER NOT NULL, PRIMARY KEY(`tag`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tag", | ||||||
|  |             "columnName": "tag", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "color", | ||||||
|  |             "columnName": "color", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "tag" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "sources", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `title` TEXT NOT NULL, `tags` TEXT NOT NULL, `spout` TEXT NOT NULL, `error` TEXT NOT NULL, `icon` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "spout", | ||||||
|  |             "columnName": "spout", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "error", | ||||||
|  |             "columnName": "error", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "items", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `datetime` TEXT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `thumbnail` TEXT, `icon` TEXT, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "datetime", | ||||||
|  |             "columnName": "datetime", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "content", | ||||||
|  |             "columnName": "content", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "starred", | ||||||
|  |             "columnName": "starred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "thumbnail", | ||||||
|  |             "columnName": "thumbnail", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": false | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": false | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "link", | ||||||
|  |             "columnName": "link", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "sourcetitle", | ||||||
|  |             "columnName": "sourcetitle", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "actions", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `articleid` TEXT NOT NULL, `read` INTEGER NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `unstarred` INTEGER NOT NULL)", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "articleId", | ||||||
|  |             "columnName": "articleid", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "read", | ||||||
|  |             "columnName": "read", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "starred", | ||||||
|  |             "columnName": "starred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unstarred", | ||||||
|  |             "columnName": "unstarred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": true | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "views": [], | ||||||
|  |     "setupQueries": [ | ||||||
|  |       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||||
|  |       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, \"9cf8b03d32f80dfd58160599a1df197d\")" | ||||||
|  |     ] | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -2,30 +2,29 @@ package apps.amine.bou.readerforselfoss | |||||||
|  |  | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.support.test.InstrumentationRegistry | import androidx.test.InstrumentationRegistry | ||||||
| import android.support.test.espresso.Espresso.onView | import androidx.test.espresso.Espresso.onView | ||||||
| import android.support.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||||
| import android.support.test.espresso.action.ViewActions.click | import androidx.test.espresso.action.ViewActions.click | ||||||
| import android.support.test.espresso.action.ViewActions.closeSoftKeyboard | import androidx.test.espresso.action.ViewActions.closeSoftKeyboard | ||||||
| import android.support.test.espresso.action.ViewActions.pressBack | import androidx.test.espresso.action.ViewActions.pressBack | ||||||
| import android.support.test.espresso.action.ViewActions.pressKey | import androidx.test.espresso.action.ViewActions.pressKey | ||||||
| import android.support.test.espresso.action.ViewActions.typeText | import androidx.test.espresso.action.ViewActions.typeText | ||||||
| import android.support.test.espresso.assertion.ViewAssertions.matches | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
| import android.support.test.espresso.contrib.DrawerActions | import androidx.test.espresso.contrib.DrawerActions | ||||||
| import android.support.test.espresso.intent.Intents | import androidx.test.espresso.intent.Intents | ||||||
| import android.support.test.espresso.intent.Intents.intended | import androidx.test.espresso.intent.Intents.intended | ||||||
| import android.support.test.espresso.intent.Intents.times | import androidx.test.espresso.intent.Intents.times | ||||||
| import android.support.test.espresso.intent.matcher.IntentMatchers.hasComponent | import androidx.test.espresso.intent.matcher.IntentMatchers.hasComponent | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.isDisplayed | import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.isRoot | import androidx.test.espresso.matcher.ViewMatchers.isRoot | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.withContentDescription | import androidx.test.espresso.matcher.ViewMatchers.withContentDescription | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.withId | import androidx.test.espresso.matcher.ViewMatchers.withId | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.withText | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
| import android.support.test.rule.ActivityTestRule | import androidx.test.rule.ActivityTestRule | ||||||
| import android.support.test.runner.AndroidJUnit4 | import androidx.test.runner.AndroidJUnit4 | ||||||
| import android.view.KeyEvent | import android.view.KeyEvent | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import com.heinrichreimersoftware.androidissuereporter.IssueReporterLauncher |  | ||||||
| import org.junit.After | import org.junit.After | ||||||
| import org.junit.Before | import org.junit.Before | ||||||
| import org.junit.Rule | import org.junit.Rule | ||||||
| @@ -92,25 +91,6 @@ class HomeActivityEspressoTest { | |||||||
|         intended(hasComponent(LoginActivity::class.java.name), times(1)) |         intended(hasComponent(LoginActivity::class.java.name), times(1)) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun drawerTesting() { |  | ||||||
|  |  | ||||||
|         rule.launchActivity(Intent()) |  | ||||||
|  |  | ||||||
|         onView(withId(R.id.material_drawer_layout)).perform(DrawerActions.open()) |  | ||||||
|  |  | ||||||
|         onView(withText(R.string.drawer_report_bug)).perform(click()) |  | ||||||
|         intended(hasComponent(IssueReporterLauncher.Activity::class.java.name)) |  | ||||||
|         onView(isRoot()).perform(pressBack()) |  | ||||||
|         onView(isRoot()).perform(pressBack()) |  | ||||||
|         intended(hasComponent(HomeActivity::class.java.name)) |  | ||||||
|  |  | ||||||
|         onView(withId(R.id.material_drawer_layout)).perform(DrawerActions.open()) |  | ||||||
|  |  | ||||||
|         onView(withId(R.id.material_drawer_layout)).perform(DrawerActions.open()) |  | ||||||
|         onView(withText(R.string.drawer_action_clear)).perform(click()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // TODO: test articles opening and actions for cards and lists |     // TODO: test articles opening and actions for cards and lists | ||||||
|  |  | ||||||
|     @After |     @After | ||||||
|   | |||||||
| @@ -1,91 +0,0 @@ | |||||||
| package apps.amine.bou.readerforselfoss |  | ||||||
|  |  | ||||||
| import android.content.Context |  | ||||||
| import android.content.Intent |  | ||||||
| import android.support.test.InstrumentationRegistry.getInstrumentation |  | ||||||
| import android.support.test.espresso.Espresso.onView |  | ||||||
| import android.support.test.espresso.action.ViewActions.click |  | ||||||
| import android.support.test.espresso.assertion.ViewAssertions.matches |  | ||||||
| import android.support.test.espresso.intent.Intents |  | ||||||
| import android.support.test.espresso.intent.Intents.intended |  | ||||||
| import android.support.test.espresso.intent.Intents.times |  | ||||||
| import android.support.test.espresso.intent.matcher.IntentMatchers.hasComponent |  | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.isDisplayed |  | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.withId |  | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.withText |  | ||||||
| import android.support.test.rule.ActivityTestRule |  | ||||||
| import android.support.test.runner.AndroidJUnit4 |  | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config |  | ||||||
| import org.junit.After |  | ||||||
| import org.junit.Before |  | ||||||
| import org.junit.Rule |  | ||||||
| import org.junit.Test |  | ||||||
| import org.junit.runner.RunWith |  | ||||||
| import java.util.* |  | ||||||
|  |  | ||||||
| @RunWith(AndroidJUnit4::class) |  | ||||||
| class IntroActivityEspressoTest { |  | ||||||
|  |  | ||||||
|     @Rule @JvmField |  | ||||||
|     val rule = ActivityTestRule(IntroActivity::class.java, true, false) |  | ||||||
|  |  | ||||||
|     @Before |  | ||||||
|     fun clearData() { |  | ||||||
|         val editor = |  | ||||||
|                 getInstrumentation().targetContext |  | ||||||
|                         .getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) |  | ||||||
|                         .edit() |  | ||||||
|         editor.clear() |  | ||||||
|         editor.commit() |  | ||||||
|  |  | ||||||
|         Intents.init() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun nextEachTimes() { |  | ||||||
|  |  | ||||||
|         rule.launchActivity(Intent()) |  | ||||||
|  |  | ||||||
|         onView(withText(R.string.intro_hello_title)).check(matches(isDisplayed())) |  | ||||||
|         onView(withId(R.id.button_next)).perform(click()) |  | ||||||
|         onView(withText(R.string.intro_needs_selfoss_message)).check(matches(isDisplayed())) |  | ||||||
|         onView(withId(R.id.button_next)).perform(click()) |  | ||||||
|         onView(withText(R.string.intro_all_set_message)).check(matches(isDisplayed())) |  | ||||||
|         onView(withId(R.id.button_next)).perform(click()) |  | ||||||
|  |  | ||||||
|         intended(hasComponent(IntroActivity::class.java.name), times(1)) |  | ||||||
|         intended(hasComponent(LoginActivity::class.java.name), times(1)) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun nextBackRandomTimes() { |  | ||||||
|         val max = 5 |  | ||||||
|         val min = 1 |  | ||||||
|  |  | ||||||
|         val random = (Random().nextInt(max + 1 - min)) + min |  | ||||||
|  |  | ||||||
|         rule.launchActivity(Intent()) |  | ||||||
|  |  | ||||||
|         onView(withText(R.string.intro_hello_title)).check(matches(isDisplayed())) |  | ||||||
|         onView(withId(R.id.button_next)).perform(click()) |  | ||||||
|  |  | ||||||
|         repeat(random) { _ -> |  | ||||||
|             onView(withText(R.string.intro_needs_selfoss_message)).check(matches(isDisplayed())) |  | ||||||
|             onView(withId(R.id.button_next)).perform(click()) |  | ||||||
|             onView(withText(R.string.intro_all_set_message)).check(matches(isDisplayed())) |  | ||||||
|             onView(withId(R.id.button_back)).perform(click()) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         onView(withId(R.id.button_next)).perform(click()) |  | ||||||
|         onView(withText(R.string.intro_all_set_message)).check(matches(isDisplayed())) |  | ||||||
|         onView(withId(R.id.button_next)).perform(click()) |  | ||||||
|  |  | ||||||
|         intended(hasComponent(IntroActivity::class.java.name), times(1)) |  | ||||||
|         intended(hasComponent(LoginActivity::class.java.name), times(1)) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @After |  | ||||||
|     fun releaseIntents() { |  | ||||||
|         Intents.release() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -2,25 +2,25 @@ package apps.amine.bou.readerforselfoss | |||||||
|  |  | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.support.test.InstrumentationRegistry | import androidx.test.InstrumentationRegistry | ||||||
| import android.support.test.espresso.Espresso.onView | import androidx.test.espresso.Espresso.onView | ||||||
| import android.support.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||||
| import android.support.test.espresso.action.ViewActions.click | import androidx.test.espresso.action.ViewActions.click | ||||||
| import android.support.test.espresso.action.ViewActions.closeSoftKeyboard | import androidx.test.espresso.action.ViewActions.closeSoftKeyboard | ||||||
| import android.support.test.espresso.action.ViewActions.pressBack | import androidx.test.espresso.action.ViewActions.pressBack | ||||||
| import android.support.test.espresso.action.ViewActions.typeText | import androidx.test.espresso.action.ViewActions.typeText | ||||||
| import android.support.test.espresso.assertion.ViewAssertions.matches | import androidx.test.espresso.assertion.ViewAssertions.matches | ||||||
| import android.support.test.espresso.intent.Intents | import androidx.test.espresso.intent.Intents | ||||||
| import android.support.test.espresso.intent.Intents.intended | import androidx.test.espresso.intent.Intents.intended | ||||||
| import android.support.test.espresso.intent.Intents.times | import androidx.test.espresso.intent.Intents.times | ||||||
| import android.support.test.espresso.intent.matcher.IntentMatchers.hasComponent | import androidx.test.espresso.intent.matcher.IntentMatchers.hasComponent | ||||||
| import android.support.test.espresso.matcher.ViewMatchers | import androidx.test.espresso.matcher.ViewMatchers | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.isRoot | import androidx.test.espresso.matcher.ViewMatchers.isRoot | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.withEffectiveVisibility | import androidx.test.espresso.matcher.ViewMatchers.withEffectiveVisibility | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.withId | import androidx.test.espresso.matcher.ViewMatchers.withId | ||||||
| import android.support.test.espresso.matcher.ViewMatchers.withText | import androidx.test.espresso.matcher.ViewMatchers.withText | ||||||
| import android.support.test.rule.ActivityTestRule | import androidx.test.rule.ActivityTestRule | ||||||
| import android.support.test.runner.AndroidJUnit4 | import androidx.test.runner.AndroidJUnit4 | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import com.mikepenz.aboutlibraries.ui.LibsActivity | import com.mikepenz.aboutlibraries.ui.LibsActivity | ||||||
| import org.junit.After | import org.junit.After | ||||||
|   | |||||||
| @@ -3,13 +3,13 @@ package apps.amine.bou.readerforselfoss | |||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.content.SharedPreferences | import android.content.SharedPreferences | ||||||
| import android.preference.PreferenceManager | import android.preference.PreferenceManager | ||||||
| import android.support.test.InstrumentationRegistry.getInstrumentation | import androidx.test.InstrumentationRegistry.getInstrumentation | ||||||
| import android.support.test.espresso.intent.Intents | import androidx.test.espresso.intent.Intents | ||||||
| import android.support.test.espresso.intent.Intents.intended | import androidx.test.espresso.intent.Intents.intended | ||||||
| import android.support.test.espresso.intent.Intents.times | import androidx.test.espresso.intent.Intents.times | ||||||
| import android.support.test.espresso.intent.matcher.IntentMatchers.hasComponent | import androidx.test.espresso.intent.matcher.IntentMatchers.hasComponent | ||||||
| import android.support.test.rule.ActivityTestRule | import androidx.test.rule.ActivityTestRule | ||||||
| import android.support.test.runner.AndroidJUnit4 | import androidx.test.runner.AndroidJUnit4 | ||||||
| import org.junit.After | import org.junit.After | ||||||
|  |  | ||||||
| import org.junit.Before | import org.junit.Before | ||||||
| @@ -45,7 +45,6 @@ class MainActivityEspressoTest { | |||||||
|         rule.launchActivity(intent) |         rule.launchActivity(intent) | ||||||
|  |  | ||||||
|         intended(hasComponent(MainActivity::class.java.name)) |         intended(hasComponent(MainActivity::class.java.name)) | ||||||
|         intended(hasComponent(IntroActivity::class.java.name)) |  | ||||||
|         intended(hasComponent(LoginActivity::class.java.name), times(0)) |         intended(hasComponent(LoginActivity::class.java.name), times(0)) | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -58,7 +57,6 @@ class MainActivityEspressoTest { | |||||||
|  |  | ||||||
|         intended(hasComponent(MainActivity::class.java.name)) |         intended(hasComponent(MainActivity::class.java.name)) | ||||||
|         intended(hasComponent(LoginActivity::class.java.name)) |         intended(hasComponent(LoginActivity::class.java.name)) | ||||||
|         intended(hasComponent(IntroActivity::class.java.name), times(0)) |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @After |     @After | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| package apps.amine.bou.readerforselfoss | package apps.amine.bou.readerforselfoss | ||||||
|  |  | ||||||
| import android.support.design.widget.TextInputLayout | import com.google.android.material.textfield.TextInputLayout | ||||||
| import android.support.test.espresso.matcher.ViewMatchers | import androidx.test.espresso.matcher.ViewMatchers | ||||||
| import android.view.View | import android.view.View | ||||||
| import org.hamcrest.Description | import org.hamcrest.Description | ||||||
| import org.hamcrest.Matcher | import org.hamcrest.Matcher | ||||||
|   | |||||||
| @@ -2,11 +2,8 @@ | |||||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|     package="apps.amine.bou.readerforselfoss"> |     package="apps.amine.bou.readerforselfoss"> | ||||||
|  |  | ||||||
|     <uses-permission android:name="android.permission.INTERNET" /> |  | ||||||
|  |  | ||||||
|     <!-- For firebase only --> |  | ||||||
|     <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> |     <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> | ||||||
|     <uses-permission android:name="android.permission.WAKE_LOCK" /> |     <uses-permission android:name="android.permission.INTERNET" /> | ||||||
|  |  | ||||||
|     <application |     <application | ||||||
|         android:name=".MyApp" |         android:name=".MyApp" | ||||||
| @@ -14,19 +11,19 @@ | |||||||
|         android:icon="@mipmap/ic_launcher" |         android:icon="@mipmap/ic_launcher" | ||||||
|         android:label="@string/app_name" |         android:label="@string/app_name" | ||||||
|         android:supportsRtl="true" |         android:supportsRtl="true" | ||||||
|  |         android:networkSecurityConfig="@xml/network_security_config" | ||||||
|         android:theme="@style/NoBar"> |         android:theme="@style/NoBar"> | ||||||
|         <activity |         <activity | ||||||
|             android:name=".MainActivity" |             android:name=".MainActivity" | ||||||
|             android:theme="@style/SplashTheme"> |             android:theme="@style/SplashTheme"> | ||||||
|             <intent-filter> |             <intent-filter> | ||||||
|                 <action android:name="android.intent.action.MAIN" /> |                 <action android:name="android.intent.action.MAIN" /> | ||||||
|  |  | ||||||
|                 <category android:name="android.intent.category.LAUNCHER" /> |                 <category android:name="android.intent.category.LAUNCHER" /> | ||||||
|             </intent-filter> |             </intent-filter> | ||||||
|         </activity> |  | ||||||
|         <activity |             <meta-data | ||||||
|             android:name=".IntroActivity" |                 android:name="android.app.shortcuts" | ||||||
|             android:theme="@style/Theme.Intro"> |                 android:resource="@xml/shortcuts" /> | ||||||
|         </activity> |         </activity> | ||||||
|         <activity |         <activity | ||||||
|             android:name=".LoginActivity" |             android:name=".LoginActivity" | ||||||
| @@ -40,7 +37,7 @@ | |||||||
|             android:parentActivityName=".HomeActivity"> |             android:parentActivityName=".HomeActivity"> | ||||||
|             <meta-data |             <meta-data | ||||||
|                 android:name="android.support.PARENT_ACTIVITY" |                 android:name="android.support.PARENT_ACTIVITY" | ||||||
|                 android:value="apps.amine.bou.readerforselfoss.HomeActivity" /> |                 android:value=".HomeActivity" /> | ||||||
|         </activity> |         </activity> | ||||||
|         <activity |         <activity | ||||||
|             android:name=".SourcesActivity" |             android:name=".SourcesActivity" | ||||||
| @@ -58,19 +55,31 @@ | |||||||
|  |  | ||||||
|             <intent-filter> |             <intent-filter> | ||||||
|                 <action android:name="android.intent.action.SEND" /> |                 <action android:name="android.intent.action.SEND" /> | ||||||
|  |  | ||||||
|                 <category android:name="android.intent.category.DEFAULT" /> |                 <category android:name="android.intent.category.DEFAULT" /> | ||||||
|  |  | ||||||
|                 <data android:mimeType="text/plain" /> |                 <data android:mimeType="text/plain" /> | ||||||
|             </intent-filter> |             </intent-filter> | ||||||
|         </activity> |         </activity> | ||||||
|         <activity |         <activity | ||||||
|             android:name=".ReaderActivity"> |             android:name=".ReaderActivity"> | ||||||
|         </activity> |         </activity> | ||||||
|  |         <activity | ||||||
|  |             android:name=".ImageActivity"> | ||||||
|  |         </activity> | ||||||
|  |  | ||||||
|         <meta-data |         <meta-data | ||||||
|             android:name="apps.amine.bou.readerforselfoss.utils.glide.SelfSignedGlideModule" |             android:name="apps.amine.bou.readerforselfoss.utils.glide.SelfSignedGlideModule" | ||||||
|             android:value="GlideModule" /> |             android:value="GlideModule" /> | ||||||
|  |  | ||||||
|  |         <meta-data android:name="android.webkit.WebView.MetricsOptOut" | ||||||
|  |             android:value="true" /> | ||||||
|  |  | ||||||
|  |         <meta-data android:name="android.webkit.WebView.EnableSafeBrowsing" | ||||||
|  |             android:value="true" /> | ||||||
|  |  | ||||||
|  |         <meta-data android:name="android.max_aspect" android:value="2.1" /> | ||||||
|  |         <meta-data | ||||||
|  |             android:name="preloaded_fonts" | ||||||
|  |             android:resource="@array/preloaded_fonts" /> | ||||||
|     </application> |     </application> | ||||||
|  |  | ||||||
| </manifest> | </manifest> | ||||||
|   | |||||||
| @@ -1,10 +1,12 @@ | |||||||
| package apps.amine.bou.readerforselfoss | package apps.amine.bou.readerforselfoss | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
|  | import android.os.Build | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.preference.PreferenceManager | import android.preference.PreferenceManager | ||||||
| import android.support.constraint.ConstraintLayout | import androidx.constraintlayout.widget.ConstraintLayout | ||||||
| import android.support.v7.app.AppCompatActivity | import androidx.appcompat.app.AppCompatActivity | ||||||
| import android.view.View | import android.view.View | ||||||
| import android.widget.AdapterView | import android.widget.AdapterView | ||||||
| import android.widget.ArrayAdapter | import android.widget.ArrayAdapter | ||||||
| @@ -16,6 +18,8 @@ import android.widget.Toast | |||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Spout | import apps.amine.bou.readerforselfoss.api.selfoss.Spout | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.Toppings | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import apps.amine.bou.readerforselfoss.utils.isBaseUrlValid | import apps.amine.bou.readerforselfoss.utils.isBaseUrlValid | ||||||
| import com.ftinc.scoop.Scoop | import com.ftinc.scoop.Scoop | ||||||
| @@ -23,29 +27,72 @@ import kotlinx.android.synthetic.main.activity_add_source.* | |||||||
| import retrofit2.Call | import retrofit2.Call | ||||||
| import retrofit2.Callback | import retrofit2.Callback | ||||||
| import retrofit2.Response | import retrofit2.Response | ||||||
|  | import android.graphics.PorterDuff | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AddSourceActivity : AppCompatActivity() { | class AddSourceActivity : AppCompatActivity() { | ||||||
|  |  | ||||||
|     private var mSpoutsValue: String? = null |     private var mSpoutsValue: String? = null | ||||||
|  |     private lateinit var api: SelfossApi | ||||||
|  |  | ||||||
|  |     private lateinit var appColors: AppColors | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         appColors = AppColors(this@AddSourceActivity) | ||||||
|  |  | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|         Scoop.getInstance().apply(this) |  | ||||||
|         setContentView(R.layout.activity_add_source) |         setContentView(R.layout.activity_add_source) | ||||||
|  |  | ||||||
|  |         val scoop = Scoop.getInstance() | ||||||
|  |         scoop.bind(this, Toppings.PRIMARY.value, toolbar) | ||||||
|  |         if  (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { | ||||||
|  |             scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val drawable = nameInput.background | ||||||
|  |         drawable.setColorFilter(appColors.colorAccent, PorterDuff.Mode.SRC_ATOP) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         // TODO: clean | ||||||
|  |         if(Build.VERSION.SDK_INT > 16) { | ||||||
|  |             nameInput.background = drawable | ||||||
|  |         } else{ | ||||||
|  |             nameInput.setBackgroundDrawable(drawable) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val drawable1 = sourceUri.background | ||||||
|  |         drawable1.setColorFilter(appColors.colorAccent, PorterDuff.Mode.SRC_ATOP) | ||||||
|  |  | ||||||
|  |         if(Build.VERSION.SDK_INT > 16) { | ||||||
|  |             sourceUri.background = drawable1 | ||||||
|  |         } else{ | ||||||
|  |             sourceUri.setBackgroundDrawable(drawable1) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val drawable2 = tags.background | ||||||
|  |         drawable2.setColorFilter(appColors.colorAccent, PorterDuff.Mode.SRC_ATOP) | ||||||
|  |  | ||||||
|  |         if(Build.VERSION.SDK_INT > 16) { | ||||||
|  |             tags.background = drawable2 | ||||||
|  |         } else{ | ||||||
|  |             tags.setBackgroundDrawable(drawable2) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         setSupportActionBar(toolbar) |         setSupportActionBar(toolbar) | ||||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) |         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) |         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||||
|  |  | ||||||
|         var api: SelfossApi? = null |  | ||||||
|  |  | ||||||
|         try { |         try { | ||||||
|             val prefs = PreferenceManager.getDefaultSharedPreferences(this) |             val prefs = PreferenceManager.getDefaultSharedPreferences(this) | ||||||
|  |             val settings = | ||||||
|  |                 getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) | ||||||
|             api = SelfossApi( |             api = SelfossApi( | ||||||
|                     this, |                 this, | ||||||
|                     this@AddSourceActivity, |                 this@AddSourceActivity, | ||||||
|                     prefs.getBoolean("isSelfSignedCert", false), |                 settings.getBoolean("isSelfSignedCert", false), | ||||||
|                     prefs.getBoolean("should_log_everything", false) |                 prefs.getString("api_timeout", "-1")!!.toLong() | ||||||
|             ) |             ) | ||||||
|         } catch (e: IllegalArgumentException) { |         } catch (e: IllegalArgumentException) { | ||||||
|             mustLoginToAddSource() |             mustLoginToAddSource() | ||||||
| @@ -53,13 +100,18 @@ class AddSourceActivity : AppCompatActivity() { | |||||||
|  |  | ||||||
|         maybeGetDetailsFromIntentSharing(intent, sourceUri, nameInput) |         maybeGetDetailsFromIntentSharing(intent, sourceUri, nameInput) | ||||||
|  |  | ||||||
|  |         saveBtn.setTextColor(appColors.colorAccent) | ||||||
|  |  | ||||||
|         saveBtn.setOnClickListener { |         saveBtn.setOnClickListener { | ||||||
|             handleSaveSource(tags, nameInput.text.toString(), sourceUri.text.toString(), api!!) |             handleSaveSource(tags, nameInput.text.toString(), sourceUri.text.toString(), api!!) | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onResume() { | ||||||
|  |         super.onResume() | ||||||
|         val config = Config(this) |         val config = Config(this) | ||||||
|  |  | ||||||
|         if (config.baseUrl.isEmpty() || !config.baseUrl.isBaseUrlValid()) { |         if (config.baseUrl.isEmpty() || !config.baseUrl.isBaseUrlValid(this@AddSourceActivity)) { | ||||||
|             mustLoginToAddSource() |             mustLoginToAddSource() | ||||||
|         } else { |         } else { | ||||||
|             handleSpoutsSpinner(spoutsSpinner, api, progress, formContainer) |             handleSpoutsSpinner(spoutsSpinner, api, progress, formContainer) | ||||||
| @@ -67,16 +119,18 @@ class AddSourceActivity : AppCompatActivity() { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleSpoutsSpinner( |     private fun handleSpoutsSpinner( | ||||||
|             spoutsSpinner: Spinner, |         spoutsSpinner: Spinner, | ||||||
|             api: SelfossApi?, |         api: SelfossApi?, | ||||||
|             mProgress: ProgressBar, |         mProgress: ProgressBar, | ||||||
|             formContainer: ConstraintLayout |         formContainer: ConstraintLayout | ||||||
|     ) { |     ) { | ||||||
|         val spoutsKV = HashMap<String, String>() |         val spoutsKV = HashMap<String, String>() | ||||||
|         spoutsSpinner.onItemSelectedListener = object : AdapterView.OnItemSelectedListener { |         spoutsSpinner.onItemSelectedListener = object : AdapterView.OnItemSelectedListener { | ||||||
|             override fun onItemSelected(adapterView: AdapterView<*>, view: View, i: Int, l: Long) { |             override fun onItemSelected(adapterView: AdapterView<*>, view: View?, i: Int, l: Long) { | ||||||
|                 val spoutName = (view as TextView).text.toString() |                 if (view != null) { | ||||||
|                 mSpoutsValue = spoutsKV[spoutName] |                     val spoutName = (view as TextView).text.toString() | ||||||
|  |                     mSpoutsValue = spoutsKV[spoutName] | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             override fun onNothingSelected(adapterView: AdapterView<*>) { |             override fun onNothingSelected(adapterView: AdapterView<*>) { | ||||||
| @@ -87,26 +141,26 @@ class AddSourceActivity : AppCompatActivity() { | |||||||
|         var items: Map<String, Spout> |         var items: Map<String, Spout> | ||||||
|         api!!.spouts().enqueue(object : Callback<Map<String, Spout>> { |         api!!.spouts().enqueue(object : Callback<Map<String, Spout>> { | ||||||
|             override fun onResponse( |             override fun onResponse( | ||||||
|                     call: Call<Map<String, Spout>>, |                 call: Call<Map<String, Spout>>, | ||||||
|                     response: Response<Map<String, Spout>> |                 response: Response<Map<String, Spout>> | ||||||
|             ) { |             ) { | ||||||
|                 if (response.body() != null) { |                 if (response.body() != null) { | ||||||
|                     items = response.body()!! |                     items = response.body()!! | ||||||
|  |  | ||||||
|                     val itemsStrings = items.map { it.value.name } |                     val itemsStrings = items.map { it.value.name } | ||||||
|                     for ((key, value) in items) { |                     for ((key, value) in items) { | ||||||
|                         spoutsKV.put(value.name, key) |                         spoutsKV[value.name] = key | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     mProgress.visibility = View.GONE |                     mProgress.visibility = View.GONE | ||||||
|                     formContainer.visibility = View.VISIBLE |                     formContainer.visibility = View.VISIBLE | ||||||
|  |  | ||||||
|                     val spinnerArrayAdapter = |                     val spinnerArrayAdapter = | ||||||
|                             ArrayAdapter( |                         ArrayAdapter( | ||||||
|                                     this@AddSourceActivity, |                             this@AddSourceActivity, | ||||||
|                                     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) | ||||||
|                     spoutsSpinner.adapter = spinnerArrayAdapter |                     spoutsSpinner.adapter = spinnerArrayAdapter | ||||||
|                 } else { |                 } else { | ||||||
| @@ -120,9 +174,9 @@ class AddSourceActivity : AppCompatActivity() { | |||||||
|  |  | ||||||
|             private fun handleProblemWithSpouts() { |             private fun handleProblemWithSpouts() { | ||||||
|                 Toast.makeText( |                 Toast.makeText( | ||||||
|                         this@AddSourceActivity, |                     this@AddSourceActivity, | ||||||
|                         R.string.cant_get_spouts, |                     R.string.cant_get_spouts, | ||||||
|                         Toast.LENGTH_SHORT |                     Toast.LENGTH_SHORT | ||||||
|                 ).show() |                 ).show() | ||||||
|                 mProgress.visibility = View.GONE |                 mProgress.visibility = View.GONE | ||||||
|             } |             } | ||||||
| @@ -130,9 +184,9 @@ class AddSourceActivity : AppCompatActivity() { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun maybeGetDetailsFromIntentSharing( |     private fun maybeGetDetailsFromIntentSharing( | ||||||
|             intent: Intent, |         intent: Intent, | ||||||
|             sourceUri: EditText, |         sourceUri: EditText, | ||||||
|             nameInput: EditText |         nameInput: EditText | ||||||
|     ) { |     ) { | ||||||
|         if (Intent.ACTION_SEND == intent.action && "text/plain" == intent.type) { |         if (Intent.ACTION_SEND == intent.action && "text/plain" == intent.type) { | ||||||
|             sourceUri.setText(intent.getStringExtra(Intent.EXTRA_TEXT)) |             sourceUri.setText(intent.getStringExtra(Intent.EXTRA_TEXT)) | ||||||
| @@ -149,38 +203,39 @@ class AddSourceActivity : AppCompatActivity() { | |||||||
|  |  | ||||||
|     private fun handleSaveSource(tags: EditText, title: String, url: String, api: SelfossApi) { |     private fun handleSaveSource(tags: EditText, title: String, url: String, api: SelfossApi) { | ||||||
|  |  | ||||||
|         val sourceDetailsAvailable = title.isEmpty() || url.isEmpty() || mSpoutsValue == null || mSpoutsValue!!.isEmpty() |         val sourceDetailsAvailable = | ||||||
|  |             title.isEmpty() || url.isEmpty() || mSpoutsValue == null || mSpoutsValue!!.isEmpty() | ||||||
|  |  | ||||||
|         if (sourceDetailsAvailable) { |         if (sourceDetailsAvailable) { | ||||||
|             Toast.makeText(this, R.string.form_not_complete, Toast.LENGTH_SHORT).show() |             Toast.makeText(this, R.string.form_not_complete, Toast.LENGTH_SHORT).show() | ||||||
|         } else { |         } else { | ||||||
|             api.createSource( |             api.createSource( | ||||||
|                     title, |                 title, | ||||||
|                     url, |                 url, | ||||||
|                     mSpoutsValue!!, |                 mSpoutsValue!!, | ||||||
|                     tags.text.toString(), |                 tags.text.toString(), | ||||||
|                     "" |                 "" | ||||||
|             ).enqueue(object : Callback<SuccessResponse> { |             ).enqueue(object : Callback<SuccessResponse> { | ||||||
|                 override fun onResponse( |                 override fun onResponse( | ||||||
|                         call: Call<SuccessResponse>, |                     call: Call<SuccessResponse>, | ||||||
|                         response: Response<SuccessResponse> |                     response: Response<SuccessResponse> | ||||||
|                 ) { |                 ) { | ||||||
|                     if (response.body() != null && response.body()!!.isSuccess) { |                     if (response.body() != null && response.body()!!.isSuccess) { | ||||||
|                         finish() |                         finish() | ||||||
|                     } else { |                     } else { | ||||||
|                         Toast.makeText( |                         Toast.makeText( | ||||||
|                                 this@AddSourceActivity, |                             this@AddSourceActivity, | ||||||
|                                 R.string.cant_create_source, |                             R.string.cant_create_source, | ||||||
|                                 Toast.LENGTH_SHORT |                             Toast.LENGTH_SHORT | ||||||
|                         ).show() |                         ).show() | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { |                 override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { | ||||||
|                     Toast.makeText( |                     Toast.makeText( | ||||||
|                             this@AddSourceActivity, |                         this@AddSourceActivity, | ||||||
|                             R.string.cant_create_source, |                         R.string.cant_create_source, | ||||||
|                             Toast.LENGTH_SHORT |                         Toast.LENGTH_SHORT | ||||||
|                     ).show() |                     ).show() | ||||||
|                 } |                 } | ||||||
|             }) |             }) | ||||||
|   | |||||||
| @@ -0,0 +1,52 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss | ||||||
|  |  | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.MenuItem | ||||||
|  | import androidx.appcompat.app.AppCompatActivity | ||||||
|  | import androidx.fragment.app.FragmentManager | ||||||
|  | import androidx.fragment.app.FragmentStatePagerAdapter | ||||||
|  | import apps.amine.bou.readerforselfoss.fragments.ImageFragment | ||||||
|  | import kotlinx.android.synthetic.main.activity_reader.* | ||||||
|  |  | ||||||
|  | class ImageActivity : AppCompatActivity() { | ||||||
|  |     private lateinit var allImages : ArrayList<String> | ||||||
|  |     private var position : Int = 0 | ||||||
|  |  | ||||||
|  |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |  | ||||||
|  |         setContentView(R.layout.activity_image) | ||||||
|  |  | ||||||
|  |         setSupportActionBar(toolBar) | ||||||
|  |         supportActionBar?.setDisplayShowTitleEnabled(false) | ||||||
|  |         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||||
|  |  | ||||||
|  |         allImages = intent.getStringArrayListExtra("allImages") | ||||||
|  |         position = intent.getIntExtra("position", 0) | ||||||
|  |  | ||||||
|  |         pager.adapter = ScreenSlidePagerAdapter(supportFragmentManager) | ||||||
|  |         pager.currentItem = position | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|  |         when (item.itemId) { | ||||||
|  |             android.R.id.home -> { | ||||||
|  |                 onBackPressed() | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return super.onOptionsItemSelected(item) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private inner class ScreenSlidePagerAdapter(fm: FragmentManager) : FragmentStatePagerAdapter(fm, FragmentStatePagerAdapter.BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) { | ||||||
|  |  | ||||||
|  |         override fun getCount(): Int { | ||||||
|  |             return allImages.size | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         override fun getItem(position: Int): ImageFragment { | ||||||
|  |             return ImageFragment.newInstance(allImages[position]) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,70 +0,0 @@ | |||||||
| package apps.amine.bou.readerforselfoss |  | ||||||
|  |  | ||||||
| import agency.tango.materialintroscreen.MaterialIntroActivity |  | ||||||
| import agency.tango.materialintroscreen.MessageButtonBehaviour |  | ||||||
| import agency.tango.materialintroscreen.SlideFragmentBuilder |  | ||||||
| import android.content.Intent |  | ||||||
| import android.net.Uri |  | ||||||
| import android.os.Bundle |  | ||||||
| import android.preference.PreferenceManager |  | ||||||
| import android.support.v7.app.AppCompatDelegate |  | ||||||
| import android.view.View |  | ||||||
|  |  | ||||||
| class IntroActivity : MaterialIntroActivity() { |  | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |  | ||||||
|         super.onCreate(savedInstanceState) |  | ||||||
|  |  | ||||||
|         AppCompatDelegate.setCompatVectorFromResourcesEnabled(true) |  | ||||||
|  |  | ||||||
|         addSlide( |  | ||||||
|                 SlideFragmentBuilder() |  | ||||||
|                         .backgroundColor(R.color.colorPrimary) |  | ||||||
|                         .buttonsColor(R.color.colorAccent) |  | ||||||
|                         .image(R.drawable.web_hi_res_512) |  | ||||||
|                         .title(getString(R.string.intro_hello_title)) |  | ||||||
|                         .description(getString(R.string.intro_hello_message)) |  | ||||||
|                         .build() |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         addSlide( |  | ||||||
|                 SlideFragmentBuilder() |  | ||||||
|                         .backgroundColor(R.color.colorAccent) |  | ||||||
|                         .buttonsColor(R.color.colorPrimary) |  | ||||||
|                         .image(R.drawable.ic_info_outline_white_48px) |  | ||||||
|                         .title(getString(R.string.intro_needs_selfoss_title)) |  | ||||||
|                         .description(getString(R.string.intro_needs_selfoss_message)) |  | ||||||
|                         .build(), |  | ||||||
|                 MessageButtonBehaviour( |  | ||||||
|                         View.OnClickListener { |  | ||||||
|                             val browserIntent = Intent( |  | ||||||
|                                     Intent.ACTION_VIEW, |  | ||||||
|                                     Uri.parse("https://selfoss.aditu.de") |  | ||||||
|                             ) |  | ||||||
|                             startActivity(browserIntent) |  | ||||||
|                         }, getString(R.string.intro_needs_selfoss_link) |  | ||||||
|                 ) |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         addSlide( |  | ||||||
|                 SlideFragmentBuilder() |  | ||||||
|                         .backgroundColor(R.color.colorPrimaryDark) |  | ||||||
|                         .buttonsColor(R.color.colorAccentDark) |  | ||||||
|                         .image(R.drawable.ic_thumb_up_white_48px) |  | ||||||
|                         .title(getString(R.string.intro_all_set_title)) |  | ||||||
|                         .description(getString(R.string.intro_all_set_message)) |  | ||||||
|                         .build() |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun onFinish() { |  | ||||||
|         super.onFinish() |  | ||||||
|         val getPrefs = PreferenceManager.getDefaultSharedPreferences(baseContext) |  | ||||||
|         val e = getPrefs.edit() |  | ||||||
|         e.putBoolean("firstStart", false) |  | ||||||
|         e.apply() |  | ||||||
|         val intent = Intent(this, LoginActivity::class.java) |  | ||||||
|         startActivity(intent) |  | ||||||
|         finish() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -6,8 +6,8 @@ import android.content.Context | |||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.content.SharedPreferences | import android.content.SharedPreferences | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.support.v7.app.AlertDialog | import androidx.appcompat.app.AlertDialog | ||||||
| import android.support.v7.app.AppCompatActivity | import androidx.appcompat.app.AppCompatActivity | ||||||
| import android.text.TextUtils | import android.text.TextUtils | ||||||
| import android.view.Menu | import android.view.Menu | ||||||
| import android.view.MenuItem | import android.view.MenuItem | ||||||
| @@ -17,11 +17,10 @@ import android.widget.TextView | |||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import apps.amine.bou.readerforselfoss.utils.isBaseUrlValid | import apps.amine.bou.readerforselfoss.utils.isBaseUrlValid | ||||||
| import com.crashlytics.android.Crashlytics | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import com.ftinc.scoop.Scoop |  | ||||||
| import com.google.firebase.analytics.FirebaseAnalytics |  | ||||||
| import com.mikepenz.aboutlibraries.Libs | import com.mikepenz.aboutlibraries.Libs | ||||||
| import com.mikepenz.aboutlibraries.LibsBuilder | import com.mikepenz.aboutlibraries.LibsBuilder | ||||||
| import kotlinx.android.synthetic.main.activity_login.* | import kotlinx.android.synthetic.main.activity_login.* | ||||||
| @@ -38,32 +37,29 @@ class LoginActivity : AppCompatActivity() { | |||||||
|  |  | ||||||
|     private lateinit var settings: SharedPreferences |     private lateinit var settings: SharedPreferences | ||||||
|     private lateinit var editor: SharedPreferences.Editor |     private lateinit var editor: SharedPreferences.Editor | ||||||
|     private lateinit var firebaseAnalytics: FirebaseAnalytics |  | ||||||
|     private lateinit var userIdentifier: String |     private lateinit var userIdentifier: String | ||||||
|     private var logErrors: Boolean = false |     private lateinit var appColors: AppColors | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         appColors = AppColors(this@LoginActivity) | ||||||
|  |  | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|         Scoop.getInstance().apply(this) |  | ||||||
|         setContentView(R.layout.activity_login) |         setContentView(R.layout.activity_login) | ||||||
|  |  | ||||||
|         setSupportActionBar(toolbar) |         setSupportActionBar(toolbar) | ||||||
|  |  | ||||||
|         handleBaseUrlFail() |         handleBaseUrlFail() | ||||||
|  |  | ||||||
|  |  | ||||||
|         settings = getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) |         settings = getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) | ||||||
|         userIdentifier = settings.getString("unique_id", "") |         userIdentifier = settings.getString("unique_id", "")!! | ||||||
|         logErrors = settings.getBoolean("login_debug", false) |  | ||||||
|  |  | ||||||
|         editor = settings.edit() |         editor = settings.edit() | ||||||
|  |  | ||||||
|         if (settings.getString("url", "").isNotEmpty()) { |         if (settings.getString("url", "")!!.isNotEmpty()) { | ||||||
|             goToMain() |             goToMain() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         firebaseAnalytics = FirebaseAnalytics.getInstance(this) |  | ||||||
|  |  | ||||||
|         handleActions() |         handleActions() | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -77,13 +73,13 @@ class LoginActivity : AppCompatActivity() { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         passwordView.setOnEditorActionListener( |         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) { | ||||||
|                         attemptLogin() |                     attemptLogin() | ||||||
|                         return@OnEditorActionListener true |                     return@OnEditorActionListener true | ||||||
|                     } |  | ||||||
|                     false |  | ||||||
|                 } |                 } | ||||||
|  |                 false | ||||||
|  |             } | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         signInButton.setOnClickListener { attemptLogin() } |         signInButton.setOnClickListener { attemptLogin() } | ||||||
| @@ -111,9 +107,9 @@ class LoginActivity : AppCompatActivity() { | |||||||
|             alertDialog.setTitle(getString(R.string.warning_wrong_url)) |             alertDialog.setTitle(getString(R.string.warning_wrong_url)) | ||||||
|             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() | ||||||
|         } |         } | ||||||
| @@ -144,7 +140,7 @@ class LoginActivity : AppCompatActivity() { | |||||||
|         var cancel = false |         var cancel = false | ||||||
|         var focusView: View? = null |         var focusView: View? = null | ||||||
|  |  | ||||||
|         if (!url.isBaseUrlValid()) { |         if (!url.isBaseUrlValid(this@LoginActivity)) { | ||||||
|             urlView.error = getString(R.string.login_url_problem) |             urlView.error = getString(R.string.login_url_problem) | ||||||
|             focusView = urlView |             focusView = urlView | ||||||
|             cancel = true |             cancel = true | ||||||
| @@ -154,16 +150,16 @@ class LoginActivity : AppCompatActivity() { | |||||||
|                 alertDialog.setTitle(getString(R.string.warning_wrong_url)) |                 alertDialog.setTitle(getString(R.string.warning_wrong_url)) | ||||||
|                 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 | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (isWithLogin || isWithHTTPLogin) { |         if (isWithLogin) { | ||||||
|             if (TextUtils.isEmpty(password)) { |             if (TextUtils.isEmpty(password)) { | ||||||
|                 passwordView.error = getString(R.string.error_invalid_password) |                 passwordView.error = getString(R.string.error_invalid_password) | ||||||
|                 focusView = passwordView |                 focusView = passwordView | ||||||
| @@ -177,6 +173,20 @@ class LoginActivity : AppCompatActivity() { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (isWithHTTPLogin) { | ||||||
|  |             if (TextUtils.isEmpty(httpPassword)) { | ||||||
|  |                 httpPasswordView.error = getString(R.string.error_invalid_password) | ||||||
|  |                 focusView = httpPasswordView | ||||||
|  |                 cancel = true | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (TextUtils.isEmpty(httpLogin)) { | ||||||
|  |                 httpLoginView.error = getString(R.string.error_field_required) | ||||||
|  |                 focusView = httpLoginView | ||||||
|  |                 cancel = true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (cancel) { |         if (cancel) { | ||||||
|             focusView?.requestFocus() |             focusView?.requestFocus() | ||||||
|         } else { |         } else { | ||||||
| @@ -191,53 +201,47 @@ class LoginActivity : AppCompatActivity() { | |||||||
|             editor.apply() |             editor.apply() | ||||||
|  |  | ||||||
|             val api = SelfossApi( |             val api = SelfossApi( | ||||||
|                     this, |                 this, | ||||||
|                     this@LoginActivity, |                 this@LoginActivity, | ||||||
|                     isWithSelfSignedCert, |                 isWithSelfSignedCert, | ||||||
|                     isWithSelfSignedCert |                 -1L | ||||||
|             ) |             ) | ||||||
|             api.login().enqueue(object : Callback<SuccessResponse> { |  | ||||||
|                 private fun preferenceError(t: Throwable) { |  | ||||||
|                     editor.remove("url") |  | ||||||
|                     editor.remove("login") |  | ||||||
|                     editor.remove("httpUserName") |  | ||||||
|                     editor.remove("password") |  | ||||||
|                     editor.remove("httpPassword") |  | ||||||
|                     editor.apply() |  | ||||||
|                     urlView.error = getString(R.string.wrong_infos) |  | ||||||
|                     loginView.error = getString(R.string.wrong_infos) |  | ||||||
|                     passwordView.error = getString(R.string.wrong_infos) |  | ||||||
|                     httpLoginView.error = getString(R.string.wrong_infos) |  | ||||||
|                     httpPasswordView.error = getString(R.string.wrong_infos) |  | ||||||
|                     if (logErrors) { |  | ||||||
|                         Crashlytics.setUserIdentifier(userIdentifier) |  | ||||||
|                         Crashlytics.log(100, "LOGIN_DEBUG_ERRROR", t.message) |  | ||||||
|                         Crashlytics.logException(t) |  | ||||||
|                         Toast.makeText( |  | ||||||
|                                 this@LoginActivity, |  | ||||||
|                                 t.message, |  | ||||||
|                                 Toast.LENGTH_LONG |  | ||||||
|                         ).show() |  | ||||||
|                     } |  | ||||||
|                     showProgress(false) |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 override fun onResponse( |             if (this@LoginActivity.isNetworkAccessible(this@LoginActivity.findViewById(R.id.loginForm))) { | ||||||
|  |                 api.login().enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                     private fun preferenceError(t: Throwable) { | ||||||
|  |                         editor.remove("url") | ||||||
|  |                         editor.remove("login") | ||||||
|  |                         editor.remove("httpUserName") | ||||||
|  |                         editor.remove("password") | ||||||
|  |                         editor.remove("httpPassword") | ||||||
|  |                         editor.apply() | ||||||
|  |                         urlView.error = getString(R.string.wrong_infos) | ||||||
|  |                         loginView.error = getString(R.string.wrong_infos) | ||||||
|  |                         passwordView.error = getString(R.string.wrong_infos) | ||||||
|  |                         httpLoginView.error = getString(R.string.wrong_infos) | ||||||
|  |                         httpPasswordView.error = getString(R.string.wrong_infos) | ||||||
|  |                         showProgress(false) | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     override fun onResponse( | ||||||
|                         call: Call<SuccessResponse>, |                         call: Call<SuccessResponse>, | ||||||
|                         response: Response<SuccessResponse> |                         response: Response<SuccessResponse> | ||||||
|                 ) { |                     ) { | ||||||
|                     if (response.body() != null && response.body()!!.isSuccess) { |                         if (response.body() != null && response.body()!!.isSuccess) { | ||||||
|                         firebaseAnalytics.logEvent(FirebaseAnalytics.Event.LOGIN, Bundle()) |                             goToMain() | ||||||
|                         goToMain() |                         } else { | ||||||
|                     } else { |                             preferenceError(Exception("No response body...")) | ||||||
|                         preferenceError(Exception("No response body...")) |                         } | ||||||
|                     } |                     } | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { |                     override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { | ||||||
|                     preferenceError(t) |                         preferenceError(t) | ||||||
|                 } |                     } | ||||||
|             }) |                 }) | ||||||
|  |             } else { | ||||||
|  |                 showProgress(false) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -246,56 +250,46 @@ class LoginActivity : AppCompatActivity() { | |||||||
|  |  | ||||||
|         loginForm.visibility = if (show) View.GONE else View.VISIBLE |         loginForm.visibility = if (show) View.GONE else View.VISIBLE | ||||||
|         loginForm |         loginForm | ||||||
|                 .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) { | ||||||
|                         loginForm.visibility = if (show) View.GONE else View.VISIBLE |                 loginForm.visibility = if (show) View.GONE else View.VISIBLE | ||||||
|                     } |             } | ||||||
|                 } |         } | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         loginProgress.visibility = if (show) View.VISIBLE else View.GONE |         loginProgress.visibility = if (show) View.VISIBLE else View.GONE | ||||||
|         loginProgress |         loginProgress | ||||||
|                 .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) { | ||||||
|                         loginProgress.visibility = if (show) View.VISIBLE else View.GONE |                 loginProgress.visibility = if (show) View.VISIBLE else View.GONE | ||||||
|                     } |             } | ||||||
|                 } |         } | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onCreateOptionsMenu(menu: Menu): Boolean { |     override fun onCreateOptionsMenu(menu: Menu): Boolean { | ||||||
|         menuInflater.inflate(R.menu.login_menu, menu) |         menuInflater.inflate(R.menu.login_menu, menu) | ||||||
|         menu.findItem(R.id.login_debug).isChecked = logErrors |  | ||||||
|         return true |         return true | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|         when (item.itemId) { |         return when (item.itemId) { | ||||||
|             R.id.about -> { |             R.id.about -> { | ||||||
|                 LibsBuilder() |                 LibsBuilder() | ||||||
|                         .withActivityStyle(Libs.ActivityStyle.LIGHT_DARK_TOOLBAR) |                     .withAboutIconShown(true) | ||||||
|                         .withAboutIconShown(true) |                     .withAboutVersionShown(true) | ||||||
|                         .withAboutVersionShown(true) |                     .start(this) | ||||||
|                         .start(this) |                 true | ||||||
|                 return true |  | ||||||
|             } |             } | ||||||
|             R.id.login_debug -> { |             else -> super.onOptionsItemSelected(item) | ||||||
|                 val newState = !item.isChecked |  | ||||||
|                 item.isChecked = newState |  | ||||||
|                 logErrors = newState |  | ||||||
|                 editor.putBoolean("login_debug", newState) |  | ||||||
|                 editor.apply() |  | ||||||
|                 return true |  | ||||||
|             } |  | ||||||
|             else -> return super.onOptionsItemSelected(item) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ package apps.amine.bou.readerforselfoss | |||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.preference.PreferenceManager | import android.preference.PreferenceManager | ||||||
| import android.support.v7.app.AppCompatActivity | import androidx.appcompat.app.AppCompatActivity | ||||||
|  |  | ||||||
| class MainActivity : AppCompatActivity() { | class MainActivity : AppCompatActivity() { | ||||||
|  |  | ||||||
| @@ -11,17 +11,9 @@ class MainActivity : AppCompatActivity() { | |||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|         setContentView(R.layout.activity_main) |         setContentView(R.layout.activity_main) | ||||||
|  |  | ||||||
|         if (PreferenceManager.getDefaultSharedPreferences(baseContext).getBoolean( |         val intent = Intent(this, LoginActivity::class.java) | ||||||
|                 "firstStart", |  | ||||||
|                 true |  | ||||||
|         )) { |  | ||||||
|             val i = Intent(this@MainActivity, IntroActivity::class.java) |  | ||||||
|             startActivity(i) |  | ||||||
|         } else { |  | ||||||
|             val intent = Intent(this, LoginActivity::class.java) |  | ||||||
|             startActivity(intent) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |         startActivity(intent) | ||||||
|         finish() |         finish() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,38 +1,32 @@ | |||||||
| package apps.amine.bou.readerforselfoss | package apps.amine.bou.readerforselfoss | ||||||
|  |  | ||||||
|  | import android.app.NotificationChannel | ||||||
|  | import android.app.NotificationManager | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.graphics.drawable.Drawable | import android.graphics.drawable.Drawable | ||||||
| import android.net.Uri | import android.net.Uri | ||||||
|  | import android.os.Build | ||||||
| import android.preference.PreferenceManager | import android.preference.PreferenceManager | ||||||
| import android.support.multidex.MultiDexApplication |  | ||||||
| import android.widget.ImageView | import android.widget.ImageView | ||||||
|  | import androidx.multidex.MultiDexApplication | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import com.anupcowkur.reservoir.Reservoir | import apps.amine.bou.readerforselfoss.utils.glide.loadMaybeBasicAuth | ||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
| import com.crashlytics.android.Crashlytics |  | ||||||
| import com.ftinc.scoop.Scoop | import com.ftinc.scoop.Scoop | ||||||
| import com.github.stkent.amplify.feedback.DefaultEmailFeedbackCollector |  | ||||||
| import com.github.stkent.amplify.feedback.GooglePlayStoreFeedbackCollector |  | ||||||
| import com.github.stkent.amplify.tracking.Amplify |  | ||||||
| import com.mikepenz.materialdrawer.util.AbstractDrawerImageLoader | import com.mikepenz.materialdrawer.util.AbstractDrawerImageLoader | ||||||
| import com.mikepenz.materialdrawer.util.DrawerImageLoader | import com.mikepenz.materialdrawer.util.DrawerImageLoader | ||||||
| import io.fabric.sdk.android.Fabric |  | ||||||
| import java.io.IOException |  | ||||||
| import java.util.UUID.randomUUID | import java.util.UUID.randomUUID | ||||||
|  |  | ||||||
| class MyApp : MultiDexApplication() { | class MyApp : MultiDexApplication() { | ||||||
|  |     private lateinit var config: Config | ||||||
|  |  | ||||||
|     override fun onCreate() { |     override fun onCreate() { | ||||||
|         super.onCreate() |         super.onCreate() | ||||||
|         Fabric.with(this, Crashlytics()) |         config = Config(baseContext) | ||||||
|  |  | ||||||
|         initAmplify() |  | ||||||
|  |  | ||||||
|         initCache() |  | ||||||
|  |  | ||||||
|         val prefs = getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) |         val prefs = getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) | ||||||
|         if (prefs.getString("unique_id", "").isEmpty()) { |         if (prefs.getString("unique_id", "")!!.isEmpty()) { | ||||||
|             val editor = prefs.edit() |             val editor = prefs.edit() | ||||||
|             editor.putString("unique_id", randomUUID().toString()) |             editor.putString("unique_id", randomUUID().toString()) | ||||||
|             editor.apply() |             editor.apply() | ||||||
| @@ -43,35 +37,39 @@ class MyApp : MultiDexApplication() { | |||||||
|         initTheme() |         initTheme() | ||||||
|  |  | ||||||
|         tryToHandleBug() |         tryToHandleBug() | ||||||
|  |  | ||||||
|  |         handleNotificationChannels() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun initAmplify() { |     private fun handleNotificationChannels() { | ||||||
|         Amplify.initSharedInstance(this) |         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { | ||||||
|                 .setPositiveFeedbackCollectors(GooglePlayStoreFeedbackCollector()) |             val notificationManager = getSystemService(NOTIFICATION_SERVICE) as NotificationManager | ||||||
|                 .setCriticalFeedbackCollectors(DefaultEmailFeedbackCollector(BuildConfig.FEEDBACK_EMAIL)) |  | ||||||
|                 .applyAllDefaultRules() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun initCache() { |             val name = getString(R.string.notification_channel_sync) | ||||||
|         try { |             val importance = NotificationManager.IMPORTANCE_LOW | ||||||
|             Reservoir.init(this, 8192) //in bytes |             val mChannel = NotificationChannel(Config.syncChannelId, name, importance) | ||||||
|         } catch (e: IOException) { |  | ||||||
|             //failure |             val newItemsChannelname = getString(R.string.new_items_channel_sync) | ||||||
|  |             val newItemsChannelimportance = NotificationManager.IMPORTANCE_DEFAULT | ||||||
|  |             val newItemsChannelmChannel = NotificationChannel(Config.newItemsChannelId, newItemsChannelname, newItemsChannelimportance) | ||||||
|  |  | ||||||
|  |             notificationManager.createNotificationChannel(mChannel) | ||||||
|  |             notificationManager.createNotificationChannel(newItemsChannelmChannel) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun initDrawerImageLoader() { |     private fun initDrawerImageLoader() { | ||||||
|         DrawerImageLoader.init(object : AbstractDrawerImageLoader() { |         DrawerImageLoader.init(object : AbstractDrawerImageLoader() { | ||||||
|             override fun set( |             override fun set( | ||||||
|                     imageView: ImageView?, |                 imageView: ImageView?, | ||||||
|                     uri: Uri?, |                 uri: Uri?, | ||||||
|                     placeholder: Drawable?, |                 placeholder: Drawable?, | ||||||
|                     tag: String? |                 tag: String? | ||||||
|             ) { |             ) { | ||||||
|                 Glide.with(imageView?.context) |                 Glide.with(imageView?.context) | ||||||
|                         .load(uri) |                     .loadMaybeBasicAuth(config, uri.toString()) | ||||||
|                         .apply(RequestOptions.fitCenterTransform().placeholder(placeholder)) |                     .apply(RequestOptions.fitCenterTransform().placeholder(placeholder)) | ||||||
|                         .into(imageView) |                     .into(imageView) | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             override fun cancel(imageView: ImageView?) { |             override fun cancel(imageView: ImageView?) { | ||||||
| @@ -86,22 +84,10 @@ class MyApp : MultiDexApplication() { | |||||||
|  |  | ||||||
|     private fun initTheme() { |     private fun initTheme() { | ||||||
|         Scoop.waffleCone() |         Scoop.waffleCone() | ||||||
|                 .addFlavor(getString(R.string.default_theme), R.style.NoBar, true) |             .addFlavor(getString(R.string.default_theme), R.style.NoBar, true) | ||||||
|                 .addFlavor(getString(R.string.default_dark_theme), R.style.NoBarDark) |             .addFlavor(getString(R.string.default_dark_theme), R.style.NoBarDark, false) | ||||||
|                 .addFlavor(getString(R.string.teal_orange_theme), R.style.NoBarTealOrange) |             .setSharedPreferences(PreferenceManager.getDefaultSharedPreferences(this)) | ||||||
|                 .addFlavor(getString(R.string.teal_orange_dark_theme), R.style.NoBarTealOrangeDark) |             .initialize() | ||||||
|                 .addFlavor(getString(R.string.cyan_pink_theme), R.style.NoBarCyanPink) |  | ||||||
|                 .addFlavor(getString(R.string.cyan_pink_dark_theme), R.style.NoBarCyanPinkDark) |  | ||||||
|                 .addFlavor(getString(R.string.grey_orange_theme), R.style.NoBarGreyOrange) |  | ||||||
|                 .addFlavor(getString(R.string.grey_orange_dark_theme), R.style.NoBarGreyOrangeDark) |  | ||||||
|                 .addFlavor(getString(R.string.blue_amber_theme), R.style.NoBarBlueAmber) |  | ||||||
|                 .addFlavor(getString(R.string.blue_amber_dark_theme), R.style.NoBarBlueAmberDark) |  | ||||||
|                 .addFlavor(getString(R.string.indigo_pink_theme), R.style.NoBarIndigoPink) |  | ||||||
|                 .addFlavor(getString(R.string.indigo_pink_dark_theme), R.style.NoBarIndigoPinkDark) |  | ||||||
|                 .addFlavor(getString(R.string.red_teal_theme), R.style.NoBarRedTeal) |  | ||||||
|                 .addFlavor(getString(R.string.red_teal_dark_theme), R.style.NoBarRedTealDark) |  | ||||||
|                 .setSharedPreferences(PreferenceManager.getDefaultSharedPreferences(this)) |  | ||||||
|                 .initialize() |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun tryToHandleBug() { |     private fun tryToHandleBug() { | ||||||
| @@ -109,8 +95,8 @@ class MyApp : MultiDexApplication() { | |||||||
|  |  | ||||||
|         Thread.setDefaultUncaughtExceptionHandler { thread, e -> |         Thread.setDefaultUncaughtExceptionHandler { thread, e -> | ||||||
|             if (e is java.lang.NoClassDefFoundError && e.stackTrace.asList().any { |             if (e is java.lang.NoClassDefFoundError && e.stackTrace.asList().any { | ||||||
|                 it.toString().contains("android.view.ViewDebug") |                     it.toString().contains("android.view.ViewDebug") | ||||||
|             }) { |                 }) { | ||||||
|                 Unit |                 Unit | ||||||
|             } else { |             } else { | ||||||
|                 oldHandler.uncaughtException(thread, e) |                 oldHandler.uncaughtException(thread, e) | ||||||
|   | |||||||
| @@ -1,209 +1,349 @@ | |||||||
| package apps.amine.bou.readerforselfoss | package apps.amine.bou.readerforselfoss | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
| import android.content.SharedPreferences | import android.content.SharedPreferences | ||||||
|  | import android.graphics.drawable.ColorDrawable | ||||||
|  | import android.os.Build | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.preference.PreferenceManager | import android.preference.PreferenceManager | ||||||
| import android.support.customtabs.CustomTabsIntent | import androidx.fragment.app.FragmentManager | ||||||
| import android.support.design.widget.FloatingActionButton | import androidx.fragment.app.FragmentStatePagerAdapter | ||||||
| import android.support.v4.widget.NestedScrollView | import androidx.core.content.ContextCompat | ||||||
| import android.support.v7.app.AppCompatActivity | import androidx.viewpager.widget.ViewPager | ||||||
| import android.text.Html | import androidx.appcompat.app.AppCompatActivity | ||||||
| import android.text.method.LinkMovementMethod | import android.view.Menu | ||||||
| import android.view.MenuItem | import android.view.MenuItem | ||||||
| import android.view.View | import android.view.ViewGroup | ||||||
| import apps.amine.bou.readerforselfoss.api.mercury.MercuryApi | import android.widget.Toast | ||||||
| import apps.amine.bou.readerforselfoss.api.mercury.ParsedContent | import androidx.fragment.app.Fragment | ||||||
| import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | import androidx.room.Room | ||||||
| import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
| import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.utils.openItemUrl | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
| import apps.amine.bou.readerforselfoss.utils.shareLink | import apps.amine.bou.readerforselfoss.fragments.ArticleFragment | ||||||
| import com.bumptech.glide.Glide | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
| import com.bumptech.glide.request.RequestOptions | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
| import com.crashlytics.android.Crashlytics | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_2_3 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_3_4 | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.Toppings | ||||||
|  | import apps.amine.bou.readerforselfoss.transformers.DepthPageTransformer | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.succeeded | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.toggleStar | ||||||
| import com.ftinc.scoop.Scoop | import com.ftinc.scoop.Scoop | ||||||
| import com.github.rubensousa.floatingtoolbar.FloatingToolbar |  | ||||||
| import kotlinx.android.synthetic.main.activity_reader.* | import kotlinx.android.synthetic.main.activity_reader.* | ||||||
| import org.sufficientlysecure.htmltextview.HtmlHttpImageGetter | import me.relex.circleindicator.CircleIndicator | ||||||
| import retrofit2.Call | import retrofit2.Call | ||||||
| import retrofit2.Callback | import retrofit2.Callback | ||||||
| import retrofit2.Response | import retrofit2.Response | ||||||
|  | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
| class ReaderActivity : AppCompatActivity() { | class ReaderActivity : AppCompatActivity() { | ||||||
|     private lateinit var mCustomTabActivityHelper: CustomTabActivityHelper |  | ||||||
|     //private lateinit var content: HtmlTextView |  | ||||||
|     private lateinit var url: String |  | ||||||
|     private lateinit var contentText: String |  | ||||||
|     private lateinit var contentSource: String |  | ||||||
|     private lateinit var contentImage: String |  | ||||||
|     private lateinit var contentTitle: String |  | ||||||
|     private lateinit var fab: FloatingActionButton |  | ||||||
|  |  | ||||||
|     override fun onStop() { |     private var markOnScroll: Boolean = false | ||||||
|         super.onStop() |     private var currentItem: Int = 0 | ||||||
|         mCustomTabActivityHelper.unbindCustomTabsService(this) |     private lateinit var userIdentifier: String | ||||||
|  |  | ||||||
|  |     private lateinit var api: SelfossApi | ||||||
|  |  | ||||||
|  |     private lateinit var toolbarMenu: Menu | ||||||
|  |  | ||||||
|  |     private lateinit var db: AppDatabase | ||||||
|  |     private lateinit var prefs: SharedPreferences | ||||||
|  |  | ||||||
|  |     private var activeAlignment: Int = 1 | ||||||
|  |     val JUSTIFY = 1 | ||||||
|  |     val ALIGN_LEFT = 2 | ||||||
|  |  | ||||||
|  |     private fun showMenuItem(willAddToFavorite: Boolean) { | ||||||
|  |         toolbarMenu.findItem(R.id.save).isVisible = willAddToFavorite | ||||||
|  |         toolbarMenu.findItem(R.id.unsave).isVisible = !willAddToFavorite | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private fun canFavorite() { | ||||||
|  |         showMenuItem(true) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun canRemoveFromFavorite() { | ||||||
|  |         showMenuItem(false) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private lateinit var editor: SharedPreferences.Editor | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|         Scoop.getInstance().apply(this) |  | ||||||
|         setContentView(R.layout.activity_reader) |         setContentView(R.layout.activity_reader) | ||||||
|  |  | ||||||
|         url = intent.getStringExtra("url") |         db = Room.databaseBuilder( | ||||||
|         contentText = intent.getStringExtra("content") |             applicationContext, | ||||||
|         contentTitle = intent.getStringExtra("title") |             AppDatabase::class.java, "selfoss-database" | ||||||
|         contentImage = intent.getStringExtra("image") |         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).addMigrations(MIGRATION_3_4).build() | ||||||
|         contentSource = intent.getStringExtra("source") |  | ||||||
|  |  | ||||||
|         fab = findViewById(R.id.fab) |         val scoop = Scoop.getInstance() | ||||||
|         val mFloatingToolbar: FloatingToolbar = findViewById(R.id.floatingToolbar) |         scoop.bind(this, Toppings.PRIMARY.value, toolBar) | ||||||
|         mFloatingToolbar.attachFab(fab) |         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { | ||||||
|  |             scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         val customTabsIntent = this@ReaderActivity.buildCustomTabsIntent() |         setSupportActionBar(toolBar) | ||||||
|         mCustomTabActivityHelper = CustomTabActivityHelper() |         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||||
|         mCustomTabActivityHelper.bindCustomTabsService(this) |         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||||
|  |  | ||||||
|         val prefs = PreferenceManager.getDefaultSharedPreferences(this) |         val settings = | ||||||
|  |             getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) | ||||||
|  |  | ||||||
|         mFloatingToolbar.setClickListener(object : FloatingToolbar.ItemClickListener { |         prefs = PreferenceManager.getDefaultSharedPreferences(this) | ||||||
|             override fun onItemClick(item: MenuItem) { |         editor = prefs.edit() | ||||||
|                 when (item.itemId) { |  | ||||||
|                     R.id.more_action -> getContentFromMercury(customTabsIntent, prefs) |         userIdentifier = prefs.getString("unique_id", "")!! | ||||||
|                     R.id.share_action -> this@ReaderActivity.shareLink(url) |         markOnScroll = prefs.getBoolean("mark_on_scroll", false) | ||||||
|                     R.id.open_action -> this@ReaderActivity.openItemUrl( |         activeAlignment = prefs.getInt("text_align", JUSTIFY) | ||||||
|                             url, |  | ||||||
|                             contentText, |         api = SelfossApi( | ||||||
|                             contentImage, |             this, | ||||||
|                             contentTitle, |             this@ReaderActivity, | ||||||
|                             contentSource, |             settings.getBoolean("isSelfSignedCert", false), | ||||||
|                             customTabsIntent, |             prefs.getString("api_timeout", "-1")!!.toLong() | ||||||
|                             false, |         ) | ||||||
|                             false, |  | ||||||
|                             this@ReaderActivity |         if (allItems.isEmpty()) { | ||||||
|                     ) |             finish() | ||||||
|                     else -> Unit |         } | ||||||
|  |  | ||||||
|  |         currentItem = intent.getIntExtra("currentItem", 0) | ||||||
|  |  | ||||||
|  |         readItem(allItems[currentItem]) | ||||||
|  |  | ||||||
|  |         pager.adapter = | ||||||
|  |                 ScreenSlidePagerAdapter(supportFragmentManager, AppColors(this@ReaderActivity)) | ||||||
|  |         pager.currentItem = currentItem | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onResume() { | ||||||
|  |         super.onResume() | ||||||
|  |  | ||||||
|  |         notifyAdapter() | ||||||
|  |  | ||||||
|  |         pager.setPageTransformer(true, DepthPageTransformer()) | ||||||
|  |         (indicator as CircleIndicator).setViewPager(pager) | ||||||
|  |  | ||||||
|  |         pager.addOnPageChangeListener( | ||||||
|  |             object : ViewPager.SimpleOnPageChangeListener() { | ||||||
|  |  | ||||||
|  |                 override fun onPageSelected(position: Int) { | ||||||
|  |  | ||||||
|  |                     if (allItems[position].starred) { | ||||||
|  |                         canRemoveFromFavorite() | ||||||
|  |                     } else { | ||||||
|  |                         canFavorite() | ||||||
|  |                     } | ||||||
|  |                     readItem(allItems[pager.currentItem]) | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|             override fun onItemLongClick(item: MenuItem?) { |     fun readItem(item: Item) { | ||||||
|  |         if (markOnScroll) { | ||||||
|  |             thread { | ||||||
|  |                 db.itemsDao().delete(item.toEntity()) | ||||||
|             } |             } | ||||||
|         }) |             if (this@ReaderActivity.isNetworkAccessible(this@ReaderActivity.findViewById(R.id.reader_activity_view))) { | ||||||
|  |                 api.markItem(item.id).enqueue( | ||||||
|  |                     object : Callback<SuccessResponse> { | ||||||
|  |                         override fun onResponse( | ||||||
|  |                             call: Call<SuccessResponse>, | ||||||
|  |                             response: Response<SuccessResponse> | ||||||
|  |                         ) { | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         override fun onFailure( | ||||||
|         if (contentText.isEmptyOrNullOrNullString()) { |                             call: Call<SuccessResponse>, | ||||||
|             getContentFromMercury(customTabsIntent, prefs) |                             t: Throwable | ||||||
|         } else { |                         ) { | ||||||
|             source.text = contentSource |                             thread { | ||||||
|             titleView.text = contentTitle |                                 db.itemsDao().insertAllItems(item.toEntity()) | ||||||
|             tryToHandleHtml(contentText, customTabsIntent, prefs) |                             } | ||||||
|  |                         } | ||||||
|             if (!contentImage.isEmptyOrNullOrNullString()) { |                     } | ||||||
|                 imageView.visibility = View.VISIBLE |                 ) | ||||||
|                 Glide |  | ||||||
|                         .with(baseContext) |  | ||||||
|                         .asBitmap() |  | ||||||
|                         .load(contentImage) |  | ||||||
|                         .apply(RequestOptions.fitCenterTransform()) |  | ||||||
|                         .into(imageView) |  | ||||||
|             } else { |             } else { | ||||||
|                 imageView.visibility = View.GONE |                 thread { | ||||||
|  |                     db.actionsDao().insertAllActions(ActionEntity(item.id, true, false, false, false)) | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         nestedScrollView.setOnScrollChangeListener( |  | ||||||
|                 NestedScrollView.OnScrollChangeListener { _, _, scrollY, _, oldScrollY -> |  | ||||||
|                     if (scrollY > oldScrollY) { |  | ||||||
|                         fab.hide() |  | ||||||
|                     } else { |  | ||||||
|                         if (mFloatingToolbar.isShowing) mFloatingToolbar.hide() else fab.show() |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         content.movementMethod = LinkMovementMethod.getInstance() |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun getContentFromMercury( |     private fun notifyAdapter() { | ||||||
|             customTabsIntent: CustomTabsIntent, |         (pager.adapter as ScreenSlidePagerAdapter).notifyDataSetChanged() | ||||||
|             prefs: SharedPreferences |     } | ||||||
|     ) { |  | ||||||
|         progressBar.visibility = View.VISIBLE |  | ||||||
|         val parser = MercuryApi( |  | ||||||
|                 BuildConfig.MERCURY_KEY, |  | ||||||
|                 prefs.getBoolean("should_log_everything", false) |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         parser.parseUrl(url).enqueue(object : Callback<ParsedContent> { |     override fun onPause() { | ||||||
|             override fun onResponse( |         super.onPause() | ||||||
|                     call: Call<ParsedContent>, |         if (markOnScroll) { | ||||||
|                     response: Response<ParsedContent> |             pager.clearOnPageChangeListeners() | ||||||
|             ) { |         } | ||||||
|                 if (response.body() != null && response.body()!!.content != null && response.body()!!.content.isNotEmpty()) { |     } | ||||||
|                     source.text = response.body()!!.domain |  | ||||||
|                     titleView.text = response.body()!!.title |  | ||||||
|                     this@ReaderActivity.url = response.body()!!.url |  | ||||||
|  |  | ||||||
|                     if (response.body()!!.content != null && !response.body()!!.content.isEmpty()) { |     override fun onSaveInstanceState(oldInstanceState: Bundle) { | ||||||
|                         tryToHandleHtml(response.body()!!.content, customTabsIntent, prefs) |         super.onSaveInstanceState(oldInstanceState) | ||||||
|                     } |         oldInstanceState!!.clear() | ||||||
|  |     } | ||||||
|  |  | ||||||
|                     if (response.body()!!.lead_image_url != null && !response.body()!!.lead_image_url.isEmpty()) { |     private inner class ScreenSlidePagerAdapter(fm: FragmentManager, val appColors: AppColors) : | ||||||
|                         imageView.visibility = View.VISIBLE |         FragmentStatePagerAdapter(fm) { | ||||||
|                         Glide |  | ||||||
|                                 .with(baseContext) |  | ||||||
|                                 .asBitmap() |  | ||||||
|                                 .load(response.body()!!.lead_image_url) |  | ||||||
|                                 .apply(RequestOptions.fitCenterTransform()) |  | ||||||
|                                 .into(imageView) |  | ||||||
|                     } else { |  | ||||||
|                         imageView.visibility = View.GONE |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     nestedScrollView.scrollTo(0, 0) |         override fun getCount(): Int { | ||||||
|  |             return allItems.size | ||||||
|  |         } | ||||||
|  |  | ||||||
|                     progressBar.visibility = View.GONE |         override fun getItem(position: Int): ArticleFragment { | ||||||
|  |             return ArticleFragment.newInstance(position, allItems) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         override fun startUpdate(container: ViewGroup) { | ||||||
|  |             super.startUpdate(container) | ||||||
|  |  | ||||||
|  |             container.background = ColorDrawable( | ||||||
|  |                 ContextCompat.getColor( | ||||||
|  |                     this@ReaderActivity, | ||||||
|  |                     appColors.colorBackground | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun alignmentMenu(showJustify: Boolean) { | ||||||
|  |         toolbarMenu.findItem(R.id.align_left).isVisible = !showJustify | ||||||
|  |         toolbarMenu.findItem(R.id.align_justify).isVisible = showJustify | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onCreateOptionsMenu(menu: Menu): Boolean { | ||||||
|  |         val inflater = menuInflater | ||||||
|  |         inflater.inflate(R.menu.reader_menu, menu) | ||||||
|  |         toolbarMenu = menu | ||||||
|  |  | ||||||
|  |         if (!allItems.isEmpty() && allItems[currentItem].starred) { | ||||||
|  |             canRemoveFromFavorite() | ||||||
|  |         } else { | ||||||
|  |             canFavorite() | ||||||
|  |         } | ||||||
|  |         if (activeAlignment == JUSTIFY) { | ||||||
|  |             alignmentMenu(false) | ||||||
|  |         } else { | ||||||
|  |             alignmentMenu(true) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|  |         fun afterSave() { | ||||||
|  |             allItems[pager.currentItem] = | ||||||
|  |                     allItems[pager.currentItem].toggleStar() | ||||||
|  |             notifyAdapter() | ||||||
|  |             canRemoveFromFavorite() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         fun afterUnsave() { | ||||||
|  |             allItems[pager.currentItem] = allItems[pager.currentItem].toggleStar() | ||||||
|  |             notifyAdapter() | ||||||
|  |             canFavorite() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         when (item.itemId) { | ||||||
|  |             android.R.id.home -> { | ||||||
|  |                 onBackPressed() | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |             R.id.save -> { | ||||||
|  |                 if (this@ReaderActivity.isNetworkAccessible(null)) { | ||||||
|  |                     api.starrItem(allItems[pager.currentItem].id) | ||||||
|  |                         .enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                             override fun onResponse( | ||||||
|  |                                 call: Call<SuccessResponse>, | ||||||
|  |                                 response: Response<SuccessResponse> | ||||||
|  |                             ) { | ||||||
|  |                                 afterSave() | ||||||
|  |                             } | ||||||
|  |  | ||||||
|  |                             override fun onFailure( | ||||||
|  |                                 call: Call<SuccessResponse>, | ||||||
|  |                                 t: Throwable | ||||||
|  |                             ) { | ||||||
|  |                                 Toast.makeText( | ||||||
|  |                                     baseContext, | ||||||
|  |                                     R.string.cant_mark_favortie, | ||||||
|  |                                     Toast.LENGTH_SHORT | ||||||
|  |                                 ).show() | ||||||
|  |                             } | ||||||
|  |                         }) | ||||||
|                 } else { |                 } else { | ||||||
|                     openInBrowserAfterFailing(customTabsIntent) |                     thread { | ||||||
|  |                         db.actionsDao().insertAllActions(ActionEntity(allItems[pager.currentItem].id, false, false, true, false)) | ||||||
|  |                         afterSave() | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |             R.id.unsave -> { | ||||||
|  |                 if (this@ReaderActivity.isNetworkAccessible(null)) { | ||||||
|  |                     api.unstarrItem(allItems[pager.currentItem].id) | ||||||
|  |                         .enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                             override fun onResponse( | ||||||
|  |                                 call: Call<SuccessResponse>, | ||||||
|  |                                 response: Response<SuccessResponse> | ||||||
|  |                             ) { | ||||||
|  |                                 afterUnsave() | ||||||
|  |                             } | ||||||
|  |  | ||||||
|             override fun onFailure( |                             override fun onFailure( | ||||||
|                     call: Call<ParsedContent>, |                                 call: Call<SuccessResponse>, | ||||||
|                     t: Throwable |                                 t: Throwable | ||||||
|             ) = openInBrowserAfterFailing(customTabsIntent) |                             ) { | ||||||
|         }) |                                 Toast.makeText( | ||||||
|     } |                                     baseContext, | ||||||
|  |                                     R.string.cant_unmark_favortie, | ||||||
|     private fun tryToHandleHtml( |                                     Toast.LENGTH_SHORT | ||||||
|             c: String, |                                 ).show() | ||||||
|             customTabsIntent: CustomTabsIntent, |                             } | ||||||
|             prefs: SharedPreferences |                         }) | ||||||
|     ) { |                 } else { | ||||||
|         try { |                     thread { | ||||||
|             content.text = Html.fromHtml(c, HtmlHttpImageGetter(content, null, true), null) |                         db.actionsDao().insertAllActions(ActionEntity(allItems[pager.currentItem].id, false, false, false, true)) | ||||||
|  |                         afterUnsave() | ||||||
|             //content.setHtml(response.body()!!.content, HtmlHttpImageGetter(content, null, true)) |                     } | ||||||
|         } catch (e: Exception) { |                 } | ||||||
|             Crashlytics.setUserIdentifier(prefs.getString("unique_id", "")) |             } | ||||||
|             Crashlytics.log(100, "CANT_TRANSFORM_TO_HTML", e.message) |             R.id.align_left -> { | ||||||
|             Crashlytics.logException(e) |                 editor.putInt("text_align", ALIGN_LEFT) | ||||||
|             openInBrowserAfterFailing(customTabsIntent) |                 editor.apply() | ||||||
|  |                 alignmentMenu(true) | ||||||
|  |                 refreshFragment() | ||||||
|  |             } | ||||||
|  |             R.id.align_justify -> { | ||||||
|  |                 editor.putInt("text_align", JUSTIFY) | ||||||
|  |                 editor.apply() | ||||||
|  |                 alignmentMenu(false) | ||||||
|  |                 refreshFragment() | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |         return super.onOptionsItemSelected(item) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun openInBrowserAfterFailing(customTabsIntent: CustomTabsIntent) { |     private fun refreshFragment() { | ||||||
|         progressBar.visibility = View.GONE |  | ||||||
|         this@ReaderActivity.openItemUrl( |  | ||||||
|                 url, |  | ||||||
|                 contentText, |  | ||||||
|                 contentImage, |  | ||||||
|                 contentTitle, |  | ||||||
|                 contentSource, |  | ||||||
|                 customTabsIntent, |  | ||||||
|                 true, |  | ||||||
|                 false, |  | ||||||
|                 this@ReaderActivity |  | ||||||
|         ) |  | ||||||
|         finish() |         finish() | ||||||
|  |         overridePendingTransition(0, 0) | ||||||
|  |         startActivity(intent) | ||||||
|  |         overridePendingTransition(0, 0) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         var allItems: ArrayList<Item> = ArrayList() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,14 +1,21 @@ | |||||||
| package apps.amine.bou.readerforselfoss | package apps.amine.bou.readerforselfoss | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
|  | import android.content.res.ColorStateList | ||||||
|  | import android.os.Build | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.preference.PreferenceManager | import android.preference.PreferenceManager | ||||||
| import android.support.v7.app.AppCompatActivity | import androidx.appcompat.app.AppCompatActivity | ||||||
| import android.support.v7.widget.LinearLayoutManager | import androidx.recyclerview.widget.LinearLayoutManager | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import apps.amine.bou.readerforselfoss.adapters.SourcesListAdapter | import apps.amine.bou.readerforselfoss.adapters.SourcesListAdapter | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Sources | import apps.amine.bou.readerforselfoss.api.selfoss.Source | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.Toppings | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import com.ftinc.scoop.Scoop | import com.ftinc.scoop.Scoop | ||||||
| import kotlinx.android.synthetic.main.activity_sources.* | import kotlinx.android.synthetic.main.activity_sources.* | ||||||
| import retrofit2.Call | import retrofit2.Call | ||||||
| @@ -17,14 +24,27 @@ import retrofit2.Response | |||||||
|  |  | ||||||
| class SourcesActivity : AppCompatActivity() { | class SourcesActivity : AppCompatActivity() { | ||||||
|  |  | ||||||
|  |     private lateinit var appColors: AppColors | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         appColors = AppColors(this@SourcesActivity) | ||||||
|  |  | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|         Scoop.getInstance().apply(this) |  | ||||||
|         setContentView(R.layout.activity_sources) |         setContentView(R.layout.activity_sources) | ||||||
|  |  | ||||||
|  |         val scoop = Scoop.getInstance() | ||||||
|  |         scoop.bind(this, Toppings.PRIMARY.value, toolbar) | ||||||
|  |         if  (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { | ||||||
|  |             scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         setSupportActionBar(toolbar) |         setSupportActionBar(toolbar) | ||||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) |         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) |         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||||
|  |  | ||||||
|  |         fab.rippleColor = appColors.colorAccentDark | ||||||
|  |         fab.backgroundTintList = ColorStateList.valueOf(appColors.colorAccent) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onStop() { |     override fun onStop() { | ||||||
| @@ -36,47 +56,51 @@ class SourcesActivity : AppCompatActivity() { | |||||||
|         super.onResume() |         super.onResume() | ||||||
|         val mLayoutManager = LinearLayoutManager(this) |         val mLayoutManager = LinearLayoutManager(this) | ||||||
|  |  | ||||||
|  |         val settings = | ||||||
|  |             getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) | ||||||
|         val prefs = PreferenceManager.getDefaultSharedPreferences(this) |         val prefs = PreferenceManager.getDefaultSharedPreferences(this) | ||||||
|  |  | ||||||
|         val api = SelfossApi( |         val api = SelfossApi( | ||||||
|                 this, |             this, | ||||||
|                 this@SourcesActivity, |             this@SourcesActivity, | ||||||
|                 prefs.getBoolean("isSelfSignedCert", false), |             settings.getBoolean("isSelfSignedCert", false), | ||||||
|                 prefs.getBoolean("should_log_everything", false) |             prefs.getString("api_timeout", "-1")!!.toLong() | ||||||
|         ) |         ) | ||||||
|         var items: ArrayList<Sources> = ArrayList() |         var items: ArrayList<Source> = ArrayList() | ||||||
|  |  | ||||||
|         recyclerView.setHasFixedSize(true) |         recyclerView.setHasFixedSize(true) | ||||||
|         recyclerView.layoutManager = mLayoutManager |         recyclerView.layoutManager = mLayoutManager | ||||||
|  |  | ||||||
|         api.sources.enqueue(object : Callback<List<Sources>> { |         if (this@SourcesActivity.isNetworkAccessible(this@SourcesActivity.findViewById(R.id.recyclerView))) { | ||||||
|             override fun onResponse( |             api.sources.enqueue(object : Callback<List<Source>> { | ||||||
|                     call: Call<List<Sources>>, |                 override fun onResponse( | ||||||
|                     response: Response<List<Sources>> |                     call: Call<List<Source>>, | ||||||
|             ) { |                     response: Response<List<Source>> | ||||||
|                 if (response.body() != null && response.body()!!.isNotEmpty()) { |                 ) { | ||||||
|                     items = response.body() as ArrayList<Sources> |                     if (response.body() != null && response.body()!!.isNotEmpty()) { | ||||||
|                 } |                         items = response.body() as ArrayList<Source> | ||||||
|                 val mAdapter = SourcesListAdapter(this@SourcesActivity, items, api) |                     } | ||||||
|                 recyclerView.adapter = mAdapter |                     val mAdapter = SourcesListAdapter(this@SourcesActivity, items, api) | ||||||
|                 mAdapter.notifyDataSetChanged() |                     recyclerView.adapter = mAdapter | ||||||
|                 if (items.isEmpty()) { |                     mAdapter.notifyDataSetChanged() | ||||||
|                     Toast.makeText( |                     if (items.isEmpty()) { | ||||||
|  |                         Toast.makeText( | ||||||
|                             this@SourcesActivity, |                             this@SourcesActivity, | ||||||
|                             R.string.nothing_here, |                             R.string.nothing_here, | ||||||
|                             Toast.LENGTH_SHORT |                             Toast.LENGTH_SHORT | ||||||
|                     ).show() |                         ).show() | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun onFailure(call: Call<List<Sources>>, t: Throwable) { |                 override fun onFailure(call: Call<List<Source>>, t: Throwable) { | ||||||
|                 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() | ||||||
|             } |                 } | ||||||
|         }) |             }) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         fab.setOnClickListener { |         fab.setOnClickListener { | ||||||
|             startActivity(Intent(this@SourcesActivity, AddSourceActivity::class.java)) |             startActivity(Intent(this@SourcesActivity, AddSourceActivity::class.java)) | ||||||
|   | |||||||
| @@ -2,56 +2,66 @@ package apps.amine.bou.readerforselfoss.adapters | |||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.graphics.Color | import androidx.cardview.widget.CardView | ||||||
| import android.support.design.widget.Snackbar | import androidx.recyclerview.widget.RecyclerView | ||||||
| import android.support.v7.widget.CardView |  | ||||||
| import android.support.v7.widget.RecyclerView |  | ||||||
| import android.text.Html |  | ||||||
| import android.view.LayoutInflater | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
| import android.widget.ImageView.ScaleType | import android.widget.ImageView.ScaleType | ||||||
| import android.widget.TextView |  | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
|  | import androidx.core.content.ContextCompat | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Item | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
| import apps.amine.bou.readerforselfoss.themes.AppColors | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.LinkOnTouchListener | ||||||
| import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | ||||||
| import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | ||||||
| import apps.amine.bou.readerforselfoss.utils.glide.bitmapCenterCrop | import apps.amine.bou.readerforselfoss.utils.glide.bitmapCenterCrop | ||||||
| import apps.amine.bou.readerforselfoss.utils.glide.circularBitmapDrawable | import apps.amine.bou.readerforselfoss.utils.glide.circularBitmapDrawable | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.openInBrowserAsNewTask | import apps.amine.bou.readerforselfoss.utils.openInBrowserAsNewTask | ||||||
| import apps.amine.bou.readerforselfoss.utils.openItemUrl | import apps.amine.bou.readerforselfoss.utils.openItemUrl | ||||||
| import apps.amine.bou.readerforselfoss.utils.shareLink | import apps.amine.bou.readerforselfoss.utils.shareLink | ||||||
| import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | ||||||
| import apps.amine.bou.readerforselfoss.utils.succeeded |  | ||||||
| import apps.amine.bou.readerforselfoss.utils.toTextDrawableString | import apps.amine.bou.readerforselfoss.utils.toTextDrawableString | ||||||
| import com.amulyakhare.textdrawable.TextDrawable | import com.amulyakhare.textdrawable.TextDrawable | ||||||
| import com.amulyakhare.textdrawable.util.ColorGenerator | import com.amulyakhare.textdrawable.util.ColorGenerator | ||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
| import com.crashlytics.android.Crashlytics |  | ||||||
| import com.like.LikeButton | import com.like.LikeButton | ||||||
| import com.like.OnLikeListener | import com.like.OnLikeListener | ||||||
| import kotlinx.android.synthetic.main.card_item.view.* | import kotlinx.android.synthetic.main.card_item.view.* | ||||||
|  | import kotlinx.android.synthetic.main.card_item.view.itemImage | ||||||
|  | import kotlinx.android.synthetic.main.card_item.view.sourceTitleAndDate | ||||||
|  | import kotlinx.android.synthetic.main.card_item.view.title | ||||||
|  | import kotlinx.android.synthetic.main.list_item.view.* | ||||||
| import retrofit2.Call | import retrofit2.Call | ||||||
| import retrofit2.Callback | import retrofit2.Callback | ||||||
| import retrofit2.Response | import retrofit2.Response | ||||||
|  | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
| class ItemCardAdapter( | class ItemCardAdapter( | ||||||
|         private val app: Activity, |     override val app: Activity, | ||||||
|         private val items: ArrayList<Item>, |     override var items: ArrayList<Item>, | ||||||
|         private val api: SelfossApi, |     override val api: SelfossApi, | ||||||
|         private val helper: CustomTabActivityHelper, |     override val db: AppDatabase, | ||||||
|         private val internalBrowser: Boolean, |     private val helper: CustomTabActivityHelper, | ||||||
|         private val articleViewer: Boolean, |     private val internalBrowser: Boolean, | ||||||
|         private val fullHeightCards: Boolean, |     private val articleViewer: Boolean, | ||||||
|         private val appColors: AppColors, |     private val fullHeightCards: Boolean, | ||||||
|         val debugReadingItems: Boolean, |     override val appColors: AppColors, | ||||||
|         val userIdentifier: String |     override val userIdentifier: String, | ||||||
| ) : RecyclerView.Adapter<ItemCardAdapter.ViewHolder>() { |     override val config: Config, | ||||||
|  |     override val updateItems: (ArrayList<Item>) -> Unit | ||||||
|  | ) : ItemsAdapter<ItemCardAdapter.ViewHolder>() { | ||||||
|     private val c: Context = app.baseContext |     private val c: Context = app.baseContext | ||||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL |     private val generator: ColorGenerator = ColorGenerator.MATERIAL | ||||||
|  |     private val imageMaxHeight: Int = | ||||||
|  |         c.resources.getDimension(R.dimen.card_image_max_height).toInt() | ||||||
|  |  | ||||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { |     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { | ||||||
|         val v = LayoutInflater.from(c).inflate(R.layout.card_item, parent, false) as CardView |         val v = LayoutInflater.from(c).inflate(R.layout.card_item, parent, false) as CardView | ||||||
| @@ -63,28 +73,47 @@ class ItemCardAdapter( | |||||||
|  |  | ||||||
|  |  | ||||||
|         holder.mView.favButton.isLiked = itm.starred |         holder.mView.favButton.isLiked = itm.starred | ||||||
|         holder.mView.title.text = Html.fromHtml(itm.title) |         holder.mView.title.text = itm.getTitleDecoded() | ||||||
|  |         holder.mView.title.setTextColor(ContextCompat.getColor( | ||||||
|  |                 c, | ||||||
|  |                 appColors.textColor | ||||||
|  |         )) | ||||||
|  |         holder.mView.title.setOnTouchListener(LinkOnTouchListener()) | ||||||
|  |  | ||||||
|  |         holder.mView.title.setLinkTextColor(appColors.colorAccent) | ||||||
|  |  | ||||||
|         holder.mView.sourceTitleAndDate.text = itm.sourceAndDateText() |         holder.mView.sourceTitleAndDate.text = itm.sourceAndDateText() | ||||||
|  |  | ||||||
|  |         holder.mView.sourceTitleAndDate.setTextColor(ContextCompat.getColor( | ||||||
|  |                 c, | ||||||
|  |                 appColors.textColor | ||||||
|  |         )) | ||||||
|  |  | ||||||
|  |         if (!fullHeightCards) { | ||||||
|  |             holder.mView.itemImage.maxHeight = imageMaxHeight | ||||||
|  |             holder.mView.itemImage.scaleType = ScaleType.CENTER_CROP | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (itm.getThumbnail(c).isEmpty()) { |         if (itm.getThumbnail(c).isEmpty()) { | ||||||
|  |             holder.mView.itemImage.visibility = View.GONE | ||||||
|             Glide.with(c).clear(holder.mView.itemImage) |             Glide.with(c).clear(holder.mView.itemImage) | ||||||
|             holder.mView.itemImage.setImageDrawable(null) |             holder.mView.itemImage.setImageDrawable(null) | ||||||
|         } else { |         } else { | ||||||
|             c.bitmapCenterCrop(itm.getThumbnail(c), holder.mView.itemImage) |             holder.mView.itemImage.visibility = View.VISIBLE | ||||||
|  |             c.bitmapCenterCrop(config, itm.getThumbnail(c), holder.mView.itemImage) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (itm.getIcon(c).isEmpty()) { |         if (itm.getIcon(c).isEmpty()) { | ||||||
|             val color = generator.getColor(itm.sourcetitle) |             val color = generator.getColor(itm.getSourceTitle()) | ||||||
|  |  | ||||||
|             val drawable = |             val drawable = | ||||||
|                     TextDrawable |                 TextDrawable | ||||||
|                             .builder() |                     .builder() | ||||||
|                             .round() |                     .round() | ||||||
|                             .build(itm.sourcetitle.toTextDrawableString(), color) |                     .build(itm.getSourceTitle().toTextDrawableString(c), color) | ||||||
|             holder.mView.sourceImage.setImageDrawable(drawable) |             holder.mView.sourceImage.setImageDrawable(drawable) | ||||||
|         } else { |         } else { | ||||||
|             c.circularBitmapDrawable(itm.getIcon(c), holder.mView.sourceImage) |             c.circularBitmapDrawable(config, itm.getIcon(c), holder.mView.sourceImage) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         holder.mView.favButton.isLiked = itm.starred |         holder.mView.favButton.isLiked = itm.starred | ||||||
| @@ -94,150 +123,78 @@ class ItemCardAdapter( | |||||||
|         return items.size |         return items.size | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun doUnmark(i: Item, position: Int) { |  | ||||||
|         val s = Snackbar |  | ||||||
|                 .make( |  | ||||||
|                         app.findViewById(R.id.coordLayout), |  | ||||||
|                         R.string.marked_as_read, |  | ||||||
|                         Snackbar.LENGTH_LONG |  | ||||||
|                 ) |  | ||||||
|                 .setAction(R.string.undo_string) { |  | ||||||
|                     items.add(position, i) |  | ||||||
|                     notifyItemInserted(position) |  | ||||||
|  |  | ||||||
|                     api.unmarkItem(i.id).enqueue(object : Callback<SuccessResponse> { |  | ||||||
|                         override fun onResponse( |  | ||||||
|                                 call: Call<SuccessResponse>, |  | ||||||
|                                 response: Response<SuccessResponse> |  | ||||||
|                         ) { |  | ||||||
|                         } |  | ||||||
|  |  | ||||||
|                         override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { |  | ||||||
|                             items.remove(i) |  | ||||||
|                             notifyItemRemoved(position) |  | ||||||
|                             doUnmark(i, position) |  | ||||||
|                         } |  | ||||||
|                     }) |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|         val view = s.view |  | ||||||
|         val tv: TextView = view.findViewById(android.support.design.R.id.snackbar_text) |  | ||||||
|         tv.setTextColor(Color.WHITE) |  | ||||||
|         s.show() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun removeItemAtIndex(position: Int) { |  | ||||||
|  |  | ||||||
|         val i = items[position] |  | ||||||
|  |  | ||||||
|         items.remove(i) |  | ||||||
|         notifyItemRemoved(position) |  | ||||||
|  |  | ||||||
|         api.markItem(i.id).enqueue(object : Callback<SuccessResponse> { |  | ||||||
|             override fun onResponse( |  | ||||||
|                     call: Call<SuccessResponse>, |  | ||||||
|                     response: Response<SuccessResponse> |  | ||||||
|             ) { |  | ||||||
|                 if (!response.succeeded() && debugReadingItems) { |  | ||||||
|                     val message = |  | ||||||
|                             "message: ${response.message()} " + |  | ||||||
|                                     "response isSuccess: ${response.isSuccessful} " + |  | ||||||
|                                     "response code: ${response.code()} " + |  | ||||||
|                                     "response message: ${response.message()} " + |  | ||||||
|                                     "response errorBody: ${response.errorBody()?.string()} " + |  | ||||||
|                                     "body success: ${response.body()?.success} " + |  | ||||||
|                                     "body isSuccess: ${response.body()?.isSuccess}" |  | ||||||
|                     Crashlytics.setUserIdentifier(userIdentifier) |  | ||||||
|                     Crashlytics.log(100, "READ_DEBUG_SUCCESS", message) |  | ||||||
|                     Crashlytics.logException(Exception("Was success, but did it work ?")) |  | ||||||
|  |  | ||||||
|                     Toast.makeText(c, message, Toast.LENGTH_LONG).show() |  | ||||||
|                 } |  | ||||||
|                 doUnmark(i, position) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { |  | ||||||
|                 if (debugReadingItems) { |  | ||||||
|                     Crashlytics.setUserIdentifier(userIdentifier) |  | ||||||
|                     Crashlytics.log(100, "READ_DEBUG_ERROR", t.message) |  | ||||||
|                     Crashlytics.logException(t) |  | ||||||
|                     Toast.makeText(c, t.message, Toast.LENGTH_LONG).show() |  | ||||||
|                 } |  | ||||||
|                 Toast.makeText( |  | ||||||
|                         app, |  | ||||||
|                         app.getString(R.string.cant_mark_read), |  | ||||||
|                         Toast.LENGTH_SHORT |  | ||||||
|                 ).show() |  | ||||||
|                 items.add(i) |  | ||||||
|                 notifyItemInserted(position) |  | ||||||
|             } |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     inner class ViewHolder(val mView: CardView) : RecyclerView.ViewHolder(mView) { |     inner class ViewHolder(val mView: CardView) : RecyclerView.ViewHolder(mView) { | ||||||
|         init { |         init { | ||||||
|             mView.setCardBackgroundColor(appColors.cardBackground) |             mView.setCardBackgroundColor(appColors.cardBackgroundColor) | ||||||
|             handleClickListeners() |             handleClickListeners() | ||||||
|             handleCustomTabActions() |             handleCustomTabActions() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private fun handleClickListeners() { |         private fun handleClickListeners() { | ||||||
|  |  | ||||||
|             if (!fullHeightCards) { |  | ||||||
|                 mView.itemImage.maxHeight = c.resources.getDimension(R.dimen.card_image_max_height).toInt() |  | ||||||
|                 mView.itemImage.scaleType = ScaleType.CENTER_CROP |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             mView.favButton.setOnLikeListener(object : OnLikeListener { |             mView.favButton.setOnLikeListener(object : OnLikeListener { | ||||||
|                 override fun liked(likeButton: LikeButton) { |                 override fun liked(likeButton: LikeButton) { | ||||||
|                     val (id) = items[adapterPosition] |                     val (id) = items[adapterPosition] | ||||||
|                     api.starrItem(id).enqueue(object : Callback<SuccessResponse> { |                     if (c.isNetworkAccessible(null)) { | ||||||
|                         override fun onResponse( |                         api.starrItem(id).enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                             override fun onResponse( | ||||||
|                                 call: Call<SuccessResponse>, |                                 call: Call<SuccessResponse>, | ||||||
|                                 response: Response<SuccessResponse> |                                 response: Response<SuccessResponse> | ||||||
|                         ) { |                             ) { | ||||||
|                         } |                             } | ||||||
|  |  | ||||||
|                         override fun onFailure( |                             override fun onFailure( | ||||||
|                                 call: Call<SuccessResponse>, |                                 call: Call<SuccessResponse>, | ||||||
|                                 t: Throwable |                                 t: Throwable | ||||||
|                         ) { |                             ) { | ||||||
|                             mView.favButton.isLiked = false |                                 mView.favButton.isLiked = false | ||||||
|                             Toast.makeText( |                                 Toast.makeText( | ||||||
|                                     c, |                                     c, | ||||||
|                                     R.string.cant_mark_favortie, |                                     R.string.cant_mark_favortie, | ||||||
|                                     Toast.LENGTH_SHORT |                                     Toast.LENGTH_SHORT | ||||||
|                             ).show() |                                 ).show() | ||||||
|  |                             } | ||||||
|  |                         }) | ||||||
|  |                     } else { | ||||||
|  |                         thread { | ||||||
|  |                             db.actionsDao().insertAllActions(ActionEntity(id, false, false, true, false)) | ||||||
|                         } |                         } | ||||||
|                     }) |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 override fun unLiked(likeButton: LikeButton) { |                 override fun unLiked(likeButton: LikeButton) { | ||||||
|                     val (id) = items[adapterPosition] |                     val (id) = items[adapterPosition] | ||||||
|                     api.unstarrItem(id).enqueue(object : Callback<SuccessResponse> { |                     if (c.isNetworkAccessible(null)) { | ||||||
|                         override fun onResponse( |                         api.unstarrItem(id).enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                             override fun onResponse( | ||||||
|                                 call: Call<SuccessResponse>, |                                 call: Call<SuccessResponse>, | ||||||
|                                 response: Response<SuccessResponse> |                                 response: Response<SuccessResponse> | ||||||
|                         ) { |                             ) { | ||||||
|                         } |                             } | ||||||
|  |  | ||||||
|                         override fun onFailure( |                             override fun onFailure( | ||||||
|                                 call: Call<SuccessResponse>, |                                 call: Call<SuccessResponse>, | ||||||
|                                 t: Throwable |                                 t: Throwable | ||||||
|                         ) { |                             ) { | ||||||
|                             mView.favButton.isLiked = true |                                 mView.favButton.isLiked = true | ||||||
|                             Toast.makeText( |                                 Toast.makeText( | ||||||
|                                     c, |                                     c, | ||||||
|                                     R.string.cant_unmark_favortie, |                                     R.string.cant_unmark_favortie, | ||||||
|                                     Toast.LENGTH_SHORT |                                     Toast.LENGTH_SHORT | ||||||
|                             ).show() |                                 ).show() | ||||||
|  |                             } | ||||||
|  |                         }) | ||||||
|  |                     } else { | ||||||
|  |                         thread { | ||||||
|  |                             db.actionsDao().insertAllActions(ActionEntity(id, false, false, false, true)) | ||||||
|                         } |                         } | ||||||
|                     }) |                     } | ||||||
|                 } |                 } | ||||||
|             }) |             }) | ||||||
|  |  | ||||||
|             mView.shareBtn.setOnClickListener { |             mView.shareBtn.setOnClickListener { | ||||||
|                 c.shareLink(items[adapterPosition].getLinkDecoded()) |                 val item = items[adapterPosition] | ||||||
|  |                 c.shareLink(item.getLinkDecoded(), item.getTitleDecoded()) | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             mView.browserBtn.setOnClickListener { |             mView.browserBtn.setOnClickListener { | ||||||
| @@ -251,15 +208,13 @@ class ItemCardAdapter( | |||||||
|  |  | ||||||
|             mView.setOnClickListener { |             mView.setOnClickListener { | ||||||
|                 c.openItemUrl( |                 c.openItemUrl( | ||||||
|                         items[adapterPosition].getLinkDecoded(), |                     items, | ||||||
|                         items[adapterPosition].content, |                     adapterPosition, | ||||||
|                         items[adapterPosition].getThumbnail(c), |                     items[adapterPosition].getLinkDecoded(), | ||||||
|                         items[adapterPosition].title, |                     customTabsIntent, | ||||||
|                         items[adapterPosition].sourceAndDateText(), |                     internalBrowser, | ||||||
|                         customTabsIntent, |                     articleViewer, | ||||||
|                         internalBrowser, |                     app | ||||||
|                         articleViewer, |  | ||||||
|                         app |  | ||||||
|                 ) |                 ) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -2,21 +2,26 @@ package apps.amine.bou.readerforselfoss.adapters | |||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.graphics.Color | import androidx.constraintlayout.widget.ConstraintLayout | ||||||
| import android.support.constraint.ConstraintLayout | import androidx.recyclerview.widget.RecyclerView | ||||||
| import android.support.design.widget.Snackbar | import android.text.Spannable | ||||||
| import android.support.v7.widget.RecyclerView | import android.text.style.ClickableSpan | ||||||
| import android.text.Html |  | ||||||
| import android.util.TypedValue | import android.util.TypedValue | ||||||
| import android.view.LayoutInflater | import android.view.LayoutInflater | ||||||
|  | import android.view.MotionEvent | ||||||
| import android.view.View | import android.view.View | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
| import android.widget.TextView | import android.widget.TextView | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
|  | import androidx.core.content.ContextCompat | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Item | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.LinkOnTouchListener | ||||||
| import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | ||||||
| import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | ||||||
| import apps.amine.bou.readerforselfoss.utils.glide.bitmapCenterCrop | import apps.amine.bou.readerforselfoss.utils.glide.bitmapCenterCrop | ||||||
| @@ -25,10 +30,9 @@ import apps.amine.bou.readerforselfoss.utils.openInBrowserAsNewTask | |||||||
| import apps.amine.bou.readerforselfoss.utils.openItemUrl | import apps.amine.bou.readerforselfoss.utils.openItemUrl | ||||||
| import apps.amine.bou.readerforselfoss.utils.shareLink | import apps.amine.bou.readerforselfoss.utils.shareLink | ||||||
| import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | ||||||
| import apps.amine.bou.readerforselfoss.utils.succeeded | import apps.amine.bou.readerforselfoss.utils.toTextDrawableString | ||||||
| import com.amulyakhare.textdrawable.TextDrawable | import com.amulyakhare.textdrawable.TextDrawable | ||||||
| import com.amulyakhare.textdrawable.util.ColorGenerator | import com.amulyakhare.textdrawable.util.ColorGenerator | ||||||
| import com.crashlytics.android.Crashlytics |  | ||||||
| import com.like.LikeButton | import com.like.LikeButton | ||||||
| import com.like.OnLikeListener | import com.like.OnLikeListener | ||||||
| import kotlinx.android.synthetic.main.list_item.view.* | import kotlinx.android.synthetic.main.list_item.view.* | ||||||
| @@ -39,25 +43,26 @@ import java.util.* | |||||||
| import kotlin.collections.ArrayList | import kotlin.collections.ArrayList | ||||||
|  |  | ||||||
| class ItemListAdapter( | class ItemListAdapter( | ||||||
|         private val app: Activity, |     override val app: Activity, | ||||||
|         private val items: ArrayList<Item>, |     override var items: ArrayList<Item>, | ||||||
|         private val api: SelfossApi, |     override val api: SelfossApi, | ||||||
|         private val helper: CustomTabActivityHelper, |     override val db: AppDatabase, | ||||||
|         private val clickBehavior: Boolean, |     private val helper: CustomTabActivityHelper, | ||||||
|         private val internalBrowser: Boolean, |     private val internalBrowser: Boolean, | ||||||
|         private val articleViewer: Boolean, |     private val articleViewer: Boolean, | ||||||
|         val debugReadingItems: Boolean, |     override val userIdentifier: String, | ||||||
|         val userIdentifier: String |     override val appColors: AppColors, | ||||||
| ) : RecyclerView.Adapter<ItemListAdapter.ViewHolder>() { |     override val config: Config, | ||||||
|  |     override val updateItems: (ArrayList<Item>) -> Unit | ||||||
|  | ) : ItemsAdapter<ItemListAdapter.ViewHolder>() { | ||||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL |     private val generator: ColorGenerator = ColorGenerator.MATERIAL | ||||||
|     private val c: Context = app.baseContext |     private val c: Context = app.baseContext | ||||||
|     private val bars: ArrayList<Boolean> = ArrayList(Collections.nCopies(items.size + 1, false)) |  | ||||||
|  |  | ||||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { |     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { | ||||||
|         val v = LayoutInflater.from(c).inflate( |         val v = LayoutInflater.from(c).inflate( | ||||||
|                 R.layout.list_item, |             R.layout.list_item, | ||||||
|                 parent, |             parent, | ||||||
|                 false |             false | ||||||
|         ) as ConstraintLayout |         ) as ConstraintLayout | ||||||
|         return ViewHolder(v) |         return ViewHolder(v) | ||||||
|     } |     } | ||||||
| @@ -66,253 +71,66 @@ class ItemListAdapter( | |||||||
|         val itm = items[position] |         val itm = items[position] | ||||||
|  |  | ||||||
|  |  | ||||||
|         holder.mView.favButton.isLiked = itm.starred |         holder.mView.title.text = itm.getTitleDecoded() | ||||||
|         holder.mView.title.text = Html.fromHtml(itm.title) |  | ||||||
|  |         holder.mView.title.setTextColor(ContextCompat.getColor( | ||||||
|  |                 c, | ||||||
|  |                 appColors.textColor | ||||||
|  |         )) | ||||||
|  |  | ||||||
|  |         holder.mView.title.setOnTouchListener(LinkOnTouchListener()) | ||||||
|  |  | ||||||
|  |         holder.mView.title.setLinkTextColor(appColors.colorAccent) | ||||||
|  |  | ||||||
|         holder.mView.sourceTitleAndDate.text = itm.sourceAndDateText() |         holder.mView.sourceTitleAndDate.text = itm.sourceAndDateText() | ||||||
|  |  | ||||||
|  |         holder.mView.sourceTitleAndDate.setTextColor(ContextCompat.getColor( | ||||||
|  |                 c, | ||||||
|  |                 appColors.textColor | ||||||
|  |         )) | ||||||
|  |  | ||||||
|         if (itm.getThumbnail(c).isEmpty()) { |         if (itm.getThumbnail(c).isEmpty()) { | ||||||
|             val sizeInInt = 46 |  | ||||||
|             val sizeInDp = TypedValue.applyDimension( |  | ||||||
|                     TypedValue.COMPLEX_UNIT_DIP, sizeInInt.toFloat(), c.resources |  | ||||||
|                     .displayMetrics |  | ||||||
|             ).toInt() |  | ||||||
|  |  | ||||||
|             val marginInInt = 16 |  | ||||||
|             val marginInDp = TypedValue.applyDimension( |  | ||||||
|                     TypedValue.COMPLEX_UNIT_DIP, marginInInt.toFloat(), c.resources |  | ||||||
|                     .displayMetrics |  | ||||||
|             ).toInt() |  | ||||||
|  |  | ||||||
|             val params = holder.mView.itemImage.layoutParams as ViewGroup.MarginLayoutParams |  | ||||||
|             params.height = sizeInDp |  | ||||||
|             params.width = sizeInDp |  | ||||||
|             params.setMargins(marginInDp, 0, 0, 0) |  | ||||||
|             holder.mView.itemImage.layoutParams = params |  | ||||||
|  |  | ||||||
|             if (itm.getIcon(c).isEmpty()) { |             if (itm.getIcon(c).isEmpty()) { | ||||||
|                 val color = generator.getColor(itm.sourcetitle) |                 val color = generator.getColor(itm.getSourceTitle()) | ||||||
|                 val textDrawable = StringBuilder() |  | ||||||
|                 for (s in itm.sourcetitle.split(" ".toRegex()).dropLastWhile { it.isEmpty() }.toTypedArray()) { |  | ||||||
|                     textDrawable.append(s[0]) |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 val builder = TextDrawable.builder().round() |                 val drawable = | ||||||
|  |                     TextDrawable | ||||||
|  |                         .builder() | ||||||
|  |                         .round() | ||||||
|  |                         .build(itm.getSourceTitle().toTextDrawableString(c), color) | ||||||
|  |  | ||||||
|                 val drawable = builder.build(textDrawable.toString(), color) |  | ||||||
|                 holder.mView.itemImage.setImageDrawable(drawable) |                 holder.mView.itemImage.setImageDrawable(drawable) | ||||||
|             } else { |             } else { | ||||||
|                 c.circularBitmapDrawable(itm.getIcon(c), holder.mView.itemImage) |                 c.circularBitmapDrawable(config, itm.getIcon(c), holder.mView.itemImage) | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             c.bitmapCenterCrop(itm.getThumbnail(c), holder.mView.itemImage) |             c.bitmapCenterCrop(config, itm.getThumbnail(c), holder.mView.itemImage) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (bars[position]) { |  | ||||||
|             holder.mView.actionBar.visibility = View.VISIBLE |  | ||||||
|         } else { |  | ||||||
|             holder.mView.actionBar.visibility = View.GONE |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         holder.mView.favButton.isLiked = itm.starred |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun getItemCount(): Int = items.size |     override fun getItemCount(): Int = items.size | ||||||
|  |  | ||||||
|     private fun doUnmark(i: Item, position: Int) { |  | ||||||
|         val s = Snackbar |  | ||||||
|                 .make( |  | ||||||
|                         app.findViewById(R.id.coordLayout), |  | ||||||
|                         R.string.marked_as_read, |  | ||||||
|                         Snackbar.LENGTH_LONG |  | ||||||
|                 ) |  | ||||||
|                 .setAction(R.string.undo_string) { |  | ||||||
|                     items.add(position, i) |  | ||||||
|                     notifyItemInserted(position) |  | ||||||
|  |  | ||||||
|                     api.unmarkItem(i.id).enqueue(object : Callback<SuccessResponse> { |  | ||||||
|                         override fun onResponse( |  | ||||||
|                                 call: Call<SuccessResponse>, |  | ||||||
|                                 response: Response<SuccessResponse> |  | ||||||
|                         ) { |  | ||||||
|                         } |  | ||||||
|  |  | ||||||
|                         override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { |  | ||||||
|                             items.remove(i) |  | ||||||
|                             notifyItemRemoved(position) |  | ||||||
|                             doUnmark(i, position) |  | ||||||
|                         } |  | ||||||
|                     }) |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|         val view = s.view |  | ||||||
|         val tv: TextView = view.findViewById(android.support.design.R.id.snackbar_text) |  | ||||||
|         tv.setTextColor(Color.WHITE) |  | ||||||
|         s.show() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun removeItemAtIndex(position: Int) { |  | ||||||
|  |  | ||||||
|         val i = items[position] |  | ||||||
|  |  | ||||||
|         items.remove(i) |  | ||||||
|         notifyItemRemoved(position) |  | ||||||
|  |  | ||||||
|         api.markItem(i.id).enqueue(object : Callback<SuccessResponse> { |  | ||||||
|             override fun onResponse( |  | ||||||
|                     call: Call<SuccessResponse>, |  | ||||||
|                     response: Response<SuccessResponse> |  | ||||||
|             ) { |  | ||||||
|                 if (!response.succeeded() && debugReadingItems) { |  | ||||||
|                     val message = |  | ||||||
|                             "message: ${response.message()} " + |  | ||||||
|                                     "response isSuccess: ${response.isSuccessful} " + |  | ||||||
|                                     "response code: ${response.code()} " + |  | ||||||
|                                     "response message: ${response.message()} " + |  | ||||||
|                                     "response errorBody: ${response.errorBody()?.string()} " + |  | ||||||
|                                     "body success: ${response.body()?.success} " + |  | ||||||
|                                     "body isSuccess: ${response.body()?.isSuccess}" |  | ||||||
|                     Crashlytics.setUserIdentifier(userIdentifier) |  | ||||||
|                     Crashlytics.log(100, "READ_DEBUG_SUCCESS", message) |  | ||||||
|                     Crashlytics.logException(Exception("Was success, but did it work ?")) |  | ||||||
|                     Toast.makeText(c, message, Toast.LENGTH_LONG).show() |  | ||||||
|                 } |  | ||||||
|                 doUnmark(i, position) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { |  | ||||||
|                 if (debugReadingItems) { |  | ||||||
|                     Crashlytics.setUserIdentifier(userIdentifier) |  | ||||||
|                     Crashlytics.log(100, "READ_DEBUG_ERROR", t.message) |  | ||||||
|                     Crashlytics.logException(t) |  | ||||||
|                     Toast.makeText(c, t.message, Toast.LENGTH_LONG).show() |  | ||||||
|                 } |  | ||||||
|                 Toast.makeText( |  | ||||||
|                         app, |  | ||||||
|                         app.getString(R.string.cant_mark_read), |  | ||||||
|                         Toast.LENGTH_SHORT |  | ||||||
|                 ).show() |  | ||||||
|                 items.add(i) |  | ||||||
|                 notifyItemInserted(position) |  | ||||||
|             } |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     inner class ViewHolder(val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) { |     inner class ViewHolder(val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) { | ||||||
|  |  | ||||||
|         init { |         init { | ||||||
|             handleClickListeners() |  | ||||||
|             handleCustomTabActions() |             handleCustomTabActions() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private fun handleClickListeners() { |  | ||||||
|  |  | ||||||
|             mView.favButton.setOnLikeListener(object : OnLikeListener { |  | ||||||
|                 override fun liked(likeButton: LikeButton) { |  | ||||||
|                     val (id) = items[adapterPosition] |  | ||||||
|                     api.starrItem(id).enqueue(object : Callback<SuccessResponse> { |  | ||||||
|                         override fun onResponse( |  | ||||||
|                                 call: Call<SuccessResponse>, |  | ||||||
|                                 response: Response<SuccessResponse> |  | ||||||
|                         ) { |  | ||||||
|                         } |  | ||||||
|  |  | ||||||
|                         override fun onFailure( |  | ||||||
|                                 call: Call<SuccessResponse>, |  | ||||||
|                                 t: Throwable |  | ||||||
|                         ) { |  | ||||||
|                             mView.favButton.isLiked = false |  | ||||||
|                             Toast.makeText( |  | ||||||
|                                     c, |  | ||||||
|                                     R.string.cant_mark_favortie, |  | ||||||
|                                     Toast.LENGTH_SHORT |  | ||||||
|                             ).show() |  | ||||||
|                         } |  | ||||||
|                     }) |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 override fun unLiked(likeButton: LikeButton) { |  | ||||||
|                     val (id) = items[adapterPosition] |  | ||||||
|                     api.unstarrItem(id).enqueue(object : Callback<SuccessResponse> { |  | ||||||
|                         override fun onResponse( |  | ||||||
|                                 call: Call<SuccessResponse>, |  | ||||||
|                                 response: Response<SuccessResponse> |  | ||||||
|                         ) { |  | ||||||
|                         } |  | ||||||
|  |  | ||||||
|                         override fun onFailure( |  | ||||||
|                                 call: Call<SuccessResponse>, |  | ||||||
|                                 t: Throwable |  | ||||||
|                         ) { |  | ||||||
|                             mView.favButton.isLiked = true |  | ||||||
|                             Toast.makeText( |  | ||||||
|                                     c, |  | ||||||
|                                     R.string.cant_unmark_favortie, |  | ||||||
|                                     Toast.LENGTH_SHORT |  | ||||||
|                             ).show() |  | ||||||
|                         } |  | ||||||
|                     }) |  | ||||||
|                 } |  | ||||||
|             }) |  | ||||||
|  |  | ||||||
|             mView.shareBtn.setOnClickListener { |  | ||||||
|                 c.shareLink(items[adapterPosition].getLinkDecoded()) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             mView.browserBtn.setOnClickListener { |  | ||||||
|                 c.openInBrowserAsNewTask(items[adapterPosition]) |  | ||||||
|  |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private fun handleCustomTabActions() { |         private fun handleCustomTabActions() { | ||||||
|             val customTabsIntent = c.buildCustomTabsIntent() |             val customTabsIntent = c.buildCustomTabsIntent() | ||||||
|             helper.bindCustomTabsService(app) |             helper.bindCustomTabsService(app) | ||||||
|  |  | ||||||
|  |             mView.setOnClickListener { | ||||||
|             if (!clickBehavior) { |                 c.openItemUrl( | ||||||
|                 mView.setOnClickListener { |                     items, | ||||||
|                     c.openItemUrl( |                     adapterPosition, | ||||||
|                             items[adapterPosition].getLinkDecoded(), |                     items[adapterPosition].getLinkDecoded(), | ||||||
|                             items[adapterPosition].content, |                     customTabsIntent, | ||||||
|                             items[adapterPosition].getThumbnail(c), |                     internalBrowser, | ||||||
|                             items[adapterPosition].title, |                     articleViewer, | ||||||
|                             items[adapterPosition].sourceAndDateText(), |                     app | ||||||
|                             customTabsIntent, |                 ) | ||||||
|                             internalBrowser, |  | ||||||
|                             articleViewer, |  | ||||||
|                             app |  | ||||||
|                     ) |  | ||||||
|                 } |  | ||||||
|                 mView.setOnLongClickListener { |  | ||||||
|                     actionBarShowHide() |  | ||||||
|                     true |  | ||||||
|                 } |  | ||||||
|             } else { |  | ||||||
|                 mView.setOnClickListener { actionBarShowHide() } |  | ||||||
|                 mView.setOnLongClickListener { |  | ||||||
|                     c.openItemUrl( |  | ||||||
|                             items[adapterPosition].getLinkDecoded(), |  | ||||||
|                             items[adapterPosition].content, |  | ||||||
|                             items[adapterPosition].getThumbnail(c), |  | ||||||
|                             items[adapterPosition].title, |  | ||||||
|                             items[adapterPosition].sourceAndDateText(), |  | ||||||
|                             customTabsIntent, |  | ||||||
|                             internalBrowser, |  | ||||||
|                             articleViewer, |  | ||||||
|                             app |  | ||||||
|                     ) |  | ||||||
|                     true |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private fun actionBarShowHide() { |  | ||||||
|             bars[adapterPosition] = true |  | ||||||
|             if (mView.actionBar.visibility == View.GONE) { |  | ||||||
|                 mView.actionBar.visibility = View.VISIBLE |  | ||||||
|             } else { |  | ||||||
|                 mView.actionBar.visibility = View.GONE |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -0,0 +1,241 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.adapters | ||||||
|  |  | ||||||
|  | import android.app.Activity | ||||||
|  | import android.graphics.Color | ||||||
|  | import com.google.android.material.snackbar.Snackbar | ||||||
|  | import androidx.recyclerview.widget.RecyclerView | ||||||
|  | import android.widget.TextView | ||||||
|  | import android.widget.Toast | ||||||
|  | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.succeeded | ||||||
|  | import retrofit2.Call | ||||||
|  | import retrofit2.Callback | ||||||
|  | import retrofit2.Response | ||||||
|  | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
|  | abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapter<VH>() { | ||||||
|  |     abstract var items: ArrayList<Item> | ||||||
|  |     abstract val api: SelfossApi | ||||||
|  |     abstract val db: AppDatabase | ||||||
|  |     abstract val userIdentifier: String | ||||||
|  |     abstract val app: Activity | ||||||
|  |     abstract val appColors: AppColors | ||||||
|  |     abstract val config: Config | ||||||
|  |     abstract val updateItems: (ArrayList<Item>) -> Unit | ||||||
|  |  | ||||||
|  |     fun updateAllItems(newItems: ArrayList<Item>) { | ||||||
|  |         items = newItems | ||||||
|  |         notifyDataSetChanged() | ||||||
|  |         updateItems(items) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun unmarkSnackbar(i: Item, position: Int) { | ||||||
|  |         val s = Snackbar | ||||||
|  |             .make( | ||||||
|  |                 app.findViewById(R.id.coordLayout), | ||||||
|  |                 R.string.marked_as_read, | ||||||
|  |                 Snackbar.LENGTH_LONG | ||||||
|  |             ) | ||||||
|  |             .setAction(R.string.undo_string) { | ||||||
|  |                 items.add(position, i) | ||||||
|  |                 thread { | ||||||
|  |                     db.itemsDao().insertAllItems(i.toEntity()) | ||||||
|  |                 } | ||||||
|  |                 notifyItemInserted(position) | ||||||
|  |                 updateItems(items) | ||||||
|  |  | ||||||
|  |                 if (app.isNetworkAccessible(null)) { | ||||||
|  |                     api.unmarkItem(i.id).enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                         override fun onResponse( | ||||||
|  |                             call: Call<SuccessResponse>, | ||||||
|  |                             response: Response<SuccessResponse> | ||||||
|  |                         ) { | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { | ||||||
|  |                             items.remove(i) | ||||||
|  |                             thread { | ||||||
|  |                                 db.itemsDao().delete(i.toEntity()) | ||||||
|  |                             } | ||||||
|  |                             notifyItemRemoved(position) | ||||||
|  |                             updateItems(items) | ||||||
|  |                         } | ||||||
|  |                     }) | ||||||
|  |                 } else { | ||||||
|  |                     thread { | ||||||
|  |                         db.actionsDao().insertAllActions(ActionEntity(i.id, false, true, false, false)) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         val view = s.view | ||||||
|  |         val tv: TextView = view.findViewById(com.google.android.material.R.id.snackbar_text) | ||||||
|  |         tv.setTextColor(Color.WHITE) | ||||||
|  |         s.show() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun markSnackbar(i: Item, position: Int) { | ||||||
|  |         val s = Snackbar | ||||||
|  |             .make( | ||||||
|  |                 app.findViewById(R.id.coordLayout), | ||||||
|  |                 R.string.marked_as_unread, | ||||||
|  |                 Snackbar.LENGTH_LONG | ||||||
|  |             ) | ||||||
|  |             .setAction(R.string.undo_string) { | ||||||
|  |                 items.add(position, i) | ||||||
|  |                 thread { | ||||||
|  |                     db.itemsDao().delete(i.toEntity()) | ||||||
|  |                 } | ||||||
|  |                 notifyItemInserted(position) | ||||||
|  |                 updateItems(items) | ||||||
|  |  | ||||||
|  |                 if (app.isNetworkAccessible(null)) { | ||||||
|  |                     api.markItem(i.id).enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                         override fun onResponse( | ||||||
|  |                             call: Call<SuccessResponse>, | ||||||
|  |                             response: Response<SuccessResponse> | ||||||
|  |                         ) { | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { | ||||||
|  |                             items.remove(i) | ||||||
|  |                             thread { | ||||||
|  |                                 db.itemsDao().insertAllItems(i.toEntity()) | ||||||
|  |                             } | ||||||
|  |                             notifyItemRemoved(position) | ||||||
|  |                             updateItems(items) | ||||||
|  |                         } | ||||||
|  |                     }) | ||||||
|  |                 } else { | ||||||
|  |                     thread { | ||||||
|  |                         db.actionsDao().insertAllActions(ActionEntity(i.id, true, false, false, false)) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         val view = s.view | ||||||
|  |         val tv: TextView = view.findViewById(com.google.android.material.R.id.snackbar_text) | ||||||
|  |         tv.setTextColor(Color.WHITE) | ||||||
|  |         s.show() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun handleItemAtIndex(position: Int) { | ||||||
|  |         if (unreadItemStatusAtIndex(position)) { | ||||||
|  |             readItemAtIndex(position) | ||||||
|  |         } else { | ||||||
|  |             unreadItemAtIndex(position) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun unreadItemStatusAtIndex(position: Int): Boolean { | ||||||
|  |         return items[position].unread | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun readItemAtIndex(position: Int) { | ||||||
|  |         val i = items[position] | ||||||
|  |         items.remove(i) | ||||||
|  |         notifyItemRemoved(position) | ||||||
|  |         updateItems(items) | ||||||
|  |  | ||||||
|  |         thread { | ||||||
|  |             db.itemsDao().delete(i.toEntity()) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (app.isNetworkAccessible(null)) { | ||||||
|  |             api.markItem(i.id).enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                 override fun onResponse( | ||||||
|  |                     call: Call<SuccessResponse>, | ||||||
|  |                     response: Response<SuccessResponse> | ||||||
|  |                 ) { | ||||||
|  |  | ||||||
|  |                     unmarkSnackbar(i, position) | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { | ||||||
|  |                     Toast.makeText( | ||||||
|  |                         app, | ||||||
|  |                         app.getString(R.string.cant_mark_read), | ||||||
|  |                         Toast.LENGTH_SHORT | ||||||
|  |                     ).show() | ||||||
|  |                     items.add(position, i) | ||||||
|  |                     notifyItemInserted(position) | ||||||
|  |                     updateItems(items) | ||||||
|  |  | ||||||
|  |                     thread { | ||||||
|  |                         db.itemsDao().insertAllItems(i.toEntity()) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|  |         } else { | ||||||
|  |             thread { | ||||||
|  |                 db.actionsDao().insertAllActions(ActionEntity(i.id, true, false, false, false)) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun unreadItemAtIndex(position: Int) { | ||||||
|  |         val i = items[position] | ||||||
|  |         items.remove(i) | ||||||
|  |         notifyItemRemoved(position) | ||||||
|  |         updateItems(items) | ||||||
|  |  | ||||||
|  |         thread { | ||||||
|  |             db.itemsDao().insertAllItems(i.toEntity()) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (app.isNetworkAccessible(null)) { | ||||||
|  |             api.unmarkItem(i.id).enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                 override fun onResponse( | ||||||
|  |                     call: Call<SuccessResponse>, | ||||||
|  |                     response: Response<SuccessResponse> | ||||||
|  |                 ) { | ||||||
|  |  | ||||||
|  |                     markSnackbar(i, position) | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { | ||||||
|  |                     Toast.makeText( | ||||||
|  |                         app, | ||||||
|  |                         app.getString(R.string.cant_mark_unread), | ||||||
|  |                         Toast.LENGTH_SHORT | ||||||
|  |                     ).show() | ||||||
|  |                     items.add(i) | ||||||
|  |                     notifyItemInserted(position) | ||||||
|  |                     updateItems(items) | ||||||
|  |  | ||||||
|  |                     thread { | ||||||
|  |                         db.itemsDao().delete(i.toEntity()) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|  |         } else { | ||||||
|  |             thread { | ||||||
|  |                 db.actionsDao().insertAllActions(ActionEntity(i.id, false, true, false, false)) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun addItemAtIndex(item: Item, position: Int) { | ||||||
|  |         items.add(position, item) | ||||||
|  |         notifyItemInserted(position) | ||||||
|  |         updateItems(items) | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun addItemsAtEnd(newItems: List<Item>) { | ||||||
|  |         val oldSize = items.size | ||||||
|  |         items.addAll(newItems) | ||||||
|  |         notifyItemRangeInserted(oldSize, newItems.size) | ||||||
|  |         updateItems(items) | ||||||
|  |  | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -2,17 +2,19 @@ package apps.amine.bou.readerforselfoss.adapters | |||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.support.constraint.ConstraintLayout | import androidx.constraintlayout.widget.ConstraintLayout | ||||||
| import android.support.v7.widget.RecyclerView | import androidx.recyclerview.widget.RecyclerView | ||||||
| import android.view.LayoutInflater | import android.view.LayoutInflater | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
| import android.widget.Button | import android.widget.Button | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Sources | import apps.amine.bou.readerforselfoss.api.selfoss.Source | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import apps.amine.bou.readerforselfoss.utils.glide.circularBitmapDrawable | import apps.amine.bou.readerforselfoss.utils.glide.circularBitmapDrawable | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.toTextDrawableString | import apps.amine.bou.readerforselfoss.utils.toTextDrawableString | ||||||
| import com.amulyakhare.textdrawable.TextDrawable | import com.amulyakhare.textdrawable.TextDrawable | ||||||
| import com.amulyakhare.textdrawable.util.ColorGenerator | import com.amulyakhare.textdrawable.util.ColorGenerator | ||||||
| @@ -22,39 +24,41 @@ import retrofit2.Callback | |||||||
| import retrofit2.Response | import retrofit2.Response | ||||||
|  |  | ||||||
| class SourcesListAdapter( | class SourcesListAdapter( | ||||||
|         private val app: Activity, |     private val app: Activity, | ||||||
|         private val items: ArrayList<Sources>, |     private val items: ArrayList<Source>, | ||||||
|         private val api: SelfossApi |     private val api: SelfossApi | ||||||
| ) : RecyclerView.Adapter<SourcesListAdapter.ViewHolder>() { | ) : RecyclerView.Adapter<SourcesListAdapter.ViewHolder>() { | ||||||
|     private val c: Context = app.baseContext |     private val c: Context = app.baseContext | ||||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL |     private val generator: ColorGenerator = ColorGenerator.MATERIAL | ||||||
|  |     private lateinit var config: Config | ||||||
|  |  | ||||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { |     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { | ||||||
|         val v = LayoutInflater.from(c).inflate( |         val v = LayoutInflater.from(c).inflate( | ||||||
|                 R.layout.source_list_item, |             R.layout.source_list_item, | ||||||
|                 parent, |             parent, | ||||||
|                 false |             false | ||||||
|         ) as ConstraintLayout |         ) as ConstraintLayout | ||||||
|         return ViewHolder(v) |         return ViewHolder(v) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onBindViewHolder(holder: ViewHolder, position: Int) { |     override fun onBindViewHolder(holder: ViewHolder, position: Int) { | ||||||
|         val itm = items[position] |         val itm = items[position] | ||||||
|  |         config = Config(c) | ||||||
|  |  | ||||||
|         if (itm.getIcon(c).isEmpty()) { |         if (itm.getIcon(c).isEmpty()) { | ||||||
|             val color = generator.getColor(itm.title) |             val color = generator.getColor(itm.getTitleDecoded()) | ||||||
|  |  | ||||||
|             val drawable = |             val drawable = | ||||||
|                     TextDrawable |                 TextDrawable | ||||||
|                             .builder() |                     .builder() | ||||||
|                             .round() |                     .round() | ||||||
|                             .build(itm.title.toTextDrawableString(), color) |                     .build(itm.getTitleDecoded().toTextDrawableString(c), color) | ||||||
|             holder.mView.itemImage.setImageDrawable(drawable) |             holder.mView.itemImage.setImageDrawable(drawable) | ||||||
|         } else { |         } else { | ||||||
|             c.circularBitmapDrawable(itm.getIcon(c), holder.mView.itemImage) |             c.circularBitmapDrawable(config, itm.getIcon(c), holder.mView.itemImage) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         holder.mView.sourceTitle.text = itm.title |         holder.mView.sourceTitle.text = itm.getTitleDecoded() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun getItemCount(): Int = items.size |     override fun getItemCount(): Int = items.size | ||||||
| @@ -70,33 +74,35 @@ class SourcesListAdapter( | |||||||
|             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) |             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) | ||||||
|  |  | ||||||
|             deleteBtn.setOnClickListener { |             deleteBtn.setOnClickListener { | ||||||
|                 val (id) = items[adapterPosition] |                 if (c.isNetworkAccessible(null)) { | ||||||
|                 api.deleteSource(id).enqueue(object : Callback<SuccessResponse> { |                     val (id) = items[adapterPosition] | ||||||
|                     override fun onResponse( |                     api.deleteSource(id).enqueue(object : Callback<SuccessResponse> { | ||||||
|  |                         override fun onResponse( | ||||||
|                             call: Call<SuccessResponse>, |                             call: Call<SuccessResponse>, | ||||||
|                             response: Response<SuccessResponse> |                             response: Response<SuccessResponse> | ||||||
|                     ) { |                         ) { | ||||||
|                         if (response.body() != null && response.body()!!.isSuccess) { |                             if (response.body() != null && response.body()!!.isSuccess) { | ||||||
|                             items.removeAt(adapterPosition) |                                 items.removeAt(adapterPosition) | ||||||
|                             notifyItemRemoved(adapterPosition) |                                 notifyItemRemoved(adapterPosition) | ||||||
|                             notifyItemRangeChanged(adapterPosition, itemCount) |                                 notifyItemRangeChanged(adapterPosition, itemCount) | ||||||
|                         } else { |                             } else { | ||||||
|                             Toast.makeText( |                                 Toast.makeText( | ||||||
|                                     app, |                                     app, | ||||||
|                                     R.string.can_delete_source, |                                     R.string.can_delete_source, | ||||||
|                                     Toast.LENGTH_SHORT |                                     Toast.LENGTH_SHORT | ||||||
|                             ).show() |                                 ).show() | ||||||
|  |                             } | ||||||
|                         } |                         } | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { |                         override fun onFailure(call: Call<SuccessResponse>, t: Throwable) { | ||||||
|                         Toast.makeText( |                             Toast.makeText( | ||||||
|                                 app, |                                 app, | ||||||
|                                 R.string.can_delete_source, |                                 R.string.can_delete_source, | ||||||
|                                 Toast.LENGTH_SHORT |                                 Toast.LENGTH_SHORT | ||||||
|                         ).show() |                             ).show() | ||||||
|                     } |                         } | ||||||
|                 }) |                     }) | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -7,33 +7,29 @@ import retrofit2.Call | |||||||
| import retrofit2.Retrofit | import retrofit2.Retrofit | ||||||
| import retrofit2.converter.gson.GsonConverterFactory | import retrofit2.converter.gson.GsonConverterFactory | ||||||
|  |  | ||||||
| class MercuryApi(private val key: String, shouldLog: Boolean) { | class MercuryApi() { | ||||||
|     private val service: MercuryService |     private val service: MercuryService | ||||||
|  |  | ||||||
|     init { |     init { | ||||||
|  |  | ||||||
|         val interceptor = HttpLoggingInterceptor() |         val interceptor = HttpLoggingInterceptor() | ||||||
|         interceptor.level = if (shouldLog) { |         interceptor.level = HttpLoggingInterceptor.Level.NONE | ||||||
|             HttpLoggingInterceptor.Level.BODY |  | ||||||
|         } else { |  | ||||||
|             HttpLoggingInterceptor.Level.NONE |  | ||||||
|         } |  | ||||||
|         val client = OkHttpClient.Builder().addInterceptor(interceptor).build() |         val client = OkHttpClient.Builder().addInterceptor(interceptor).build() | ||||||
|  |  | ||||||
|         val gson = GsonBuilder() |         val gson = GsonBuilder() | ||||||
|                 .setLenient() |             .setLenient() | ||||||
|                 .create() |             .create() | ||||||
|         val retrofit = |         val retrofit = | ||||||
|                 Retrofit |             Retrofit | ||||||
|                         .Builder() |                 .Builder() | ||||||
|                         .baseUrl("https://mercury.postlight.com") |                 .baseUrl("https://www.amine-bou.fr") | ||||||
|                         .client(client) |                 .client(client) | ||||||
|                         .addConverterFactory(GsonConverterFactory.create(gson)) |                 .addConverterFactory(GsonConverterFactory.create(gson)) | ||||||
|                         .build() |                 .build() | ||||||
|         service = retrofit.create(MercuryService::class.java) |         service = retrofit.create(MercuryService::class.java) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun parseUrl(url: String): Call<ParsedContent> { |     fun parseUrl(url: String): Call<ParsedContent> { | ||||||
|         return service.parseUrl(url, this.key) |         return service.parseUrl(url) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,39 +5,40 @@ import android.os.Parcelable | |||||||
| import com.google.gson.annotations.SerializedName | import com.google.gson.annotations.SerializedName | ||||||
|  |  | ||||||
| class ParsedContent( | class ParsedContent( | ||||||
|         @SerializedName("title") val title: String, |     @SerializedName("title") val title: String, | ||||||
|         @SerializedName("content") val content: String, |     @SerializedName("content") val content: String?, | ||||||
|         @SerializedName("date_published") val date_published: String, |     @SerializedName("date_published") val date_published: String, | ||||||
|         @SerializedName("lead_image_url") val lead_image_url: String, |     @SerializedName("lead_image_url") val lead_image_url: String?, | ||||||
|         @SerializedName("dek") val dek: String, |     @SerializedName("dek") val dek: String, | ||||||
|         @SerializedName("url") val url: String, |     @SerializedName("url") val url: String, | ||||||
|         @SerializedName("domain") val domain: String, |     @SerializedName("domain") val domain: String, | ||||||
|         @SerializedName("excerpt") val excerpt: String, |     @SerializedName("excerpt") val excerpt: String, | ||||||
|         @SerializedName("total_pages") val total_pages: Int, |     @SerializedName("total_pages") val total_pages: Int, | ||||||
|         @SerializedName("rendered_pages") val rendered_pages: Int, |     @SerializedName("rendered_pages") val rendered_pages: Int, | ||||||
|         @SerializedName("next_page_url") val next_page_url: String |     @SerializedName("next_page_url") val next_page_url: String | ||||||
| ) : Parcelable { | ) : Parcelable { | ||||||
|  |  | ||||||
|     companion object { |     companion object { | ||||||
|         @JvmField |         @JvmField | ||||||
|         val CREATOR: Parcelable.Creator<ParsedContent> = object : Parcelable.Creator<ParsedContent> { |         val CREATOR: Parcelable.Creator<ParsedContent> = | ||||||
|             override fun createFromParcel(source: Parcel): ParsedContent = ParsedContent(source) |             object : Parcelable.Creator<ParsedContent> { | ||||||
|             override fun newArray(size: Int): Array<ParsedContent?> = arrayOfNulls(size) |                 override fun createFromParcel(source: Parcel): ParsedContent = ParsedContent(source) | ||||||
|         } |                 override fun newArray(size: Int): Array<ParsedContent?> = arrayOfNulls(size) | ||||||
|  |             } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     constructor(source: Parcel) : this( |     constructor(source: Parcel) : this( | ||||||
|             title = source.readString(), |         title = source.readString().orEmpty(), | ||||||
|             content = source.readString(), |         content = source.readString(), | ||||||
|             date_published = source.readString(), |         date_published = source.readString().orEmpty(), | ||||||
|             lead_image_url = source.readString(), |         lead_image_url = source.readString(), | ||||||
|             dek = source.readString(), |         dek = source.readString().orEmpty(), | ||||||
|             url = source.readString(), |         url = source.readString().orEmpty(), | ||||||
|             domain = source.readString(), |         domain = source.readString().orEmpty(), | ||||||
|             excerpt = source.readString(), |         excerpt = source.readString().orEmpty(), | ||||||
|             total_pages = source.readInt(), |         total_pages = source.readInt(), | ||||||
|             rendered_pages = source.readInt(), |         rendered_pages = source.readInt(), | ||||||
|             next_page_url = source.readString() |         next_page_url = source.readString().orEmpty() | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     override fun describeContents() = 0 |     override fun describeContents() = 0 | ||||||
|   | |||||||
| @@ -6,6 +6,6 @@ import retrofit2.http.Header | |||||||
| import retrofit2.http.Query | import retrofit2.http.Query | ||||||
|  |  | ||||||
| interface MercuryService { | interface MercuryService { | ||||||
|     @GET("parser") |     @GET("parser.php") | ||||||
|     fun parseUrl(@Query("url") url: String, @Header("x-api-key") key: String): Call<ParsedContent> |     fun parseUrl(@Query("link") link: String): Call<ParsedContent> | ||||||
| } | } | ||||||
|   | |||||||
| @@ -10,13 +10,13 @@ internal class BooleanTypeAdapter : JsonDeserializer<Boolean> { | |||||||
|  |  | ||||||
|     @Throws(JsonParseException::class) |     @Throws(JsonParseException::class) | ||||||
|     override fun deserialize( |     override fun deserialize( | ||||||
|             json: JsonElement, |         json: JsonElement, | ||||||
|             typeOfT: Type, |         typeOfT: Type, | ||||||
|             context: JsonDeserializationContext |         context: JsonDeserializationContext | ||||||
|     ): Boolean? = |     ): Boolean? = | ||||||
|             try { |         try { | ||||||
|                 json.asInt == 1 |             json.asInt == 1 | ||||||
|             } catch (e: Exception) { |         } catch (e: Exception) { | ||||||
|                 json.asBoolean |             json.asBoolean | ||||||
|             } |         } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -12,18 +12,20 @@ import com.burgstaller.okhttp.digest.CachingAuthenticator | |||||||
| import com.burgstaller.okhttp.digest.Credentials | import com.burgstaller.okhttp.digest.Credentials | ||||||
| import com.burgstaller.okhttp.digest.DigestAuthenticator | import com.burgstaller.okhttp.digest.DigestAuthenticator | ||||||
| import com.google.gson.GsonBuilder | import com.google.gson.GsonBuilder | ||||||
| import okhttp3.OkHttpClient | import okhttp3.* | ||||||
| import okhttp3.logging.HttpLoggingInterceptor | import okhttp3.logging.HttpLoggingInterceptor | ||||||
| import retrofit2.Call | import retrofit2.Call | ||||||
| import retrofit2.Retrofit | import retrofit2.Retrofit | ||||||
| import retrofit2.converter.gson.GsonConverterFactory | import retrofit2.converter.gson.GsonConverterFactory | ||||||
|  | import java.net.SocketTimeoutException | ||||||
| import java.util.concurrent.ConcurrentHashMap | import java.util.concurrent.ConcurrentHashMap | ||||||
|  | import java.util.concurrent.TimeUnit | ||||||
|  |  | ||||||
| class SelfossApi( | class SelfossApi( | ||||||
|         c: Context, |     c: Context, | ||||||
|         callingActivity: Activity, |     callingActivity: Activity?, | ||||||
|         isWithSelfSignedCert: Boolean, |     isWithSelfSignedCert: Boolean, | ||||||
|         shouldLog: Boolean |     timeout: Long | ||||||
| ) { | ) { | ||||||
|  |  | ||||||
|     private lateinit var service: SelfossService |     private lateinit var service: SelfossService | ||||||
| @@ -32,25 +34,45 @@ class SelfossApi( | |||||||
|     private val password: String |     private val password: String | ||||||
|  |  | ||||||
|     fun OkHttpClient.Builder.maybeWithSelfSigned(isWithSelfSignedCert: Boolean): OkHttpClient.Builder = |     fun OkHttpClient.Builder.maybeWithSelfSigned(isWithSelfSignedCert: Boolean): OkHttpClient.Builder = | ||||||
|             if (isWithSelfSignedCert) { |         if (isWithSelfSignedCert) { | ||||||
|                 getUnsafeHttpClient() |             getUnsafeHttpClient() | ||||||
|             } else { |         } else { | ||||||
|                 this |             this | ||||||
|             } |         } | ||||||
|  |  | ||||||
|  |     fun OkHttpClient.Builder.maybeWithSettingsTimeout(timeout: Long): OkHttpClient.Builder = | ||||||
|  |         if (timeout != -1L) { | ||||||
|  |             this.readTimeout(timeout, TimeUnit.SECONDS) | ||||||
|  |                 .connectTimeout(timeout, TimeUnit.SECONDS) | ||||||
|  |         } else { | ||||||
|  |             this | ||||||
|  |         } | ||||||
|  |  | ||||||
|     fun Credentials.createAuthenticator(): DispatchingAuthenticator = |     fun Credentials.createAuthenticator(): DispatchingAuthenticator = | ||||||
|             DispatchingAuthenticator.Builder() |         DispatchingAuthenticator.Builder() | ||||||
|                     .with("digest", DigestAuthenticator(this)) |             .with("digest", DigestAuthenticator(this)) | ||||||
|                     .with("basic", BasicAuthenticator(this)) |             .with("basic", BasicAuthenticator(this)) | ||||||
|                     .build() |             .build() | ||||||
|  |  | ||||||
|     fun DispatchingAuthenticator.getHttpClien(isWithSelfSignedCert: Boolean): OkHttpClient.Builder { |     fun DispatchingAuthenticator.getHttpClien(isWithSelfSignedCert: Boolean, timeout: Long): OkHttpClient.Builder { | ||||||
|         val authCache = ConcurrentHashMap<String, CachingAuthenticator>() |         val authCache = ConcurrentHashMap<String, CachingAuthenticator>() | ||||||
|         return OkHttpClient |         return OkHttpClient | ||||||
|                 .Builder() |             .Builder() | ||||||
|                 .maybeWithSelfSigned(isWithSelfSignedCert) |             .maybeWithSettingsTimeout(timeout) | ||||||
|                 .authenticator(CachingAuthenticatorDecorator(this, authCache)) |             .maybeWithSelfSigned(isWithSelfSignedCert) | ||||||
|                 .addInterceptor(AuthenticationCacheInterceptor(authCache)) |             .authenticator(CachingAuthenticatorDecorator(this, authCache)) | ||||||
|  |             .addInterceptor(AuthenticationCacheInterceptor(authCache)) | ||||||
|  |             .addInterceptor(object: Interceptor { | ||||||
|  |                 override fun intercept(chain: Interceptor.Chain): Response { | ||||||
|  |                     val request: Request = chain.request() | ||||||
|  |                     val response: Response = chain.proceed(request) | ||||||
|  |  | ||||||
|  |                     if (response.code() == 408) { | ||||||
|  |                         return response | ||||||
|  |                     } | ||||||
|  |                     return response | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     init { |     init { | ||||||
| @@ -58,97 +80,122 @@ class SelfossApi( | |||||||
|         password = config.userPassword |         password = config.userPassword | ||||||
|  |  | ||||||
|         val authenticator = |         val authenticator = | ||||||
|                 Credentials( |             Credentials( | ||||||
|                         config.httpUserLogin, |                 config.httpUserLogin, | ||||||
|                         config.httpUserPassword |                 config.httpUserPassword | ||||||
|                 ).createAuthenticator() |             ).createAuthenticator() | ||||||
|  |  | ||||||
|         val gson = |         val gson = | ||||||
|                 GsonBuilder() |             GsonBuilder() | ||||||
|                         .registerTypeAdapter(Boolean::class.javaPrimitiveType, BooleanTypeAdapter()) |                 .registerTypeAdapter(Boolean::class.javaPrimitiveType, BooleanTypeAdapter()) | ||||||
|                         .setLenient() |                 .registerTypeAdapter(SelfossTagType::class.java, SelfossTagTypeTypeAdapter()) | ||||||
|                         .create() |                 .setLenient() | ||||||
|  |                 .create() | ||||||
|  |  | ||||||
|         val logging = HttpLoggingInterceptor() |         val logging = HttpLoggingInterceptor() | ||||||
|  |  | ||||||
|         logging.level = if (shouldLog) { |  | ||||||
|             HttpLoggingInterceptor.Level.BODY |  | ||||||
|         } else { |  | ||||||
|             HttpLoggingInterceptor.Level.NONE |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         val httpClient = authenticator.getHttpClien(isWithSelfSignedCert) |         logging.level = HttpLoggingInterceptor.Level.NONE | ||||||
|  |         val httpClient = authenticator.getHttpClien(isWithSelfSignedCert, timeout) | ||||||
|  |  | ||||||
|         httpClient.addInterceptor(logging) |         val timeoutCode = 504 | ||||||
|  |         httpClient | ||||||
|  |                 .addInterceptor { chain -> | ||||||
|  |                     val res = chain.proceed(chain.request()) | ||||||
|  |                     if (res.code() == timeoutCode) { | ||||||
|  |                         throw SocketTimeoutException("timeout") | ||||||
|  |                     } | ||||||
|  |                     res | ||||||
|  |                 } | ||||||
|  |                 .addInterceptor(logging) | ||||||
|  |                 .addInterceptor { chain -> | ||||||
|  |                     val request = chain.request() | ||||||
|  |                     try { | ||||||
|  |                         chain.proceed(request) | ||||||
|  |                     } catch (e: SocketTimeoutException) { | ||||||
|  |                         Response.Builder() | ||||||
|  |                                 .code(timeoutCode) | ||||||
|  |                                 .protocol(Protocol.HTTP_2) | ||||||
|  |                                 .body(ResponseBody.create(MediaType.parse("text/plain"), "")) | ||||||
|  |                                 .message("") | ||||||
|  |                                 .request(request) | ||||||
|  |                                 .build() | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|         try { |         try { | ||||||
|             val retrofit = |             val retrofit = | ||||||
|                     Retrofit |                 Retrofit | ||||||
|                             .Builder() |                     .Builder() | ||||||
|                             .baseUrl(config.baseUrl) |                     .baseUrl(config.baseUrl) | ||||||
|                             .client(httpClient.build()) |                     .client(httpClient.build()) | ||||||
|                             .addConverterFactory(GsonConverterFactory.create(gson)) |                     .addConverterFactory(GsonConverterFactory.create(gson)) | ||||||
|                             .build() |                     .build() | ||||||
|             service = retrofit.create(SelfossService::class.java) |             service = retrofit.create(SelfossService::class.java) | ||||||
|         } catch (e: IllegalArgumentException) { |         } catch (e: IllegalArgumentException) { | ||||||
|             Config.logoutAndRedirect(c, callingActivity, config.settings.edit(), baseUrlFail = true) |             if (callingActivity != null) { | ||||||
|  |                 Config.logoutAndRedirect(c, callingActivity, config.settings.edit(), baseUrlFail = true) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun login(): Call<SuccessResponse> = |     fun login(): Call<SuccessResponse> = | ||||||
|             service.loginToSelfoss(config.userLogin, config.userPassword) |         service.loginToSelfoss(config.userLogin, config.userPassword) | ||||||
|  |  | ||||||
|     fun readItems( |     fun readItems( | ||||||
|             tag: String?, |         tag: String?, | ||||||
|             sourceId: Long?, |         sourceId: Long?, | ||||||
|             search: String?, |         search: String?, | ||||||
|             itemsNumber: Int, |         itemsNumber: Int, | ||||||
|             offset: Int |         offset: Int | ||||||
|     ): Call<List<Item>> = |     ): Call<List<Item>> = | ||||||
|             getItems("read", tag, sourceId, search, itemsNumber, offset) |         getItems("read", tag, sourceId, search, itemsNumber, offset) | ||||||
|  |  | ||||||
|     fun newItems( |     fun newItems( | ||||||
|             tag: String?, |         tag: String?, | ||||||
|             sourceId: Long?, |         sourceId: Long?, | ||||||
|             search: String?, |         search: String?, | ||||||
|             itemsNumber: Int, |         itemsNumber: Int, | ||||||
|             offset: Int |         offset: Int | ||||||
|     ): Call<List<Item>> = |     ): Call<List<Item>> = | ||||||
|             getItems("unread", tag, sourceId, search, itemsNumber, offset) |         getItems("unread", tag, sourceId, search, itemsNumber, offset) | ||||||
|  |  | ||||||
|     fun starredItems( |     fun starredItems( | ||||||
|             tag: String?, |         tag: String?, | ||||||
|             sourceId: Long?, |         sourceId: Long?, | ||||||
|             search: String?, |         search: String?, | ||||||
|             itemsNumber: Int, |         itemsNumber: Int, | ||||||
|             offset: Int |         offset: Int | ||||||
|     ): Call<List<Item>> = |     ): Call<List<Item>> = | ||||||
|             getItems("starred", tag, sourceId, search, itemsNumber, offset) |         getItems("starred", tag, sourceId, search, itemsNumber, offset) | ||||||
|  |  | ||||||
|  |     fun allItems(): Call<List<Item>> = | ||||||
|  |         service.allItems(userName, password) | ||||||
|  |  | ||||||
|     private fun getItems( |     private fun getItems( | ||||||
|             type: String, |         type: String, | ||||||
|             tag: String?, |         tag: String?, | ||||||
|             sourceId: Long?, |         sourceId: Long?, | ||||||
|             search: String?, |         search: String?, | ||||||
|             items: Int, |         items: Int, | ||||||
|             offset: Int |         offset: Int | ||||||
|     ): Call<List<Item>> = |     ): Call<List<Item>> = | ||||||
|             service.getItems(type, tag, sourceId, search, userName, password, items, offset) |         service.getItems(type, tag, sourceId, search, userName, password, items, offset) | ||||||
|  |  | ||||||
|     fun markItem(itemId: String): Call<SuccessResponse> = |     fun markItem(itemId: String): Call<SuccessResponse> = | ||||||
|             service.markAsRead(itemId, userName, password) |         service.markAsRead(itemId, userName, password) | ||||||
|  |  | ||||||
|     fun unmarkItem(itemId: String): Call<SuccessResponse> = |     fun unmarkItem(itemId: String): Call<SuccessResponse> = | ||||||
|             service.unmarkAsRead(itemId, userName, password) |         service.unmarkAsRead(itemId, userName, password) | ||||||
|  |  | ||||||
|     fun readAll(ids: List<String>): Call<SuccessResponse> = |     fun readAll(ids: List<String>): Call<SuccessResponse> = | ||||||
|             service.markAllAsRead(ids, userName, password) |         service.markAllAsRead(ids, userName, password) | ||||||
|  |  | ||||||
|     fun starrItem(itemId: String): Call<SuccessResponse> = |     fun starrItem(itemId: String): Call<SuccessResponse> = | ||||||
|             service.starr(itemId, userName, password) |         service.starr(itemId, userName, password) | ||||||
|  |  | ||||||
|     fun unstarrItem(itemId: String): Call<SuccessResponse> = |     fun unstarrItem(itemId: String): Call<SuccessResponse> = | ||||||
|             service.unstarr(itemId, userName, password) |         service.unstarr(itemId, userName, password) | ||||||
|  |  | ||||||
|     val stats: Call<Stats> |     val stats: Call<Stats> | ||||||
|         get() = service.stats(userName, password) |         get() = service.stats(userName, password) | ||||||
| @@ -157,23 +204,23 @@ class SelfossApi( | |||||||
|         get() = service.tags(userName, password) |         get() = service.tags(userName, password) | ||||||
|  |  | ||||||
|     fun update(): Call<String> = |     fun update(): Call<String> = | ||||||
|             service.update(userName, password) |         service.update(userName, password) | ||||||
|  |  | ||||||
|     val sources: Call<List<Sources>> |     val sources: Call<List<Source>> | ||||||
|         get() = service.sources(userName, password) |         get() = service.sources(userName, password) | ||||||
|  |  | ||||||
|     fun deleteSource(id: String): Call<SuccessResponse> = |     fun deleteSource(id: String): Call<SuccessResponse> = | ||||||
|             service.deleteSource(id, userName, password) |         service.deleteSource(id, userName, password) | ||||||
|  |  | ||||||
|     fun spouts(): Call<Map<String, Spout>> = |     fun spouts(): Call<Map<String, Spout>> = | ||||||
|             service.spouts(userName, password) |         service.spouts(userName, password) | ||||||
|  |  | ||||||
|     fun createSource( |     fun createSource( | ||||||
|             title: String, |         title: String, | ||||||
|             url: String, |         url: String, | ||||||
|             spout: String, |         spout: String, | ||||||
|             tags: String, |         tags: String, | ||||||
|             filter: String |         filter: String | ||||||
|     ): Call<SuccessResponse> = |     ): Call<SuccessResponse> = | ||||||
|             service.createSource(title, url, spout, tags, filter, userName, password) |         service.createSource(title, url, spout, tags, filter, userName, password) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,26 +4,32 @@ import android.content.Context | |||||||
| import android.net.Uri | import android.net.Uri | ||||||
| import android.os.Parcel | import android.os.Parcel | ||||||
| import android.os.Parcelable | import android.os.Parcelable | ||||||
|  | import android.text.Html | ||||||
|  | import android.webkit.URLUtil | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  |  | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | ||||||
|  | import com.bumptech.glide.Glide | ||||||
|  | import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||||
|  | import com.bumptech.glide.request.RequestOptions | ||||||
| import com.google.gson.annotations.SerializedName | import com.google.gson.annotations.SerializedName | ||||||
|  |  | ||||||
| private fun constructUrl(config: Config?, path: String, file: String): String { | private fun constructUrl(config: Config?, path: String, file: String?): String { | ||||||
|     val baseUriBuilder = Uri.parse(config!!.baseUrl).buildUpon() |  | ||||||
|     baseUriBuilder.appendPath(path).appendPath(file) |  | ||||||
|  |  | ||||||
|     return if (file.isEmptyOrNullOrNullString()) { |     return if (file.isEmptyOrNullOrNullString()) { | ||||||
|         "" |         "" | ||||||
|     } else { |     } else { | ||||||
|  |         val baseUriBuilder = Uri.parse(config!!.baseUrl).buildUpon() | ||||||
|  |         baseUriBuilder.appendPath(path).appendPath(file) | ||||||
|  |  | ||||||
|         baseUriBuilder.toString() |         baseUriBuilder.toString() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| data class Tag( | data class Tag( | ||||||
|         @SerializedName("tag") val tag: String, |     @SerializedName("tag") val tag: String, | ||||||
|         @SerializedName("color") val color: String, |     @SerializedName("color") val color: String, | ||||||
|         @SerializedName("unread") val unread: Int |     @SerializedName("unread") val unread: Int | ||||||
| ) | ) | ||||||
|  |  | ||||||
| class SuccessResponse(@SerializedName("success") val success: Boolean) { | class SuccessResponse(@SerializedName("success") val success: Boolean) { | ||||||
| @@ -32,23 +38,23 @@ class SuccessResponse(@SerializedName("success") val success: Boolean) { | |||||||
| } | } | ||||||
|  |  | ||||||
| class Stats( | class Stats( | ||||||
|         @SerializedName("total") val total: Int, |     @SerializedName("total") val total: Int, | ||||||
|         @SerializedName("unread") val unread: Int, |     @SerializedName("unread") val unread: Int, | ||||||
|         @SerializedName("starred") val starred: Int |     @SerializedName("starred") val starred: Int | ||||||
| ) | ) | ||||||
|  |  | ||||||
| data class Spout( | data class Spout( | ||||||
|         @SerializedName("name") val name: String, |     @SerializedName("name") val name: String, | ||||||
|         @SerializedName("description") val description: String |     @SerializedName("description") val description: String | ||||||
| ) | ) | ||||||
|  |  | ||||||
| data class Sources( | data class Source( | ||||||
|         @SerializedName("id") val id: String, |     @SerializedName("id") val id: String, | ||||||
|         @SerializedName("title") val title: String, |     @SerializedName("title") val title: String, | ||||||
|         @SerializedName("tags") val tags: String, |     @SerializedName("tags") val tags: SelfossTagType, | ||||||
|         @SerializedName("spout") val spout: String, |     @SerializedName("spout") val spout: String, | ||||||
|         @SerializedName("error") val error: String, |     @SerializedName("error") val error: String, | ||||||
|         @SerializedName("icon") val icon: String |     @SerializedName("icon") val icon: String | ||||||
| ) { | ) { | ||||||
|     var config: Config? = null |     var config: Config? = null | ||||||
|  |  | ||||||
| @@ -58,19 +64,24 @@ data class Sources( | |||||||
|         } |         } | ||||||
|         return constructUrl(config, "favicons", icon) |         return constructUrl(config, "favicons", icon) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     fun getTitleDecoded(): String { | ||||||
|  |         return Html.fromHtml(title).toString() | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| data class Item( | data class Item( | ||||||
|         @SerializedName("id") val id: String, |     @SerializedName("id") val id: String, | ||||||
|         @SerializedName("datetime") val datetime: String, |     @SerializedName("datetime") val datetime: String, | ||||||
|         @SerializedName("title") val title: String, |     @SerializedName("title") val title: String, | ||||||
|         @SerializedName("content") val content: String, |     @SerializedName("content") val content: String, | ||||||
|         @SerializedName("unread") val unread: Boolean, |     @SerializedName("unread") val unread: Boolean, | ||||||
|         @SerializedName("starred") val starred: Boolean, |     @SerializedName("starred") var starred: Boolean, | ||||||
|         @SerializedName("thumbnail") val thumbnail: String, |     @SerializedName("thumbnail") val thumbnail: String?, | ||||||
|         @SerializedName("icon") val icon: String, |     @SerializedName("icon") val icon: String?, | ||||||
|         @SerializedName("link") val link: String, |     @SerializedName("link") val link: String, | ||||||
|         @SerializedName("sourcetitle") val sourcetitle: String |     @SerializedName("sourcetitle") val sourcetitle: String, | ||||||
|  |     @SerializedName("tags") val tags: SelfossTagType | ||||||
| ) : Parcelable { | ) : Parcelable { | ||||||
|  |  | ||||||
|     var config: Config? = null |     var config: Config? = null | ||||||
| @@ -83,16 +94,17 @@ data class Item( | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     constructor(source: Parcel) : this( |     constructor(source: Parcel) : this( | ||||||
|             id = source.readString(), |         id = source.readString().orEmpty(), | ||||||
|             datetime = source.readString(), |         datetime = source.readString().orEmpty(), | ||||||
|             title = source.readString(), |         title = source.readString().orEmpty(), | ||||||
|             content = source.readString(), |         content = source.readString().orEmpty(), | ||||||
|             unread = 0.toByte() != source.readByte(), |         unread = 0.toByte() != source.readByte(), | ||||||
|             starred = 0.toByte() != source.readByte(), |         starred = 0.toByte() != source.readByte(), | ||||||
|             thumbnail = source.readString(), |         thumbnail = source.readString(), | ||||||
|             icon = source.readString(), |         icon = source.readString(), | ||||||
|             link = source.readString(), |         link = source.readString().orEmpty(), | ||||||
|             sourcetitle = source.readString() |         sourcetitle = source.readString().orEmpty(), | ||||||
|  |         tags = if (source.readParcelable<SelfossTagType>(ClassLoader.getSystemClassLoader()) != null) source.readParcelable(ClassLoader.getSystemClassLoader())!! else SelfossTagType("") | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     override fun describeContents() = 0 |     override fun describeContents() = 0 | ||||||
| @@ -108,6 +120,7 @@ data class Item( | |||||||
|         dest.writeString(icon) |         dest.writeString(icon) | ||||||
|         dest.writeString(link) |         dest.writeString(link) | ||||||
|         dest.writeString(sourcetitle) |         dest.writeString(sourcetitle) | ||||||
|  |         dest.writeParcelable(tags, flags) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getIcon(app: Context): String { |     fun getIcon(app: Context): String { | ||||||
| @@ -124,18 +137,57 @@ data class Item( | |||||||
|         return constructUrl(config, "thumbnails", thumbnail) |         return constructUrl(config, "thumbnails", thumbnail) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     fun getImages() : ArrayList<String> { | ||||||
|  |         var allImages = ArrayList<String>() | ||||||
|  |  | ||||||
|  |         for ( image in Jsoup.parse(content).getElementsByTag("img")) { | ||||||
|  |             allImages.add(image.attr("src")) | ||||||
|  |         } | ||||||
|  |         return allImages | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun preloadImages(context: Context) : Boolean { | ||||||
|  |         val imageUrls = this.getImages() | ||||||
|  |  | ||||||
|  |         val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         try { | ||||||
|  |             for (url in imageUrls) { | ||||||
|  |                 if ( URLUtil.isValidUrl(url)) { | ||||||
|  |                     val image = Glide.with(context).asBitmap() | ||||||
|  |                             .apply(glideOptions) | ||||||
|  |                             .load(url).submit().get() | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } catch (e : Error) { | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getTitleDecoded(): String { | ||||||
|  |         return Html.fromHtml(title).toString() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getSourceTitle(): String { | ||||||
|  |         return Html.fromHtml(sourcetitle).toString() | ||||||
|  |     } | ||||||
|  |  | ||||||
|     // TODO: maybe find a better way to handle these kind of urls |     // TODO: maybe find a better way to handle these kind of urls | ||||||
|     fun getLinkDecoded(): String { |     fun getLinkDecoded(): String { | ||||||
|         var stringUrl: String |         var stringUrl: String | ||||||
|         stringUrl = if (link.startsWith("http://news.google.com/news/") || link.startsWith("https://news.google.com/news/")) { |         stringUrl = | ||||||
|             if (link.contains("&url=")) { |                 if (link.startsWith("http://news.google.com/news/") || link.startsWith("https://news.google.com/news/")) { | ||||||
|                 link.substringAfter("&url=") |                     if (link.contains("&url=")) { | ||||||
|             } else { |                         link.substringAfter("&url=") | ||||||
|                 this.link.replace("&", "&") |                     } else { | ||||||
|             } |                         this.link.replace("&", "&") | ||||||
|         } else { |                     } | ||||||
|             this.link.replace("&", "&") |                 } else { | ||||||
|         } |                     this.link.replace("&", "&") | ||||||
|  |                 } | ||||||
|  |  | ||||||
|         // handle :443 => https |         // handle :443 => https | ||||||
|         if (stringUrl.contains(":443")) { |         if (stringUrl.contains(":443")) { | ||||||
| @@ -144,9 +196,32 @@ data class Item( | |||||||
|  |  | ||||||
|         // handle url not starting with http |         // handle url not starting with http | ||||||
|         if (stringUrl.startsWith("//")) { |         if (stringUrl.startsWith("//")) { | ||||||
|             stringUrl = "http:" + stringUrl |             stringUrl = "http:$stringUrl" | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return stringUrl |         return stringUrl | ||||||
|     } |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | data class SelfossTagType(val tags: String) : Parcelable { | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         @JvmField val CREATOR: Parcelable.Creator<SelfossTagType> = | ||||||
|  |             object : Parcelable.Creator<SelfossTagType> { | ||||||
|  |                 override fun createFromParcel(source: Parcel): SelfossTagType = | ||||||
|  |                     SelfossTagType(source) | ||||||
|  |  | ||||||
|  |                 override fun newArray(size: Int): Array<SelfossTagType?> = arrayOfNulls(size) | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     constructor(source: Parcel) : this( | ||||||
|  |         tags = source.readString().orEmpty() | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     override fun describeContents() = 0 | ||||||
|  |  | ||||||
|  |     override fun writeToParcel(dest: Parcel, flags: Int) { | ||||||
|  |         dest.writeString(tags) | ||||||
|  |     } | ||||||
| } | } | ||||||
| @@ -17,102 +17,108 @@ internal interface SelfossService { | |||||||
|  |  | ||||||
|     @GET("items") |     @GET("items") | ||||||
|     fun getItems( |     fun getItems( | ||||||
|             @Query("type") type: String, |         @Query("type") type: String, | ||||||
|             @Query("tag") tag: String?, |         @Query("tag") tag: String?, | ||||||
|             @Query("source") source: Long?, |         @Query("source") source: Long?, | ||||||
|             @Query("search") search: String?, |         @Query("search") search: String?, | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String, |         @Query("password") password: String, | ||||||
|             @Query("items") items: Int, |         @Query("items") items: Int, | ||||||
|             @Query("offset") offset: Int |         @Query("offset") offset: Int | ||||||
|  |     ): Call<List<Item>> | ||||||
|  |  | ||||||
|  |     @GET("items") | ||||||
|  |     fun allItems( | ||||||
|  |         @Query("username") username: String, | ||||||
|  |         @Query("password") password: String | ||||||
|     ): Call<List<Item>> |     ): Call<List<Item>> | ||||||
|  |  | ||||||
|     @Headers("Content-Type: application/x-www-form-urlencoded") |     @Headers("Content-Type: application/x-www-form-urlencoded") | ||||||
|     @POST("mark/{id}") |     @POST("mark/{id}") | ||||||
|     fun markAsRead( |     fun markAsRead( | ||||||
|             @Path("id") id: String, |         @Path("id") id: String, | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<SuccessResponse> |     ): Call<SuccessResponse> | ||||||
|  |  | ||||||
|     @Headers("Content-Type: application/x-www-form-urlencoded") |     @Headers("Content-Type: application/x-www-form-urlencoded") | ||||||
|     @POST("unmark/{id}") |     @POST("unmark/{id}") | ||||||
|     fun unmarkAsRead( |     fun unmarkAsRead( | ||||||
|             @Path("id") id: String, |         @Path("id") id: String, | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<SuccessResponse> |     ): Call<SuccessResponse> | ||||||
|  |  | ||||||
|     @FormUrlEncoded |     @FormUrlEncoded | ||||||
|     @POST("mark") |     @POST("mark") | ||||||
|     fun markAllAsRead( |     fun markAllAsRead( | ||||||
|             @Field("ids[]") ids: List<String>, |         @Field("ids[]") ids: List<String>, | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<SuccessResponse> |     ): Call<SuccessResponse> | ||||||
|  |  | ||||||
|     @Headers("Content-Type: application/x-www-form-urlencoded") |     @Headers("Content-Type: application/x-www-form-urlencoded") | ||||||
|     @POST("starr/{id}") |     @POST("starr/{id}") | ||||||
|     fun starr( |     fun starr( | ||||||
|             @Path("id") id: String, |         @Path("id") id: String, | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<SuccessResponse> |     ): Call<SuccessResponse> | ||||||
|  |  | ||||||
|     @Headers("Content-Type: application/x-www-form-urlencoded") |     @Headers("Content-Type: application/x-www-form-urlencoded") | ||||||
|     @POST("unstarr/{id}") |     @POST("unstarr/{id}") | ||||||
|     fun unstarr( |     fun unstarr( | ||||||
|             @Path("id") id: String, |         @Path("id") id: String, | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<SuccessResponse> |     ): Call<SuccessResponse> | ||||||
|  |  | ||||||
|     @GET("stats") |     @GET("stats") | ||||||
|     fun stats( |     fun stats( | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<Stats> |     ): Call<Stats> | ||||||
|  |  | ||||||
|     @GET("tags") |     @GET("tags") | ||||||
|     fun tags( |     fun tags( | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<List<Tag>> |     ): Call<List<Tag>> | ||||||
|  |  | ||||||
|     @GET("update") |     @GET("update") | ||||||
|     fun update( |     fun update( | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<String> |     ): Call<String> | ||||||
|  |  | ||||||
|     @GET("sources/spouts") |     @GET("sources/spouts") | ||||||
|     fun spouts( |     fun spouts( | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<Map<String, Spout>> |     ): Call<Map<String, Spout>> | ||||||
|  |  | ||||||
|     @GET("sources/list") |     @GET("sources/list") | ||||||
|     fun sources( |     fun sources( | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<List<Sources>> |     ): Call<List<Source>> | ||||||
|  |  | ||||||
|     @DELETE("source/{id}") |     @DELETE("source/{id}") | ||||||
|     fun deleteSource( |     fun deleteSource( | ||||||
|             @Path("id") id: String, |         @Path("id") id: String, | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<SuccessResponse> |     ): Call<SuccessResponse> | ||||||
|  |  | ||||||
|     @FormUrlEncoded |     @FormUrlEncoded | ||||||
|     @POST("source") |     @POST("source") | ||||||
|     fun createSource( |     fun createSource( | ||||||
|             @Field("title") title: String, |         @Field("title") title: String, | ||||||
|             @Field("url") url: String, |         @Field("url") url: String, | ||||||
|             @Field("spout") spout: String, |         @Field("spout") spout: String, | ||||||
|             @Field("tags") tags: String, |         @Field("tags") tags: String, | ||||||
|             @Field("filter") filter: String, |         @Field("filter") filter: String, | ||||||
|             @Query("username") username: String, |         @Query("username") username: String, | ||||||
|             @Query("password") password: String |         @Query("password") password: String | ||||||
|     ): Call<SuccessResponse> |     ): Call<SuccessResponse> | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,22 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.api.selfoss | ||||||
|  |  | ||||||
|  | import com.google.gson.JsonDeserializationContext | ||||||
|  | import com.google.gson.JsonDeserializer | ||||||
|  | import com.google.gson.JsonElement | ||||||
|  | import com.google.gson.JsonParseException | ||||||
|  | import java.lang.reflect.Type | ||||||
|  |  | ||||||
|  | internal class SelfossTagTypeTypeAdapter : JsonDeserializer<SelfossTagType> { | ||||||
|  |  | ||||||
|  |     @Throws(JsonParseException::class) | ||||||
|  |     override fun deserialize( | ||||||
|  |         json: JsonElement, | ||||||
|  |         typeOfT: Type, | ||||||
|  |         context: JsonDeserializationContext | ||||||
|  |     ): SelfossTagType? = | ||||||
|  |         if (json.isJsonArray) { | ||||||
|  |             SelfossTagType(json.asJsonArray.joinToString(",") { it.toString() }) | ||||||
|  |         } else { | ||||||
|  |             SelfossTagType(json.toString()) | ||||||
|  |         } | ||||||
|  | } | ||||||
| @@ -0,0 +1,150 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.background | ||||||
|  |  | ||||||
|  | import android.app.NotificationManager | ||||||
|  | import android.app.PendingIntent | ||||||
|  | import android.content.Context | ||||||
|  | import android.content.Intent | ||||||
|  | import android.preference.PreferenceManager | ||||||
|  | import androidx.core.app.NotificationCompat | ||||||
|  | import androidx.core.app.NotificationCompat.PRIORITY_DEFAULT | ||||||
|  | import androidx.core.app.NotificationCompat.PRIORITY_LOW | ||||||
|  | import androidx.room.Room | ||||||
|  | import androidx.work.Worker | ||||||
|  | import androidx.work.WorkerParameters | ||||||
|  | import apps.amine.bou.readerforselfoss.MainActivity | ||||||
|  | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_2_3 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_3_4 | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | ||||||
|  | import retrofit2.Call | ||||||
|  | import retrofit2.Callback | ||||||
|  | import retrofit2.Response | ||||||
|  | import java.util.* | ||||||
|  | import kotlin.concurrent.schedule | ||||||
|  | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
|  | class LoadingWorker(val context: Context, params: WorkerParameters) : Worker(context, params) { | ||||||
|  |     lateinit var db: AppDatabase | ||||||
|  |  | ||||||
|  |     override fun doWork(): Result { | ||||||
|  |         if (context.isNetworkAccessible(null)) { | ||||||
|  |  | ||||||
|  |             val notificationManager = | ||||||
|  |                 applicationContext.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager | ||||||
|  |  | ||||||
|  |             val notification = NotificationCompat.Builder(applicationContext, Config.syncChannelId) | ||||||
|  |                 .setContentTitle(context.getString(R.string.loading_notification_title)) | ||||||
|  |                 .setContentText(context.getString(R.string.loading_notification_text)) | ||||||
|  |                 .setOngoing(true) | ||||||
|  |                 .setPriority(PRIORITY_LOW) | ||||||
|  |                 .setChannelId(Config.syncChannelId) | ||||||
|  |                 .setSmallIcon(R.drawable.ic_stat_cloud_download_black_24dp) | ||||||
|  |  | ||||||
|  |             notificationManager.notify(1, notification.build()) | ||||||
|  |  | ||||||
|  |             val settings = | ||||||
|  |                 this.context.getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) | ||||||
|  |             val sharedPref = PreferenceManager.getDefaultSharedPreferences(this.context) | ||||||
|  |             val notifyNewItems = sharedPref.getBoolean("notify_new_items", false) | ||||||
|  |  | ||||||
|  |             db = Room.databaseBuilder( | ||||||
|  |                 applicationContext, | ||||||
|  |                 AppDatabase::class.java, "selfoss-database" | ||||||
|  |             ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).addMigrations(MIGRATION_3_4).build() | ||||||
|  |  | ||||||
|  |             val api = SelfossApi( | ||||||
|  |                 this.context, | ||||||
|  |                 null, | ||||||
|  |                 settings.getBoolean("isSelfSignedCert", false), | ||||||
|  |                 sharedPref.getString("api_timeout", "-1")!!.toLong() | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             api.allItems().enqueue(object : Callback<List<Item>> { | ||||||
|  |                 override fun onFailure(call: Call<List<Item>>, t: Throwable) { | ||||||
|  |                     Timer("", false).schedule(4000) { | ||||||
|  |                         notificationManager.cancel(1) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 override fun onResponse( | ||||||
|  |                     call: Call<List<Item>>, | ||||||
|  |                     response: Response<List<Item>> | ||||||
|  |                 ) { | ||||||
|  |                     thread { | ||||||
|  |                         if (response.body() != null) { | ||||||
|  |                             val apiItems = (response.body() as ArrayList<Item>) | ||||||
|  |                             db.itemsDao().deleteAllItems() | ||||||
|  |                             db.itemsDao() | ||||||
|  |                                 .insertAllItems(*(apiItems.map { it.toEntity() }).toTypedArray()) | ||||||
|  |  | ||||||
|  |                             val newSize = apiItems.filter { it.unread }.size | ||||||
|  |                             if (notifyNewItems && newSize > 0) { | ||||||
|  |  | ||||||
|  |                                 val intent = Intent(context, MainActivity::class.java).apply { | ||||||
|  |                                     flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK | ||||||
|  |                                 } | ||||||
|  |                                 val pendingIntent: PendingIntent = PendingIntent.getActivity(context, 0, intent, 0) | ||||||
|  |  | ||||||
|  |                                 val newItemsNotification = NotificationCompat.Builder(applicationContext, Config.newItemsChannelId) | ||||||
|  |                                     .setContentTitle(context.getString(R.string.new_items_notification_title)) | ||||||
|  |                                     .setContentText(context.getString(R.string.new_items_notification_text, newSize)) | ||||||
|  |                                     .setPriority(PRIORITY_DEFAULT) | ||||||
|  |                                     .setChannelId(Config.newItemsChannelId) | ||||||
|  |                                     .setContentIntent(pendingIntent) | ||||||
|  |                                     .setAutoCancel(true) | ||||||
|  |                                     .setSmallIcon(R.drawable.ic_tab_fiber_new_black_24dp) | ||||||
|  |  | ||||||
|  |                                 Timer("", false).schedule(4000) { | ||||||
|  |                                     notificationManager.notify(2, newItemsNotification.build()) | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                             apiItems.map {it.preloadImages(context)} | ||||||
|  |                         } | ||||||
|  |                         Timer("", false).schedule(4000) { | ||||||
|  |                             notificationManager.cancel(1) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|  |             thread { | ||||||
|  |                 val actions = db.actionsDao().actions() | ||||||
|  |  | ||||||
|  |                 actions.forEach { action -> | ||||||
|  |                     when { | ||||||
|  |                         action.read -> doAndReportOnFail(api.markItem(action.articleId), action) | ||||||
|  |                         action.unread -> doAndReportOnFail(api.unmarkItem(action.articleId), action) | ||||||
|  |                         action.starred -> doAndReportOnFail(api.starrItem(action.articleId), action) | ||||||
|  |                         action.unstarred -> doAndReportOnFail( | ||||||
|  |                             api.unstarrItem(action.articleId), | ||||||
|  |                             action | ||||||
|  |                         ) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return Result.success() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun <T> doAndReportOnFail(call: Call<T>, action: ActionEntity) { | ||||||
|  |         call.enqueue(object : Callback<T> { | ||||||
|  |             override fun onResponse( | ||||||
|  |                 call: Call<T>, | ||||||
|  |                 response: Response<T> | ||||||
|  |             ) { | ||||||
|  |                 thread { | ||||||
|  |                     db.actionsDao().delete(action) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             override fun onFailure(call: Call<T>, t: Throwable) { | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,592 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.fragments | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import android.content.Intent | ||||||
|  | import android.content.SharedPreferences | ||||||
|  | import android.content.res.ColorStateList | ||||||
|  | import android.content.res.TypedArray | ||||||
|  | import android.graphics.Bitmap | ||||||
|  | import android.graphics.Typeface | ||||||
|  | import android.graphics.drawable.ColorDrawable | ||||||
|  | import android.net.Uri | ||||||
|  | import android.os.Build | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.preference.PreferenceManager | ||||||
|  | import android.view.* | ||||||
|  | import android.webkit.* | ||||||
|  | import androidx.browser.customtabs.CustomTabsIntent | ||||||
|  | import com.google.android.material.floatingactionbutton.FloatingActionButton | ||||||
|  | import androidx.fragment.app.Fragment | ||||||
|  | import androidx.core.content.ContextCompat | ||||||
|  | import androidx.core.widget.NestedScrollView | ||||||
|  | import androidx.appcompat.app.AlertDialog | ||||||
|  | import androidx.core.content.res.ResourcesCompat | ||||||
|  | import androidx.room.Room | ||||||
|  | import apps.amine.bou.readerforselfoss.ImageActivity | ||||||
|  | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import apps.amine.bou.readerforselfoss.api.mercury.MercuryApi | ||||||
|  | import apps.amine.bou.readerforselfoss.api.mercury.ParsedContent | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_2_3 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_3_4 | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.glide.loadMaybeBasicAuth | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.glide.getBitmapInputStream | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.openItemUrl | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.shareLink | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.succeeded | ||||||
|  | import com.bumptech.glide.Glide | ||||||
|  | import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||||
|  | import com.bumptech.glide.request.RequestOptions | ||||||
|  | import com.github.rubensousa.floatingtoolbar.FloatingToolbar | ||||||
|  | import kotlinx.android.synthetic.main.fragment_article.view.* | ||||||
|  | import retrofit2.Call | ||||||
|  | import retrofit2.Callback | ||||||
|  | import retrofit2.Response | ||||||
|  | import java.net.MalformedURLException | ||||||
|  | import java.net.URL | ||||||
|  | import java.util.concurrent.ExecutionException | ||||||
|  | import kotlin.collections.ArrayList | ||||||
|  | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
|  | class ArticleFragment : Fragment() { | ||||||
|  |     private lateinit var pageNumber: Number | ||||||
|  |     private var fontSize: Int = 16 | ||||||
|  |     private lateinit var allItems: ArrayList<Item> | ||||||
|  |     private var mCustomTabActivityHelper: CustomTabActivityHelper? = null; | ||||||
|  |     private lateinit var url: String | ||||||
|  |     private lateinit var contentText: String | ||||||
|  |     private lateinit var contentSource: String | ||||||
|  |     private lateinit var contentImage: String | ||||||
|  |     private lateinit var contentTitle: String | ||||||
|  |     private lateinit var allImages : ArrayList<String> | ||||||
|  |     private lateinit var editor: SharedPreferences.Editor | ||||||
|  |     private lateinit var fab: FloatingActionButton | ||||||
|  |     private lateinit var appColors: AppColors | ||||||
|  |     private lateinit var db: AppDatabase | ||||||
|  |     private lateinit var textAlignment: String | ||||||
|  |     private lateinit var config: Config | ||||||
|  |  | ||||||
|  |     private var rootView: ViewGroup? = null | ||||||
|  |  | ||||||
|  |     private lateinit var prefs: SharedPreferences | ||||||
|  |  | ||||||
|  |     private var typeface: Typeface? = null | ||||||
|  |     private var resId: Int = 0 | ||||||
|  |     private var font = "" | ||||||
|  |  | ||||||
|  |     override fun onStop() { | ||||||
|  |         super.onStop() | ||||||
|  |         if (mCustomTabActivityHelper != null) { | ||||||
|  |             mCustomTabActivityHelper!!.unbindCustomTabsService(activity) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         appColors = AppColors(activity!!) | ||||||
|  |         config = Config(activity!!) | ||||||
|  |  | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |  | ||||||
|  |         pageNumber = arguments!!.getInt(ARG_POSITION) | ||||||
|  |         allItems = arguments!!.getParcelableArrayList<Item>(ARG_ITEMS) as ArrayList<Item> | ||||||
|  |  | ||||||
|  |         db = Room.databaseBuilder( | ||||||
|  |             context!!, | ||||||
|  |             AppDatabase::class.java, "selfoss-database" | ||||||
|  |         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).addMigrations(MIGRATION_3_4).build() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onCreateView( | ||||||
|  |         inflater: LayoutInflater, | ||||||
|  |         container: ViewGroup?, | ||||||
|  |         savedInstanceState: Bundle? | ||||||
|  |     ): View? { | ||||||
|  |         try { | ||||||
|  |             rootView = inflater | ||||||
|  |                 .inflate(R.layout.fragment_article, container, false) as ViewGroup | ||||||
|  |  | ||||||
|  |             url = allItems[pageNumber.toInt()].getLinkDecoded() | ||||||
|  |             contentText = allItems[pageNumber.toInt()].content | ||||||
|  |             contentTitle = allItems[pageNumber.toInt()].getTitleDecoded() | ||||||
|  |             contentImage = allItems[pageNumber.toInt()].getThumbnail(activity!!) | ||||||
|  |             contentSource = allItems[pageNumber.toInt()].sourceAndDateText() | ||||||
|  |             allImages = allItems[pageNumber.toInt()].getImages() | ||||||
|  |  | ||||||
|  |             prefs = PreferenceManager.getDefaultSharedPreferences(activity) | ||||||
|  |             editor = prefs.edit() | ||||||
|  |             fontSize = prefs.getString("reader_font_size", "16")!!.toInt() | ||||||
|  |  | ||||||
|  |             font = prefs.getString("reader_font", "")!! | ||||||
|  |             if (font.isNotEmpty()) { | ||||||
|  |                 resId = context!!.resources.getIdentifier(font, "font", context!!.packageName) | ||||||
|  |                 typeface = try { | ||||||
|  |                     ResourcesCompat.getFont(context!!, resId)!! | ||||||
|  |                 } catch (e: java.lang.Exception) { | ||||||
|  |                     // ACRA.getErrorReporter().maybeHandleSilentException(Throwable("Font loading issue: ${e.message}"), context!!) | ||||||
|  |                     // Just to be sure | ||||||
|  |                     null | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             refreshAlignment() | ||||||
|  |  | ||||||
|  |             val settings = activity!!.getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) | ||||||
|  |  | ||||||
|  |             val api = SelfossApi( | ||||||
|  |                 context!!, | ||||||
|  |                 activity!!, | ||||||
|  |                 settings.getBoolean("isSelfSignedCert", false), | ||||||
|  |                 prefs.getString("api_timeout", "-1")!!.toLong() | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             fab = rootView!!.fab | ||||||
|  |  | ||||||
|  |             fab.backgroundTintList = ColorStateList.valueOf(appColors.colorAccent) | ||||||
|  |  | ||||||
|  |             fab.rippleColor = appColors.colorAccentDark | ||||||
|  |  | ||||||
|  |             val floatingToolbar: FloatingToolbar = rootView!!.floatingToolbar | ||||||
|  |             floatingToolbar.attachFab(fab) | ||||||
|  |  | ||||||
|  |             floatingToolbar.background = ColorDrawable(appColors.colorAccent) | ||||||
|  |  | ||||||
|  |             val customTabsIntent = activity!!.buildCustomTabsIntent() | ||||||
|  |             mCustomTabActivityHelper = CustomTabActivityHelper() | ||||||
|  |             mCustomTabActivityHelper!!.bindCustomTabsService(activity) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |             floatingToolbar.setClickListener( | ||||||
|  |                 object : FloatingToolbar.ItemClickListener { | ||||||
|  |                     override fun onItemClick(item: MenuItem) { | ||||||
|  |                         when (item.itemId) { | ||||||
|  |                             R.id.more_action -> getContentFromMercury(customTabsIntent, prefs) | ||||||
|  |                             R.id.share_action -> activity!!.shareLink(url, contentTitle) | ||||||
|  |                             R.id.open_action -> activity!!.openItemUrl( | ||||||
|  |                                 allItems, | ||||||
|  |                                 pageNumber.toInt(), | ||||||
|  |                                 url, | ||||||
|  |                                 customTabsIntent, | ||||||
|  |                                 false, | ||||||
|  |                                 false, | ||||||
|  |                                 activity!! | ||||||
|  |                             ) | ||||||
|  |                             R.id.unread_action -> if ((context != null && context!!.isNetworkAccessible(null)) || context == null) { | ||||||
|  |                                 api.unmarkItem(allItems[pageNumber.toInt()].id).enqueue( | ||||||
|  |                                     object : Callback<SuccessResponse> { | ||||||
|  |                                         override fun onResponse( | ||||||
|  |                                             call: Call<SuccessResponse>, | ||||||
|  |                                             response: Response<SuccessResponse> | ||||||
|  |                                         ) { | ||||||
|  |                                         } | ||||||
|  |  | ||||||
|  |                                         override fun onFailure( | ||||||
|  |                                             call: Call<SuccessResponse>, | ||||||
|  |                                             t: Throwable | ||||||
|  |                                         ) { | ||||||
|  |                                         } | ||||||
|  |                                     } | ||||||
|  |                                 ) | ||||||
|  |                             } else { | ||||||
|  |                                 thread { | ||||||
|  |                                     db.actionsDao().insertAllActions(ActionEntity(allItems[pageNumber.toInt()].id, false, true, false, false)) | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                             else -> Unit | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     override fun onItemLongClick(item: MenuItem?) { | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             rootView!!.source.text = contentSource | ||||||
|  |             if (typeface != null) { | ||||||
|  |                 rootView!!.source.typeface = typeface | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (contentText.isEmptyOrNullOrNullString()) { | ||||||
|  |                 getContentFromMercury(customTabsIntent, prefs) | ||||||
|  |             } else { | ||||||
|  |                 rootView!!.titleView.text = contentTitle | ||||||
|  |                 if (typeface != null) { | ||||||
|  |                     rootView!!.titleView.typeface = typeface | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 htmlToWebview() | ||||||
|  |  | ||||||
|  |                 if (!contentImage.isEmptyOrNullOrNullString() && context != null) { | ||||||
|  |                     rootView!!.imageView.visibility = View.VISIBLE | ||||||
|  |                     Glide | ||||||
|  |                         .with(context!!) | ||||||
|  |                         .asBitmap() | ||||||
|  |                         .loadMaybeBasicAuth(config, contentImage) | ||||||
|  |                         .apply(RequestOptions.fitCenterTransform()) | ||||||
|  |                         .into(rootView!!.imageView) | ||||||
|  |                 } else { | ||||||
|  |                     rootView!!.imageView.visibility = View.GONE | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             rootView!!.nestedScrollView.setOnScrollChangeListener( | ||||||
|  |                 NestedScrollView.OnScrollChangeListener { _, _, scrollY, _, oldScrollY -> | ||||||
|  |                     if (scrollY > oldScrollY) { | ||||||
|  |                         fab.hide() | ||||||
|  |                     } else { | ||||||
|  |                         if (floatingToolbar.isShowing) floatingToolbar.hide() else fab.show() | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |         } catch (e: InflateException) { | ||||||
|  |             AlertDialog.Builder(context!!) | ||||||
|  |                 .setMessage(context!!.getString(R.string.webview_dialog_issue_message)) | ||||||
|  |                 .setTitle(context!!.getString(R.string.webview_dialog_issue_title)) | ||||||
|  |                 .setPositiveButton(android.R.string.ok | ||||||
|  |                 ) { dialog, which -> | ||||||
|  |                     val sharedPref = PreferenceManager.getDefaultSharedPreferences(context!!) | ||||||
|  |                     val editor = sharedPref.edit() | ||||||
|  |                     editor.putBoolean("prefer_article_viewer", false) | ||||||
|  |                     editor.commit() | ||||||
|  |                     activity!!.finish() | ||||||
|  |                 } | ||||||
|  |                 .create() | ||||||
|  |                 .show() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return rootView | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun refreshAlignment() { | ||||||
|  |         textAlignment = when (prefs.getInt("text_align", 1)) { | ||||||
|  |             1 -> "justify" | ||||||
|  |             2 -> "left" | ||||||
|  |             else -> "justify" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun getContentFromMercury( | ||||||
|  |         customTabsIntent: CustomTabsIntent, | ||||||
|  |         prefs: SharedPreferences | ||||||
|  |     ) { | ||||||
|  |         if ((context != null && context!!.isNetworkAccessible(null)) || context == null) { | ||||||
|  |             rootView!!.progressBar.visibility = View.VISIBLE | ||||||
|  |             val parser = MercuryApi() | ||||||
|  |  | ||||||
|  |             parser.parseUrl(url).enqueue( | ||||||
|  |                 object : Callback<ParsedContent> { | ||||||
|  |                     override fun onResponse( | ||||||
|  |                         call: Call<ParsedContent>, | ||||||
|  |                         response: Response<ParsedContent> | ||||||
|  |                     ) { | ||||||
|  |                         // TODO: clean all the following after finding the mercury content issue | ||||||
|  |                         try { | ||||||
|  |                             if (response.body() != null && response.body()!!.content != null && !response.body()!!.content.isNullOrEmpty()) { | ||||||
|  |                                 try { | ||||||
|  |                                     rootView!!.titleView.text = response.body()!!.title | ||||||
|  |                                     if (typeface != null) { | ||||||
|  |                                         rootView!!.titleView.typeface = typeface | ||||||
|  |                                     } | ||||||
|  |                                     try { | ||||||
|  |                                         // Note: Mercury may return relative urls... If it does the url val will not be changed. | ||||||
|  |                                         URL(response.body()!!.url) | ||||||
|  |                                         url = response.body()!!.url | ||||||
|  |                                     } catch (e: MalformedURLException) { | ||||||
|  |                                         // Mercury returned a relative url. We do nothing. | ||||||
|  |                                     } | ||||||
|  |                                 } catch (e: Exception) { | ||||||
|  |                                 } | ||||||
|  |  | ||||||
|  |                                 try { | ||||||
|  |                                     contentText = response.body()!!.content.orEmpty() | ||||||
|  |                                     htmlToWebview() | ||||||
|  |                                 } catch (e: Exception) { | ||||||
|  |                                 } | ||||||
|  |  | ||||||
|  |                                 try { | ||||||
|  |                                     if (response.body()!!.lead_image_url != null && !response.body()!!.lead_image_url.isNullOrEmpty() && context != null) { | ||||||
|  |                                         rootView!!.imageView.visibility = View.VISIBLE | ||||||
|  |                                         try { | ||||||
|  |                                             Glide | ||||||
|  |                                                 .with(context!!) | ||||||
|  |                                                 .asBitmap() | ||||||
|  |                                                 .loadMaybeBasicAuth(config, response.body()!!.lead_image_url.orEmpty()) | ||||||
|  |                                                 .apply(RequestOptions.fitCenterTransform()) | ||||||
|  |                                                 .into(rootView!!.imageView) | ||||||
|  |                                         } catch (e: Exception) { | ||||||
|  |                                         } | ||||||
|  |                                     } else { | ||||||
|  |                                         rootView!!.imageView.visibility = View.GONE | ||||||
|  |                                     } | ||||||
|  |                                 } catch (e: Exception) { | ||||||
|  |                                     if (context != null) { | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |  | ||||||
|  |                                 try { | ||||||
|  |                                     rootView!!.nestedScrollView.scrollTo(0, 0) | ||||||
|  |  | ||||||
|  |                                     rootView!!.progressBar.visibility = View.GONE | ||||||
|  |                                 } catch (e: Exception) { | ||||||
|  |                                     if (context != null) { | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |                             } else { | ||||||
|  |                                 try { | ||||||
|  |                                     openInBrowserAfterFailing(customTabsIntent) | ||||||
|  |                                 } catch (e: Exception) { | ||||||
|  |                                     if (context != null) { | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                         } catch (e: Exception) { | ||||||
|  |                             if (context != null) { | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     override fun onFailure( | ||||||
|  |                         call: Call<ParsedContent>, | ||||||
|  |                         t: Throwable | ||||||
|  |                     ) = openInBrowserAfterFailing(customTabsIntent) | ||||||
|  |                 } | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun htmlToWebview() { | ||||||
|  |         val stringColor = String.format("#%06X", 0xFFFFFF and appColors.colorAccent) | ||||||
|  |  | ||||||
|  |         val attrs: IntArray = intArrayOf(android.R.attr.fontFamily) | ||||||
|  |         val a: TypedArray = context!!.obtainStyledAttributes(resId, attrs) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         rootView!!.webcontent.settings.standardFontFamily = a.getString(0) | ||||||
|  |         rootView!!.webcontent.visibility = View.VISIBLE | ||||||
|  |         val (textColor, backgroundColor) = if (appColors.isDarkTheme) { | ||||||
|  |             if (context != null) { | ||||||
|  |                 rootView!!.webcontent.setBackgroundColor( | ||||||
|  |                     ContextCompat.getColor( | ||||||
|  |                         context!!, | ||||||
|  |                         R.color.dark_webview | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |                 Pair(ContextCompat.getColor(context!!, R.color.dark_webview_text), ContextCompat.getColor(context!!, R.color.dark_webview)) | ||||||
|  |             } else { | ||||||
|  |                 Pair(null, null) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             if (context != null) { | ||||||
|  |                 rootView!!.webcontent.setBackgroundColor( | ||||||
|  |                     ContextCompat.getColor( | ||||||
|  |                         context!!, | ||||||
|  |                         R.color.light_webview | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |                 Pair(ContextCompat.getColor(context!!, R.color.light_webview_text), ContextCompat.getColor(context!!, R.color.light_webview)) | ||||||
|  |             } else { | ||||||
|  |                 Pair(null, null) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val stringTextColor: String = if (textColor != null) { | ||||||
|  |             String.format("#%06X", 0xFFFFFF and textColor) | ||||||
|  |         } else { | ||||||
|  |             "#000000" | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val stringBackgroundColor = if (backgroundColor != null) { | ||||||
|  |             String.format("#%06X", 0xFFFFFF and backgroundColor) | ||||||
|  |         } else { | ||||||
|  |             "#FFFFFF" | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         rootView!!.webcontent.settings.useWideViewPort = true | ||||||
|  |         rootView!!.webcontent.settings.loadWithOverviewMode = true | ||||||
|  |         rootView!!.webcontent.settings.javaScriptEnabled = false | ||||||
|  |  | ||||||
|  |         rootView!!.webcontent.webViewClient = object : WebViewClient() { | ||||||
|  |             override fun shouldOverrideUrlLoading(view: WebView?, url : String): Boolean { | ||||||
|  |                 if (rootView!!.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||||
|  |                     rootView!!.context.startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) | ||||||
|  |                 } | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             override fun shouldInterceptRequest(view: WebView?, url: String): WebResourceResponse? { | ||||||
|  |                 val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|  |                 if (url.toLowerCase().contains(".jpg") || url.toLowerCase().contains(".jpeg")) { | ||||||
|  |                     try { | ||||||
|  |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|  |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.JPEG)) | ||||||
|  |                     }catch ( e : ExecutionException) {} | ||||||
|  |                 } | ||||||
|  |                 else if (url.toLowerCase().contains(".png")) { | ||||||
|  |                     try { | ||||||
|  |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|  |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.PNG)) | ||||||
|  |                     }catch ( e : ExecutionException) {} | ||||||
|  |                 } | ||||||
|  |                 else if (url.toLowerCase().contains(".webp")) { | ||||||
|  |                     try { | ||||||
|  |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|  |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.WEBP)) | ||||||
|  |                     }catch ( e : ExecutionException) {} | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 return super.shouldInterceptRequest(view, url) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val gestureDetector = GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { | ||||||
|  |             override fun onSingleTapUp(e: MotionEvent?): Boolean { | ||||||
|  |                 return performClick() | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |         rootView!!.webcontent.setOnTouchListener { _, event -> gestureDetector.onTouchEvent(event)} | ||||||
|  |  | ||||||
|  |         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { | ||||||
|  |             rootView!!.webcontent.settings.layoutAlgorithm = | ||||||
|  |                     WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING | ||||||
|  |         } else { | ||||||
|  |             rootView!!.webcontent.settings.layoutAlgorithm = WebSettings.LayoutAlgorithm.SINGLE_COLUMN | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         var baseUrl: String? = null | ||||||
|  |  | ||||||
|  |         try { | ||||||
|  |             val itemUrl = URL(url) | ||||||
|  |             baseUrl = itemUrl.protocol + "://" + itemUrl.host | ||||||
|  |         } catch (e: MalformedURLException) { | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val fontName =  when (font) { | ||||||
|  |             getString(R.string.open_sans_font_id) -> "Open Sans" | ||||||
|  |             getString(R.string.roboto_font_id) -> "Roboto" | ||||||
|  |             else -> "" | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val fontLinkAndStyle = if (font.isNotEmpty()) { | ||||||
|  |             """<link href="https://fonts.googleapis.com/css?family=${fontName.replace(" ", "+")}" rel="stylesheet"> | ||||||
|  |                 |<style> | ||||||
|  |                 |   * { | ||||||
|  |                 |       font-family: '$fontName'; | ||||||
|  |                 |   } | ||||||
|  |                 |</style> | ||||||
|  |             """.trimMargin() | ||||||
|  |         } else { | ||||||
|  |             "" | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         rootView!!.webcontent.loadDataWithBaseURL( | ||||||
|  |             baseUrl, | ||||||
|  |             """<html> | ||||||
|  |                 |<head> | ||||||
|  |                 |   <meta name="viewport" content="width=device-width, initial-scale=1"> | ||||||
|  |                 |   <style> | ||||||
|  |                 |      img { | ||||||
|  |                 |        display: inline-block; | ||||||
|  |                 |        height: auto; | ||||||
|  |                 |        width: 100%; | ||||||
|  |                 |        max-width: 100%; | ||||||
|  |                 |      } | ||||||
|  |                 |      a { | ||||||
|  |                 |        color: $stringColor !important; | ||||||
|  |                 |      } | ||||||
|  |                 |      *:not(a) { | ||||||
|  |                 |        color: $stringTextColor; | ||||||
|  |                 |      } | ||||||
|  |                 |      * { | ||||||
|  |                 |        font-size: ${fontSize}px; | ||||||
|  |                 |        text-align: $textAlignment; | ||||||
|  |                 |        word-break: break-word; | ||||||
|  |                 |        overflow:hidden; | ||||||
|  |                 |        line-height: 1.5em; | ||||||
|  |                 |        background-color: $stringBackgroundColor; | ||||||
|  |                 |      } | ||||||
|  |                 |      body, html { | ||||||
|  |                 |        background-color: $stringBackgroundColor !important; | ||||||
|  |                 |        border-color: $stringBackgroundColor  !important; | ||||||
|  |                 |        padding: 0 !important; | ||||||
|  |                 |        margin: 0 !important; | ||||||
|  |                 |      } | ||||||
|  |                 |      a, pre, code { | ||||||
|  |                 |        text-align: $textAlignment; | ||||||
|  |                 |      } | ||||||
|  |                 |      pre, code { | ||||||
|  |                 |        white-space: pre-wrap; | ||||||
|  |                 |        width:100%; | ||||||
|  |                 |        background-color: $stringBackgroundColor; | ||||||
|  |                 |      } | ||||||
|  |                 |   </style> | ||||||
|  |                 |   $fontLinkAndStyle | ||||||
|  |                 |</head> | ||||||
|  |                 |<body> | ||||||
|  |                 |   $contentText | ||||||
|  |                 |</body>""".trimMargin(), | ||||||
|  |             "text/html", | ||||||
|  |             "utf-8", | ||||||
|  |             null | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun openInBrowserAfterFailing(customTabsIntent: CustomTabsIntent) { | ||||||
|  |         rootView!!.progressBar.visibility = View.GONE | ||||||
|  |         activity!!.openItemUrl( | ||||||
|  |             allItems, | ||||||
|  |             pageNumber.toInt(), | ||||||
|  |             url, | ||||||
|  |             customTabsIntent, | ||||||
|  |             true, | ||||||
|  |             false, | ||||||
|  |             activity!! | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         private const val ARG_POSITION = "position" | ||||||
|  |         private const val ARG_ITEMS = "items" | ||||||
|  |  | ||||||
|  |         fun newInstance( | ||||||
|  |             position: Int, | ||||||
|  |             allItems: ArrayList<Item> | ||||||
|  |         ): ArticleFragment { | ||||||
|  |             val fragment = ArticleFragment() | ||||||
|  |             val args = Bundle() | ||||||
|  |             args.putInt(ARG_POSITION, position) | ||||||
|  |             args.putParcelableArrayList(ARG_ITEMS, allItems) | ||||||
|  |             fragment.arguments = args | ||||||
|  |             return fragment | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun performClick(): Boolean { | ||||||
|  |         if (rootView!!.webcontent.hitTestResult.type == WebView.HitTestResult.IMAGE_TYPE || | ||||||
|  |                 rootView!!.webcontent.hitTestResult.type == WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||||
|  |  | ||||||
|  |             val position : Int = allImages.indexOf(rootView!!.webcontent.hitTestResult.extra) | ||||||
|  |  | ||||||
|  |             val intent = Intent(activity, ImageActivity::class.java) | ||||||
|  |             intent.putExtra("allImages", allImages) | ||||||
|  |             intent.putExtra("position", position) | ||||||
|  |             startActivity(intent) | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,49 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.fragments | ||||||
|  |  | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.* | ||||||
|  | import androidx.fragment.app.Fragment | ||||||
|  | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import com.bumptech.glide.Glide | ||||||
|  | import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||||
|  | import com.bumptech.glide.request.RequestOptions | ||||||
|  | import kotlinx.android.synthetic.main.fragment_image.view.* | ||||||
|  |  | ||||||
|  | class ImageFragment : Fragment() { | ||||||
|  |  | ||||||
|  |     private lateinit var imageUrl : String | ||||||
|  |     private val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|  |  | ||||||
|  |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |  | ||||||
|  |         imageUrl = arguments!!.getString("imageUrl")!! | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { | ||||||
|  |         val view : View = inflater.inflate(R.layout.fragment_image, container, false) | ||||||
|  |  | ||||||
|  |         view.photoView.visibility = View.VISIBLE | ||||||
|  |         Glide.with(activity) | ||||||
|  |                 .asBitmap() | ||||||
|  |                 .apply(glideOptions) | ||||||
|  |                 .load(imageUrl) | ||||||
|  |                 .into(view.photoView) | ||||||
|  |  | ||||||
|  |         return view | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         private const val ARG_IMAGE = "imageUrl" | ||||||
|  |  | ||||||
|  |         fun newInstance( | ||||||
|  |                 imageUrl : String | ||||||
|  |         ): ImageFragment { | ||||||
|  |             val fragment = ImageFragment() | ||||||
|  |             val args = Bundle() | ||||||
|  |             args.putString(ARG_IMAGE, imageUrl) | ||||||
|  |             fragment.arguments = args | ||||||
|  |             return fragment | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,23 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.dao | ||||||
|  |  | ||||||
|  | import androidx.room.Dao | ||||||
|  | import androidx.room.Delete | ||||||
|  | import androidx.room.Insert | ||||||
|  | import androidx.room.OnConflictStrategy | ||||||
|  | import androidx.room.Query | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
|  |  | ||||||
|  | @Dao | ||||||
|  | interface ActionsDao { | ||||||
|  |     @Query("SELECT * FROM actions order by id asc") | ||||||
|  |     fun actions(): List<ActionEntity> | ||||||
|  |  | ||||||
|  |     @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|  |     fun insertAllActions(vararg actions: ActionEntity) | ||||||
|  |  | ||||||
|  |     @Query("DELETE FROM actions WHERE articleid = :article_id AND read = 1") | ||||||
|  |     fun deleteReadActionForArticle(article_id: String) | ||||||
|  |  | ||||||
|  |     @Delete | ||||||
|  |     fun delete(action: ActionEntity) | ||||||
|  | } | ||||||
| @@ -0,0 +1,36 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.dao | ||||||
|  |  | ||||||
|  | import androidx.room.Delete | ||||||
|  | import androidx.room.Dao | ||||||
|  | import androidx.room.Insert | ||||||
|  | import androidx.room.OnConflictStrategy | ||||||
|  | import androidx.room.Query | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.SourceEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.TagEntity | ||||||
|  |  | ||||||
|  | @Dao | ||||||
|  | interface DrawerDataDao { | ||||||
|  |     @Query("SELECT * FROM tags") | ||||||
|  |     fun tags(): List<TagEntity> | ||||||
|  |  | ||||||
|  |     @Query("SELECT * FROM sources") | ||||||
|  |     fun sources(): List<SourceEntity> | ||||||
|  |  | ||||||
|  |     @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|  |     fun insertAllTags(vararg tags: TagEntity) | ||||||
|  |  | ||||||
|  |     @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|  |     fun insertAllSources(vararg sources: SourceEntity) | ||||||
|  |  | ||||||
|  |     @Query("DELETE FROM tags") | ||||||
|  |     fun deleteAllTags() | ||||||
|  |  | ||||||
|  |     @Query("DELETE FROM sources") | ||||||
|  |     fun deleteAllSources() | ||||||
|  |  | ||||||
|  |     @Delete | ||||||
|  |     fun deleteTag(tag: TagEntity) | ||||||
|  |  | ||||||
|  |     @Delete | ||||||
|  |     fun deleteSource(source: SourceEntity) | ||||||
|  | } | ||||||
| @@ -0,0 +1,29 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.dao | ||||||
|  |  | ||||||
|  | import androidx.room.Dao | ||||||
|  | import androidx.room.Delete | ||||||
|  | import androidx.room.Insert | ||||||
|  | import androidx.room.OnConflictStrategy | ||||||
|  | import androidx.room.Query | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ItemEntity | ||||||
|  | import androidx.room.Update | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @Dao | ||||||
|  | interface ItemsDao { | ||||||
|  |     @Query("SELECT * FROM items order by id desc") | ||||||
|  |     fun items(): List<ItemEntity> | ||||||
|  |  | ||||||
|  |     @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|  |     fun insertAllItems(vararg items: ItemEntity) | ||||||
|  |  | ||||||
|  |     @Query("DELETE FROM items") | ||||||
|  |     fun deleteAllItems() | ||||||
|  |  | ||||||
|  |     @Delete | ||||||
|  |     fun delete(item: ItemEntity) | ||||||
|  |  | ||||||
|  |     @Update | ||||||
|  |     fun updateItem(item: ItemEntity) | ||||||
|  | } | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.database | ||||||
|  |  | ||||||
|  | import androidx.room.RoomDatabase | ||||||
|  | import androidx.room.Database | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.dao.ActionsDao | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.dao.DrawerDataDao | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.dao.ItemsDao | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ItemEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.SourceEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.TagEntity | ||||||
|  |  | ||||||
|  | @Database(entities = [TagEntity::class, SourceEntity::class, ItemEntity::class, ActionEntity::class], version = 4) | ||||||
|  | abstract class AppDatabase : RoomDatabase() { | ||||||
|  |     abstract fun drawerDataDao(): DrawerDataDao | ||||||
|  |  | ||||||
|  |     abstract fun itemsDao(): ItemsDao | ||||||
|  |  | ||||||
|  |     abstract fun actionsDao(): ActionsDao | ||||||
|  | } | ||||||
| @@ -0,0 +1,22 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.entities | ||||||
|  |  | ||||||
|  | import androidx.room.ColumnInfo | ||||||
|  | import androidx.room.Entity | ||||||
|  | import androidx.room.PrimaryKey | ||||||
|  |  | ||||||
|  | @Entity(tableName = "actions") | ||||||
|  | data class ActionEntity( | ||||||
|  |     @ColumnInfo(name = "articleid") | ||||||
|  |     val articleId: String, | ||||||
|  |     @ColumnInfo(name = "read") | ||||||
|  |     val read: Boolean, | ||||||
|  |     @ColumnInfo(name = "unread") | ||||||
|  |     val unread: Boolean, | ||||||
|  |     @ColumnInfo(name = "starred") | ||||||
|  |     var starred: Boolean, | ||||||
|  |     @ColumnInfo(name = "unstarred") | ||||||
|  |     var unstarred: Boolean | ||||||
|  | ) { | ||||||
|  |     @PrimaryKey(autoGenerate = true) | ||||||
|  |     var id: Int = 0 | ||||||
|  | } | ||||||
| @@ -0,0 +1,33 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.entities | ||||||
|  |  | ||||||
|  | import androidx.room.ColumnInfo | ||||||
|  | import androidx.room.Entity | ||||||
|  | import androidx.room.PrimaryKey | ||||||
|  |  | ||||||
|  | @Entity(tableName = "tags") | ||||||
|  | data class TagEntity( | ||||||
|  |     @PrimaryKey | ||||||
|  |     @ColumnInfo(name = "tag") | ||||||
|  |     val tag: String, | ||||||
|  |     @ColumnInfo(name = "color") | ||||||
|  |     val color: String, | ||||||
|  |     @ColumnInfo(name = "unread") | ||||||
|  |     val unread: Int | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | @Entity(tableName = "sources") | ||||||
|  | data class SourceEntity( | ||||||
|  |     @PrimaryKey | ||||||
|  |     @ColumnInfo(name = "id") | ||||||
|  |     val id: String, | ||||||
|  |     @ColumnInfo(name = "title") | ||||||
|  |     val title: String, | ||||||
|  |     @ColumnInfo(name = "tags") | ||||||
|  |     val tags: String, | ||||||
|  |     @ColumnInfo(name = "spout") | ||||||
|  |     val spout: String, | ||||||
|  |     @ColumnInfo(name = "error") | ||||||
|  |     val error: String, | ||||||
|  |     @ColumnInfo(name = "icon") | ||||||
|  |     val icon: String | ||||||
|  | ) | ||||||
| @@ -0,0 +1,32 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.entities | ||||||
|  |  | ||||||
|  | import androidx.room.ColumnInfo | ||||||
|  | import androidx.room.Entity | ||||||
|  | import androidx.room.PrimaryKey | ||||||
|  |  | ||||||
|  | @Entity(tableName = "items") | ||||||
|  | data class ItemEntity( | ||||||
|  |     @PrimaryKey | ||||||
|  |     @ColumnInfo(name = "id") | ||||||
|  |     val id: String, | ||||||
|  |     @ColumnInfo(name = "datetime") | ||||||
|  |     val datetime: String, | ||||||
|  |     @ColumnInfo(name = "title") | ||||||
|  |     val title: String, | ||||||
|  |     @ColumnInfo(name = "content") | ||||||
|  |     val content: String, | ||||||
|  |     @ColumnInfo(name = "unread") | ||||||
|  |     val unread: Boolean, | ||||||
|  |     @ColumnInfo(name = "starred") | ||||||
|  |     var starred: Boolean, | ||||||
|  |     @ColumnInfo(name = "thumbnail") | ||||||
|  |     val thumbnail: String?, | ||||||
|  |     @ColumnInfo(name = "icon") | ||||||
|  |     val icon: String?, | ||||||
|  |     @ColumnInfo(name = "link") | ||||||
|  |     val link: String, | ||||||
|  |     @ColumnInfo(name = "sourcetitle") | ||||||
|  |     val sourcetitle: String, | ||||||
|  |     @ColumnInfo(name = "tags") | ||||||
|  |     val tags: String | ||||||
|  | ) | ||||||
| @@ -0,0 +1,34 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.migrations | ||||||
|  |  | ||||||
|  | import androidx.sqlite.db.SupportSQLiteDatabase | ||||||
|  | import androidx.room.migration.Migration | ||||||
|  |  | ||||||
|  | val MIGRATION_1_2: Migration = object : Migration(1, 2) { | ||||||
|  |     override fun migrate(database: SupportSQLiteDatabase) { | ||||||
|  |         database.execSQL("CREATE TABLE IF NOT EXISTS `items` (`id` TEXT NOT NULL, `datetime` TEXT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `thumbnail` TEXT NOT NULL, `icon` TEXT NOT NULL, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))") | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | val MIGRATION_2_3: Migration = object : Migration(2, 3) { | ||||||
|  |     override fun migrate(database: SupportSQLiteDatabase) { | ||||||
|  |         database.execSQL("CREATE TABLE IF NOT EXISTS `actions` (`id` INTEGER NOT NULL, `articleid` TEXT NOT NULL, `read` INTEGER NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `unstarred` INTEGER NOT NULL, PRIMARY KEY(`id`))") | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | val MIGRATION_3_4: Migration = object : Migration(3, 4) { | ||||||
|  |     override fun migrate(database: SupportSQLiteDatabase) { | ||||||
|  |         // @see https://stackoverflow.com/questions/57392015/how-to-migrate-not-null-table-column-into-null-in-android-room-database | ||||||
|  |         // Create the new table | ||||||
|  |         database.execSQL("CREATE TABLE IF NOT EXISTS `itemstmp` (`id` TEXT NOT NULL, `datetime` TEXT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `thumbnail` TEXT, `icon` TEXT, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))") | ||||||
|  |  | ||||||
|  |         // Copy the data | ||||||
|  |         database.execSQL( | ||||||
|  |                 "INSERT INTO itemstmp (`id`, `datetime`, `title`, `content`, `unread`, `starred`, `thumbnail`, `icon`, `link`, `sourcetitle`, `tags`) SELECT `id`, `datetime`, `title`, `content`, `unread`, `starred`, `thumbnail`, `icon`, `link`, `sourcetitle`, `tags` FROM items") | ||||||
|  |  | ||||||
|  |         // Remove the old table | ||||||
|  |         database.execSQL("DROP TABLE items") | ||||||
|  |  | ||||||
|  |         // Change the table name to the correct one | ||||||
|  |         database.execSQL("ALTER TABLE itemstmp RENAME TO items") | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,38 +1,43 @@ | |||||||
| package apps.amine.bou.readerforselfoss.settings; | package apps.amine.bou.readerforselfoss.settings; | ||||||
|  |  | ||||||
| import android.content.res.Configuration; | import android.content.res.Configuration; | ||||||
|  | import android.os.Build; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.preference.PreferenceActivity; | import android.preference.PreferenceActivity; | ||||||
| import android.support.annotation.LayoutRes; | import androidx.annotation.LayoutRes; | ||||||
| import android.support.annotation.NonNull; | import androidx.annotation.NonNull; | ||||||
| import android.support.annotation.Nullable; | import androidx.annotation.Nullable; | ||||||
| import android.support.design.widget.AppBarLayout; | import com.google.android.material.appbar.AppBarLayout; | ||||||
| import android.support.v7.app.ActionBar; | import androidx.appcompat.app.ActionBar; | ||||||
| import android.support.v7.app.AppCompatDelegate; | import androidx.appcompat.app.AppCompatDelegate; | ||||||
| import android.support.v7.widget.Toolbar; | import androidx.appcompat.widget.Toolbar; | ||||||
| import android.view.LayoutInflater; | import android.view.LayoutInflater; | ||||||
| import android.view.MenuInflater; | import android.view.MenuInflater; | ||||||
| import android.view.View; | import android.view.View; | ||||||
| import android.view.ViewGroup; | import android.view.ViewGroup; | ||||||
| import android.widget.LinearLayout; | import android.widget.LinearLayout; | ||||||
|  |  | ||||||
| import apps.amine.bou.readerforselfoss.R; |  | ||||||
| import com.ftinc.scoop.Scoop; | import com.ftinc.scoop.Scoop; | ||||||
|  |  | ||||||
|  | import apps.amine.bou.readerforselfoss.R; | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors; | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.Toppings; | ||||||
|  |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * A {@link PreferenceActivity} which implements and proxies the necessary calls |  * A {@link PreferenceActivity} which implements and proxies the necessary calls | ||||||
|  * to be used with AppCompat. |  * to be used with AppCompat. | ||||||
|  */ |  */ | ||||||
| public abstract class AppCompatPreferenceActivity extends PreferenceActivity { //NOSONAR | public abstract class AppCompatPreferenceActivity extends PreferenceActivity { | ||||||
|  |  | ||||||
|     private AppCompatDelegate mDelegate; |     private AppCompatDelegate mDelegate; | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void onCreate(Bundle savedInstanceState) { |     protected void onCreate(Bundle savedInstanceState) { | ||||||
|  |         new AppColors(this); | ||||||
|  |  | ||||||
|         getDelegate().installViewFactory(); |         getDelegate().installViewFactory(); | ||||||
|         getDelegate().onCreate(savedInstanceState); |         getDelegate().onCreate(savedInstanceState); | ||||||
|         Scoop.getInstance().apply(this); |  | ||||||
|         super.onCreate(savedInstanceState); |         super.onCreate(savedInstanceState); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -40,9 +45,16 @@ public abstract class AppCompatPreferenceActivity extends PreferenceActivity { / | |||||||
|     protected void onPostCreate(Bundle savedInstanceState) { |     protected void onPostCreate(Bundle savedInstanceState) { | ||||||
|         super.onPostCreate(savedInstanceState); |         super.onPostCreate(savedInstanceState); | ||||||
|  |  | ||||||
|         LinearLayout root = (LinearLayout)findViewById(android.R.id.list).getParent().getParent().getParent(); |         LinearLayout root = (LinearLayout) findViewById(android.R.id.list).getParent().getParent().getParent(); | ||||||
|         AppBarLayout bar = (AppBarLayout) LayoutInflater.from(this).inflate(R.layout.settings_toolbar, root, false); |         AppBarLayout bar = (AppBarLayout) LayoutInflater.from(this).inflate(R.layout.settings_toolbar, root, false); | ||||||
|         Toolbar toolbar = bar.findViewById(R.id.toolbar); |         Toolbar toolbar = bar.findViewById(R.id.toolbar); | ||||||
|  |  | ||||||
|  |         Scoop scoop = Scoop.getInstance(); | ||||||
|  |         scoop.bind(this, Toppings.PRIMARY.getValue(), toolbar); | ||||||
|  |         if  (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { | ||||||
|  |             scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.getValue()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         setSupportActionBar(toolbar); |         setSupportActionBar(toolbar); | ||||||
|         getSupportActionBar().setDisplayHomeAsUpEnabled(true); |         getSupportActionBar().setDisplayHomeAsUpEnabled(true); | ||||||
|         getSupportActionBar().setDisplayShowHomeEnabled(true); |         getSupportActionBar().setDisplayShowHomeEnabled(true); | ||||||
|   | |||||||
| @@ -1,8 +1,6 @@ | |||||||
| package apps.amine.bou.readerforselfoss.settings; | package apps.amine.bou.readerforselfoss.settings; | ||||||
|  |  | ||||||
|  |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| import android.annotation.TargetApi; | import android.annotation.TargetApi; | ||||||
| import android.content.ClipData; | import android.content.ClipData; | ||||||
| import android.content.ClipboardManager; | import android.content.ClipboardManager; | ||||||
| @@ -10,6 +8,7 @@ import android.content.Context; | |||||||
| import android.content.Intent; | import android.content.Intent; | ||||||
| import android.content.SharedPreferences; | import android.content.SharedPreferences; | ||||||
| import android.content.res.Configuration; | import android.content.res.Configuration; | ||||||
|  | import android.content.res.Resources; | ||||||
| import android.net.Uri; | import android.net.Uri; | ||||||
| import android.os.Build; | import android.os.Build; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| @@ -21,16 +20,22 @@ import android.preference.PreferenceActivity; | |||||||
| import android.preference.PreferenceFragment; | import android.preference.PreferenceFragment; | ||||||
| import android.preference.PreferenceManager; | import android.preference.PreferenceManager; | ||||||
| import android.preference.SwitchPreference; | import android.preference.SwitchPreference; | ||||||
| import android.support.v7.app.ActionBar; | import androidx.appcompat.app.ActionBar; | ||||||
|  | import android.text.Editable; | ||||||
| import android.text.InputFilter; | import android.text.InputFilter; | ||||||
| import android.text.Spanned; | import android.text.Spanned; | ||||||
|  | import android.text.TextWatcher; | ||||||
|  | import android.util.Log; | ||||||
|  | import android.view.Menu; | ||||||
|  | import android.view.MenuInflater; | ||||||
| import android.view.MenuItem; | import android.view.MenuItem; | ||||||
| import android.widget.Toast; | import android.widget.Toast; | ||||||
|  |  | ||||||
| import apps.amine.bou.readerforselfoss.BuildConfig; | import java.util.List; | ||||||
|  |  | ||||||
| import apps.amine.bou.readerforselfoss.R; | import apps.amine.bou.readerforselfoss.R; | ||||||
|  | import apps.amine.bou.readerforselfoss.themes.AppColors; | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config; | import apps.amine.bou.readerforselfoss.utils.Config; | ||||||
| import com.ftinc.scoop.ui.ScoopSettingsActivity; |  | ||||||
|  |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -44,7 +49,7 @@ import com.ftinc.scoop.ui.ScoopSettingsActivity; | |||||||
|  * href="http://developer.android.com/guide/topics/ui/settings.html">Settings |  * href="http://developer.android.com/guide/topics/ui/settings.html">Settings | ||||||
|  * API Guide</a> for more information on developing a Settings UI. |  * API Guide</a> for more information on developing a Settings UI. | ||||||
|  */ |  */ | ||||||
| public class SettingsActivity extends AppCompatPreferenceActivity { //NOSONAR | public class SettingsActivity extends AppCompatPreferenceActivity { | ||||||
|     /** |     /** | ||||||
|      * A preference value change listener that updates the preference's summary |      * A preference value change listener that updates the preference's summary | ||||||
|      * to reflect its new value. |      * to reflect its new value. | ||||||
| @@ -90,6 +95,7 @@ public class SettingsActivity extends AppCompatPreferenceActivity { //NOSONAR | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void onCreate(Bundle savedInstanceState) { |     protected void onCreate(Bundle savedInstanceState) { | ||||||
|  |         new AppColors(this); | ||||||
|         super.onCreate(savedInstanceState); |         super.onCreate(savedInstanceState); | ||||||
|         setupActionBar(); |         setupActionBar(); | ||||||
|     } |     } | ||||||
| @@ -120,6 +126,27 @@ public class SettingsActivity extends AppCompatPreferenceActivity { //NOSONAR | |||||||
|     @TargetApi(Build.VERSION_CODES.HONEYCOMB) |     @TargetApi(Build.VERSION_CODES.HONEYCOMB) | ||||||
|     public void onBuildHeaders(List<Header> target) { |     public void onBuildHeaders(List<Header> target) { | ||||||
|         loadHeadersFromResource(R.xml.pref_headers, target); |         loadHeadersFromResource(R.xml.pref_headers, target); | ||||||
|  |  | ||||||
|  |         AppColors appColors = new AppColors(this); | ||||||
|  |         if (appColors != null && appColors.isDarkTheme()) { | ||||||
|  |             for (Header header : target) { | ||||||
|  |                 tryLoadIconDark(header); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void tryLoadIconDark(Header header){ | ||||||
|  |         try{ | ||||||
|  |             if (header.fragmentArguments != null) { | ||||||
|  |                 String iconDark = header.fragmentArguments.getString("iconDark"); | ||||||
|  |                 int iconDarkId = getResources().getIdentifier(iconDark, "drawable", getPackageName()); | ||||||
|  |                 if (iconDarkId != 0) { | ||||||
|  |                     header.iconRes = iconDarkId; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             Log.e("SettingsActivity", "Can not load dark icon", e); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -130,8 +157,11 @@ public class SettingsActivity extends AppCompatPreferenceActivity { //NOSONAR | |||||||
|     protected boolean isValidFragment(String fragmentName) { |     protected boolean isValidFragment(String fragmentName) { | ||||||
|         return PreferenceFragment.class.getName().equals(fragmentName) |         return PreferenceFragment.class.getName().equals(fragmentName) | ||||||
|                 || GeneralPreferenceFragment.class.getName().equals(fragmentName) |                 || GeneralPreferenceFragment.class.getName().equals(fragmentName) | ||||||
|                 || DebugPreferenceFragment.class.getName().equals(fragmentName) |                 || ArticleViewerPreferenceFragment.class.getName().equals(fragmentName) | ||||||
|                 || LinksPreferenceFragment.class.getName().equals(fragmentName); |                 || OfflinePreferenceFragment.class.getName().equals(fragmentName) | ||||||
|  |                 || ExperimentalPreferenceFragment.class.getName().equals(fragmentName) | ||||||
|  |                 || LinksPreferenceFragment.class.getName().equals(fragmentName) | ||||||
|  |                 || ThemePreferenceFragment.class.getName().equals(fragmentName); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -146,83 +176,64 @@ public class SettingsActivity extends AppCompatPreferenceActivity { //NOSONAR | |||||||
|             addPreferencesFromResource(R.xml.pref_general); |             addPreferencesFromResource(R.xml.pref_general); | ||||||
|             setHasOptionsMenu(true); |             setHasOptionsMenu(true); | ||||||
|  |  | ||||||
|             SwitchPreference cardViewActive = (SwitchPreference) findPreference("card_view_active"); |  | ||||||
|             final SwitchPreference tabOnTap = (SwitchPreference) findPreference("tab_on_tap"); |  | ||||||
|             tabOnTap.setEnabled(!cardViewActive.isChecked()); |  | ||||||
|             cardViewActive.setOnPreferenceChangeListener(new OnPreferenceChangeListener() { |  | ||||||
|                 public boolean onPreferenceChange(Preference preference, Object newValue){ |  | ||||||
|                     boolean isEnabled = (Boolean) newValue; |  | ||||||
|                     tabOnTap.setEnabled(!isEnabled); |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             EditTextPreference itemsNumber = (EditTextPreference) findPreference("prefer_api_items_number"); |             EditTextPreference itemsNumber = (EditTextPreference) findPreference("prefer_api_items_number"); | ||||||
|             itemsNumber.getEditText().setFilters(new InputFilter[]{ |             itemsNumber.getEditText().setFilters(new InputFilter[]{ | ||||||
|                 new InputFilter (){ |                     new InputFilter() { | ||||||
|  |  | ||||||
|                     @Override |                         @Override | ||||||
|                     public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) { |                         public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) { | ||||||
|                         try { |                             try { | ||||||
|                             int input = Integer.parseInt(dest.toString() + source.toString()); |                                 int input = Integer.parseInt(dest.toString() + source.toString()); | ||||||
|                             if (input <= 200 && input >0) |                                 if (input <= 200 && input > 0) | ||||||
|                                 return null; |                                     return null; | ||||||
|                         } catch (NumberFormatException nfe) { |                             } catch (NumberFormatException nfe) { | ||||||
|                             Toast.makeText(getActivity(), R.string.items_number_should_be_number, Toast.LENGTH_LONG).show(); |                                 Toast.makeText(getActivity(), R.string.items_number_should_be_number, Toast.LENGTH_LONG).show(); | ||||||
|  |                             } | ||||||
|  |                             return ""; | ||||||
|                         } |                         } | ||||||
|                         return ""; |  | ||||||
|                     } |                     } | ||||||
|                 } |  | ||||||
|             }); |             }); | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |  | ||||||
|             int id = item.getItemId(); |  | ||||||
|             if (id == android.R.id.home) { |  | ||||||
|                 getActivity().finish(); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|             return super.onOptionsItemSelected(item); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @TargetApi(Build.VERSION_CODES.HONEYCOMB) |     @TargetApi(Build.VERSION_CODES.HONEYCOMB) | ||||||
|     public static class DebugPreferenceFragment extends PreferenceFragment { |     public static class ArticleViewerPreferenceFragment extends PreferenceFragment { | ||||||
|         @Override |         @Override | ||||||
|         public void onCreate(Bundle savedInstanceState) { |         public void onCreate(Bundle savedInstanceState) { | ||||||
|             super.onCreate(savedInstanceState); |             super.onCreate(savedInstanceState); | ||||||
|             addPreferencesFromResource(R.xml.pref_debug); |             addPreferencesFromResource(R.xml.pref_viewer); | ||||||
|             setHasOptionsMenu(true); |             setHasOptionsMenu(true); | ||||||
|  |  | ||||||
|             SharedPreferences pref = getActivity().getSharedPreferences(Config.Companion.getSettingsName(), Context.MODE_PRIVATE); |             final EditTextPreference fontSize = (EditTextPreference) findPreference("reader_font_size"); | ||||||
|             final String id = pref.getString("unique_id", "..."); |             fontSize.getEditText().addTextChangedListener(new TextWatcher() { | ||||||
|  |  | ||||||
|             final Preference identifier = findPreference("debug_identifier"); |  | ||||||
|             final ClipboardManager clipboard = (ClipboardManager) |  | ||||||
|                     getActivity().getSystemService(Context.CLIPBOARD_SERVICE); |  | ||||||
|  |  | ||||||
|             identifier.setOnPreferenceClickListener(new OnPreferenceClickListener() { |  | ||||||
|                 @Override |                 @Override | ||||||
|                 public boolean onPreferenceClick(Preference preference) { |                 public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {} | ||||||
|                     ClipData clip = ClipData.newPlainText("Selfoss unique id", id); |  | ||||||
|                     clipboard.setPrimaryClip(clip); |  | ||||||
|  |  | ||||||
|                     Toast.makeText(getActivity(), R.string.unique_id_to_clipboard, Toast.LENGTH_LONG).show(); |                 @Override | ||||||
|                     return true; |                 public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {} | ||||||
|  |  | ||||||
|  |                 @Override | ||||||
|  |                 public void afterTextChanged(Editable editable) { | ||||||
|  |                     try { | ||||||
|  |                         fontSize.getEditText().setTextSize(Integer.parseInt(editable.toString())); | ||||||
|  |                     } catch (NumberFormatException e) {} | ||||||
|                 } |                 } | ||||||
|             }); |             }); | ||||||
|             identifier.setTitle(id); |             fontSize.getEditText().setFilters(new InputFilter[]{ | ||||||
|         } |                     new InputFilter() { | ||||||
|  |  | ||||||
|         @Override |                         @Override | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |                         public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) { | ||||||
|             int id = item.getItemId(); |                             try { | ||||||
|             if (id == android.R.id.home) { |                                 int input = Integer.parseInt(dest.toString() + source.toString()); | ||||||
|                 getActivity().finish(); |                                 if (input > 0) | ||||||
|                 return true; |                                     return null; | ||||||
|             } |                             } catch (NumberFormatException nfe) {} | ||||||
|             return super.onOptionsItemSelected(item); |                             return ""; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |             }); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -243,10 +254,10 @@ public class SettingsActivity extends AppCompatPreferenceActivity { //NOSONAR | |||||||
|             addPreferencesFromResource(R.xml.pref_links); |             addPreferencesFromResource(R.xml.pref_links); | ||||||
|             setHasOptionsMenu(true); |             setHasOptionsMenu(true); | ||||||
|  |  | ||||||
|             findPreference( "trackerLink" ).setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() { |             findPreference("trackerLink").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() { | ||||||
|                 @Override |                 @Override | ||||||
|                 public boolean onPreferenceClick(Preference preference) { |                 public boolean onPreferenceClick(Preference preference) { | ||||||
|                     openUrl(Uri.parse(BuildConfig.TRACKER_URL)); |                     openUrl(Uri.parse(Config.trackerUrl)); | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|             }); |             }); | ||||||
| @@ -254,7 +265,7 @@ public class SettingsActivity extends AppCompatPreferenceActivity { //NOSONAR | |||||||
|             findPreference("sourceLink").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() { |             findPreference("sourceLink").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() { | ||||||
|                 @Override |                 @Override | ||||||
|                 public boolean onPreferenceClick(Preference preference) { |                 public boolean onPreferenceClick(Preference preference) { | ||||||
|                     openUrl(Uri.parse(BuildConfig.SOURCE_URL)); |                     openUrl(Uri.parse(Config.sourceUrl)); | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|             }); |             }); | ||||||
| @@ -262,39 +273,71 @@ public class SettingsActivity extends AppCompatPreferenceActivity { //NOSONAR | |||||||
|             findPreference("translation").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() { |             findPreference("translation").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() { | ||||||
|                 @Override |                 @Override | ||||||
|                 public boolean onPreferenceClick(Preference preference) { |                 public boolean onPreferenceClick(Preference preference) { | ||||||
|                     openUrl(Uri.parse(BuildConfig.TRANSLATION_URL)); |                     openUrl(Uri.parse(Config.translationUrl)); | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @TargetApi(Build.VERSION_CODES.HONEYCOMB) | ||||||
|  |     public static class ThemePreferenceFragment extends PreferenceFragment { | ||||||
|  |         @Override | ||||||
|  |         public void onCreate(Bundle savedInstanceState) { | ||||||
|  |             super.onCreate(savedInstanceState); | ||||||
|  |             addPreferencesFromResource(R.xml.pref_theme); | ||||||
|  |             setHasOptionsMenu(true); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |         public boolean onOptionsItemSelected(MenuItem item) { | ||||||
|             int id = item.getItemId(); |             int id = item.getItemId(); | ||||||
|             if (id == android.R.id.home) { |             if (id == R.id.clear) { | ||||||
|                 getActivity().finish(); |                 SharedPreferences pref = PreferenceManager.getDefaultSharedPreferences(getActivity()); | ||||||
|                 return true; |                 SharedPreferences.Editor editor = pref.edit(); | ||||||
|  |                 editor.remove("color_primary"); | ||||||
|  |                 editor.remove("color_primary_dark"); | ||||||
|  |                 editor.remove("color_accent"); | ||||||
|  |                 editor.remove("color_accent_dark"); | ||||||
|  |                 editor.remove("dark_theme"); | ||||||
|  |                 editor.apply(); | ||||||
|  |                 getActivity().recreate(); | ||||||
|             } |             } | ||||||
|             return super.onOptionsItemSelected(item); |             return super.onOptionsItemSelected(item); | ||||||
|         } |         } | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |         @Override | ||||||
|     public void onHeaderClick(Header header, int position) { |         public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { | ||||||
|         super.onHeaderClick(header, position); |             inflater.inflate(R.menu.settings_theme, menu); | ||||||
|         if (header.id == R.id.theme_change) { |  | ||||||
|             Intent intent = ScoopSettingsActivity.createIntent(getApplicationContext()); |  | ||||||
|             intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); |  | ||||||
|             getApplicationContext().startActivity(intent); |  | ||||||
|             finish(); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @TargetApi(Build.VERSION_CODES.HONEYCOMB) | ||||||
|  |     public static class OfflinePreferenceFragment extends PreferenceFragment { | ||||||
|  |         @Override | ||||||
|  |         public void onCreate(Bundle savedInstanceState) { | ||||||
|  |             super.onCreate(savedInstanceState); | ||||||
|  |             addPreferencesFromResource(R.xml.pref_offline); | ||||||
|  |             setHasOptionsMenu(true); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @TargetApi(Build.VERSION_CODES.HONEYCOMB) | ||||||
|  |     public static class ExperimentalPreferenceFragment extends PreferenceFragment { | ||||||
|  |         @Override | ||||||
|  |         public void onCreate(Bundle savedInstanceState) { | ||||||
|  |             super.onCreate(savedInstanceState); | ||||||
|  |             addPreferencesFromResource(R.xml.pref_experimental); | ||||||
|  |             setHasOptionsMenu(true); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onOptionsItemSelected(MenuItem item) { |     public boolean onOptionsItemSelected(MenuItem item) { | ||||||
|         int id = item.getItemId(); |         int id = item.getItemId(); | ||||||
|         if (id == android.R.id.home) { |         if (id == android.R.id.home) { | ||||||
|             finish(); |             super.onBackPressed(); | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|         return super.onOptionsItemSelected(item); |         return super.onOptionsItemSelected(item); | ||||||
|   | |||||||
| @@ -2,49 +2,75 @@ package apps.amine.bou.readerforselfoss.themes | |||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.support.annotation.ColorInt | import android.preference.PreferenceManager | ||||||
|  | import androidx.annotation.ColorInt | ||||||
|  | import androidx.appcompat.view.ContextThemeWrapper | ||||||
| import android.util.TypedValue | import android.util.TypedValue | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.ViewGroup | ||||||
|  |  | ||||||
| class AppColors(a: Activity) { | class AppColors(a: Activity) { | ||||||
|     @ColorInt val accent: Int |  | ||||||
|     @ColorInt val dark: Int |     @ColorInt val colorPrimary: Int | ||||||
|     @ColorInt val primary: Int |     @ColorInt val colorPrimaryDark: Int | ||||||
|     @ColorInt val cardBackground: Int |     @ColorInt val colorAccent: Int | ||||||
|     @ColorInt val windowBackground: Int |     @ColorInt val colorAccentDark: Int | ||||||
|  |     @ColorInt val cardBackgroundColor: Int | ||||||
|  |     @ColorInt val colorBackground: Int | ||||||
|  |     @ColorInt val textColor: Int | ||||||
|     val isDarkTheme: Boolean |     val isDarkTheme: Boolean | ||||||
|  |  | ||||||
|     init { |     init { | ||||||
|  |         val sharedPref = PreferenceManager.getDefaultSharedPreferences(a) | ||||||
|  |  | ||||||
|  |         colorPrimary = | ||||||
|  |                 sharedPref.getInt( | ||||||
|  |                     "color_primary", | ||||||
|  |                     a.resources.getColor(R.color.colorPrimary) | ||||||
|  |                 ) | ||||||
|  |         colorPrimaryDark = | ||||||
|  |                 sharedPref.getInt( | ||||||
|  |                     "color_primary_dark", | ||||||
|  |                     a.resources.getColor(R.color.colorPrimaryDark) | ||||||
|  |                 ) | ||||||
|  |         colorAccent = | ||||||
|  |                 sharedPref.getInt( | ||||||
|  |                     "color_accent", | ||||||
|  |                     a.resources.getColor(R.color.colorAccent) | ||||||
|  |                 ) | ||||||
|  |         colorAccentDark = | ||||||
|  |                 sharedPref.getInt( | ||||||
|  |                     "color_accent_dark", | ||||||
|  |                     a.resources.getColor(R.color.colorAccentDark) | ||||||
|  |                 ) | ||||||
|  |         isDarkTheme = | ||||||
|  |                 sharedPref.getBoolean( | ||||||
|  |                     "dark_theme", | ||||||
|  |                     false | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |         colorBackground = if (isDarkTheme) { | ||||||
|  |             a.setTheme(R.style.NoBarDark) | ||||||
|  |             R.color.darkBackground | ||||||
|  |         } else { | ||||||
|  |             a.setTheme(R.style.NoBar) | ||||||
|  |             android.R.color.background_light | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         textColor = if (isDarkTheme) { | ||||||
|  |             R.color.md_white_1000 | ||||||
|  |         } else { | ||||||
|  |             R.color.md_grey_900 | ||||||
|  |         } | ||||||
|  |  | ||||||
|         val wrapper = Context::class.java |         val wrapper = Context::class.java | ||||||
|         val method = wrapper!!.getMethod("getThemeResId") |         val method = wrapper!!.getMethod("getThemeResId") | ||||||
|         method.isAccessible = true |         method.isAccessible = true | ||||||
|  |  | ||||||
|         isDarkTheme = when (method.invoke(a.baseContext)) { |  | ||||||
|             R.style.NoBarTealOrangeDark, |  | ||||||
|             R.style.NoBarDark, |  | ||||||
|             R.style.NoBarBlueAmberDark, |  | ||||||
|             R.style.NoBarGreyOrangeDark, |  | ||||||
|             R.style.NoBarIndigoPinkDark, |  | ||||||
|             R.style.NoBarRedTealDark, |  | ||||||
|             R.style.NoBarCyanPinkDark -> true |  | ||||||
|             else -> false |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         val typedAccent = TypedValue() |  | ||||||
|         val typedAccentDark = TypedValue() |  | ||||||
|         val typedPrimary = TypedValue() |  | ||||||
|         val typedCardBackground = TypedValue() |         val typedCardBackground = TypedValue() | ||||||
|         val typedWindowBackground = TypedValue() |  | ||||||
|  |  | ||||||
|         a.theme.resolveAttribute(R.attr.colorAccent, typedAccent, true) |  | ||||||
|         a.theme.resolveAttribute(R.attr.colorAccent, typedAccent, true) |  | ||||||
|         a.theme.resolveAttribute(R.attr.colorPrimary, typedPrimary, true) |  | ||||||
|         a.theme.resolveAttribute(R.attr.cardBackgroundColor, typedCardBackground, true) |         a.theme.resolveAttribute(R.attr.cardBackgroundColor, typedCardBackground, true) | ||||||
|         a.theme.resolveAttribute(android.R.attr.colorBackground, typedWindowBackground, true) |  | ||||||
|         accent = typedAccent.data |         cardBackgroundColor = typedCardBackground.data | ||||||
|         dark = typedAccentDark.data |  | ||||||
|         primary = typedPrimary.data |  | ||||||
|         cardBackground = typedCardBackground.data |  | ||||||
|         windowBackground = typedWindowBackground.data |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.themes | ||||||
|  |  | ||||||
|  | enum class Toppings(val value: Int) { | ||||||
|  |     PRIMARY(1), | ||||||
|  |     PRIMARY_DARK(2), | ||||||
|  |     ACCENT(3), | ||||||
|  |     ACCENT_DARK(4) | ||||||
|  | } | ||||||
| @@ -0,0 +1,43 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.transformers | ||||||
|  |  | ||||||
|  | import androidx.viewpager.widget.ViewPager | ||||||
|  | import android.view.View | ||||||
|  |  | ||||||
|  | class DepthPageTransformer : ViewPager.PageTransformer { | ||||||
|  |  | ||||||
|  |     override fun transformPage(view: View, position: Float) { | ||||||
|  |         val pageWidth = view.width | ||||||
|  |  | ||||||
|  |         when { | ||||||
|  |             position < -1 -> // [-Infinity,-1) | ||||||
|  |                 // This page is way off-screen to the left. | ||||||
|  |                 view.alpha = 0F | ||||||
|  |             position <= 0 -> { // [-1,0] | ||||||
|  |                 // Use the default slide transition when moving to the left page | ||||||
|  |                 view.alpha = 1F | ||||||
|  |                 view.translationX = 0F | ||||||
|  |                 view.scaleX = 1F | ||||||
|  |                 view.scaleY = 1F | ||||||
|  |             } | ||||||
|  |             position <= 1 -> { // (0,1] | ||||||
|  |                 // Fade the page out. | ||||||
|  |                 view.alpha = 1 - position | ||||||
|  |  | ||||||
|  |                 // Counteract the default slide transition | ||||||
|  |                 view.translationX = pageWidth * -position | ||||||
|  |  | ||||||
|  |                 // Scale the page down (between MIN_SCALE and 1) | ||||||
|  |                 val scaleFactor = MIN_SCALE + (1 - MIN_SCALE) * (1 - Math.abs(position)) | ||||||
|  |                 view.scaleX = scaleFactor | ||||||
|  |                 view.scaleY = scaleFactor | ||||||
|  |             } | ||||||
|  |             else -> // (1,+Infinity] | ||||||
|  |                 // This page is way off-screen to the right. | ||||||
|  |                 view.alpha = 0F | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         private val MIN_SCALE = 0.75f | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -4,4 +4,4 @@ import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | |||||||
| import retrofit2.Response | import retrofit2.Response | ||||||
|  |  | ||||||
| fun Response<SuccessResponse>.succeeded(): Boolean = | fun Response<SuccessResponse>.succeeded(): Boolean = | ||||||
|         this.code() === 200 && this.body() != null && this.body()!!.isSuccess |     this.code() === 200 && this.body() != null && this.body()!!.isSuccess | ||||||
| @@ -2,59 +2,10 @@ package apps.amine.bou.readerforselfoss.utils | |||||||
|  |  | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.content.SharedPreferences |  | ||||||
| import android.net.Uri |  | ||||||
| import android.support.v7.app.AlertDialog |  | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
| import com.google.firebase.remoteconfig.FirebaseRemoteConfig |  | ||||||
|  |  | ||||||
| fun String?.isEmptyOrNullOrNullString(): Boolean = | fun String?.isEmptyOrNullOrNullString(): Boolean = | ||||||
|         this == null || this == "null" || this.isEmpty() |     this == null || this == "null" || this.isEmpty() | ||||||
|  |  | ||||||
| fun Context.checkApkVersion( |  | ||||||
|         settings: SharedPreferences, |  | ||||||
|         editor: SharedPreferences.Editor, |  | ||||||
|         mFirebaseRemoteConfig: FirebaseRemoteConfig |  | ||||||
| ) = { |  | ||||||
|     fun isThereAnUpdate() { |  | ||||||
|         val APK_LINK = "github_apk" |  | ||||||
|  |  | ||||||
|         val apkLink = mFirebaseRemoteConfig.getString(APK_LINK) |  | ||||||
|         val storedLink = settings.getString(APK_LINK, "") |  | ||||||
|         if (apkLink != storedLink && !apkLink.isEmpty()) { |  | ||||||
|             val alertDialog = AlertDialog.Builder(this).create() |  | ||||||
|             alertDialog.setTitle(getString(R.string.new_apk_available_title)) |  | ||||||
|             alertDialog.setMessage(getString(R.string.new_apk_available_message)) |  | ||||||
|             alertDialog.setButton( |  | ||||||
|                     AlertDialog.BUTTON_POSITIVE, |  | ||||||
|                     getString(R.string.new_apk_available_get) |  | ||||||
|             ) { _, _ -> |  | ||||||
|                 editor.putString(APK_LINK, apkLink) |  | ||||||
|                 editor.apply() |  | ||||||
|                 val browserIntent = Intent(Intent.ACTION_VIEW, Uri.parse(apkLink)) |  | ||||||
|                 startActivity(browserIntent) |  | ||||||
|             } |  | ||||||
|             alertDialog.setButton( |  | ||||||
|                     AlertDialog.BUTTON_NEUTRAL, getString(R.string.new_apk_available_no), |  | ||||||
|                     { dialog, _ -> |  | ||||||
|                         editor.putString(APK_LINK, apkLink) |  | ||||||
|                         editor.apply() |  | ||||||
|                         dialog.dismiss() |  | ||||||
|                     } |  | ||||||
|             ) |  | ||||||
|             alertDialog.show() |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     mFirebaseRemoteConfig.fetch(43200) |  | ||||||
|             .addOnCompleteListener { task -> |  | ||||||
|                 if (task.isSuccessful) { |  | ||||||
|                     mFirebaseRemoteConfig.activateFetched() |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 isThereAnUpdate() |  | ||||||
|             } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun String.longHash(): Long { | fun String.longHash(): Long { | ||||||
|     var h = 98764321261L |     var h = 98764321261L | ||||||
| @@ -68,22 +19,23 @@ fun String.longHash(): Long { | |||||||
| } | } | ||||||
|  |  | ||||||
| fun String.toStringUriWithHttp(): String = | fun String.toStringUriWithHttp(): String = | ||||||
|         if (!this.startsWith("https://") && !this.startsWith("http://")) { |     if (!this.startsWith("https://") && !this.startsWith("http://")) { | ||||||
|             "http://" + this |         "http://" + this | ||||||
|         } else { |     } else { | ||||||
|             this |         this | ||||||
|         } |     } | ||||||
|  |  | ||||||
| fun Context.shareLink(itemUrl: 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 | ||||||
|     sendIntent.putExtra(Intent.EXTRA_TEXT, itemUrl.toStringUriWithHttp()) |     sendIntent.putExtra(Intent.EXTRA_TEXT, itemUrl.toStringUriWithHttp()) | ||||||
|  |     sendIntent.putExtra(Intent.EXTRA_SUBJECT, itemTitle) | ||||||
|     sendIntent.type = "text/plain" |     sendIntent.type = "text/plain" | ||||||
|     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) | ||||||
|     ) |     ) | ||||||
| } | } | ||||||
| @@ -11,28 +11,40 @@ class Config(c: Context) { | |||||||
|     val settings: SharedPreferences = c.getSharedPreferences(settingsName, Context.MODE_PRIVATE) |     val settings: SharedPreferences = c.getSharedPreferences(settingsName, Context.MODE_PRIVATE) | ||||||
|  |  | ||||||
|     val baseUrl: String |     val baseUrl: String | ||||||
|         get() = settings.getString("url", "") |         get() = settings.getString("url", "")!! | ||||||
|  |  | ||||||
|     val userLogin: String |     val userLogin: String | ||||||
|         get() = settings.getString("login", "") |         get() = settings.getString("login", "")!! | ||||||
|  |  | ||||||
|     val userPassword: String |     val userPassword: String | ||||||
|         get() = settings.getString("password", "") |         get() = settings.getString("password", "")!! | ||||||
|  |  | ||||||
|     val httpUserLogin: String |     val httpUserLogin: String | ||||||
|         get() = settings.getString("httpUserName", "") |         get() = settings.getString("httpUserName", "")!! | ||||||
|  |  | ||||||
|     val httpUserPassword: String |     val httpUserPassword: String | ||||||
|         get() = settings.getString("httpPassword", "") |         get() = settings.getString("httpPassword", "")!! | ||||||
|  |  | ||||||
|     companion object { |     companion object { | ||||||
|         val settingsName = "paramsselfoss" |         const val settingsName = "paramsselfoss" | ||||||
|  |  | ||||||
|  |         const val feedbackEmail = "aminecmi@gmail.com" | ||||||
|  |  | ||||||
|  |         const val translationUrl = "https://crwd.in/readerforselfoss" | ||||||
|  |  | ||||||
|  |         const val sourceUrl = "https://github.com/aminecmi/ReaderforSelfoss" | ||||||
|  |  | ||||||
|  |         const val trackerUrl = "https://github.com/aminecmi/ReaderforSelfoss/issues" | ||||||
|  |  | ||||||
|  |         const val syncChannelId = "sync-channel-id" | ||||||
|  |  | ||||||
|  |         const val newItemsChannelId = "new-items-channel-id" | ||||||
|  |  | ||||||
|         fun logoutAndRedirect( |         fun logoutAndRedirect( | ||||||
|                 c: Context, |             c: Context, | ||||||
|                 callingActivity: Activity, |             callingActivity: Activity, | ||||||
|                 editor: SharedPreferences.Editor, |             editor: SharedPreferences.Editor, | ||||||
|                 baseUrlFail: Boolean = false |             baseUrlFail: Boolean = false | ||||||
|         ): Boolean { |         ): Boolean { | ||||||
|             editor.remove("url") |             editor.remove("url") | ||||||
|             editor.remove("login") |             editor.remove("login") | ||||||
|   | |||||||
| @@ -7,37 +7,37 @@ import javax.net.ssl.SSLContext | |||||||
| import javax.net.ssl.TrustManager | import javax.net.ssl.TrustManager | ||||||
| import javax.net.ssl.X509TrustManager | import javax.net.ssl.X509TrustManager | ||||||
|  |  | ||||||
| fun getUnsafeHttpClient() = | fun getUnsafeHttpClient(): OkHttpClient.Builder = | ||||||
|         try { |     try { | ||||||
|             // Create a trust manager that does not validate certificate chains |         // Create a trust manager that does not validate certificate chains | ||||||
|             val trustAllCerts = arrayOf<TrustManager>(object : X509TrustManager { |         val trustAllCerts = arrayOf<TrustManager>(object : X509TrustManager { | ||||||
|                 override fun getAcceptedIssuers(): Array<X509Certificate> = |             override fun getAcceptedIssuers(): Array<X509Certificate> = | ||||||
|                         arrayOf() |                 arrayOf() | ||||||
|  |  | ||||||
|                 @Throws(CertificateException::class) |             @Throws(CertificateException::class) | ||||||
|                 override fun checkClientTrusted( |             override fun checkClientTrusted( | ||||||
|                         chain: Array<java.security.cert.X509Certificate>, |                 chain: Array<java.security.cert.X509Certificate>, | ||||||
|                         authType: String |                 authType: String | ||||||
|                 ) { |             ) { | ||||||
|                 } |             } | ||||||
|  |  | ||||||
|                 @Throws(CertificateException::class) |             @Throws(CertificateException::class) | ||||||
|                 override fun checkServerTrusted( |             override fun checkServerTrusted( | ||||||
|                         chain: Array<java.security.cert.X509Certificate>, |                 chain: Array<java.security.cert.X509Certificate>, | ||||||
|                         authType: String |                 authType: String | ||||||
|                 ) { |             ) { | ||||||
|                 } |             } | ||||||
|             }) |         }) | ||||||
|  |  | ||||||
|             // Install the all-trusting trust manager |         // Install the all-trusting trust manager | ||||||
|             val sslContext = SSLContext.getInstance("SSL") |         val sslContext = SSLContext.getInstance("SSL") | ||||||
|             sslContext.init(null, trustAllCerts, java.security.SecureRandom()) |         sslContext.init(null, trustAllCerts, java.security.SecureRandom()) | ||||||
|  |  | ||||||
|             val sslSocketFactory = sslContext.socketFactory |         val sslSocketFactory = sslContext.socketFactory | ||||||
|  |  | ||||||
|             OkHttpClient.Builder() |         OkHttpClient.Builder() | ||||||
|                     .sslSocketFactory(sslSocketFactory, trustAllCerts[0] as X509TrustManager) |             .sslSocketFactory(sslSocketFactory, trustAllCerts[0] as X509TrustManager) | ||||||
|                     .hostnameVerifier { _, _ -> true } |             .hostnameVerifier { _, _ -> true } | ||||||
|         } catch (e: Exception) { |     } catch (e: Exception) { | ||||||
|             throw RuntimeException(e) |         throw RuntimeException(e) | ||||||
|         } |     } | ||||||
| @@ -1,15 +1,20 @@ | |||||||
| package apps.amine.bou.readerforselfoss.utils | package apps.amine.bou.readerforselfoss.utils | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
| import android.text.format.DateUtils | import android.text.format.DateUtils | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Item | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossTagType | ||||||
| import java.text.ParseException | import java.text.ParseException | ||||||
| import java.text.SimpleDateFormat | import java.text.SimpleDateFormat | ||||||
| import java.util.* | import java.util.* | ||||||
|  |  | ||||||
| fun String.toTextDrawableString(): String { | fun String.toTextDrawableString(c: Context): String { | ||||||
|     val textDrawable = StringBuilder() |     val textDrawable = StringBuilder() | ||||||
|     for (s in this.split(" ".toRegex()).dropLastWhile { it.isEmpty() }.toTypedArray()) { |     for (s in this.split(" ".toRegex()).filter { !it.isEmpty() }.toTypedArray()) { | ||||||
|         textDrawable.append(s[0]) |         try { | ||||||
|  |             textDrawable.append(s[0]) | ||||||
|  |         } catch (e: StringIndexOutOfBoundsException) { | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|     return textDrawable.toString() |     return textDrawable.toString() | ||||||
| } | } | ||||||
| @@ -17,15 +22,29 @@ fun String.toTextDrawableString(): String { | |||||||
| fun Item.sourceAndDateText(): String { | fun Item.sourceAndDateText(): String { | ||||||
|     val formattedDate: String = try { |     val formattedDate: String = try { | ||||||
|         " " + DateUtils.getRelativeTimeSpanString( |         " " + DateUtils.getRelativeTimeSpanString( | ||||||
|                 SimpleDateFormat("yyyy-MM-dd HH:mm:ss").parse(this.datetime).time, |             SimpleDateFormat("yyyy-MM-dd HH:mm:ss").parse(this.datetime).time, | ||||||
|                 Date().time, |             Date().time, | ||||||
|                 DateUtils.MINUTE_IN_MILLIS, |             DateUtils.MINUTE_IN_MILLIS, | ||||||
|                 DateUtils.FORMAT_ABBREV_RELATIVE |             DateUtils.FORMAT_ABBREV_RELATIVE | ||||||
|         ) |         ) | ||||||
|     } catch (e: ParseException) { |     } catch (e: ParseException) { | ||||||
|         e.printStackTrace() |         e.printStackTrace() | ||||||
|         "" |         "" | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return this.sourcetitle + formattedDate |     return this.getSourceTitle() + formattedDate | ||||||
| } | } | ||||||
|  |  | ||||||
|  | fun Item.toggleStar(): Item { | ||||||
|  |     this.starred = !this.starred | ||||||
|  |     return this | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun List<Item>.flattenTags(): List<Item> = | ||||||
|  |     this.flatMap { | ||||||
|  |         val item = it | ||||||
|  |         val tags: List<String> = it.tags.tags.split(",") | ||||||
|  |         tags.map { t -> | ||||||
|  |             item.copy(tags = SelfossTagType(t.trim())) | ||||||
|  |         } | ||||||
|  |     } | ||||||
| @@ -2,12 +2,18 @@ package apps.amine.bou.readerforselfoss.utils | |||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
| import android.app.PendingIntent | import android.app.PendingIntent | ||||||
|  | import android.content.ActivityNotFoundException | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.graphics.BitmapFactory | import android.graphics.BitmapFactory | ||||||
| import android.net.Uri | import android.net.Uri | ||||||
| import android.support.customtabs.CustomTabsIntent | import android.text.Spannable | ||||||
|  | import android.text.style.ClickableSpan | ||||||
|  | import androidx.browser.customtabs.CustomTabsIntent | ||||||
| import android.util.Patterns | import android.util.Patterns | ||||||
|  | import android.view.MotionEvent | ||||||
|  | import android.view.View | ||||||
|  | import android.widget.TextView | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
| import apps.amine.bou.readerforselfoss.ReaderActivity | import apps.amine.bou.readerforselfoss.ReaderActivity | ||||||
| @@ -20,10 +26,10 @@ fun Context.buildCustomTabsIntent(): CustomTabsIntent { | |||||||
|     val actionIntent = Intent(Intent.ACTION_SEND) |     val actionIntent = Intent(Intent.ACTION_SEND) | ||||||
|     actionIntent.type = "text/plain" |     actionIntent.type = "text/plain" | ||||||
|     val createPendingShareIntent: PendingIntent = PendingIntent.getActivity( |     val createPendingShareIntent: PendingIntent = PendingIntent.getActivity( | ||||||
|             this, |         this, | ||||||
|             0, |         0, | ||||||
|             actionIntent, |         actionIntent, | ||||||
|             0 |         0 | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     val intentBuilder = CustomTabsIntent.Builder() |     val intentBuilder = CustomTabsIntent.Builder() | ||||||
| @@ -35,14 +41,14 @@ fun Context.buildCustomTabsIntent(): CustomTabsIntent { | |||||||
|  |  | ||||||
|  |  | ||||||
|     intentBuilder.setStartAnimations( |     intentBuilder.setStartAnimations( | ||||||
|             this, |         this, | ||||||
|             R.anim.slide_in_right, |         R.anim.slide_in_right, | ||||||
|             R.anim.slide_out_left |         R.anim.slide_out_left | ||||||
|     ) |     ) | ||||||
|     intentBuilder.setExitAnimations( |     intentBuilder.setExitAnimations( | ||||||
|             this, |         this, | ||||||
|             android.R.anim.slide_in_left, |         android.R.anim.slide_in_left, | ||||||
|             android.R.anim.slide_out_right |         android.R.anim.slide_out_right | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     val closeicon = BitmapFactory.decodeResource(resources, R.drawable.ic_close_white_24dp) |     val closeicon = BitmapFactory.decodeResource(resources, R.drawable.ic_close_white_24dp) | ||||||
| @@ -50,8 +56,8 @@ fun Context.buildCustomTabsIntent(): CustomTabsIntent { | |||||||
|  |  | ||||||
|     val shareLabel = this.getString(R.string.label_share) |     val shareLabel = this.getString(R.string.label_share) | ||||||
|     val icon = BitmapFactory.decodeResource( |     val icon = BitmapFactory.decodeResource( | ||||||
|             resources, |         resources, | ||||||
|             R.drawable.ic_share_white_24dp |         R.drawable.ic_share_white_24dp | ||||||
|     ) |     ) | ||||||
|     intentBuilder.setActionButton(icon, shareLabel, createPendingShareIntent) |     intentBuilder.setActionButton(icon, shareLabel, createPendingShareIntent) | ||||||
|  |  | ||||||
| @@ -59,37 +65,24 @@ fun Context.buildCustomTabsIntent(): CustomTabsIntent { | |||||||
| } | } | ||||||
|  |  | ||||||
| fun Context.openItemUrlInternally( | fun Context.openItemUrlInternally( | ||||||
|         linkDecoded: String, |     allItems: ArrayList<Item>, | ||||||
|         content: String, |     currentItem: Int, | ||||||
|         image: String, |     linkDecoded: String, | ||||||
|         title: String, |     customTabsIntent: CustomTabsIntent, | ||||||
|         source: String, |     articleViewer: Boolean, | ||||||
|         customTabsIntent: CustomTabsIntent, |     app: Activity | ||||||
|         articleViewer: Boolean, |  | ||||||
|         app: Activity |  | ||||||
| ) { | ) { | ||||||
|     if (articleViewer) { |     if (articleViewer) { | ||||||
|  |         ReaderActivity.allItems = allItems | ||||||
|         val intent = Intent(this, ReaderActivity::class.java) |         val intent = Intent(this, ReaderActivity::class.java) | ||||||
|  |         intent.putExtra("currentItem", currentItem) | ||||||
|         /*DragDismissIntentBuilder(this) |  | ||||||
|                 .setFullscreenOnTablets(true)      // defaults to false, tablets will have padding on each side |  | ||||||
|                 .setDragElasticity(DragDismissIntentBuilder.DragElasticity.NORMAL)  // Larger elasticities will make it easier to dismiss. |  | ||||||
|                 .setDrawUnderStatusBar(true) |  | ||||||
|                 .build(intent)*/ |  | ||||||
|  |  | ||||||
|  |  | ||||||
|         intent.putExtra("url", linkDecoded) |  | ||||||
|         intent.putExtra("content", content) |  | ||||||
|         intent.putExtra("title", title) |  | ||||||
|         intent.putExtra("image", image) |  | ||||||
|         intent.putExtra("source", source) |  | ||||||
|         app.startActivity(intent) |         app.startActivity(intent) | ||||||
|     } else { |     } else { | ||||||
|         try { |         try { | ||||||
|             CustomTabActivityHelper.openCustomTab( |             CustomTabActivityHelper.openCustomTab( | ||||||
|                     app, |                 app, | ||||||
|                     customTabsIntent, |                 customTabsIntent, | ||||||
|                     Uri.parse(linkDecoded) |                 Uri.parse(linkDecoded) | ||||||
|             ) { _, uri -> |             ) { _, uri -> | ||||||
|                 val intent = Intent(Intent.ACTION_VIEW, uri) |                 val intent = Intent(Intent.ACTION_VIEW, uri) | ||||||
|                 intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK |                 intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||||
| @@ -102,36 +95,32 @@ fun Context.openItemUrlInternally( | |||||||
| } | } | ||||||
|  |  | ||||||
| fun Context.openItemUrl( | fun Context.openItemUrl( | ||||||
|         linkDecoded: String, |     allItems: ArrayList<Item>, | ||||||
|         content: String, |     currentItem: Int, | ||||||
|         image: String, |     linkDecoded: String, | ||||||
|         title: String, |     customTabsIntent: CustomTabsIntent, | ||||||
|         source: String, |     internalBrowser: Boolean, | ||||||
|         customTabsIntent: CustomTabsIntent, |     articleViewer: Boolean, | ||||||
|         internalBrowser: Boolean, |     app: Activity | ||||||
|         articleViewer: Boolean, |  | ||||||
|         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 (!internalBrowser) { |         if (!internalBrowser) { | ||||||
|             openInBrowser(linkDecoded, app) |             openInBrowser(linkDecoded, app) | ||||||
|         } else { |         } else { | ||||||
|             this.openItemUrlInternally( |             this.openItemUrlInternally( | ||||||
|                     linkDecoded, |                 allItems, | ||||||
|                     content, |                 currentItem, | ||||||
|                     image, |                 linkDecoded, | ||||||
|                     title, |                 customTabsIntent, | ||||||
|                     source, |                 articleViewer, | ||||||
|                     customTabsIntent, |                 app | ||||||
|                     articleViewer, |  | ||||||
|                     app |  | ||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -140,13 +129,17 @@ fun Context.openItemUrl( | |||||||
| private fun openInBrowser(linkDecoded: String, app: Activity) { | private fun openInBrowser(linkDecoded: String, app: Activity) { | ||||||
|     val intent = Intent(Intent.ACTION_VIEW) |     val intent = Intent(Intent.ACTION_VIEW) | ||||||
|     intent.data = Uri.parse(linkDecoded) |     intent.data = Uri.parse(linkDecoded) | ||||||
|     app.startActivity(intent) |     try { | ||||||
|  |         app.startActivity(intent) | ||||||
|  |     } catch (e: ActivityNotFoundException) { | ||||||
|  |         Toast.makeText(app.baseContext, e.message, Toast.LENGTH_LONG).show() | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| fun String.isUrlValid(): Boolean = | fun String.isUrlValid(): Boolean = | ||||||
|         HttpUrl.parse(this) != null && Patterns.WEB_URL.matcher(this).matches() |     HttpUrl.parse(this) != null && Patterns.WEB_URL.matcher(this).matches() | ||||||
|  |  | ||||||
| fun String.isBaseUrlValid(): Boolean { | fun String.isBaseUrlValid(ctx: Context): Boolean { | ||||||
|     val baseUrl = HttpUrl.parse(this) |     val baseUrl = HttpUrl.parse(this) | ||||||
|     var existsAndEndsWithSlash = false |     var existsAndEndsWithSlash = false | ||||||
|     if (baseUrl != null) { |     if (baseUrl != null) { | ||||||
| @@ -163,3 +156,40 @@ fun Context.openInBrowserAsNewTask(i: Item) { | |||||||
|     intent.data = Uri.parse(i.getLinkDecoded().toStringUriWithHttp()) |     intent.data = Uri.parse(i.getLinkDecoded().toStringUriWithHttp()) | ||||||
|     startActivity(intent) |     startActivity(intent) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class LinkOnTouchListener: View.OnTouchListener { | ||||||
|  |     override fun onTouch(v: View?, event: MotionEvent?): Boolean { | ||||||
|  |         var ret = false | ||||||
|  |         val widget: TextView = v as TextView | ||||||
|  |         val text: CharSequence = widget.text | ||||||
|  |         val stext = Spannable.Factory.getInstance().newSpannable(text) | ||||||
|  |  | ||||||
|  |         val action = event!!.action | ||||||
|  |  | ||||||
|  |         if (action == MotionEvent.ACTION_UP || | ||||||
|  |             action == MotionEvent.ACTION_DOWN) { | ||||||
|  |             var x: Float = event.x | ||||||
|  |             var y: Float = event.y | ||||||
|  |  | ||||||
|  |             x -= widget.totalPaddingLeft | ||||||
|  |             y -= widget.totalPaddingTop | ||||||
|  |  | ||||||
|  |             x += widget.scrollX | ||||||
|  |             y += widget.scrollY | ||||||
|  |  | ||||||
|  |             val layout = widget.layout | ||||||
|  |             val line = layout.getLineForVertical(y.toInt()) | ||||||
|  |             val off = layout.getOffsetForHorizontal(line, x) | ||||||
|  |  | ||||||
|  |             val link = stext.getSpans(off, off, ClickableSpan::class.java) | ||||||
|  |  | ||||||
|  |             if (link.isNotEmpty()) { | ||||||
|  |                 if (action == MotionEvent.ACTION_UP) { | ||||||
|  |                     link[0].onClick(widget) | ||||||
|  |                 } | ||||||
|  |                 ret = true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return ret | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,53 +0,0 @@ | |||||||
| package apps.amine.bou.readerforselfoss.utils |  | ||||||
|  |  | ||||||
| import android.content.Context |  | ||||||
| import android.support.design.widget.CoordinatorLayout |  | ||||||
| import android.support.design.widget.FloatingActionButton |  | ||||||
| import android.util.AttributeSet |  | ||||||
| import android.view.View |  | ||||||
|  |  | ||||||
| class ScrollAwareFABBehavior( |  | ||||||
|         context: Context, |  | ||||||
|         attrs: AttributeSet |  | ||||||
| ) : CoordinatorLayout.Behavior<FloatingActionButton>() { |  | ||||||
|  |  | ||||||
|     override fun onStartNestedScroll( |  | ||||||
|             coordinatorLayout: CoordinatorLayout, |  | ||||||
|             child: FloatingActionButton, |  | ||||||
|             directTargetChild: View, |  | ||||||
|             target: View, |  | ||||||
|             nestedScrollAxes: Int |  | ||||||
|     ): Boolean { |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun onNestedScroll( |  | ||||||
|             coordinatorLayout: CoordinatorLayout, |  | ||||||
|             child: FloatingActionButton, |  | ||||||
|             target: View, |  | ||||||
|             dxConsumed: Int, |  | ||||||
|             dyConsumed: Int, |  | ||||||
|             dxUnconsumed: Int, |  | ||||||
|             dyUnconsumed: Int |  | ||||||
|     ) { |  | ||||||
|         super.onNestedScroll( |  | ||||||
|                 coordinatorLayout, |  | ||||||
|                 child, |  | ||||||
|                 target, |  | ||||||
|                 dxConsumed, |  | ||||||
|                 dyConsumed, |  | ||||||
|                 dxUnconsumed, |  | ||||||
|                 dyUnconsumed |  | ||||||
|         ) |  | ||||||
|         if (dyConsumed > 0 && child.visibility == View.VISIBLE) { |  | ||||||
|             child.hide(object : FloatingActionButton.OnVisibilityChangedListener() { |  | ||||||
|                 override fun onHidden(fab: FloatingActionButton?) { |  | ||||||
|                     super.onHidden(fab) |  | ||||||
|                     fab!!.visibility = View.INVISIBLE |  | ||||||
|                 } |  | ||||||
|             }) |  | ||||||
|         } else if (dyConsumed < 0 && child.visibility != View.VISIBLE) { |  | ||||||
|             child.show() |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.utils | ||||||
|  |  | ||||||
|  | import android.content.res.Resources | ||||||
|  |  | ||||||
|  | val Int.toPx: Int | ||||||
|  |     get() = (this * Resources.getSystem().displayMetrics.density).toInt() | ||||||
|  |  | ||||||
|  | val Int.toDp: Int | ||||||
|  |     get() = (this / Resources.getSystem().displayMetrics.density).toInt() | ||||||
| @@ -9,4 +9,4 @@ fun TextBadgeItem.removeBadge(): TextBadgeItem { | |||||||
| } | } | ||||||
|  |  | ||||||
| fun TextBadgeItem.maybeShow(): TextBadgeItem = | fun TextBadgeItem.maybeShow(): TextBadgeItem = | ||||||
|         if (this.isHidden) this.show() else this |     if (this.isHidden) this.show() else this | ||||||
|   | |||||||
| @@ -1,15 +1,15 @@ | |||||||
| package apps.amine.bou.readerforselfoss.utils.customtabs; | package apps.amine.bou.readerforselfoss.utils.customtabs; | ||||||
|  |  | ||||||
|  |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| import android.app.Activity; | import android.app.Activity; | ||||||
| import android.net.Uri; | import android.net.Uri; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.support.customtabs.CustomTabsClient; | import androidx.browser.customtabs.CustomTabsClient; | ||||||
| import android.support.customtabs.CustomTabsIntent; | import androidx.browser.customtabs.CustomTabsIntent; | ||||||
| import android.support.customtabs.CustomTabsServiceConnection; | import androidx.browser.customtabs.CustomTabsServiceConnection; | ||||||
| import android.support.customtabs.CustomTabsSession; | import androidx.browser.customtabs.CustomTabsSession; | ||||||
|  |  | ||||||
|  | import java.util.List; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * This is a helper class to manage the connection to the Custom Tabs Service. |  * This is a helper class to manage the connection to the Custom Tabs Service. | ||||||
| @@ -23,15 +23,15 @@ public class CustomTabActivityHelper implements ServiceConnectionCallback { | |||||||
|     /** |     /** | ||||||
|      * Opens the URL on a Custom Tab if possible. Otherwise fallsback to opening it on a WebView. |      * Opens the URL on a Custom Tab if possible. Otherwise fallsback to opening it on a WebView. | ||||||
|      * |      * | ||||||
|      * @param activity The host activity. |      * @param activity         The host activity. | ||||||
|      * @param customTabsIntent a CustomTabsIntent to be used if Custom Tabs is available. |      * @param customTabsIntent a CustomTabsIntent to be used if Custom Tabs is available. | ||||||
|      * @param uri the Uri to be opened. |      * @param uri              the Uri to be opened. | ||||||
|      * @param fallback a CustomTabFallback to be used if Custom Tabs is not available. |      * @param fallback         a CustomTabFallback to be used if Custom Tabs is not available. | ||||||
|      */ |      */ | ||||||
|     public static void openCustomTab(Activity activity, |     public static void openCustomTab(Activity activity, | ||||||
|             CustomTabsIntent customTabsIntent, |                                      CustomTabsIntent customTabsIntent, | ||||||
|             Uri uri, |                                      Uri uri, | ||||||
|             CustomTabFallback fallback) { |                                      CustomTabFallback fallback) { | ||||||
|         String packageName = CustomTabsHelper.getPackageNameToUse(activity); |         String packageName = CustomTabsHelper.getPackageNameToUse(activity); | ||||||
|  |  | ||||||
|         //If we cant find a package name, it means theres no browser that supports |         //If we cant find a package name, it means theres no browser that supports | ||||||
| @@ -48,6 +48,7 @@ public class CustomTabActivityHelper implements ServiceConnectionCallback { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Unbinds the Activity from the Custom Tabs Service. |      * Unbinds the Activity from the Custom Tabs Service. | ||||||
|  |      * | ||||||
|      * @param activity the activity that is connected to the service. |      * @param activity the activity that is connected to the service. | ||||||
|      */ |      */ | ||||||
|     public void unbindCustomTabsService(Activity activity) { |     public void unbindCustomTabsService(Activity activity) { | ||||||
| @@ -74,6 +75,7 @@ public class CustomTabActivityHelper implements ServiceConnectionCallback { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Register a Callback to be called when connected or disconnected from the Custom Tabs Service. |      * Register a Callback to be called when connected or disconnected from the Custom Tabs Service. | ||||||
|  |      * | ||||||
|      * @param connectionCallback |      * @param connectionCallback | ||||||
|      */ |      */ | ||||||
|     public void setConnectionCallback(ConnectionCallback connectionCallback) { |     public void setConnectionCallback(ConnectionCallback connectionCallback) { | ||||||
| @@ -82,6 +84,7 @@ public class CustomTabActivityHelper implements ServiceConnectionCallback { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Binds the Activity to the Custom Tabs Service. |      * Binds the Activity to the Custom Tabs Service. | ||||||
|  |      * | ||||||
|      * @param activity the activity to be binded to the service. |      * @param activity the activity to be binded to the service. | ||||||
|      */ |      */ | ||||||
|     public void bindCustomTabsService(Activity activity) { |     public void bindCustomTabsService(Activity activity) { | ||||||
| @@ -95,16 +98,15 @@ public class CustomTabActivityHelper implements ServiceConnectionCallback { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @see {@link CustomTabsSession#mayLaunchUrl(Uri, Bundle, List)}. |  | ||||||
|      * @return true if call to mayLaunchUrl was accepted. |      * @return true if call to mayLaunchUrl was accepted. | ||||||
|  |      * @see {@link CustomTabsSession#mayLaunchUrl(Uri, Bundle, List)}. | ||||||
|      */ |      */ | ||||||
|     public boolean mayLaunchUrl(Uri uri, Bundle extras, List<Bundle> otherLikelyBundles) { |     public boolean mayLaunchUrl(Uri uri, Bundle extras, List<Bundle> otherLikelyBundles) { | ||||||
|         if (mClient == null) return false; |         if (mClient == null) return false; | ||||||
|  |  | ||||||
|         CustomTabsSession session = getSession(); |         CustomTabsSession session = getSession(); | ||||||
|         if (session == null) return false; |         return session != null && session.mayLaunchUrl(uri, extras, otherLikelyBundles); | ||||||
|  |  | ||||||
|         return session.mayLaunchUrl(uri, extras, otherLikelyBundles); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -142,9 +144,8 @@ public class CustomTabActivityHelper implements ServiceConnectionCallback { | |||||||
|      */ |      */ | ||||||
|     public interface CustomTabFallback { |     public interface CustomTabFallback { | ||||||
|         /** |         /** | ||||||
|          * |  | ||||||
|          * @param activity The Activity that wants to open the Uri. |          * @param activity The Activity that wants to open the Uri. | ||||||
|          * @param uri The uri to be opened by the fallback. |          * @param uri      The uri to be opened by the fallback. | ||||||
|          */ |          */ | ||||||
|         void openUri(Activity activity, Uri uri); |         void openUri(Activity activity, Uri uri); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,19 +1,19 @@ | |||||||
| package apps.amine.bou.readerforselfoss.utils.customtabs; | package apps.amine.bou.readerforselfoss.utils.customtabs; | ||||||
|  |  | ||||||
|  |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| import android.content.Context; | import android.content.Context; | ||||||
| import android.content.Intent; | import android.content.Intent; | ||||||
| import android.content.IntentFilter; | import android.content.IntentFilter; | ||||||
| import android.content.pm.PackageManager; | import android.content.pm.PackageManager; | ||||||
| import android.content.pm.ResolveInfo; | import android.content.pm.ResolveInfo; | ||||||
| import android.net.Uri; | import android.net.Uri; | ||||||
| import android.support.customtabs.CustomTabsService; | import androidx.browser.customtabs.CustomTabsService; | ||||||
| import android.text.TextUtils; | import android.text.TextUtils; | ||||||
| import android.util.Log; | import android.util.Log; | ||||||
|  |  | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.List; | ||||||
|  |  | ||||||
| import apps.amine.bou.readerforselfoss.utils.customtabs.helpers.KeepAliveService; | import apps.amine.bou.readerforselfoss.utils.customtabs.helpers.KeepAliveService; | ||||||
|  |  | ||||||
| @SuppressWarnings("ALL") | @SuppressWarnings("ALL") | ||||||
| @@ -28,7 +28,8 @@ class CustomTabsHelper { | |||||||
|  |  | ||||||
|     private static String sPackageNameToUse; |     private static String sPackageNameToUse; | ||||||
|  |  | ||||||
|     private CustomTabsHelper() {} |     private CustomTabsHelper() { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public static void addKeepAliveExtra(Context context, Intent intent) { |     public static void addKeepAliveExtra(Context context, Intent intent) { | ||||||
|         Intent keepAliveIntent = new Intent().setClassName( |         Intent keepAliveIntent = new Intent().setClassName( | ||||||
| @@ -40,7 +41,7 @@ class CustomTabsHelper { | |||||||
|      * Goes through all apps that handle VIEW intents and have a warmup service. Picks |      * Goes through all apps that handle VIEW intents and have a warmup service. Picks | ||||||
|      * the one chosen by the user if there is one, otherwise makes a best effort to return a |      * the one chosen by the user if there is one, otherwise makes a best effort to return a | ||||||
|      * valid package name. |      * valid package name. | ||||||
|      * |      * <p> | ||||||
|      * This is <strong>not</strong> threadsafe. |      * This is <strong>not</strong> threadsafe. | ||||||
|      * |      * | ||||||
|      * @param context {@link Context} to use for accessing {@link PackageManager}. |      * @param context {@link Context} to use for accessing {@link PackageManager}. | ||||||
| @@ -94,6 +95,7 @@ class CustomTabsHelper { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Used to check whether there is a specialized handler for a given intent. |      * Used to check whether there is a specialized handler for a given intent. | ||||||
|  |      * | ||||||
|      * @param intent The intent to check with. |      * @param intent The intent to check with. | ||||||
|      * @return Whether there is a specialized handler for the given intent. |      * @return Whether there is a specialized handler for the given intent. | ||||||
|      */ |      */ | ||||||
|   | |||||||
| @@ -1,11 +1,11 @@ | |||||||
| package apps.amine.bou.readerforselfoss.utils.customtabs; | package apps.amine.bou.readerforselfoss.utils.customtabs; | ||||||
|  |  | ||||||
|  |  | ||||||
| import java.lang.ref.WeakReference; |  | ||||||
|  |  | ||||||
| import android.content.ComponentName; | import android.content.ComponentName; | ||||||
| import android.support.customtabs.CustomTabsClient; | import androidx.browser.customtabs.CustomTabsClient; | ||||||
| import android.support.customtabs.CustomTabsServiceConnection; | import androidx.browser.customtabs.CustomTabsServiceConnection; | ||||||
|  |  | ||||||
|  | import java.lang.ref.WeakReference; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Implementation for the CustomTabsServiceConnection that avoids leaking the |  * Implementation for the CustomTabsServiceConnection that avoids leaking the | ||||||
|   | |||||||
| @@ -1,12 +1,13 @@ | |||||||
| package apps.amine.bou.readerforselfoss.utils.customtabs; | package apps.amine.bou.readerforselfoss.utils.customtabs; | ||||||
|  |  | ||||||
|  |  | ||||||
| import android.support.customtabs.CustomTabsClient; | import androidx.browser.customtabs.CustomTabsClient; | ||||||
|  |  | ||||||
|  |  | ||||||
| public interface ServiceConnectionCallback { | public interface ServiceConnectionCallback { | ||||||
|     /** |     /** | ||||||
|      * Called when the service is connected. |      * Called when the service is connected. | ||||||
|  |      * | ||||||
|      * @param client a CustomTabsClient |      * @param client a CustomTabsClient | ||||||
|      */ |      */ | ||||||
|     void onServiceConnected(CustomTabsClient client); |     void onServiceConnected(CustomTabsClient client); | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| /* From https://github.com/mikepenz/MaterialDrawer/blob/develop/app/src/main/java/com/mikepenz/materialdrawer/app/drawerItems/CustomBaseViewHolder.java */ | /* From https://github.com/mikepenz/MaterialDrawer/blob/develop/app/src/main/java/com/mikepenz/materialdrawer/app/drawerItems/CustomBaseViewHolder.java */ | ||||||
| package apps.amine.bou.readerforselfoss.utils.drawer | package apps.amine.bou.readerforselfoss.utils.drawer | ||||||
|  |  | ||||||
| import android.support.v7.widget.RecyclerView | import androidx.recyclerview.widget.RecyclerView | ||||||
| import android.view.View | import android.view.View | ||||||
| import android.widget.ImageView | import android.widget.ImageView | ||||||
| import android.widget.TextView | import android.widget.TextView | ||||||
|   | |||||||
| @@ -1,111 +0,0 @@ | |||||||
| /* From https://github.com/mikepenz/MaterialDrawer/blob/develop/app/src/main/java/com/mikepenz/materialdrawer/app/drawerItems/CustomUrlBasePrimaryDrawerItem.java */ |  | ||||||
| package apps.amine.bou.readerforselfoss.utils.drawer |  | ||||||
|  |  | ||||||
| import android.net.Uri |  | ||||||
| import android.support.annotation.ColorInt |  | ||||||
| import android.support.annotation.ColorRes |  | ||||||
| import android.support.annotation.StringRes |  | ||||||
| import android.support.v7.widget.RecyclerView |  | ||||||
|  |  | ||||||
| import com.mikepenz.materialdrawer.holder.ColorHolder |  | ||||||
| import com.mikepenz.materialdrawer.holder.ImageHolder |  | ||||||
| import com.mikepenz.materialdrawer.holder.StringHolder |  | ||||||
| import com.mikepenz.materialdrawer.model.BaseDrawerItem |  | ||||||
| import com.mikepenz.materialdrawer.util.DrawerImageLoader |  | ||||||
| import com.mikepenz.materialdrawer.util.DrawerUIUtils |  | ||||||
| import com.mikepenz.materialize.util.UIUtils |  | ||||||
|  |  | ||||||
| abstract class CustomUrlBasePrimaryDrawerItem<T, VH : RecyclerView.ViewHolder> : BaseDrawerItem<T, VH>() { |  | ||||||
|     fun withIcon(url: String): T { |  | ||||||
|         this.icon = ImageHolder(url) |  | ||||||
|         return this as T |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun withIcon(uri: Uri): T { |  | ||||||
|         this.icon = ImageHolder(uri) |  | ||||||
|         return this as T |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     var description: StringHolder? = null |  | ||||||
|         private set |  | ||||||
|     var descriptionTextColor: ColorHolder? = null |  | ||||||
|         private set |  | ||||||
|  |  | ||||||
|     fun withDescription(description: String): T { |  | ||||||
|         this.description = StringHolder(description) |  | ||||||
|         return this as T |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun withDescription(@StringRes descriptionRes: Int): T { |  | ||||||
|         this.description = StringHolder(descriptionRes) |  | ||||||
|         return this as T |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun withDescriptionTextColor(@ColorInt color: Int): T { |  | ||||||
|         this.descriptionTextColor = ColorHolder.fromColor(color) |  | ||||||
|         return this as T |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun withDescriptionTextColorRes(@ColorRes colorRes: Int): T { |  | ||||||
|         this.descriptionTextColor = ColorHolder.fromColorRes(colorRes) |  | ||||||
|         return this as T |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * a helper method to have the logic for all secondaryDrawerItems only once |  | ||||||
|  |  | ||||||
|      * @param viewHolder |  | ||||||
|      */ |  | ||||||
|     protected fun bindViewHelper(viewHolder: CustomBaseViewHolder) { |  | ||||||
|         val ctx = viewHolder.itemView.context |  | ||||||
|  |  | ||||||
|         //set the identifier from the drawerItem here. It can be used to run tests |  | ||||||
|         viewHolder.itemView.id = hashCode() |  | ||||||
|  |  | ||||||
|         //set the item selected if it is |  | ||||||
|         viewHolder.itemView.isSelected = isSelected |  | ||||||
|  |  | ||||||
|         //get the correct color for the background |  | ||||||
|         val selectedColor = getSelectedColor(ctx) |  | ||||||
|         //get the correct color for the text |  | ||||||
|         val color = getColor(ctx) |  | ||||||
|         val selectedTextColor = getSelectedTextColor(ctx) |  | ||||||
|         //get the correct color for the icon |  | ||||||
|         val iconColor = getIconColor(ctx) |  | ||||||
|         val selectedIconColor = getSelectedIconColor(ctx) |  | ||||||
|  |  | ||||||
|         //set the background for the item |  | ||||||
|         UIUtils.setBackground( |  | ||||||
|                 viewHolder.view, |  | ||||||
|                 UIUtils.getSelectableBackground(ctx, selectedColor, true) |  | ||||||
|         ) |  | ||||||
|         //set the text for the name |  | ||||||
|         StringHolder.applyTo(this.getName(), viewHolder.name) |  | ||||||
|         //set the text for the description or hide |  | ||||||
|         StringHolder.applyToOrHide(this.description, viewHolder.description) |  | ||||||
|  |  | ||||||
|         //set the colors for textViews |  | ||||||
|         viewHolder.name.setTextColor(getTextColorStateList(color, selectedTextColor)) |  | ||||||
|         //set the description text color |  | ||||||
|         ColorHolder.applyToOr( |  | ||||||
|                 descriptionTextColor, |  | ||||||
|                 viewHolder.description, |  | ||||||
|                 getTextColorStateList(color, selectedTextColor) |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         //define the typeface for our textViews |  | ||||||
|         if (getTypeface() != null) { |  | ||||||
|             viewHolder.name.typeface = getTypeface() |  | ||||||
|             viewHolder.description.typeface = getTypeface() |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         //we make sure we reset the image first before setting the new one in case there is an empty one |  | ||||||
|         DrawerImageLoader.getInstance().cancelImage(viewHolder.icon) |  | ||||||
|         viewHolder.icon.setImageBitmap(null) |  | ||||||
|         //get the drawables for our icon and set it |  | ||||||
|         ImageHolder.applyTo(icon, viewHolder.icon, "customUrlItem") |  | ||||||
|  |  | ||||||
|         //for android API 17 --> Padding not applied via xml |  | ||||||
|         DrawerUIUtils.setDrawerVerticalPadding(viewHolder.view) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,92 +0,0 @@ | |||||||
| /* From https://github.com/mikepenz/MaterialDrawer/blob/develop/app/src/main/java/com/mikepenz/materialdrawer/app/drawerItems/CustomUrlPrimaryDrawerItem.java */ |  | ||||||
| package apps.amine.bou.readerforselfoss.utils.drawer |  | ||||||
|  |  | ||||||
| import android.support.annotation.LayoutRes |  | ||||||
| import android.support.annotation.StringRes |  | ||||||
| import android.view.View |  | ||||||
| import android.widget.TextView |  | ||||||
| import apps.amine.bou.readerforselfoss.R |  | ||||||
| import com.mikepenz.materialdrawer.holder.BadgeStyle |  | ||||||
| import com.mikepenz.materialdrawer.holder.StringHolder |  | ||||||
| import com.mikepenz.materialdrawer.model.interfaces.ColorfulBadgeable |  | ||||||
|  |  | ||||||
| class CustomUrlPrimaryDrawerItem : CustomUrlBasePrimaryDrawerItem<CustomUrlPrimaryDrawerItem, CustomUrlPrimaryDrawerItem.ViewHolder>(), ColorfulBadgeable<CustomUrlPrimaryDrawerItem> { |  | ||||||
|     protected var mBadge: StringHolder = StringHolder("") |  | ||||||
|     protected var mBadgeStyle = BadgeStyle() |  | ||||||
|  |  | ||||||
|     override fun withBadge(badge: StringHolder): CustomUrlPrimaryDrawerItem { |  | ||||||
|         this.mBadge = badge |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun withBadge(badge: String): CustomUrlPrimaryDrawerItem { |  | ||||||
|         this.mBadge = StringHolder(badge) |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun withBadge(@StringRes badgeRes: Int): CustomUrlPrimaryDrawerItem { |  | ||||||
|         this.mBadge = StringHolder(badgeRes) |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun withBadgeStyle(badgeStyle: BadgeStyle): CustomUrlPrimaryDrawerItem { |  | ||||||
|         this.mBadgeStyle = badgeStyle |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun getBadge(): StringHolder { |  | ||||||
|         return mBadge |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun getBadgeStyle(): BadgeStyle { |  | ||||||
|         return mBadgeStyle |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun getType(): Int { |  | ||||||
|         return R.id.material_drawer_item_custom_url_item |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @LayoutRes |  | ||||||
|     override fun getLayoutRes(): Int { |  | ||||||
|         return R.layout.material_drawer_item_primary |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun bindView(viewHolder: ViewHolder, payloads: List<*>?) { |  | ||||||
|         super.bindView(viewHolder, payloads) |  | ||||||
|  |  | ||||||
|         val ctx = viewHolder.itemView.context |  | ||||||
|  |  | ||||||
|         //bind the basic view parts |  | ||||||
|         bindViewHelper(viewHolder) |  | ||||||
|  |  | ||||||
|         //set the text for the badge or hide |  | ||||||
|         val badgeVisible = StringHolder.applyToOrHide(mBadge, viewHolder.badge) |  | ||||||
|         //style the badge if it is visible |  | ||||||
|         if (badgeVisible) { |  | ||||||
|             mBadgeStyle.style( |  | ||||||
|                     viewHolder.badge, |  | ||||||
|                     getTextColorStateList(getColor(ctx), getSelectedTextColor(ctx)) |  | ||||||
|             ) |  | ||||||
|             viewHolder.badgeContainer.visibility = View.VISIBLE |  | ||||||
|         } else { |  | ||||||
|             viewHolder.badgeContainer.visibility = View.GONE |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         //define the typeface for our textViews |  | ||||||
|         if (getTypeface() != null) { |  | ||||||
|             viewHolder.badge.typeface = getTypeface() |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         //call the onPostBindView method to trigger post bind view actions (like the listener to modify the item if required) |  | ||||||
|         onPostBindView(this, viewHolder.itemView) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun getViewHolder(v: View): ViewHolder { |  | ||||||
|         return ViewHolder(v) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     class ViewHolder(view: View) : CustomBaseViewHolder(view) { |  | ||||||
|         val badgeContainer: View = view.findViewById(R.id.material_drawer_badge_container) |  | ||||||
|         val badge: TextView = view.findViewById(R.id.material_drawer_badge) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -2,38 +2,68 @@ package apps.amine.bou.readerforselfoss.utils.glide | |||||||
|  |  | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.graphics.Bitmap | import android.graphics.Bitmap | ||||||
| import android.support.v4.graphics.drawable.RoundedBitmapDrawableFactory | import android.graphics.drawable.Drawable | ||||||
|  | import android.util.Base64 | ||||||
|  | import androidx.core.graphics.drawable.RoundedBitmapDrawableFactory | ||||||
| import android.widget.ImageView | import android.widget.ImageView | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
|  | import com.bumptech.glide.RequestBuilder | ||||||
|  | import com.bumptech.glide.RequestManager | ||||||
|  | import com.bumptech.glide.load.model.GlideUrl | ||||||
|  | import com.bumptech.glide.load.model.LazyHeaders | ||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
| import com.bumptech.glide.request.target.BitmapImageViewTarget | import com.bumptech.glide.request.target.BitmapImageViewTarget | ||||||
|  | import java.io.ByteArrayInputStream | ||||||
|  | import java.io.ByteArrayOutputStream | ||||||
|  | import java.io.InputStream | ||||||
|  |  | ||||||
| fun Context.bitmapCenterCrop(url: String, iv: ImageView) = | fun Context.bitmapCenterCrop(config: Config, url: String, iv: ImageView) = | ||||||
|         Glide.with(this) |     Glide.with(this) | ||||||
|                 .asBitmap() |         .asBitmap() | ||||||
|                 .load(url) |         .loadMaybeBasicAuth(config, url) | ||||||
|                 .apply(RequestOptions.centerCropTransform()) |         .apply(RequestOptions.centerCropTransform()) | ||||||
|                 .into(iv) |         .into(iv) | ||||||
|  |  | ||||||
| fun Context.bitmapFitCenter(url: String, iv: ImageView) = | fun Context.circularBitmapDrawable(config: Config, url: String, iv: ImageView) = | ||||||
|         Glide.with(this) |     Glide.with(this) | ||||||
|                 .asBitmap() |         .asBitmap() | ||||||
|                 .load(url) |         .loadMaybeBasicAuth(config, url) | ||||||
|                 .apply(RequestOptions.fitCenterTransform()) |         .apply(RequestOptions.centerCropTransform()) | ||||||
|                 .into(iv) |         .into(object : BitmapImageViewTarget(iv) { | ||||||
|  |             override fun setResource(resource: Bitmap?) { | ||||||
|  |                 val circularBitmapDrawable = RoundedBitmapDrawableFactory.create( | ||||||
|  |                     resources, | ||||||
|  |                     resource | ||||||
|  |                 ) | ||||||
|  |                 circularBitmapDrawable.isCircular = true | ||||||
|  |                 iv.setImageDrawable(circularBitmapDrawable) | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
| fun Context.circularBitmapDrawable(url: String, iv: ImageView) = | fun RequestBuilder<Bitmap>.loadMaybeBasicAuth(config: Config, url: String): RequestBuilder<Bitmap> { | ||||||
|         Glide.with(this) |     val builder: LazyHeaders.Builder = LazyHeaders.Builder() | ||||||
|                 .asBitmap() |     if (config.httpUserLogin.isNotEmpty() || config.httpUserPassword.isNotEmpty()) { | ||||||
|                 .load(url) |         val basicAuth = "Basic " + Base64.encodeToString("${config.httpUserLogin}:${config.httpUserPassword}".toByteArray(), Base64.NO_WRAP) | ||||||
|                 .apply(RequestOptions.centerCropTransform()) |         builder.addHeader("Authorization", basicAuth) | ||||||
|                 .into(object : BitmapImageViewTarget(iv) { |     } | ||||||
|                     override fun setResource(resource: Bitmap?) { |     val glideUrl = GlideUrl(url, builder.build()) | ||||||
|                         val circularBitmapDrawable = RoundedBitmapDrawableFactory.create( |     return this.load(glideUrl) | ||||||
|                                 resources, | } | ||||||
|                                 resource |  | ||||||
|                         ) | fun RequestManager.loadMaybeBasicAuth(config: Config, url: String): RequestBuilder<Drawable> { | ||||||
|                         circularBitmapDrawable.isCircular = true |     val builder: LazyHeaders.Builder = LazyHeaders.Builder() | ||||||
|                         iv.setImageDrawable(circularBitmapDrawable) |     if (config.httpUserLogin.isNotEmpty() || config.httpUserPassword.isNotEmpty()) { | ||||||
|                     } |         val basicAuth = "Basic " + Base64.encodeToString("${config.httpUserLogin}:${config.httpUserPassword}".toByteArray(), Base64.NO_WRAP) | ||||||
|                 }) |         builder.addHeader("Authorization", basicAuth) | ||||||
|  |     } | ||||||
|  |     val glideUrl = GlideUrl(url, builder.build()) | ||||||
|  |     return this.load(glideUrl) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun getBitmapInputStream(bitmap:Bitmap,compressFormat: Bitmap.CompressFormat): InputStream { | ||||||
|  |     val byteArrayOutputStream = ByteArrayOutputStream() | ||||||
|  |     bitmap.compress(compressFormat, 80, byteArrayOutputStream) | ||||||
|  |     val bitmapData: ByteArray = byteArrayOutputStream.toByteArray() | ||||||
|  |     return ByteArrayInputStream(bitmapData) | ||||||
|  | } | ||||||
| @@ -23,9 +23,9 @@ class SelfSignedGlideModule : GlideModule { | |||||||
|                 val client = getUnsafeHttpClient().build() |                 val client = getUnsafeHttpClient().build() | ||||||
|  |  | ||||||
|                 registry?.append( |                 registry?.append( | ||||||
|                         GlideUrl::class.java, |                     GlideUrl::class.java, | ||||||
|                         InputStream::class.java, |                     InputStream::class.java, | ||||||
|                         com.bumptech.glide.integration.okhttp3.OkHttpUrlLoader.Factory(client) |                     com.bumptech.glide.integration.okhttp3.OkHttpUrlLoader.Factory(client) | ||||||
|                 ) |                 ) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -0,0 +1,45 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.utils.network | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import android.graphics.Color | ||||||
|  | import android.net.ConnectivityManager | ||||||
|  | import android.net.NetworkInfo | ||||||
|  | import android.view.View | ||||||
|  | import android.widget.TextView | ||||||
|  | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import com.google.android.material.snackbar.Snackbar | ||||||
|  |  | ||||||
|  | var snackBarShown = false | ||||||
|  | var view: View? = null | ||||||
|  | lateinit var s: Snackbar | ||||||
|  |  | ||||||
|  | fun Context.isNetworkAccessible(v: View?, overrideOffline: Boolean = false): Boolean { | ||||||
|  |     val cm = this.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager | ||||||
|  |     val activeNetwork: NetworkInfo? = cm.activeNetworkInfo | ||||||
|  |     val networkIsAccessible = activeNetwork != null && activeNetwork.isConnectedOrConnecting | ||||||
|  |  | ||||||
|  |     if (v != null && (!networkIsAccessible || overrideOffline) && (!snackBarShown || v != view)) { | ||||||
|  |         view = v | ||||||
|  |         s = Snackbar | ||||||
|  |             .make( | ||||||
|  |                 v, | ||||||
|  |                 R.string.no_network_connectivity, | ||||||
|  |                 Snackbar.LENGTH_INDEFINITE | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |         s.setAction(android.R.string.ok) { | ||||||
|  |             snackBarShown = false | ||||||
|  |             s.dismiss() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val view = s.view | ||||||
|  |         val tv: TextView = view.findViewById(com.google.android.material.R.id.snackbar_text) | ||||||
|  |         tv.setTextColor(Color.WHITE) | ||||||
|  |         s.show() | ||||||
|  |         snackBarShown = true | ||||||
|  |     } | ||||||
|  |     if (snackBarShown && networkIsAccessible && !overrideOffline) { | ||||||
|  |         s.dismiss() | ||||||
|  |     } | ||||||
|  |     return if(overrideOffline) overrideOffline else networkIsAccessible | ||||||
|  | } | ||||||
| @@ -0,0 +1,73 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.utils.persistence | ||||||
|  |  | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossTagType | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.Source | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.Tag | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ItemEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.SourceEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.TagEntity | ||||||
|  |  | ||||||
|  | fun TagEntity.toView(): Tag = | ||||||
|  |         Tag( | ||||||
|  |             this.tag, | ||||||
|  |             this.color, | ||||||
|  |             this.unread | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  | fun SourceEntity.toView(): Source = | ||||||
|  |         Source( | ||||||
|  |             this.id, | ||||||
|  |             this.title, | ||||||
|  |             SelfossTagType(this.tags), | ||||||
|  |             this.spout, | ||||||
|  |             this.error, | ||||||
|  |             this.icon | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  | fun Source.toEntity(): SourceEntity = | ||||||
|  |         SourceEntity( | ||||||
|  |             this.id, | ||||||
|  |             this.getTitleDecoded(), | ||||||
|  |             this.tags.tags, | ||||||
|  |             this.spout, | ||||||
|  |             this.error, | ||||||
|  |             this.icon.orEmpty() | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  | fun Tag.toEntity(): TagEntity = | ||||||
|  |         TagEntity( | ||||||
|  |             this.tag, | ||||||
|  |             this.color, | ||||||
|  |             this.unread | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  | fun ItemEntity.toView(): Item = | ||||||
|  |         Item( | ||||||
|  |             this.id, | ||||||
|  |             this.datetime, | ||||||
|  |             this.title, | ||||||
|  |             this.content, | ||||||
|  |             this.unread, | ||||||
|  |             this.starred, | ||||||
|  |             this.thumbnail, | ||||||
|  |             this.icon, | ||||||
|  |             this.link, | ||||||
|  |             this.sourcetitle, | ||||||
|  |             SelfossTagType(this.tags) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  | fun Item.toEntity(): ItemEntity = | ||||||
|  |     ItemEntity( | ||||||
|  |         this.id, | ||||||
|  |         this.datetime, | ||||||
|  |         this.getTitleDecoded(), | ||||||
|  |         this.content, | ||||||
|  |         this.unread, | ||||||
|  |         this.starred, | ||||||
|  |         this.thumbnail, | ||||||
|  |         this.icon, | ||||||
|  |         this.link, | ||||||
|  |         this.getSourceTitle(), | ||||||
|  |         this.tags.tags | ||||||
|  |     ) | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |         android:width="24dp" | ||||||
|  |         android:height="24dp" | ||||||
|  |         android:viewportWidth="24.0" | ||||||
|  |         android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FFFFFFFF" | ||||||
|  |         android:pathData="M3,21h18v-2L3,19v2zM3,17h18v-2L3,15v2zM3,13h18v-2L3,11v2zM3,9h18L21,7L3,7v2zM3,3v2h18L21,3L3,3z"/> | ||||||
|  | </vector> | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |         android:width="24dp" | ||||||
|  |         android:height="24dp" | ||||||
|  |         android:viewportWidth="24.0" | ||||||
|  |         android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FFFFFFFF" | ||||||
|  |         android:pathData="M15,15L3,15v2h12v-2zM15,7L3,7v2h12L15,7zM3,13h18v-2L3,11v2zM3,21h18v-2L3,19v2zM3,3v2h18L21,3L3,3z"/> | ||||||
|  | </vector> | ||||||
| Before Width: | Height: | Size: 680 B | 
| Before Width: | Height: | Size: 134 B | 
| Before Width: | Height: | Size: 239 B | 
| Before Width: | Height: | Size: 271 B | 
| Before Width: | Height: | Size: 216 B | 
| Before Width: | Height: | Size: 221 B | 
| Before Width: | Height: | Size: 458 B | 
| Before Width: | Height: | Size: 275 B | 
| Before Width: | Height: | Size: 361 B | 
| Before Width: | Height: | Size: 301 B | 
| Before Width: | Height: | Size: 355 B | 
| Before Width: | Height: | Size: 551 B | 
| Before Width: | Height: | Size: 204 B | 
| Before Width: | Height: | Size: 187 B | 
| Before Width: | Height: | Size: 422 B | 
| Before Width: | Height: | Size: 473 B | 
| Before Width: | Height: | Size: 498 B | 
| Before Width: | Height: | Size: 453 B | 
| Before Width: | Height: | Size: 398 B | 
| Before Width: | Height: | Size: 397 B | 
| Before Width: | Height: | Size: 442 B | 
| Before Width: | Height: | Size: 116 B | 
| Before Width: | Height: | Size: 174 B | 
| Before Width: | Height: | Size: 212 B | 
| Before Width: | Height: | Size: 136 B |