Files
apidash/lib/widgets/codegen_previewer.dart
Ankit Mahato e3084e8844 fixes
2024-03-23 14:19:24 +05:30

180 lines
5.3 KiB
Dart

import 'package:flutter/material.dart';
import 'package:highlighter/highlighter.dart' show highlight;
import 'package:apidash/consts.dart';
import 'package:apidash/utils/utils.dart';
import 'code_previewer.dart';
import 'widgets.dart'
show CopyButton, DropdownButtonCodegenLanguage, SaveInDownloadsButton;
class CodeGenPreviewer extends StatefulWidget {
const CodeGenPreviewer({
super.key,
required this.code,
required this.theme,
this.language,
this.textStyle,
this.padding = EdgeInsets.zero,
});
final String code;
final String? language;
final TextStyle? textStyle;
final EdgeInsetsGeometry padding;
final Map<String, TextStyle> theme;
@override
State<CodeGenPreviewer> createState() => _CodeGenPreviewerState();
}
class _CodeGenPreviewerState extends State<CodeGenPreviewer> {
static const _rootKey = 'root';
static const _defaultFontColor = Color(0xff000000);
late TextStyle textStyle;
final ScrollController controllerH = ScrollController();
final ScrollController controllerV = ScrollController();
@override
void dispose() {
controllerH.dispose();
controllerV.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
final spans = generateSpans(widget.code, widget.language, widget.theme);
textStyle = TextStyle(
color: widget.theme[_rootKey]?.color ?? _defaultFontColor,
);
if (widget.textStyle != null) {
textStyle = textStyle.merge(widget.textStyle);
}
return Padding(
padding: widget.padding,
child: Scrollbar(
thickness: 10,
thumbVisibility: true,
controller: controllerV,
child: Scrollbar(
notificationPredicate: (notification) => notification.depth == 1,
thickness: 10,
thumbVisibility: true,
controller: controllerH,
child: SingleChildScrollView(
controller: controllerV,
child: SingleChildScrollView(
scrollDirection: Axis.horizontal,
controller: controllerH,
child: Column(
children: [
Row(
children: [
SelectionArea(
child: Text.rich(
TextSpan(
children: spans,
style: textStyle,
),
softWrap: false,
),
),
],
),
],
),
),
),
),
),
);
}
}
List<TextSpan> generateSpans(
String code, String? language, Map<String, TextStyle> theme) {
var parsed = highlight.parse(code, language: language);
var spans = convert(parsed.nodes!, theme);
return spans;
}
class ViewCodePane extends StatelessWidget {
const ViewCodePane({
super.key,
required this.code,
required this.codegenLanguage,
required this.onChangedCodegenLanguage,
});
final String code;
final CodegenLanguage codegenLanguage;
final Function(CodegenLanguage?) onChangedCodegenLanguage;
@override
Widget build(BuildContext context) {
var codeTheme = Theme.of(context).brightness == Brightness.light
? kLightCodeTheme
: kDarkCodeTheme;
final textContainerdecoration = BoxDecoration(
color: Color.alphaBlend(
(Theme.of(context).brightness == Brightness.dark
? Theme.of(context).colorScheme.onPrimaryContainer
: Theme.of(context).colorScheme.primaryContainer)
.withOpacity(kForegroundOpacity),
Theme.of(context).colorScheme.surface),
border: Border.all(color: Theme.of(context).colorScheme.surfaceVariant),
borderRadius: kBorderRadius8,
);
return LayoutBuilder(
builder: (BuildContext context, BoxConstraints constraints) {
var showLabel = showButtonLabelsInViewCodePane(
constraints.maxWidth,
);
return Padding(
padding: kP10,
child: Column(
children: [
SizedBox(
height: kHeaderHeight,
child: Row(
children: [
Expanded(
child: DropdownButtonCodegenLanguage(
codegenLanguage: codegenLanguage,
onChanged: onChangedCodegenLanguage,
),
),
CopyButton(
toCopy: code,
showLabel: showLabel,
),
SaveInDownloadsButton(
content: stringToBytes(code),
ext: codegenLanguage.ext,
showLabel: showLabel,
)
],
),
),
kVSpacer10,
Expanded(
child: Container(
width: double.maxFinite,
padding: kP8,
decoration: textContainerdecoration,
child: CodeGenPreviewer(
code: code,
theme: codeTheme,
language: codegenLanguage.codeHighlightLang,
textStyle: kCodeStyle,
),
),
),
],
),
);
},
);
}
}