Как нарисовать виджет на холсте во Flutter?
Есть ли способ нарисовать Widget
в данной позиции на Canvas
?
В частности, я хочу нарисовать дочерние виджеты Marker
связан с FlutterMap
на отдельном Canvas
перед фактическим FlutterMap
виджет. Вот попытка создатьCustomPainter
это сделало бы это, но я не могу понять, как на самом деле рисовать виджеты на холсте. С использованиемRenderObject
требует PaintingContext
, который я не знаю, как создать / получить:
class MarkerPainter extends CustomPainter {
MapController mc;
BuildContext context;
List<Marker> markers;
MarkerPainter(this.context, this.mc, this.markers);
@override
void paint(Canvas canvas, Size size) {
if( markers != null && markers.isNotEmpty ){
for(int i=0; i<markers.length; i++){
Marker marker = markers[i];
Offset o = myCalculateOffsetFromLatLng(marker.point, mc, context);
// Won't work, this needs a PaintingContext...
marker.builder(context).createElement().renderObject.paint(context, o);
}
}
}
@override
bool shouldRepaint(MarkerPainter oldDelegate) => oldDelegate.markers != markers;
}
1 ответ
Вы не можете сделать это с помощью CustomPainter.
Этот класс - всего лишь упрощение реального дела: RenderObject, у которого есть доступ ко всему, что связано с художником (и макетом + многое другое).
Что вам нужно сделать, так это вместо CustomPainter создать RenderBox (2d RenderObject)
В вашем случае вы хотите нарисовать список виджетов. В этой ситуации вам нужно будет создать:
- MultiChildRenderObjectWidget, виджет, который принимает
children
рисовать - RenderBox которые смешивают в RenderBoxContainerDefaultsMixin, который добавляет необходимое для RenderBox для обработки списка детей.
Подводя итог, виджет использовался следующим образом:
MyExample(
children: [
Text('foo'),
Text('bar'),
],
),
будет записано так:
import 'package:flutter/rendering.dart';
import 'package:flutter/material.dart';
class MyExample extends MultiChildRenderObjectWidget {
MyExample({
Key key,
@required List<Widget> children,
}) : super(key: key, children: children);
@override
RenderMyExample createRenderObject(BuildContext context) {
return RenderMyExample();
}
}
class MyExampleParentData extends ContainerBoxParentData<RenderBox> {}
class RenderMyExample extends RenderBox
with ContainerRenderObjectMixin<RenderBox, MyExampleParentData> {
@override
void setupParentData(RenderObject child) {
if (child.parentData is! MyExampleParentData) {
child.parentData = MyExampleParentData();
}
}
@override
void performLayout() {
size = constraints.biggest;
for (var child = firstChild; child != null; child = childAfter(child)) {
child.layout(
// limit children to a max height of 50
constraints.copyWith(maxHeight: 50),
);
}
}
@override
void paint(PaintingContext context, Offset offset) {
// Paints all children in order vertically, separated by 50px
var verticalOffset = .0;
for (var child = firstChild; child != null; child = childAfter(child)) {
context.paintChild(child, offset + Offset(0, verticalOffset));
verticalOffset += 50;
}
}
}