123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833 |
- import 'dart:convert';
- import 'package:fis_jsonrpc/rpc.dart';
- import 'package:flutter/material.dart';
- import 'package:get/get.dart';
- import 'package:vnoteapp/components/button.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/cachedRecord.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_blood_sugar.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_body_temperature.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_body_weight.dart';
- import 'package:vnoteapp/pages/check/widgets/exam_configurable/exam_boold_oxygen.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;
- final Widget? followUpWidget;
- final String? patientCode;
- final String? examData;
- const ConfigurableCard({
- super.key,
- required this.cardKey,
- required this.callBack,
- this.followUpWidget,
- this.patientCode,
- this.examData,
- });
- @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 _cachedRecordManager = Get.find<ICachedRecordManager>();
- final arrowHeight = math.tan(120 / 180) * (_height / 2);
- List<String> deviceList = ['Temp', 'GLU', 'NIBP', 'SpO2', 'BMI'];
- Map<String, dynamic> deviceCached = {};
- @override
- void initState() {
- super.initState();
- initTemplate();
- }
- @override
- void dispose() {
- super.dispose();
- }
- Future<void> initTemplate() async {
- await fetchTemplateIndex();
- await fetchTemplate(widget.cardKey);
- if (!['GXY', 'TNB'].contains(widget.cardKey)) {
- fetchTemplateData();
- }
- }
- /// 读取诊疗展示的缓存
- Future<String?> readCachedRecord(String currentDevice) async {
- String? value = await _cachedRecordManager.readCachedRecord(
- currentDevice,
- widget.patientCode!,
- 'ZLZS',
- );
- return value;
- }
- /// 读取体检的缓存
- Future<String?> readCachedCheck() async {
- String? value = await _cachedRecordManager.readCachedRecord(
- widget.cardKey,
- widget.patientCode!,
- 'EXAM',
- );
- return value;
- }
- Future<void> readCached() async {
- for (var element in deviceList) {
- String? value = await readCachedRecord(element);
- if (value?.isNotEmpty ?? false) {
- deviceCached.addAll(jsonDecode(value!));
- }
- }
- }
- Future<void> fetchTemplateIndex() async {
- try {
- /// 获取模板的键值对
- String? templates;
- templates = await _templateManager.readTemplate('templateRelation');
- templateRelation = jsonDecode(templates!);
- setState(() {});
- } catch (error) {
- print('发生错误: $error');
- }
- }
- Future<void> fetchTemplateData() async {
- String? value = await readCachedCheck();
- await readCached();
- if (deviceCached.isNotEmpty) {
- formValue = deviceCached;
- setState(() {});
- return;
- }
- if (value?.isNotEmpty ?? false) {
- formValue = jsonDecode(value!);
- }
- /// 这逻辑需要优化
- // if (widget.examData?.isNotEmpty ?? false) {
- // formValue = jsonDecode(widget.examData!);
- // }
- formValue.forEach(
- (key, value) {
- if (value is List) {
- formValue[key] = List<String>.from(formValue[key]);
- }
- },
- );
- setState(() {});
- }
- 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));
- setState(() {});
- } catch (error) {
- print('发生错误: $error');
- }
- }
- @override
- Widget build(BuildContext context) {
- return Scaffold(
- key: scaffoldKey,
- endDrawer: VDynamicDrawerWrapper(scaffoldKey: scaffoldKey),
- resizeToAvoidBottomInset: false,
- body: Column(
- children: [
- Row(
- crossAxisAlignment: CrossAxisAlignment.start,
- children: [
- const SizedBox(
- width: 16,
- ),
- Container(
- margin: const EdgeInsets.only(top: 8),
- width: 130,
- height: 54,
- child: VButton(
- onTap: () => {Get.back()},
- child: const Row(
- mainAxisAlignment: MainAxisAlignment.center,
- children: [
- Icon(Icons.arrow_back_ios_new, size: 24),
- SizedBox(
- width: 8,
- ),
- Text("返回", style: TextStyle(fontSize: 20)),
- ],
- ),
- ),
- ),
- Expanded(
- child: Container(
- padding:
- const EdgeInsets.symmetric(vertical: 16, horizontal: 10),
- child: _buildTitleList(),
- ),
- ),
- Container(
- margin: const EdgeInsets.only(top: 8, right: 16),
- width: 130,
- height: 54,
- child: VButton(
- onTap: () {
- widget
- .callBack(
- widget.cardKey,
- templateRelation[widget.cardKey]!,
- jsonEncode(formValue),
- )
- .call();
- Get.back();
- },
- child: const Row(
- mainAxisAlignment: MainAxisAlignment.center,
- children: [
- Icon(Icons.save, size: 24),
- SizedBox(
- width: 8,
- ),
- Text("保存", style: TextStyle(fontSize: 20)),
- ],
- ),
- ),
- ),
- ],
- ),
- Expanded(
- child: Stack(
- children: [
- Row(
- mainAxisAlignment: MainAxisAlignment.start,
- crossAxisAlignment: CrossAxisAlignment.start,
- children: [
- _buildDiagram(),
- _buildContent(),
- ],
- ),
- if (currentTitleIndex != currentTemplate.length - 1)
- _buildPositionedButton(
- () async {
- currentTitleIndex++;
- setState(() {});
- },
- right: -30,
- ),
- ],
- ),
- )
- ],
- ),
- );
- }
- 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,
- 'weight': _buildBodyWeight,
- 'sugar': _buildBodySugar,
- 'bloodOxygen': _buildBloodOxygen,
- };
- 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;
- }
- bool currentTemplateOptionsIsNotEmpty =
- currentTemplate[currentTitleIndex].options?.isNotEmpty ?? false;
- List<Widget> items = List.generate(itemCount, (index) {
- FormObject? currentFormObject =
- currentTemplate[currentTitleIndex].children?[index];
- int span = currentFormObject?.span ?? 12;
- //父结构的options不等于空或null
- if (true) {
- //子结构的options若是无值则取父类的options值
- if (currentTemplateOptionsIsNotEmpty) {
- currentFormObject!.options =
- currentTemplate[currentTitleIndex].options;
- }
- }
- 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, // 设置子小部件之间的间距
- spacing: -12,
- 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 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: [
- /// TODO BAKA-优化
- ['GXY', 'TNB'].contains(widget.cardKey)
- ? widget.followUpWidget!
- : Container(
- alignment: Alignment.topCenter,
- margin: const EdgeInsets.all(16).copyWith(top: 0),
- child: Image.asset(
- 'assets/images/exam/normalMeasurementChart.png',
- height: double.infinity,
- fit: BoxFit.fitWidth, // 设置图像的适应方式
- ),
- ),
- if (currentTitleIndex != 0)
- _buildPositionedButton(
- () async {
- if (currentTitleIndex == 0) {
- Get.back();
- } else {
- currentTitleIndex--;
- setState(() {});
- }
- },
- left: -30,
- ),
- ],
- ),
- );
- }
- /// 按钮
- Widget _buildPositionedButton(Function onTap, {double? right, double? left}) {
- return Positioned(
- right: right,
- left: left,
- bottom: 0,
- top: 0,
- child: Container(
- width: 100,
- height: 100,
- alignment: Alignment.centerLeft,
- child: InkWell(
- onTap: () => onTap.call(),
- child: Container(
- width: 100,
- height: 100,
- padding: const EdgeInsets.all(20),
- alignment:
- right.isNull ? Alignment.centerRight : Alignment.centerLeft,
- decoration: BoxDecoration(
- borderRadius: BorderRadius.circular(50),
- color: Theme.of(context).primaryColor.withOpacity(
- .8,
- ),
- ),
- child: Image.asset(
- right.isNull
- ? "assets/images/exam/left-arrow.png"
- : "assets/images/exam/right-arrow.png",
- width: 40,
- height: 40,
- color: Colors.white,
- fit: BoxFit.contain,
- ),
- ),
- ),
- ),
- );
- }
- /// 主页面
- 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!] ?? '';
- if ((formValue['Height']?.isNotEmpty ?? false) &&
- (formValue['Weight']?.isNotEmpty ?? false)) {
- formValue['Bmi'] = (double.parse(formValue['Weight']) /
- ((double.parse(formValue['Height']) / 100) *
- (double.parse(formValue['Height']) / 100)))
- .toStringAsFixed(2);
- }
- 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) {
- return const ExamBloodPressure();
- }
- /// 单选框组件
- 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!] ?? '';
- void bodyTemperatureInput(String value) {
- formValue[currentFormObject.key!] = value;
- currentInputValue = formValue[currentFormObject.key!];
- setState(() {});
- }
- return ExamBodyTemperature(
- currentInputValue: currentInputValue,
- bodyTemperatureInput: bodyTemperatureInput,
- currentFormObject: currentFormObject,
- );
- }
- /// 血氧
- Widget _buildBloodOxygen(FormObject currentFormObject) {
- Map<String, dynamic> currentValue = formValue;
- void bloodOxygenInput(Map<String, dynamic> bloodOxygen) {
- formValue['Pulse_Frequency'] = bloodOxygen['Pulse_Frequency'];
- formValue['Spo2'] = bloodOxygen['Spo2'];
- currentValue = bloodOxygen;
- setState(() {});
- }
- return ExamBloodOxygen(
- currentValue: currentValue,
- bloodOxygenInput: bloodOxygenInput,
- );
- }
- /// 体重
- Widget _buildBodyWeight(FormObject currentFormObject) {
- String currentInputValue = formValue[currentFormObject.key!] ?? '';
- void bodyWeightInput(String value) {
- formValue[currentFormObject.key!] = value;
- currentInputValue = formValue[currentFormObject.key!];
- setState(() {});
- }
- return ExamBodyWeight(
- currentInputValue: currentInputValue,
- bodyWeightInput: bodyWeightInput,
- currentFormObject: currentFormObject,
- );
- }
- /// 血糖
- Widget _buildBodySugar(FormObject currentFormObject) {
- String currentInputValue = formValue[currentFormObject.key!] ?? '';
- void bloodSugarInput(String value) {
- formValue[currentFormObject.key!] = value;
- currentInputValue = formValue[currentFormObject.key!];
- setState(() {});
- }
- return ExamBloodSugar(
- currentInputValue: currentInputValue,
- bloodSugarInput: bloodSugarInput,
- 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;
- // }
|