Added Formula Button (for maths support) (#900)

* added basic math keyboard

* added formula to sample data!

* created new variable for showFormulaButton
pull/901/head
Garv Shah 3 years ago committed by GitHub
parent ac1cf592c3
commit a707918f7f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 7
      example/assets/sample_data.json
  2. 3
      lib/src/models/documents/nodes/embeddable.dart
  3. 20
      lib/src/widgets/embeds/default_embed_builder.dart
  4. 14
      lib/src/widgets/toolbar.dart
  5. 69
      lib/src/widgets/toolbar/formula_button.dart
  6. 1
      pubspec.yaml

@ -536,5 +536,10 @@
},
{
"insert": "\n"
},
{
"insert": {
"formula": "x=\\frac{-b\\pm\\sqrt{b^2-4ac}}{2a}"
}
}
]
]

@ -38,6 +38,9 @@ class BlockEmbed extends Embeddable {
static const String videoType = 'video';
static BlockEmbed video(String videoUrl) => BlockEmbed(videoType, videoUrl);
static const String formulaType = 'formula';
static BlockEmbed formula(String formula) => BlockEmbed(formulaType, formula);
static const String customType = 'custom';
static BlockEmbed custom(CustomBlockEmbed customBlock) =>
BlockEmbed(customType, customBlock.toJsonString());

@ -1,7 +1,9 @@
import 'package:flutter/cupertino.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:gallery_saver/gallery_saver.dart';
import 'package:math_keyboard/math_keyboard.dart';
import 'package:tuple/tuple.dart';
import '../../models/documents/attribute.dart';
@ -154,6 +156,24 @@ Widget defaultEmbedBuilder(
readOnly: readOnly,
onVideoInit: onVideoInit,
);
case BlockEmbed.formulaType:
final mathController = MathFieldEditingController();
return Focus(
onFocusChange: (hasFocus) {
if (hasFocus) {
// If the MathField is tapped, hides the built in keyboard
SystemChannels.textInput.invokeMethod('TextInput.hide');
print(mathController.currentEditingValue());
}
},
child: MathField(
controller: mathController,
variables: const ['x', 'y', 'z'],
onChanged: (value) {},
onSubmitted: (value) {},
),
);
default:
throw UnimplementedError(
'Embeddable type "${node.value.type}" is not supported by default '

@ -14,6 +14,7 @@ import 'toolbar/arrow_indicated_button_list.dart';
import 'toolbar/camera_button.dart';
import 'toolbar/clear_format_button.dart';
import 'toolbar/color_button.dart';
import 'toolbar/formula_button.dart';
import 'toolbar/history_button.dart';
import 'toolbar/image_button.dart';
import 'toolbar/image_video_utils.dart';
@ -109,6 +110,7 @@ class QuillToolbar extends StatelessWidget implements PreferredSizeWidget {
bool multiRowsDisplay = true,
bool showImageButton = true,
bool showVideoButton = true,
bool showFormulaButton = false,
bool showCameraButton = true,
bool showDirection = false,
bool showSearchButton = true,
@ -364,6 +366,18 @@ class QuillToolbar extends StatelessWidget implements PreferredSizeWidget {
webVideoPickImpl: webVideoPickImpl,
iconTheme: iconTheme,
),
if (showFormulaButton)
FormulaButton(
icon: Icons.functions,
iconSize: toolbarIconSize,
controller: controller,
onImagePickCallback: onImagePickCallback,
filePickImpl: filePickImpl,
webImagePickImpl: webImagePickImpl,
mediaPickSettingSelector: mediaPickSettingSelector,
iconTheme: iconTheme,
dialogTheme: dialogTheme,
),
if (showDividers &&
isButtonGroupShown[0] &&
(isButtonGroupShown[1] ||

@ -0,0 +1,69 @@
import 'package:flutter/material.dart';
import 'package:image_picker/image_picker.dart';
import '../../models/documents/nodes/embeddable.dart';
import '../../models/themes/quill_dialog_theme.dart';
import '../../models/themes/quill_icon_theme.dart';
import '../controller.dart';
import '../toolbar.dart';
class FormulaButton extends StatelessWidget {
const FormulaButton({
required this.icon,
required this.controller,
this.iconSize = kDefaultIconSize,
this.onImagePickCallback,
this.fillColor,
this.filePickImpl,
this.webImagePickImpl,
this.mediaPickSettingSelector,
this.iconTheme,
this.dialogTheme,
Key? key,
}) : super(key: key);
final IconData icon;
final double iconSize;
final Color? fillColor;
final QuillController controller;
final OnImagePickCallback? onImagePickCallback;
final WebImagePickImpl? webImagePickImpl;
final FilePickImpl? filePickImpl;
final MediaPickSettingSelector? mediaPickSettingSelector;
final QuillIconTheme? iconTheme;
final QuillDialogTheme? dialogTheme;
@override
Widget build(BuildContext context) {
final theme = Theme.of(context);
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color;
final iconFillColor =
iconTheme?.iconUnselectedFillColor ?? (fillColor ?? theme.canvasColor);
return QuillIconButton(
icon: Icon(icon, size: iconSize, color: iconColor),
highlightElevation: 0,
hoverElevation: 0,
size: iconSize * 1.77,
fillColor: iconFillColor,
borderRadius: iconTheme?.borderRadius ?? 2,
onPressed: () => _onPressedHandler(context),
);
}
Future<void> _onPressedHandler(BuildContext context) async {
final index = controller.selection.baseOffset;
final length = controller.selection.extentOffset - index;
controller.replaceText(index, length, BlockEmbed.formula(''), null);
}
}

@ -30,6 +30,7 @@ dependencies:
gallery_saver: ^2.3.2
device_info_plus: ^4.0.0
platform: ^3.1.0
math_keyboard: ^0.1.6
dev_dependencies:
flutter_test:

Loading…
Cancel
Save