Compare commits
	
		
			16 Commits
		
	
	
		
			794500355a
			...
			v125020581
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| a90ccec707 | |||
| 2564b19726 | |||
| 61c7bb20cc | |||
| 6a0f5baf0a | |||
| 39f9505c00 | |||
| 6a6d447456 | |||
| 0bb4fe6aed | |||
| 7df4c3368c | |||
| c69635b5ae | |||
| 3a829df70e | |||
| 7a0202689f | |||
| b20f6888f5 | |||
| 6b96eb358d | |||
| dfc1bf9fa3 | |||
| b173664ff0 | |||
| bc20a421ae | 
| @@ -26,9 +26,10 @@ jobs: | ||||
|       - uses: KengoTODA/actions-setup-docker-compose@v1 | ||||
|         with: | ||||
|           version: "2.23.3" | ||||
|       - name: run selfoss | ||||
|         run: | | ||||
|           docker compose -f .gitea/workflows/assets/docker-compose.yml up -d | ||||
|       #      TESTS ARE RUN LOCALLY | ||||
|       #      - name: run selfoss | ||||
|       #        run: | | ||||
|       #          docker compose -f .gitea/workflows/assets/docker-compose.yml up -d | ||||
|       - name: coverage | ||||
|         run: | | ||||
|           ./gradlew :koverHtmlReport | ||||
| @@ -39,7 +40,8 @@ jobs: | ||||
|           retention-days: 1 | ||||
|           overwrite: true | ||||
|           include-hidden-files: true | ||||
|       - name: Clean | ||||
|         if: always() | ||||
|         run: | | ||||
|           docker compose -f .gitea/workflows/assets/docker-compose.yml stop | ||||
| #      TESTS ARE RUN LOCALLY | ||||
| #      - name: Clean | ||||
| #        if: always() | ||||
| #        run: | | ||||
| #          docker compose -f .gitea/workflows/assets/docker-compose.yml stop | ||||
|   | ||||
| @@ -16,6 +16,7 @@ jobs: | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|           ref: master | ||||
|       - name: Config git | ||||
|         run: | | ||||
|           git config --global user.email aminecmi+giteadrone@pm.me | ||||
| @@ -50,7 +51,7 @@ jobs: | ||||
|           followtags: true | ||||
|           ssh_key: ${{ secrets.PRIVATE_KEY }} | ||||
|           tags: true | ||||
|           branch: release | ||||
|           branch: master | ||||
|       - name: copy file via ssh password | ||||
|         uses: appleboy/scp-action@v0.1.7 | ||||
|         with: | ||||
|   | ||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -324,3 +324,5 @@ crowdin.properties | ||||
|  | ||||
| .kotlin/ | ||||
| build-cache/ | ||||
|  | ||||
| act | ||||
|   | ||||
							
								
								
									
										31
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										31
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,3 +1,34 @@ | ||||
| **v125020471 | ||||
|  | ||||
| - chore: no more docker-compose. | ||||
| - bump: gradle plugin. | ||||
| - Merge pull request 'fix: check index exists.' (#183) from fix-index into master | ||||
| - fix: check index exists. | ||||
| - Changelog for v125020411 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125020411 | ||||
|  | ||||
| - Merge pull request 'bump' (#182) from bump into master | ||||
| - chore: non transiant R classes. | ||||
| - Merge pull request 'fix: One more missing context.' (#181) from fix-one-more-context into master | ||||
| - bump | ||||
| - fix: One more missing context. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125010241 | ||||
|  | ||||
| - Merge pull request 'fix: Link not opening.' (#178) from fix-open-link into master | ||||
| - refactor: context fragments issues. | ||||
| - logs: Context issues. | ||||
| - fix: Handle empty url issue, again. | ||||
| - fix: Link not opening. | ||||
| - Changelog for v125010201 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125010201 | ||||
|  | ||||
| - fix: Handle empty url issue. | ||||
|   | ||||
| @@ -56,7 +56,7 @@ class HomeActivityTest { | ||||
|     fun testMenuActions() { | ||||
|         onView(withId(R.id.action_search)).perform(click()) | ||||
|         onView( | ||||
|             withId(R.id.search_src_text), | ||||
|             withId(com.google.android.material.R.id.search_src_text), | ||||
|         ).check(matches(isFocused())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|  | ||||
|   | ||||
| @@ -161,6 +161,7 @@ class ReaderActivity : | ||||
|                     override fun onPageSelected(position: Int) { | ||||
|                         super.onPageSelected(position) | ||||
|  | ||||
|                         if (!allItems.isNullOrEmpty() && allItems.size >= position) { | ||||
|                             if (allItems[position].starred) { | ||||
|                                 canRemoveFromFavorite() | ||||
|                             } else { | ||||
| @@ -168,6 +169,7 @@ class ReaderActivity : | ||||
|                             } | ||||
|                             readItem(allItems[position]) | ||||
|                         } | ||||
|                     } | ||||
|                 }, | ||||
|             ) | ||||
|         } | ||||
|   | ||||
| @@ -75,7 +75,7 @@ class ArticleFragment : | ||||
|     private var colorSurface: Int = 0 | ||||
|     private var fontSize: Int = DEFAULT_FONT_SIZE | ||||
|     private lateinit var item: SelfossModel.Item | ||||
|     private lateinit var url: String | ||||
|     private var url: String? = null | ||||
|     private lateinit var contentText: String | ||||
|     private lateinit var contentSource: String | ||||
|     private lateinit var contentImage: String | ||||
| @@ -118,8 +118,8 @@ class ArticleFragment : | ||||
|                 e.sendSilentlyWithAcra() | ||||
|             } | ||||
|  | ||||
|             colorOnSurface = getColorFromAttr(R.attr.colorOnSurface) | ||||
|             colorSurface = getColorFromAttr(R.attr.colorSurface) | ||||
|             colorOnSurface = getColorFromAttr(com.google.android.material.R.attr.colorOnSurface) | ||||
|             colorSurface = getColorFromAttr(com.google.android.material.R.attr.colorSurface) | ||||
|  | ||||
|             contentText = item.content | ||||
|             contentTitle = item.title.getHtmlDecoded() | ||||
| @@ -168,8 +168,8 @@ class ArticleFragment : | ||||
|  | ||||
|     private fun handleContent() { | ||||
|         if (contentText.isEmptyOrNullOrNullString()) { | ||||
|             if (repository.isNetworkAvailable()) { | ||||
|                 getContentFromMercury() | ||||
|             if (repository.isNetworkAvailable() && url.isUrlValid()) { | ||||
|                 getContentFromMercury(url!!) | ||||
|             } | ||||
|         } else { | ||||
|             binding.titleView.text = contentTitle | ||||
| @@ -193,14 +193,13 @@ class ArticleFragment : | ||||
|         fab.mainFabClosedIconColor = colorOnSurface | ||||
|         fab.mainFabOpenedIconColor = colorOnSurface | ||||
|  | ||||
|         maybeIfContext { context -> handleFloatingToolbarActionItems(context) } | ||||
|         maybeIfContext { handleFloatingToolbarActionItems(it) } | ||||
|  | ||||
|         fab.setOnActionSelectedListener { actionItem -> | ||||
|             when (actionItem.id) { | ||||
|                 R.id.share_action -> requireActivity().shareLink(url, contentTitle) | ||||
|                 R.id.open_action -> requireActivity().openItemUrlInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|                 R.id.unread_action -> | ||||
|                     try { | ||||
|                     if (this@ArticleFragment.item.unread) { | ||||
|                         CoroutineScope(Dispatchers.IO).launch { | ||||
|                             repository.markAsRead(this@ArticleFragment.item) | ||||
| @@ -219,15 +218,14 @@ class ArticleFragment : | ||||
|                             repository.unmarkAsRead(this@ArticleFragment.item) | ||||
|                         } | ||||
|                         this@ArticleFragment.item.unread = true | ||||
|                         maybeIfContext { | ||||
|                             Toast | ||||
|                                 .makeText( | ||||
|                                     context, | ||||
|                                     it, | ||||
|                                     R.string.marked_as_unread, | ||||
|                                     Toast.LENGTH_LONG, | ||||
|                                 ).show() | ||||
|                         } | ||||
|                     } catch (e: IllegalStateException) { | ||||
|                         e.sendSilentlyWithAcraWithName("Toolbar context required is null") | ||||
|                     } | ||||
|  | ||||
|                 else -> Unit | ||||
| @@ -273,7 +271,7 @@ class ArticleFragment : | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:SwallowedException") | ||||
|     private fun getContentFromMercury() { | ||||
|     private fun getContentFromMercury(url: String) { | ||||
|         binding.progressBar.visibility = View.VISIBLE | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -313,8 +311,8 @@ class ArticleFragment : | ||||
|  | ||||
|     private fun handleLeadImage(leadImageUrl: String?) { | ||||
|         if (!leadImageUrl.isNullOrEmpty()) { | ||||
|             binding.imageView.visibility = View.VISIBLE | ||||
|             maybeIfContext { | ||||
|                 binding.imageView.visibility = View.VISIBLE | ||||
|                 it.bitmapFitCenter(leadImageUrl, binding.imageView, appSettingsService) | ||||
|             } | ||||
|         } else { | ||||
| @@ -397,13 +395,12 @@ class ArticleFragment : | ||||
|                 WHITE_COLOR_HEX and (if (colorOnSurface != DATA_NULL_UNDEFINED) colorOnSurface else 0), | ||||
|             ) | ||||
|  | ||||
|         try { | ||||
|         binding.webcontent.settings.useWideViewPort = true | ||||
|         binding.webcontent.settings.loadWithOverviewMode = true | ||||
|         binding.webcontent.settings.javaScriptEnabled = false | ||||
|  | ||||
|         handleImageLoading() | ||||
|  | ||||
|         try { | ||||
|             val gestureDetector = | ||||
|                 GestureDetector( | ||||
|                     activity, | ||||
| @@ -417,33 +414,34 @@ class ArticleFragment : | ||||
|                     event, | ||||
|                 ) | ||||
|             } | ||||
|  | ||||
|             binding.webcontent.settings.layoutAlgorithm = | ||||
|                 WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING | ||||
|         } catch (e: IllegalStateException) { | ||||
|             e.sendSilentlyWithAcraWithName("Context is null but wasn't, and that's causing issues with webview config") | ||||
|             e.sendSilentlyWithAcraWithName("Gesture detector issue ?") | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|         binding.webcontent.settings.layoutAlgorithm = | ||||
|             WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING | ||||
|  | ||||
|         var baseUrl: String? = null | ||||
|         try { | ||||
|                 val itemUrl = URL(url) | ||||
|             val itemUrl = URL(url.orEmpty()) | ||||
|             baseUrl = itemUrl.protocol + "://" + itemUrl.host | ||||
|         } catch (e: MalformedURLException) { | ||||
|                 e.sendSilentlyWithAcraWithName("htmlToWebview > $url") | ||||
|             e.sendSilentlyWithAcraWithName("htmlToWebview > ${url.orEmpty()}") | ||||
|         } | ||||
|  | ||||
|             val fontName = | ||||
|         val fontName: String = | ||||
|             maybeIfContext { | ||||
|                 when (font) { | ||||
|                     getString(R.string.open_sans_font_id) -> "Open Sans" | ||||
|                     getString(R.string.roboto_font_id) -> "Roboto" | ||||
|                     getString(R.string.source_code_pro_font_id) -> "Source Code Pro" | ||||
|                     it.getString(R.string.open_sans_font_id) -> "Open Sans" | ||||
|                     it.getString(R.string.roboto_font_id) -> "Roboto" | ||||
|                     it.getString(R.string.source_code_pro_font_id) -> "Source Code Pro" | ||||
|                     else -> "" | ||||
|                 } | ||||
|             }?.toString().orEmpty() | ||||
|  | ||||
|         val fontLinkAndStyle = | ||||
|                 if (font.isNotEmpty()) { | ||||
|             if (fontName.isNotEmpty()) { | ||||
|                 """<link href="https://fonts.googleapis.com/css?family=${ | ||||
|                     fontName.replace( | ||||
|                         " ", | ||||
| @@ -459,7 +457,7 @@ class ArticleFragment : | ||||
|             } else { | ||||
|                 "" | ||||
|             } | ||||
|  | ||||
|         try { | ||||
|             binding.webcontent.loadDataWithBaseURL( | ||||
|                 baseUrl, | ||||
|                 """<html> | ||||
|   | ||||
| @@ -83,7 +83,15 @@ class FilterSheetFragment : | ||||
|         val sourceGroup = binding.sourcesGroup | ||||
|  | ||||
|         repository.getSourcesDetailsOrStats().forEachIndexed { _, source -> | ||||
|             val c = Chip(context) | ||||
|             val c: Chip? = | ||||
|                 maybeIfContext { | ||||
|                     Chip(it) | ||||
|                 } as Chip? | ||||
|  | ||||
|             if (c == null) { | ||||
|                 return | ||||
|             } | ||||
|  | ||||
|             c.ellipsize = TextUtils.TruncateAt.END | ||||
|  | ||||
|             maybeIfContext { | ||||
| @@ -145,7 +153,11 @@ class FilterSheetFragment : | ||||
|         val tags = repository.getTags() | ||||
|  | ||||
|         tags.forEachIndexed { _, tag -> | ||||
|             val c = Chip(context) | ||||
|             val c: Chip? = maybeIfContext { Chip(it) } as Chip? | ||||
|             if (c == null) { | ||||
|                 return | ||||
|             } | ||||
|  | ||||
|             c.ellipsize = TextUtils.TruncateAt.END | ||||
|             c.text = tag.tag | ||||
|  | ||||
|   | ||||
| @@ -11,13 +11,14 @@ import bou.amine.apps.readerforselfossv2.android.utils.acra.sendSilentlyWithAcra | ||||
| import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | ||||
|  | ||||
| fun Context.shareLink( | ||||
|     itemUrl: String, | ||||
|     itemUrl: String?, | ||||
|     itemTitle: String, | ||||
| ) { | ||||
|     if (itemUrl.isUrlValid()) { | ||||
|         val sendIntent = Intent() | ||||
|         sendIntent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|         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" | ||||
|         startActivity( | ||||
| @@ -27,6 +28,7 @@ fun Context.shareLink( | ||||
|                     getString(R.string.share), | ||||
|                 ).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK), | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ColorInt | ||||
|   | ||||
| @@ -15,12 +15,12 @@ import android.widget.Toast | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.ReaderActivity | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | ||||
| import bou.amine.apps.readerforselfossv2.utils.isEmptyOrNullOrNullString | ||||
| import okhttp3.HttpUrl.Companion.toHttpUrlOrNull | ||||
|  | ||||
| fun Context.openItemUrl( | ||||
|     currentItem: Int, | ||||
|     linkDecoded: String, | ||||
|     linkDecoded: String?, | ||||
|     articleViewer: Boolean, | ||||
|     app: Activity, | ||||
| ) { | ||||
| @@ -37,12 +37,13 @@ fun Context.openItemUrl( | ||||
|             intent.putExtra("currentItem", currentItem) | ||||
|             app.startActivity(intent) | ||||
|         } else { | ||||
|             this.openUrlInBrowserAsNewTask(linkDecoded) | ||||
|             this.openUrlInBrowserAsNewTask(linkDecoded!!) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun String.isUrlValid(): Boolean = this.toHttpUrlOrNull() != null && Patterns.WEB_URL.matcher(this).matches() | ||||
| fun String?.isUrlValid(): Boolean = | ||||
|     !this.isEmptyOrNullOrNullString() && this!!.toHttpUrlOrNull() != null && Patterns.WEB_URL.matcher(this).matches() | ||||
|  | ||||
| fun String.isBaseUrlInvalid(): Boolean { | ||||
|     val baseUrl = this.toHttpUrlOrNull() | ||||
| @@ -56,14 +57,16 @@ fun String.isBaseUrlInvalid(): Boolean { | ||||
| } | ||||
|  | ||||
| fun Context.openItemUrlInBrowserAsNewTask(i: SelfossModel.Item) { | ||||
|     this.openUrlInBrowserAsNewTask(i.getLinkDecoded().toStringUriWithHttp()) | ||||
|     this.openUrlInBrowserAsNewTask(i.getLinkDecoded()) | ||||
| } | ||||
|  | ||||
| fun Context.openUrlInBrowserAsNewTask(url: String) { | ||||
| fun Context.openUrlInBrowserAsNewTask(url: String?) { | ||||
|     if (url.isUrlValid()) { | ||||
|         val intent = Intent(Intent.ACTION_VIEW) | ||||
|         intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|         intent.data = Uri.parse(url) | ||||
|         this.mayBeStartActivity(intent) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun Context.openUrlInBrowser(url: String) { | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| plugins { | ||||
|     //trick: for the same plugin versions in all sub-modules | ||||
|     id("com.android.application").version("8.7.3").apply(false) | ||||
|     id("com.android.library").version("8.7.3").apply(false) | ||||
|     // trick: for the same plugin versions in all sub-modules | ||||
|     id("com.android.application").version("8.8.1").apply(false) | ||||
|     id("com.android.library").version("8.8.1").apply(false) | ||||
|     id("org.jetbrains.kotlin.android").version("2.1.0").apply(false) | ||||
|     kotlin("multiplatform").version("2.1.0").apply(false) | ||||
|     id("com.mikepenz.aboutlibraries.plugin").version("10.5.1").apply(false) | ||||
| @@ -16,7 +16,6 @@ allprojects { | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| tasks.register("clean", Delete::class) { | ||||
|     delete(layout.buildDirectory) | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,8 @@ | ||||
| **v125010241** | ||||
|  | ||||
| - Merge pull request 'fix: Link not opening.' (#178) from fix-open-link into master | ||||
| - refactor: context fragments issues. | ||||
| - logs: Context issues. | ||||
| - fix: Handle empty url issue, again. | ||||
| - fix: Link not opening. | ||||
| - Changelog for v125010201 | ||||
| @@ -0,0 +1,7 @@ | ||||
| **v125020411** | ||||
|  | ||||
| - Merge pull request 'bump' (#182) from bump into master | ||||
| - chore: non transiant R classes. | ||||
| - Merge pull request 'fix: One more missing context.' (#181) from fix-one-more-context into master | ||||
| - bump | ||||
| - fix: One more missing context. | ||||
| @@ -0,0 +1,7 @@ | ||||
| **v125020471** | ||||
|  | ||||
| - chore: no more docker-compose. | ||||
| - bump: gradle plugin. | ||||
| - Merge pull request 'fix: check index exists.' (#183) from fix-index into master | ||||
| - fix: check index exists. | ||||
| - Changelog for v125020411 | ||||
| @@ -19,7 +19,7 @@ kotlin.code.style=official | ||||
| android.useAndroidX=true | ||||
| #android.nonTransitiveRClass=true | ||||
| android.enableJetifier=false | ||||
| android.nonTransitiveRClass=false | ||||
| android.nonTransitiveRClass=true | ||||
| #MPP | ||||
| kotlin.mpp.enableCInteropCommonization=true | ||||
| org.gradle.parallel=true | ||||
|   | ||||
							
								
								
									
										4
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| #Mon Nov 25 22:48:24 CET 2024 | ||||
| #Sun Feb 09 14:44:52 CET 2025 | ||||
| distributionBase=GRADLE_USER_HOME | ||||
| distributionPath=wrapper/dists | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.10.2-bin.zip | ||||
| zipStoreBase=GRADLE_USER_HOME | ||||
| zipStorePath=wrapper/dists | ||||
|   | ||||
| @@ -127,8 +127,8 @@ class SelfossModel { | ||||
|         val tags: List<String>, | ||||
|         val author: String? = null, | ||||
|     ) { | ||||
|         fun getLinkDecoded(): String { | ||||
|             var stringUrl: String | ||||
|         fun getLinkDecoded(): String? { | ||||
|             var stringUrl: String? | ||||
|             stringUrl = | ||||
|                 if (link.contains("//news.google.com/news/") && link.contains("&url=")) { | ||||
|                     link.substringAfter("&url=") | ||||
| @@ -146,11 +146,7 @@ class SelfossModel { | ||||
|                 stringUrl = "http:$stringUrl" | ||||
|             } | ||||
|  | ||||
|             if (stringUrl.isEmptyOrNullOrNullString()) { | ||||
|                 throw ModelException("Link $link was translated to $stringUrl, but was empty. Handle this.") | ||||
|             } | ||||
|  | ||||
|             return stringUrl | ||||
|             return if (stringUrl.isEmptyOrNullOrNullString()) null else stringUrl | ||||
|         } | ||||
|  | ||||
|         fun sourceAuthorAndDate(): String { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user