Skip to content

Commit 06604fa

Browse files
committed
Add OudsSwitch
1 parent 5a42528 commit 06604fa

36 files changed

+793
-185
lines changed

NOTICE.txt

+11
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,8 @@ app/src/main/res/drawable-hdpi/il_components_link.png
3737
app/src/main/res/drawable-hdpi/il_components_link_dark.png
3838
app/src/main/res/drawable-hdpi/il_components_radiobutton.png
3939
app/src/main/res/drawable-hdpi/il_components_radiobutton_dark.png
40+
app/src/main/res/drawable-hdpi/il_components_switch.png
41+
app/src/main/res/drawable-hdpi/il_components_switch_dark.png
4042
app/src/main/res/drawable-hdpi/il_tokens_grid_column_margin.png
4143
app/src/main/res/drawable-hdpi/il_tokens_grid_column_margin_dark.png
4244
app/src/main/res/drawable-hdpi/il_tokens_grid_max_width.png
@@ -51,6 +53,8 @@ app/src/main/res/drawable-mdpi/il_components_link.png
5153
app/src/main/res/drawable-mdpi/il_components_link_dark.png
5254
app/src/main/res/drawable-mdpi/il_components_radiobutton.png
5355
app/src/main/res/drawable-mdpi/il_components_radiobutton_dark.png
56+
app/src/main/res/drawable-mdpi/il_components_switch.png
57+
app/src/main/res/drawable-mdpi/il_components_switch_dark.png
5458
app/src/main/res/drawable-mdpi/il_tokens_grid_column_margin.png
5559
app/src/main/res/drawable-mdpi/il_tokens_grid_column_margin_dark.png
5660
app/src/main/res/drawable-mdpi/il_tokens_grid_max_width.png
@@ -65,6 +69,8 @@ app/src/main/res/drawable-xhdpi/il_components_link.png
6569
app/src/main/res/drawable-xhdpi/il_components_link_dark.png
6670
app/src/main/res/drawable-xhdpi/il_components_radiobutton.png
6771
app/src/main/res/drawable-xhdpi/il_components_radiobutton_dark.png
72+
app/src/main/res/drawable-xhdpi/il_components_switch.png
73+
app/src/main/res/drawable-xhdpi/il_components_switch_dark.png
6874
app/src/main/res/drawable-xhdpi/il_tokens_grid_column_margin.png
6975
app/src/main/res/drawable-xhdpi/il_tokens_grid_column_margin_dark.png
7076
app/src/main/res/drawable-xhdpi/il_tokens_grid_max_width.png
@@ -79,6 +85,8 @@ app/src/main/res/drawable-xxhdpi/il_components_link.png
7985
app/src/main/res/drawable-xxhdpi/il_components_link_dark.png
8086
app/src/main/res/drawable-xxhdpi/il_components_radiobutton.png
8187
app/src/main/res/drawable-xxhdpi/il_components_radiobutton_dark.png
88+
app/src/main/res/drawable-xxhdpi/il_components_switch.png
89+
app/src/main/res/drawable-xxhdpi/il_components_switch_dark.png
8290
app/src/main/res/drawable-xxhdpi/il_tokens_grid_column_margin.png
8391
app/src/main/res/drawable-xxhdpi/il_tokens_grid_column_margin_dark.png
8492
app/src/main/res/drawable-xxhdpi/il_tokens_grid_max_width.png
@@ -93,6 +101,8 @@ app/src/main/res/drawable-xxxhdpi/il_components_link.png
93101
app/src/main/res/drawable-xxxhdpi/il_components_link_dark.png
94102
app/src/main/res/drawable-xxxhdpi/il_components_radiobutton.png
95103
app/src/main/res/drawable-xxxhdpi/il_components_radiobutton_dark.png
104+
app/src/main/res/drawable-xxxhdpi/il_components_switch.png
105+
app/src/main/res/drawable-xxxhdpi/il_components_switch_dark.png
96106
app/src/main/res/drawable-xxxhdpi/il_tokens_grid_column_margin.png
97107
app/src/main/res/drawable-xxxhdpi/il_tokens_grid_column_margin_dark.png
98108
app/src/main/res/drawable-xxxhdpi/il_tokens_grid_max_width.png
@@ -106,6 +116,7 @@ core/src/main/res/drawable/checkbox_indeterminate.xml
106116
core/src/main/res/drawable/checkbox_selected.xml
107117
core/src/main/res/drawable/chevron_left.xml
108118
core/src/main/res/drawable/radiobutton_selected.xml
119+
core/src/main/res/drawable/switch_selected.xml
109120

110121
docs/assets/logo-icon.svg
111122

app/src/main/java/com/orange/ouds/app/ui/components/Component.kt

+10
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ import com.orange.ouds.app.ui.components.checkbox.CheckboxItemDemoScreen
2222
import com.orange.ouds.app.ui.components.link.LinkDemoScreen
2323
import com.orange.ouds.app.ui.components.radiobutton.RadioButtonDemoScreen
2424
import com.orange.ouds.app.ui.components.radiobutton.RadioButtonItemDemoScreen
25+
import com.orange.ouds.app.ui.components.switch.SwitchDemoScreen
2526
import com.orange.ouds.app.ui.utilities.LightDarkResourceId
2627

2728
val components = Component::class.sealedSubclasses.mapNotNull { it.objectInstance }
@@ -68,6 +69,13 @@ sealed class Component(
6869
R.string.app_components_radioButton_description_text,
6970
listOf(Variant.RadioButton, Variant.RadioButtonItem)
7071
)
72+
73+
data object Switch : Component(
74+
R.string.app_components_switch_label,
75+
LightDarkResourceId(R.drawable.il_components_switch, R.drawable.il_components_switch_dark),
76+
R.string.app_components_switch_description_text,
77+
listOf(Variant.Switch, Variant.SwitchItem)
78+
)
7179
}
7280

7381
sealed class Variant(
@@ -89,4 +97,6 @@ sealed class Variant(
8997

9098
data object RadioButton : Variant(R.string.app_components_radioButton_radioButton_label, { RadioButtonDemoScreen() })
9199
data object RadioButtonItem : Variant(R.string.app_components_radioButton_radioButtonItem_label, { RadioButtonItemDemoScreen() })
100+
data object Switch : Variant(R.string.app_components_switch_switch_label, { SwitchDemoScreen() })
101+
data object SwitchItem : Variant(R.string.app_components_switch_switchItem_label, {})
92102
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
/*
2+
* Software Name: OUDS Android
3+
* SPDX-FileCopyrightText: Copyright (c) Orange SA
4+
* SPDX-License-Identifier: MIT
5+
*
6+
* This software is distributed under the MIT license,
7+
* the text of which is available at https://opensource.org/license/MIT/
8+
* or see the "LICENSE" file for more details.
9+
*
10+
* Software description: Android library of reusable graphical components
11+
*/
12+
13+
package com.orange.ouds.app.ui.components.switch
14+
15+
import androidx.compose.foundation.layout.padding
16+
import androidx.compose.material3.ExperimentalMaterial3Api
17+
import androidx.compose.material3.rememberBottomSheetScaffoldState
18+
import androidx.compose.runtime.Composable
19+
import androidx.compose.ui.Modifier
20+
import androidx.compose.ui.res.stringResource
21+
import com.orange.ouds.app.R
22+
import com.orange.ouds.app.ui.components.enabledArgument
23+
import com.orange.ouds.app.ui.utilities.composable.CodeSnippet
24+
import com.orange.ouds.app.ui.utilities.composable.CustomizationBottomSheetScaffold
25+
import com.orange.ouds.app.ui.utilities.composable.CustomizationSwitchListItem
26+
import com.orange.ouds.app.ui.utilities.composable.DemoScreen
27+
import com.orange.ouds.app.ui.utilities.composable.LightDarkDemo
28+
import com.orange.ouds.core.component.OudsSwitch
29+
import com.orange.ouds.core.theme.OudsTheme
30+
import com.orange.ouds.core.utilities.OudsPreview
31+
import com.orange.ouds.foundation.utilities.UiModePreviews
32+
33+
@OptIn(ExperimentalMaterial3Api::class)
34+
@Composable
35+
fun SwitchDemoScreen() = DemoScreen(rememberSwitchDemoState()) {
36+
CustomizationBottomSheetScaffold(
37+
bottomSheetScaffoldState = rememberBottomSheetScaffoldState(),
38+
bottomSheetContent = {
39+
CustomizationSwitchListItem(
40+
label = stringResource(R.string.app_common_enabled_label),
41+
checked = enabled,
42+
onCheckedChange = { enabled = it }
43+
)
44+
}
45+
) {
46+
LightDarkDemo {
47+
SwitchDemo(state = this@DemoScreen)
48+
}
49+
50+
SwitchDemoCodeSnippet(
51+
state = this@DemoScreen,
52+
modifier = Modifier
53+
.padding(horizontal = OudsTheme.grids.margin, vertical = OudsTheme.spaces.fixed.medium)
54+
.padding(top = OudsTheme.spaces.fixed.medium)
55+
)
56+
}
57+
}
58+
59+
@Composable
60+
private fun SwitchDemo(state: SwitchDemoState) {
61+
with(state) {
62+
OudsSwitch(
63+
checked = checked,
64+
onCheckedChange = { checked = it },
65+
enabled = enabled
66+
)
67+
}
68+
}
69+
70+
@Composable
71+
private fun SwitchDemoCodeSnippet(state: SwitchDemoState, modifier: Modifier = Modifier) {
72+
CodeSnippet(modifier = modifier) {
73+
with(state) {
74+
functionCall("OudsSwitch") {
75+
typedArgument("checked", checked)
76+
lambdaArgument("onCheckedChange") {
77+
comment("Change state")
78+
}
79+
enabledArgument(enabled)
80+
}
81+
}
82+
}
83+
}
84+
85+
@UiModePreviews.Default
86+
@Composable
87+
private fun PreviewSwitchDemoScreen() = OudsPreview {
88+
SwitchDemoScreen()
89+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
/*
2+
* Software Name: OUDS Android
3+
* SPDX-FileCopyrightText: Copyright (c) Orange SA
4+
* SPDX-License-Identifier: MIT
5+
*
6+
* This software is distributed under the MIT license,
7+
* the text of which is available at https://opensource.org/license/MIT/
8+
* or see the "LICENSE" file for more details.
9+
*
10+
* Software description: Android library of reusable graphical components
11+
*/
12+
13+
package com.orange.ouds.app.ui.components.switch
14+
15+
import androidx.compose.runtime.Composable
16+
import androidx.compose.runtime.getValue
17+
import androidx.compose.runtime.mutableStateOf
18+
import androidx.compose.runtime.saveable.mapSaver
19+
import androidx.compose.runtime.saveable.rememberSaveable
20+
import androidx.compose.runtime.setValue
21+
22+
@Composable
23+
fun rememberSwitchDemoState(
24+
checked: Boolean = false,
25+
enabled: Boolean = true
26+
) = rememberSaveable(checked, enabled, saver = SwitchDemoState.Saver) {
27+
SwitchDemoState(checked, enabled)
28+
}
29+
30+
class SwitchDemoState(
31+
checked: Boolean,
32+
enabled: Boolean
33+
) {
34+
35+
companion object {
36+
37+
val Saver = run {
38+
val checkedKey = "checked"
39+
val enabledKey = "enabled"
40+
mapSaver(
41+
save = { state ->
42+
mapOf(
43+
checkedKey to state.checked,
44+
enabledKey to state.enabled
45+
)
46+
},
47+
restore = { map ->
48+
SwitchDemoState(
49+
map[checkedKey] as Boolean,
50+
map[enabledKey] as Boolean
51+
)
52+
}
53+
)
54+
}
55+
}
56+
57+
var checked: Boolean by mutableStateOf(checked)
58+
59+
var enabled: Boolean by mutableStateOf(enabled)
60+
}
Loading
Loading
Loading
Loading
Loading
Loading
Loading
Loading
Loading
Loading

app/src/main/res/values/strings.xml

+6
Original file line numberDiff line numberDiff line change
@@ -154,6 +154,12 @@
154154
<string name="app_components_radioButton_radioButtonItem_outlined_label">Outlined</string>
155155
<string name="app_components_radioButton_radioButtonItem_additionalText_label">Additional text</string>
156156

157+
<!-- Components: switch -->
158+
<string name="app_components_switch_label">Switch</string>
159+
<string name="app_components_switch_description_text">A switch allows the user to toggle between two states, typically "on" and "off." It is often represented as a button or a slider that changes position or color to indicate the current state. Switches are used to enable or disable features, options, or settings in an intuitive and visual manner.</string>
160+
<string name="app_components_switch_switch_label">Switch</string>
161+
<string name="app_components_switch_switchItem_label">Switch item</string>
162+
157163
<!-- About -->
158164
<string name="app_about_name_label">Design System Toolbox</string>
159165
<string name="app_about_pullRequest_label">Pull request\u0020</string>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
/*
2+
* Software Name: OUDS Android
3+
* SPDX-FileCopyrightText: Copyright (c) Orange SA
4+
* SPDX-License-Identifier: MIT
5+
*
6+
* This software is distributed under the MIT license,
7+
* the text of which is available at https://opensource.org/license/MIT/
8+
* or see the "LICENSE" file for more details.
9+
*
10+
* Software description: Android library of reusable graphical components
11+
*/
12+
13+
package com.orange.ouds.core.component
14+
15+
import androidx.compose.ui.Modifier
16+
import androidx.compose.ui.platform.testTag
17+
import androidx.compose.ui.test.assertIsOff
18+
import androidx.compose.ui.test.junit4.createComposeRule
19+
import androidx.compose.ui.test.onNodeWithTag
20+
import androidx.compose.ui.test.performClick
21+
import com.orange.ouds.core.extension.setOudsContent
22+
import org.junit.Rule
23+
import org.junit.Test
24+
import org.mockito.kotlin.mock
25+
import org.mockito.kotlin.verify
26+
27+
internal class OudsSwitchButtonTest {
28+
29+
@get:Rule
30+
val composeTestRule = createComposeRule()
31+
32+
@Test
33+
fun oudsSwitch_click_succeeds() {
34+
with(composeTestRule) {
35+
val testTag = "OudsSwitch"
36+
val checked = false
37+
val onCheckedChange = mock<(Boolean) -> Unit>()
38+
39+
setOudsContent {
40+
OudsSwitch(
41+
checked = checked,
42+
onCheckedChange = onCheckedChange,
43+
modifier = Modifier.testTag(testTag)
44+
)
45+
}
46+
47+
onNodeWithTag(testTag).assertIsOff()
48+
onNodeWithTag(testTag).performClick()
49+
verify(onCheckedChange).invoke(!checked)
50+
}
51+
}
52+
}

0 commit comments

Comments
 (0)