parent
8d4a9e519e
commit
8dccdc56c4
@ -0,0 +1,3 @@
|
|||||||
|
description: This file stores settings for Dart & Flutter DevTools.
|
||||||
|
documentation: https://docs.flutter.dev/tools/devtools/extensions#configure-extension-enablement-states
|
||||||
|
extensions:
|
@ -0,0 +1,30 @@
|
|||||||
|
// api/api_collection.dart
|
||||||
|
|
||||||
|
import 'package:dio/dio.dart';
|
||||||
|
import 'package:music_player_miao/models/universal_bean.dart';
|
||||||
|
|
||||||
|
const String _CollectionURL = 'http://flyingpig.fun:10010/collections';
|
||||||
|
|
||||||
|
class CollectionApiMusic {
|
||||||
|
final Dio dio = Dio();
|
||||||
|
|
||||||
|
/// 添加收藏
|
||||||
|
Future<UniversalBean> addCollection({
|
||||||
|
required int musicId,
|
||||||
|
required String Authorization,
|
||||||
|
}) async {
|
||||||
|
|
||||||
|
Response response = await dio.post(
|
||||||
|
_CollectionURL,
|
||||||
|
queryParameters: {'musicId': musicId},
|
||||||
|
options: Options(headers: {
|
||||||
|
'Authorization': Authorization,
|
||||||
|
'Content-Type': 'application/json;charset=UTF-8'
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
print(response.data);
|
||||||
|
return UniversalBean.formMap(response.data); // 将返回的数据转换为 UniversalBean 对象
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -1,211 +1,260 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import '../api/api_music_likes.dart'; // 导入点赞 API
|
||||||
|
import '../api/api_collection.dart'; // 导入收藏 API
|
||||||
|
import '../view_model/comment_page.dart'; // 导入评论页面
|
||||||
|
|
||||||
class RankSongsRow extends StatelessWidget {
|
class RankSongsRow extends StatelessWidget {
|
||||||
final Map sObj;
|
final Map sObj;
|
||||||
final VoidCallback onPressedPlay;
|
final VoidCallback onPressedPlay;
|
||||||
final VoidCallback onPressed;
|
final VoidCallback onPressed;
|
||||||
final String? rank;
|
final String? rank;
|
||||||
|
|
||||||
const RankSongsRow({
|
const RankSongsRow({
|
||||||
super.key,
|
super.key,
|
||||||
required this.sObj,
|
required this.sObj,
|
||||||
required this.onPressed,
|
required this.onPressed,
|
||||||
required this.onPressedPlay, required this.rank,
|
required this.onPressedPlay,
|
||||||
|
required this.rank,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
rank: sObj["rank"];
|
rank: sObj["rank"];
|
||||||
return
|
return Column(
|
||||||
Column(
|
children: [
|
||||||
children: [
|
Row(
|
||||||
Row(
|
children: [
|
||||||
// mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
Row(
|
||||||
children: [
|
children: [
|
||||||
Row(
|
SizedBox(
|
||||||
children: [
|
width: 15,
|
||||||
SizedBox(
|
child: RichText(
|
||||||
width: 15,
|
text: TextSpan(
|
||||||
child: RichText(
|
text: sObj["rank"],
|
||||||
text: TextSpan(
|
style: TextStyle(
|
||||||
text: sObj["rank"],
|
fontSize: getRankFontSize(sObj["rank"]),
|
||||||
style: TextStyle(
|
fontWeight: FontWeight.w700,
|
||||||
fontSize: getRankFontSize(sObj["rank"]),
|
color: Color(0xffCE0000),
|
||||||
fontWeight: FontWeight.w700,
|
|
||||||
color: Color(0xffCE0000),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(width: 10,),
|
),
|
||||||
ClipRRect(
|
const SizedBox(width: 10),
|
||||||
borderRadius: BorderRadius.circular(10),
|
ClipRRect(
|
||||||
child: Image.asset(
|
borderRadius: BorderRadius.circular(10),
|
||||||
sObj["image"],
|
child: Image.asset(
|
||||||
width: 80,
|
sObj["image"],
|
||||||
height: 80,
|
width: 80,
|
||||||
fit: BoxFit.cover,
|
height: 80,
|
||||||
),
|
fit: BoxFit.cover,
|
||||||
),
|
),
|
||||||
|
),
|
||||||
const SizedBox(width: 20,),
|
const SizedBox(width: 20),
|
||||||
SizedBox(
|
SizedBox(
|
||||||
width: 170,
|
width: 170,
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
sObj["name"],
|
sObj["name"],
|
||||||
maxLines: 1,
|
maxLines: 1,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
color: Colors.black,
|
color: Colors.black,
|
||||||
fontSize: 16,
|
fontSize: 16,
|
||||||
fontWeight: FontWeight.w400),
|
fontWeight: FontWeight.w400,
|
||||||
),
|
),
|
||||||
Text(
|
),
|
||||||
sObj["artists"],
|
Text(
|
||||||
maxLines: 1,
|
sObj["artists"],
|
||||||
style: TextStyle(color: Colors.black, fontSize: 14),
|
maxLines: 1,
|
||||||
)
|
style: TextStyle(color: Colors.black, fontSize: 14),
|
||||||
],
|
)
|
||||||
),
|
],
|
||||||
),
|
),
|
||||||
const SizedBox(width: 20,),
|
|
||||||
|
|
||||||
|
|
||||||
],
|
|
||||||
),
|
|
||||||
IconButton(
|
|
||||||
onPressed: (){
|
|
||||||
_bottomSheet(context);
|
|
||||||
},
|
|
||||||
icon: Image.asset(
|
|
||||||
"assets/img/More.png",
|
|
||||||
width: 25,
|
|
||||||
height: 25,
|
|
||||||
|
|
||||||
),
|
),
|
||||||
|
const SizedBox(width: 20),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
_bottomSheet(context);
|
||||||
|
},
|
||||||
|
icon: Image.asset(
|
||||||
|
"assets/img/More.png",
|
||||||
|
width: 25,
|
||||||
|
height: 25,
|
||||||
),
|
),
|
||||||
const SizedBox(height: 20,)
|
),
|
||||||
],
|
const SizedBox(height: 20)
|
||||||
),
|
],
|
||||||
const SizedBox(height: 10,)
|
),
|
||||||
],
|
const SizedBox(height: 10)
|
||||||
|
],
|
||||||
);
|
);
|
||||||
|
|
||||||
}
|
}
|
||||||
getRankFontSize(String rank) {
|
|
||||||
|
double getRankFontSize(String rank) {
|
||||||
switch (rank) {
|
switch (rank) {
|
||||||
case '1': case '2':case '3':
|
case '1':
|
||||||
return 30.0;
|
case '2':
|
||||||
|
case '3':
|
||||||
|
return 30.0;
|
||||||
default:
|
default:
|
||||||
return 20.0;
|
return 20.0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Future _bottomSheet(BuildContext context){
|
|
||||||
|
Future _bottomSheet(BuildContext context) {
|
||||||
return showModalBottomSheet(
|
return showModalBottomSheet(
|
||||||
context: context,
|
context: context,
|
||||||
backgroundColor: Colors.white,
|
backgroundColor: Colors.white,
|
||||||
shape: const RoundedRectangleBorder(borderRadius: BorderRadius.vertical(top: Radius.circular(30))),
|
shape: const RoundedRectangleBorder(
|
||||||
builder: (context) =>Container(
|
borderRadius: BorderRadius.vertical(top: Radius.circular(30))),
|
||||||
height: 210,
|
builder: (context) => Container(
|
||||||
padding: const EdgeInsets.only(top: 20),
|
height: 210,
|
||||||
child: Column(
|
padding: const EdgeInsets.only(top: 20),
|
||||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
child: Column(
|
||||||
children: [
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
Row(
|
children: [
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
Row(
|
||||||
children: [
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
Column(
|
children: [
|
||||||
children: [
|
Column(
|
||||||
IconButton(
|
children: [
|
||||||
onPressed: (){},
|
IconButton(
|
||||||
icon: Image.asset("assets/img/list_add.png"),
|
onPressed: () {
|
||||||
iconSize: 60,
|
// TODO: 加入歌单的逻辑
|
||||||
),
|
},
|
||||||
Text("加入歌单")
|
icon: Image.asset("assets/img/list_add.png"),
|
||||||
],
|
iconSize: 60,
|
||||||
),
|
),
|
||||||
Column(
|
Text("加入歌单"),
|
||||||
children: [
|
],
|
||||||
IconButton(
|
|
||||||
onPressed: (){},
|
|
||||||
icon: Image.asset("assets/img/list_download.png"),
|
|
||||||
iconSize: 60,
|
|
||||||
),
|
|
||||||
Text("下载")
|
|
||||||
],
|
|
||||||
),
|
|
||||||
Column(
|
|
||||||
children: [
|
|
||||||
IconButton(
|
|
||||||
onPressed: (){},
|
|
||||||
icon: Image.asset("assets/img/list_collection.png"),
|
|
||||||
iconSize: 60,
|
|
||||||
),
|
|
||||||
Text("收藏")
|
|
||||||
],
|
|
||||||
),
|
|
||||||
Column(
|
|
||||||
children: [
|
|
||||||
IconButton(
|
|
||||||
onPressed: (){},
|
|
||||||
icon: Image.asset("assets/img/list_good.png"),
|
|
||||||
iconSize: 60,
|
|
||||||
),
|
|
||||||
Text("点赞")
|
|
||||||
],
|
|
||||||
),
|
|
||||||
Column(
|
|
||||||
children: [
|
|
||||||
IconButton(
|
|
||||||
onPressed: (){},
|
|
||||||
icon: Image.asset("assets/img/list_comment.png"),
|
|
||||||
iconSize: 60,
|
|
||||||
),
|
|
||||||
Text("评论")
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10,),
|
|
||||||
ElevatedButton(
|
|
||||||
onPressed: () {
|
|
||||||
// Get.to(()=>const MainTabView());
|
|
||||||
},
|
|
||||||
child: Text(
|
|
||||||
"查看详情页",
|
|
||||||
style: const TextStyle(color:Colors.black,fontSize: 18),
|
|
||||||
),
|
),
|
||||||
style: ElevatedButton.styleFrom(
|
Column(
|
||||||
backgroundColor: const Color(0xffE6F4F1),
|
children: [
|
||||||
padding: const EdgeInsets.symmetric(vertical: 8),
|
IconButton(
|
||||||
tapTargetSize: MaterialTapTargetSize.shrinkWrap,
|
onPressed: () {
|
||||||
shape: RoundedRectangleBorder(
|
// TODO: 下载的逻辑
|
||||||
borderRadius: BorderRadius.zero,
|
},
|
||||||
),
|
icon: Image.asset("assets/img/list_download.png"),
|
||||||
|
iconSize: 60,
|
||||||
|
),
|
||||||
|
Text("下载"),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
|
Column(
|
||||||
|
children: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: () async {
|
||||||
|
// 点击收藏按钮时调用收藏 API
|
||||||
|
await _toggleCollect();
|
||||||
|
},
|
||||||
|
icon: Image.asset("assets/img/list_collection.png"),
|
||||||
|
iconSize: 60,
|
||||||
|
),
|
||||||
|
Text("收藏"),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
Column(
|
||||||
|
children: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: () async {
|
||||||
|
// 点击点赞按钮时调用点赞 API
|
||||||
|
await _toggleLike();
|
||||||
|
},
|
||||||
|
icon: Image.asset("assets/img/list_good.png"),
|
||||||
|
iconSize: 60,
|
||||||
|
),
|
||||||
|
Text("点赞"),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
Column(
|
||||||
|
children: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
// 跳转到评论页面
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => const CommentPage(),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
icon: Image.asset("assets/img/list_comment.png"),
|
||||||
|
iconSize: 60,
|
||||||
|
),
|
||||||
|
Text("评论"),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(height: 10),
|
||||||
|
ElevatedButton(
|
||||||
|
onPressed: () {
|
||||||
|
// TODO: 查看详情页的逻辑
|
||||||
|
},
|
||||||
|
child: Text(
|
||||||
|
"查看详情页",
|
||||||
|
style: const TextStyle(color: Colors.black, fontSize: 18),
|
||||||
),
|
),
|
||||||
ElevatedButton(
|
style: ElevatedButton.styleFrom(
|
||||||
onPressed: () =>Navigator.pop(context),
|
backgroundColor: const Color(0xffE6F4F1),
|
||||||
child: Text(
|
padding: const EdgeInsets.symmetric(vertical: 8),
|
||||||
"取消",
|
tapTargetSize: MaterialTapTargetSize.shrinkWrap,
|
||||||
style: const TextStyle(color:Colors.black,fontSize: 18),
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.zero,
|
||||||
),
|
),
|
||||||
style: ElevatedButton.styleFrom(
|
),
|
||||||
backgroundColor: const Color(0xff429482),
|
),
|
||||||
padding: const EdgeInsets.symmetric(vertical: 8),
|
ElevatedButton(
|
||||||
tapTargetSize: MaterialTapTargetSize.shrinkWrap,
|
onPressed: () => Navigator.pop(context),
|
||||||
shape: RoundedRectangleBorder(
|
child: Text(
|
||||||
borderRadius: BorderRadius.zero,
|
"取消",
|
||||||
),
|
style: const TextStyle(color: Colors.black, fontSize: 18),
|
||||||
|
),
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: const Color(0xff429482),
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 8),
|
||||||
|
tapTargetSize: MaterialTapTargetSize.shrinkWrap,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.zero,
|
||||||
),
|
),
|
||||||
|
|
||||||
),
|
),
|
||||||
],
|
),
|
||||||
),
|
],
|
||||||
)
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 点赞功能
|
||||||
|
Future<void> _toggleLike() async {
|
||||||
|
final api = LikesApiMusic(); // 实例化点赞 API
|
||||||
|
try {
|
||||||
|
String authorizationToken = 'eyJhbGciOiJIUzI1NiJ9.eyJqdGkiOiI1ZDBmY2Q3ZThlYmY0N2QzOThlNmVjNDQ0ZTM5NTAxNSIsInN1YiI6IjEiLCJpc3MiOiJmbHlpbmdwaWciLCJpYXQiOjE3MzEwNDM3NTgsImV4cCI6MTczMzYzNTc1OH0.5jfhZtK46YNSC7KCaBWiPxSLO7Ym6ntBXnQwfsvMrCw'; // 替换为实际的授权 Token
|
||||||
|
await api.likesMusic(
|
||||||
|
musicId: sObj['id'], // 使用当前音乐的 ID
|
||||||
|
Authorization: authorizationToken,
|
||||||
|
);
|
||||||
|
print('Liked music ID: ${sObj['id']}');
|
||||||
|
} catch (e) {
|
||||||
|
print('Error liking music: $e');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 收藏功能
|
||||||
|
Future<void> _toggleCollect() async {
|
||||||
|
final api = CollectionApiMusic(); // 实例化收藏 API
|
||||||
|
try {
|
||||||
|
String authorizationToken = 'eyJhbGciOiJIUzI1NiJ9.eyJqdGkiOiI1ZDBmY2Q3ZThlYmY0N2QzOThlNmVjNDQ0ZTM5NTAxNSIsInN1YiI6IjEiLCJpc3MiOiJmbHlpbmdwaWciLCJpYXQiOjE3MzEwNDM3NTgsImV4cCI6MTczMzYzNTc1OH0.5jfhZtK46YNSC7KCaBWiPxSLO7Ym6ntBXnQwfsvMrCw'; // 替换为实际的授权 Token
|
||||||
|
await api.addCollection(
|
||||||
|
musicId: sObj['id'], // 使用当前音乐的 ID
|
||||||
|
Authorization: authorizationToken,
|
||||||
|
);
|
||||||
|
print('Collected music ID: ${sObj['id']}');
|
||||||
|
} catch (e) {
|
||||||
|
print('Error collecting music: $e');
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,76 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
class CommentPage extends StatefulWidget {
|
||||||
|
const CommentPage({Key? key}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
_CommentPageState createState() => _CommentPageState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CommentPageState extends State<CommentPage> {
|
||||||
|
final TextEditingController _controller = TextEditingController();
|
||||||
|
final List<String> _comments = ["这是第一个评论", "很喜欢这首歌!"]; // 初始评论示例
|
||||||
|
|
||||||
|
void _addComment() {
|
||||||
|
if (_controller.text.isNotEmpty) {
|
||||||
|
setState(() {
|
||||||
|
_comments.insert(0, _controller.text); // 插入到顶部
|
||||||
|
});
|
||||||
|
_controller.clear(); // 清空输入框
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: const Text('评论'),
|
||||||
|
),
|
||||||
|
body: Column(
|
||||||
|
children: [
|
||||||
|
// 评论列表区域
|
||||||
|
Expanded(
|
||||||
|
child: ListView.builder(
|
||||||
|
reverse: true, // 新评论显示在顶部
|
||||||
|
itemCount: _comments.length,
|
||||||
|
itemBuilder: (context, index) {
|
||||||
|
return ListTile(
|
||||||
|
leading: const CircleAvatar(
|
||||||
|
child: Icon(Icons.person),
|
||||||
|
),
|
||||||
|
title: Text(_comments[index]),
|
||||||
|
subtitle: Text('${DateTime.now().difference(DateTime.now().subtract(Duration(minutes: index * 5))).inMinutes} 分钟前'),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Divider(),
|
||||||
|
// 评论输入区域
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.all(8.0),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Expanded(
|
||||||
|
child: TextField(
|
||||||
|
controller: _controller,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
hintText: '输入你的评论...',
|
||||||
|
border: OutlineInputBorder(
|
||||||
|
borderRadius: BorderRadius.circular(10.0),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
onSubmitted: (_) => _addComment(), // 按回车提交评论
|
||||||
|
),
|
||||||
|
),
|
||||||
|
IconButton(
|
||||||
|
icon: const Icon(Icons.send, color: Colors.blue),
|
||||||
|
onPressed: _addComment,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue