123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 |
- // Copyright (c) 2018, the Zefyr project authors. Please see the AUTHORS file
- // for details. All rights reserved. Use of this source code is governed by a
- // BSD-style license that can be found in the LICENSE file.
- import 'package:flutter/widgets.dart';
-
- import 'controller.dart';
- import 'editable_text.dart';
- import 'image.dart';
- import 'scaffold.dart';
- import 'scope.dart';
- import 'theme.dart';
- import 'toolbar.dart';
-
- /// Widget for editing Zefyr documents.
- class ZefyrEditor extends StatefulWidget {
- const ZefyrEditor({
- Key key,
- @required this.controller,
- @required this.focusNode,
- this.autofocus: true,
- this.enabled: true,
- this.padding: const EdgeInsets.symmetric(horizontal: 16.0),
- this.toolbarDelegate,
- this.imageDelegate,
- this.physics,
- }) : super(key: key);
-
- final ZefyrController controller;
- final FocusNode focusNode;
- final bool autofocus;
- final bool enabled;
- final ZefyrToolbarDelegate toolbarDelegate;
- final ZefyrImageDelegate imageDelegate;
- final ScrollPhysics physics;
-
- /// Padding around editable area.
- final EdgeInsets padding;
-
- @override
- _ZefyrEditorState createState() => new _ZefyrEditorState();
- }
-
- class _ZefyrEditorState extends State<ZefyrEditor> {
- ZefyrImageDelegate _imageDelegate;
- ZefyrScope _scope;
- ZefyrThemeData _themeData;
- GlobalKey<ZefyrToolbarState> _toolbarKey;
- ZefyrScaffoldState _scaffold;
-
- bool get hasToolbar => _toolbarKey != null;
-
- void showToolbar() {
- assert(_toolbarKey == null);
- _toolbarKey = GlobalKey();
- _scaffold.showToolbar(buildToolbar);
- }
-
- void hideToolbar() {
- if (_toolbarKey == null) return;
- _scaffold.hideToolbar();
- _toolbarKey = null;
- }
-
- Widget buildToolbar(BuildContext context) {
- return ZefyrTheme(
- data: _themeData,
- child: ZefyrToolbar(
- key: _toolbarKey,
- editor: _scope,
- delegate: widget.toolbarDelegate,
- ),
- );
- }
-
- void _handleChange() {
- if (_scope.focusOwner == FocusOwner.none) {
- hideToolbar();
- } else if (!hasToolbar) {
- showToolbar();
- } else {
- // TODO: is there a nicer way to do this?
- WidgetsBinding.instance.addPostFrameCallback((_) {
- _toolbarKey?.currentState?.markNeedsRebuild();
- });
- }
- }
-
- @override
- void initState() {
- super.initState();
- _imageDelegate = widget.imageDelegate ?? new ZefyrDefaultImageDelegate();
- }
-
- @override
- void didUpdateWidget(ZefyrEditor oldWidget) {
- super.didUpdateWidget(oldWidget);
- _scope.controller = widget.controller;
- _scope.focusNode = widget.focusNode;
- if (widget.imageDelegate != oldWidget.imageDelegate) {
- _imageDelegate = widget.imageDelegate ?? new ZefyrDefaultImageDelegate();
- _scope.imageDelegate = _imageDelegate;
- }
- }
-
- @override
- void didChangeDependencies() {
- super.didChangeDependencies();
- final parentTheme = ZefyrTheme.of(context, nullOk: true);
- final fallbackTheme = ZefyrThemeData.fallback(context);
- _themeData = (parentTheme != null)
- ? fallbackTheme.merge(parentTheme)
- : fallbackTheme;
-
- if (_scope == null) {
- _scope = ZefyrScope.editable(
- imageDelegate: _imageDelegate,
- controller: widget.controller,
- focusNode: widget.focusNode,
- focusScope: FocusScope.of(context),
- );
- _scope.addListener(_handleChange);
- } else {
- final focusScope = FocusScope.of(context);
- _scope.focusScope = focusScope;
- }
-
- final scaffold = ZefyrScaffold.of(context);
- if (_scaffold != scaffold) {
- bool didHaveToolbar = hasToolbar;
- hideToolbar();
- _scaffold = scaffold;
- if (didHaveToolbar) showToolbar();
- }
- }
-
- @override
- void dispose() {
- hideToolbar();
- _scope.removeListener(_handleChange);
- _scope.dispose();
- super.dispose();
- }
-
- @override
- Widget build(BuildContext context) {
- Widget editable = new ZefyrEditableText(
- controller: _scope.controller,
- focusNode: _scope.focusNode,
- imageDelegate: _scope.imageDelegate,
- autofocus: widget.autofocus,
- enabled: widget.enabled,
- padding: widget.padding,
- physics: widget.physics,
- );
-
- return ZefyrTheme(
- data: _themeData,
- child: ZefyrScopeAccess(
- scope: _scope,
- child: editable,
- ),
- );
- }
- }
|