Trình đơn

Trình đơn thả xuống cho phép người dùng nhấp vào một biểu tượng, trường văn bản hoặc thành phần khác, sau đó chọn trong danh sách các tuỳ chọn trên một nền tảng tạm thời. Hướng dẫn này mô tả cách tạo cả trình đơn cơ bản và trình đơn phức tạp hơn bằng các đường phân chia và biểu tượng.

Một trình đơn thả xuống có hai tuỳ chọn hiển thị. Biểu tượng có ba dấu chấm dọc cho biết rằng khi nhấp vào biểu tượng đó, trình đơn sẽ mở ra.
Hình 1. Một trình đơn thả xuống cơ bản có hai mục được liệt kê.

Nền tảng API

Sử dụng các thành phần DropdownMenu, DropdownMenuItemIconButton để triển khai trình đơn thả xuống tuỳ chỉnh. Các thành phần DropdownMenuDropdownMenuItem được dùng để hiển thị các mục trong trình đơn, còn IconButton là trình kích hoạt để hiển thị hoặc ẩn trình đơn thả xuống.

Các tham số chính cho thành phần DropdownMenu bao gồm:

  • expanded: Cho biết trình đơn có hiển thị hay không.
  • onDismissRequest: Dùng để xử lý việc đóng trình đơn.
  • content: Nội dung có thể kết hợp của trình đơn, thường chứa các thành phần kết hợp DropdownMenuItem.

Các tham số chính cho DropdownMenuItem bao gồm:

  • text: Xác định nội dung hiển thị trong mục trình đơn.
  • onClick: Lệnh gọi lại để xử lý hoạt động tương tác với mục trong trình đơn.

Tạo trình đơn thả xuống cơ bản

Đoạn mã sau đây minh hoạ cách triển khai DropdownMenu tối thiểu:

@Composable
fun MinimalDropdownMenu() {
    var expanded by remember { mutableStateOf(false) }
    Box(
        modifier = Modifier
            .padding(16.dp)
    ) {
        IconButton(onClick = { expanded = !expanded }) {
            Icon(Icons.Default.MoreVert, contentDescription = "More options")
        }
        DropdownMenu(
            expanded = expanded,
            onDismissRequest = { expanded = false }
        ) {
            DropdownMenuItem(
                text = { Text("Option 1") },
                onClick = { /* Do something... */ }
            )
            DropdownMenuItem(
                text = { Text("Option 2") },
                onClick = { /* Do something... */ }
            )
        }
    }
}

Các điểm chính về mã

  • Xác định một DropdownMenu cơ bản chứa hai mục trong trình đơn.
  • Tham số expanded kiểm soát chế độ hiển thị của trình đơn là mở rộng hoặc thu gọn.
  • Tham số onDismissRequest xác định một lệnh gọi lại sẽ thực thi khi người dùng đóng trình đơn.
  • Thành phần kết hợp DropdownMenuItem đại diện cho các mục có thể chọn trong trình đơn thả xuống.
  • IconButton kích hoạt việc mở rộng và thu gọn trình đơn.

Kết quả

Trình đơn thả xuống được kích hoạt bằng một biểu tượng có ba dấu chấm dọc. Trình đơn hiển thị hai tuỳ chọn có thể chọn là Tuỳ chọn 1 và Tuỳ chọn 2.
Hình 2. Một trình đơn thả xuống tối giản chỉ có hai tuỳ chọn.

Tạo trình đơn thả xuống dài hơn

DropdownMenu có thể cuộn theo mặc định nếu không thể hiển thị tất cả các mục trong trình đơn cùng một lúc. Đoạn mã sau đây tạo một trình đơn thả xuống có thể cuộn dài hơn:

@Composable
fun LongBasicDropdownMenu() {
    var expanded by remember { mutableStateOf(false) }
    // Placeholder list of 100 strings for demonstration
    val menuItemData = List(100) { "Option ${it + 1}" }

    Box(
        modifier = Modifier
            .padding(16.dp)
    ) {
        IconButton(onClick = { expanded = !expanded }) {
            Icon(Icons.Default.MoreVert, contentDescription = "More options")
        }
        DropdownMenu(
            expanded = expanded,
            onDismissRequest = { expanded = false }
        ) {
            menuItemData.forEach { option ->
                DropdownMenuItem(
                    text = { Text(option) },
                    onClick = { /* Do something... */ }
                )
            }
        }
    }
}

Các điểm chính về mã

  • DropdownMenu có thể cuộn được khi tổng chiều cao của nội dung vượt quá không gian có sẵn. Mã này tạo một DropdownMenu có thể cuộn hiển thị 100 mục phần giữ chỗ.
  • Vòng lặp forEach tạo động các thành phần kết hợp DropdownMenuItem. Các mục này không được tạo từng phần, tức là tất cả 100 mục trình đơn thả xuống đều được tạo và tồn tại trong thành phần.
  • IconButton kích hoạt việc mở rộng và thu gọn DropdownMenu khi được nhấp vào.
  • Lambda onClick trong mỗi DropdownMenuItem cho phép bạn xác định hành động được thực hiện khi người dùng chọn một mục trong trình đơn.

Kết quả

Đoạn mã trước tạo ra trình đơn cuộn sau:

Một trình đơn thả xuống có nhiều tuỳ chọn, yêu cầu cuộn để xem tất cả các mục.
Hình 3. Một trình đơn thả xuống dài, có thể cuộn.

Tạo trình đơn thả xuống dài hơn bằng các đường phân chia

Đoạn mã sau đây cho thấy cách triển khai trình đơn thả xuống nâng cao hơn. Trong đoạn mã này, các biểu tượng đầu và cuối được thêm vào các mục trong trình đơn và các đường phân chia tách biệt các nhóm mục trong trình đơn.

@Composable
fun DropdownMenuWithDetails() {
    var expanded by remember { mutableStateOf(false) }

    Box(
        modifier = Modifier
            .fillMaxWidth()
            .padding(16.dp)
    ) {
        IconButton(onClick = { expanded = !expanded }) {
            Icon(Icons.Default.MoreVert, contentDescription = "More options")
        }
        DropdownMenu(
            expanded = expanded,
            onDismissRequest = { expanded = false }
        ) {
            // First section
            DropdownMenuItem(
                text = { Text("Profile") },
                leadingIcon = { Icon(Icons.Outlined.Person, contentDescription = null) },
                onClick = { /* Do something... */ }
            )
            DropdownMenuItem(
                text = { Text("Settings") },
                leadingIcon = { Icon(Icons.Outlined.Settings, contentDescription = null) },
                onClick = { /* Do something... */ }
            )

            HorizontalDivider()

            // Second section
            DropdownMenuItem(
                text = { Text("Send Feedback") },
                leadingIcon = { Icon(Icons.Outlined.Feedback, contentDescription = null) },
                trailingIcon = { Icon(Icons.AutoMirrored.Outlined.Send, contentDescription = null) },
                onClick = { /* Do something... */ }
            )

            HorizontalDivider()

            // Third section
            DropdownMenuItem(
                text = { Text("About") },
                leadingIcon = { Icon(Icons.Outlined.Info, contentDescription = null) },
                onClick = { /* Do something... */ }
            )
            DropdownMenuItem(
                text = { Text("Help") },
                leadingIcon = { Icon(Icons.AutoMirrored.Outlined.Help, contentDescription = null) },
                trailingIcon = { Icon(Icons.AutoMirrored.Outlined.OpenInNew, contentDescription = null) },
                onClick = { /* Do something... */ }
            )
        }
    }
}

Mã này xác định DropdownMenu trong Box.

Các điểm chính về mã

  • Tham số leadingIcontrailingIcon thêm các biểu tượng vào đầu và cuối DropdownMenuItem.
  • IconButton kích hoạt việc mở rộng trình đơn.
  • DropdownMenu chứa một số thành phần kết hợp DropdownMenuItem, mỗi thành phần đại diện cho một thao tác có thể chọn.
  • Thành phần kết hợp HorizontalDivider chèn một đường kẻ ngang để phân tách các nhóm mục trong trình đơn.

Kết quả

Đoạn mã trước tạo một trình đơn thả xuống có biểu tượng và đường phân chia:

Một trình đơn thả xuống có các mục Hồ sơ, Cài đặt, Gửi ý kiến phản hồi, Giới thiệu và
Hình 4. Một trình đơn thả xuống được chia thành các phần có biểu tượng ở đầu và cuối.

Tài nguyên khác