Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improve OSS licenses screen #1018

Open
wants to merge 48 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 30 commits
Commits
Show all changes
48 commits
Select commit Hold shift + click to select a range
47ec25d
Add feature/licenses module
fumiya-kume Aug 29, 2023
830e3b0
Merge branch 'main' into kuu/improve-oss-licenses-screen
fumiya-kume Sep 3, 2023
aa0e419
Add empty oss license screen
fumiya-kume Sep 3, 2023
f537b83
Add empty oss license ViewModel
fumiya-kume Sep 3, 2023
fde39dc
Combine screen and ViewModel
fumiya-kume Sep 3, 2023
15c6b3b
Setup basis of license screen
fumiya-kume Sep 3, 2023
810acb4
Remove feature/lincenses
fumiya-kume Sep 3, 2023
2a326c1
Remove feature/licenses relational file
fumiya-kume Sep 3, 2023
b717b7d
Move classes
fumiya-kume Sep 3, 2023
6ddec12
Add logic to load the licenses data
fumiya-kume Sep 3, 2023
e4e55ff
Add Scaffold for OSS licenses screen
fumiya-kume Sep 3, 2023
71db515
Add UI implementation
fumiya-kume Sep 3, 2023
d2794b9
Add license detail implementation
fumiya-kume Sep 3, 2023
54e8302
Fix code format
fumiya-kume Sep 3, 2023
8cea72a
Add OssLicense screen
fumiya-kume Sep 12, 2023
0294a88
Add OssDetailLicense screen
fumiya-kume Sep 12, 2023
bde016a
Merge branch 'main' into kuu/improve-oss-licenses-screen
fumiya-kume Sep 12, 2023
dc641f3
Move License to core/model module
fumiya-kume Sep 12, 2023
c18fb1d
Oss screen move to feature / about
fumiya-kume Sep 12, 2023
1ae5174
Fix code format
fumiya-kume Sep 12, 2023
753f9a7
Update DI config with using the optional binding
fumiya-kume Sep 14, 2023
c605ed3
Delete unnecessary Dagger module
fumiya-kume Sep 14, 2023
35316a8
Rename unintentional method name for the navigation
fumiya-kume Sep 14, 2023
c7a3413
Remove un-necessary method
fumiya-kume Sep 14, 2023
be4982d
Cleanup fake class
fumiya-kume Sep 14, 2023
a7c4a4c
Cleanup duplicate logic for id of license
fumiya-kume Sep 14, 2023
6afe1e6
Using multi language support feature for the screen title
fumiya-kume Sep 14, 2023
1d0da32
Update library grouping logic
fumiya-kume Sep 14, 2023
6207404
Fix code format
fumiya-kume Sep 14, 2023
3793f2e
Merge branch 'main' into kuu/improve-oss-licenses-screen
fumiya-kume Sep 15, 2023
2c4d86f
Rename default implementation for Repository
fumiya-kume Sep 16, 2023
b172de9
Update DI config for OssLicenseDataSource
fumiya-kume Sep 16, 2023
5e5c12e
Refactoring DataFlow
fumiya-kume Sep 16, 2023
65ffeeb
Fix logic
fumiya-kume Sep 16, 2023
6e246d2
Remove duplicated libraries
fumiya-kume Sep 16, 2023
723baf9
Refactoring
fumiya-kume Sep 16, 2023
aaad120
Remove unnecessary code
fumiya-kume Sep 16, 2023
0aae559
Merge branch 'main' into kuu/improve-oss-licenses-screen
fumiya-kume Sep 16, 2023
4ba7ae4
Merge branch 'main' into kuu/improve-oss-licenses-screen
fumiya-kume Sep 16, 2023
c0c4f4d
Fix visibility
fumiya-kume Sep 16, 2023
1819113
Fix visibility
fumiya-kume Sep 16, 2023
2e08ce5
Merge branch 'main' into kuu/improve-oss-licenses-screen
fumiya-kume Sep 17, 2023
a2e5eb3
Merge branch 'main' into kuu/improve-oss-licenses-screen
fumiya-kume Sep 23, 2023
4183d96
Add screenshot testing for oss license screen
fumiya-kume Sep 23, 2023
d94c87e
Add screenshot testing for oss license detail screen
fumiya-kume Sep 23, 2023
a26cc92
revert comment out
fumiya-kume Sep 23, 2023
1255554
Try to update the build variant when CI
fumiya-kume Sep 23, 2023
ecd9b9f
Revert variant change
fumiya-kume Sep 23, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,11 +1,16 @@
package io.github.droidkaigi.confsched2023

import android.content.Context
import dagger.Module
import dagger.Provides
import dagger.hilt.InstallIn
import dagger.hilt.android.qualifiers.ApplicationContext
import dagger.hilt.components.SingletonComponent
import io.github.droidkaigi.confsched2023.data.di.AppAndroidBuildConfig
import io.github.droidkaigi.confsched2023.data.di.AppAndroidOssLicenseConfig
import io.github.droidkaigi.confsched2023.license.OssLicenseRepositoryImpl
import io.github.droidkaigi.confsched2023.model.BuildConfigProvider
import io.github.droidkaigi.confsched2023.model.OssLicenseRepository
import javax.inject.Singleton

@InstallIn(SingletonComponent::class)
Expand All @@ -15,6 +20,13 @@ class AppModule {
@Singleton
@AppAndroidBuildConfig
fun provideBuildConfigProvider(): BuildConfigProvider = AppBuildConfigProvider()

@Provides
@Singleton
@AppAndroidOssLicenseConfig
fun provideOssLicenseRepositoryProvider(
@ApplicationContext context: Context,
): OssLicenseRepository = OssLicenseRepositoryImpl(context)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We are using Default〜 prefix for the concrete class 🙏

}

class AppBuildConfigProvider(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,11 @@ import co.touchlab.kermit.Logger
import com.google.android.gms.oss.licenses.OssLicensesMenuActivity
import io.github.droidkaigi.confsched2023.about.aboutScreenRoute
import io.github.droidkaigi.confsched2023.about.navigateAboutScreen
import io.github.droidkaigi.confsched2023.about.navigateOssLicenseDetailScreen
import io.github.droidkaigi.confsched2023.about.navigateOssLicenseScreen
import io.github.droidkaigi.confsched2023.about.nestedAboutScreen
import io.github.droidkaigi.confsched2023.about.ossLicenseDetailScreen
import io.github.droidkaigi.confsched2023.about.ossLicenseScreen
import io.github.droidkaigi.confsched2023.achievements.achievementsScreenRoute
import io.github.droidkaigi.confsched2023.achievements.navigateAchievementsScreen
import io.github.droidkaigi.confsched2023.achievements.nestedAchievementsScreen
Expand Down Expand Up @@ -125,6 +129,19 @@ private fun KaigiNavHost(
onBackClick = navController::popBackStack,
onStaffClick = externalNavController::navigate,
)
ossLicenseScreen(
onLicenseClick = { license ->
navController.navigateOssLicenseDetailScreen(license)
},
onUpClick = {
navController.navigateUp()
},
)
ossLicenseDetailScreen(
onUpClick = {
navController.navigateUp()
},
)
// For KMP, we are not using navigation abstraction for contributors screen
composable(contributorsScreenRoute) {
val lifecycleOwner = LocalLifecycleOwner.current
Expand Down Expand Up @@ -171,7 +188,7 @@ private fun NavGraphBuilder.mainScreen(
Sponsors -> navController.navigateSponsorsScreen()
CodeOfConduct -> { externalNavController.navigate(url = "$portalBaseUrl/about/code-of-conduct") }
Contributors -> navController.navigate(contributorsScreenRoute)
License -> externalNavController.navigateToLicenseScreen()
License -> navController.navigateOssLicenseScreen()
Medium -> externalNavController.navigate(url = "https://medium.com/droidkaigi")
PrivacyPolicy -> {
externalNavController.navigate(url = "$portalBaseUrl/about/privacy")
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
package io.github.droidkaigi.confsched2023.license

import android.content.Context
import io.github.droidkaigi.confsched2023.R.raw
import io.github.droidkaigi.confsched2023.model.License
import io.github.droidkaigi.confsched2023.model.OssLicenseRepository
import kotlinx.collections.immutable.PersistentList
import kotlinx.collections.immutable.persistentListOf
import kotlinx.collections.immutable.toPersistentList
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.MutableStateFlow
import okio.BufferedSource
import okio.buffer
import okio.source
import javax.inject.Inject

class OssLicenseRepositoryImpl @Inject constructor(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

How about moving the logic to DefaultOssLicenseDataSource to simplify the app's implementation? I think this is similar to an API, since both handle data parsing. Therefore, I suggest transferring the code to DefaultOssLicenseDataSource.

class DefaultOssLicenseDataSource @Inject constructor(
    private val context: Context,
) : OssLicenseDataSource {

    override suspend fun licenseFlow(): OssLicense {
        return withContext(context = Dispatchers.IO) {
            val details = readLicensesFile().toRowList()
            val metadata = readLicensesMetaFile().toRowList().parseToLibraryItem(
                details = details,
            )
            val groupList = metadata.distinctBy { it.name }.groupByCategory()
                .map {
                    OssLicenseGroup(
                        title = it.key,
                        licenses = it.value,
                    )
                }.toPersistentList()
            OssLicense(groupList)
        }
    }

    private fun List<License>.groupByCategory(): Map<String, List<License>> {
        val categoryList = listOf(
            "Android Support",
            "Android Datastore",
            "Android ",
            "Compose UI",
            "Compose Material3",
            "Compose ",
            "AndroidX lifecycle",
            "AndroidX ",
            "Kotlin",
            "Dagger",
            "Firebase",
            "Ktorfit",
            "okhttp",
            "ktor",
        )
        return groupBy { license ->
            categoryList.firstOrNull {
                license.name.startsWith(
                    prefix = it,
                    ignoreCase = true,
                )
            } ?: "etc"
        }
    }

    private fun readLicensesMetaFile(): BufferedSource {
        return context.resources.openRawResource(raw.third_party_license_metadata)
            .source()
            .buffer()
    }

    private fun readLicensesFile(): BufferedSource {
        return context.resources.openRawResource(raw.third_party_licenses)
            .source()
            .buffer()
    }

    private fun List<String>.parseToLibraryItem(details: List<String>): List<License> {
        return mapIndexed { index, value ->
            val (position, name) = value.split(' ', limit = 2)
            val (offset, length) = position.split(':').map { it.toInt() }
            val id = name.replace(' ', '-')
            License(
                name = name,
                id = id,
                offset = offset,
                length = length,
                detail = details[index],
            )
        }
    }

    private fun BufferedSource.toRowList(): List<String> {
        val list: MutableList<String> = mutableListOf()
        while (true) {
            val line = readUtf8Line() ?: break
            list.add(line)
        }
        return list
    }
}

private val context: Context,
) : OssLicenseRepository {

private val licenseMetaStateFlow =
MutableStateFlow<PersistentList<License>>(persistentListOf())

private val licenseDetailStateFlow = MutableStateFlow<List<String>>(emptyList())

override fun licenseMetaData(): Flow<PersistentList<License>> {
licenseMetaStateFlow.value = readLicensesMetaFile()
.toRowList()
.parseToLibraryItem()
.toPersistentList()
return licenseMetaStateFlow
}

override fun licenseDetailData(): Flow<List<String>> {
licenseDetailStateFlow.value = readLicensesFile()
.toRowList()
return licenseDetailStateFlow
}

private fun readLicensesMetaFile(): BufferedSource {
return context.resources.openRawResource(raw.third_party_license_metadata)
.source()
.buffer()
}

private fun readLicensesFile(): BufferedSource {
return context.resources.openRawResource(raw.third_party_licenses)
.source()
.buffer()
}

private fun List<String>.parseToLibraryItem(): List<License> {
return map {
val (position, name) = it.split(' ', limit = 2)
val (offset, length) = position.split(':').map { it.toInt() }
val id = name.replace(' ', '-')
License(id = id, name = name, offset = offset, length = length)
}
}

private fun BufferedSource.toRowList(): List<String> {
val list: MutableList<String> = mutableListOf()
while (true) {
val line = readUtf8Line() ?: break
list.add(line)
}
return list
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,22 @@ import dagger.Provides
import dagger.hilt.InstallIn
import dagger.hilt.components.SingletonComponent
import io.github.droidkaigi.confsched2023.model.BuildConfigProvider
import io.github.droidkaigi.confsched2023.model.License
import io.github.droidkaigi.confsched2023.model.OssLicenseRepository
import kotlinx.collections.immutable.PersistentList
import kotlinx.collections.immutable.persistentListOf
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.flowOf
import java.util.Optional
import javax.inject.Qualifier
import javax.inject.Singleton

@Qualifier
annotation class AppAndroidBuildConfig

@Qualifier
annotation class AppAndroidOssLicenseConfig

@Module
@InstallIn(SingletonComponent::class)
class BuildConfigProviderModule {
Expand All @@ -25,6 +34,16 @@ class BuildConfigProviderModule {
} else {
EmptyBuildConfigProvider
}

@Provides
@Singleton
fun provideOssLicenseRepositoryProvider(
@AppAndroidOssLicenseConfig ossLicenseRepository: Optional<OssLicenseRepository>,
): OssLicenseRepository = if (ossLicenseRepository.isPresent) {
ossLicenseRepository.get()
} else {
EmptyOssLicenseRepository
}
}

@InstallIn(SingletonComponent::class)
Expand All @@ -35,6 +54,19 @@ abstract class AppAndroidBuildConfigModule {
abstract fun bindBuildConfigProvider(): BuildConfigProvider
}

@InstallIn(SingletonComponent::class)
@Module
abstract class AppAndroidOssLicenseModule {
@BindsOptionalOf
@AppAndroidOssLicenseConfig
abstract fun bindOssLicenseProvider(): OssLicenseRepository
}

private object EmptyBuildConfigProvider : BuildConfigProvider {
override val versionName: String = ""
}

private object EmptyOssLicenseRepository : OssLicenseRepository {
override fun licenseMetaData(): Flow<PersistentList<License>> = flowOf(persistentListOf())
override fun licenseDetailData(): Flow<List<String>> = flowOf(emptyList())
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
package io.github.droidkaigi.confsched2023.model

import kotlinx.collections.immutable.PersistentList
import kotlinx.collections.immutable.persistentListOf

data class License(
val id: String,
val name: String,
val offset: Int,
val length: Int,
val detail: String = "",
)

data class OssLicenseGroup(
val title: String,
val licenses: List<License>,
val expand: Boolean = false,
)

data class OssLicense(
val groupList: PersistentList<OssLicenseGroup> = persistentListOf(),
)
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
package io.github.droidkaigi.confsched2023.model

import kotlinx.collections.immutable.PersistentList
import kotlinx.coroutines.flow.Flow

public interface OssLicenseRepository {
public fun licenseMetaData(): Flow<PersistentList<License>>

public fun licenseDetailData(): Flow<List<String>>
}
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ sealed class AboutStrings : Strings<AboutStrings>(Bindings) {
data object OthersTitle : AboutStrings()
data object CodeOfConduct : AboutStrings()
data object License : AboutStrings()
data object LicenseScreenTitle : AboutStrings()
data object PrivacyPolicy : AboutStrings()
data object AppVersion : AboutStrings()
data object LicenceDescription : AboutStrings()
Expand All @@ -42,6 +43,7 @@ sealed class AboutStrings : Strings<AboutStrings>(Bindings) {
OthersTitle -> "Others"
CodeOfConduct -> "行動規範"
License -> "ライセンス"
LicenseScreenTitle -> "ライセンス"
PrivacyPolicy -> "プライバシーポリシー"
AppVersion -> "アプリバージョン"
LicenceDescription -> "The Android robot is reproduced or modified from work created and shared by Google and used according to terms described in the Creative Commons 3.0 Attribution License."
Expand All @@ -63,6 +65,7 @@ sealed class AboutStrings : Strings<AboutStrings>(Bindings) {
OthersTitle -> bindings.defaultBinding(item, bindings)
CodeOfConduct -> "Code Of Conduct"
License -> "License"
LicenseScreenTitle -> "License"
PrivacyPolicy -> "Privacy Policy"
AppVersion -> "App Version"
LicenceDescription -> bindings.defaultBinding(item, bindings)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
package io.github.droidkaigi.confsched2023.about

import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.rememberScrollState
import androidx.compose.foundation.verticalScroll
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.filled.ArrowBack
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.IconButton
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Scaffold
import androidx.compose.material3.Text
import androidx.compose.material3.TopAppBar
import androidx.compose.runtime.Composable
import androidx.compose.runtime.collectAsState
import androidx.compose.runtime.getValue
import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp
import androidx.hilt.navigation.compose.hiltViewModel
import androidx.navigation.NavController
import androidx.navigation.NavGraphBuilder
import androidx.navigation.compose.composable
import io.github.droidkaigi.confsched2023.model.License

const val ossLicenseDetailScreenRouteNameArgument = "name"
const val ossLicenseDetailScreenRoute = "osslicense"
fun NavGraphBuilder.ossLicenseDetailScreen(onUpClick: () -> Unit) {
composable("$ossLicenseDetailScreenRoute/{$ossLicenseDetailScreenRouteNameArgument}") {
OssLicenseDetailScreen(
onUpClick = onUpClick,
)
}
}

fun NavController.navigateOssLicenseDetailScreen(
license: License,
) {
navigate("$ossLicenseDetailScreenRoute/${license.id}")
}

data class OssLicenseDetailScreenUiState(
val ossLicense: License? = null,
)

@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun OssLicenseDetailScreen(
modifier: Modifier = Modifier,
viewModel: OssLicenseDetailViewModel = hiltViewModel<OssLicenseDetailViewModel>(),
onUpClick: () -> Unit,
) {
val uiState by viewModel.uiState.collectAsState()

Scaffold(
modifier = modifier,
topBar = {
TopAppBar(
title = {
Text(text = "OSS ライセンス")
},
navigationIcon = {
IconButton(onClick = { onUpClick() }) {
Icon(
imageVector = Icons.Default.ArrowBack,
contentDescription = "back button",
)
}
},
)
},
) { paddingValues ->
Box(modifier = Modifier.padding(paddingValues)) {
Column(
modifier = Modifier
.verticalScroll(rememberScrollState())
.padding(horizontal = 8.dp),
) {
uiState.ossLicense?.run {
Text(
text = this.detail,
style = MaterialTheme.typography.bodyMedium,
)
}
}
}
}
}
Loading