Close Menu
geekfence.comgeekfence.com
    What's Hot

    Edge browser leaves passwords exposed in plain text, says researcher

    May 5, 2026

    The Download: inside the Musk v. Altman trial, and AI for democracy

    May 5, 2026

    Charter offers second line on same mobile device

    May 5, 2026
    Facebook X (Twitter) Instagram
    • About Us
    • Contact Us
    Facebook Instagram
    geekfence.comgeekfence.com
    • Home
    • UK Tech News
    • AI
    • Big Data
    • Cyber Security
      • Cloud Computing
      • iOS Development
    • IoT
    • Mobile
    • Software
      • Software Development
      • Software Engineering
    • Technology
      • Green Technology
      • Nanotechnology
    • Telecom
    geekfence.comgeekfence.com
    Home»iOS Development»ios – Flutter UI not matching design: top-right badge not properly attached to card border
    iOS Development

    ios – Flutter UI not matching design: top-right badge not properly attached to card border

    AdminBy AdminMay 5, 2026No Comments3 Mins Read2 Views
    Facebook Twitter Pinterest LinkedIn Telegram Tumblr Email
    ios – Flutter UI not matching design: top-right badge not properly attached to card border
    Share
    Facebook Twitter LinkedIn Pinterest Email


    Here is what I am trying to do:

    Image

    The screenshot is taken from a iPhone 15 Pro:

    Image

    I’m trying to build a pricing UI in Flutter, but the layout is not matching the design—specifically the top-right badge (“Starter Plan”).

    class _PricingScreenState extends State {
      final PageController _controller = PageController();
      int currentPage = 0;
    
      final List plans = [
        PlanModel(
          title: "Basic Plan",
          price: "\$13 USD",
          subtitle: "per user/month",
          tag: "Starter Plan",
          buttonText: "Get Basic - \$13/month",
          features: [
            "",
            "",
            "",
            "",
            "",
            "",
            "",
            "",
            "",
          ],
          isPremium: false,
        ),
      ];
    
      @override
      Widget build(BuildContext context) {
        return Scaffold(
          backgroundColor: Colors.white,
          body: SafeArea(
            child: Column(
              children: [
                Padding(
                  padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 16),
                  child: Row(
                    mainAxisAlignment: MainAxisAlignment.spaceBetween,
                    children: [
                      Text(
                        currentPage == 2 ? "Customize Plan" : "Choose Your Plan",
                        style: const TextStyle(
                          fontSize: 26,
                          fontWeight: FontWeight.w800,
                        ),
                      ),
                    ],
                  ),
                ),
    
                // PAGES
                Expanded(
                  child: PageView.builder(
                    controller: _controller,
                    itemCount: plans.length,
                    onPageChanged: (index) {
                      setState(() => currentPage = index);
                    },
                    itemBuilder: (context, index) {
                      return _PlanCard(plan: plans[index]);
                    },
                  ),
                ),
              ],
            ),
          ),
        );
      }
    }
    
    class _PlanCard extends StatelessWidget {
      final PlanModel plan;
    
      const _PlanCard({required this.plan});
    
      @override
      Widget build(BuildContext context) {
        final isPremium = plan.isPremium;
        final isCustom = plan.isCustom;
    
        return Padding(
          padding: const EdgeInsets.symmetric(horizontal: 20),
          child: Column(
            children: [
    
              Expanded(
                child: Stack(
                  clipBehavior: Clip.none,
                  children: [
    
                    Container(
                      width: double.infinity,
                      decoration: BoxDecoration(
                        color: isPremium
                            ? const Color(0xFF4A90E2)
                            : const Color(0xFFF5F5F7),
                        borderRadius: BorderRadius.circular(20),
                      ),
                      padding: const EdgeInsets.fromLTRB(22, 28, 22, 22),
                      child: Column(
                        crossAxisAlignment: CrossAxisAlignment.start,
                        children: [
    
                          Text(
                            plan.title,
                            style: TextStyle(
                              color: isPremium ? Colors.white : Colors.black,
                              fontWeight: FontWeight.w600,
                            ),
                          ),
    
                          const SizedBox(height: 10),
    
                          if (!isCustom)
                            Text(
                              plan.price,
                              style: TextStyle(
                                fontSize: 34,
                                fontWeight: FontWeight.bold,
                                color:
                                isPremium ? Colors.white : Colors.black,
                              ),
                            ),
    
                          const SizedBox(height: 5),
    
                          Text(
                            plan.subtitle,
                            style: TextStyle(
                              color: isPremium
                                  ? Colors.white70
                                  : Colors.grey,
                            ),
                          ),
    
                          const SizedBox(height: 20),
    
                          if (!isCustom)
                            const Divider(),
    
                          const SizedBox(height: 10),
    
                          Expanded(
                            child: ListView.builder(
                              itemCount: plan.features.length,
                              itemBuilder: (_, i) => _FeatureRow(
                                text: plan.features[i],
                                isPremium: isPremium,
                                isCustom: isCustom,
                              ),
                            ),
                          ),
                        ],
                      ),
                    ),
    
                    // TAG BADGE
                    Positioned(
                      top: -12,
                      right: 16,
                      child: Container(
                        padding: const EdgeInsets.symmetric(
                            horizontal: 14, vertical: 6),
                        decoration: BoxDecoration(
                          color: Colors.white,
                          borderRadius: BorderRadius.circular(30),
                        ),
                        child: Text(plan.tag),
                      ),
                    )
                  ],
                ),
              ),
            ],
          ),
        );
      }
    }
    
    class _FeatureRow extends StatelessWidget {
      final String text;
      final bool isPremium;
      final bool isCustom;
    
      const _FeatureRow({
        required this.text,
        this.isPremium = false,
        this.isCustom = false,
      });
    
      @override
      Widget build(BuildContext context) {
        return Padding(
          padding: const EdgeInsets.symmetric(vertical: 6),
          child: Row(
            children: [
              CircleAvatar(
                radius: 10,
                backgroundColor:
                isPremium || isCustom ? Colors.yellow : Colors.green,
                child: Icon(
                  Icons.check,
                  size: 12,
                  color: Colors.black,
                ),
              ),
              const SizedBox(width: 10),
              Expanded(
                child: Text(
                  text,
                  style: TextStyle(
                    color: isPremium ? Colors.white : Colors.black,
                  ),
                ),
              ),
            ],
          ),
        );
      }
    }
    

    How can I make the badge look attached to the card corner instead of floating?
    Should I use ClipPath / CustomPainter for this?
    What is the best approach to achieve this kind of connected UI shape in Flutter?



    Source link

    Share. Facebook Twitter Pinterest LinkedIn Tumblr Email

    Related Posts

    Four Green Checkmarks: GitHub CI for macOS, iOS, Linux, and Windows

    May 1, 2026

    Swift enum all values – The.Swift.Dev.

    April 30, 2026

    ios – AppShortcut is shown in Shortcuts app but not in Spotlight

    April 29, 2026

    How to make a Swift framework?

    April 25, 2026

    ios – AVCaptureDevice.exposurePointOfInterest readback does not match set value on iPad Pro M4 / M3(UltraWide)and Gen2,3,4(TrueDepth) using resizeAspectFill

    April 24, 2026

    How I use FlowDeck to let my AI agent build and run my apps – Donny Wals

    April 21, 2026
    Top Posts

    Understanding U-Net Architecture in Deep Learning

    November 25, 202534 Views

    Hard-braking events as indicators of road segment crash risk

    January 14, 202626 Views

    Redefining AI efficiency with extreme compression

    March 25, 202625 Views
    Don't Miss

    Edge browser leaves passwords exposed in plain text, says researcher

    May 5, 2026

    A Norwegian researcher has identified an issue with Microsoft Edge’s Password Manager that could be…

    The Download: inside the Musk v. Altman trial, and AI for democracy

    May 5, 2026

    Charter offers second line on same mobile device

    May 5, 2026

    Strengthening cyber capacity in Kenya: A new toolkit with lessons for the region

    May 5, 2026
    Stay In Touch
    • Facebook
    • Instagram
    About Us

    At GeekFence, we are a team of tech-enthusiasts, industry watchers and content creators who believe that technology isn’t just about gadgets—it’s about how innovation transforms our lives, work and society. We’ve come together to build a place where readers, thinkers and industry insiders can converge to explore what’s next in tech.

    Our Picks

    Edge browser leaves passwords exposed in plain text, says researcher

    May 5, 2026

    The Download: inside the Musk v. Altman trial, and AI for democracy

    May 5, 2026

    Subscribe to Updates

    Please enable JavaScript in your browser to complete this form.
    Loading
    • About Us
    • Contact Us
    • Disclaimer
    • Privacy Policy
    • Terms and Conditions
    © 2026 Geekfence.All Rigt Reserved.

    Type above and press Enter to search. Press Esc to cancel.