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

Fix #545: Low fi onboarding part 1 #558

Merged
merged 38 commits into from
Feb 19, 2020
Merged
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
38 commits
Select commit Hold shift + click to select a range
b7150e8
Initial file introduction
Dec 12, 2019
7bc3b49
ViewModel amnd ViewPager introduction
Dec 12, 2019
2f98f99
Basic implementation finished
Dec 12, 2019
4c6d02e
Added images
Dec 13, 2019
c60d69c
Resolve merge conflicts
Dec 16, 2019
55f129a
New implementation
Dec 16, 2019
827a0a2
Button click listener added
Dec 16, 2019
42067d3
Skip button functionality added
Dec 16, 2019
985c531
Final implememntation
Dec 16, 2019
1bb3dc7
Added test cases
Dec 16, 2019
a345db9
Self review nit changes
Dec 16, 2019
439af56
Updated test cases- Not perfect
Dec 17, 2019
db52f8e
Indicator dots implementation change and test cases updated
Dec 18, 2019
0b07b22
Merge remote-tracking branch 'upstream/develop' into low-fi-onboardin…
Dec 19, 2019
c84679b
Added TODO
Dec 19, 2019
cbbf30b
Nit change
Dec 19, 2019
0a7bf10
Merge remote-tracking branch 'upstream/develop' into low-fi-onboardin…
Dec 24, 2019
219da2d
Resolve merge conflicts
Dec 24, 2019
d9116e6
Changes as per Ben's suggestion
Dec 24, 2019
2419ed9
Nit changes to OnboardingSlideViewModel
Dec 24, 2019
33bea19
Updated test cases
Jan 8, 2020
eb41f5b
Added contentDescription
Jan 9, 2020
4f7861e
Updates image test cases
Jan 9, 2020
fc8be33
All test cases fixed
Jan 10, 2020
fc6033e
Nit change
Jan 10, 2020
50eb1f4
Merge branch 'develop' of https://github.com/oppia/oppia-android into…
nikitamarysolomanpvt Jan 21, 2020
7f5910c
nit
nikitamarysolomanpvt Jan 22, 2020
eb85c75
nit
nikitamarysolomanpvt Jan 22, 2020
161c216
change in package structure, and changed on-board to single word onbo…
nikitamarysolomanpvt Feb 11, 2020
1ba63ea
changed on-boarded to single word onboarded
nikitamarysolomanpvt Feb 11, 2020
58c3f39
changed on-board to single word onboard in text class method names
nikitamarysolomanpvt Feb 11, 2020
e4b7854
changed on-board to single word onboard in comments
nikitamarysolomanpvt Feb 11, 2020
897b00e
nit
nikitamarysolomanpvt Feb 12, 2020
b4db282
Resolve merge conflicts with develop
Feb 14, 2020
46cea68
Merge remote-tracking branch 'upstream/domain-onboarding-flag-part-1'…
Feb 14, 2020
193840e
Onboarding flow full lowfi implementation
Feb 14, 2020
8bfb842
Updated code
Feb 18, 2020
5d70d01
Updated test cases
Feb 19, 2020
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
13 changes: 8 additions & 5 deletions app/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.INTERNET" />

<!-- TODO(#56): Reenable landscape support. -->
<application
android:name=".application.OppiaApplication"
Expand All @@ -26,6 +25,10 @@
android:name=".mydownloads.MyDownloadsActivity"
android:screenOrientation="portrait"
android:theme="@style/OppiaThemeWithoutActionBar" />
<activity
android:name=".onboarding.OnboardingActivity"
android:screenOrientation="portrait"
android:theme="@style/OppiaThemeWithoutActionBar" />
<activity
android:name=".player.exploration.ExplorationActivity"
android:screenOrientation="portrait"
Expand All @@ -46,8 +49,8 @@
android:theme="@style/OppiaThemeWithoutActionBar" />
<activity
android:name=".profile.ProfileActivity"
android:theme="@style/OppiaThemeWithoutActionBar"
android:screenOrientation="portrait" />
android:screenOrientation="portrait"
android:theme="@style/OppiaThemeWithoutActionBar" />
<activity
android:name=".settings.profile.ProfileEditActivity"
android:screenOrientation="portrait" />
Expand Down Expand Up @@ -81,8 +84,10 @@
android:name=".story.testing.StoryFragmentTestActivity"
android:theme="@style/OppiaThemeWithoutActionBar" />
<activity android:name=".testing.BindableAdapterTestActivity" />
<activity android:name=".testing.ConceptCardFragmentTestActivity" />
<activity android:name=".testing.ContentCardTestActivity" />
<activity android:name=".testing.ContinuePlayingFragmentTestActivity" />
<activity android:name=".testing.ExplorationInjectionActivity" />
<activity android:name=".testing.ExplorationTestActivity" />
<activity android:name=".testing.HtmlParserTestActivity" />
<activity android:name=".testing.InputInteractionViewTestActivity" />
Expand All @@ -92,8 +97,6 @@
<activity
android:name=".testing.TopicTestActivityForStory"
android:theme="@style/OppiaThemeWithoutActionBar" />
<activity android:name=".testing.ExplorationInjectionActivity" />
<activity android:name=".testing.ConceptCardFragmentTestActivity" />
<activity
android:name=".topic.questionplayer.QuestionPlayerActivity"
android:screenOrientation="portrait" />
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import dagger.Subcomponent
import org.oppia.app.fragment.FragmentComponent
import org.oppia.app.home.HomeActivity
import org.oppia.app.home.continueplaying.ContinuePlayingActivity
import org.oppia.app.onboarding.OnboardingActivity
import org.oppia.app.mydownloads.MyDownloadsActivity
import org.oppia.app.player.exploration.ExplorationActivity
import org.oppia.app.profile.AddProfileActivity
Expand Down Expand Up @@ -62,6 +63,7 @@ interface ActivityComponent {
fun inject(homeActivity: HomeActivity)
fun inject(htmlParserTestActivity: HtmlParserTestActivity)
fun inject(myDownloadsActivity: MyDownloadsActivity)
fun inject(onboardingActivity: OnboardingActivity)
fun inject(pinPasswordActivity: PinPasswordActivity)
fun inject(profileActivity: ProfileActivity)
fun inject(questionPlayerActivity: QuestionPlayerActivity)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import dagger.BindsInstance
import dagger.Subcomponent
import org.oppia.app.home.HomeFragment
import org.oppia.app.home.continueplaying.ContinuePlayingFragment
import org.oppia.app.onboarding.OnboardingFragment
import org.oppia.app.mydownloads.DownloadsTabFragment
import org.oppia.app.mydownloads.MyDownloadsFragment
import org.oppia.app.mydownloads.UpdatesTabFragment
Expand Down Expand Up @@ -50,6 +51,7 @@ interface FragmentComponent {
fun inject(explorationFragment: ExplorationFragment)
fun inject(homeFragment: HomeFragment)
fun inject(myDownloadsFragment: MyDownloadsFragment)
fun inject(onboardingFragment: OnboardingFragment)
fun inject(profileChooserFragment: ProfileChooserFragment)
fun inject(questionPlayerFragment: QuestionPlayerFragment)
fun inject(resetPinDialogFragment: ResetPinDialogFragment)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
package org.oppia.app.onboarding

import android.content.Context
import androidx.databinding.ObservableField
import androidx.lifecycle.ViewModel
import org.oppia.app.R
import org.oppia.app.viewmodel.ObservableViewModel

/** [ViewModel] for slide in onboarding flow. */
class OnboardingSlideViewModel(val context: Context, val index: Int) : ObservableViewModel() {
rt4914 marked this conversation as resolved.
Show resolved Hide resolved
val slideImage = ObservableField<Int>(R.drawable.ic_onboarding_0)
val title = ObservableField<String>(context.resources.getString(R.string.slide_0_title))
val description = ObservableField<String>(context.resources.getString(R.string.slide_0_description))

init {
slideChanged(index)
}

private fun slideChanged(index: Int) {
rt4914 marked this conversation as resolved.
Show resolved Hide resolved
when (index) {
0 -> {
slideImage.set(R.drawable.ic_onboarding_0)
title.set(context.resources.getString(R.string.slide_0_title))
description.set(context.resources.getString(R.string.slide_0_description))
}
1 -> {
slideImage.set(R.drawable.ic_onboarding_1)
title.set(context.resources.getString(R.string.slide_1_title))
description.set(context.resources.getString(R.string.slide_1_description))
}
2 -> {
slideImage.set(R.drawable.ic_onboarding_2)
title.set(context.resources.getString(R.string.slide_2_title))
description.set(context.resources.getString(R.string.slide_2_description))
}
3 -> {
slideImage.set(R.drawable.ic_onboarding_3)
title.set(context.resources.getString(R.string.slide_3_title))
description.set(context.resources.getString(R.string.slide_3_description))
}
else -> {
slideImage.set(R.drawable.ic_onboarding_0)
rt4914 marked this conversation as resolved.
Show resolved Hide resolved
title.set(context.resources.getString(R.string.slide_3_title))
description.set(context.resources.getString(R.string.slide_3_description))
}
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
package org.oppia.app.onboarding

import androidx.databinding.ObservableField
import androidx.lifecycle.ViewModel
import org.oppia.app.viewmodel.ObservableViewModel
import javax.inject.Inject

/** [ViewModel] for [OnboardingFragment]. */
class OnboardingViewModel @Inject constructor() : ObservableViewModel() {
val slideNumber = ObservableField<Int>(0)

fun slideChanged(index: Int) {
slideNumber.set(index)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
package org.oppia.app.onboarding

import android.os.Bundle
import org.oppia.app.activity.InjectableAppCompatActivity
import org.oppia.app.profile.ProfileActivity
import javax.inject.Inject

/** Activity to contain the onbaording flow for learners. */
class OnboardingActivity : InjectableAppCompatActivity(), RouteToProfileListener {
@Inject lateinit var onboardingActivityPresenter: OnboardingActivityPresenter

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
activityComponent.inject(this)
onboardingActivityPresenter.handleOnCreate()
}

override fun routeToProfile() {
startActivity(ProfileActivity.createProfileActivityIntent(this))
finish()
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
package org.oppia.app.onboarding

import androidx.appcompat.app.AppCompatActivity
import org.oppia.app.R
import org.oppia.app.activity.ActivityScope
import javax.inject.Inject

/** The presenter for [OnboardingActivity]. */
@ActivityScope
class OnboardingActivityPresenter @Inject constructor(private val activity: AppCompatActivity) {
fun handleOnCreate() {
activity.setContentView(R.layout.onboarding_activity)
if (getOnboardingFragment() == null) {
activity.supportFragmentManager.beginTransaction().add(
R.id.onboarding_fragment_placeholder,
OnboardingFragment()
).commitNow()
}
}

private fun getOnboardingFragment(): OnboardingFragment? {
return activity.supportFragmentManager.findFragmentById(R.id.onboarding_fragment_placeholder) as OnboardingFragment?
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
package org.oppia.app.onboarding

import android.content.Context
import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import org.oppia.app.fragment.InjectableFragment
import javax.inject.Inject

/** Fragment that contains an onboarding flow of the app. */
class OnboardingFragment : InjectableFragment() {
@Inject lateinit var onboardingFragmentPresenter: OnboardingFragmentPresenter

override fun onAttach(context: Context) {
super.onAttach(context)
fragmentComponent.inject(this)
}

override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
return onboardingFragmentPresenter.handleCreateView(inflater, container)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
package org.oppia.app.onboarding

import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.appcompat.app.AppCompatActivity
import androidx.fragment.app.Fragment
import androidx.viewpager.widget.ViewPager
import kotlinx.android.synthetic.main.onboarding_fragment.view.*
import org.oppia.app.databinding.OnboardingFragmentBinding
import org.oppia.app.fragment.FragmentScope
import org.oppia.app.viewmodel.ViewModelProvider
import javax.inject.Inject

/** The presenter for [OnboardingFragment]. */
@FragmentScope
class OnboardingFragmentPresenter @Inject constructor(
activity: AppCompatActivity,
private val fragment: Fragment,
private val viewModelProvider: ViewModelProvider<OnboardingViewModel>
) {
private lateinit var onboardingPagerAdapter: OnboardingPagerAdapter
private val routeToProfileListener = activity as RouteToProfileListener
private lateinit var binding: OnboardingFragmentBinding
private lateinit var slidesViewPager: ViewPager
fun handleCreateView(inflater: LayoutInflater, container: ViewGroup?): View? {
rt4914 marked this conversation as resolved.
Show resolved Hide resolved
binding = OnboardingFragmentBinding.inflate(inflater, container, /* attachToRoot= */ false)
// NB: Both the view model and lifecycle owner must be set in order to correctly bind LiveData elements to
// data-bound view models.
binding.let {
it.lifecycleOwner = fragment
it.presenter = this
it.viewModel = getOnboardingViewModel()
}
slidesViewPager = binding.root.onboarding_slide_view_pager as ViewPager
rt4914 marked this conversation as resolved.
Show resolved Hide resolved
setUpViewPager(slidesViewPager)
return binding.root
}

private fun setUpViewPager(viewPager: ViewPager) {
onboardingPagerAdapter = OnboardingPagerAdapter(fragment.requireContext())
viewPager.adapter = onboardingPagerAdapter
viewPager.addOnPageChangeListener(object : ViewPager.OnPageChangeListener {
override fun onPageScrollStateChanged(state: Int) {
}

override fun onPageScrolled(position: Int, positionOffset: Float, positionOffsetPixels: Int) {
}

override fun onPageSelected(position: Int) {
getOnboardingViewModel().slideChanged(position)
}
})
}

fun clickOnGetStarted() {
routeToProfileListener.routeToProfile()
}

fun clickOnSkip() {
getOnboardingViewModel().slideChanged(3)
rt4914 marked this conversation as resolved.
Show resolved Hide resolved
slidesViewPager.currentItem = 3
}

private fun getOnboardingViewModel(): OnboardingViewModel {
return viewModelProvider.getForFragment(fragment, OnboardingViewModel::class.java)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
package org.oppia.app.onboarding

import android.content.Context
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.constraintlayout.widget.ConstraintLayout
import androidx.viewpager.widget.PagerAdapter
import org.oppia.app.databinding.OnboardingSlideBinding

/** Adapter to control the slide details in onboarding flow. */
class OnboardingPagerAdapter(val context: Context) : PagerAdapter() {
override fun instantiateItem(container: ViewGroup, position: Int): Any {
val binding = OnboardingSlideBinding.inflate(LayoutInflater.from(context), container, false)
val onboardingSlideViewModel = OnboardingSlideViewModel(context, position)
binding.viewModel = onboardingSlideViewModel
container.addView(binding.root)
return binding.root
}

override fun getCount(): Int {
return 4
}

override fun isViewFromObject(view: View, `object`: Any): Boolean {
return view == `object`
rt4914 marked this conversation as resolved.
Show resolved Hide resolved
}

override fun destroyItem(container: ViewGroup, position: Int, `object`: Any) {
container.removeView(`object` as ConstraintLayout)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
package org.oppia.app.onboarding

/** Listener for when an activity should route to [ProfileActivity]. */
interface RouteToProfileListener {
fun routeToProfile()
rt4914 marked this conversation as resolved.
Show resolved Hide resolved
}
9 changes: 9 additions & 0 deletions app/src/main/java/org/oppia/app/profile/ProfileActivity.kt
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
package org.oppia.app.profile

import android.content.Context
import android.content.Intent
import android.os.Bundle
import kotlinx.coroutines.ExperimentalCoroutinesApi
import org.oppia.app.activity.InjectableAppCompatActivity
Expand All @@ -16,4 +18,11 @@ class ProfileActivity : InjectableAppCompatActivity() {
activityComponent.inject(this)
profileActivityPresenter.handleOnCreate()
}

companion object {
/** Returns a new [Intent] to route to [ProfileActivity]. */
fun createProfileActivityIntent(context: Context): Intent {
return Intent(context, ProfileActivity::class.java)
}
}
}
3 changes: 2 additions & 1 deletion app/src/main/java/org/oppia/app/splash/SplashActivity.kt
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import android.os.Bundle
import android.view.WindowManager
import androidx.appcompat.app.AppCompatActivity
import org.oppia.app.R
import org.oppia.app.onboarding.OnboardingActivity
import org.oppia.app.profile.ProfileActivity

/** An activity that shows a temporary loading page until the app is fully loaded then navigates to [ProfileActivity]. */
Expand All @@ -15,7 +16,7 @@ class SplashActivity : AppCompatActivity() {
setContentView(R.layout.splash_activity)

window.setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
val intent = Intent(this@SplashActivity, ProfileActivity::class.java)
val intent = Intent(this@SplashActivity, OnboardingActivity::class.java)
startActivity(intent)
finish()
}
Expand Down
Loading