|
| 1 | +/* |
| 2 | + * Copyright (c) SRG SSR. All rights reserved. |
| 3 | + * License information is available from the LICENSE file. |
| 4 | + */ |
| 5 | +package ch.srgssr.pillarbox.demo.ui.showcases.misc |
| 6 | + |
| 7 | +import androidx.compose.foundation.layout.Arrangement |
| 8 | +import androidx.compose.foundation.layout.Column |
| 9 | +import androidx.compose.foundation.layout.Row |
| 10 | +import androidx.compose.foundation.layout.Spacer |
| 11 | +import androidx.compose.foundation.layout.aspectRatio |
| 12 | +import androidx.compose.foundation.layout.fillMaxSize |
| 13 | +import androidx.compose.foundation.layout.fillMaxWidth |
| 14 | +import androidx.compose.foundation.layout.padding |
| 15 | +import androidx.compose.foundation.layout.size |
| 16 | +import androidx.compose.foundation.text.input.TextFieldLineLimits |
| 17 | +import androidx.compose.foundation.text.input.rememberTextFieldState |
| 18 | +import androidx.compose.foundation.text.input.setTextAndPlaceCursorAtEnd |
| 19 | +import androidx.compose.material.icons.Icons |
| 20 | +import androidx.compose.material.icons.filled.Audiotrack |
| 21 | +import androidx.compose.material.icons.filled.Videocam |
| 22 | +import androidx.compose.material3.DropdownMenuItem |
| 23 | +import androidx.compose.material3.ExperimentalMaterial3Api |
| 24 | +import androidx.compose.material3.ExposedDropdownMenuAnchorType |
| 25 | +import androidx.compose.material3.ExposedDropdownMenuBox |
| 26 | +import androidx.compose.material3.ExposedDropdownMenuDefaults |
| 27 | +import androidx.compose.material3.Icon |
| 28 | +import androidx.compose.material3.MaterialTheme |
| 29 | +import androidx.compose.material3.OutlinedIconToggleButton |
| 30 | +import androidx.compose.material3.Text |
| 31 | +import androidx.compose.material3.TextField |
| 32 | +import androidx.compose.runtime.Composable |
| 33 | +import androidx.compose.runtime.getValue |
| 34 | +import androidx.compose.runtime.mutableStateOf |
| 35 | +import androidx.compose.runtime.remember |
| 36 | +import androidx.compose.runtime.setValue |
| 37 | +import androidx.compose.ui.Alignment |
| 38 | +import androidx.compose.ui.Modifier |
| 39 | +import androidx.lifecycle.viewmodel.compose.viewModel |
| 40 | +import ch.srgssr.pillarbox.demo.ui.player.PlayerView |
| 41 | +import ch.srgssr.pillarbox.demo.ui.theme.paddings |
| 42 | + |
| 43 | +/** |
| 44 | + * A showcase demonstrating how switching audio / video live. |
| 45 | + */ |
| 46 | +@Composable |
| 47 | +fun AudioVideoSwitchShowcase(audioVideoLiveSwitchViewModel: AudioVideoLiveSwitchViewModel = viewModel()) { |
| 48 | + val player = audioVideoLiveSwitchViewModel.player |
| 49 | + Column(modifier = Modifier.fillMaxSize(), verticalArrangement = Arrangement.spacedBy(MaterialTheme.paddings.medium)) { |
| 50 | + PlayerView( |
| 51 | + player = player, |
| 52 | + modifier = Modifier |
| 53 | + .fillMaxWidth() |
| 54 | + .aspectRatio(16 / 9f) |
| 55 | + ) |
| 56 | + Row( |
| 57 | + modifier = Modifier.fillMaxWidth().padding(horizontal = MaterialTheme.paddings.baseline), |
| 58 | + verticalAlignment = Alignment.CenterVertically, |
| 59 | + horizontalArrangement = Arrangement.End |
| 60 | + ) { |
| 61 | + ContentTypeToggleButton( |
| 62 | + currentContentType = audioVideoLiveSwitchViewModel.currentContentType, |
| 63 | + toggle = audioVideoLiveSwitchViewModel::toggleContentType |
| 64 | + ) |
| 65 | + Spacer(modifier = Modifier.size(MaterialTheme.paddings.baseline)) |
| 66 | + ContentSelector( |
| 67 | + contents = AudioVideoLiveSwitchViewModel.contents, |
| 68 | + currentContent = audioVideoLiveSwitchViewModel.currentContent, |
| 69 | + loadContent = audioVideoLiveSwitchViewModel::load, |
| 70 | + ) |
| 71 | + } |
| 72 | + } |
| 73 | +} |
| 74 | + |
| 75 | +@OptIn(ExperimentalMaterial3Api::class) |
| 76 | +@Composable |
| 77 | +private fun ContentSelector( |
| 78 | + contents: List<AudioVideoLiveSwitchViewModel.LiveContent>, |
| 79 | + currentContent: AudioVideoLiveSwitchViewModel.LiveContent?, |
| 80 | + loadContent: (AudioVideoLiveSwitchViewModel.LiveContent) -> Unit, |
| 81 | + modifier: Modifier = Modifier |
| 82 | +) { |
| 83 | + val textFieldState = rememberTextFieldState(currentContent?.label ?: "Select") |
| 84 | + var expanded by remember { mutableStateOf(false) } |
| 85 | + ExposedDropdownMenuBox(modifier = modifier, expanded = expanded, onExpandedChange = { |
| 86 | + expanded = it |
| 87 | + }) { |
| 88 | + TextField( |
| 89 | + // The `menuAnchor` modifier must be passed to the text field to handle |
| 90 | + // expanding/collapsing the menu on click. A read-only text field has |
| 91 | + // the anchor type `PrimaryNotEditable`. |
| 92 | + modifier = Modifier.menuAnchor(ExposedDropdownMenuAnchorType.PrimaryNotEditable), |
| 93 | + state = textFieldState, |
| 94 | + readOnly = true, |
| 95 | + lineLimits = TextFieldLineLimits.SingleLine, |
| 96 | + label = { Text("Channel") }, |
| 97 | + trailingIcon = { ExposedDropdownMenuDefaults.TrailingIcon(expanded = expanded) }, |
| 98 | + colors = ExposedDropdownMenuDefaults.textFieldColors(), |
| 99 | + ) |
| 100 | + ExposedDropdownMenu(expanded = expanded, onDismissRequest = { expanded = false }) { |
| 101 | + contents.forEach { content -> |
| 102 | + DropdownMenuItem( |
| 103 | + text = { Text(content.label, style = MaterialTheme.typography.bodyLarge) }, |
| 104 | + onClick = { |
| 105 | + textFieldState.setTextAndPlaceCursorAtEnd(content.label) |
| 106 | + expanded = false |
| 107 | + loadContent(content) |
| 108 | + }, |
| 109 | + contentPadding = ExposedDropdownMenuDefaults.ItemContentPadding, |
| 110 | + ) |
| 111 | + } |
| 112 | + } |
| 113 | + } |
| 114 | +} |
| 115 | + |
| 116 | +@Composable |
| 117 | +private fun ContentTypeToggleButton( |
| 118 | + currentContentType: AudioVideoLiveSwitchViewModel.ContentType, |
| 119 | + toggle: () -> Unit, |
| 120 | + modifier: Modifier = Modifier, |
| 121 | +) { |
| 122 | + OutlinedIconToggleButton( |
| 123 | + modifier = modifier, |
| 124 | + checked = currentContentType == AudioVideoLiveSwitchViewModel.ContentType.Video, |
| 125 | + onCheckedChange = { |
| 126 | + toggle() |
| 127 | + } |
| 128 | + ) { |
| 129 | + val icon = when (currentContentType) { |
| 130 | + AudioVideoLiveSwitchViewModel.ContentType.Video -> Icons.Default.Videocam |
| 131 | + AudioVideoLiveSwitchViewModel.ContentType.Audio -> Icons.Default.Audiotrack |
| 132 | + } |
| 133 | + Icon(icon, "Toggle video or audio") |
| 134 | + } |
| 135 | +} |
0 commit comments