Я думаю, что вопрос проясняет вопрос, однако мне нужен png из рисунка, нарисованного на экране с помощью canvas, как мне это сделать, вот мой код ... Я пробовал все в настоящее время из-за временных ограничений, которые я использую Перекрасьте границу, чтобы сделать снимок экрана с полным стеком, однако мне нужна лучшая функциональность. Есть какие-то указатели? ... Я пытался искать, но пока не нашел каких-либо решений. Я работаю над своего рода подписью подпись пользователя и размещение его в Интернете в виде png, я могу заставить пользователя рисовать на экране и делать скриншот всего стека, но не могу получить Png только из нарисованного шаблона
class SignaturePainter extends CustomPainter {
Paint _paint;
SignaturePainter(this.points, this._paint);
final List<Offset> points;
List<Offset> offsetPoints = List();
void paint(Canvas canvas, Size size) {
for (int i = 0; i < points.length - 1; i++) {
if (points[i] != null && points[i + 1] != null)
canvas.drawLine(points[i], points[i + 1], _paint);
if (points[i] != null && points[i + 1] == null) {
offsetPoints.clear();
offsetPoints.add(points[i]);
offsetPoints.add(Offset(points[i].dx + 0.1, points[i].dy + 0.1));
canvas.drawPoints(ui.PointMode.points, offsetPoints, _paint);
}
}
}
bool shouldRepaint(SignaturePainter other) => other.points != points;
}
class Signature extends StatefulWidget {
SignatureState createState() => new SignatureState();
}
class SignatureState extends State<Signature> implements ClearScreen {
List<Offset> _points = <Offset>[];
Paint backgroundPaint, foregroundPaint;
GlobalKey globalKey = GlobalKey();
@override
void initState() {
backgroundPaint = new Paint()
..color = Colors.white
..strokeCap = StrokeCap.round
..strokeWidth = 5.0;
foregroundPaint = new Paint()
..color = Colors.amber
..strokeCap = StrokeCap.round
..strokeWidth = 5.0;
super.initState();
}
Widget build(BuildContext context) {
return Scaffold(
body: Container(
decoration: appGradient,
height: MediaQuery.of(context).size.height,
width: MediaQuery.of(context).size.width,
child: RepaintBoundary(
key: globalKey,
child: Stack(
children: [
GestureDetector(
onPanUpdate: (DragUpdateDetails details) {
RenderBox referenceBox = context.findRenderObject();
Offset localPosition =
referenceBox.globalToLocal(details.globalPosition);
setState(() {
_points = new List.from(_points)..add(localPosition);
});
},
onPanEnd: (DragEndDetails details) => _points.add(null),
onPanStart: (details) {
setState(() {
RenderBox referenceBox = context.findRenderObject();
Offset localPosition =
referenceBox.globalToLocal(details.globalPosition);
_points = List.from(_points)..add(localPosition);
});
},
),
CustomPaint(
painter: SignaturePainter(_points, backgroundPaint),
),
Positioned(
bottom: 0.0,
child: Container(
width: MediaQuery.of(context).size.width,
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Button("Proceed", Alignment.bottomLeft, this, 0),
Button("Redo", Alignment.bottomRight, this, 1),
],
),
],
),
),
),
],
),
),
),
);
}
@override
void clearScreen() {
setState(() {
_points.clear();
});
}
@override
void performProceedTapAction() async {
RenderRepaintBoundary boundary =
globalKey.currentContext.findRenderObject();
ui.Image image = await boundary.toImage();
ByteData byteData = await image.toByteData(format: ui.ImageByteFormat.png);
Uint8List pngBytes = byteData.buffer.asUint8List();
var extern = await getExternalStorageDirectory();
String path = extern.path;
int random = Random(10000).nextInt(1000000);
File file = new File("$path/image-$random.png");
await file.writeAsBytes(pngBytes).then((onValue) {
print(onValue);
});
}
}
abstract class ClearScreen {
void clearScreen();
void performProceedTapAction();
}
class Button extends StatelessWidget {
final String buttonText;
final Alignment alignment;
final ClearScreen mListener;
final flag;
const Button(this.buttonText, this.alignment, this.mListener, this.flag);
@override
Widget build(BuildContext context) {
return Align(
alignment: alignment,
child: Container(
height: 60,
width: MediaQuery.of(context).size.width / 2,
child: Card(
elevation: 10,
color: Color.fromARGB(255, 149, 208, 158),
shape:
RoundedRectangleBorder(borderRadius: BorderRadius.circular(30)),
child: InkWell(
onTap: () {
if (flag == 1) {
mListener.clearScreen();
} else if (flag == 0) {
mListener.performProceedTapAction();
}
},
child: Center(
child: Text(
buttonText,
style: TextStyle(
color: Colors.white,
fontWeight: FontWeight.w300,
fontSize: 20),
)),
),
),
),
);
}
}