image_compare_slider 2.8.0 copy "image_compare_slider: ^2.8.0" to clipboard
image_compare_slider: ^2.8.0 copied to clipboard

Easily compare two images with a slider and a draggable line/handle, fully customizable.

example/lib/main.dart

import 'dart:ui';

import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:image_compare_slider/image_compare_slider.dart';

void main() => runApp(const MyApp());

class MyApp extends StatefulWidget {
  const MyApp({super.key});

  @override
  State<MyApp> createState() => _MyAppState();
}

class _MyAppState extends State<MyApp> {
  @override
  Widget build(BuildContext context) {
    return const MaterialApp(
      debugShowCheckedModeBanner: false,
      home: _App(),
    );
  }
}

class _App extends StatefulWidget {
  const _App();

  @override
  State<_App> createState() => _AppState();
}

class _AppState extends State<_App> {
  SliderDirection direction = SliderDirection.leftToRight;
  Color dividerColor = Colors.white;
  Color handleColor = Colors.white;
  Color handleOutlineColor = Colors.white;
  double dividerWidth = 2;
  bool reactOnHover = false;
  bool hideHandle = false;
  double position = 0.5;
  double handlePosition = 0.5;
  double handleSizeHeight = 75;
  double handleSizeWidth = 7.5;
  bool handleFollowsP = false;
  bool fillHandle = true;
  double handleRadius = 10;
  Color? itemOneColor;
  Color? itemTwoColor;
  BlendMode itemOneBlendMode = BlendMode.overlay;
  BlendMode itemTwoBlendMode = BlendMode.darken;
  Widget Function(Widget)? itemOneWrapper;
  Widget Function(Widget)? itemTwoWrapper;

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      backgroundColor: Colors.grey.shade900,
      appBar: AppBar(
        backgroundColor: Colors.grey.shade900,
        title: const Text('Image Compare Slider'),
        elevation: 0,
        centerTitle: false,
        actions: [
          IconButton(
            icon: const Icon(Icons.info_outline),
            onPressed: () => showDialog(
              context: context,
              builder: (context) => AlertDialog(
                backgroundColor: Colors.grey.shade900,
                title: const Text(
                  'Image Compare Slider',
                  style: TextStyle(color: Colors.white),
                ),
                content: const Text(
                  'This is a Flutter package that allows you to compare two images with a fully customizable slider.',
                  style: TextStyle(color: Colors.white),
                ),
                actions: [
                  TextButton(
                    onPressed: () => Navigator.pop(context),
                    child: const Text('Close'),
                  ),
                ],
              ),
            ),
          ),
        ],
      ),
      body: SingleChildScrollView(
        child: Padding(
          padding: const EdgeInsets.all(20.0),
          child: Column(
            crossAxisAlignment: CrossAxisAlignment.start,
            children: [
              const Text('Last Arrangements',
                  style: TextStyle(color: Colors.white, fontSize: 30)),
              const Text(
                'Miriam Raya',
                style: TextStyle(color: Colors.blueAccent, fontSize: 15),
              ),
              const SizedBox(height: 20),
              Center(
                child: Container(
                  height: 250,
                  decoration: BoxDecoration(
                    borderRadius: BorderRadius.circular(10),
                  ),
                  clipBehavior: Clip.hardEdge,
                  child: ImageCompareSlider(
                    itemOne: Image.asset(
                      'assets/images/render_oc.png',
                      colorBlendMode: itemOneBlendMode,
                      color: itemOneColor,
                    ),
                    itemTwo: Image.asset(
                      'assets/images/render-big.jpeg',
                      colorBlendMode: itemTwoBlendMode,
                      color: itemTwoColor,
                    ),

                    /* Optional */
                    changePositionOnHover: reactOnHover,
                    direction: direction,
                    dividerColor: dividerColor,
                    handleColor: handleColor,
                    handleOutlineColor: handleOutlineColor,
                    dividerWidth: dividerWidth,
                    position: position,
                    onPositionChange: (p) => setState(() => position = p),
                    hideHandle: hideHandle,
                    handlePosition: handlePosition,
                    fillHandle: fillHandle,
                    handleSize: Size(handleSizeWidth, handleSizeHeight),
                    handleRadius:
                        BorderRadius.all(Radius.circular(handleRadius)),
                    itemOneBuilder: (child, context) =>
                        itemOneWrapper?.call(child) ?? child,
                    itemTwoBuilder: (child, context) =>
                        itemTwoWrapper?.call(child) ?? child,
                    handleFollowsPosition: handleFollowsP,
                  ),
                ),
              ),
              const SizedBox(height: 10),
              slider('Position: ${position.toStringAsFixed(2)}', position, (v) {
                setState(() => position = v);
              }),
              Center(
                child: switcher('React on hover', reactOnHover, (v) {
                  setState(() => reactOnHover = v);
                }),
              ),
              _DividerWithText(
                text: 'Divider',
                children: [
                  Row(
                    crossAxisAlignment: CrossAxisAlignment.start,
                    children: [
                      Column(
                        children: [
                          slider('Width: ${dividerWidth.toStringAsFixed(2)}',
                              dividerWidth, (v) {
                            setState(() => dividerWidth = v);
                          }, max: 25, min: 0),
                          slider(
                            'R',
                            dividerColor.red.toDouble(),
                            (v) => setState(() =>
                                dividerColor = dividerColor.withRed(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'G',
                            dividerColor.green.toDouble(),
                            (v) => setState(() => dividerColor =
                                dividerColor.withGreen(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'B',
                            dividerColor.blue.toDouble(),
                            (v) => setState(() => dividerColor =
                                dividerColor.withBlue(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'A',
                            dividerColor.opacity,
                            (v) => setState(() =>
                                dividerColor = dividerColor.withOpacity(v)),
                            max: 1,
                            min: 0,
                          ),
                        ],
                      ),
                      const Spacer(),
                      directionArrow(),
                      const Spacer(),
                    ],
                  ),
                ],
              ),
              _DividerWithText(
                text: 'Handle',
                children: [
                  Row(
                    mainAxisAlignment: MainAxisAlignment.center,
                    children: [
                      Column(
                        children: [
                          const Text(
                            'Color',
                            style: TextStyle(
                              color: Colors.white,
                              fontSize: 15,
                              fontWeight: FontWeight.w500,
                            ),
                          ),
                          slider(
                            'R',
                            handleColor.red.toDouble(),
                            (v) => setState(() =>
                                handleColor = handleColor.withRed(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'G',
                            handleColor.green.toDouble(),
                            (v) => setState(() =>
                                handleColor = handleColor.withGreen(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'B',
                            handleColor.blue.toDouble(),
                            (v) => setState(() =>
                                handleColor = handleColor.withBlue(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'A',
                            handleColor.opacity,
                            (v) => setState(
                                () => handleColor = handleColor.withOpacity(v)),
                            max: 1,
                            min: 0,
                          ),
                        ],
                      ),
                      Column(
                        children: [
                          const Text(
                            'Outline',
                            style: TextStyle(
                              color: Colors.white,
                              fontSize: 15,
                              fontWeight: FontWeight.w500,
                            ),
                          ),
                          slider(
                            'R',
                            handleOutlineColor.red.toDouble(),
                            (v) => setState(() => handleOutlineColor =
                                handleOutlineColor.withRed(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'G',
                            handleOutlineColor.green.toDouble(),
                            (v) => setState(() => handleOutlineColor =
                                handleOutlineColor.withGreen(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'B',
                            handleOutlineColor.blue.toDouble(),
                            (v) => setState(() => handleOutlineColor =
                                handleOutlineColor.withBlue(v.toInt())),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'A',
                            handleOutlineColor.opacity,
                            (v) => setState(() => handleOutlineColor =
                                handleOutlineColor.withOpacity(v)),
                            max: 1,
                            min: 0,
                          ),
                        ],
                      ),
                    ],
                  ),
                  switcher('Hide handle', hideHandle, (v) {
                    setState(() => hideHandle = v);
                  }),
                  switcher('Fill handle', fillHandle, (v) {
                    setState(() => fillHandle = v);
                  }),
                  switcher('Follows position', handleFollowsP, (v) {
                    setState(() => handleFollowsP = v);
                  }),
                  slider('Position: ${handlePosition.toStringAsFixed(2)}',
                      handlePosition, (v) {
                    setState(() => handlePosition = v);
                  }),
                  slider('Radius: ${handleRadius.toStringAsFixed(2)}',
                      handleRadius, (v) {
                    setState(() => handleRadius = v);
                  }, max: 50, min: 0),
                  slider('Size H: ${handleSizeHeight.toStringAsFixed(2)}',
                      handleSizeHeight, (v) {
                    setState(() => handleSizeHeight = v);
                  }, max: 100, min: 0),
                  slider('Size W: ${handleSizeWidth.toStringAsFixed(2)}',
                      handleSizeWidth, (v) {
                    setState(() => handleSizeWidth = v);
                  }, max: 100, min: -50),
                ],
              ),
              _DividerWithText(
                text: 'Items',
                children: [
                  Row(
                    mainAxisAlignment: MainAxisAlignment.spaceAround,
                    children: [
                      Column(
                        children: [
                          slider(
                            'R',
                            itemOneColor?.red.toDouble() ?? 0,
                            (v) => setState(() => itemOneColor =
                                itemOneColor?.withRed(v.toInt()) ??
                                    Colors.white),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'G',
                            itemOneColor?.green.toDouble() ?? 0,
                            (v) => setState(() => itemOneColor =
                                itemOneColor?.withGreen(v.toInt()) ??
                                    Colors.white),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'B',
                            itemOneColor?.blue.toDouble() ?? 0,
                            (v) => setState(() => itemOneColor =
                                itemOneColor?.withBlue(v.toInt()) ??
                                    Colors.white),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'A',
                            itemOneColor?.opacity.toDouble() ?? 0,
                            (v) => setState(() => itemOneColor =
                                itemOneColor?.withOpacity(v) ?? Colors.white),
                            max: 1,
                            min: 0,
                          ),
                          PopupMenuButton<BlendMode>(
                            color: Colors.grey.shade800,
                            child: Row(
                              children: [
                                text('Blend mode: '),
                                text(itemOneBlendMode
                                    .toString()
                                    .split('.')
                                    .last),
                              ],
                            ),
                            onSelected: (v) =>
                                setState(() => itemOneBlendMode = v),
                            itemBuilder: (context) => BlendMode.values
                                .map(
                                  (e) => PopupMenuItem(
                                    value: e,
                                    child: text(e.toString().split('.')[1]),
                                  ),
                                )
                                .toList(),
                          ),
                          const SizedBox(height: 10),
                          switcher('Add Blur', itemOneWrapper != null, (v) {
                            setState(
                              () => itemOneWrapper = v
                                  ? (child) => ImageFiltered(
                                        imageFilter: ImageFilter.blur(
                                            sigmaX: 2, sigmaY: 5),
                                        child: child,
                                      )
                                  : null,
                            );
                          }),
                        ],
                      ),
                      Column(
                        children: [
                          slider(
                            'R',
                            itemTwoColor?.red.toDouble() ?? 0,
                            (v) => setState(() => itemTwoColor =
                                itemTwoColor?.withRed(v.toInt()) ??
                                    Colors.white),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'G',
                            itemTwoColor?.green.toDouble() ?? 0,
                            (v) => setState(() => itemTwoColor =
                                itemTwoColor?.withGreen(v.toInt()) ??
                                    Colors.white),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'B',
                            itemTwoColor?.blue.toDouble() ?? 0,
                            (v) => setState(() => itemTwoColor =
                                itemTwoColor?.withBlue(v.toInt()) ??
                                    Colors.white),
                            max: 255,
                            min: 0,
                          ),
                          slider(
                            'A',
                            itemTwoColor?.opacity.toDouble() ?? 0,
                            (v) => setState(() => itemTwoColor =
                                itemTwoColor?.withOpacity(v) ?? Colors.white),
                            max: 1,
                            min: 0,
                          ),
                          PopupMenuButton<BlendMode>(
                            color: Colors.grey.shade800,
                            child: Row(
                              children: [
                                text('Blend mode: '),
                                text(itemTwoBlendMode
                                    .toString()
                                    .split('.')
                                    .last),
                              ],
                            ),
                            onSelected: (v) =>
                                setState(() => itemTwoBlendMode = v),
                            itemBuilder: (context) => BlendMode.values
                                .map(
                                  (e) => PopupMenuItem(
                                    value: e,
                                    child: text(e.toString().split('.')[1]),
                                  ),
                                )
                                .toList(),
                          ),
                          const SizedBox(height: 10),
                          switcher('Add Color Filter', itemTwoWrapper != null,
                              (v) {
                            setState(
                              () => itemTwoWrapper = v
                                  ? (child) => ColorFiltered(
                                        colorFilter: const ColorFilter.mode(
                                            Colors.red, BlendMode.color),
                                        child: child,
                                      )
                                  : null,
                            );
                          }),
                        ],
                      )
                    ],
                  )
                ],
              ),
              const SizedBox(height: 100)
            ],
          ),
        ),
      ),
      /*
      bottomNavigationBar: BottomAppBar(
        color: Colors.grey.shade900,
        child: const Padding(
          padding: EdgeInsets.all(20.0),
          child: Text(
            '© 2022 Miriam Raya\nConcept by: Lynn Chen',
            style: TextStyle(color: Colors.white),
          ),
        ),
      ),*/
    );
  }

  Widget slider(String title, double value, Function(double) onChanged,
      {double max = 1, double min = 0}) {
    return Row(
      children: [
        text(title),
        const SizedBox(width: 5),
        SizedBox(
          //width: 100,
          child: CupertinoSlider(
            value: value,
            max: max,
            min: min,
            onChanged: onChanged,
          ),
        ),
      ],
    );
  }

  Widget switcher(String title, bool value, Function(bool) onChanged) {
    return Row(
      mainAxisAlignment: MainAxisAlignment.spaceBetween,
      children: [
        text(title),
        const SizedBox(width: 10),
        SizedBox(
          child: CupertinoSwitch(
            value: value,
            onChanged: onChanged,
          ),
        ),
      ],
    );
  }

  Widget text(String text) {
    return Text(text,
        style: const TextStyle(
          color: Colors.white,
          fontSize: 15,
          fontWeight: FontWeight.w500,
        ));
  }

  Widget directionArrow() {
    return Column(
      mainAxisAlignment: MainAxisAlignment.spaceEvenly,
      children: List.generate(
        SliderDirection.values.length,
        (index) {
          final dir = SliderDirection.values[index];
          late IconData icon;

          switch (dir) {
            case SliderDirection.leftToRight:
              icon = Icons.arrow_forward;
              break;
            case SliderDirection.rightToLeft:
              icon = Icons.arrow_back;
              break;
            case SliderDirection.topToBottom:
              icon = Icons.arrow_downward;
              break;
            case SliderDirection.bottomToTop:
              icon = Icons.arrow_upward;
              break;
          }
          return IconButton(
            icon: Icon(
              icon,
              color: dir == direction ? Colors.blueAccent : Colors.white,
            ),
            onPressed: () => setState(() => direction = dir),
          );
        },
      ),
    );
  }
}

class _DividerWithText extends StatefulWidget {
  const _DividerWithText({
    required this.text,
    required this.children,
  });

  final String text;
  final List<Widget> children;

  @override
  State<_DividerWithText> createState() => _DividerWithTextState();
}

class _DividerWithTextState extends State<_DividerWithText> {
  bool show = false;
  @override
  Widget build(BuildContext context) {
    return Column(
      children: [
        CupertinoButton(
          padding: const EdgeInsets.all(0),
          onPressed: () => setState(() => show = !show),
          child: Row(children: [
            const Expanded(child: Divider(color: Colors.white)),
            Padding(
              padding: const EdgeInsets.symmetric(horizontal: 10),
              child: Text(
                widget.text,
                style: const TextStyle(
                  color: Colors.white,
                  fontSize: 15,
                  fontWeight: FontWeight.w500,
                ),
              ),
            ),
            const Expanded(child: Divider(color: Colors.white)),
          ]),
        ),
        if (show) ...widget.children,
      ],
    );
  }
}
101
likes
160
points
412
downloads
screenshot

Publisher

unverified uploader

Weekly Downloads

Easily compare two images with a slider and a draggable line/handle, fully customizable.

Repository (GitHub)
View/report issues
Contributing

Documentation

API reference

License

MIT (license)

Dependencies

flutter

More

Packages that depend on image_compare_slider