mirror of
https://github.com/kiwix/kiwix-android.git
synced 2025-09-08 06:42:21 -04:00
Merge pull request #4258 from kiwix/Fixes#4250
Migrated the `AddNoteDialog` to jetpack.
This commit is contained in:
commit
d4ebe42d49
@ -19,6 +19,7 @@
|
|||||||
package org.kiwix.kiwixmobile.note
|
package org.kiwix.kiwixmobile.note
|
||||||
|
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
|
import androidx.compose.ui.test.junit4.createComposeRule
|
||||||
import androidx.core.content.edit
|
import androidx.core.content.edit
|
||||||
import androidx.core.net.toUri
|
import androidx.core.net.toUri
|
||||||
import androidx.lifecycle.Lifecycle
|
import androidx.lifecycle.Lifecycle
|
||||||
@ -33,6 +34,7 @@ import androidx.test.uiautomator.UiDevice
|
|||||||
import com.google.android.apps.common.testing.accessibility.framework.AccessibilityCheckResultUtils.matchesCheck
|
import com.google.android.apps.common.testing.accessibility.framework.AccessibilityCheckResultUtils.matchesCheck
|
||||||
import com.google.android.apps.common.testing.accessibility.framework.AccessibilityCheckResultUtils.matchesViews
|
import com.google.android.apps.common.testing.accessibility.framework.AccessibilityCheckResultUtils.matchesViews
|
||||||
import com.google.android.apps.common.testing.accessibility.framework.checks.DuplicateClickableBoundsCheck
|
import com.google.android.apps.common.testing.accessibility.framework.checks.DuplicateClickableBoundsCheck
|
||||||
|
import com.google.android.apps.common.testing.accessibility.framework.checks.SpeakableTextPresentCheck
|
||||||
import com.google.android.apps.common.testing.accessibility.framework.checks.TouchTargetSizeCheck
|
import com.google.android.apps.common.testing.accessibility.framework.checks.TouchTargetSizeCheck
|
||||||
import leakcanary.LeakAssertions
|
import leakcanary.LeakAssertions
|
||||||
import org.hamcrest.Matchers.allOf
|
import org.hamcrest.Matchers.allOf
|
||||||
@ -64,6 +66,9 @@ class NoteFragmentTest : BaseActivityTest() {
|
|||||||
|
|
||||||
private lateinit var kiwixMainActivity: KiwixMainActivity
|
private lateinit var kiwixMainActivity: KiwixMainActivity
|
||||||
|
|
||||||
|
@get:Rule
|
||||||
|
val composeTestRule = createComposeRule()
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
override fun waitForIdle() {
|
override fun waitForIdle() {
|
||||||
UiDevice.getInstance(InstrumentationRegistry.getInstrumentation()).apply {
|
UiDevice.getInstance(InstrumentationRegistry.getInstrumentation()).apply {
|
||||||
@ -107,7 +112,8 @@ class NoteFragmentTest : BaseActivityTest() {
|
|||||||
allOf(
|
allOf(
|
||||||
matchesCheck(TouchTargetSizeCheck::class.java),
|
matchesCheck(TouchTargetSizeCheck::class.java),
|
||||||
matchesViews(withContentDescription("More options"))
|
matchesViews(withContentDescription("More options"))
|
||||||
)
|
),
|
||||||
|
matchesCheck(SpeakableTextPresentCheck::class.java)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -133,16 +139,16 @@ class NoteFragmentTest : BaseActivityTest() {
|
|||||||
StandardActions.closeDrawer() // close the drawer if open before running the test cases.
|
StandardActions.closeDrawer() // close the drawer if open before running the test cases.
|
||||||
note {
|
note {
|
||||||
clickOnNoteMenuItem(context)
|
clickOnNoteMenuItem(context)
|
||||||
assertNoteDialogDisplayed()
|
assertNoteDialogDisplayed(composeTestRule)
|
||||||
writeDemoNote()
|
writeDemoNote(composeTestRule)
|
||||||
saveNote()
|
saveNote(composeTestRule)
|
||||||
pressBack()
|
pressBack()
|
||||||
openNoteFragment()
|
openNoteFragment()
|
||||||
assertToolbarExist()
|
assertToolbarExist()
|
||||||
assertNoteRecyclerViewExist()
|
assertNoteRecyclerViewExist()
|
||||||
clickOnSavedNote()
|
clickOnSavedNote()
|
||||||
clickOnOpenNote()
|
clickOnOpenNote()
|
||||||
assertNoteSaved()
|
assertNoteSaved(composeTestRule)
|
||||||
// to close the note dialog.
|
// to close the note dialog.
|
||||||
pressBack()
|
pressBack()
|
||||||
// to close the notes fragment.
|
// to close the notes fragment.
|
||||||
@ -166,7 +172,7 @@ class NoteFragmentTest : BaseActivityTest() {
|
|||||||
assertNoteRecyclerViewExist()
|
assertNoteRecyclerViewExist()
|
||||||
clickOnSavedNote()
|
clickOnSavedNote()
|
||||||
clickOnOpenNote()
|
clickOnOpenNote()
|
||||||
assertNoteSaved()
|
assertNoteSaved(composeTestRule)
|
||||||
pressBack()
|
pressBack()
|
||||||
}
|
}
|
||||||
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.N_MR1) {
|
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.N_MR1) {
|
||||||
@ -182,16 +188,16 @@ class NoteFragmentTest : BaseActivityTest() {
|
|||||||
note {
|
note {
|
||||||
assertHomePageIsLoadedOfTestZimFile()
|
assertHomePageIsLoadedOfTestZimFile()
|
||||||
clickOnNoteMenuItem(context)
|
clickOnNoteMenuItem(context)
|
||||||
assertNoteDialogDisplayed()
|
assertNoteDialogDisplayed(composeTestRule)
|
||||||
writeDemoNote()
|
writeDemoNote(composeTestRule)
|
||||||
saveNote()
|
saveNote(composeTestRule)
|
||||||
pressBack()
|
pressBack()
|
||||||
openNoteFragment()
|
openNoteFragment()
|
||||||
assertToolbarExist()
|
assertToolbarExist()
|
||||||
assertNoteRecyclerViewExist()
|
assertNoteRecyclerViewExist()
|
||||||
clickOnSavedNote()
|
clickOnSavedNote()
|
||||||
clickOnOpenNote()
|
clickOnOpenNote()
|
||||||
assertNoteSaved()
|
assertNoteSaved(composeTestRule)
|
||||||
// to close the note dialog.
|
// to close the note dialog.
|
||||||
pressBack()
|
pressBack()
|
||||||
// to close the notes fragment.
|
// to close the notes fragment.
|
||||||
@ -201,23 +207,25 @@ class NoteFragmentTest : BaseActivityTest() {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun testNoteEntryIsRemovedFromDatabaseWhenDeletedInAddNoteDialog() {
|
fun testNoteEntryIsRemovedFromDatabaseWhenDeletedInAddNoteDialog() {
|
||||||
deletePreviouslySavedNotes()
|
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.N_MR1) {
|
||||||
loadZimFileInReader("testzim.zim")
|
deletePreviouslySavedNotes()
|
||||||
note {
|
loadZimFileInReader("testzim.zim")
|
||||||
clickOnNoteMenuItem(context)
|
note {
|
||||||
assertNoteDialogDisplayed()
|
clickOnNoteMenuItem(context)
|
||||||
writeDemoNote()
|
assertNoteDialogDisplayed(composeTestRule)
|
||||||
saveNote()
|
writeDemoNote(composeTestRule)
|
||||||
pressBack()
|
saveNote(composeTestRule)
|
||||||
openNoteFragment()
|
pressBack()
|
||||||
assertToolbarExist()
|
openNoteFragment()
|
||||||
assertNoteRecyclerViewExist()
|
assertToolbarExist()
|
||||||
clickOnSavedNote()
|
assertNoteRecyclerViewExist()
|
||||||
clickOnOpenNote()
|
clickOnSavedNote()
|
||||||
assertNoteSaved()
|
clickOnOpenNote()
|
||||||
clickOnDeleteIcon()
|
assertNoteSaved(composeTestRule)
|
||||||
pressBack()
|
clickOnDeleteIcon(composeTestRule)
|
||||||
assertNoNotesTextDisplayed()
|
pressBack()
|
||||||
|
assertNoNotesTextDisplayed()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -228,9 +236,9 @@ class NoteFragmentTest : BaseActivityTest() {
|
|||||||
// Save a note.
|
// Save a note.
|
||||||
note {
|
note {
|
||||||
clickOnNoteMenuItem(context)
|
clickOnNoteMenuItem(context)
|
||||||
assertNoteDialogDisplayed()
|
assertNoteDialogDisplayed(composeTestRule)
|
||||||
writeDemoNote()
|
writeDemoNote(composeTestRule)
|
||||||
saveNote()
|
saveNote(composeTestRule)
|
||||||
pressBack()
|
pressBack()
|
||||||
}
|
}
|
||||||
// Delete that note from "Note" screen.
|
// Delete that note from "Note" screen.
|
||||||
@ -238,8 +246,8 @@ class NoteFragmentTest : BaseActivityTest() {
|
|||||||
// Test the note file is deleted or not.
|
// Test the note file is deleted or not.
|
||||||
note {
|
note {
|
||||||
clickOnNoteMenuItem(context)
|
clickOnNoteMenuItem(context)
|
||||||
assertNoteDialogDisplayed()
|
assertNoteDialogDisplayed(composeTestRule)
|
||||||
assertNotDoesNotExist()
|
assertNotDoesNotExist(composeTestRule)
|
||||||
pressBack()
|
pressBack()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,14 +19,17 @@
|
|||||||
package org.kiwix.kiwixmobile.note
|
package org.kiwix.kiwixmobile.note
|
||||||
|
|
||||||
import android.content.Context
|
import android.content.Context
|
||||||
|
import androidx.compose.ui.test.assertTextContains
|
||||||
|
import androidx.compose.ui.test.assertTextEquals
|
||||||
|
import androidx.compose.ui.test.junit4.ComposeContentTestRule
|
||||||
|
import androidx.compose.ui.test.onNodeWithTag
|
||||||
|
import androidx.compose.ui.test.performClick
|
||||||
|
import androidx.compose.ui.test.performTextReplacement
|
||||||
import androidx.recyclerview.widget.RecyclerView
|
import androidx.recyclerview.widget.RecyclerView
|
||||||
import androidx.test.espresso.Espresso.closeSoftKeyboard
|
import androidx.test.espresso.Espresso.closeSoftKeyboard
|
||||||
import androidx.test.espresso.Espresso.onView
|
import androidx.test.espresso.Espresso.onView
|
||||||
import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu
|
import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu
|
||||||
import androidx.test.espresso.action.ViewActions.clearText
|
|
||||||
import androidx.test.espresso.action.ViewActions.click
|
import androidx.test.espresso.action.ViewActions.click
|
||||||
import androidx.test.espresso.action.ViewActions.typeText
|
|
||||||
import androidx.test.espresso.assertion.ViewAssertions.doesNotExist
|
|
||||||
import androidx.test.espresso.contrib.RecyclerViewActions.actionOnItemAtPosition
|
import androidx.test.espresso.contrib.RecyclerViewActions.actionOnItemAtPosition
|
||||||
import androidx.test.espresso.matcher.ViewMatchers
|
import androidx.test.espresso.matcher.ViewMatchers
|
||||||
import androidx.test.espresso.matcher.ViewMatchers.withContentDescription
|
import androidx.test.espresso.matcher.ViewMatchers.withContentDescription
|
||||||
@ -41,6 +44,10 @@ import org.kiwix.kiwixmobile.Findable.StringId.TextId
|
|||||||
import org.kiwix.kiwixmobile.Findable.Text
|
import org.kiwix.kiwixmobile.Findable.Text
|
||||||
import org.kiwix.kiwixmobile.Findable.ViewId
|
import org.kiwix.kiwixmobile.Findable.ViewId
|
||||||
import org.kiwix.kiwixmobile.core.R
|
import org.kiwix.kiwixmobile.core.R
|
||||||
|
import org.kiwix.kiwixmobile.core.main.ADD_NOTE_TEXT_FILED_TESTING_TAG
|
||||||
|
import org.kiwix.kiwixmobile.core.main.DELETE_MENU_BUTTON_TESTING_TAG
|
||||||
|
import org.kiwix.kiwixmobile.core.main.SAVE_MENU_BUTTON_TESTING_TAG
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.components.TOOLBAR_TITLE_TESTING_TAG
|
||||||
import org.kiwix.kiwixmobile.testutils.TestUtils
|
import org.kiwix.kiwixmobile.testutils.TestUtils
|
||||||
import org.kiwix.kiwixmobile.testutils.TestUtils.testFlakyView
|
import org.kiwix.kiwixmobile.testutils.TestUtils.testFlakyView
|
||||||
import org.kiwix.kiwixmobile.utils.StandardActions.openDrawer
|
import org.kiwix.kiwixmobile.utils.StandardActions.openDrawer
|
||||||
@ -49,7 +56,6 @@ fun note(func: NoteRobot.() -> Unit) = NoteRobot().apply(func)
|
|||||||
|
|
||||||
class NoteRobot : BaseRobot() {
|
class NoteRobot : BaseRobot() {
|
||||||
private val noteText = "Test Note"
|
private val noteText = "Test Note"
|
||||||
private val editTextId = R.id.add_note_edit_text
|
|
||||||
|
|
||||||
fun assertToolbarExist() {
|
fun assertToolbarExist() {
|
||||||
isVisible(ViewId(R.id.toolbar))
|
isVisible(ViewId(R.id.toolbar))
|
||||||
@ -71,19 +77,39 @@ class NoteRobot : BaseRobot() {
|
|||||||
clickOn(TextId(R.string.take_notes))
|
clickOn(TextId(R.string.take_notes))
|
||||||
}
|
}
|
||||||
|
|
||||||
fun assertNoteDialogDisplayed() {
|
fun assertNoteDialogDisplayed(composeTestRule: ComposeContentTestRule) {
|
||||||
pauseForBetterTestPerformance()
|
testFlakyView({
|
||||||
isVisible(TextId(R.string.note))
|
composeTestRule.waitForIdle()
|
||||||
|
composeTestRule.onNodeWithTag(TOOLBAR_TITLE_TESTING_TAG)
|
||||||
|
.assertTextEquals(context.getString(R.string.note))
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fun writeDemoNote() {
|
fun writeDemoNote(composeTestRule: ComposeContentTestRule) {
|
||||||
onView(withId(editTextId)).perform(clearText(), typeText(noteText))
|
testFlakyView({
|
||||||
closeSoftKeyboard()
|
composeTestRule.waitForIdle()
|
||||||
|
// Click on the TextField to focus it
|
||||||
|
composeTestRule.onNodeWithTag(ADD_NOTE_TEXT_FILED_TESTING_TAG)
|
||||||
|
.assertExists("TextField not found in dialog")
|
||||||
|
.performClick()
|
||||||
|
.performTextReplacement(noteText)
|
||||||
|
|
||||||
|
composeTestRule.waitForIdle()
|
||||||
|
|
||||||
|
composeTestRule.onNodeWithTag(ADD_NOTE_TEXT_FILED_TESTING_TAG)
|
||||||
|
.assertTextContains(noteText, substring = true)
|
||||||
|
|
||||||
|
// Close the keyboard after typing
|
||||||
|
closeSoftKeyboard()
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fun saveNote() {
|
fun saveNote(composeTestRule: ComposeContentTestRule) {
|
||||||
pauseForBetterTestPerformance()
|
testFlakyView({
|
||||||
clickOn(ViewId(R.id.save_note))
|
composeTestRule.waitForIdle()
|
||||||
|
composeTestRule.onNodeWithTag(SAVE_MENU_BUTTON_TESTING_TAG)
|
||||||
|
.performClick()
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fun openNoteFragment() {
|
fun openNoteFragment() {
|
||||||
@ -106,19 +132,30 @@ class NoteRobot : BaseRobot() {
|
|||||||
testFlakyView({ clickOn(Text("OPEN NOTE")) })
|
testFlakyView({ clickOn(Text("OPEN NOTE")) })
|
||||||
}
|
}
|
||||||
|
|
||||||
fun assertNoteSaved() {
|
fun assertNoteSaved(composeTestRule: ComposeContentTestRule) {
|
||||||
// This is flaky since it is shown in a dialog and sometimes
|
// This is flaky since it is shown in a dialog and sometimes
|
||||||
// UIDevice does not found the view immediately due to rendering process.
|
// UIDevice does not found the view immediately due to rendering process.
|
||||||
testFlakyView({ isVisible(Text(noteText)) })
|
testFlakyView({
|
||||||
|
composeTestRule.waitForIdle()
|
||||||
|
composeTestRule.onNodeWithTag(ADD_NOTE_TEXT_FILED_TESTING_TAG)
|
||||||
|
.assertTextEquals(noteText)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fun assertNotDoesNotExist() {
|
fun assertNotDoesNotExist(composeTestRule: ComposeContentTestRule) {
|
||||||
testFlakyView({ onView(withText(noteText)).check(doesNotExist()) })
|
testFlakyView({
|
||||||
|
composeTestRule.waitForIdle()
|
||||||
|
composeTestRule.onNodeWithTag(ADD_NOTE_TEXT_FILED_TESTING_TAG)
|
||||||
|
.assertTextContains("", ignoreCase = true)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fun clickOnDeleteIcon() {
|
fun clickOnDeleteIcon(composeTestRule: ComposeContentTestRule) {
|
||||||
pauseForBetterTestPerformance()
|
testFlakyView({
|
||||||
testFlakyView({ clickOn(ViewId(R.id.delete_note)) })
|
composeTestRule.waitForIdle()
|
||||||
|
composeTestRule.onNodeWithTag(DELETE_MENU_BUTTON_TESTING_TAG)
|
||||||
|
.performClick()
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fun clickOnTrashIcon() {
|
fun clickOnTrashIcon() {
|
||||||
|
@ -41,7 +41,6 @@ import org.kiwix.kiwixmobile.core.compat.ResolveInfoFlagsCompat
|
|||||||
import org.kiwix.kiwixmobile.core.dao.NewBookDao
|
import org.kiwix.kiwixmobile.core.dao.NewBookDao
|
||||||
import org.kiwix.kiwixmobile.core.extensions.toast
|
import org.kiwix.kiwixmobile.core.extensions.toast
|
||||||
import org.kiwix.kiwixmobile.core.reader.ZimReaderContainer
|
import org.kiwix.kiwixmobile.core.reader.ZimReaderContainer
|
||||||
import org.kiwix.kiwixmobile.core.ui.theme.KiwixTheme
|
|
||||||
import org.kiwix.kiwixmobile.core.utils.CRASH_AND_FEEDBACK_EMAIL_ADDRESS
|
import org.kiwix.kiwixmobile.core.utils.CRASH_AND_FEEDBACK_EMAIL_ADDRESS
|
||||||
import org.kiwix.kiwixmobile.core.utils.LanguageUtils.Companion.getCurrentLocale
|
import org.kiwix.kiwixmobile.core.utils.LanguageUtils.Companion.getCurrentLocale
|
||||||
import org.kiwix.kiwixmobile.core.utils.files.FileLogger
|
import org.kiwix.kiwixmobile.core.utils.files.FileLogger
|
||||||
@ -92,15 +91,13 @@ open class ErrorActivity : BaseActivity() {
|
|||||||
checkBoxItems = remember {
|
checkBoxItems = remember {
|
||||||
getCrashCheckBoxItems().map { it.first to mutableStateOf(it.second) }
|
getCrashCheckBoxItems().map { it.first to mutableStateOf(it.second) }
|
||||||
}
|
}
|
||||||
KiwixTheme {
|
ErrorActivityScreen(
|
||||||
ErrorActivityScreen(
|
crashTitle,
|
||||||
crashTitle,
|
crashDescription,
|
||||||
crashDescription,
|
checkBoxItems,
|
||||||
checkBoxItems,
|
{ restartApp() },
|
||||||
{ restartApp() },
|
{ sendDetailsOnMail() }
|
||||||
{ sendDetailsOnMail() }
|
)
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -41,7 +41,6 @@ import androidx.compose.ui.Alignment
|
|||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.graphics.ImageBitmap
|
import androidx.compose.ui.graphics.ImageBitmap
|
||||||
import androidx.compose.ui.platform.LocalContext
|
import androidx.compose.ui.platform.LocalContext
|
||||||
import androidx.compose.ui.res.colorResource
|
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import androidx.compose.ui.text.style.TextAlign
|
import androidx.compose.ui.text.style.TextAlign
|
||||||
import com.tonyodev.fetch2.R.string
|
import com.tonyodev.fetch2.R.string
|
||||||
@ -49,6 +48,10 @@ import org.kiwix.kiwixmobile.core.R
|
|||||||
import org.kiwix.kiwixmobile.core.extensions.loadBitmapFromMipmap
|
import org.kiwix.kiwixmobile.core.extensions.loadBitmapFromMipmap
|
||||||
import org.kiwix.kiwixmobile.core.ui.components.CrashCheckBox
|
import org.kiwix.kiwixmobile.core.ui.components.CrashCheckBox
|
||||||
import org.kiwix.kiwixmobile.core.ui.components.KiwixButton
|
import org.kiwix.kiwixmobile.core.ui.components.KiwixButton
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.AlabasterWhite
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.ErrorActivityBackground
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.KiwixTheme
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.White
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.CRASH_IMAGE_SIZE
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.CRASH_IMAGE_SIZE
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.EIGHT_DP
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.EIGHT_DP
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.SEVENTEEN_DP
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.SEVENTEEN_DP
|
||||||
@ -64,63 +67,91 @@ fun ErrorActivityScreen(
|
|||||||
noThanksButtonClickListener: () -> Unit,
|
noThanksButtonClickListener: () -> Unit,
|
||||||
sendDetailsButtonClickListener: () -> Unit
|
sendDetailsButtonClickListener: () -> Unit
|
||||||
) {
|
) {
|
||||||
Column(
|
KiwixTheme {
|
||||||
modifier = Modifier
|
Column(
|
||||||
.fillMaxSize()
|
|
||||||
.background(colorResource(id = R.color.error_activity_background))
|
|
||||||
.systemBarsPadding()
|
|
||||||
.imePadding()
|
|
||||||
.padding(SIXTEEN_DP),
|
|
||||||
horizontalAlignment = Alignment.CenterHorizontally
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
text = stringResource(crashTitleStringId),
|
|
||||||
style = MaterialTheme.typography.headlineSmall,
|
|
||||||
color = colorResource(id = R.color.alabaster_white),
|
|
||||||
modifier = Modifier.padding(top = SIXTY_DP, start = EIGHT_DP, end = EIGHT_DP)
|
|
||||||
)
|
|
||||||
|
|
||||||
Image(
|
|
||||||
bitmap = ImageBitmap.loadBitmapFromMipmap(LocalContext.current, R.mipmap.ic_launcher),
|
|
||||||
contentDescription = stringResource(id = string.app_name),
|
|
||||||
modifier = Modifier
|
modifier = Modifier
|
||||||
.height(CRASH_IMAGE_SIZE)
|
.fillMaxSize()
|
||||||
.padding(top = TWELVE_DP, start = EIGHT_DP, end = EIGHT_DP)
|
.background(ErrorActivityBackground)
|
||||||
)
|
.systemBarsPadding()
|
||||||
|
.imePadding()
|
||||||
Text(
|
.padding(SIXTEEN_DP),
|
||||||
text = stringResource(messageStringId),
|
horizontalAlignment = Alignment.CenterHorizontally
|
||||||
style = MaterialTheme.typography.bodyMedium,
|
|
||||||
textAlign = TextAlign.Center,
|
|
||||||
color = colorResource(id = R.color.white),
|
|
||||||
modifier = Modifier.padding(start = EIGHT_DP, top = EIGHT_DP, end = EIGHT_DP)
|
|
||||||
)
|
|
||||||
|
|
||||||
Column(modifier = Modifier.weight(1f).padding(top = SEVENTEEN_DP, bottom = EIGHT_DP)) {
|
|
||||||
LazyColumn(
|
|
||||||
modifier = Modifier.fillMaxWidth()
|
|
||||||
) {
|
|
||||||
itemsIndexed(checkBoxData) { _, item ->
|
|
||||||
CrashCheckBox(item.first to item.second)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Row(
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.navigationBarsPadding(),
|
|
||||||
horizontalArrangement = Arrangement.SpaceEvenly
|
|
||||||
) {
|
) {
|
||||||
KiwixButton(
|
CrashTitle(crashTitleStringId)
|
||||||
buttonTextId = R.string.crash_button_confirm,
|
AppIcon()
|
||||||
clickListener = { sendDetailsButtonClickListener.invoke() },
|
CrashMessage(messageStringId)
|
||||||
|
CrashCheckBoxList(
|
||||||
|
checkBoxData,
|
||||||
|
Modifier
|
||||||
|
.weight(1f)
|
||||||
|
.padding(top = SEVENTEEN_DP, bottom = EIGHT_DP)
|
||||||
)
|
)
|
||||||
|
|
||||||
KiwixButton(
|
// Buttons on the ErrorActivity.
|
||||||
clickListener = { noThanksButtonClickListener.invoke() },
|
Row(
|
||||||
buttonTextId = R.string.no_thanks
|
modifier = Modifier
|
||||||
)
|
.fillMaxWidth()
|
||||||
|
.navigationBarsPadding(),
|
||||||
|
horizontalArrangement = Arrangement.SpaceEvenly
|
||||||
|
) {
|
||||||
|
KiwixButton(
|
||||||
|
buttonTextId = R.string.crash_button_confirm,
|
||||||
|
clickListener = { sendDetailsButtonClickListener.invoke() },
|
||||||
|
)
|
||||||
|
|
||||||
|
KiwixButton(
|
||||||
|
clickListener = { noThanksButtonClickListener.invoke() },
|
||||||
|
buttonTextId = R.string.no_thanks
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun CrashTitle(
|
||||||
|
@StringRes titleId: Int
|
||||||
|
) {
|
||||||
|
Text(
|
||||||
|
text = stringResource(titleId),
|
||||||
|
style = MaterialTheme.typography.headlineSmall,
|
||||||
|
color = AlabasterWhite,
|
||||||
|
modifier = Modifier.padding(top = SIXTY_DP, start = EIGHT_DP, end = EIGHT_DP)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun AppIcon() {
|
||||||
|
Image(
|
||||||
|
bitmap = ImageBitmap.loadBitmapFromMipmap(LocalContext.current, R.mipmap.ic_launcher),
|
||||||
|
contentDescription = stringResource(id = string.app_name),
|
||||||
|
modifier = Modifier
|
||||||
|
.height(CRASH_IMAGE_SIZE)
|
||||||
|
.padding(top = TWELVE_DP, start = EIGHT_DP, end = EIGHT_DP)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun CrashMessage(
|
||||||
|
@StringRes messageId: Int
|
||||||
|
) {
|
||||||
|
Text(
|
||||||
|
text = stringResource(messageId),
|
||||||
|
style = MaterialTheme.typography.bodyMedium,
|
||||||
|
textAlign = TextAlign.Center,
|
||||||
|
color = White,
|
||||||
|
modifier = Modifier.padding(start = EIGHT_DP, top = EIGHT_DP, end = EIGHT_DP)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun CrashCheckBoxList(
|
||||||
|
checkBoxData: List<Pair<Int, MutableState<Boolean>>>,
|
||||||
|
modifier: Modifier
|
||||||
|
) {
|
||||||
|
LazyColumn(modifier = modifier) {
|
||||||
|
itemsIndexed(checkBoxData) { _, item ->
|
||||||
|
CrashCheckBox(item.first to item.second)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,45 @@
|
|||||||
|
/*
|
||||||
|
* Kiwix Android
|
||||||
|
* Copyright (c) 2025 Kiwix <android.kiwix.org>
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.kiwix.kiwixmobile.core.extensions
|
||||||
|
|
||||||
|
import androidx.compose.material3.SnackbarDuration
|
||||||
|
import androidx.compose.material3.SnackbarHostState
|
||||||
|
import androidx.compose.material3.SnackbarResult
|
||||||
|
import kotlinx.coroutines.CoroutineScope
|
||||||
|
import kotlinx.coroutines.launch
|
||||||
|
|
||||||
|
fun SnackbarHostState.snack(
|
||||||
|
message: String,
|
||||||
|
actionLabel: String? = null,
|
||||||
|
actionClick: (() -> Unit)? = null,
|
||||||
|
// Default duration is 4 seconds.
|
||||||
|
snackbarDuration: SnackbarDuration = SnackbarDuration.Short,
|
||||||
|
lifecycleScope: CoroutineScope
|
||||||
|
) {
|
||||||
|
lifecycleScope.launch {
|
||||||
|
val result = showSnackbar(
|
||||||
|
message = message,
|
||||||
|
actionLabel = actionLabel?.uppercase(),
|
||||||
|
duration = snackbarDuration
|
||||||
|
)
|
||||||
|
if (result == SnackbarResult.ActionPerformed) {
|
||||||
|
actionClick?.invoke()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -19,38 +19,43 @@ package org.kiwix.kiwixmobile.core.main
|
|||||||
|
|
||||||
import android.Manifest
|
import android.Manifest
|
||||||
import android.app.Dialog
|
import android.app.Dialog
|
||||||
import android.content.Context
|
|
||||||
import android.content.Intent
|
import android.content.Intent
|
||||||
import android.content.pm.PackageManager
|
import android.content.pm.PackageManager
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
import android.os.Bundle
|
import android.os.Bundle
|
||||||
import android.text.Editable
|
|
||||||
import android.view.LayoutInflater
|
import android.view.LayoutInflater
|
||||||
import android.view.MenuItem
|
|
||||||
import android.view.View
|
import android.view.View
|
||||||
import android.view.ViewGroup
|
import android.view.ViewGroup
|
||||||
import android.view.inputmethod.InputMethodManager
|
|
||||||
import android.widget.EditText
|
|
||||||
import android.widget.Toast
|
import android.widget.Toast
|
||||||
import androidx.activity.OnBackPressedCallback
|
import androidx.activity.OnBackPressedCallback
|
||||||
import androidx.appcompat.widget.Toolbar
|
import androidx.compose.material.icons.Icons
|
||||||
|
import androidx.compose.material.icons.filled.Delete
|
||||||
|
import androidx.compose.material.icons.filled.Share
|
||||||
|
import androidx.compose.material3.SnackbarHostState
|
||||||
|
import androidx.compose.runtime.mutableStateOf
|
||||||
|
import androidx.compose.runtime.remember
|
||||||
|
import androidx.compose.ui.platform.ComposeView
|
||||||
|
import androidx.compose.ui.text.TextRange
|
||||||
|
import androidx.compose.ui.text.input.TextFieldValue
|
||||||
import androidx.core.content.ContextCompat
|
import androidx.core.content.ContextCompat
|
||||||
import androidx.core.content.FileProvider
|
import androidx.core.content.FileProvider
|
||||||
import androidx.fragment.app.DialogFragment
|
import androidx.fragment.app.DialogFragment
|
||||||
|
import androidx.lifecycle.lifecycleScope
|
||||||
import org.kiwix.kiwixmobile.core.CoreApp.Companion.coreComponent
|
import org.kiwix.kiwixmobile.core.CoreApp.Companion.coreComponent
|
||||||
import org.kiwix.kiwixmobile.core.CoreApp.Companion.instance
|
import org.kiwix.kiwixmobile.core.CoreApp.Companion.instance
|
||||||
import org.kiwix.kiwixmobile.core.R
|
import org.kiwix.kiwixmobile.core.R
|
||||||
import org.kiwix.kiwixmobile.core.databinding.DialogAddNoteBinding
|
|
||||||
import org.kiwix.kiwixmobile.core.extensions.closeKeyboard
|
import org.kiwix.kiwixmobile.core.extensions.closeKeyboard
|
||||||
import org.kiwix.kiwixmobile.core.extensions.getToolbarNavigationIcon
|
|
||||||
import org.kiwix.kiwixmobile.core.extensions.setToolTipWithContentDescription
|
|
||||||
import org.kiwix.kiwixmobile.core.extensions.snack
|
import org.kiwix.kiwixmobile.core.extensions.snack
|
||||||
import org.kiwix.kiwixmobile.core.extensions.toast
|
import org.kiwix.kiwixmobile.core.extensions.toast
|
||||||
import org.kiwix.kiwixmobile.core.page.notes.adapter.NoteListItem
|
import org.kiwix.kiwixmobile.core.page.notes.adapter.NoteListItem
|
||||||
import org.kiwix.kiwixmobile.core.reader.ZimReaderContainer
|
import org.kiwix.kiwixmobile.core.reader.ZimReaderContainer
|
||||||
import org.kiwix.kiwixmobile.core.reader.ZimReaderSource
|
import org.kiwix.kiwixmobile.core.reader.ZimReaderSource
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.components.NavigationIcon
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.models.ActionMenuItem
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.models.IconItem
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.models.IconItem.Drawable
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.models.IconItem.Vector
|
||||||
import org.kiwix.kiwixmobile.core.utils.SharedPreferenceUtil
|
import org.kiwix.kiwixmobile.core.utils.SharedPreferenceUtil
|
||||||
import org.kiwix.kiwixmobile.core.utils.SimpleTextWatcher
|
|
||||||
import org.kiwix.kiwixmobile.core.utils.dialog.AlertDialogShower
|
import org.kiwix.kiwixmobile.core.utils.dialog.AlertDialogShower
|
||||||
import org.kiwix.kiwixmobile.core.utils.dialog.KiwixDialog
|
import org.kiwix.kiwixmobile.core.utils.dialog.KiwixDialog
|
||||||
import org.kiwix.kiwixmobile.core.utils.files.Log
|
import org.kiwix.kiwixmobile.core.utils.files.Log
|
||||||
@ -77,13 +82,14 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
private lateinit var zimFileUrl: String
|
private lateinit var zimFileUrl: String
|
||||||
private var articleTitle: String? = null
|
private var articleTitle: String? = null
|
||||||
|
|
||||||
|
private val menuItems = mutableStateOf(actionMenuItems())
|
||||||
|
private val noteText = mutableStateOf(TextFieldValue(""))
|
||||||
|
private lateinit var snackBarHostState: SnackbarHostState
|
||||||
|
|
||||||
// Corresponds to "ArticleUrl" of "{External Storage}/Kiwix/Notes/ZimFileName/ArticleUrl.txt"
|
// Corresponds to "ArticleUrl" of "{External Storage}/Kiwix/Notes/ZimFileName/ArticleUrl.txt"
|
||||||
private lateinit var articleNoteFileName: String
|
private lateinit var articleNoteFileName: String
|
||||||
private var noteFileExists = false
|
|
||||||
private var noteEdited = false
|
private var noteEdited = false
|
||||||
|
|
||||||
private var dialogNoteAddNoteBinding: DialogAddNoteBinding? = null
|
|
||||||
|
|
||||||
// Keeps track of state of the note (whether edited since last save)
|
// Keeps track of state of the note (whether edited since last save)
|
||||||
// Stores path to directory for the currently open zim's notes
|
// Stores path to directory for the currently open zim's notes
|
||||||
private var zimNotesDirectory: String? = null
|
private var zimNotesDirectory: String? = null
|
||||||
@ -100,16 +106,6 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
@Inject
|
@Inject
|
||||||
lateinit var mainRepositoryActions: MainRepositoryActions
|
lateinit var mainRepositoryActions: MainRepositoryActions
|
||||||
|
|
||||||
private val toolbar by lazy {
|
|
||||||
dialogNoteAddNoteBinding?.root?.findViewById<Toolbar>(R.id.toolbar)
|
|
||||||
}
|
|
||||||
|
|
||||||
private val saveItem by lazy { toolbar?.menu?.findItem(R.id.save_note) }
|
|
||||||
|
|
||||||
private val shareItem by lazy { toolbar?.menu?.findItem(R.id.share_note) }
|
|
||||||
|
|
||||||
private val deleteItem by lazy { toolbar?.menu?.findItem(R.id.delete_note) }
|
|
||||||
|
|
||||||
private var noteListItem: NoteListItem? = null
|
private var noteListItem: NoteListItem? = null
|
||||||
private var zimReaderSource: ZimReaderSource? = null
|
private var zimReaderSource: ZimReaderSource? = null
|
||||||
private var favicon: String? = null
|
private var favicon: String? = null
|
||||||
@ -174,10 +170,81 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
inflater: LayoutInflater,
|
inflater: LayoutInflater,
|
||||||
container: ViewGroup?,
|
container: ViewGroup?,
|
||||||
savedInstanceState: Bundle?
|
savedInstanceState: Bundle?
|
||||||
): View? {
|
): View? = ComposeView(requireContext()).apply {
|
||||||
super.onCreateView(inflater, container, savedInstanceState)
|
setContent {
|
||||||
dialogNoteAddNoteBinding = DialogAddNoteBinding.inflate(inflater, container, false)
|
snackBarHostState = remember { SnackbarHostState() }
|
||||||
return dialogNoteAddNoteBinding?.root
|
AddNoteDialogScreen(
|
||||||
|
articleTitle.toString(),
|
||||||
|
navigationIcon = {
|
||||||
|
NavigationIcon(
|
||||||
|
iconItem = IconItem.Drawable(R.drawable.ic_close_white_24dp),
|
||||||
|
onClick = {
|
||||||
|
exitAddNoteDialog()
|
||||||
|
closeKeyboard()
|
||||||
|
}
|
||||||
|
)
|
||||||
|
},
|
||||||
|
noteText = noteText.value,
|
||||||
|
actionMenuItems = menuItems.value,
|
||||||
|
onTextChange = { textInputFiled ->
|
||||||
|
enableSaveAndShareMenuButtonAndSetTextEdited(textInputFiled)
|
||||||
|
},
|
||||||
|
snackBarHostState = snackBarHostState
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun updateMenuItem(vararg contentDescription: Int, isEnabled: Boolean) {
|
||||||
|
menuItems.value = menuItems.value.map { item ->
|
||||||
|
if (contentDescription.contains(item.contentDescription)) {
|
||||||
|
item.copy(isEnabled = isEnabled)
|
||||||
|
} else {
|
||||||
|
item
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun actionMenuItems() = listOf(
|
||||||
|
ActionMenuItem(
|
||||||
|
Vector(Icons.Default.Delete),
|
||||||
|
R.string.delete,
|
||||||
|
{ deleteNote() },
|
||||||
|
isEnabled = false,
|
||||||
|
testingTag = DELETE_MENU_BUTTON_TESTING_TAG
|
||||||
|
),
|
||||||
|
ActionMenuItem(
|
||||||
|
Vector(Icons.Default.Share),
|
||||||
|
R.string.share,
|
||||||
|
{ shareNote() },
|
||||||
|
isEnabled = false,
|
||||||
|
testingTag = SHARE_MENU_BUTTON_TESTING_TAG
|
||||||
|
),
|
||||||
|
ActionMenuItem(
|
||||||
|
Drawable(R.drawable.ic_save),
|
||||||
|
R.string.save,
|
||||||
|
{ saveNote() },
|
||||||
|
isEnabled = false,
|
||||||
|
testingTag = SAVE_MENU_BUTTON_TESTING_TAG
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Updates the note text and enables the save/share menu buttons when the user edits the text.
|
||||||
|
*
|
||||||
|
* ⚠️ Jetpack Compose triggers `onTextChange` not only when the text changes
|
||||||
|
* but also when the cursor position moves due to recomposition or rendering updates.
|
||||||
|
* This function ensures that menu buttons are only enabled when the actual text is modified,
|
||||||
|
* preventing unnecessary state updates caused by cursor movement.
|
||||||
|
*
|
||||||
|
* @param textFieldValue The latest value of the TextField, including text and cursor position.
|
||||||
|
*/
|
||||||
|
private fun enableSaveAndShareMenuButtonAndSetTextEdited(textFieldValue: TextFieldValue) {
|
||||||
|
if (noteText.value.text != textFieldValue.text) {
|
||||||
|
noteEdited = true
|
||||||
|
enableSaveNoteMenuItem()
|
||||||
|
enableShareNoteMenuItem()
|
||||||
|
}
|
||||||
|
noteText.value = textFieldValue
|
||||||
}
|
}
|
||||||
|
|
||||||
private val zimNoteDirectoryName: String
|
private val zimNoteDirectoryName: String
|
||||||
@ -211,7 +278,7 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
|
|
||||||
// Add onBackPressedCallBack to respond to user pressing 'Back' button on navigation bar
|
// Add onBackPressedCallBack to respond to user pressing 'Back' button on navigation bar
|
||||||
override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
|
override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
|
||||||
val dialog = Dialog(requireContext(), theme)
|
val dialog = Dialog(requireContext())
|
||||||
requireActivity().onBackPressedDispatcher.addCallback(onBackPressedCallBack)
|
requireActivity().onBackPressedDispatcher.addCallback(onBackPressedCallBack)
|
||||||
return dialog
|
return dialog
|
||||||
}
|
}
|
||||||
@ -230,58 +297,28 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
// Closing unedited note dialog straightaway
|
// Closing unedited note dialog straightaway
|
||||||
dismissAddNoteDialog()
|
dismissAddNoteDialog()
|
||||||
}
|
}
|
||||||
if (dialogNoteAddNoteBinding?.addNoteEditText?.isFocused == true) {
|
|
||||||
dialogNoteAddNoteBinding?.addNoteEditText?.clearFocus()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun disableMenuItems() {
|
private fun disableMenuItems() {
|
||||||
if (toolbar?.menu != null) {
|
updateMenuItem(R.string.delete, R.string.share, R.string.save, isEnabled = false)
|
||||||
saveItem?.isEnabled = false
|
|
||||||
shareItem?.isEnabled = false
|
|
||||||
deleteItem?.isEnabled = false
|
|
||||||
saveItem?.icon?.alpha = DISABLE_ICON_ITEM_ALPHA
|
|
||||||
shareItem?.icon?.alpha = DISABLE_ICON_ITEM_ALPHA
|
|
||||||
deleteItem?.icon?.alpha = DISABLE_ICON_ITEM_ALPHA
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "Toolbar without inflated menu")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun disableSaveNoteMenuItem() {
|
private fun disableSaveNoteMenuItem() {
|
||||||
if (toolbar?.menu != null) {
|
updateMenuItem(R.string.save, isEnabled = false)
|
||||||
saveItem?.isEnabled = false
|
|
||||||
saveItem?.icon?.alpha = DISABLE_ICON_ITEM_ALPHA
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "Toolbar without inflated menu")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun enableSaveNoteMenuItem() {
|
private fun enableSaveNoteMenuItem() {
|
||||||
if (toolbar?.menu != null && isZimFileExist()) {
|
if (isZimFileExist()) {
|
||||||
saveItem?.isEnabled = true
|
updateMenuItem(R.string.save, isEnabled = true)
|
||||||
saveItem?.icon?.alpha = ENABLE_ICON_ITEM_ALPHA
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "Toolbar without inflated menu")
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun enableDeleteNoteMenuItem() {
|
private fun enableDeleteNoteMenuItem() {
|
||||||
if (toolbar?.menu != null) {
|
updateMenuItem(R.string.delete, isEnabled = true)
|
||||||
deleteItem?.isEnabled = true
|
|
||||||
deleteItem?.icon?.alpha = ENABLE_ICON_ITEM_ALPHA
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "Toolbar without inflated menu")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun enableShareNoteMenuItem() {
|
private fun enableShareNoteMenuItem() {
|
||||||
if (toolbar?.menu != null) {
|
updateMenuItem(R.string.share, isEnabled = true)
|
||||||
shareItem?.isEnabled = true
|
|
||||||
shareItem?.icon?.alpha = ENABLE_ICON_ITEM_ALPHA
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "Toolbar without inflated menu")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onViewCreated(
|
override fun onViewCreated(
|
||||||
@ -289,62 +326,13 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
savedInstanceState: Bundle?
|
savedInstanceState: Bundle?
|
||||||
) {
|
) {
|
||||||
super.onViewCreated(view, savedInstanceState)
|
super.onViewCreated(view, savedInstanceState)
|
||||||
toolbar?.apply {
|
|
||||||
setTitle(R.string.note)
|
|
||||||
setNavigationIcon(R.drawable.ic_close_white_24dp)
|
|
||||||
setNavigationOnClickListener {
|
|
||||||
exitAddNoteDialog()
|
|
||||||
closeKeyboard()
|
|
||||||
}
|
|
||||||
// set the navigation close button contentDescription
|
|
||||||
getToolbarNavigationIcon()?.setToolTipWithContentDescription(
|
|
||||||
getString(R.string.toolbar_back_button_content_description)
|
|
||||||
)
|
|
||||||
setOnMenuItemClickListener { item: MenuItem ->
|
|
||||||
when (item.itemId) {
|
|
||||||
R.id.share_note -> shareNote()
|
|
||||||
R.id.save_note -> saveNote(dialogNoteAddNoteBinding?.addNoteEditText?.text.toString())
|
|
||||||
R.id.delete_note -> deleteNote()
|
|
||||||
}
|
|
||||||
true
|
|
||||||
}
|
|
||||||
inflateMenu(R.menu.menu_add_note_dialog)
|
|
||||||
}
|
|
||||||
// 'Share' disabled for empty notes, 'Save' disabled for unedited notes
|
// 'Share' disabled for empty notes, 'Save' disabled for unedited notes
|
||||||
disableMenuItems()
|
disableMenuItems()
|
||||||
dialogNoteAddNoteBinding?.addNoteTextView?.text = articleTitle
|
|
||||||
|
|
||||||
// Show the previously saved note if it exists
|
// Show the previously saved note if it exists
|
||||||
displayNote()
|
displayNote()
|
||||||
dialogNoteAddNoteBinding?.addNoteEditText?.addTextChangedListener(
|
|
||||||
SimpleTextWatcher { _, _, _, _ ->
|
|
||||||
noteEdited = true
|
|
||||||
enableSaveNoteMenuItem()
|
|
||||||
enableShareNoteMenuItem()
|
|
||||||
}
|
|
||||||
)
|
|
||||||
if (!noteFileExists) {
|
|
||||||
// Prepare for input in case of empty/new note
|
|
||||||
dialogNoteAddNoteBinding?.addNoteEditText?.apply {
|
|
||||||
requestFocus()
|
|
||||||
showKeyboard(this)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Suppress("MagicNumber")
|
private fun saveNote() {
|
||||||
private fun showKeyboard(editText: EditText) {
|
|
||||||
val inputMethodManager =
|
|
||||||
requireContext().getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManager
|
|
||||||
editText.postDelayed(
|
|
||||||
{
|
|
||||||
inputMethodManager.showSoftInput(editText, InputMethodManager.SHOW_IMPLICIT)
|
|
||||||
},
|
|
||||||
100
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun saveNote(noteText: String) {
|
|
||||||
/* String content of the EditText, given by noteText, is saved into the text file given by:
|
/* String content of the EditText, given by noteText, is saved into the text file given by:
|
||||||
* "{External Storage}/Kiwix/Notes/ZimFileTitle/ArticleTitle.txt"
|
* "{External Storage}/Kiwix/Notes/ZimFileTitle/ArticleTitle.txt"
|
||||||
* */
|
* */
|
||||||
@ -375,7 +363,7 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
|
|
||||||
// Save note text-file code:
|
// Save note text-file code:
|
||||||
try {
|
try {
|
||||||
noteFile.writeText(noteText)
|
noteFile.writeText(noteText.value.text)
|
||||||
context.toast(R.string.note_save_successful, Toast.LENGTH_SHORT)
|
context.toast(R.string.note_save_successful, Toast.LENGTH_SHORT)
|
||||||
noteEdited = false // As no unsaved changes remain
|
noteEdited = false // As no unsaved changes remain
|
||||||
enableDeleteNoteMenuItem()
|
enableDeleteNoteMenuItem()
|
||||||
@ -433,15 +421,16 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
val noteFile =
|
val noteFile =
|
||||||
File(notesFolder.absolutePath, "$articleNoteFileName.txt")
|
File(notesFolder.absolutePath, "$articleNoteFileName.txt")
|
||||||
val noteDeleted = noteFile.delete()
|
val noteDeleted = noteFile.delete()
|
||||||
val noteText = dialogNoteAddNoteBinding?.addNoteEditText?.text.toString()
|
val editedNoteText = noteText.value.text
|
||||||
if (noteDeleted) {
|
if (noteDeleted) {
|
||||||
dialogNoteAddNoteBinding?.addNoteEditText?.text?.clear()
|
noteText.value = TextFieldValue("")
|
||||||
mainRepositoryActions.deleteNote(getNoteTitle())
|
mainRepositoryActions.deleteNote(getNoteTitle())
|
||||||
disableMenuItems()
|
disableMenuItems()
|
||||||
view?.snack(
|
snackBarHostState.snack(
|
||||||
stringId = R.string.note_delete_successful,
|
message = requireActivity().getString(R.string.note_delete_successful),
|
||||||
actionStringId = R.string.undo,
|
actionLabel = requireActivity().getString(R.string.undo),
|
||||||
actionClick = { restoreDeletedNote(noteText) }
|
actionClick = { restoreDeletedNote(editedNoteText) },
|
||||||
|
lifecycleScope = lifecycleScope
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
context.toast(R.string.note_delete_unsuccessful, Toast.LENGTH_LONG)
|
context.toast(R.string.note_delete_unsuccessful, Toast.LENGTH_LONG)
|
||||||
@ -449,7 +438,12 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun restoreDeletedNote(text: String) {
|
private fun restoreDeletedNote(text: String) {
|
||||||
dialogNoteAddNoteBinding?.addNoteEditText?.setText(text)
|
val restoreNoteTextFieldValue = TextFieldValue(
|
||||||
|
text = text,
|
||||||
|
// Moves cursor to end
|
||||||
|
selection = TextRange(text.length)
|
||||||
|
)
|
||||||
|
enableSaveAndShareMenuButtonAndSetTextEdited(restoreNoteTextFieldValue)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* String content of the note text file given at:
|
/* String content of the note text file given at:
|
||||||
@ -466,15 +460,8 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun readNoteFromFile(noteFile: File) {
|
private fun readNoteFromFile(noteFile: File) {
|
||||||
noteFileExists = true
|
val noteFileText = noteFile.readText()
|
||||||
val contents = noteFile.readText()
|
noteText.value = TextFieldValue(noteFileText, selection = TextRange(noteFileText.length))
|
||||||
dialogNoteAddNoteBinding?.addNoteEditText?.apply {
|
|
||||||
setText(contents) // Display the note content
|
|
||||||
text?.takeIf(Editable::isNotEmpty)?.let { text ->
|
|
||||||
val selection = text.length - 1
|
|
||||||
setSelection(selection.coerceAtLeast(0))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
enableShareNoteMenuItem() // As note content exists which can be shared
|
enableShareNoteMenuItem() // As note content exists which can be shared
|
||||||
enableDeleteNoteMenuItem()
|
enableDeleteNoteMenuItem()
|
||||||
if (!isZimFileExist()) {
|
if (!isZimFileExist()) {
|
||||||
@ -490,7 +477,7 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
* */
|
* */
|
||||||
if (noteEdited && isZimFileExist()) {
|
if (noteEdited && isZimFileExist()) {
|
||||||
// Save edited note before sharing the text file
|
// Save edited note before sharing the text file
|
||||||
saveNote(dialogNoteAddNoteBinding?.addNoteEditText?.text.toString())
|
saveNote()
|
||||||
}
|
}
|
||||||
val noteFile = File("$zimNotesDirectory$articleNoteFileName.txt")
|
val noteFile = File("$zimNotesDirectory$articleNoteFileName.txt")
|
||||||
if (noteFile.exists()) {
|
if (noteFile.exists()) {
|
||||||
@ -526,7 +513,6 @@ class AddNoteDialog : DialogFragment() {
|
|||||||
override fun onDestroyView() {
|
override fun onDestroyView() {
|
||||||
super.onDestroyView()
|
super.onDestroyView()
|
||||||
mainRepositoryActions.dispose()
|
mainRepositoryActions.dispose()
|
||||||
dialogNoteAddNoteBinding = null
|
|
||||||
onBackPressedCallBack.remove()
|
onBackPressedCallBack.remove()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,139 @@
|
|||||||
|
/*
|
||||||
|
* Kiwix Android
|
||||||
|
* Copyright (c) 2025 Kiwix <android.kiwix.org>
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.kiwix.kiwixmobile.core.main
|
||||||
|
|
||||||
|
import androidx.compose.foundation.background
|
||||||
|
import androidx.compose.foundation.layout.Column
|
||||||
|
import androidx.compose.foundation.layout.fillMaxWidth
|
||||||
|
import androidx.compose.foundation.layout.heightIn
|
||||||
|
import androidx.compose.foundation.layout.imePadding
|
||||||
|
import androidx.compose.foundation.layout.padding
|
||||||
|
import androidx.compose.foundation.text.KeyboardOptions
|
||||||
|
import androidx.compose.material3.HorizontalDivider
|
||||||
|
import androidx.compose.material3.MaterialTheme
|
||||||
|
import androidx.compose.material3.Scaffold
|
||||||
|
import androidx.compose.material3.SnackbarHostState
|
||||||
|
import androidx.compose.material3.Text
|
||||||
|
import androidx.compose.material3.TextField
|
||||||
|
import androidx.compose.material3.TextFieldDefaults
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.graphics.Color
|
||||||
|
import androidx.compose.ui.graphics.RectangleShape
|
||||||
|
import androidx.compose.ui.platform.testTag
|
||||||
|
import androidx.compose.ui.res.stringResource
|
||||||
|
import androidx.compose.ui.text.input.KeyboardCapitalization
|
||||||
|
import androidx.compose.ui.text.input.KeyboardType
|
||||||
|
import androidx.compose.ui.text.input.TextFieldValue
|
||||||
|
import org.kiwix.kiwixmobile.core.R
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.components.KiwixAppBar
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.components.KiwixSnackbarHost
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.models.ActionMenuItem
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.KiwixDialogTheme
|
||||||
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.FIVE_DP
|
||||||
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.FOUR_DP
|
||||||
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.MINIMUM_HEIGHT_OF_NOTE_TEXT_FILED
|
||||||
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.TEN_DP
|
||||||
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.TWENTY_DP
|
||||||
|
|
||||||
|
const val ADD_NOTE_TEXT_FILED_TESTING_TAG = "addNoteTextFiledTestingTag"
|
||||||
|
const val SAVE_MENU_BUTTON_TESTING_TAG = "saveMenuButtonTestingTag"
|
||||||
|
const val SHARE_MENU_BUTTON_TESTING_TAG = "shareMenuButtonTestingTag"
|
||||||
|
const val DELETE_MENU_BUTTON_TESTING_TAG = "deleteMenuButtonTestingTag"
|
||||||
|
|
||||||
|
@Suppress("ComposableLambdaParameterNaming")
|
||||||
|
@Composable
|
||||||
|
fun AddNoteDialogScreen(
|
||||||
|
articleTitle: String,
|
||||||
|
noteText: TextFieldValue,
|
||||||
|
actionMenuItems: List<ActionMenuItem>,
|
||||||
|
onTextChange: (TextFieldValue) -> Unit,
|
||||||
|
snackBarHostState: SnackbarHostState,
|
||||||
|
navigationIcon: @Composable () -> Unit
|
||||||
|
) {
|
||||||
|
KiwixDialogTheme {
|
||||||
|
Scaffold(
|
||||||
|
snackbarHost = { KiwixSnackbarHost(snackbarHostState = snackBarHostState) }
|
||||||
|
) { paddingValues ->
|
||||||
|
Column(
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.background(Color.Transparent)
|
||||||
|
.imePadding()
|
||||||
|
.padding(paddingValues),
|
||||||
|
) {
|
||||||
|
KiwixAppBar(R.string.note, navigationIcon, actionMenuItems)
|
||||||
|
ArticleTitleText(articleTitle)
|
||||||
|
HorizontalDivider(
|
||||||
|
modifier = Modifier.padding(vertical = FIVE_DP),
|
||||||
|
color = MaterialTheme.colorScheme.onSurface
|
||||||
|
)
|
||||||
|
NoteTextField(
|
||||||
|
noteText = noteText,
|
||||||
|
onTextChange = onTextChange
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun ArticleTitleText(articleTitle: String) {
|
||||||
|
Text(
|
||||||
|
text = articleTitle,
|
||||||
|
maxLines = 3,
|
||||||
|
style = MaterialTheme.typography.bodyLarge,
|
||||||
|
modifier = Modifier.padding(top = TEN_DP, start = TWENTY_DP, end = TWENTY_DP)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun NoteTextField(
|
||||||
|
noteText: TextFieldValue,
|
||||||
|
onTextChange: (TextFieldValue) -> Unit
|
||||||
|
) {
|
||||||
|
TextField(
|
||||||
|
value = noteText,
|
||||||
|
onValueChange = { onTextChange(it) },
|
||||||
|
maxLines = 6,
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.heightIn(min = MINIMUM_HEIGHT_OF_NOTE_TEXT_FILED)
|
||||||
|
.padding(bottom = TEN_DP)
|
||||||
|
.padding(horizontal = FOUR_DP)
|
||||||
|
.testTag(ADD_NOTE_TEXT_FILED_TESTING_TAG),
|
||||||
|
placeholder = { Text(text = stringResource(R.string.note)) },
|
||||||
|
singleLine = false,
|
||||||
|
shape = RectangleShape,
|
||||||
|
keyboardOptions = KeyboardOptions(
|
||||||
|
autoCorrectEnabled = true,
|
||||||
|
capitalization = KeyboardCapitalization.Sentences,
|
||||||
|
keyboardType = KeyboardType.Text
|
||||||
|
),
|
||||||
|
textStyle = MaterialTheme.typography.bodyLarge,
|
||||||
|
colors = TextFieldDefaults.colors(
|
||||||
|
focusedContainerColor = Color.Transparent,
|
||||||
|
unfocusedContainerColor = Color.Transparent,
|
||||||
|
disabledContainerColor = Color.Transparent,
|
||||||
|
errorContainerColor = Color.Transparent,
|
||||||
|
focusedIndicatorColor = Color.Transparent,
|
||||||
|
unfocusedIndicatorColor = Color.Transparent
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
@ -31,12 +31,13 @@ import org.kiwix.kiwixmobile.core.utils.files.FileUtils.isFileDescriptorCanOpenW
|
|||||||
import org.kiwix.libzim.Archive
|
import org.kiwix.libzim.Archive
|
||||||
import org.kiwix.libzim.FdInput
|
import org.kiwix.libzim.FdInput
|
||||||
import java.io.File
|
import java.io.File
|
||||||
|
import java.io.Serializable
|
||||||
|
|
||||||
class ZimReaderSource(
|
class ZimReaderSource(
|
||||||
val file: File? = null,
|
val file: File? = null,
|
||||||
val uri: Uri? = null,
|
val uri: Uri? = null,
|
||||||
val assetFileDescriptorList: List<AssetFileDescriptor>? = null
|
val assetFileDescriptorList: List<AssetFileDescriptor>? = null
|
||||||
) {
|
) : Serializable {
|
||||||
constructor(uri: Uri) : this(
|
constructor(uri: Uri) : this(
|
||||||
uri = uri,
|
uri = uri,
|
||||||
assetFileDescriptorList = getAssetFileDescriptorFromUri(CoreApp.instance, uri)
|
assetFileDescriptorList = getAssetFileDescriptorFromUri(CoreApp.instance, uri)
|
||||||
|
@ -29,9 +29,10 @@ import androidx.compose.runtime.Composable
|
|||||||
import androidx.compose.runtime.MutableState
|
import androidx.compose.runtime.MutableState
|
||||||
import androidx.compose.ui.Alignment
|
import androidx.compose.ui.Alignment
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.res.colorResource
|
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import org.kiwix.kiwixmobile.core.R
|
import org.kiwix.kiwixmobile.core.ui.theme.DenimBlue200
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.ErrorActivityBackground
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.White
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.CRASH_CHECKBOX_START_PADDING
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.CRASH_CHECKBOX_START_PADDING
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.CRASH_CHECKBOX_TOP_PADDING
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.CRASH_CHECKBOX_TOP_PADDING
|
||||||
|
|
||||||
@ -47,15 +48,15 @@ fun CrashCheckBox(checkBoxItem: Pair<Int, MutableState<Boolean>>) {
|
|||||||
checked = checkBoxItem.second.value,
|
checked = checkBoxItem.second.value,
|
||||||
onCheckedChange = { checkBoxItem.second.value = it },
|
onCheckedChange = { checkBoxItem.second.value = it },
|
||||||
colors = CheckboxDefaults.colors(
|
colors = CheckboxDefaults.colors(
|
||||||
checkedColor = colorResource(id = R.color.denim_blue200),
|
checkedColor = DenimBlue200,
|
||||||
checkmarkColor = colorResource(id = R.color.error_activity_background),
|
checkmarkColor = ErrorActivityBackground,
|
||||||
uncheckedColor = colorResource(R.color.denim_blue200)
|
uncheckedColor = DenimBlue200
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
Text(
|
Text(
|
||||||
style = MaterialTheme.typography.bodyMedium,
|
style = MaterialTheme.typography.bodyMedium,
|
||||||
text = stringResource(id = checkBoxItem.first),
|
text = stringResource(id = checkBoxItem.first),
|
||||||
color = colorResource(id = R.color.white),
|
color = White,
|
||||||
modifier = Modifier.padding(start = CRASH_CHECKBOX_TOP_PADDING)
|
modifier = Modifier.padding(start = CRASH_CHECKBOX_TOP_PADDING)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,122 @@
|
|||||||
|
/*
|
||||||
|
* Kiwix Android
|
||||||
|
* Copyright (c) 2025 Kiwix <android.kiwix.org>
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.kiwix.kiwixmobile.core.ui.components
|
||||||
|
|
||||||
|
import androidx.annotation.StringRes
|
||||||
|
import androidx.compose.foundation.background
|
||||||
|
import androidx.compose.foundation.isSystemInDarkTheme
|
||||||
|
import androidx.compose.foundation.layout.Row
|
||||||
|
import androidx.compose.foundation.layout.Spacer
|
||||||
|
import androidx.compose.foundation.layout.fillMaxWidth
|
||||||
|
import androidx.compose.foundation.layout.height
|
||||||
|
import androidx.compose.foundation.layout.padding
|
||||||
|
import androidx.compose.material3.Icon
|
||||||
|
import androidx.compose.material3.IconButton
|
||||||
|
import androidx.compose.material3.MaterialTheme
|
||||||
|
import androidx.compose.material3.Text
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.ui.Alignment
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.graphics.Color
|
||||||
|
import androidx.compose.ui.graphics.vector.rememberVectorPainter
|
||||||
|
import androidx.compose.ui.platform.testTag
|
||||||
|
import androidx.compose.ui.res.painterResource
|
||||||
|
import androidx.compose.ui.res.stringResource
|
||||||
|
import androidx.compose.ui.text.font.FontWeight.Companion.SemiBold
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.models.ActionMenuItem
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.models.IconItem
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.Black
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.KiwixTheme
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.MineShaftGray350
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.White
|
||||||
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.KIWIX_APP_BAR_HEIGHT
|
||||||
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.SIXTEEN_DP
|
||||||
|
|
||||||
|
const val TOOLBAR_TITLE_TESTING_TAG = "toolbarTitle"
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun KiwixAppBar(
|
||||||
|
@StringRes titleId: Int,
|
||||||
|
navigationIcon: @Composable () -> Unit,
|
||||||
|
actionMenuItems: List<ActionMenuItem> = emptyList(),
|
||||||
|
// Optional search bar, used in fragments that require it
|
||||||
|
searchBar: (@Composable () -> Unit)? = null
|
||||||
|
) {
|
||||||
|
KiwixTheme {
|
||||||
|
Row(
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.height(KIWIX_APP_BAR_HEIGHT)
|
||||||
|
.background(color = Black),
|
||||||
|
verticalAlignment = Alignment.CenterVertically
|
||||||
|
) {
|
||||||
|
navigationIcon()
|
||||||
|
searchBar?.let {
|
||||||
|
// Display the search bar when provided
|
||||||
|
it()
|
||||||
|
} ?: run {
|
||||||
|
// Otherwise, show the title
|
||||||
|
AppBarTitle(titleId)
|
||||||
|
}
|
||||||
|
Spacer(Modifier.weight(1f))
|
||||||
|
ActionMenu(actionMenuItems)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun AppBarTitle(
|
||||||
|
@StringRes titleId: Int
|
||||||
|
) {
|
||||||
|
val appBarTitleColor = if (isSystemInDarkTheme()) {
|
||||||
|
MineShaftGray350
|
||||||
|
} else {
|
||||||
|
White
|
||||||
|
}
|
||||||
|
Text(
|
||||||
|
text = stringResource(titleId),
|
||||||
|
color = appBarTitleColor,
|
||||||
|
style = MaterialTheme.typography.titleLarge.copy(fontWeight = SemiBold),
|
||||||
|
modifier = Modifier
|
||||||
|
.padding(horizontal = SIXTEEN_DP)
|
||||||
|
.testTag(TOOLBAR_TITLE_TESTING_TAG)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun ActionMenu(actionMenuItems: List<ActionMenuItem>) {
|
||||||
|
Row {
|
||||||
|
actionMenuItems.forEach { menuItem ->
|
||||||
|
IconButton(
|
||||||
|
enabled = menuItem.isEnabled,
|
||||||
|
onClick = menuItem.onClick,
|
||||||
|
modifier = Modifier.testTag(menuItem.testingTag)
|
||||||
|
) {
|
||||||
|
Icon(
|
||||||
|
painter = when (val icon = menuItem.icon) {
|
||||||
|
is IconItem.Vector -> rememberVectorPainter(icon.imageVector)
|
||||||
|
is IconItem.Drawable -> painterResource(icon.drawableRes)
|
||||||
|
},
|
||||||
|
contentDescription = stringResource(menuItem.contentDescription),
|
||||||
|
tint = if (menuItem.isEnabled) menuItem.iconTint else Color.Gray
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -26,10 +26,10 @@ import androidx.compose.material3.MaterialTheme
|
|||||||
import androidx.compose.material3.Text
|
import androidx.compose.material3.Text
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.graphics.Color
|
|
||||||
import androidx.compose.ui.res.colorResource
|
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import org.kiwix.kiwixmobile.core.R
|
import androidx.compose.ui.text.font.FontWeight.Companion.SemiBold
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.DenimBlue800
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.White
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.BUTTON_DEFAULT_ELEVATION
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.BUTTON_DEFAULT_ELEVATION
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.BUTTON_DEFAULT_PADDING
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.BUTTON_DEFAULT_PADDING
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.BUTTON_PRESSED_ELEVATION
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.BUTTON_PRESSED_ELEVATION
|
||||||
@ -46,8 +46,8 @@ fun KiwixButton(
|
|||||||
Button(
|
Button(
|
||||||
onClick = { clickListener.invoke() },
|
onClick = { clickListener.invoke() },
|
||||||
colors = ButtonDefaults.buttonColors(
|
colors = ButtonDefaults.buttonColors(
|
||||||
containerColor = colorResource(id = R.color.denim_blue800),
|
containerColor = DenimBlue800,
|
||||||
contentColor = Color.White
|
contentColor = White
|
||||||
),
|
),
|
||||||
modifier = Modifier.padding(BUTTON_DEFAULT_PADDING),
|
modifier = Modifier.padding(BUTTON_DEFAULT_PADDING),
|
||||||
shape = MaterialTheme.shapes.extraSmall,
|
shape = MaterialTheme.shapes.extraSmall,
|
||||||
@ -58,7 +58,8 @@ fun KiwixButton(
|
|||||||
) {
|
) {
|
||||||
Text(
|
Text(
|
||||||
text = stringResource(id = buttonTextId).uppercase(),
|
text = stringResource(id = buttonTextId).uppercase(),
|
||||||
letterSpacing = DEFAULT_LETTER_SPACING
|
letterSpacing = DEFAULT_LETTER_SPACING,
|
||||||
|
style = MaterialTheme.typography.labelLarge.copy(fontWeight = SemiBold)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,51 @@
|
|||||||
|
/*
|
||||||
|
* Kiwix Android
|
||||||
|
* Copyright (c) 2025 Kiwix <android.kiwix.org>
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.kiwix.kiwixmobile.core.ui.components
|
||||||
|
|
||||||
|
import androidx.compose.foundation.isSystemInDarkTheme
|
||||||
|
import androidx.compose.material3.MaterialTheme
|
||||||
|
import androidx.compose.material3.Snackbar
|
||||||
|
import androidx.compose.material3.SnackbarHost
|
||||||
|
import androidx.compose.material3.SnackbarHostState
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.DenimBlue400
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A custom SnackbarHost for displaying snackbars with theme-aware action button colors.
|
||||||
|
*
|
||||||
|
* This function ensures that the action button color follows the app's theme:
|
||||||
|
* - In **light mode**, the action button color is `DenimBlue400`.
|
||||||
|
* - In **dark mode**, the action button color is `surface`, similar to the XML-based styling.
|
||||||
|
*
|
||||||
|
* @param snackbarHostState The state that controls the Snackbar display.
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
fun KiwixSnackbarHost(snackbarHostState: SnackbarHostState) {
|
||||||
|
val actionColor = if (isSystemInDarkTheme()) {
|
||||||
|
MaterialTheme.colorScheme.surface
|
||||||
|
} else {
|
||||||
|
DenimBlue400
|
||||||
|
}
|
||||||
|
SnackbarHost(hostState = snackbarHostState) { snackbarData ->
|
||||||
|
Snackbar(
|
||||||
|
snackbarData = snackbarData,
|
||||||
|
actionColor = actionColor
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,63 @@
|
|||||||
|
/*
|
||||||
|
* Kiwix Android
|
||||||
|
* Copyright (c) 2025 Kiwix <android.kiwix.org>
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.kiwix.kiwixmobile.core.ui.components
|
||||||
|
|
||||||
|
import androidx.annotation.StringRes
|
||||||
|
import androidx.compose.material.icons.Icons
|
||||||
|
import androidx.compose.material.icons.automirrored.filled.ArrowBack
|
||||||
|
import androidx.compose.material3.Icon
|
||||||
|
import androidx.compose.material3.IconButton
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.ui.graphics.Color
|
||||||
|
import androidx.compose.ui.graphics.vector.rememberVectorPainter
|
||||||
|
import androidx.compose.ui.res.painterResource
|
||||||
|
import androidx.compose.ui.res.stringResource
|
||||||
|
import org.kiwix.kiwixmobile.core.R
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.models.IconItem
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.White
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A composable function that renders a navigation icon, which can be either a vector
|
||||||
|
* or drawable image.
|
||||||
|
*
|
||||||
|
* @param iconItem The icon to be displayed. It can be either a vector (`IconItem.Vector`)
|
||||||
|
* or a drawable (`IconItem.Drawable`).
|
||||||
|
* @param onClick Callback function triggered when the icon is clicked.
|
||||||
|
* @param contentDescription A string resource ID for accessibility purposes, describing
|
||||||
|
* the icon's function.
|
||||||
|
* @param iconTint The color used to tint the icon. Default is white.
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
fun NavigationIcon(
|
||||||
|
iconItem: IconItem = IconItem.Vector(Icons.AutoMirrored.Filled.ArrowBack),
|
||||||
|
onClick: () -> Unit,
|
||||||
|
@StringRes contentDescription: Int = R.string.toolbar_back_button_content_description,
|
||||||
|
iconTint: Color = White
|
||||||
|
) {
|
||||||
|
IconButton(onClick = onClick) {
|
||||||
|
Icon(
|
||||||
|
painter = when (val icon = iconItem) {
|
||||||
|
is IconItem.Vector -> rememberVectorPainter(icon.imageVector)
|
||||||
|
is IconItem.Drawable -> painterResource(icon.drawableRes)
|
||||||
|
},
|
||||||
|
contentDescription = stringResource(contentDescription),
|
||||||
|
tint = iconTint
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,32 @@
|
|||||||
|
/*
|
||||||
|
* Kiwix Android
|
||||||
|
* Copyright (c) 2025 Kiwix <android.kiwix.org>
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.kiwix.kiwixmobile.core.ui.models
|
||||||
|
|
||||||
|
import androidx.annotation.StringRes
|
||||||
|
import androidx.compose.ui.graphics.Color
|
||||||
|
import org.kiwix.kiwixmobile.core.ui.theme.White
|
||||||
|
|
||||||
|
data class ActionMenuItem(
|
||||||
|
val icon: IconItem,
|
||||||
|
@StringRes val contentDescription: Int,
|
||||||
|
val onClick: () -> Unit,
|
||||||
|
val iconTint: Color = White,
|
||||||
|
val isEnabled: Boolean = true,
|
||||||
|
val testingTag: String
|
||||||
|
)
|
@ -0,0 +1,29 @@
|
|||||||
|
/*
|
||||||
|
* Kiwix Android
|
||||||
|
* Copyright (c) 2025 Kiwix <android.kiwix.org>
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.kiwix.kiwixmobile.core.ui.models
|
||||||
|
|
||||||
|
import androidx.annotation.DrawableRes
|
||||||
|
import androidx.compose.ui.graphics.vector.ImageVector
|
||||||
|
|
||||||
|
sealed class IconItem {
|
||||||
|
data class Vector(val imageVector: ImageVector) : IconItem()
|
||||||
|
data class Drawable(
|
||||||
|
@DrawableRes val drawableRes: Int
|
||||||
|
) : IconItem()
|
||||||
|
}
|
@ -26,6 +26,7 @@ val MineShaftGray350 = Color(0xFFD6D6D6)
|
|||||||
val MineShaftGray500 = Color(0xFF9E9E9E)
|
val MineShaftGray500 = Color(0xFF9E9E9E)
|
||||||
val ScorpionGray = Color(0xFF5A5A5A)
|
val ScorpionGray = Color(0xFF5A5A5A)
|
||||||
val MineShaftGray600 = Color(0xFF737373)
|
val MineShaftGray600 = Color(0xFF737373)
|
||||||
|
val MineShaftGray700 = Color(0xFF424242)
|
||||||
val MineShaftGray850 = Color(0xFF303030)
|
val MineShaftGray850 = Color(0xFF303030)
|
||||||
val MineShaftGray900 = Color(0xFF212121)
|
val MineShaftGray900 = Color(0xFF212121)
|
||||||
val Black = Color(0xFF000000)
|
val Black = Color(0xFF000000)
|
||||||
|
@ -34,7 +34,8 @@ private val DarkColorScheme = darkColorScheme(
|
|||||||
onSecondary = MineShaftGray350,
|
onSecondary = MineShaftGray350,
|
||||||
onBackground = White,
|
onBackground = White,
|
||||||
onSurface = White,
|
onSurface = White,
|
||||||
onError = White
|
onError = White,
|
||||||
|
onTertiary = MineShaftGray500
|
||||||
)
|
)
|
||||||
|
|
||||||
private val LightColorScheme = lightColorScheme(
|
private val LightColorScheme = lightColorScheme(
|
||||||
@ -47,7 +48,8 @@ private val LightColorScheme = lightColorScheme(
|
|||||||
onSecondary = ScorpionGray,
|
onSecondary = ScorpionGray,
|
||||||
onBackground = Black,
|
onBackground = Black,
|
||||||
onSurface = Black,
|
onSurface = Black,
|
||||||
onError = AlabasterWhite
|
onError = AlabasterWhite,
|
||||||
|
onTertiary = MineShaftGray600
|
||||||
)
|
)
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
@ -67,3 +69,27 @@ fun KiwixTheme(
|
|||||||
typography = KiwixTypography
|
typography = KiwixTypography
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A custom MaterialTheme specifically for dialogs in the Kiwix app.
|
||||||
|
*
|
||||||
|
* This theme applies a modified dark mode background for dialogs while keeping
|
||||||
|
* the rest of the color scheme unchanged. In light mode, it uses the
|
||||||
|
* standard app theme(KiwixTheme).
|
||||||
|
*/
|
||||||
|
@Composable
|
||||||
|
fun KiwixDialogTheme(
|
||||||
|
darkTheme: Boolean = isSystemInDarkTheme(),
|
||||||
|
content: @Composable () -> Unit
|
||||||
|
) {
|
||||||
|
val colorScheme = when {
|
||||||
|
darkTheme -> DarkColorScheme.copy(background = MineShaftGray700)
|
||||||
|
else -> LightColorScheme
|
||||||
|
}
|
||||||
|
MaterialTheme(
|
||||||
|
colorScheme = colorScheme,
|
||||||
|
content = content,
|
||||||
|
shapes = shapes,
|
||||||
|
typography = KiwixTypography
|
||||||
|
)
|
||||||
|
}
|
||||||
|
@ -20,7 +20,6 @@ package org.kiwix.kiwixmobile.core.ui.theme
|
|||||||
|
|
||||||
import androidx.compose.material3.Typography
|
import androidx.compose.material3.Typography
|
||||||
import androidx.compose.ui.text.TextStyle
|
import androidx.compose.ui.text.TextStyle
|
||||||
import androidx.compose.ui.text.font.FontWeight
|
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.LARGE_BODY_TEXT_SIZE
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.LARGE_BODY_TEXT_SIZE
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.LARGE_HEADLINE_TEXT_SIZE
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.LARGE_HEADLINE_TEXT_SIZE
|
||||||
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.LARGE_LABEL_TEXT_SIZE
|
import org.kiwix.kiwixmobile.core.utils.ComposeDimens.LARGE_LABEL_TEXT_SIZE
|
||||||
@ -50,10 +49,7 @@ val KiwixTypography = Typography(
|
|||||||
bodyLarge = TextStyle(fontSize = LARGE_BODY_TEXT_SIZE),
|
bodyLarge = TextStyle(fontSize = LARGE_BODY_TEXT_SIZE),
|
||||||
bodyMedium = TextStyle(fontSize = MEDIUM_BODY_TEXT_SIZE),
|
bodyMedium = TextStyle(fontSize = MEDIUM_BODY_TEXT_SIZE),
|
||||||
bodySmall = TextStyle(fontSize = SMALL_BODY_TEXT_SIZE),
|
bodySmall = TextStyle(fontSize = SMALL_BODY_TEXT_SIZE),
|
||||||
labelLarge = TextStyle(
|
labelLarge = TextStyle(fontSize = LARGE_LABEL_TEXT_SIZE),
|
||||||
fontSize = LARGE_LABEL_TEXT_SIZE,
|
|
||||||
fontWeight = FontWeight.SemiBold,
|
|
||||||
),
|
|
||||||
labelMedium = TextStyle(fontSize = MEDIUM_LABEL_TEXT_SIZE),
|
labelMedium = TextStyle(fontSize = MEDIUM_LABEL_TEXT_SIZE),
|
||||||
labelSmall = TextStyle(fontSize = SMALL_LABEL_TEXT_SIZE),
|
labelSmall = TextStyle(fontSize = SMALL_LABEL_TEXT_SIZE),
|
||||||
)
|
)
|
||||||
|
@ -33,16 +33,19 @@ object ComposeDimens {
|
|||||||
val BUTTON_DEFAULT_PADDING = 4.dp
|
val BUTTON_DEFAULT_PADDING = 4.dp
|
||||||
|
|
||||||
// Error screen dimens
|
// Error screen dimens
|
||||||
val CRASH_TITLE_TEXT_SIZE = 24.sp
|
|
||||||
val CRASH_MESSAGE_TEXT_SIZE = 14.sp
|
|
||||||
val CRASH_IMAGE_SIZE = 70.dp
|
val CRASH_IMAGE_SIZE = 70.dp
|
||||||
|
|
||||||
|
// KiwixAppBar(Toolbar) height
|
||||||
|
val KIWIX_APP_BAR_HEIGHT = 56.dp
|
||||||
|
|
||||||
// Padding & Margins
|
// Padding & Margins
|
||||||
val SIXTY_DP = 60.dp
|
val SIXTY_DP = 60.dp
|
||||||
val THIRTY_TWO_DP = 30.dp
|
val THIRTY_TWO_DP = 30.dp
|
||||||
|
val TWENTY_DP = 20.dp
|
||||||
val SEVENTEEN_DP = 17.dp
|
val SEVENTEEN_DP = 17.dp
|
||||||
val SIXTEEN_DP = 16.dp
|
val SIXTEEN_DP = 16.dp
|
||||||
val TWELVE_DP = 12.dp
|
val TWELVE_DP = 12.dp
|
||||||
|
val TEN_DP = 10.dp
|
||||||
val EIGHT_DP = 8.dp
|
val EIGHT_DP = 8.dp
|
||||||
val FIVE_DP = 5.dp
|
val FIVE_DP = 5.dp
|
||||||
val FOUR_DP = 4.dp
|
val FOUR_DP = 4.dp
|
||||||
@ -76,4 +79,7 @@ object ComposeDimens {
|
|||||||
val LARGE_LABEL_TEXT_SIZE = 14.sp
|
val LARGE_LABEL_TEXT_SIZE = 14.sp
|
||||||
val MEDIUM_LABEL_TEXT_SIZE = 12.sp
|
val MEDIUM_LABEL_TEXT_SIZE = 12.sp
|
||||||
val SMALL_LABEL_TEXT_SIZE = 10.sp
|
val SMALL_LABEL_TEXT_SIZE = 10.sp
|
||||||
|
|
||||||
|
// AddNoteDialog dimens
|
||||||
|
val MINIMUM_HEIGHT_OF_NOTE_TEXT_FILED = 120.dp
|
||||||
}
|
}
|
||||||
|
@ -1,61 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
|
||||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="match_parent"
|
|
||||||
android:background="@android:color/transparent">
|
|
||||||
|
|
||||||
<include layout="@layout/layout_standard_app_bar" />
|
|
||||||
|
|
||||||
<ScrollView
|
|
||||||
android:layout_width="0dp"
|
|
||||||
android:layout_height="0dp"
|
|
||||||
app:layout_constraintBottom_toBottomOf="parent"
|
|
||||||
app:layout_constraintStart_toStartOf="parent"
|
|
||||||
app:layout_constraintEnd_toEndOf="parent"
|
|
||||||
app:layout_constraintTop_toBottomOf="@+id/app_bar">
|
|
||||||
|
|
||||||
<LinearLayout
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:orientation="vertical">
|
|
||||||
|
|
||||||
<TextView
|
|
||||||
android:id="@+id/add_note_text_view"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:background="@android:color/transparent"
|
|
||||||
android:hint="@string/wiki_article_title"
|
|
||||||
android:maxLines="3"
|
|
||||||
android:paddingStart="20dp"
|
|
||||||
android:paddingTop="10dp"
|
|
||||||
android:paddingEnd="20dp"
|
|
||||||
android:textAppearance="?textAppearanceSubtitle1" />
|
|
||||||
|
|
||||||
<View
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="1dp"
|
|
||||||
android:layout_marginTop="5dp"
|
|
||||||
android:layout_marginBottom="5dp"
|
|
||||||
android:background="#000000" />
|
|
||||||
|
|
||||||
<EditText
|
|
||||||
android:id="@+id/add_note_edit_text"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:background="@android:color/transparent"
|
|
||||||
android:gravity="top|start"
|
|
||||||
android:hint="@string/note"
|
|
||||||
android:inputType="textMultiLine|textCapSentences|textAutoCorrect"
|
|
||||||
android:minLines="6"
|
|
||||||
android:paddingStart="20dp"
|
|
||||||
android:paddingTop="5dp"
|
|
||||||
android:paddingEnd="20dp"
|
|
||||||
android:paddingBottom="10dp"
|
|
||||||
android:scrollbars="vertical"
|
|
||||||
android:textAppearance="@style/TextAppearance.MaterialComponents.Body1"
|
|
||||||
android:importantForAutofill="no" />
|
|
||||||
|
|
||||||
</LinearLayout>
|
|
||||||
</ScrollView>
|
|
||||||
</androidx.constraintlayout.widget.ConstraintLayout>
|
|
Loading…
x
Reference in New Issue
Block a user