-
-
Save zach-klippenstein/7ae8874db304f957d6bb91263e292117 to your computer and use it in GitHub Desktop.
| import android.annotation.SuppressLint | |
| import androidx.compose.animation.core.animateDpAsState | |
| import androidx.compose.animation.core.animateFloatAsState | |
| import androidx.compose.foundation.Canvas | |
| import androidx.compose.foundation.background | |
| import androidx.compose.foundation.gestures.awaitFirstDown | |
| import androidx.compose.foundation.gestures.forEachGesture | |
| import androidx.compose.foundation.gestures.horizontalDrag | |
| import androidx.compose.foundation.layout.Arrangement.spacedBy | |
| import androidx.compose.foundation.layout.Box | |
| import androidx.compose.foundation.layout.Column | |
| import androidx.compose.foundation.layout.Row | |
| import androidx.compose.foundation.layout.fillMaxSize | |
| import androidx.compose.foundation.layout.fillMaxWidth | |
| import androidx.compose.foundation.layout.padding | |
| import androidx.compose.foundation.layout.wrapContentWidth | |
| import androidx.compose.foundation.selection.selectableGroup | |
| import androidx.compose.foundation.shape.RoundedCornerShape | |
| import androidx.compose.material.LocalTextStyle | |
| import androidx.compose.material.MaterialTheme | |
| import androidx.compose.material.Surface | |
| import androidx.compose.material.Text | |
| import androidx.compose.runtime.Composable | |
| import androidx.compose.runtime.CompositionLocalProvider | |
| import androidx.compose.runtime.getValue | |
| import androidx.compose.runtime.mutableStateOf | |
| import androidx.compose.runtime.remember | |
| import androidx.compose.runtime.setValue | |
| import androidx.compose.ui.Modifier | |
| import androidx.compose.ui.composed | |
| import androidx.compose.ui.draw.alpha | |
| import androidx.compose.ui.draw.shadow | |
| import androidx.compose.ui.geometry.Offset | |
| import androidx.compose.ui.geometry.Rect | |
| import androidx.compose.ui.graphics.Color | |
| import androidx.compose.ui.graphics.TransformOrigin | |
| import androidx.compose.ui.graphics.graphicsLayer | |
| import androidx.compose.ui.input.pointer.AwaitPointerEventScope | |
| import androidx.compose.ui.input.pointer.PointerEventPass | |
| import androidx.compose.ui.input.pointer.PointerInputChange | |
| import androidx.compose.ui.input.pointer.changedToUp | |
| import androidx.compose.ui.input.pointer.pointerInput | |
| import androidx.compose.ui.input.pointer.positionChangeConsumed | |
| import androidx.compose.ui.layout.Layout | |
| import androidx.compose.ui.semantics.Role | |
| import androidx.compose.ui.semantics.onClick | |
| import androidx.compose.ui.semantics.role | |
| import androidx.compose.ui.semantics.selected | |
| import androidx.compose.ui.semantics.semantics | |
| import androidx.compose.ui.semantics.stateDescription | |
| import androidx.compose.ui.text.TextStyle | |
| import androidx.compose.ui.text.font.FontWeight | |
| import androidx.compose.ui.text.style.TextOverflow.Ellipsis | |
| import androidx.compose.ui.tooling.preview.Preview | |
| import androidx.compose.ui.unit.Constraints | |
| import androidx.compose.ui.unit.Density | |
| import androidx.compose.ui.unit.IntOffset | |
| import androidx.compose.ui.unit.dp | |
| @Preview | |
| @Composable fun SegmentedDemo() { | |
| MaterialTheme { | |
| Surface { | |
| Column(Modifier.padding(16.dp), verticalArrangement = spacedBy(16.dp)) { | |
| Text("SEGMENTS", style = MaterialTheme.typography.caption) | |
| val twoSegments = remember { listOf("Foo", "Bar") } | |
| var selectedTwoSegment by remember { mutableStateOf(twoSegments.first()) } | |
| SegmentedControl( | |
| twoSegments, | |
| selectedTwoSegment, | |
| onSegmentSelected = { selectedTwoSegment = it } | |
| ) { | |
| SegmentText(it) | |
| } | |
| val threeSegments = remember { listOf("Foo", "Bar", "Some very long string") } | |
| var selectedThreeSegment by remember { mutableStateOf(threeSegments.first()) } | |
| SegmentedControl( | |
| threeSegments, | |
| selectedThreeSegment, | |
| onSegmentSelected = { selectedThreeSegment = it } | |
| ) { | |
| SegmentText(it) | |
| } | |
| } | |
| } | |
| } | |
| } | |
| private const val NO_SEGMENT_INDEX = -1 | |
| /** Padding inside the track. */ | |
| private val TRACK_PADDING = 2.dp | |
| private val TRACK_COLOR = Color.LightGray.copy(alpha = .5f) | |
| /** Additional padding to inset segments and the thumb when pressed. */ | |
| private val PRESSED_TRACK_PADDING = 1.dp | |
| /** Padding inside individual segments. */ | |
| private val SEGMENT_PADDING = 5.dp | |
| /** Alpha to use to indicate pressed state when unselected segments are pressed. */ | |
| private const val PRESSED_UNSELECTED_ALPHA = .6f | |
| private val BACKGROUND_SHAPE = RoundedCornerShape(8.dp) | |
| @Composable fun <T : Any> SegmentedControl( | |
| segments: List<T>, | |
| selectedSegment: T, | |
| onSegmentSelected: (T) -> Unit, | |
| modifier: Modifier = Modifier, | |
| content: @Composable (T) -> Unit | |
| ) { | |
| val state = remember { SegmentedControlState() } | |
| state.segmentCount = segments.size | |
| state.selectedSegment = segments.indexOf(selectedSegment) | |
| state.onSegmentSelected = { onSegmentSelected(segments[it]) } | |
| // Animate between whole-number indices so we don't need to do pixel calculations. | |
| val selectedIndexOffset by animateFloatAsState(state.selectedSegment.toFloat()) | |
| // Use a custom layout so that we can measure the thumb using the height of the segments. The thumb | |
| // is whole composable that draws itself – this layout is just responsible for placing it under | |
| // the correct segment. | |
| Layout( | |
| content = { | |
| // Each of these produces a single measurable. | |
| Thumb(state) | |
| Dividers(state) | |
| Segments(state, segments, content) | |
| }, | |
| modifier = modifier | |
| .fillMaxWidth() | |
| .then(state.inputModifier) | |
| .background(TRACK_COLOR, BACKGROUND_SHAPE) | |
| .padding(TRACK_PADDING) | |
| ) { measurables, constraints -> | |
| val (thumbMeasurable, dividersMeasurable, segmentsMeasurable) = measurables | |
| // Measure the segments first so we know how tall to make the thumb. | |
| val segmentsPlaceable = segmentsMeasurable.measure(constraints) | |
| state.updatePressedScale(segmentsPlaceable.height, this) | |
| // Now we can measure the thumb and dividers to be the right size. | |
| val thumbPlaceable = thumbMeasurable.measure( | |
| Constraints.fixed( | |
| width = segmentsPlaceable.width / segments.size, | |
| height = segmentsPlaceable.height | |
| ) | |
| ) | |
| val dividersPlaceable = dividersMeasurable.measure( | |
| Constraints.fixed( | |
| width = segmentsPlaceable.width, | |
| height = segmentsPlaceable.height | |
| ) | |
| ) | |
| layout(segmentsPlaceable.width, segmentsPlaceable.height) { | |
| val segmentWidth = segmentsPlaceable.width / segments.size | |
| // Place the thumb first since it should be drawn below the segments. | |
| thumbPlaceable.placeRelative( | |
| x = (selectedIndexOffset * segmentWidth).toInt(), | |
| y = 0 | |
| ) | |
| dividersPlaceable.placeRelative(IntOffset.Zero) | |
| segmentsPlaceable.placeRelative(IntOffset.Zero) | |
| } | |
| } | |
| } | |
| /** | |
| * Wrapper around [Text] that is configured to display appropriately inside of a [SegmentedControl]. | |
| */ | |
| @Composable fun SegmentText(text: String) { | |
| Text(text, maxLines = 1, overflow = Ellipsis) | |
| } | |
| /** | |
| * Draws the thumb (selected indicator) on a [SegmentedControl] track, underneath the [Segments]. | |
| */ | |
| @Composable private fun Thumb(state: SegmentedControlState) { | |
| Box( | |
| Modifier | |
| .then( | |
| state.segmentScaleModifier( | |
| pressed = state.pressedSegment == state.selectedSegment, | |
| segment = state.selectedSegment | |
| ) | |
| ) | |
| .shadow(4.dp, BACKGROUND_SHAPE) | |
| .background(Color.White, BACKGROUND_SHAPE) | |
| ) | |
| } | |
| /** | |
| * Draws dividers between segments. No dividers are drawn around the selected segment. | |
| */ | |
| @Composable private fun Dividers(state: SegmentedControlState) { | |
| // Animate each divider independently. | |
| val alphas = (0 until state.segmentCount).map { i -> | |
| val selectionAdjacent = i == state.selectedSegment || i - 1 == state.selectedSegment | |
| animateFloatAsState(if (selectionAdjacent) 0f else 1f) | |
| } | |
| Canvas(Modifier.fillMaxSize()) { | |
| val segmentWidth = size.width / state.segmentCount | |
| val dividerPadding = TRACK_PADDING + PRESSED_TRACK_PADDING | |
| alphas.forEachIndexed { i, alpha -> | |
| val x = i * segmentWidth | |
| drawLine( | |
| Color.White, | |
| alpha = alpha.value, | |
| start = Offset(x, dividerPadding.toPx()), | |
| end = Offset(x, size.height - dividerPadding.toPx()) | |
| ) | |
| } | |
| } | |
| } | |
| /** | |
| * Draws the actual segments in a [SegmentedControl]. | |
| */ | |
| @Composable private fun <T> Segments( | |
| state: SegmentedControlState, | |
| segments: List<T>, | |
| content: @Composable (T) -> Unit | |
| ) { | |
| CompositionLocalProvider( | |
| LocalTextStyle provides TextStyle(fontWeight = FontWeight.Medium) | |
| ) { | |
| Row( | |
| horizontalArrangement = spacedBy(TRACK_PADDING), | |
| modifier = Modifier | |
| .fillMaxWidth() | |
| .selectableGroup() | |
| ) { | |
| segments.forEachIndexed { i, segment -> | |
| val isSelected = i == state.selectedSegment | |
| val isPressed = i == state.pressedSegment | |
| // Unselected presses are represented by fading. | |
| val alpha by animateFloatAsState(if (!isSelected && isPressed) PRESSED_UNSELECTED_ALPHA else 1f) | |
| // We can't use Modifier.selectable because it does way too much: it does its own input | |
| // handling and wires into Compose's indicaiton/interaction system, which we don't want because | |
| // we've got our own indication mechanism. | |
| val semanticsModifier = Modifier.semantics(mergeDescendants = true) { | |
| selected = isSelected | |
| role = Role.Button | |
| onClick { state.onSegmentSelected(i); true } | |
| stateDescription = if (isSelected) "Selected" else "Not selected" | |
| } | |
| Box( | |
| Modifier | |
| // Divide space evenly between all segments. | |
| .weight(1f) | |
| .then(semanticsModifier) | |
| .padding(SEGMENT_PADDING) | |
| // Draw pressed indication when not selected. | |
| .alpha(alpha) | |
| // Selected presses are represented by scaling. | |
| .then(state.segmentScaleModifier(isPressed && isSelected, i)) | |
| // Center the segment content. | |
| .wrapContentWidth() | |
| ) { | |
| content(segment) | |
| } | |
| } | |
| } | |
| } | |
| } | |
| private class SegmentedControlState { | |
| var segmentCount by mutableStateOf(0) | |
| var selectedSegment by mutableStateOf(0) | |
| var onSegmentSelected: (Int) -> Unit by mutableStateOf({}) | |
| var pressedSegment by mutableStateOf(NO_SEGMENT_INDEX) | |
| /** | |
| * Scale factor that should be used to scale pressed segments (both the segment itself and the | |
| * thumb). When this scale is applied, exactly [PRESSED_TRACK_PADDING] will be added around the | |
| * element's usual size. | |
| */ | |
| var pressedSelectedScale by mutableStateOf(1f) | |
| private set | |
| /** | |
| * Calculates the scale factor we need to use for pressed segments to get the desired padding. | |
| */ | |
| fun updatePressedScale(controlHeight: Int, density: Density) { | |
| with(density) { | |
| val pressedPadding = PRESSED_TRACK_PADDING * 2 | |
| val pressedHeight = controlHeight - pressedPadding.toPx() | |
| pressedSelectedScale = pressedHeight / controlHeight | |
| } | |
| } | |
| /** | |
| * Returns a [Modifier] that will scale an element so that it gets [PRESSED_TRACK_PADDING] extra | |
| * padding around it. The scale will be animated. | |
| * | |
| * The scale is also performed around either the left or right edge of the element if the [segment] | |
| * is the first or last segment, respectively. In those cases, the scale will also be translated so | |
| * that [PRESSED_TRACK_PADDING] will be added on the left or right edge. | |
| */ | |
| @SuppressLint("ModifierFactoryExtensionFunction") | |
| fun segmentScaleModifier( | |
| pressed: Boolean, | |
| segment: Int, | |
| ): Modifier = Modifier.composed { | |
| val scale by animateFloatAsState(if (pressed) pressedSelectedScale else 1f) | |
| val xOffset by animateDpAsState(if (pressed) PRESSED_TRACK_PADDING else 0.dp) | |
| graphicsLayer { | |
| this.scaleX = scale | |
| this.scaleY = scale | |
| // Scales on the ends should gravitate to that edge. | |
| this.transformOrigin = TransformOrigin( | |
| pivotFractionX = when (segment) { | |
| 0 -> 0f | |
| segmentCount - 1 -> 1f | |
| else -> .5f | |
| }, | |
| pivotFractionY = .5f | |
| ) | |
| // But should still move inwards to keep the pressed padding consistent with top and bottom. | |
| this.translationX = when (segment) { | |
| 0 -> xOffset.toPx() | |
| segmentCount - 1 -> -xOffset.toPx() | |
| else -> 0f | |
| } | |
| } | |
| } | |
| /** | |
| * A [Modifier] that will listen for touch gestures and update the selected and pressed properties | |
| * of this state appropriately. | |
| * | |
| * Input will be reset if the [segmentCount] changes. | |
| */ | |
| val inputModifier = Modifier.pointerInput(segmentCount) { | |
| val segmentWidth = size.width / segmentCount | |
| // Helper to calculate which segment an event occured in. | |
| fun segmentIndex(change: PointerInputChange): Int = | |
| ((change.position.x / size.width.toFloat()) * segmentCount) | |
| .toInt() | |
| .coerceIn(0, segmentCount - 1) | |
| forEachGesture { | |
| awaitPointerEventScope { | |
| val down = awaitFirstDown() | |
| pressedSegment = segmentIndex(down) | |
| val downOnSelected = pressedSegment == selectedSegment | |
| val segmentBounds = Rect( | |
| left = pressedSegment * segmentWidth.toFloat(), | |
| right = (pressedSegment + 1) * segmentWidth.toFloat(), | |
| top = 0f, | |
| bottom = size.height.toFloat() | |
| ) | |
| // Now that the pointer is down, the rest of the gesture depends on whether the segment that | |
| // was "pressed" was selected. | |
| if (downOnSelected) { | |
| // When the selected segment is pressed, it can be dragged to other segments to animate the | |
| // thumb moving and the segments scaling. | |
| horizontalDrag(down.id) { change -> | |
| pressedSegment = segmentIndex(change) | |
| // Notify the SegmentedControl caller when the pointer changes segments. | |
| if (pressedSegment != selectedSegment) { | |
| onSegmentSelected(pressedSegment) | |
| } | |
| } | |
| } else { | |
| // When an unselected segment is pressed, we just animate the alpha of the segment while | |
| // the pointer is down. No dragging is supported. | |
| waitForUpOrCancellation(inBounds = segmentBounds) | |
| // Null means the gesture was cancelled (e.g. dragged out of bounds). | |
| ?.let { onSegmentSelected(pressedSegment) } | |
| } | |
| // In either case, once the gesture is cancelled, stop showing the pressed indication. | |
| pressedSegment = NO_SEGMENT_INDEX | |
| } | |
| } | |
| } | |
| } | |
| /** | |
| * Copy of nullary waitForUpOrCancellation that works with bounds that may not be at 0,0. | |
| */ | |
| private suspend fun AwaitPointerEventScope.waitForUpOrCancellation(inBounds: Rect): PointerInputChange? { | |
| while (true) { | |
| val event = awaitPointerEvent(PointerEventPass.Main) | |
| if (event.changes.all { it.changedToUp() }) { | |
| // All pointers are up | |
| return event.changes[0] | |
| } | |
| if (event.changes.any { it.consumed.downChange || !inBounds.contains(it.position) }) { | |
| return null // Canceled | |
| } | |
| // Check for cancel by position consumption. We can look on the Final pass of the | |
| // existing pointer event because it comes after the Main pass we checked above. | |
| val consumeCheck = awaitPointerEvent(PointerEventPass.Final) | |
| if (consumeCheck.changes.any { it.positionChangeConsumed() }) { | |
| return null | |
| } | |
| } | |
| } |
| Apache License | |
| Version 2.0, January 2004 | |
| http://www.apache.org/licenses/ | |
| TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | |
| 1. Definitions. | |
| "License" shall mean the terms and conditions for use, reproduction, | |
| and distribution as defined by Sections 1 through 9 of this document. | |
| "Licensor" shall mean the copyright owner or entity authorized by | |
| the copyright owner that is granting the License. | |
| "Legal Entity" shall mean the union of the acting entity and all | |
| other entities that control, are controlled by, or are under common | |
| control with that entity. For the purposes of this definition, | |
| "control" means (i) the power, direct or indirect, to cause the | |
| direction or management of such entity, whether by contract or | |
| otherwise, or (ii) ownership of fifty percent (50%) or more of the | |
| outstanding shares, or (iii) beneficial ownership of such entity. | |
| "You" (or "Your") shall mean an individual or Legal Entity | |
| exercising permissions granted by this License. | |
| "Source" form shall mean the preferred form for making modifications, | |
| including but not limited to software source code, documentation | |
| source, and configuration files. | |
| "Object" form shall mean any form resulting from mechanical | |
| transformation or translation of a Source form, including but | |
| not limited to compiled object code, generated documentation, | |
| and conversions to other media types. | |
| "Work" shall mean the work of authorship, whether in Source or | |
| Object form, made available under the License, as indicated by a | |
| copyright notice that is included in or attached to the work | |
| (an example is provided in the Appendix below). | |
| "Derivative Works" shall mean any work, whether in Source or Object | |
| form, that is based on (or derived from) the Work and for which the | |
| editorial revisions, annotations, elaborations, or other modifications | |
| represent, as a whole, an original work of authorship. For the purposes | |
| of this License, Derivative Works shall not include works that remain | |
| separable from, or merely link (or bind by name) to the interfaces of, | |
| the Work and Derivative Works thereof. | |
| "Contribution" shall mean any work of authorship, including | |
| the original version of the Work and any modifications or additions | |
| to that Work or Derivative Works thereof, that is intentionally | |
| submitted to Licensor for inclusion in the Work by the copyright owner | |
| or by an individual or Legal Entity authorized to submit on behalf of | |
| the copyright owner. For the purposes of this definition, "submitted" | |
| means any form of electronic, verbal, or written communication sent | |
| to the Licensor or its representatives, including but not limited to | |
| communication on electronic mailing lists, source code control systems, | |
| and issue tracking systems that are managed by, or on behalf of, the | |
| Licensor for the purpose of discussing and improving the Work, but | |
| excluding communication that is conspicuously marked or otherwise | |
| designated in writing by the copyright owner as "Not a Contribution." | |
| "Contributor" shall mean Licensor and any individual or Legal Entity | |
| on behalf of whom a Contribution has been received by Licensor and | |
| subsequently incorporated within the Work. | |
| 2. Grant of Copyright License. Subject to the terms and conditions of | |
| this License, each Contributor hereby grants to You a perpetual, | |
| worldwide, non-exclusive, no-charge, royalty-free, irrevocable | |
| copyright license to reproduce, prepare Derivative Works of, | |
| publicly display, publicly perform, sublicense, and distribute the | |
| Work and such Derivative Works in Source or Object form. | |
| 3. Grant of Patent License. Subject to the terms and conditions of | |
| this License, each Contributor hereby grants to You a perpetual, | |
| worldwide, non-exclusive, no-charge, royalty-free, irrevocable | |
| (except as stated in this section) patent license to make, have made, | |
| use, offer to sell, sell, import, and otherwise transfer the Work, | |
| where such license applies only to those patent claims licensable | |
| by such Contributor that are necessarily infringed by their | |
| Contribution(s) alone or by combination of their Contribution(s) | |
| with the Work to which such Contribution(s) was submitted. If You | |
| institute patent litigation against any entity (including a | |
| cross-claim or counterclaim in a lawsuit) alleging that the Work | |
| or a Contribution incorporated within the Work constitutes direct | |
| or contributory patent infringement, then any patent licenses | |
| granted to You under this License for that Work shall terminate | |
| as of the date such litigation is filed. | |
| 4. Redistribution. You may reproduce and distribute copies of the | |
| Work or Derivative Works thereof in any medium, with or without | |
| modifications, and in Source or Object form, provided that You | |
| meet the following conditions: | |
| (a) You must give any other recipients of the Work or | |
| Derivative Works a copy of this License; and | |
| (b) You must cause any modified files to carry prominent notices | |
| stating that You changed the files; and | |
| (c) You must retain, in the Source form of any Derivative Works | |
| that You distribute, all copyright, patent, trademark, and | |
| attribution notices from the Source form of the Work, | |
| excluding those notices that do not pertain to any part of | |
| the Derivative Works; and | |
| (d) If the Work includes a "NOTICE" text file as part of its | |
| distribution, then any Derivative Works that You distribute must | |
| include a readable copy of the attribution notices contained | |
| within such NOTICE file, excluding those notices that do not | |
| pertain to any part of the Derivative Works, in at least one | |
| of the following places: within a NOTICE text file distributed | |
| as part of the Derivative Works; within the Source form or | |
| documentation, if provided along with the Derivative Works; or, | |
| within a display generated by the Derivative Works, if and | |
| wherever such third-party notices normally appear. The contents | |
| of the NOTICE file are for informational purposes only and | |
| do not modify the License. You may add Your own attribution | |
| notices within Derivative Works that You distribute, alongside | |
| or as an addendum to the NOTICE text from the Work, provided | |
| that such additional attribution notices cannot be construed | |
| as modifying the License. | |
| You may add Your own copyright statement to Your modifications and | |
| may provide additional or different license terms and conditions | |
| for use, reproduction, or distribution of Your modifications, or | |
| for any such Derivative Works as a whole, provided Your use, | |
| reproduction, and distribution of the Work otherwise complies with | |
| the conditions stated in this License. | |
| 5. Submission of Contributions. Unless You explicitly state otherwise, | |
| any Contribution intentionally submitted for inclusion in the Work | |
| by You to the Licensor shall be under the terms and conditions of | |
| this License, without any additional terms or conditions. | |
| Notwithstanding the above, nothing herein shall supersede or modify | |
| the terms of any separate license agreement you may have executed | |
| with Licensor regarding such Contributions. | |
| 6. Trademarks. This License does not grant permission to use the trade | |
| names, trademarks, service marks, or product names of the Licensor, | |
| except as required for reasonable and customary use in describing the | |
| origin of the Work and reproducing the content of the NOTICE file. | |
| 7. Disclaimer of Warranty. Unless required by applicable law or | |
| agreed to in writing, Licensor provides the Work (and each | |
| Contributor provides its Contributions) on an "AS IS" BASIS, | |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | |
| implied, including, without limitation, any warranties or conditions | |
| of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | |
| PARTICULAR PURPOSE. You are solely responsible for determining the | |
| appropriateness of using or redistributing the Work and assume any | |
| risks associated with Your exercise of permissions under this License. | |
| 8. Limitation of Liability. In no event and under no legal theory, | |
| whether in tort (including negligence), contract, or otherwise, | |
| unless required by applicable law (such as deliberate and grossly | |
| negligent acts) or agreed to in writing, shall any Contributor be | |
| liable to You for damages, including any direct, indirect, special, | |
| incidental, or consequential damages of any character arising as a | |
| result of this License or out of the use or inability to use the | |
| Work (including but not limited to damages for loss of goodwill, | |
| work stoppage, computer failure or malfunction, or any and all | |
| other commercial damages or losses), even if such Contributor | |
| has been advised of the possibility of such damages. | |
| 9. Accepting Warranty or Additional Liability. While redistributing | |
| the Work or Derivative Works thereof, You may choose to offer, | |
| and charge a fee for, acceptance of support, warranty, indemnity, | |
| or other liability obligations and/or rights consistent with this | |
| License. However, in accepting such obligations, You may act only | |
| on Your own behalf and on Your sole responsibility, not on behalf | |
| of any other Contributor, and only if You agree to indemnify, | |
| defend, and hold each Contributor harmless for any liability | |
| incurred by, or claims asserted against, such Contributor by reason | |
| of your accepting any such warranty or additional liability. | |
| END OF TERMS AND CONDITIONS | |
| APPENDIX: How to apply the Apache License to your work. | |
| To apply the Apache License to your work, attach the following | |
| boilerplate notice, with the fields enclosed by brackets "[]" | |
| replaced with your own identifying information. (Don't include | |
| the brackets!) The text should be enclosed in the appropriate | |
| comment syntax for the file format. We also recommend that a | |
| file or class name and description of purpose be included on the | |
| same "printed page" as the copyright notice for easier | |
| identification within third-party archives. | |
| Copyright [yyyy] [name of copyright owner] | |
| Licensed under the Apache License, Version 2.0 (the "License"); | |
| you may not use this file except in compliance with the License. | |
| You may obtain a copy of the License at | |
| http://www.apache.org/licenses/LICENSE-2.0 | |
| Unless required by applicable law or agreed to in writing, software | |
| distributed under the License is distributed on an "AS IS" BASIS, | |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
| See the License for the specific language governing permissions and | |
| limitations under the License. |
Bug: A divider is rendered on the left of the first item.
Filed an issue to request the waitForUpOrCancellation in this gist.
Hi,
thanks for your work, that's exactly what I need! I'd like to use that component in my application. Under what license is it published?
Cheers
Michael
@zach-klippenstein is there a license you have for this gist?
I just added a license file, it’s Apache 2.
Thanks for adding the license!
Bug: A divider is rendered on the left of the first item.
@zach-klippenstein do you have any idea what causes that divider bug? I've been trying to figure it out, but no progress thus far
I didnt really want dividers anyway so I just used Color.Transparent instead https://gist.github.com/zach-klippenstein/7ae8874db304f957d6bb91263e292117#file-_segmentedcontrol-kt-L216 and now the vertical bar is gone.
I agreed with @sphrak, just deleted all related to dividers and it works perfectly, thanks so much
Corrected Dividers. It seems the problem is gone.
@Composable
private fun Dividers(state: SegmentedControlState) {
// Animate each divider independently.
val alphas = (1 until state.segmentCount).map { i ->
val selectionAdjacent = i == state.selectedSegment || i - 1 == state.selectedSegment
animateFloatAsState(if (selectionAdjacent) 0f else 1f)
}
Canvas(Modifier.fillMaxSize()) {
val segmentWidth = size.width / state.segmentCount
val dividerPadding = TRACK_PADDING + PRESSED_TRACK_PADDING
alphas.forEachIndexed { i, alpha ->
val x = (i + 1) * segmentWidth
drawLine(
Color.White,
alpha = alpha.value,
start = Offset(x, dividerPadding.toPx()),
end = Offset(x, size.height - dividerPadding.toPx())
)
}
}
}
how to add swipe like pager between segments🙏
Here's what it looks like:
