Back to Posts

Adding Borders to Specific Parts of a Container in Flutter

8 min read
<div style="text-align: center;"> <img src="" alt="Partial Border Example" width="300" /> </div>

Flutter provides several ways to add borders to specific parts of a container. This guide will show you various techniques to achieve this, from simple to advanced implementations.

Basic Techniques

1. Using Border Class

class PartialBorderContainer extends StatelessWidget {
  const PartialBorderContainer({super.key});

  @override
  Widget build(BuildContext context) {
    return Container(
      width: 200,
      height: 200,
      decoration: BoxDecoration(
        border: Border(
          top: BorderSide(color: Colors.blue, width: 2),
          bottom: BorderSide(color: Colors.blue, width: 2),
        ),
      ),
      child: const Center(child: Text('Top and Bottom Borders')),
    );
  }
}

2. Using BoxDecoration

class BoxDecorationBorder extends StatelessWidget {
  const BoxDecorationBorder({super.key});

  @override
  Widget build(BuildContext context) {
    return Container(
      width: 200,
      height: 200,
      decoration: BoxDecoration(
        border: Border.all(color: Colors.blue, width: 2),
        borderRadius: const BorderRadius.only(
          topLeft: Radius.circular(10),
          bottomRight: Radius.circular(10),
        ),
      ),
      child: const Center(child: Text('Custom Border Radius')),
    );
  }
}

Advanced Techniques

1. Custom Painter for Complex Borders

class CustomBorderPainter extends CustomPainter {
  final Color color;
  final double strokeWidth;

  CustomBorderPainter({required this.color, this.strokeWidth = 2.0});

  @override
  void paint(Canvas canvas, Size size) {
    final paint = Paint()
      ..color = color
      ..strokeWidth = strokeWidth
      ..style = PaintingStyle.stroke;

    final path = Path()
      ..moveTo(0, size.height * 0.2)
      ..lineTo(size.width * 0.3, size.height * 0.2)
      ..moveTo(size.width * 0.7, size.height * 0.2)
      ..lineTo(size.width, size.height * 0.2)
      ..moveTo(0, size.height * 0.8)
      ..lineTo(size.width * 0.3, size.height * 0.8)
      ..moveTo(size.width * 0.7, size.height * 0.8)
      ..lineTo(size.width, size.height * 0.8);

    canvas.drawPath(path, paint);
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) => false;
}

class CustomBorderContainer extends StatelessWidget {
  const CustomBorderContainer({super.key});

  @override
  Widget build(BuildContext context) {
    return CustomPaint(
      painter: CustomBorderPainter(color: Colors.blue),
      child: Container(
        width: 200,
        height: 200,
        child: const Center(child: Text('Custom Border')),
      ),
    );
  }
}

2. Gradient Borders

class GradientBorderContainer extends StatelessWidget {
  const GradientBorderContainer({super.key});

  @override
  Widget build(BuildContext context) {
    return Container(
      width: 200,
      height: 200,
      decoration: BoxDecoration(
        gradient: LinearGradient(
          colors: [Colors.blue, Colors.purple],
          begin: Alignment.topLeft,
          end: Alignment.bottomRight,
        ),
        borderRadius: BorderRadius.circular(10),
      ),
      padding: const EdgeInsets.all(2),
      child: Container(
        decoration: BoxDecoration(
          color: Colors.white,
          borderRadius: BorderRadius.circular(8),
        ),
        child: const Center(child: Text('Gradient Border')),
      ),
    );
  }
}

Special Cases

1. Dashed Borders

class DashedBorderPainter extends CustomPainter {
  final Color color;
  final double strokeWidth;
  final double dashWidth;
  final double dashSpace;

  DashedBorderPainter({
    required this.color,
    this.strokeWidth = 2.0,
    this.dashWidth = 5.0,
    this.dashSpace = 3.0,
  });

  @override
  void paint(Canvas canvas, Size size) {
    final paint = Paint()
      ..color = color
      ..strokeWidth = strokeWidth
      ..style = PaintingStyle.stroke;

    void drawDashedLine(Offset start, Offset end) {
      final path = Path();
      final distance = (end - start).distance;
      final dashCount = (distance / (dashWidth + dashSpace)).floor();
      
      for (var i = 0; i < dashCount; i++) {
        final startOffset = start + (end - start) * (i * (dashWidth + dashSpace) / distance);
        final endOffset = start + (end - start) * ((i * (dashWidth + dashSpace) + dashWidth) / distance);
        path.moveTo(startOffset.dx, startOffset.dy);
        path.lineTo(endOffset.dx, endOffset.dy);
      }
      
      canvas.drawPath(path, paint);
    }

    drawDashedLine(Offset(0, 0), Offset(size.width, 0));
    drawDashedLine(Offset(size.width, 0), Offset(size.width, size.height));
    drawDashedLine(Offset(size.width, size.height), Offset(0, size.height));
    drawDashedLine(Offset(0, size.height), Offset(0, 0));
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) => false;
}

2. Animated Borders

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

  @override
  State<AnimatedBorderContainer> createState() => _AnimatedBorderContainerState();
}

class _AnimatedBorderContainerState extends State<AnimatedBorderContainer>
    with SingleTickerProviderStateMixin {
  late AnimationController _controller;
  late Animation<double> _animation;

  @override
  void initState() {
    super.initState();
    _controller = AnimationController(
      duration: const Duration(seconds: 2),
      vsync: this,
    )..repeat(reverse: true);
    _animation = Tween<double>(begin: 0, end: 1).animate(_controller);
  }

  @override
  void dispose() {
    _controller.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return AnimatedBuilder(
      animation: _animation,
      builder: (context, child) {
        return Container(
          width: 200,
          height: 200,
          decoration: BoxDecoration(
            border: Border.all(
              color: Color.lerp(Colors.blue, Colors.purple, _animation.value)!,
              width: 2,
            ),
          ),
          child: const Center(child: Text('Animated Border')),
        );
      },
    );
  }
}

Best Practices

  1. Performance

    • Use appropriate border types
    • Consider using CustomPainter for complex borders
    • Optimize animations
    • Cache expensive calculations
  2. Visual Consistency

    • Maintain consistent border styles
    • Use appropriate border widths
    • Consider color schemes
    • Test on different backgrounds
  3. Accessibility

    • Ensure sufficient contrast
    • Consider color blindness
    • Test with screen readers
    • Provide alternative styles
  4. User Experience

    • Use subtle borders
    • Consider loading states
    • Implement proper animations
    • Provide visual feedback

Common Issues and Solutions

  1. Border Overlap

    // Use proper border radius
    Container(
      decoration: BoxDecoration(
        border: Border.all(color: Colors.blue),
        borderRadius: BorderRadius.circular(10),
      ),
    )
  2. Performance Issues

    // Use RepaintBoundary for complex borders
    RepaintBoundary(
      child: CustomPaint(
        painter: CustomBorderPainter(),
        child: Container(...),
      ),
    )
  3. Animation Performance

    // Use AnimatedBuilder for efficient animations
    AnimatedBuilder(
      animation: _animation,
      builder: (context, child) {
        return Container(...);
      },
    )

Conclusion

Adding partial borders to containers can significantly enhance your app's visual appeal. Remember to:

  • Choose appropriate border types
  • Consider performance implications
  • Maintain visual consistency
  • Test across different devices

Happy coding!