123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678 |
- import 'dart:convert';
- import 'package:fis_jsonrpc/rpc.dart';
- import 'package:flutter/material.dart';
- import 'package:get/get.dart';
- import 'package:vnoteapp/components/dialog_input.dart';
- import 'package:vnoteapp/components/dialog_number.dart';
- import 'package:vnoteapp/components/dynamic_drawer.dart';
- import 'package:vnoteapp/managers/interfaces/template.dart';
- import 'package:vnoteapp/pages/check/models/form.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_blood_pressure.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_body_temperature.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_check_box.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_input.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_number_input.dart';
- import 'dart:math' as math;
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_radio.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_radio_score.dart';
- const double _width = 170;
- const double _height = 38;
- String tw = '00.0';
- class ConfigurableCard extends StatefulWidget {
- final String cardKey;
- final Function(String, String, dynamic) callBack;
- const ConfigurableCard({
- super.key,
- required this.cardKey,
- required this.callBack,
- });
- @override
- State<ConfigurableCard> createState() => _ConfigurableFormState();
- }
- class _ConfigurableFormState extends State<ConfigurableCard> {
- /// 当前最新的模板的键值对
- Map<String, dynamic> templateRelation = {};
- /// 当前模板数据
- List<FormObject> currentTemplate = [];
- /// 当前title的下标
- int currentTitleIndex = 0;
- Map<String, dynamic> formValue = {};
- var scaffoldKey = GlobalKey<ScaffoldState>();
- final _templateManager = Get.find<ITemplateManager>();
- final arrowHeight = math.tan(120 / 180) * (_height / 2);
- @override
- void initState() {
- super.initState();
- fetchTemplateIndex();
- }
- @override
- void dispose() {
- super.dispose();
- }
- Future<void> fetchTemplateIndex() async {
- try {
- /// 获取模板的键值对
- String? templates;
- templates = await _templateManager.readTemplate('templateRelation');
- templateRelation = jsonDecode(templates!);
- fetchTemplate(widget.cardKey);
- } catch (error) {
- print('发生错误: $error');
- }
- }
- Future<void> fetchTemplate(String key) async {
- try {
- if (templateRelation[key] == null) {
- currentTemplate = [];
- setState(() {});
- return;
- }
- var template =
- await _templateManager.readTemplate(templateRelation[key]!);
- String templateContent =
- TemplateDTO.fromJson(jsonDecode(template!)).templateContent!;
- List<Map<String, dynamic>> list =
- jsonDecode(templateContent).cast<Map<String, dynamic>>();
- for (var i in list) {
- if (i['children'] != null) {
- List<FormObject> currentChildren = [];
- for (var j in i['children']) {
- currentChildren.add(FormObject.fromJson(j));
- }
- i['children'] = currentChildren;
- }
- currentTemplate.add(FormObject.fromJson(i));
- }
- // var list = jsonDecode(templateContent).cast<Map<String, dynamic>>();
- // if (list == null) {
- // currentTemplate = [];
- // } else {
- // currentTemplate = updateChildren(list);
- // }
- // TextStorage t = TextStorage(
- // fileName: key,
- // directory: "template",
- // );
- // await t.save(jsonEncode(currentTemplate));
- print(currentTemplate);
- setState(() {});
- } catch (error) {
- print('发生错误: $error');
- }
- }
- @override
- Widget build(BuildContext context) {
- return Scaffold(
- key: scaffoldKey,
- endDrawer: VDynamicDrawerWrapper(scaffoldKey: scaffoldKey),
- resizeToAvoidBottomInset: false,
- body: Column(
- children: [
- Container(
- width: double.infinity,
- padding: const EdgeInsets.all(20),
- child: _buildTitleList(),
- ),
- Expanded(
- child: Stack(
- children: [
- Row(
- mainAxisAlignment: MainAxisAlignment.start,
- crossAxisAlignment: CrossAxisAlignment.start,
- children: [
- _buildDiagram(),
- _buildContent(),
- ],
- ),
- _buildPositionedButton(
- currentTitleIndex == currentTemplate.length - 1
- ? "保存"
- : "下一步",
- () async {
- if (currentTitleIndex == currentTemplate.length - 1) {
- widget
- .callBack(
- widget.cardKey,
- templateRelation[widget.cardKey]!,
- jsonEncode(formValue),
- )
- .call();
- return;
- }
- currentTitleIndex++;
- setState(() {});
- },
- right: 0,
- ),
- ],
- ),
- )
- ],
- ),
- );
- }
- Widget buildSingleItem(Widget item, int span) {
- return FractionallySizedBox(
- widthFactor: span == 24 ? 1 : 0.5,
- child: item,
- );
- }
- Widget buildWidget(FormObject? currentFormObject) {
- Map<String, Widget Function(FormObject)> widgetMap = {
- 'checkbox': _buildCheckBox,
- 'numberInput': _buildNumberInput,
- 'input': _buildInput,
- 'radio': _buildRadio,
- 'radioScore': _buildRadioScore,
- 'bloodPressure': _buildBloodPressure,
- 'bodyTemperature': _buildBodyTemperature,
- };
- Widget Function(FormObject) builder =
- widgetMap[currentFormObject?.type] ?? _buildInput;
- return builder(currentFormObject!);
- }
- Widget waterCardList() {
- int itemCount = 0;
- if (currentTemplate.isNotEmpty) {
- itemCount = currentTemplate[currentTitleIndex].children?.length ?? 0;
- }
- List<Widget> items = List.generate(itemCount, (index) {
- FormObject? currentFormObject =
- currentTemplate[currentTitleIndex].children?[index];
- int span = currentFormObject?.span ?? 12;
- return buildSingleItem(buildWidget(currentFormObject), span);
- });
- return Scrollbar(
- thumbVisibility: true,
- child: SingleChildScrollView(
- child: Container(
- alignment: Alignment.topCenter,
- padding: const EdgeInsets.all(15),
- child: Wrap(
- runSpacing: 20, // 纵向元素间距
- alignment: WrapAlignment.start,
- children: items,
- ),
- ),
- ),
- );
- }
- /// title标签
- Widget _buildTitleList() {
- return Wrap(
- runSpacing: 10, // 设置子小部件之间的间距
- alignment: WrapAlignment.start,
- children: currentTemplate.asMap().entries.map(
- (e) {
- /// TODO 这边需要改下
- MaterialColor currentColors = Colors.grey;
- e.value.children?.forEach((element) {
- if (formValue.containsKey(element.key)) {
- currentColors = Colors.green;
- }
- });
- return Transform.translate(
- offset: Offset((-arrowHeight + 2) * e.key, 0),
- child: TitleClipRect(
- title: e.value.label ?? '',
- color: currentTitleIndex == e.key ? null : currentColors,
- arrowHeight: arrowHeight,
- clickTitle: () {
- currentTitleIndex = e.key;
- setState(() {});
- },
- ),
- );
- },
- ).toList(),
- );
- }
- /// 示意图
- Widget _buildDiagram() {
- return Expanded(
- flex: 1,
- child: Stack(
- children: [
- Image.asset(
- 'assets/images/exam/test.png',
- height: double.infinity,
- fit: BoxFit.fitWidth, // 设置图像的适应方式
- ),
- _buildPositionedButton(
- currentTitleIndex == 0 ? "返回" : "上一步",
- () async {
- if (currentTitleIndex == 0) {
- Get.back();
- } else {
- currentTitleIndex--;
- setState(() {});
- }
- },
- ),
- ],
- ),
- );
- }
- /// 按钮
- Widget _buildPositionedButton(String title, Function onTap,
- {double? right, double? left}) {
- return Positioned(
- right: right,
- left: left,
- bottom: 0,
- child: Container(
- width: 150,
- padding: const EdgeInsets.symmetric(vertical: 30),
- child: Align(
- alignment: Alignment.bottomCenter,
- child: SizedBox(
- width: 100,
- height: 100,
- child: ElevatedButton(
- style: ButtonStyle(
- backgroundColor: MaterialStatePropertyAll(
- Colors.white.withOpacity(
- .8,
- ),
- ),
- shape: MaterialStatePropertyAll(
- RoundedRectangleBorder(
- borderRadius: BorderRadius.circular(50),
- ),
- ),
- ),
- onPressed: () => onTap.call(),
- child: Text(
- title,
- style:
- const TextStyle(fontWeight: FontWeight.bold, fontSize: 16),
- ),
- ),
- ),
- ),
- ),
- );
- }
- /// 主页面
- Widget _buildContent() {
- return Expanded(
- flex: 2,
- child: waterCardList(),
- );
- }
- /// 多选框组件
- Widget _buildCheckBox(FormObject currentFormObject) {
- List<Option> options = currentFormObject.options ?? [];
- List<String> currentSelectedCheckBox =
- formValue[currentFormObject.key!] ?? [];
- void selectCheckBoxChange(Option e) {
- if (currentSelectedCheckBox.contains(e.value)) {
- currentSelectedCheckBox.remove(e.value);
- } else {
- currentSelectedCheckBox.add(e.value ?? '');
- }
- formValue[currentFormObject.key!] = currentSelectedCheckBox;
- setState(() {});
- }
- return ExamCheckBox(
- options: options,
- currentSelectedCheckBox: currentSelectedCheckBox,
- currentFormObject: currentFormObject,
- selectCheckBoxChange: selectCheckBoxChange,
- );
- }
- /// 数字输入框组件
- Widget _buildNumberInput(FormObject currentFormObject) {
- String currentInputValue = formValue[currentFormObject.key!] ?? '';
- Future<void> commonInput() async {
- String? result = await VDialogNumber(
- title: currentFormObject.label,
- initialValue: formValue[currentFormObject.key],
- ).show();
- if (result?.isNotEmpty ?? false) {
- formValue[currentFormObject.key!] = result;
- currentInputValue = formValue[currentFormObject.key!];
- setState(() {});
- }
- }
- void specialInput(String value) {
- formValue[currentFormObject.key!] = value;
- currentInputValue = formValue[currentFormObject.key!];
- setState(() {});
- }
- return ExamNumberInput(
- currentInputValue: currentInputValue,
- commonInput: commonInput,
- specialInput: specialInput,
- currentFormObject: currentFormObject,
- );
- }
- Widget _buildInput(FormObject currentFormObject) {
- String currentInputValue = formValue[currentFormObject.key!] ?? '';
- Future<void> commonInput() async {
- String? result = await VDialogInput(
- title: currentFormObject.label,
- initialValue: formValue[currentFormObject.key],
- ).show();
- if (result?.isNotEmpty ?? false) {
- formValue[currentFormObject.key!] = result;
- currentInputValue = formValue[currentFormObject.key!];
- setState(() {});
- }
- }
- return ExamInput(
- currentInputValue: currentInputValue,
- commonInput: commonInput,
- currentFormObject: currentFormObject,
- );
- }
- /// 血压组件
- Widget _buildBloodPressure(FormObject currentFormObject) {
- String? highPressureKey = currentFormObject.childrenKey?.first;
- String? lowPressureKey = currentFormObject.childrenKey?.last;
- List<String> currentBloodPressureValue = [
- formValue[highPressureKey] ?? '',
- formValue[lowPressureKey] ?? ''
- ];
- Future<void> commonBloodPressure() async {
- String? result = await VDialogBloodPressure(
- title: currentFormObject.label,
- placeholder: '请输入',
- initialValue: currentBloodPressureValue,
- ).show();
- if (result?.isNotEmpty ?? false) {
- List resultList = jsonDecode(result!);
- formValue[highPressureKey!] = resultList.first;
- formValue[lowPressureKey!] = resultList.last;
- currentBloodPressureValue = [
- formValue[highPressureKey] ?? '',
- formValue[lowPressureKey] ?? ''
- ];
- formValue['Left'] = '111';
- setState(() {});
- }
- }
- return ExamBloodPressure(
- currentFormObject: currentFormObject,
- currentBloodPressureValue: currentBloodPressureValue,
- commonBloodPressure: commonBloodPressure,
- );
- }
- /// 单选框组件
- Widget _buildRadio(FormObject currentFormObject) {
- List<Option> options = currentFormObject.options ?? [];
- String currentSelected = formValue[currentFormObject.key!] ?? "";
- void selectRaidoChange(Option e) {
- currentSelected = e.value ?? '';
- formValue[currentFormObject.key!] = currentSelected;
- setState(() {});
- }
- return ExamRadio(
- options: options,
- currentFormObject: currentFormObject,
- selectRaidoChange: selectRaidoChange,
- currentSelected: currentSelected,
- );
- }
- Widget _buildRadioScore(FormObject currentFormObject) {
- print(currentFormObject.toJson());
- List<Option> options = currentFormObject.options ?? [];
- String currentSelected =
- formValue[currentFormObject.childrenKey!.first] ?? "";
- void selectRaidoChange(Option e) {
- currentSelected = e.value ?? '';
- formValue[currentFormObject.childrenKey!.first] = currentSelected;
- setState(() {});
- }
- return ExamRadioScore(
- options: options,
- currentFormObject: currentFormObject,
- selectRaidoChange: selectRaidoChange,
- currentSelected: currentSelected,
- );
- }
- /// 数字输入框组件
- Widget _buildBodyTemperature(FormObject currentFormObject) {
- String currentInputValue = formValue[currentFormObject.key!] ?? '';
- Future<void> commonInput() async {
- String? result = await VDialogNumber(
- title: currentFormObject.label,
- initialValue: formValue[currentFormObject.key],
- ).show();
- if (result?.isNotEmpty ?? false) {
- formValue[currentFormObject.key!] = result;
- currentInputValue = formValue[currentFormObject.key!];
- setState(() {});
- }
- }
- void specialInput(String value) {
- formValue[currentFormObject.key!] = value;
- currentInputValue = formValue[currentFormObject.key!];
- setState(() {});
- }
- return ExamBodyTemperature(
- currentInputValue: currentInputValue,
- commonInput: commonInput,
- specialInput: specialInput,
- currentFormObject: currentFormObject,
- );
- }
- }
- class TitleClipRect extends StatelessWidget {
- const TitleClipRect({
- super.key,
- this.color = Colors.grey,
- required this.title,
- required this.arrowHeight,
- required this.clickTitle,
- });
- final Color? color;
- final String title;
- final double arrowHeight;
- final Function clickTitle;
- @override
- Widget build(BuildContext context) {
- return InkWell(
- onTap: () {
- clickTitle.call();
- },
- customBorder: HoleShapeBorder(arrowHeight),
- child: Card(
- margin: const EdgeInsets.all(0),
- shape: HoleShapeBorder(arrowHeight),
- color: color,
- elevation: color == null ? 5 : 0,
- shadowColor: Theme.of(context).primaryColor.withOpacity(0.8),
- child: ClipPath(
- clipper: _TitleClipPath(arrowHeight),
- child: Container(
- width: _width,
- height: _height,
- padding: const EdgeInsets.symmetric(
- horizontal: 15,
- ),
- decoration: BoxDecoration(
- boxShadow: [
- BoxShadow(
- color: Theme.of(context).primaryColor.withOpacity(1),
- ),
- ],
- color: color,
- ),
- alignment: Alignment.center,
- child: FittedBox(
- child: Text(
- title,
- style: const TextStyle(color: Colors.white, fontSize: 20),
- ),
- ),
- ),
- ),
- ),
- );
- }
- }
- class _TitleClipPath extends CustomClipper<Path> {
- final double arrowHeight;
- _TitleClipPath(this.arrowHeight);
- @override
- Path getClip(Size size) {
- final height = size.height;
- final arrowBase = height / 2;
- // final arrowPLine = math.tan(120 / 180) * arrowBase;
- final path = Path();
- path.moveTo(0, 0); // 左上角
- path.lineTo(size.width - arrowHeight, 0); // 右上角
- path.lineTo(size.width, arrowBase); // 右端点
- path.lineTo(size.width - arrowHeight, height); // 右下角
- path.lineTo(0, height); // 左下角
- path.lineTo(arrowHeight, arrowBase); // 左端点
- path.lineTo(0, 0); // 左上角
- return path;
- }
- @override
- bool shouldReclip(covariant CustomClipper<Path> oldClipper) {
- return false;
- }
- }
- class HoleShapeBorder extends ShapeBorder {
- final Offset offset;
- final double size;
- final double arrowHeight;
- const HoleShapeBorder(this.arrowHeight,
- {this.offset = const Offset(0, 0), this.size = 0});
- @override
- EdgeInsetsGeometry get dimensions => throw UnimplementedError();
- @override
- void paint(Canvas canvas, Rect rect, {TextDirection? textDirection}) {}
- @override
- Path getOuterPath(Rect rect, {TextDirection? textDirection}) {
- var path = Path();
- final height = rect.size.height;
- final arrowBase = height / 2;
- path.moveTo(0, 0); // 左上角
- path.lineTo(rect.size.width - arrowHeight, 0); // 右上角
- path.lineTo(rect.size.width, arrowBase); // 右端点
- path.lineTo(rect.size.width - arrowHeight, height); // 右下角
- path.lineTo(0, height); // 左下角
- path.lineTo(arrowHeight, arrowBase); // 左端点
- path.lineTo(0, 0); // 左上角
- return path;
- }
- @override
- ShapeBorder scale(double t) {
- // TODO: implement scale
- throw UnimplementedError();
- }
- @override
- Path getInnerPath(Rect rect, {TextDirection? textDirection}) {
- // TODO: implement getInnerPath
- throw UnimplementedError();
- }
- }
- // ClipPath(
- // clipper: TriangleClipper(),
- // child: Container(
- // width: 50,
- // height: 50,
- // padding:
- // const EdgeInsets.all(
- // 2),
- // alignment:
- // Alignment.topRight,
- // decoration:
- // const BoxDecoration(
- // color: Colors.blue,
- // borderRadius:
- // BorderRadius.only(
- // topRight:
- // Radius.circular(
- // 8,
- // ),
- // ),
- // ),
- // child: const Icon(
- // Icons.check,
- // color: Colors.white,
- // ),
- // ))
- // class TriangleClipper extends CustomClipper<Path> {
- // @override
- // Path getClip(Size size) {
- // final path = Path()
- // ..moveTo(size.width, 0)
- // ..lineTo(0, 0)
- // ..lineTo(size.width, size.height)
- // ..close();
- // return path;
- // }
- // @override
- // bool shouldReclip(CustomClipper<Path> oldClipper) => false;
- // }
|