From afb5ff4f586f495e619356df2a9878f2f28ac3c3 Mon Sep 17 00:00:00 2001 From: Hamza Israr <71447999+HamzaIsrar12@users.noreply.github.com> Date: Thu, 12 Sep 2024 14:22:14 +0500 Subject: [PATCH] fix: Analytics events improvements & Add missing ones --- .../main/java/org/openedx/app/AppAnalytics.kt | 4 + .../main/java/org/openedx/app/MainFragment.kt | 3 +- .../java/org/openedx/app/MainViewModel.kt | 4 + .../java/org/openedx/app/di/ScreenModule.kt | 4 +- .../container/CourseContainerViewModel.kt | 1 + .../container/CourseContainerViewModelTest.kt | 36 +++++++++ .../presentation/DashboardAnalytics.kt | 8 +- .../learn/presentation/LearnFragment.kt | 73 +++++++------------ .../learn/presentation/LearnUIState.kt | 5 ++ .../learn/presentation/LearnViewModel.kt | 42 ++++++++++- 10 files changed, 127 insertions(+), 53 deletions(-) create mode 100644 dashboard/src/main/java/org/openedx/learn/presentation/LearnUIState.kt diff --git a/app/src/main/java/org/openedx/app/AppAnalytics.kt b/app/src/main/java/org/openedx/app/AppAnalytics.kt index a122e79c1..0fe3ed4be 100644 --- a/app/src/main/java/org/openedx/app/AppAnalytics.kt +++ b/app/src/main/java/org/openedx/app/AppAnalytics.kt @@ -12,6 +12,10 @@ enum class AppAnalyticsEvent(val eventName: String, val biValue: String) { "Launch", "edx.bi.app.launch" ), + LEARN( + "MainDashboard:Learn", + "edx.bi.app.main_dashboard.learn" + ), DISCOVER( "MainDashboard:Discover", "edx.bi.app.main_dashboard.discover" diff --git a/app/src/main/java/org/openedx/app/MainFragment.kt b/app/src/main/java/org/openedx/app/MainFragment.kt index 4011b3a04..8a3a476cb 100644 --- a/app/src/main/java/org/openedx/app/MainFragment.kt +++ b/app/src/main/java/org/openedx/app/MainFragment.kt @@ -46,6 +46,7 @@ class MainFragment : Fragment(R.layout.fragment_main) { binding.bottomNavView.setOnItemSelectedListener { when (it.itemId) { R.id.fragmentLearn -> { + viewModel.logLearnTabClickedEvent() binding.viewPager.setCurrentItem(0, false) } @@ -89,7 +90,7 @@ class MainFragment : Fragment(R.layout.fragment_main) { putString(ARG_INFO_TYPE, "") } - when (requireArguments().getString(ARG_OPEN_TAB, HomeTab.LEARN.name)) { + when (requireArguments().getString(ARG_OPEN_TAB, "")) { HomeTab.LEARN.name, HomeTab.PROGRAMS.name -> { binding.bottomNavView.selectedItemId = R.id.fragmentLearn diff --git a/app/src/main/java/org/openedx/app/MainViewModel.kt b/app/src/main/java/org/openedx/app/MainViewModel.kt index ff24f4ff8..dc51a4daa 100644 --- a/app/src/main/java/org/openedx/app/MainViewModel.kt +++ b/app/src/main/java/org/openedx/app/MainViewModel.kt @@ -49,6 +49,10 @@ class MainViewModel( _isBottomBarEnabled.value = enable } + fun logLearnTabClickedEvent() { + logScreenEvent(AppAnalyticsEvent.LEARN) + } + fun logDiscoveryTabClickedEvent() { logScreenEvent(AppAnalyticsEvent.DISCOVER) } diff --git a/app/src/main/java/org/openedx/app/di/ScreenModule.kt b/app/src/main/java/org/openedx/app/di/ScreenModule.kt index 31ebf741e..d799c266b 100644 --- a/app/src/main/java/org/openedx/app/di/ScreenModule.kt +++ b/app/src/main/java/org/openedx/app/di/ScreenModule.kt @@ -157,7 +157,9 @@ val screenModule = module { ) } viewModel { AllEnrolledCoursesViewModel(get(), get(), get(), get(), get(), get(), get()) } - viewModel { LearnViewModel(get(), get(), get()) } + viewModel { (openTab: String) -> + LearnViewModel(openTab, get(), get(), get()) + } factory { DiscoveryRepository(get(), get(), get()) } factory { DiscoveryInteractor(get()) } diff --git a/course/src/main/java/org/openedx/course/presentation/container/CourseContainerViewModel.kt b/course/src/main/java/org/openedx/course/presentation/container/CourseContainerViewModel.kt index a743730ec..7e6bf3adb 100644 --- a/course/src/main/java/org/openedx/course/presentation/container/CourseContainerViewModel.kt +++ b/course/src/main/java/org/openedx/course/presentation/container/CourseContainerViewModel.kt @@ -312,6 +312,7 @@ class CourseContainerViewModel( private fun courseDashboardViewed() { logCourseContainerEvent(CourseAnalyticsEvent.DASHBOARD) + courseTabClickedEvent() } private fun courseTabClickedEvent() { diff --git a/course/src/test/java/org/openedx/course/presentation/container/CourseContainerViewModelTest.kt b/course/src/test/java/org/openedx/course/presentation/container/CourseContainerViewModelTest.kt index 98cf58a8b..c16b9acb9 100644 --- a/course/src/test/java/org/openedx/course/presentation/container/CourseContainerViewModelTest.kt +++ b/course/src/test/java/org/openedx/course/presentation/container/CourseContainerViewModelTest.kt @@ -259,6 +259,12 @@ class CourseContainerViewModelTest { any() ) } returns Unit + every { + analytics.logScreenEvent( + CourseAnalyticsEvent.HOME_TAB.eventName, + any() + ) + } returns Unit viewModel.fetchCourseDetails() advanceUntilIdle() @@ -269,6 +275,12 @@ class CourseContainerViewModelTest { any() ) } + verify(exactly = 1) { + analytics.logScreenEvent( + CourseAnalyticsEvent.HOME_TAB.eventName, + any() + ) + } assert(!viewModel.refreshing.value) assert(viewModel.courseAccessStatus.value == CourseAccessError.UNKNOWN) } @@ -299,6 +311,12 @@ class CourseContainerViewModelTest { any() ) } returns Unit + every { + analytics.logScreenEvent( + CourseAnalyticsEvent.HOME_TAB.eventName, + any() + ) + } returns Unit viewModel.fetchCourseDetails() advanceUntilIdle() @@ -309,6 +327,12 @@ class CourseContainerViewModelTest { any() ) } + verify(exactly = 1) { + analytics.logScreenEvent( + CourseAnalyticsEvent.HOME_TAB.eventName, + any() + ) + } assert(viewModel.errorMessage.value == null) assert(!viewModel.refreshing.value) assert(viewModel.courseAccessStatus.value != null) @@ -339,6 +363,12 @@ class CourseContainerViewModelTest { any() ) } returns Unit + every { + analytics.logScreenEvent( + CourseAnalyticsEvent.HOME_TAB.eventName, + any() + ) + } returns Unit viewModel.fetchCourseDetails() advanceUntilIdle() coVerify(exactly = 0) { courseApi.getEnrollmentDetails(any()) } @@ -348,6 +378,12 @@ class CourseContainerViewModelTest { any() ) } + verify(exactly = 1) { + analytics.logScreenEvent( + CourseAnalyticsEvent.HOME_TAB.eventName, + any() + ) + } assert(viewModel.errorMessage.value == null) assert(!viewModel.refreshing.value) diff --git a/dashboard/src/main/java/org/openedx/dashboard/presentation/DashboardAnalytics.kt b/dashboard/src/main/java/org/openedx/dashboard/presentation/DashboardAnalytics.kt index cf7097a64..066b8ff73 100644 --- a/dashboard/src/main/java/org/openedx/dashboard/presentation/DashboardAnalytics.kt +++ b/dashboard/src/main/java/org/openedx/dashboard/presentation/DashboardAnalytics.kt @@ -7,12 +7,12 @@ interface DashboardAnalytics { enum class DashboardAnalyticsEvent(val eventName: String, val biValue: String) { MY_COURSES( - "MainDashboard:My Courses", - "edx.bi.app.main_dashboard.my_course" + "Learn:My Courses", + "edx.bi.app.main_dashboard.learn.my_course" ), MY_PROGRAMS( - "MainDashboard:My Programs", - "edx.bi.app.main_dashboard.my_program" + "Learn:My Programs", + "edx.bi.app.main_dashboard.learn.my_programs" ), } diff --git a/dashboard/src/main/java/org/openedx/learn/presentation/LearnFragment.kt b/dashboard/src/main/java/org/openedx/learn/presentation/LearnFragment.kt index a0e304170..fcbb5c13d 100644 --- a/dashboard/src/main/java/org/openedx/learn/presentation/LearnFragment.kt +++ b/dashboard/src/main/java/org/openedx/learn/presentation/LearnFragment.kt @@ -22,7 +22,7 @@ import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.ExpandMore import androidx.compose.material.icons.filled.ManageAccounts import androidx.compose.runtime.Composable -import androidx.compose.runtime.LaunchedEffect +import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember @@ -31,7 +31,6 @@ import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.draw.rotate import androidx.compose.ui.graphics.Color -import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.res.stringResource import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.Dp @@ -42,6 +41,7 @@ import androidx.fragment.app.FragmentManager import androidx.viewpager2.widget.ViewPager2 import org.koin.androidx.compose.koinViewModel import org.koin.androidx.viewmodel.ext.android.viewModel +import org.koin.core.parameter.parametersOf import org.openedx.core.adapter.NavigationFragmentAdapter import org.openedx.core.presentation.global.viewBinding import org.openedx.core.ui.crop @@ -60,24 +60,29 @@ import org.openedx.core.R as CoreR class LearnFragment : Fragment(R.layout.fragment_learn) { private val binding by viewBinding(FragmentLearnBinding::bind) - private val viewModel by viewModel() + private val viewModel by viewModel { + parametersOf(requireArguments().getString(ARG_OPEN_TAB, LearnTab.COURSES.name)) + } private lateinit var adapter: NavigationFragmentAdapter override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) initViewPager() - val openTab = requireArguments().getString(ARG_OPEN_TAB, LearnTab.COURSES.name) - val defaultLearnType = if (openTab == LearnTab.PROGRAMS.name) { - LearnType.PROGRAMS - } else { - LearnType.COURSES - } binding.header.setContent { OpenEdXTheme { + val uiState by viewModel.uiState.collectAsState() + binding.viewPager.setCurrentItem( + when (uiState.learnType) { + LearnType.COURSES -> 0 + LearnType.PROGRAMS -> 1 + }, false + ) Header( fragmentManager = requireParentFragment().parentFragmentManager, - defaultLearnType = defaultLearnType, - viewPager = binding.viewPager + selectedLearnType = uiState.learnType, + onUpdateLearnType = { learnType -> + viewModel.updateLearnType(learnType) + }, ) } } @@ -93,23 +98,12 @@ class LearnFragment : Fragment(R.layout.fragment_learn) { } binding.viewPager.adapter = adapter binding.viewPager.setUserInputEnabled(false) - - binding.viewPager.registerOnPageChangeCallback(object : ViewPager2.OnPageChangeCallback() { - override fun onPageSelected(position: Int) { - super.onPageSelected(position) - if (LearnType.COURSES.ordinal == position) { - viewModel.logMyCoursesTabClickedEvent() - } else { - viewModel.logMyProgramsTabClickedEvent() - } - } - }) } companion object { private const val ARG_OPEN_TAB = "open_tab" fun newInstance( - openTab: String = LearnTab.COURSES.name + openTab: String = LearnTab.COURSES.name, ): LearnFragment { val fragment = LearnFragment() fragment.arguments = bundleOf( @@ -123,8 +117,8 @@ class LearnFragment : Fragment(R.layout.fragment_learn) { @Composable private fun Header( fragmentManager: FragmentManager, - defaultLearnType: LearnType, - viewPager: ViewPager2, + selectedLearnType: LearnType, + onUpdateLearnType: (LearnType) -> Unit ) { val viewModel: LearnViewModel = koinViewModel() val windowSize = rememberWindowSize() @@ -156,8 +150,8 @@ private fun Header( modifier = Modifier .align(Alignment.Start) .padding(horizontal = 16.dp), - defaultLearnType = defaultLearnType, - viewPager = viewPager + selectedLearnType = selectedLearnType, + onUpdateLearnType = onUpdateLearnType ) } } @@ -200,25 +194,15 @@ private fun Title( @Composable private fun LearnDropdownMenu( modifier: Modifier = Modifier, - defaultLearnType: LearnType, - viewPager: ViewPager2, + selectedLearnType: LearnType, + onUpdateLearnType: (LearnType) -> Unit ) { var expanded by remember { mutableStateOf(false) } - var currentValue by remember { mutableStateOf(defaultLearnType) } val iconRotation by animateFloatAsState( targetValue = if (expanded) 180f else 0f, label = "" ) - LaunchedEffect(currentValue) { - viewPager.setCurrentItem( - when (currentValue) { - LearnType.COURSES -> 0 - LearnType.PROGRAMS -> 1 - }, false - ) - } - Column( modifier = modifier ) { @@ -230,7 +214,7 @@ private fun LearnDropdownMenu( verticalAlignment = Alignment.CenterVertically ) { Text( - text = stringResource(id = currentValue.title), + text = stringResource(id = selectedLearnType.title), color = MaterialTheme.appColors.textDark, style = MaterialTheme.appTypography.titleSmall ) @@ -261,7 +245,7 @@ private fun LearnDropdownMenu( for (learnType in LearnType.entries) { val background: Color val textColor: Color - if (currentValue == learnType) { + if (selectedLearnType == learnType) { background = MaterialTheme.appColors.primary textColor = MaterialTheme.appColors.primaryButtonText } else { @@ -272,7 +256,7 @@ private fun LearnDropdownMenu( modifier = Modifier .background(background), onClick = { - currentValue = learnType + onUpdateLearnType(learnType) expanded = false } ) { @@ -303,10 +287,9 @@ private fun HeaderPreview() { @Composable private fun LearnDropdownMenuPreview() { OpenEdXTheme { - val context = LocalContext.current LearnDropdownMenu( - defaultLearnType = LearnType.COURSES, - viewPager = ViewPager2(context) + selectedLearnType = LearnType.COURSES, + onUpdateLearnType = {} ) } } diff --git a/dashboard/src/main/java/org/openedx/learn/presentation/LearnUIState.kt b/dashboard/src/main/java/org/openedx/learn/presentation/LearnUIState.kt new file mode 100644 index 000000000..934caa374 --- /dev/null +++ b/dashboard/src/main/java/org/openedx/learn/presentation/LearnUIState.kt @@ -0,0 +1,5 @@ +package org.openedx.learn.presentation + +import org.openedx.learn.LearnType + +data class LearnUIState(val learnType: LearnType) diff --git a/dashboard/src/main/java/org/openedx/learn/presentation/LearnViewModel.kt b/dashboard/src/main/java/org/openedx/learn/presentation/LearnViewModel.kt index ee38caf75..ba91b14ff 100644 --- a/dashboard/src/main/java/org/openedx/learn/presentation/LearnViewModel.kt +++ b/dashboard/src/main/java/org/openedx/learn/presentation/LearnViewModel.kt @@ -1,5 +1,11 @@ package org.openedx.learn.presentation +import androidx.lifecycle.viewModelScope +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.flow.update +import kotlinx.coroutines.launch import androidx.fragment.app.FragmentManager import org.openedx.DashboardNavigator import org.openedx.core.config.Config @@ -8,12 +14,26 @@ import org.openedx.dashboard.presentation.DashboardAnalyticsEvent import org.openedx.dashboard.presentation.DashboardAnalyticsKey import org.openedx.dashboard.presentation.DashboardRouter import org.openedx.foundation.presentation.BaseViewModel +import org.openedx.learn.LearnType class LearnViewModel( + openTab: String, private val config: Config, private val dashboardRouter: DashboardRouter, private val analytics: DashboardAnalytics, ) : BaseViewModel() { + private val _uiState = MutableStateFlow( + LearnUIState( + if (openTab == LearnTab.PROGRAMS.name) { + LearnType.PROGRAMS + } else { + LearnType.COURSES + } + ) + ) + + val uiState: StateFlow + get() = _uiState.asStateFlow() private val dashboardType get() = config.getDashboardConfig().getType() val isProgramTypeWebView get() = config.getProgramConfig().isViewTypeWebView() @@ -26,11 +46,29 @@ class LearnViewModel( val getProgramFragment get() = dashboardRouter.getProgramFragment() - fun logMyCoursesTabClickedEvent() { + init { + viewModelScope.launch { + _uiState.collect { uiState -> + if (uiState.learnType == LearnType.COURSES) { + logMyCoursesTabClickedEvent() + } else { + logMyProgramsTabClickedEvent() + } + } + } + } + + fun updateLearnType(learnType: LearnType) { + viewModelScope.launch { + _uiState.update { it.copy(learnType = learnType) } + } + } + + private fun logMyCoursesTabClickedEvent() { logScreenEvent(DashboardAnalyticsEvent.MY_COURSES) } - fun logMyProgramsTabClickedEvent() { + private fun logMyProgramsTabClickedEvent() { logScreenEvent(DashboardAnalyticsEvent.MY_PROGRAMS) }